diff --git a/packages/sdk/CHANGELOG.md b/packages/sdk/CHANGELOG.md index 1b713cd3..00714b45 100644 --- a/packages/sdk/CHANGELOG.md +++ b/packages/sdk/CHANGELOG.md @@ -2,6 +2,10 @@ All notable changes to the IBM® CICS® Plug-in for Zowe CLI will be documented in this file. +## Recent Changes + +- Enhancement: Add optional query parameters on getResource SDK method. [#168](https://github.com/zowe/cics-for-zowe-client/issues/168) + ## `6.1.0` - Enhancement: Made the region name optional on the getResource SDK method. [#162](https://github.com/zowe/cics-for-zowe-client/issues/162) diff --git a/packages/sdk/__tests__/__unit__/get/Get.resource.unit.test.ts b/packages/sdk/__tests__/__unit__/get/Get.resource.unit.test.ts index 893736d7..b4b1b4fa 100644 --- a/packages/sdk/__tests__/__unit__/get/Get.resource.unit.test.ts +++ b/packages/sdk/__tests__/__unit__/get/Get.resource.unit.test.ts @@ -184,5 +184,74 @@ describe("CMCI - Get resource", () => { expect(response).toContain(content); expect(deleteSpy).toHaveBeenCalledWith(dummySession, endPoint, []); }); + + it("should be able to get a resource with SUMMONLY specified", async () => { + resourceParms.cicsPlex = "plex1"; + resourceParms.regionName = "reg1"; + resourceParms.queryParams = { + summonly: true, + }; + endPoint = `/${CicsCmciConstants.CICS_SYSTEM_MANAGEMENT}/${resource}/plex1/reg1?SUMMONLY`; + response = await getResource(dummySession, resourceParms); + + expect(response).toContain(content); + expect(deleteSpy).toHaveBeenCalledWith(dummySession, endPoint, []); + }); + + it("should be able to get a resource with NODISCARD specified", async () => { + resourceParms.cicsPlex = "plex1"; + resourceParms.regionName = "reg1"; + resourceParms.queryParams = { + nodiscard: true, + }; + endPoint = `/${CicsCmciConstants.CICS_SYSTEM_MANAGEMENT}/${resource}/plex1/reg1?NODISCARD`; + response = await getResource(dummySession, resourceParms); + + expect(response).toContain(content); + expect(deleteSpy).toHaveBeenCalledWith(dummySession, endPoint, []); + }); + + it("should be able to get a resource with OVERRIDEWARNINGCOUNT specified", async () => { + resourceParms.cicsPlex = "plex1"; + resourceParms.regionName = "reg1"; + resourceParms.queryParams = { + overrideWarningCount: true, + }; + endPoint = `/${CicsCmciConstants.CICS_SYSTEM_MANAGEMENT}/${resource}/plex1/reg1?OVERRIDEWARNINGCOUNT`; + response = await getResource(dummySession, resourceParms); + + expect(response).toContain(content); + expect(deleteSpy).toHaveBeenCalledWith(dummySession, endPoint, []); + }); + + it("should be able to get a resource with all query params specified", async () => { + resourceParms.cicsPlex = "plex1"; + resourceParms.regionName = "reg1"; + resourceParms.queryParams = { + overrideWarningCount: true, + summonly: true, + nodiscard: true, + }; + endPoint = `/${CicsCmciConstants.CICS_SYSTEM_MANAGEMENT}/${resource}/plex1/reg1?SUMMONLY&NODISCARD&OVERRIDEWARNINGCOUNT`; + response = await getResource(dummySession, resourceParms); + + expect(response).toContain(content); + expect(deleteSpy).toHaveBeenCalledWith(dummySession, endPoint, []); + }); + + it("should be able to get a resource with no context and all query params specified", async () => { + resourceParms.cicsPlex = undefined; + resourceParms.regionName = undefined; + resourceParms.queryParams = { + overrideWarningCount: true, + summonly: true, + nodiscard: true, + }; + endPoint = `/${CicsCmciConstants.CICS_SYSTEM_MANAGEMENT}/${resource}/?SUMMONLY&NODISCARD&OVERRIDEWARNINGCOUNT`; + response = await getResource(dummySession, resourceParms); + + expect(response).toContain(content); + expect(deleteSpy).toHaveBeenCalledWith(dummySession, endPoint, []); + }); }); }); diff --git a/packages/sdk/__tests__/__unit__/utils/Utils.unit.test.ts b/packages/sdk/__tests__/__unit__/utils/Utils.unit.test.ts index 56d26200..de9e9971 100644 --- a/packages/sdk/__tests__/__unit__/utils/Utils.unit.test.ts +++ b/packages/sdk/__tests__/__unit__/utils/Utils.unit.test.ts @@ -223,5 +223,195 @@ describe("Utils - getResourceUri", () => { expect(error).toBeUndefined(); expect(response).toEqual("/CICSSystemManagement/resource1/cicsplex1/region1?CRITERIA=(NAME%3Dtest1)&PARAMETER=PARAM%3Dtest2"); }); + + it("should be able to get a resource uri with SUMMONLY specified", async () => { + try { + const options: IGetResourceUriOptions = { + cicsPlex: "cicsplex1", + regionName: "region1", + queryParams: { + summonly: true, + } + }; + + response = Utils.getResourceUri("resource1", options); + } catch (err) { + error = err; + } + + expect(response).toBeDefined(); + expect(error).toBeUndefined(); + expect(response).toEqual("/CICSSystemManagement/resource1/cicsplex1/region1?SUMMONLY"); + }); + + it("should be able to get a resource uri with SUMMONLY specified to false", async () => { + try { + const options: IGetResourceUriOptions = { + cicsPlex: "cicsplex1", + regionName: "region1", + queryParams: { + summonly: false, + } + }; + + response = Utils.getResourceUri("resource1", options); + } catch (err) { + error = err; + } + + expect(response).toBeDefined(); + expect(error).toBeUndefined(); + expect(response).toEqual("/CICSSystemManagement/resource1/cicsplex1/region1"); + }); + + it("should be able to get a resource uri with NODISCARD specified", async () => { + try { + const options: IGetResourceUriOptions = { + cicsPlex: "cicsplex1", + regionName: "region1", + queryParams: { + nodiscard: true, + } + }; + + response = Utils.getResourceUri("resource1", options); + } catch (err) { + error = err; + } + + expect(response).toBeDefined(); + expect(error).toBeUndefined(); + expect(response).toEqual("/CICSSystemManagement/resource1/cicsplex1/region1?NODISCARD"); + }); + + it("should be able to get a resource uri with NODISCARD specified", async () => { + try { + const options: IGetResourceUriOptions = { + cicsPlex: "cicsplex1", + regionName: "region1", + queryParams: { + nodiscard: false, + } + }; + + response = Utils.getResourceUri("resource1", options); + } catch (err) { + error = err; + } + + expect(response).toBeDefined(); + expect(error).toBeUndefined(); + expect(response).toEqual("/CICSSystemManagement/resource1/cicsplex1/region1"); + }); + + it("should be able to get a resource uri with OVERRIDEWARNINGCOUNT specified", async () => { + try { + const options: IGetResourceUriOptions = { + cicsPlex: "cicsplex1", + regionName: "region1", + queryParams: { + overrideWarningCount: true, + } + }; + + response = Utils.getResourceUri("resource1", options); + } catch (err) { + error = err; + } + + expect(response).toBeDefined(); + expect(error).toBeUndefined(); + expect(response).toEqual("/CICSSystemManagement/resource1/cicsplex1/region1?OVERRIDEWARNINGCOUNT"); + }); + + it("should be able to get a resource uri with OVERRIDEWARNINGCOUNT specified to false", async () => { + try { + const options: IGetResourceUriOptions = { + cicsPlex: "cicsplex1", + regionName: "region1", + queryParams: { + overrideWarningCount: false, + } + }; + + response = Utils.getResourceUri("resource1", options); + } catch (err) { + error = err; + } + + expect(response).toBeDefined(); + expect(error).toBeUndefined(); + expect(response).toEqual("/CICSSystemManagement/resource1/cicsplex1/region1"); + }); + + it("should be able to get a resource uri with all query params specified", async () => { + try { + const options: IGetResourceUriOptions = { + cicsPlex: "cicsplex1", + regionName: "region1", + queryParams: { + summonly: true, + nodiscard: true, + overrideWarningCount: true, + } + }; + + response = Utils.getResourceUri("resource1", options); + } catch (err) { + error = err; + } + + expect(response).toBeDefined(); + expect(error).toBeUndefined(); + expect(response).toEqual("/CICSSystemManagement/resource1/cicsplex1/region1?SUMMONLY&NODISCARD&OVERRIDEWARNINGCOUNT"); + }); + + it("should be able to get a resource uri with all query params specified and no context", async () => { + try { + const options: IGetResourceUriOptions = { + queryParams: { + summonly: true, + nodiscard: true, + overrideWarningCount: true, + } + }; + + response = Utils.getResourceUri("resource1", options); + } catch (err) { + error = err; + } + + expect(response).toBeDefined(); + expect(error).toBeUndefined(); + expect(response).toEqual("/CICSSystemManagement/resource1/?SUMMONLY&NODISCARD&OVERRIDEWARNINGCOUNT"); + }); + }); +}); + +describe('Utils - enforceParentheses', () => { + + it("should add brackets when none exist", () => { + const output = Utils.enforceParentheses("input"); + expect(output).toEqual("(input)"); + }); + + it("should add first bracket when end exists", () => { + const output = Utils.enforceParentheses("input with spaces)"); + expect(output).toEqual("(input with spaces)"); + }); + + it("should add last bracket when first exists", () => { + const output = Utils.enforceParentheses("(input with spec1@| characters"); + expect(output).toEqual("(input with spec1@| characters)"); + }); + + it("should do nothing when both brackets exist", () => { + const output = Utils.enforceParentheses("(fully covered)"); + expect(output).toEqual("(fully covered)"); + }); + + it("should do nothing when multiple brackets exist", () => { + const output = Utils.enforceParentheses("((()))"); + expect(output).toEqual("((()))"); }); }); diff --git a/packages/sdk/src/constants/CicsCmci.constants.ts b/packages/sdk/src/constants/CicsCmci.constants.ts index aaa36c3e..2a18507d 100644 --- a/packages/sdk/src/constants/CicsCmci.constants.ts +++ b/packages/sdk/src/constants/CicsCmci.constants.ts @@ -12,7 +12,7 @@ /** * Constants to be used by the API */ -export const CicsCmciConstants: { [key: string]: any } = { +export const CicsCmciConstants = { /** * Specifies the required part of the REST interface URI */ @@ -79,6 +79,11 @@ export const CicsCmciConstants: { [key: string]: any } = { */ NO_DISCARD: "NODISCARD", + /** + * OVERRIDEWARNINGCOUNT parameter + */ + OVERRIDE_WARNING_COUNT: "OVERRIDEWARNINGCOUNT", + /** * CRITERIA parameter */ diff --git a/packages/sdk/src/doc/IGetResourceUriOptions.ts b/packages/sdk/src/doc/IGetResourceUriOptions.ts index c4d9c4ad..3d4cac03 100644 --- a/packages/sdk/src/doc/IGetResourceUriOptions.ts +++ b/packages/sdk/src/doc/IGetResourceUriOptions.ts @@ -9,6 +9,8 @@ * */ +import { IResourceQueryParams } from "./IResourceQueryParms"; + export interface IGetResourceUriOptions { /** @@ -39,4 +41,9 @@ export interface IGetResourceUriOptions { * "CSDGROUP(D*)" */ parameter?: string; + + /** + * Query parameters to be used in the HTTP request + */ + queryParams?: IResourceQueryParams; } diff --git a/packages/sdk/src/doc/IResourceParms.ts b/packages/sdk/src/doc/IResourceParms.ts index 12f47081..8bf01310 100644 --- a/packages/sdk/src/doc/IResourceParms.ts +++ b/packages/sdk/src/doc/IResourceParms.ts @@ -9,38 +9,11 @@ * */ -export interface IResourceParms { +import { IGetResourceUriOptions } from "./IGetResourceUriOptions"; + +export interface IResourceParms extends IGetResourceUriOptions { /** * The name of the resource */ name: string; - - /** - * Criteria by which to filter the records - * - * Examples: - * "TRANID=TRAN" - * "PROGRAM=PRG*" - * "NAME=C* AND PROGRAM=D*" - */ - criteria?: string; - - /** - * Parameter by which to refine the records - * - * Example: - * "CSDGROUP(GRP1)" - * "CSDGROUP(D*)" - */ - parameter?: string; - - /** - * The name of the CICS region of the program - */ - regionName?: string; - - /** - * CICS Plex of the program - */ - cicsPlex?: string; } diff --git a/packages/sdk/src/doc/IResourceQueryParms.ts b/packages/sdk/src/doc/IResourceQueryParms.ts new file mode 100644 index 00000000..e008f3a9 --- /dev/null +++ b/packages/sdk/src/doc/IResourceQueryParms.ts @@ -0,0 +1,25 @@ +/** + * This program and the accompanying materials are made available under the terms of the + * Eclipse Public License v2.0 which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-v20.html + * + * SPDX-License-Identifier: EPL-2.0 + * + * Copyright Contributors to the Zowe Project. + * + */ + +export interface IResourceQueryParams { + /** + * Include SUMMONLY query parameter + */ + summonly?: boolean; + /** + * Include NODISCARD query parameter + */ + nodiscard?: boolean; + /** + * Include OVERRIDEWARNINGCOUNT query parameter + */ + overrideWarningCount?: boolean; +} diff --git a/packages/sdk/src/methods/get/Get.ts b/packages/sdk/src/methods/get/Get.ts index fb60be42..9d624d5a 100644 --- a/packages/sdk/src/methods/get/Get.ts +++ b/packages/sdk/src/methods/get/Get.ts @@ -10,8 +10,8 @@ */ import { AbstractSession, ImperativeExpect, Logger } from "@zowe/imperative"; +import { ICMCIApiResponse, IGetResourceUriOptions, IResourceParms } from "../../doc"; import { CicsCmciRestClient } from "../../rest"; -import { ICMCIApiResponse, IResourceParms, IGetResourceUriOptions } from "../../doc"; import { Utils } from "../../utils"; /** @@ -30,10 +30,11 @@ export async function getResource(session: AbstractSession, parms: IResourceParm Logger.getAppLogger().debug("Attempting to get resource(s) with the following parameters:\n%s", JSON.stringify(parms)); const options: IGetResourceUriOptions = { - "cicsPlex": parms.cicsPlex, - "regionName": parms.regionName, - "criteria": parms.criteria, - "parameter": parms.parameter + cicsPlex: parms.cicsPlex, + regionName: parms.regionName, + criteria: parms.criteria, + parameter: parms.parameter, + queryParams: parms.queryParams, }; const cmciResource = Utils.getResourceUri(parms.name, options); diff --git a/packages/sdk/src/utils/Utils.ts b/packages/sdk/src/utils/Utils.ts index 9be391a4..5135a789 100644 --- a/packages/sdk/src/utils/Utils.ts +++ b/packages/sdk/src/utils/Utils.ts @@ -25,7 +25,7 @@ export class Utils { * @param {string} resourceName - CMCI resource name * @param {IGetResourceUriOptions} options - CMCI resource options */ - public static getResourceUri(resourceName: string, options?: IGetResourceUriOptions) : string { + public static getResourceUri(resourceName: string, options?: IGetResourceUriOptions): string { ImperativeExpect.toBeDefinedAndNonBlank(resourceName, "CICS Resource name", "CICS resource name is required"); let delimiter = "?"; // initial delimiter @@ -35,20 +35,42 @@ export class Utils { let cmciResource = `/${CicsCmciConstants.CICS_SYSTEM_MANAGEMENT}/${resourceName}/${cicsPlex}${region}`; - if (options != null) { - if (options.criteria != null && options.criteria.length > 0) { - const addParentheses = options.criteria.charAt(0) !== '('; + if (options && options.criteria) { + cmciResource += `${delimiter}${CicsCmciConstants.CRITERIA}=${this.enforceParentheses(encodeURIComponent(options.criteria))}`; + delimiter = "&"; + } - cmciResource = cmciResource + delimiter + "CRITERIA=" + (addParentheses ? "(": "") + - encodeURIComponent(options.criteria) + (addParentheses ? ")": "") ; - delimiter = "&"; - } + if (options && options.parameter) { + cmciResource += `${delimiter}PARAMETER=${encodeURIComponent(options.parameter)}`; + delimiter = "&"; + } - if (options.parameter != null && options.parameter.length > 0) { - cmciResource = cmciResource + delimiter + "PARAMETER=" + encodeURIComponent(options.parameter); - } + if (options && options.queryParams && options.queryParams.summonly) { + cmciResource += `${delimiter}${CicsCmciConstants.SUMM_ONLY}`; + delimiter = "&"; + } + + if (options && options.queryParams && options.queryParams.nodiscard) { + cmciResource += `${delimiter}${CicsCmciConstants.NO_DISCARD}`; + delimiter = "&"; + } + + if (options && options.queryParams && options.queryParams.overrideWarningCount) { + cmciResource += `${delimiter}${CicsCmciConstants.OVERRIDE_WARNING_COUNT}`; + delimiter = "&"; } return cmciResource; } + + public static enforceParentheses(input: string): string { + if (!input.startsWith('(') && !input.endsWith(')')) { + return `(${input})`; + } else if (input.startsWith('(') && !input.endsWith(')')) { + return `${input})`; + } else if (!input.startsWith('(') && input.endsWith(')')) { + return `(${input}`; + } + return input; + } }