diff --git a/docs/docs/cmd/spo/listitem/listitem-attachment-get.mdx b/docs/docs/cmd/spo/listitem/listitem-attachment-get.mdx new file mode 100644 index 00000000000..5e40d700df2 --- /dev/null +++ b/docs/docs/cmd/spo/listitem/listitem-attachment-get.mdx @@ -0,0 +1,104 @@ +import Global from '/docs/cmd/_global.mdx'; +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +# spo listitem attachment get + +Gets an attachment from a list item + +## Usage + +```sh +m365 spo listitem attachment get [options] +``` + +## Options + +```md definition-list +`-u, --webUrl ` +: URL of the site where the list item is located. + +`--listId [listId]` +: ID of the list. Specify either `listTitle`, `listId` or `listUrl`. + +`--listTitle [listTitle]` +: Title of the list. Specify either `listTitle`, `listId` or `listUrl`. + +`--listUrl [listUrl]` +: Server- or site-relative URL of the list. Specify either `listTitle`, `listId` or `listUrl`. + +`--listItemId ` +: The ID of the list item. + +`-n, --fileName ` +: Name of the file to get. +``` + + + +## Examples + +Get an attachment from a list item by using list title. + +```sh +m365 spo listitem attachment get --webUrl https://contoso.sharepoint.com/sites/project-x --listTitle "Demo List" --listItemId 147 --fileName "File1.jpg" +``` + +Get an attachment from a list item by using list URL. + +```sh +m365 spo listitem attachment get --webUrl https://contoso.sharepoint.com/sites/project-x --listUrl "/sites/project-x/Lists/DemoList" --listItemId 147 --fileName "File1.jpg" +``` + +## Response + + + + + ```json + { + "FileName": "File1.jpg", + "FileNameAsPath": { + "DecodedUrl": "File1.jpg" + }, + "ServerRelativePath": { + "DecodedUrl": "/sites/project-x/Lists/DemoListAttachments/147/File1.jpg" + }, + "ServerRelativeUrl": "/sites/project-x/Lists/DemoListAttachments/147/File1.jpg" + } + ``` + + + + + ```text + FileName : File1.jpg + FileNameAsPath : {"DecodedUrl":"File1.jpg"} + ServerRelativePath: {"DecodedUrl":"/sites/project-x/Lists/DemoListAttachments/147/File1.jpg"} + ServerRelativeUrl : /sites/project-x/Lists/DemoListAttachments/147/File1.jpg + ``` + + + + + ```csv + FileName,ServerRelativeUrl + File1.jpg,/sites/project-x/Lists/DemoListAttachments/147/File1.jpg + ``` + + + + + ```md + # spo listitem attachment get --webUrl "https://contoso.sharepoint.com/sites/project-x" --listTitle "PnP PowerShell List" --listItemId "1" --fileName "File1.jpg" + + Date: 7/20/2023 + + Property | Value + ---------|------- + FileName | File1.jpg + ServerRelativeUrl | /sites/project-x/Lists/DemoListAttachments/147/File1.jpg + ``` + + + diff --git a/docs/src/config/sidebars.js b/docs/src/config/sidebars.js index 83ce9e2b3d3..54e1a9a36e9 100644 --- a/docs/src/config/sidebars.js +++ b/docs/src/config/sidebars.js @@ -2621,6 +2621,11 @@ const sidebars = { label: 'listitem remove', id: 'cmd/spo/listitem/listitem-remove' }, + { + type: 'doc', + label: 'listitem attachment get', + id: 'cmd/spo/listitem/listitem-attachment-get' + }, { type: 'doc', label: 'listitem attachment list', diff --git a/src/m365/spo/commands.ts b/src/m365/spo/commands.ts index aa2ad3e06cc..29363776144 100644 --- a/src/m365/spo/commands.ts +++ b/src/m365/spo/commands.ts @@ -159,6 +159,7 @@ export default { LIST_WEBHOOK_REMOVE: `${prefix} list webhook remove`, LIST_WEBHOOK_SET: `${prefix} list webhook set`, LISTITEM_ADD: `${prefix} listitem add`, + LISTITEM_ATTACHMENT_GET: `${prefix} listitem attachment get`, LISTITEM_ATTACHMENT_LIST: `${prefix} listitem attachment list`, LISTITEM_BATCH_ADD: `${prefix} listitem batch add`, LISTITEM_BATCH_SET: `${prefix} listitem batch set`, diff --git a/src/m365/spo/commands/listitem/listitem-attachment-get.spec.ts b/src/m365/spo/commands/listitem/listitem-attachment-get.spec.ts new file mode 100644 index 00000000000..7871d74b085 --- /dev/null +++ b/src/m365/spo/commands/listitem/listitem-attachment-get.spec.ts @@ -0,0 +1,209 @@ +import assert from 'assert'; +import sinon from 'sinon'; +import { telemetry } from '../../../../telemetry.js'; +import auth from '../../../../Auth.js'; +import { Cli } from '../../../../cli/Cli.js'; +import { CommandInfo } from '../../../../cli/CommandInfo.js'; +import { Logger } from '../../../../cli/Logger.js'; +import { CommandError } from '../../../../Command.js'; +import request from '../../../../request.js'; +import { formatting } from '../../../../utils/formatting.js'; +import { pid } from '../../../../utils/pid.js'; +import { session } from '../../../../utils/session.js'; +import { sinonUtil } from '../../../../utils/sinonUtil.js'; +import { urlUtil } from '../../../../utils/urlUtil.js'; +import commands from '../../commands.js'; +import command from './listitem-attachment-get.js'; + +describe(commands.LISTITEM_ATTACHMENT_GET, () => { + const webUrl = 'https://contoso.sharepoint.com/sites/project-x'; + const listId = '4fc5ba1e-18b7-49e0-81fe-54515cc2eede'; + const listTitle = 'Demo List'; + const listUrl = '/sites/project-x/Lists/DemoList'; + const listItemId = 147; + const fileName = 'File1.jpg'; + const listServerRelativeUrl: string = urlUtil.getServerRelativePath(webUrl, listUrl); + + let cli: Cli; + let log: any[]; + let logger: Logger; + let loggerLogSpy: sinon.SinonSpy; + let commandInfo: CommandInfo; + + const attachmentResponse = { + "FileName": "File1.jpg", + "FileNameAsPath": { + "DecodedUrl": "File1.jpg" + }, + "ServerRelativePath": { + "DecodedUrl": "/sites/project-x/Lists/DemoListAttachments/147/File1.jpg" + }, + "ServerRelativeUrl": "/sites/project-x/Lists/DemoListAttachments/147/File1.jpg" + }; + + const getFakes = async (opts: any) => { + if ((opts.url as string).indexOf('/_api/web/lists') > -1) { + return attachmentResponse; + } + + if (opts.url === `https://contoso.sharepoint.com/sites/project-x/_api/web/GetList('${formatting.encodeQueryParameter(listServerRelativeUrl)}')/items(147)/AttachmentFiles('${fileName}')`) { + return attachmentResponse; + } + + throw 'Invalid request'; + }; + + before(() => { + cli = Cli.getInstance(); + sinon.stub(auth, 'restoreAuth').callsFake(() => Promise.resolve()); + sinon.stub(telemetry, 'trackEvent').callsFake(() => { }); + sinon.stub(pid, 'getProcessName').callsFake(() => ''); + sinon.stub(session, 'getId').callsFake(() => ''); + auth.service.connected = true; + commandInfo = Cli.getCommandInfo(command); + }); + + beforeEach(() => { + log = []; + logger = { + log: async (msg: string) => { + log.push(msg); + }, + logRaw: async (msg: string) => { + log.push(msg); + }, + logToStderr: async (msg: string) => { + log.push(msg); + } + }; + loggerLogSpy = sinon.spy(logger, 'log'); + sinon.stub(cli, 'getSettingWithDefaultValue').callsFake(((settingName, defaultValue) => defaultValue)); + }); + + afterEach(() => { + sinonUtil.restore([ + request.get, + cli.getSettingWithDefaultValue + ]); + }); + + after(() => { + sinon.restore(); + auth.service.connected = false; + }); + + it('has correct name', () => { + assert.strictEqual(command.name, commands.LISTITEM_ATTACHMENT_GET); + }); + + it('has a description', () => { + assert.notStrictEqual(command.description, null); + }); + + it('supports specifying URL', () => { + const options = command.options; + let containsTypeOption = false; + options.forEach(o => { + if (o.option.indexOf('') > -1) { + containsTypeOption = true; + } + }); + assert(containsTypeOption); + }); + + it('fails validation if the webUrl option is not a valid SharePoint site URL', async () => { + const actual = await command.validate({ options: { webUrl: 'foo', listTitle: 'Demo List', listItemId: listItemId, fileName: fileName } }, commandInfo); + assert.notStrictEqual(actual, true); + }); + + it('passes validation if the webUrl option is a valid SharePoint site URL', async () => { + const actual = await command.validate({ options: { webUrl: 'https://contoso.sharepoint.com', listTitle: 'Demo List', listItemId: listItemId, fileName: fileName } }, commandInfo); + assert(actual); + }); + + it('fails validation if the listId option is not a valid GUID', async () => { + const actual = await command.validate({ options: { webUrl: 'https://contoso.sharepoint.com', listId: 'foo', listItemId: listItemId, fileName: fileName } }, commandInfo); + assert.notStrictEqual(actual, true); + }); + + it('passes validation if the listId option is a valid GUID', async () => { + const actual = await command.validate({ options: { webUrl: 'https://contoso.sharepoint.com', listId: listId, listItemId: listItemId, fileName: fileName } }, commandInfo); + assert(actual); + }); + + it('fails validation if the specified listItemId is not a number', async () => { + const actual = await command.validate({ options: { webUrl: 'https://contoso.sharepoint.com', listTitle: 'Demo List', listItemId: 'a', fileName: fileName } }, commandInfo); + assert.notStrictEqual(actual, true); + }); + + it('returns attachment from a list item by listId', async () => { + sinon.stub(request, 'get').callsFake(getFakes); + + const options: any = { + debug: true, + webUrl: webUrl, + listId: listId, + listItemId: listItemId, + fileName: fileName + }; + + await command.action(logger, { options: options } as any); + assert(loggerLogSpy.calledWith(attachmentResponse)); + }); + + it('returns attachment from a list item by listTitle', async () => { + sinon.stub(request, 'get').callsFake(getFakes); + + const options: any = { + debug: true, + webUrl: webUrl, + listTitle: listTitle, + listItemId: listItemId, + fileName: fileName + }; + + await command.action(logger, { options: options } as any); + assert(loggerLogSpy.calledWith(attachmentResponse)); + }); + + it('returns attachment from a list item by listUrl', async () => { + sinon.stub(request, 'get').callsFake(getFakes); + + const options: any = { + debug: true, + webUrl: webUrl, + listUrl: listUrl, + listItemId: listItemId, + fileName: fileName + }; + + await command.action(logger, { options: options } as any); + assert(loggerLogSpy.calledWith(attachmentResponse)); + }); + + it('correctly handles random API error', async () => { + sinon.stub(request, 'get').callsFake(() => Promise.reject('An error has occurred')); + + const options: any = { + webUrl: webUrl, + listId: listId, + listItemId: listItemId, + fileName: fileName + }; + + await assert.rejects(command.action(logger, { options: options } as any), new CommandError('An error has occurred')); + }); + + it('correctly handles no attachment found', async () => { + sinon.stub(request, 'get').rejects(new Error('Specified argument was out of the range of valid values.\r\nParameter name: fileName')); + + await assert.rejects(command.action(logger, { + options: { + webUrl: webUrl, + listId: listId, + listItemId: listItemId, + fileName: fileName + } + } as any), new CommandError('Specified argument was out of the range of valid values.\r\nParameter name: fileName')); + }); +}); \ No newline at end of file diff --git a/src/m365/spo/commands/listitem/listitem-attachment-get.ts b/src/m365/spo/commands/listitem/listitem-attachment-get.ts new file mode 100644 index 00000000000..6dbebec7080 --- /dev/null +++ b/src/m365/spo/commands/listitem/listitem-attachment-get.ts @@ -0,0 +1,132 @@ +import { Logger } from '../../../../cli/Logger.js'; +import GlobalOptions from '../../../../GlobalOptions.js'; +import request, { CliRequestOptions } from '../../../../request.js'; +import { formatting } from '../../../../utils/formatting.js'; +import { urlUtil } from '../../../../utils/urlUtil.js'; +import { validation } from '../../../../utils/validation.js'; +import SpoCommand from '../../../base/SpoCommand.js'; +import commands from '../../commands.js'; + +interface CommandArgs { + options: Options; +} + +interface Options extends GlobalOptions { + webUrl: string; + listId?: string; + listTitle?: string; + listUrl?: string; + listItemId: string; + fileName: string; +} + +class SpoListItemAttachmentGetCommand extends SpoCommand { + public get name(): string { + return commands.LISTITEM_ATTACHMENT_GET; + } + + public get description(): string { + return 'Gets an attachment from a list item'; + } + + constructor() { + super(); + + this.#initTelemetry(); + this.#initOptions(); + this.#initValidators(); + this.#initOptionSets(); + } + + #initTelemetry(): void { + this.telemetry.push((args: CommandArgs) => { + Object.assign(this.telemetryProperties, { + listId: typeof args.options.listId !== 'undefined', + listTitle: typeof args.options.listTitle !== 'undefined', + listUrl: typeof args.options.listUrl !== 'undefined' + }); + }); + } + + #initOptions(): void { + this.options.unshift( + { + option: '-u, --webUrl ' + }, + { + option: '--listId [listId]' + }, + { + option: '--listTitle [listTitle]' + }, + { + option: '--listUrl [listUrl]' + }, + { + option: '--listItemId ' + }, + { + option: '-n, --fileName ' + } + ); + } + + #initValidators(): void { + this.validators.push( + async (args: CommandArgs) => { + const isValidSharePointUrl: boolean | string = validation.isValidSharePointUrl(args.options.webUrl); + if (isValidSharePointUrl !== true) { + return isValidSharePointUrl; + } + + if (args.options.listId && !validation.isValidGuid(args.options.listId)) { + return `${args.options.listId} in option listId is not a valid GUID`; + } + + if (isNaN(parseInt(args.options.listItemId))) { + return `${args.options.listItemId} is not a number`; + } + + return true; + } + ); + } + + #initOptionSets(): void { + this.optionSets.push({ options: ['listId', 'listTitle', 'listUrl'] }); + } + + public async commandAction(logger: Logger, args: CommandArgs): Promise { + let requestUrl = `${args.options.webUrl}/_api/web`; + + if (args.options.listId) { + requestUrl += `/lists(guid'${formatting.encodeQueryParameter(args.options.listId)}')`; + } + else if (args.options.listTitle) { + requestUrl += `/lists/getByTitle('${formatting.encodeQueryParameter(args.options.listTitle)}')`; + } + else if (args.options.listUrl) { + const listServerRelativeUrl: string = urlUtil.getServerRelativePath(args.options.webUrl, args.options.listUrl); + requestUrl += `/GetList('${formatting.encodeQueryParameter(listServerRelativeUrl)}')`; + } + + const requestOptions: CliRequestOptions = { + url: `${requestUrl}/items(${args.options.listItemId})/AttachmentFiles('${args.options.fileName}')`, + method: 'GET', + headers: { + 'accept': 'application/json;odata=nometadata' + }, + responseType: 'json' + }; + + try { + const attachmentFile = await request.get(requestOptions); + await logger.log(attachmentFile); + } + catch (err: any) { + this.handleRejectedODataJsonPromise(err); + } + } +} + +export default new SpoListItemAttachmentGetCommand(); \ No newline at end of file