From 80b69a7cb1e45bf425df07cc0ef30e79b4e88aa2 Mon Sep 17 00:00:00 2001 From: Timothy Johnson Date: Tue, 12 Mar 2024 16:40:17 -0400 Subject: [PATCH 1/2] Fix api for uploading buffer to uss file Signed-off-by: Timothy Johnson --- packages/zosfiles/CHANGELOG.md | 5 + .../methods/upload/Upload.unit.test.ts | 128 ++++++++++++------ .../zosfiles/src/methods/upload/Upload.ts | 47 +++++-- 3 files changed, 127 insertions(+), 53 deletions(-) diff --git a/packages/zosfiles/CHANGELOG.md b/packages/zosfiles/CHANGELOG.md index 05d088b17c..f784f0615c 100644 --- a/packages/zosfiles/CHANGELOG.md +++ b/packages/zosfiles/CHANGELOG.md @@ -2,6 +2,11 @@ All notable changes to the Zowe z/OS files SDK package will be documented in this file. +## Recent Changes + +- LTS Breaking: Changed return type of `Upload.bufferToUssFile` to return `IZosFilesResponse` object instead of string. [#2089](https://github.com/zowe/zowe-cli/pull/2089) +- BugFix: Fixed `Upload.bufferToUssFile` not normalizing new lines when uploading plain text. [#2089](https://github.com/zowe/zowe-cli/pull/2089) + ## `8.0.0-next.202403041352` - BugFix: Updated engine to Node 18.12.0. [#2074](https://github.com/zowe/zowe-cli/pull/2074) diff --git a/packages/zosfiles/__tests__/__unit__/methods/upload/Upload.unit.test.ts b/packages/zosfiles/__tests__/__unit__/methods/upload/Upload.unit.test.ts index d8ce8b49a0..7522a2d00f 100644 --- a/packages/zosfiles/__tests__/__unit__/methods/upload/Upload.unit.test.ts +++ b/packages/zosfiles/__tests__/__unit__/methods/upload/Upload.unit.test.ts @@ -121,7 +121,7 @@ describe("z/OS Files - Upload", () => { expect(error.causeErrors).toBe(rootError); }); - it("return with proper response", async () => { + it("should return with proper response", async () => { const testReturn = {}; const testPath = "test/path"; lstatSpy.mockImplementationOnce((somePath, callback: any) => { @@ -141,7 +141,7 @@ describe("z/OS Files - Upload", () => { expect(dataSetSpy).toHaveBeenCalledTimes(1); expect(dataSetSpy).toHaveBeenLastCalledWith(dummySession, testPath, dsName, {}); }); - it("return with proper response with responseTimeout", async () => { + it("should return with proper response with responseTimeout", async () => { const testReturn = {}; const testPath = "test/path"; lstatSpy.mockImplementationOnce((somePath, callback: any) => { @@ -227,7 +227,7 @@ describe("z/OS Files - Upload", () => { expect(error.causeErrors).toBe(rootError); }); - it("return with proper message when path is pointing to a file", async () => { + it("should return with proper message when path is pointing to a file", async () => { lstatSpy.mockImplementationOnce((somePath, callback: any) => { callback(null, {isFile: () => false}); }); @@ -244,7 +244,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error.message).toContain(ZosFilesMessages.missingInputDir.message); }); - it("return with proper response", async () => { + it("should return with proper response", async () => { const testReturn = {}; const testPath = "test/path"; isDirSpy.mockReturnValueOnce(true); @@ -264,7 +264,7 @@ describe("z/OS Files - Upload", () => { expect(dataSetSpy).toHaveBeenCalledTimes(1); expect(dataSetSpy).toHaveBeenLastCalledWith(dummySession, testPath, dsName, {}); }); - it("return with proper response with responseTimeout", async () => { + it("should return with proper response with responseTimeout", async () => { const testReturn = {}; const testPath = "test/path"; isDirSpy.mockReturnValueOnce(true); @@ -284,7 +284,7 @@ describe("z/OS Files - Upload", () => { expect(dataSetSpy).toHaveBeenCalledTimes(1); expect(dataSetSpy).toHaveBeenLastCalledWith(dummySession, testPath, dsName, {responseTimeout: 5}); }); - it("return with proper response with encoding", async () => { + it("should return with proper response with encoding", async () => { const encoding = "1048"; const testReturn = {}; const testPath = "test/path"; @@ -346,7 +346,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error.message).toContain(ZosFilesMessages.missingDatasetName.message); }); - it("return error that throw by the ZosmfRestClient", async () => { + it("should return error that is thrown by the ZosmfRestClient", async () => { const buffer: Buffer = Buffer.from("testing"); const testError = new ImperativeError({ msg: "test error" @@ -364,7 +364,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error).toBe(testError); }); - it("return with proper response when upload buffer to a data set", async () => { + it("should return with proper response when upload buffer to a data set", async () => { const buffer: Buffer = Buffer.from("testing"); const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_DS_FILES, dsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.ACCEPT_ENCODING]; @@ -383,7 +383,7 @@ describe("z/OS Files - Upload", () => { reqHeaders, writeData: buffer}); }); - it("return with proper response when upload buffer to a PDS member", async () => { + it("should return with proper response when upload buffer to a PDS member", async () => { const buffer: Buffer = Buffer.from("testing"); const testDsName = `${dsName}(member)`; const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_DS_FILES, testDsName); @@ -403,6 +403,27 @@ describe("z/OS Files - Upload", () => { reqHeaders, writeData: buffer}); }); + it("should normalize new lines when upload buffer to a data set", async () => { + const buffer: Buffer = Buffer.from("testing\r\ntesting2"); + const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_DS_FILES, dsName); + const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.ACCEPT_ENCODING]; + + try { + response = await Upload.bufferToDataSet(dummySession, buffer, dsName); + } catch (err) { + error = err; + } + + expect(error).toBeUndefined(); + expect(response).toBeDefined(); + + const normalizedData = ZosFilesUtils.normalizeNewline(buffer); + expect(buffer.length).not.toBe(normalizedData.length); + expect(zosmfPutFullSpy).toHaveBeenCalledTimes(1); + expect(zosmfPutFullSpy).toHaveBeenCalledWith(dummySession, {resource: endpoint, + reqHeaders, + writeData: normalizedData}); + }); describe("Using optional parameters", () => { let buffer: Buffer; let uploadOptions: IUploadOptions; @@ -517,7 +538,7 @@ describe("z/OS Files - Upload", () => { reqHeaders, writeData: buffer}); }); - it("return with proper response when uploading with 'recall nowait' option", async () => { + it("should return with proper response when uploading with 'recall nowait' option", async () => { // Unit test for no wait option uploadOptions.recall = "nowait"; reqHeaders.push(ZosmfHeaders.X_IBM_MIGRATED_RECALL_NO_WAIT); @@ -536,7 +557,7 @@ describe("z/OS Files - Upload", () => { reqHeaders, writeData: buffer}); }); - it("return with proper response when uploading with 'recall error' option", async () => { + it("should return with proper response when uploading with 'recall error' option", async () => { // Unit test for no error option uploadOptions.recall = "error"; reqHeaders.push(ZosmfHeaders.X_IBM_MIGRATED_RECALL_ERROR); @@ -555,7 +576,7 @@ describe("z/OS Files - Upload", () => { reqHeaders, writeData: buffer}); }); - it("return with proper response when uploading with non-exiting recall option", async () => { + it("should return with proper response when uploading with non-exiting recall option", async () => { // Unit test default value uploadOptions.recall = "non-existing"; reqHeaders.push(ZosmfHeaders.X_IBM_MIGRATED_RECALL_NO_WAIT); @@ -574,7 +595,7 @@ describe("z/OS Files - Upload", () => { reqHeaders, writeData: buffer}); }); - it("return with proper response when uploading with pass 'etag' option", async () => { + it("should return with proper response when uploading with pass 'etag' option", async () => { // Unit test for pass etag option uploadOptions.etag = etagValue; reqHeaders.push({"If-Match" : uploadOptions.etag}); @@ -593,7 +614,7 @@ describe("z/OS Files - Upload", () => { reqHeaders, writeData: buffer}); }); - it("return with proper response when uploading with return 'etag' option", async () => { + it("should return with proper response when uploading with return 'etag' option", async () => { zosmfPutFullSpy.mockImplementationOnce(async () => fakeResponseWithEtag); // Unit test for return etag option reqHeaders.push(ZosmfHeaders.X_IBM_RETURN_ETAG); @@ -613,7 +634,7 @@ describe("z/OS Files - Upload", () => { writeData: buffer, dataToReturn: [CLIENT_PROPERTY.response]}); }); - it("return with proper response when uploading with responseTimeout option", async () => { + it("should return with proper response when uploading with responseTimeout option", async () => { uploadOptions.responseTimeout = 5; reqHeaders.push({[ZosmfHeaders.X_IBM_RESPONSE_TIMEOUT]: "5"}); @@ -632,7 +653,7 @@ describe("z/OS Files - Upload", () => { writeData: buffer}); }); }); - it("return with proper response when upload dataset with specify volume option", async () => { + it("should return with proper response when upload dataset with specify volume option", async () => { const buffer: Buffer = Buffer.from("testing"); const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_DS_FILES, `-(TEST)`, dsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.ACCEPT_ENCODING]; @@ -683,7 +704,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error.message).toContain(ZosFilesMessages.missingDatasetName.message); }); - it("return error that throw by the ZosmfRestClient", async () => { + it("should return error that throw by the ZosmfRestClient", async () => { const testError = new ImperativeError({ msg: "test error" }); @@ -700,7 +721,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error).toBe(testError); }); - it("return with proper response when upload stream to a data set", async () => { + it("should return with proper response when upload stream to a data set", async () => { const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_DS_FILES, dsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.ACCEPT_ENCODING]; @@ -719,7 +740,7 @@ describe("z/OS Files - Upload", () => { reqHeaders, requestStream: inputStream}); }); - it("return with proper response when upload stream to a PDS member", async () => { + it("should return with proper response when upload stream to a PDS member", async () => { const testDsName = `${dsName}(member)`; const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_DS_FILES, testDsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.ACCEPT_ENCODING]; @@ -739,7 +760,7 @@ describe("z/OS Files - Upload", () => { reqHeaders, requestStream: inputStream}); }); - it("return with proper response when upload stream to a data set with optional parameters 1", async () => { + it("should return with proper response when upload stream to a data set with optional parameters 1", async () => { const uploadOptions: IUploadOptions = { binary: true }; @@ -928,7 +949,7 @@ describe("z/OS Files - Upload", () => { requestStream: inputStream, dataToReturn: [CLIENT_PROPERTY.response]}); }); - it("return with proper response when upload stream to a data set with optional parameters 2", async () => { + it("should return with proper response when upload stream to a data set with optional parameters 2", async () => { const uploadOptions: IUploadOptions = { record: true }; @@ -1101,7 +1122,7 @@ describe("z/OS Files - Upload", () => { requestStream: inputStream, dataToReturn: [CLIENT_PROPERTY.response]}); }); - it("return with proper response when upload dataset with specify volume option", async () => { + it("should return with proper response when upload dataset with specify volume option", async () => { const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_DS_FILES, `-(TEST)`, dsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.ACCEPT_ENCODING]; const uploadOptions: IUploadOptions = { @@ -1528,8 +1549,8 @@ describe("z/OS Files - Upload", () => { }); describe("bufferToUssFile", () => { - const zosmfExpectSpy = jest.spyOn(ZosmfRestClient, "putExpectString"); - let USSresponse: string; + const zosmfExpectSpy = jest.spyOn(ZosmfRestClient, "putExpectFullResponse"); + let USSresponse: IZosFilesResponse; beforeEach(() => { USSresponse = undefined; error = undefined; @@ -1548,7 +1569,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error.message).toContain(ZosFilesMessages.missingUSSFileName.message); }); - it("return error that throw by the ZosmfRestClient", async () => { + it("should return error that is thrown by the ZosmfRestClient", async () => { const testError = new ImperativeError({ msg: "test error" }); @@ -1565,7 +1586,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error).toBe(testError); }); - it("return error that thrown when the 'record' option is specified", async () => { + it("should return error that is thrown when the 'record' option is specified", async () => { const record = true; try { @@ -1578,7 +1599,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error.message).toContain(ZosFilesMessages.unsupportedDataType.message); }); - it("return with proper response when upload USS file", async () => { + it("should return with proper response when upload USS file", async () => { const data: Buffer = Buffer.from("testing"); const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); const headers = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.TEXT_PLAIN, ZosmfHeaders.ACCEPT_ENCODING]; @@ -1593,9 +1614,9 @@ describe("z/OS Files - Upload", () => { expect(USSresponse).toBeDefined(); expect(zosmfExpectSpy).toHaveBeenCalledTimes(1); - expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, endpoint, headers, data); + expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, { reqHeaders: headers, resource: endpoint, writeData: data }); }); - it("return with proper response when upload USS file with responseTimeout", async () => { + it("should return with proper response when upload USS file with responseTimeout", async () => { const data: Buffer = Buffer.from("testing"); const responseTimeout = 5; const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); @@ -1618,9 +1639,9 @@ describe("z/OS Files - Upload", () => { expect(USSresponse).toBeDefined(); expect(zosmfExpectSpy).toHaveBeenCalledTimes(1); - expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, endpoint, headers, data); + expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, { reqHeaders: headers, resource: endpoint, writeData: data }); }); - it("return with proper response when upload USS file in binary", async () => { + it("should return with proper response when upload USS file in binary", async () => { const data: Buffer = Buffer.from("testing"); const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); const headers = [ZosmfHeaders.OCTET_STREAM, ZosmfHeaders.X_IBM_BINARY, ZosmfHeaders.ACCEPT_ENCODING]; @@ -1635,9 +1656,9 @@ describe("z/OS Files - Upload", () => { expect(USSresponse).toBeDefined(); expect(zosmfExpectSpy).toHaveBeenCalledTimes(1); - expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, endpoint, headers, data); + expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, { reqHeaders: headers, resource: endpoint, writeData: data }); }); - it("return with proper response when upload USS file with Etag", async () => { + it("should return with proper response when upload USS file with Etag", async () => { const data: Buffer = Buffer.from("testing"); const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); const headers = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.TEXT_PLAIN, ZosmfHeaders.ACCEPT_ENCODING, {"If-Match": etagValue}]; @@ -1656,7 +1677,7 @@ describe("z/OS Files - Upload", () => { expect(USSresponse).toBeDefined(); expect(zosmfExpectSpy).toHaveBeenCalledTimes(1); - expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, endpoint, headers, data); + expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, { reqHeaders: headers, resource: endpoint, writeData: data }); }); it("should set local encoding if specified", async () => { const data: Buffer = Buffer.from("testing"); @@ -1676,7 +1697,26 @@ describe("z/OS Files - Upload", () => { expect(USSresponse).toBeDefined(); expect(zosmfExpectSpy).toHaveBeenCalledTimes(1); - expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, endpoint, headers, data); + expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, { reqHeaders: headers, resource: endpoint, writeData: data }); + }); + it("should normalize new lines when upload USS file", async () => { + const data: Buffer = Buffer.from("testing\r\ntesting2"); + const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); + const headers = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.TEXT_PLAIN, ZosmfHeaders.ACCEPT_ENCODING]; + + try { + USSresponse = await Upload.bufferToUssFile(dummySession, dsName, data); + } catch (err) { + error = err; + } + + expect(error).toBeUndefined(); + expect(USSresponse).toBeDefined(); + + const normalizedData = ZosFilesUtils.normalizeNewline(data); + expect(data.length).not.toBe(normalizedData.length); + expect(zosmfExpectSpy).toHaveBeenCalledTimes(1); + expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, { reqHeaders: headers, resource: endpoint, writeData: normalizedData }); }); }); @@ -1718,7 +1758,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error.message).toContain(ZosFilesMessages.missingUSSFileName.message); }); - it("return error that is thrown by the ZosmfRestClient", async () => { + it("should return error that is thrown by the ZosmfRestClient", async () => { const testError = new ImperativeError({ msg: "test error" }); @@ -1735,7 +1775,7 @@ describe("z/OS Files - Upload", () => { expect(error).toBeDefined(); expect(error).toBe(testError); }); - it("return with proper response when upload USS file", async () => { + it("should return with proper response when upload USS file", async () => { const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.TEXT_PLAIN, ZosmfHeaders.ACCEPT_ENCODING]; @@ -1756,7 +1796,7 @@ describe("z/OS Files - Upload", () => { normalizeRequestNewLines: true}); expect(chtagSpy).toHaveBeenCalledTimes(0); }); - it("return with proper response when upload USS file with responseTimeout", async () => { + it("should return with proper response when upload USS file with responseTimeout", async () => { const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.TEXT_PLAIN, ZosmfHeaders.ACCEPT_ENCODING, {[ZosmfHeaders.X_IBM_RESPONSE_TIMEOUT]: "5"}]; @@ -1778,7 +1818,7 @@ describe("z/OS Files - Upload", () => { normalizeRequestNewLines: true}); expect(chtagSpy).toHaveBeenCalledTimes(0); }); - it("return with proper response when upload USS file in binary", async () => { + it("should return with proper response when upload USS file in binary", async () => { const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); const reqHeaders = [ZosmfHeaders.OCTET_STREAM, ZosmfHeaders.X_IBM_BINARY, ZosmfHeaders.ACCEPT_ENCODING]; @@ -1801,7 +1841,7 @@ describe("z/OS Files - Upload", () => { expect(chtagSpy).toHaveBeenCalledTimes(1); expect(chtagSpy).toHaveBeenCalledWith(dummySession, dsName, Tag.BINARY); }); - it("return with proper response when upload USS file with Etag", async () => { + it("should return with proper response when upload USS file with Etag", async () => { const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.TEXT_PLAIN, ZosmfHeaders.ACCEPT_ENCODING, {"If-Match": etagValue}]; @@ -1823,7 +1863,7 @@ describe("z/OS Files - Upload", () => { normalizeRequestNewLines: true}); expect(chtagSpy).toHaveBeenCalledTimes(0); }); - it("return with proper response when upload USS file and request Etag back", async () => { + it("should return with proper response when upload USS file and request Etag back", async () => { const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); const reqHeaders = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.TEXT_PLAIN, ZosmfHeaders.ACCEPT_ENCODING, ZosmfHeaders.X_IBM_RETURN_ETAG]; zosmfExpectFullSpy.mockImplementationOnce(async () => fakeResponseWithEtag); @@ -1976,7 +2016,7 @@ describe("z/OS Files - Upload", () => { expect(error.additionalDetails).toEqual(rootError.toString()); expect(error.causeErrors).toBe(rootError); }); - it("return with proper response when upload USS file", async () => { + it("should return with proper response when upload USS file", async () => { try { USSresponse = await Upload.fileToUssFile(dummySession, inputFile, "file"); } catch (err) { @@ -1992,7 +2032,7 @@ describe("z/OS Files - Upload", () => { expect(streamToUssFileSpy).toHaveBeenCalledTimes(1); expect(streamToUssFileSpy).toHaveBeenCalledWith(dummySession, "file", null, {}); }); - it("return with proper response when upload USS file with responseTimeout", async () => { + it("should return with proper response when upload USS file with responseTimeout", async () => { try { USSresponse = await Upload.fileToUssFile(dummySession, inputFile, "file", {responseTimeout: 5}); } catch (err) { @@ -2008,7 +2048,7 @@ describe("z/OS Files - Upload", () => { expect(streamToUssFileSpy).toHaveBeenCalledTimes(1); expect(streamToUssFileSpy).toHaveBeenCalledWith(dummySession, "file", null, {responseTimeout: 5}); }); - it("return with proper response when upload USS file including Etag", async () => { + it("should return with proper response when upload USS file including Etag", async () => { const streamResponse: IZosFilesResponse = { success: true, commandResponse: undefined as any, diff --git a/packages/zosfiles/src/methods/upload/Upload.ts b/packages/zosfiles/src/methods/upload/Upload.ts index 5c90d5c31c..570796a722 100644 --- a/packages/zosfiles/src/methods/upload/Upload.ts +++ b/packages/zosfiles/src/methods/upload/Upload.ts @@ -447,33 +447,62 @@ export class Upload { * Upload content to USS file * @param {AbstractSession} session - z/OS connection info * @param {string} ussname - Name of the USS file to write to - * @param {Buffer} buffer - Data to be written + * @param {Buffer} fileBuffer - Data to be written * @param {IUploadOptions} [options={}] - Uploading options - * @returns {Promise} + * @returns {Promise} */ public static async bufferToUssFile(session: AbstractSession, ussname: string, - buffer: Buffer, - options: IUploadOptions = {}) { + fileBuffer: Buffer, + options: IUploadOptions = {}): Promise { ImperativeExpect.toNotBeEqual(options.record, true, ZosFilesMessages.unsupportedDataType.message); options.binary = options.binary ? options.binary : false; ImperativeExpect.toNotBeNullOrUndefined(ussname, ZosFilesMessages.missingUSSFileName.message); ussname = ZosFilesUtils.sanitizeUssPathForRestCall(ussname); - const parameters: string = ZosFilesConstants.RES_USS_FILES + "/" + ussname; - const headers: IHeaderContent[] = this.generateHeadersBasedOnOptions(options, "buffer"); - return ZosmfRestClient.putExpectString(session, ZosFilesConstants.RESOURCE + parameters, headers, buffer); + const endpoint = ZosFilesConstants.RESOURCE + ZosFilesConstants.RES_USS_FILES + "/" + ussname; + const reqHeaders: IHeaderContent[] = this.generateHeadersBasedOnOptions(options, "buffer"); + + if (!options.binary) { + fileBuffer = ZosFilesUtils.normalizeNewline(fileBuffer); + } + + // Options to use the buffer to write a file + const requestOptions: IOptionsFullResponse = { + resource: endpoint, + reqHeaders, + writeData: fileBuffer + }; + + // If requestor needs etag, add header + get "response" back + if (options.returnEtag) { + requestOptions.dataToReturn = [CLIENT_PROPERTY.response]; + } + const uploadRequest: IRestClientResponse = await ZosmfRestClient.putExpectFullResponse(session, requestOptions); + + // By default, apiResponse is empty when uploading + const apiResponse: any = {}; + + // Return Etag in apiResponse, if requested + if (options.returnEtag) { + apiResponse.etag = uploadRequest.response.headers.etag; + } + + return { + success: true, + commandResponse: ZosFilesMessages.dataSetUploadedSuccessfully.message, + apiResponse + }; } /** * Upload content to USS file * @param {AbstractSession} session - z/OS connection info * @param {string} ussname - Name of the USS file to write to - * @param {Buffer} uploadStream - Data to be written + * @param {Stream} uploadStream - Data to be written * @param {IUploadOptions} [options={}] - Uploading options * @returns {Promise} - A response indicating the outcome */ - public static async streamToUssFile(session: AbstractSession, ussname: string, uploadStream: Readable, From 29c548d24697469498267d5247af4b6a8924cb28 Mon Sep 17 00:00:00 2001 From: Timothy Johnson Date: Tue, 12 Mar 2024 16:59:14 -0400 Subject: [PATCH 2/2] Add test for returning etag from bufferToUssFile Signed-off-by: Timothy Johnson --- .../methods/upload/Upload.unit.test.ts | 25 +++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/packages/zosfiles/__tests__/__unit__/methods/upload/Upload.unit.test.ts b/packages/zosfiles/__tests__/__unit__/methods/upload/Upload.unit.test.ts index 7522a2d00f..b755d19f2f 100644 --- a/packages/zosfiles/__tests__/__unit__/methods/upload/Upload.unit.test.ts +++ b/packages/zosfiles/__tests__/__unit__/methods/upload/Upload.unit.test.ts @@ -1550,6 +1550,10 @@ describe("z/OS Files - Upload", () => { describe("bufferToUssFile", () => { const zosmfExpectSpy = jest.spyOn(ZosmfRestClient, "putExpectFullResponse"); + const fakeResponseWithEtag = { + data: Buffer.from(dsName), + response: { headers: { etag: etagValue } } + }; let USSresponse: IZosFilesResponse; beforeEach(() => { USSresponse = undefined; @@ -1679,6 +1683,27 @@ describe("z/OS Files - Upload", () => { expect(zosmfExpectSpy).toHaveBeenCalledTimes(1); expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, { reqHeaders: headers, resource: endpoint, writeData: data }); }); + it("should return with proper response when upload USS file and request Etag back", async () => { + const data: Buffer = Buffer.from("testing"); + const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName); + const headers = [ZosmfHeaders.X_IBM_TEXT, ZosmfHeaders.TEXT_PLAIN, ZosmfHeaders.ACCEPT_ENCODING, ZosmfHeaders.X_IBM_RETURN_ETAG]; + zosmfExpectSpy.mockImplementationOnce(async () => fakeResponseWithEtag); + try { + USSresponse = await Upload.bufferToUssFile(dummySession, dsName, data, {returnEtag: true}); + } catch (err) { + error = err; + } + + expect(error).toBeUndefined(); + expect(USSresponse).toBeDefined(); + expect(USSresponse.success).toBeTruthy(); + expect(USSresponse.apiResponse.etag).toBeDefined(); + expect(USSresponse.apiResponse.etag).toEqual(etagValue); + + expect(zosmfExpectSpy).toHaveBeenCalledTimes(1); + expect(zosmfExpectSpy).toHaveBeenCalledWith(dummySession, { reqHeaders: headers, resource: endpoint, writeData: data, + dataToReturn: [CLIENT_PROPERTY.response] }); + }); it("should set local encoding if specified", async () => { const data: Buffer = Buffer.from("testing"); const endpoint = path.posix.join(ZosFilesConstants.RESOURCE, ZosFilesConstants.RES_USS_FILES, dsName);