From ddfee7e038d1abc9a38a7187a70dc24245eff4ec Mon Sep 17 00:00:00 2001 From: Elson9 Date: Thu, 7 Sep 2023 02:33:38 -0700 Subject: [PATCH 1/7] fixes request access button not being displayed when the Product has an inactive environment (#879) --- .../api-product-item/api-product-item.tsx | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/src/nextapp/components/api-product-item/api-product-item.tsx b/src/nextapp/components/api-product-item/api-product-item.tsx index 649a85bab..72a4c8c73 100644 --- a/src/nextapp/components/api-product-item/api-product-item.tsx +++ b/src/nextapp/components/api-product-item/api-product-item.tsx @@ -63,15 +63,15 @@ const ApiProductItem: React.FC = ({ )} - {!isPublic && !isTiered && ( + {!isTiered && ( <> {isPublic && ( - + )} {!isPublic && ( Date: Thu, 7 Sep 2023 02:36:27 -0700 Subject: [PATCH 2/7] Feature - gwa cli v2 (#845) Co-authored-by: James Elson Co-authored-by: Elson9 --- src/api-openapi.js | 23 +- src/api-proxy.js | 52 +++-- ...plocalhost4180managernamespaces-85071c.gql | 7 + src/batch/data-rules.js | 4 +- src/batch/feed-worker.ts | 7 +- .../transformations/toStringDefaultArray.ts | 4 + src/controllers/ioc/keystoneInjector.ts | 4 +- src/controllers/v2/GatewayController.ts | 14 ++ src/controllers/v2/NamespaceController.ts | 51 ++++- src/controllers/v2/openapi.yaml | 89 +++++++- src/controllers/v2/routes.ts | 96 ++++++++- src/controllers/v2/types-extra.ts | 9 + src/controllers/v2/types.ts | 13 +- src/lists/CredentialIssuer.js | 9 +- src/lists/extensions/CredentialIssuerExt.ts | 2 +- src/lists/extensions/Namespace.ts | 67 +++--- src/lists/extensions/ServiceAccount.ts | 6 +- src/lists/extensions/UserExt.ts | 12 +- .../new-namespace/new-namespace.tsx | 2 +- src/nextapp/shared/types/query.types.ts | 7 +- src/package-lock.json | 199 ++++++++++++------ src/package.json | 5 +- src/server.ts | 4 +- src/services/identifiers.ts | 10 +- src/services/keycloak/client-service.ts | 5 + src/services/keycloak/group-service.ts | 2 +- src/services/keycloak/namespace-details.ts | 27 ++- src/services/keystone/types.ts | 7 +- src/services/report/data/namespaces.ts | 1 + src/services/report/output/structure.ts | 5 + .../uma2/resource-registration-service.ts | 3 + src/services/utils.ts | 1 - src/services/workflow/client-shared-idp.ts | 76 ++++++- src/services/workflow/delete-namespace.ts | 4 + src/services/workflow/get-namespaces.ts | 2 + src/services/workflow/index.ts | 6 +- src/test/integrated/uma2/resource.ts | 11 +- src/test/services/identifiers.test.js | 11 + src/tsoa-v2.json | 5 + 39 files changed, 653 insertions(+), 209 deletions(-) create mode 100644 src/authz/graphql-whitelist/httplocalhost4180managernamespaces-85071c.gql create mode 100644 src/test/services/identifiers.test.js diff --git a/src/api-openapi.js b/src/api-openapi.js index bcf052c44..b2d1f99a2 100644 --- a/src/api-openapi.js +++ b/src/api-openapi.js @@ -23,13 +23,6 @@ class ApiOpenapiApp { RegisterRoutes(app); - // RFC 8631 service-desc link relation - // https://datatracker.ietf.org/doc/html/rfc8631 - app.get('/ds/api', (req, res) => { - res.setHeader('Link', '; rel="service-desc"'); - res.status(204).end(); - }); - app.get('/ds/api/openapi.yaml', (req, res) => { res.setHeader('Content-Type', 'application/yaml'); res.send(spec); @@ -54,14 +47,9 @@ class ApiOpenapiApp { */ specObject.components.securitySchemes.jwt.flows.clientCredentials.tokenUrl = `${process.env.OIDC_ISSUER}/protocol/openid-connect/token`; - RegisterRoutes(app); + specObject.components.securitySchemes.openid.openIdConnectUrl = `${process.env.OIDC_ISSUER}/.well-known/openid-configuration`; - // RFC 8631 service-desc link relation - // https://datatracker.ietf.org/doc/html/rfc8631 - app.get('/ds/api/v2', (req, res) => { - res.setHeader('Link', '; rel="service-desc"'); - res.status(204).end(); - }); + RegisterRoutes(app); app.get('/ds/api/v2/openapi.yaml', (req, res) => { res.setHeader('Content-Type', 'application/yaml'); @@ -86,6 +74,13 @@ class ApiOpenapiApp { this.prepareV2(app); this.prepareV1(app); + // RFC 8631 service-desc link relation + // https://datatracker.ietf.org/doc/html/rfc8631 + app.get('/ds/api', (req, res) => { + res.setHeader('Link', '; rel="service-desc"'); + res.status(204).end(); + }); + app.use(function errorHandler(err, req, res, next) { if (err instanceof UnauthorizedError) { return res.status(err.status).json({ diff --git a/src/api-proxy.js b/src/api-proxy.js index 3c7d89558..8540c1521 100644 --- a/src/api-proxy.js +++ b/src/api-proxy.js @@ -1,4 +1,3 @@ - const express = require('express'); const pathModule = require('path'); @@ -11,31 +10,38 @@ class ApiProxyApp { prepareMiddleware({ keystone }) { const app = express(); - - const apiProxy = createProxyMiddleware({ - target: this._gwaApiUrl, - changeOrigin: true, - pathRewrite: { '^/gw/api/': '/v2/' }, - onProxyReq: (proxyReq, req) => { - // console.log(req.headers) - // proxyReq.removeHeader("cookie"); - proxyReq.removeHeader("cookie"); - proxyReq.setHeader('Accept', 'application/json') - proxyReq.setHeader('Authorization', `Bearer ${req.header('x-forwarded-access-token')}`) }, - onError:(err, req, res, target) => { - console.log(err) - res.writeHead(400, { - 'Content-Type': 'text/plain', - }); - res.end('error reaching api'); - } - }) - app.all(/^\/gw\/api\//, apiProxy) + + const apiProxy = createProxyMiddleware({ + target: this._gwaApiUrl, + changeOrigin: true, + logLevel: 'debug', + pathRewrite: { '^/gw/api/': '/v2/' }, + onProxyReq: (proxyReq, req) => { + //console.log(req.headers) + // proxyReq.removeHeader("cookie"); + proxyReq.removeHeader('cookie'); + proxyReq.setHeader('Accept', 'application/json'); + proxyReq.setHeader( + 'Authorization', + 'authorization' in req.headers + ? req.header('authorization') + : `Bearer ${req.header('x-forwarded-access-token')}` + ); + }, + onError: (err, req, res, target) => { + console.log(err); + res.writeHead(400, { + 'Content-Type': 'text/plain', + }); + res.end('error reaching api'); + }, + }); + app.all(/^\/gw\/api\//, apiProxy); return app; } } module.exports = { - ApiProxyApp, -}; \ No newline at end of file + ApiProxyApp, +}; diff --git a/src/authz/graphql-whitelist/httplocalhost4180managernamespaces-85071c.gql b/src/authz/graphql-whitelist/httplocalhost4180managernamespaces-85071c.gql new file mode 100644 index 000000000..5fdb60eee --- /dev/null +++ b/src/authz/graphql-whitelist/httplocalhost4180managernamespaces-85071c.gql @@ -0,0 +1,7 @@ + + mutation CreateNamespace($name: String!) { + createNamespace(name: $name) { + id + name + } + } diff --git a/src/batch/data-rules.js b/src/batch/data-rules.js index 847310919..241e38969 100644 --- a/src/batch/data-rules.js +++ b/src/batch/data-rules.js @@ -198,7 +198,7 @@ const metadata = { Namespace: { query: 'allNamespaces', refKey: 'extRefId', - sync: ['name'], + sync: ['name', 'displayName'], transformations: { // members: { // name: 'connectExclusiveList', @@ -474,7 +474,7 @@ const metadata = { resourceScopes: { name: 'toStringDefaultArray' }, clientRoles: { name: 'toStringDefaultArray' }, clientMappers: { name: 'toStringDefaultArray' }, - environmentDetails: { name: 'toString' }, + environmentDetails: { name: 'toStringDefaultArray' }, inheritFrom: { name: 'connectOne', list: 'allSharedIdPs', diff --git a/src/batch/feed-worker.ts b/src/batch/feed-worker.ts index 76122ea68..b33db0978 100644 --- a/src/batch/feed-worker.ts +++ b/src/batch/feed-worker.ts @@ -476,9 +476,14 @@ export const syncRecords = async function ( } if (Object.keys(data).length === 0) { logger.debug('[%s] [%s] no update', entity, localRecord.id); + const firstChildResult = childResults + .filter((r) => r.result != 'no-change') + .pop(); return { status: 200, - result: 'no-change', + result: firstChildResult + ? firstChildResult.result + '-child' + : 'no-change', id: localRecord['id'], childResults, ownedBy: diff --git a/src/batch/transformations/toStringDefaultArray.ts b/src/batch/transformations/toStringDefaultArray.ts index 41973c2ae..a07fa9214 100644 --- a/src/batch/transformations/toStringDefaultArray.ts +++ b/src/batch/transformations/toStringDefaultArray.ts @@ -7,6 +7,10 @@ export function toStringDefaultArray( inputData: any, key: string ) { + // if new and not passed, then set an empty array as a default + if (inputData[key] == null && currentData == null) { + return '[]'; + } return inputData[key] == null || (currentData != null && currentData[key] === stringify(inputData[key])) ? null diff --git a/src/controllers/ioc/keystoneInjector.ts b/src/controllers/ioc/keystoneInjector.ts index 1d6e4b41f..a29f9dd4c 100644 --- a/src/controllers/ioc/keystoneInjector.ts +++ b/src/controllers/ioc/keystoneInjector.ts @@ -41,12 +41,14 @@ export class KeystoneService { public createContext(request: any, skipAccessControl: boolean = false): any { const _scopes = scopes(request.user.scope); + const identityProvider = request.user.identity_provider; + const identity = { id: null, name: resolveName(request.user), username: resolveUsername(request.user), namespace: request.params.ns, - roles: JSON.stringify(scopesToRoles(null, _scopes)), + roles: JSON.stringify(scopesToRoles(identityProvider, _scopes)), scopes: _scopes, userId: null, } as any; diff --git a/src/controllers/v2/GatewayController.ts b/src/controllers/v2/GatewayController.ts index 94a7f5c1c..79b0b7a54 100644 --- a/src/controllers/v2/GatewayController.ts +++ b/src/controllers/v2/GatewayController.ts @@ -9,6 +9,8 @@ import { Security, Body, Tags, + FormField, + UploadedFile, } from 'tsoa'; import { KeystoneService } from '../ioc/keystoneInjector'; import { inject, injectable } from 'tsyringe'; @@ -20,6 +22,7 @@ import { removeKeys, } from '../../batch/feed-worker'; import { GatewayRoute } from './types'; +import { PublishResult } from './types-extra'; @injectable() @Route('/namespaces/{ns}/gateway') @@ -31,6 +34,17 @@ export class GatewayController extends Controller { this.keystone = _keystone; } + @Put() + @OperationId('publish-gateway-config') + @Security('jwt', ['Gateway.Config']) + public async put( + @FormField() dryRun: boolean, + @UploadedFile() configFile: Express.Multer.File + ): Promise { + // stub - gwa-api implements this + return { error: 'Stub - not implemented' }; + } + /** * Get a summary of your Gateway Services * > `Required Scope:` Namespace.Manage diff --git a/src/controllers/v2/NamespaceController.ts b/src/controllers/v2/NamespaceController.ts index 5a1007aff..1ef0715d6 100644 --- a/src/controllers/v2/NamespaceController.ts +++ b/src/controllers/v2/NamespaceController.ts @@ -9,13 +9,15 @@ import { Tags, Delete, Query, + Post, + Body, } from 'tsoa'; import { ValidateError, FieldErrors } from 'tsoa'; import { KeystoneService } from '../ioc/keystoneInjector'; import { inject, injectable } from 'tsyringe'; import { gql } from 'graphql-request'; import { WorkbookService } from '../../services/report/workbook.service'; -import { Namespace } from '../../services/keystone/types'; +import { Namespace, NamespaceInput } from '../../services/keystone/types'; import { Readable } from 'stream'; import { @@ -33,6 +35,7 @@ import { Activity } from './types'; import { getActivity } from '../../services/keystone/activity'; import { transformActivity } from '../../services/workflow'; import { ActivityDetail } from './types-extra'; + const logger = Logger('controllers.Namespace'); /** @@ -107,7 +110,7 @@ export class NamespaceController extends Controller { query: list, }); logger.debug('Result %j', result); - return result.data.allNamespaces.map((ns: Namespace) => ns.name); + return result.data.allNamespaces.map((ns: Namespace) => ns.name).sort(); } /** @@ -136,6 +139,40 @@ export class NamespaceController extends Controller { return result.data.namespace; } + /** + * Create a namespace + * + * @summary Create Namespace + * @param ns + * @param request + * @returns + */ + @Post() + @OperationId('create-namespace') + @Security('jwt', []) + public async create( + @Request() request: any, + @Body() vars: NamespaceInput + ): Promise { + const result = await this.keystone.executeGraphQL({ + context: this.keystone.createContext(request), + query: createNS, + variables: vars, + }); + logger.debug('Result %j', result); + if (result.errors) { + const errors: FieldErrors = {}; + result.errors.forEach((err: any, ind: number) => { + errors[`d${ind}`] = { message: err.message }; + }); + throw new ValidateError(errors, 'Unable to create namespace'); + } + return { + name: result.data.createNamespace.name, + displayName: result.data.createNamespace.displayName, + }; + } + /** * Delete the namespace * > `Required Scope:` Namespace.Manage @@ -214,6 +251,7 @@ const item = gql` query Namespace($ns: String!) { namespace(ns: $ns) { name + displayName scopes { name } @@ -234,3 +272,12 @@ const deleteNS = gql` forceDeleteNamespace(namespace: $ns, force: $force) } `; + +const createNS = gql` + mutation CreateNamespace($name: String, $displayName: String) { + createNamespace(name: $name, displayName: $displayName) { + name + displayName + } + } +`; diff --git a/src/controllers/v2/openapi.yaml b/src/controllers/v2/openapi.yaml index 5c6ad39cd..306e8d3ac 100644 --- a/src/controllers/v2/openapi.yaml +++ b/src/controllers/v2/openapi.yaml @@ -188,6 +188,16 @@ components: - tag2 organization: ministry-of-citizens-services organizationUnit: databc + PublishResult: + properties: + message: + type: string + results: + type: string + error: + type: string + type: object + additionalProperties: false GatewayServiceRefID: type: string GatewayRouteRefID: @@ -277,9 +287,7 @@ components: clientRegistration: managed clientId: a-client-id clientSecret: a-client-secret - CredentialIssuerRefID: - type: string - UserRefID: + undefinedRefID: type: string CredentialIssuer: properties: @@ -338,9 +346,9 @@ components: type: string type: array inheritFrom: - $ref: '#/components/schemas/CredentialIssuerRefID' + $ref: '#/components/schemas/undefinedRefID' owner: - $ref: '#/components/schemas/UserRefID' + $ref: '#/components/schemas/undefinedRefID' type: object additionalProperties: false example: @@ -350,7 +358,7 @@ components: clientAuthenticator: client-secret mode: auto environmentDetails: [] - owner: acope@idir + owner: janis@gov.bc.ca Maybe_Scalars-at-String_: type: string nullable: true @@ -414,6 +422,8 @@ components: $ref: '#/components/schemas/Maybe_Scalars-at-String_' scopes: $ref: '#/components/schemas/Maybe_Array_Maybe_UmaScope___' + displayName: + $ref: '#/components/schemas/Maybe_Scalars-at-String_' name: type: string id: @@ -426,6 +436,13 @@ components: required: - name type: object + NamespaceInput: + properties: + displayName: + $ref: '#/components/schemas/Maybe_Scalars-at-String_' + name: + $ref: '#/components/schemas/Maybe_Scalars-at-String_' + type: object ActivityDetail: properties: id: @@ -539,6 +556,8 @@ components: type: string LegalRefID: type: string + CredentialIssuerRefID: + type: string Environment: properties: appId: @@ -621,6 +640,10 @@ components: description: 'Authz Portal Login' scheme: bearer bearerFormat: JWT + openid: + type: openIdConnect + description: 'OIDC Login' + openIdConnectUrl: 'https://well_known_endpoint' info: title: 'APS Directory API' version: 1.1.0 @@ -961,6 +984,37 @@ paths: schema: type: string '/namespaces/{ns}/gateway': + put: + operationId: publish-gateway-config + responses: + '200': + description: Ok + content: + application/json: + schema: + $ref: '#/components/schemas/PublishResult' + tags: + - 'Gateway Services' + security: + - + jwt: + - Gateway.Config + parameters: [] + requestBody: + required: true + content: + multipart/form-data: + schema: + type: object + properties: + dryRun: + type: string + configFile: + type: string + format: binary + required: + - dryRun + - configFile get: operationId: get-gateway-routes responses: @@ -1140,6 +1194,29 @@ paths: - jwt: [] parameters: [] + post: + operationId: create-namespace + responses: + '200': + description: Ok + content: + application/json: + schema: + $ref: '#/components/schemas/Namespace' + description: 'Create a namespace' + summary: 'Create Namespace' + tags: + - Namespaces + security: + - + jwt: [] + parameters: [] + requestBody: + required: true + content: + application/json: + schema: + $ref: '#/components/schemas/NamespaceInput' '/namespaces/{ns}': get: operationId: namespace-profile diff --git a/src/controllers/v2/routes.ts b/src/controllers/v2/routes.ts index 21bb4b1ec..9f05c5aee 100644 --- a/src/controllers/v2/routes.ts +++ b/src/controllers/v2/routes.ts @@ -34,6 +34,8 @@ const promiseAny = require('promise.any'); import { iocContainer } from './../ioc'; import { IocContainer, IocContainerFactory } from '@tsoa/runtime'; import * as express from 'express'; +const multer = require('multer'); +const upload = multer(); // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa @@ -128,6 +130,16 @@ const models: TsoaRoute.Models = { "additionalProperties": false, }, // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa + "PublishResult": { + "dataType": "refObject", + "properties": { + "message": {"dataType":"string"}, + "results": {"dataType":"string"}, + "error": {"dataType":"string"}, + }, + "additionalProperties": false, + }, + // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa "GatewayServiceRefID": { "dataType": "refAlias", "type": {"dataType":"string","validators":{}}, @@ -185,12 +197,7 @@ const models: TsoaRoute.Models = { "additionalProperties": false, }, // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa - "CredentialIssuerRefID": { - "dataType": "refAlias", - "type": {"dataType":"string","validators":{}}, - }, - // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa - "UserRefID": { + "undefinedRefID": { "dataType": "refAlias", "type": {"dataType":"string","validators":{}}, }, @@ -215,8 +222,8 @@ const models: TsoaRoute.Models = { "resourceScopes": {"dataType":"array","array":{"dataType":"string"}}, "clientRoles": {"dataType":"array","array":{"dataType":"string"}}, "clientMappers": {"dataType":"array","array":{"dataType":"string"}}, - "inheritFrom": {"ref":"CredentialIssuerRefID"}, - "owner": {"ref":"UserRefID"}, + "inheritFrom": {"ref":"undefinedRefID"}, + "owner": {"ref":"undefinedRefID"}, }, "additionalProperties": false, }, @@ -263,7 +270,12 @@ const models: TsoaRoute.Models = { // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa "Namespace": { "dataType": "refAlias", - "type": {"dataType":"nestedObjectLiteral","nestedProperties":{"orgAdmins":{"ref":"Maybe_Array_Maybe_Scalars-at-String___"},"orgNoticeViewed":{"ref":"Maybe_Scalars-at-Boolean_"},"orgEnabled":{"ref":"Maybe_Scalars-at-Boolean_"},"orgUpdatedAt":{"ref":"Maybe_Scalars-at-Float_"},"orgUnit":{"ref":"Maybe_Scalars-at-JSON_"},"org":{"ref":"Maybe_Scalars-at-JSON_"},"permProtectedNs":{"ref":"Maybe_Scalars-at-String_"},"permDataPlane":{"ref":"Maybe_Scalars-at-String_"},"permDomains":{"ref":"Maybe_Array_Maybe_Scalars-at-String___"},"prodEnvId":{"ref":"Maybe_Scalars-at-String_"},"scopes":{"ref":"Maybe_Array_Maybe_UmaScope___"},"name":{"dataType":"string","required":true},"id":{"ref":"Maybe_Scalars-at-String_"},"__typename":{"dataType":"enum","enums":["Namespace"]}},"validators":{}}, + "type": {"dataType":"nestedObjectLiteral","nestedProperties":{"orgAdmins":{"ref":"Maybe_Array_Maybe_Scalars-at-String___"},"orgNoticeViewed":{"ref":"Maybe_Scalars-at-Boolean_"},"orgEnabled":{"ref":"Maybe_Scalars-at-Boolean_"},"orgUpdatedAt":{"ref":"Maybe_Scalars-at-Float_"},"orgUnit":{"ref":"Maybe_Scalars-at-JSON_"},"org":{"ref":"Maybe_Scalars-at-JSON_"},"permProtectedNs":{"ref":"Maybe_Scalars-at-String_"},"permDataPlane":{"ref":"Maybe_Scalars-at-String_"},"permDomains":{"ref":"Maybe_Array_Maybe_Scalars-at-String___"},"prodEnvId":{"ref":"Maybe_Scalars-at-String_"},"scopes":{"ref":"Maybe_Array_Maybe_UmaScope___"},"displayName":{"ref":"Maybe_Scalars-at-String_"},"name":{"dataType":"string","required":true},"id":{"ref":"Maybe_Scalars-at-String_"},"__typename":{"dataType":"enum","enums":["Namespace"]}},"validators":{}}, + }, + // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa + "NamespaceInput": { + "dataType": "refAlias", + "type": {"dataType":"nestedObjectLiteral","nestedProperties":{"displayName":{"ref":"Maybe_Scalars-at-String_"},"name":{"ref":"Maybe_Scalars-at-String_"}},"validators":{}}, }, // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa "ActivityDetail": { @@ -355,6 +367,11 @@ const models: TsoaRoute.Models = { "type": {"dataType":"string","validators":{}}, }, // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa + "CredentialIssuerRefID": { + "dataType": "refAlias", + "type": {"dataType":"string","validators":{}}, + }, + // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa "Environment": { "dataType": "refObject", "properties": { @@ -780,6 +797,37 @@ export function RegisterRoutes(app: express.Router) { } }); // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa + app.put('/ds/api/v2/namespaces/:ns/gateway', + authenticateMiddleware([{"jwt":["Gateway.Config"]}]), + upload.single('configFile'), + + async function GatewayController_put(request: any, response: any, next: any) { + const args = { + dryRun: {"in":"formData","name":"dryRun","required":true,"dataType":"string"}, + configFile: {"in":"formData","name":"configFile","required":true,"dataType":"file"}, + }; + + // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa + + let validatedArgs: any[] = []; + try { + validatedArgs = getValidatedArgs(args, request, response); + + const container: IocContainer = typeof iocContainer === 'function' ? (iocContainer as IocContainerFactory)(request) : iocContainer; + + const controller: any = await container.get(GatewayController); + if (typeof controller['setStatus'] === 'function') { + controller.setStatus(undefined); + } + + + const promise = controller.put.apply(controller, validatedArgs as any); + promiseHandler(controller, promise, response, undefined, next); + } catch (err) { + return next(err); + } + }); + // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa app.get('/ds/api/v2/namespaces/:ns/gateway', authenticateMiddleware([{"jwt":["Namespace.Manage"]}]), @@ -1019,6 +1067,36 @@ export function RegisterRoutes(app: express.Router) { } }); // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa + app.post('/ds/api/v2/namespaces', + authenticateMiddleware([{"jwt":[]}]), + + async function NamespaceController_create(request: any, response: any, next: any) { + const args = { + request: {"in":"request","name":"request","required":true,"dataType":"object"}, + vars: {"in":"body","name":"vars","required":true,"ref":"NamespaceInput"}, + }; + + // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa + + let validatedArgs: any[] = []; + try { + validatedArgs = getValidatedArgs(args, request, response); + + const container: IocContainer = typeof iocContainer === 'function' ? (iocContainer as IocContainerFactory)(request) : iocContainer; + + const controller: any = await container.get(NamespaceController); + if (typeof controller['setStatus'] === 'function') { + controller.setStatus(undefined); + } + + + const promise = controller.create.apply(controller, validatedArgs as any); + promiseHandler(controller, promise, response, undefined, next); + } catch (err) { + return next(err); + } + }); + // WARNING: This file was auto-generated with tsoa. Please do not modify it. Re-run tsoa to re-generate this file: https://github.com/lukeautry/tsoa app.delete('/ds/api/v2/namespaces/:ns', authenticateMiddleware([{"jwt":["Namespace.Manage"]}]), diff --git a/src/controllers/v2/types-extra.ts b/src/controllers/v2/types-extra.ts index ecc3eede2..438ddde0d 100644 --- a/src/controllers/v2/types-extra.ts +++ b/src/controllers/v2/types-extra.ts @@ -10,3 +10,12 @@ export interface ActivityDetail { activityAt: Scalars['DateTime']; blob?: any; } + +/** + * @tsoaModel + */ +export interface PublishResult { + message?: string; + results?: string; + error?: string; +} diff --git a/src/controllers/v2/types.ts b/src/controllers/v2/types.ts index 7e4c9839c..366bbdbfb 100644 --- a/src/controllers/v2/types.ts +++ b/src/controllers/v2/types.ts @@ -134,6 +134,7 @@ export interface Alert { export interface Namespace { extRefId?: string; // Primary Key name?: string; + displayName?: string; } @@ -315,7 +316,7 @@ export interface Environment { * "clientAuthenticator": "client-secret", * "mode": "auto", * "environmentDetails": [], - * "owner": "acope@idir" + * "owner": "janis@gov.bc.ca" * } */ export interface CredentialIssuer { @@ -336,8 +337,8 @@ export interface CredentialIssuer { resourceScopes?: string[]; clientRoles?: string[]; clientMappers?: string[]; - inheritFrom?: CredentialIssuerRefID; - owner?: UserRefID; + inheritFrom?: undefinedRefID; + owner?: undefinedRefID; } @@ -477,6 +478,7 @@ export interface User { name?: string; email?: string; legalsAgreed?: UserLegalsAgreed[]; + provider?: string; } @@ -554,3 +556,8 @@ export type OrganizationUnitRefID = string * @tsoaModel */ export type UserRefID = string + +/** + * @tsoaModel + */ +export type undefinedRefID = string diff --git a/src/lists/CredentialIssuer.js b/src/lists/CredentialIssuer.js index 1b51a1103..6147b2c5b 100644 --- a/src/lists/CredentialIssuer.js +++ b/src/lists/CredentialIssuer.js @@ -26,9 +26,10 @@ const { const { syncSharedIdp, addClientsToSharedIdP, + DeleteClientsFromSharedIdP, } = require('../services/workflow'); const { Logger } = require('../logger'); -const { kebabCase } = require('lodash'); +const kebabCase = require('just-kebab-case'); const logger = Logger('lists.credentialissuer'); module.exports = { @@ -249,6 +250,12 @@ module.exports = { }, afterDelete: async function ({ existingItem, context }) { + await DeleteClientsFromSharedIdP( + context, + existingItem.clientId, + `${existingItem.inheritFrom}` + ); + await new StructuredActivityService( context, context.authedItem['namespace'] diff --git a/src/lists/extensions/CredentialIssuerExt.ts b/src/lists/extensions/CredentialIssuerExt.ts index f44463ff1..5913f0e27 100644 --- a/src/lists/extensions/CredentialIssuerExt.ts +++ b/src/lists/extensions/CredentialIssuerExt.ts @@ -1,5 +1,5 @@ const { EnforcementPoint } = require('../../authz/enforcement'); -import { kebabCase } from 'lodash'; +import kebabCase from 'just-kebab-case'; import { generateEnvDetails, lookupSharedIssuers, diff --git a/src/lists/extensions/Namespace.ts b/src/lists/extensions/Namespace.ts index 9bb8b4e7c..59c815c5c 100644 --- a/src/lists/extensions/Namespace.ts +++ b/src/lists/extensions/Namespace.ts @@ -7,7 +7,6 @@ import { ResourceSetInput, } from '../../services/uma2'; import { - getOrganizationUnit, lookupProductEnvironmentServicesBySlug, lookupUsersByUsernames, recordActivity, @@ -18,7 +17,6 @@ import { getNamespaceResourceSets, isUserBasedResourceOwners, doClientLoginForCredentialIssuer, - EnvironmentContext, getOrgPoliciesForResource, } from './Common'; import type { TokenExchangeResult } from './Common'; @@ -32,17 +30,12 @@ import { DeleteNamespaceValidate, } from '../../services/workflow/delete-namespace'; import { GWAService } from '../../services/gwaapi'; -import { - camelCaseAttributes, - transformSingleValueAttributes, -} from '../../services/utils'; +import { regExprValidation } from '../../services/utils'; import getSubjectToken from '../../auth/auth-token'; import { GroupAccessService, NamespaceService, } from '../../services/org-groups'; -import { IssuerEnvironmentConfig } from '../../services/workflow/types'; -import { Keystone } from '@keystonejs/keystone'; import { Logger } from '../../logger'; import { getGwaProductEnvironment } from '../../services/workflow'; import { NotificationService } from '../../services/notification/notification.service'; @@ -54,6 +47,7 @@ import { getResource, transformOrgAndOrgUnit, } from '../../services/keycloak/namespace-details'; +import { newNamespaceID } from '../../services/identifiers'; const logger = Logger('ext.Namespace'); @@ -69,6 +63,7 @@ const typeNamespace = ` type Namespace { id: String name: String!, + displayName: String, scopes: [UMAScope], prodEnvId: String, permDomains: [String], @@ -85,7 +80,8 @@ type Namespace { const typeNamespaceInput = ` input NamespaceInput { - name: String!, + name: String, + displayName: String } `; @@ -422,7 +418,8 @@ module.exports = { }, }, { - schema: 'createNamespace(namespace: String!): Namespace', + schema: + 'createNamespace(name: String, displayName: String): Namespace', resolver: async ( item: any, args: any, @@ -431,11 +428,13 @@ module.exports = { { query, access }: any ) => { const namespaceValidationRule = '^[a-z][a-z0-9-]{4,14}$'; - const re = new RegExp(namespaceValidationRule); - assert.strictEqual( - re.test(args.namespace), - true, - 'Namespace name must be between 5 and 15 alpha-numeric lowercase characters and begin with an alphabet.' + + const newNS = args.name ? args.name : newNamespaceID(); + + regExprValidation( + namespaceValidationRule, + newNS, + 'Namespace name must be between 5 and 15 alpha-numeric lowercase characters and start and end with an alphabet.' ); const noauthContext = context.createContext({ @@ -458,7 +457,7 @@ module.exports = { envCtx.issuerEnvConfig.clientId, envCtx.issuerEnvConfig.clientSecret ); - await nsService.checkNamespaceAvailable(args.namespace); + await nsService.checkNamespaceAvailable(newNS); // This function gets all resources but also sets the accessToken in envCtx // which we need to create the resource set @@ -478,7 +477,8 @@ module.exports = { 'CredentialIssuer.Admin', ]; const res = { - name: args.namespace, + name: newNS, + displayName: args.displayName, type: 'namespace', resource_scopes: scopes, ownerManagedAccess: true, @@ -491,12 +491,18 @@ module.exports = { envCtx.issuerEnvConfig.issuerUrl, envCtx.accessToken ); - await permissionApi.createPermission( - rset.id, - envCtx.subjectUuid, - true, - 'Namespace.Manage' - ); + for (const scope of [ + 'Namespace.Manage', + 'CredentialIssuer.Admin', + 'GatewayConfig.Publish', + ]) { + await permissionApi.createPermission( + rset.id, + envCtx.subjectUuid, + true, + scope + ); + } } const kcGroupService = new KeycloakGroupService( @@ -507,27 +513,24 @@ module.exports = { envCtx.issuerEnvConfig.clientSecret ); - await kcGroupService.createIfMissing('ns', args.namespace); + await kcGroupService.createIfMissing('ns', newNS); await recordActivity( context.sudo(), 'create', 'Namespace', - args.namespace, - `Created ${args.namespace} namespace`, + newNS, + `Created ${newNS} namespace`, 'success', JSON.stringify({ message: '{actor} created {ns} namespace', params: { actor: context.authedItem.name, - ns: args.namespace, + ns: newNS, }, }), - args.namespace, - [ - `Namespace:${args.namespace}`, - `actor:${context.authedItem.name}`, - ] + newNS, + [`Namespace:${newNS}`, `actor:${context.authedItem.name}`] ); return rset; diff --git a/src/lists/extensions/ServiceAccount.ts b/src/lists/extensions/ServiceAccount.ts index f2aa9917e..ed50e19ee 100644 --- a/src/lists/extensions/ServiceAccount.ts +++ b/src/lists/extensions/ServiceAccount.ts @@ -81,11 +81,13 @@ module.exports = { info: any, { query, access }: any ) => { - context.skipAccessControl = true; + const noauthContext = context.createContext({ + skipAccessControl: true, + }); const productEnvironmentSlug = process.env.GWA_PROD_ENV_SLUG; const { newCredentials } = await CreateServiceAccount( - context, + noauthContext, productEnvironmentSlug, context.req.user.namespace, args.resourceId, diff --git a/src/lists/extensions/UserExt.ts b/src/lists/extensions/UserExt.ts index 76b33e03d..6d3784ab4 100644 --- a/src/lists/extensions/UserExt.ts +++ b/src/lists/extensions/UserExt.ts @@ -1,16 +1,6 @@ const { EnforcementPoint } = require('../../authz/enforcement'); import { lookupProviderUserByEmail } from '../../services/keystone/user'; -import { kebabCase } from 'lodash'; -import { - generateEnvDetails, - lookupSharedIssuers, -} from '../../services/keystone'; -import { - CredentialIssuer, - CredentialIssuerWhereInput, - User, - UserWhereInput, -} from '../../services/keystone/types'; +import { User, UserWhereInput } from '../../services/keystone/types'; module.exports = { extensions: [ diff --git a/src/nextapp/components/new-namespace/new-namespace.tsx b/src/nextapp/components/new-namespace/new-namespace.tsx index 9ed28b46b..bc376436a 100644 --- a/src/nextapp/components/new-namespace/new-namespace.tsx +++ b/src/nextapp/components/new-namespace/new-namespace.tsx @@ -133,7 +133,7 @@ export default NewNamespace; const mutation = gql` mutation CreateNamespace($name: String!) { - createNamespace(namespace: $name) { + createNamespace(name: $name) { id name } diff --git a/src/nextapp/shared/types/query.types.ts b/src/nextapp/shared/types/query.types.ts index d34a32302..5d5de8201 100644 --- a/src/nextapp/shared/types/query.types.ts +++ b/src/nextapp/shared/types/query.types.ts @@ -5326,7 +5326,8 @@ export type MutationUpdateCurrentNamespaceArgs = { export type MutationCreateNamespaceArgs = { - namespace: Scalars['String']; + name?: Maybe; + displayName?: Maybe; }; @@ -5415,6 +5416,7 @@ export type Namespace = { __typename?: 'Namespace'; id?: Maybe; name: Scalars['String']; + displayName?: Maybe; scopes?: Maybe>>; prodEnvId?: Maybe; permDomains?: Maybe>>; @@ -5429,7 +5431,8 @@ export type Namespace = { }; export type NamespaceInput = { - name: Scalars['String']; + name?: Maybe; + displayName?: Maybe; }; /** A keystone list */ diff --git a/src/package-lock.json b/src/package-lock.json index bd8cafbc2..3e70e7aa5 100644 --- a/src/package-lock.json +++ b/src/package-lock.json @@ -50,7 +50,7 @@ "graphql": "^15.4.0", "graphql-request": "^3.4.0", "graphql-tools": "^7.0.2", - "http-proxy-middleware": "^1.1.2", + "http-proxy-middleware": "^2.0.6", "isomorphic-unfetch": "^3.1.0", "js-yaml": "^4.1.0", "json-stable-stringify": "^1.0.2", @@ -15831,9 +15831,9 @@ "integrity": "sha512-EqX+YQxINb+MeXaIqYDASb6U6FCHbWjkj4a1CKDBks3d/QiB2+PqBLyO72vLDgAO1wUI4O+9gweRcQK11bTL/w==" }, "node_modules/@types/http-proxy": { - "version": "1.17.5", - "resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.5.tgz", - "integrity": "sha512-GNkDE7bTv6Sf8JbV2GksknKOsk7OznNYHSdrtvPJXO0qJ9odZig6IZKUi5RFGi6d1bf6dgIAe4uXi3DBc7069Q==", + "version": "1.17.11", + "resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.11.tgz", + "integrity": "sha512-HC8G7c1WmaF2ekqpnFq626xd3Zz0uvaqFmBJNRZCGEZCXkvSdJoNFn/8Ygbd9fKNQj8UzLdCETaI0UWPAjK7IA==", "dependencies": { "@types/node": "*" } @@ -28528,18 +28528,26 @@ } }, "node_modules/http-proxy-middleware": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-1.1.2.tgz", - "integrity": "sha512-YRFUeOG3q85FJjAaYVJUoNRW9a73SDlOtAyQOS5PHLr18QeZ/vEhxywNoOPiEO8BxCegz4RXzTHcvyLEGB78UA==", + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-2.0.6.tgz", + "integrity": "sha512-ya/UeJ6HVBYxrgYotAZo1KvPWlgB48kUJLDePFeneHsVujFaW5WNj2NgWCAE//B1Dl02BIfYlpNgBy8Kf8Rjmw==", "dependencies": { - "@types/http-proxy": "^1.17.5", + "@types/http-proxy": "^1.17.8", "http-proxy": "^1.18.1", "is-glob": "^4.0.1", "is-plain-obj": "^3.0.0", "micromatch": "^4.0.2" }, "engines": { - "node": ">=8.0.0" + "node": ">=12.0.0" + }, + "peerDependencies": { + "@types/express": "^4.17.13" + }, + "peerDependenciesMeta": { + "@types/express": { + "optional": true + } } }, "node_modules/http2-wrapper": { @@ -49255,7 +49263,8 @@ "@apollographql/apollo-tools": { "version": "0.5.3", "resolved": "https://registry.npmjs.org/@apollographql/apollo-tools/-/apollo-tools-0.5.3.tgz", - "integrity": "sha512-VcsXHfTFoCodDAgJZxN04GdFK1kqOhZQnQY/9Fa147P+I8xfvOSz5d+lKAPB+hwSgBNyd7ncAKGIs4+utbL+yA==" + "integrity": "sha512-VcsXHfTFoCodDAgJZxN04GdFK1kqOhZQnQY/9Fa147P+I8xfvOSz5d+lKAPB+hwSgBNyd7ncAKGIs4+utbL+yA==", + "requires": {} }, "@apollographql/graphql-playground-html": { "version": "1.6.29", @@ -51676,7 +51685,8 @@ "@chakra-ui/css-reset": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/@chakra-ui/css-reset/-/css-reset-1.0.0.tgz", - "integrity": "sha512-UaPsImGHvCgFO3ayp6Ugafu2/3/EG8wlW/8Y9Ihfk1UFv8cpV+3BfWKmuZ7IcmxcBL9dkP6E8p3/M1T0FB92hg==" + "integrity": "sha512-UaPsImGHvCgFO3ayp6Ugafu2/3/EG8wlW/8Y9Ihfk1UFv8cpV+3BfWKmuZ7IcmxcBL9dkP6E8p3/M1T0FB92hg==", + "requires": {} }, "@chakra-ui/descendant": { "version": "2.0.0", @@ -53410,7 +53420,8 @@ "@graphql-typed-document-node/core": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/@graphql-typed-document-node/core/-/core-3.1.0.tgz", - "integrity": "sha512-wYn6r8zVZyQJ6rQaALBEln5B1pzxb9shV5Ef97kTvn6yVGrqyXVnDqnU24MXnFubR+rZjBY9NWuxX3FB2sTsjg==" + "integrity": "sha512-wYn6r8zVZyQJ6rQaALBEln5B1pzxb9shV5Ef97kTvn6yVGrqyXVnDqnU24MXnFubR+rZjBY9NWuxX3FB2sTsjg==", + "requires": {} }, "@hapi/accept": { "version": "5.0.1", @@ -57575,7 +57586,8 @@ "slate-react-placeholder": { "version": "0.2.9", "resolved": "https://registry.npmjs.org/slate-react-placeholder/-/slate-react-placeholder-0.2.9.tgz", - "integrity": "sha512-YSJ9Gb4tGpbzPje3eNKtu26hWM8ApxTk9RzjK+6zfD5V/RMTkuWONk24y6c9lZk0OAYNZNUmrnb/QZfU3j9nag==" + "integrity": "sha512-YSJ9Gb4tGpbzPje3eNKtu26hWM8ApxTk9RzjK+6zfD5V/RMTkuWONk24y6c9lZk0OAYNZNUmrnb/QZfU3j9nag==", + "requires": {} } } }, @@ -57655,7 +57667,8 @@ "react-codemirror2": { "version": "7.2.1", "resolved": "https://registry.npmjs.org/react-codemirror2/-/react-codemirror2-7.2.1.tgz", - "integrity": "sha512-t7YFmz1AXdlImgHXA9Ja0T6AWuopilub24jRaQdPVbzUJVNKIYuy3uCFZYa7CE5S3UW6SrSa5nAqVQvtzRF9gw==" + "integrity": "sha512-t7YFmz1AXdlImgHXA9Ja0T6AWuopilub24jRaQdPVbzUJVNKIYuy3uCFZYa7CE5S3UW6SrSa5nAqVQvtzRF9gw==", + "requires": {} } } }, @@ -58302,7 +58315,8 @@ "version": "1.6.22", "resolved": "https://registry.npmjs.org/@mdx-js/react/-/react-1.6.22.tgz", "integrity": "sha512-TDoPum4SHdfPiGSAaRBw7ECyI8VaHpK8GJugbJIJuqyh6kzw9ZLJZW3HGL3NNrJGxcAixUvqROm+YuQOo5eXtg==", - "dev": true + "dev": true, + "requires": {} }, "@mdx-js/util": { "version": "1.6.22", @@ -58358,7 +58372,8 @@ "@n1ru4l/graphql-live-query": { "version": "0.9.0", "resolved": "https://registry.npmjs.org/@n1ru4l/graphql-live-query/-/graphql-live-query-0.9.0.tgz", - "integrity": "sha512-BTpWy1e+FxN82RnLz4x1+JcEewVdfmUhV1C6/XYD5AjS7PQp9QFF7K8bCD6gzPTr2l+prvqOyVueQhFJxB1vfg==" + "integrity": "sha512-BTpWy1e+FxN82RnLz4x1+JcEewVdfmUhV1C6/XYD5AjS7PQp9QFF7K8bCD6gzPTr2l+prvqOyVueQhFJxB1vfg==", + "requires": {} }, "@next/env": { "version": "9.5.5", @@ -58546,7 +58561,8 @@ "@primer/octicons-react": { "version": "11.3.0", "resolved": "https://registry.npmjs.org/@primer/octicons-react/-/octicons-react-11.3.0.tgz", - "integrity": "sha512-4sVhkrBKuj3h+PFw69yOyO/l3nQB/mm95V+Kz7LRSlIrbZr6hZarZD5Ft4ewdONPROkIHQM/6KSK90+OAimxsQ==" + "integrity": "sha512-4sVhkrBKuj3h+PFw69yOyO/l3nQB/mm95V+Kz7LRSlIrbZr6hZarZD5Ft4ewdONPROkIHQM/6KSK90+OAimxsQ==", + "requires": {} }, "@prisma/bar": { "version": "0.0.1", @@ -59851,7 +59867,8 @@ "version": "8.5.0", "resolved": "https://registry.npmjs.org/ws/-/ws-8.5.0.tgz", "integrity": "sha512-BWX0SWVgLPzYwF8lTzEy1egjhS4S4OEAHfsO8o65WOVsrnSRGaSiUaa9e0ggGlkMTtBlmOpEXiie9RUcBO86qg==", - "dev": true + "dev": true, + "requires": {} } } }, @@ -60202,7 +60219,8 @@ "version": "2.2.2", "resolved": "https://registry.npmjs.org/react-docgen-typescript/-/react-docgen-typescript-2.2.2.tgz", "integrity": "sha512-tvg2ZtOpOi6QDwsb3GZhOjDkkX0h8Z2gipvTg6OVMUyoYoURhEiRNePT8NZItTVCDh39JJHnLdfCOkzoLbFnTg==", - "dev": true + "dev": true, + "requires": {} } } }, @@ -61128,9 +61146,9 @@ "integrity": "sha512-EqX+YQxINb+MeXaIqYDASb6U6FCHbWjkj4a1CKDBks3d/QiB2+PqBLyO72vLDgAO1wUI4O+9gweRcQK11bTL/w==" }, "@types/http-proxy": { - "version": "1.17.5", - "resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.5.tgz", - "integrity": "sha512-GNkDE7bTv6Sf8JbV2GksknKOsk7OznNYHSdrtvPJXO0qJ9odZig6IZKUi5RFGi6d1bf6dgIAe4uXi3DBc7069Q==", + "version": "1.17.11", + "resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.11.tgz", + "integrity": "sha512-HC8G7c1WmaF2ekqpnFq626xd3Zz0uvaqFmBJNRZCGEZCXkvSdJoNFn/8Ygbd9fKNQj8UzLdCETaI0UWPAjK7IA==", "requires": { "@types/node": "*" } @@ -62358,7 +62376,8 @@ "version": "5.3.1", "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.1.tgz", "integrity": "sha512-K0Ptm/47OKfQRpNQ2J/oIN/3QYiK6FwW+eJbILhsdxh2WTLdl+30o8aGdTbm5JbffpFFAg/g+zi1E+jvJha5ng==", - "dev": true + "dev": true, + "requires": {} }, "acorn-node": { "version": "1.8.2", @@ -62479,12 +62498,14 @@ "ajv-errors": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/ajv-errors/-/ajv-errors-1.0.1.tgz", - "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==" + "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==", + "requires": {} }, "ajv-keywords": { "version": "3.5.2", "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", - "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==" + "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", + "requires": {} }, "ally.js": { "version": "1.4.1", @@ -62782,7 +62803,8 @@ "apollo-server-errors": { "version": "2.5.0", "resolved": "https://registry.npmjs.org/apollo-server-errors/-/apollo-server-errors-2.5.0.tgz", - "integrity": "sha512-lO5oTjgiC3vlVg2RKr3RiXIIQ5pGXBFxYGGUkKDhTud3jMIhs+gel8L8zsEjKaKxkjHhCQAA/bcEfYiKkGQIvA==" + "integrity": "sha512-lO5oTjgiC3vlVg2RKr3RiXIIQ5pGXBFxYGGUkKDhTud3jMIhs+gel8L8zsEjKaKxkjHhCQAA/bcEfYiKkGQIvA==", + "requires": {} }, "apollo-server-express": { "version": "2.25.3", @@ -63660,7 +63682,8 @@ "version": "0.3.7", "resolved": "https://registry.npmjs.org/babel-plugin-named-asset-import/-/babel-plugin-named-asset-import-0.3.7.tgz", "integrity": "sha512-squySRkf+6JGnvjoUtDEjSREJEBirnXi9NqP6rjSYsylxQxqBTz+pkmf395i9E2zsvmYUaI40BHo6SqZUdydlw==", - "dev": true + "dev": true, + "requires": {} }, "babel-plugin-polyfill-corejs2": { "version": "0.3.1", @@ -66072,7 +66095,8 @@ "create-react-context": { "version": "0.1.6", "resolved": "https://registry.npmjs.org/create-react-context/-/create-react-context-0.1.6.tgz", - "integrity": "sha512-eCnYYEUEc5i32LHwpE/W7NlddOB9oHwsPaWtWzYtflNkkwa3IfindIcoXdVWs12zCbwaMCavKNu84EXogVIWHw==" + "integrity": "sha512-eCnYYEUEc5i32LHwpE/W7NlddOB9oHwsPaWtWzYtflNkkwa3IfindIcoXdVWs12zCbwaMCavKNu84EXogVIWHw==", + "requires": {} }, "create-require": { "version": "1.1.1", @@ -66235,7 +66259,8 @@ "icss-utils": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-5.1.0.tgz", - "integrity": "sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==" + "integrity": "sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==", + "requires": {} }, "lru-cache": { "version": "7.8.0", @@ -66255,7 +66280,8 @@ "postcss-modules-extract-imports": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-3.0.0.tgz", - "integrity": "sha512-bdHleFnP3kZ4NYDhuGlVK+CMrQ/pqUm8bx/oGL93K6gVwiclvX5x0n76fYMKuIGKzlABOy13zsvqjb0f92TEXw==" + "integrity": "sha512-bdHleFnP3kZ4NYDhuGlVK+CMrQ/pqUm8bx/oGL93K6gVwiclvX5x0n76fYMKuIGKzlABOy13zsvqjb0f92TEXw==", + "requires": {} }, "postcss-modules-local-by-default": { "version": "4.0.0", @@ -67877,7 +67903,8 @@ "version": "7.1.0", "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-7.1.0.tgz", "integrity": "sha512-9sm5/PxaFG7qNJvJzTROMM1Bk1ozXVTKI0buKOyb0Bsr1hrwi0H/TzxF/COtf1uxikIK8SwhX7K6zg78jAzbeA==", - "dev": true + "dev": true, + "requires": {} }, "eslint-plugin-jest": { "version": "24.1.3", @@ -67942,7 +67969,8 @@ "version": "4.2.0", "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-4.2.0.tgz", "integrity": "sha512-623WEiZJqxR7VdxFCKLI6d6LLpwJkGPYKODnkH3D7WpOG5KM8yWueBd8TLsNAetEJNF5iJmolaAKO3F8yzyVBQ==", - "dev": true + "dev": true, + "requires": {} }, "eslint-scope": { "version": "5.1.1", @@ -70061,7 +70089,8 @@ "graphql-executor": { "version": "0.0.22", "resolved": "https://registry.npmjs.org/graphql-executor/-/graphql-executor-0.0.22.tgz", - "integrity": "sha512-WbKSnSHFn6REKKH4T6UAwDM3mLUnYMQlQLNG0Fw+Lkb3ilCnL3m5lkJ7411LAI9sF7BvPbthovVZhsEUh9Xfag==" + "integrity": "sha512-WbKSnSHFn6REKKH4T6UAwDM3mLUnYMQlQLNG0Fw+Lkb3ilCnL3m5lkJ7411LAI9sF7BvPbthovVZhsEUh9Xfag==", + "requires": {} }, "graphql-extensions": { "version": "0.15.0", @@ -70192,7 +70221,8 @@ "ws": { "version": "8.5.0", "resolved": "https://registry.npmjs.org/ws/-/ws-8.5.0.tgz", - "integrity": "sha512-BWX0SWVgLPzYwF8lTzEy1egjhS4S4OEAHfsO8o65WOVsrnSRGaSiUaa9e0ggGlkMTtBlmOpEXiie9RUcBO86qg==" + "integrity": "sha512-BWX0SWVgLPzYwF8lTzEy1egjhS4S4OEAHfsO8o65WOVsrnSRGaSiUaa9e0ggGlkMTtBlmOpEXiie9RUcBO86qg==", + "requires": {} } } }, @@ -70272,7 +70302,8 @@ "graphql-ws": { "version": "5.7.0", "resolved": "https://registry.npmjs.org/graphql-ws/-/graphql-ws-5.7.0.tgz", - "integrity": "sha512-8yYuvnyqIjlJ/WfebOyu2GSOQeFauRxnfuTveY9yvrDGs2g3kR9Nv4gu40AKvRHbXlSJwTbMJ6dVxAtEyKwVRA==" + "integrity": "sha512-8yYuvnyqIjlJ/WfebOyu2GSOQeFauRxnfuTveY9yvrDGs2g3kR9Nv4gu40AKvRHbXlSJwTbMJ6dVxAtEyKwVRA==", + "requires": {} }, "minimatch": { "version": "4.2.1", @@ -70444,7 +70475,8 @@ "ws": { "version": "8.5.0", "resolved": "https://registry.npmjs.org/ws/-/ws-8.5.0.tgz", - "integrity": "sha512-BWX0SWVgLPzYwF8lTzEy1egjhS4S4OEAHfsO8o65WOVsrnSRGaSiUaa9e0ggGlkMTtBlmOpEXiie9RUcBO86qg==" + "integrity": "sha512-BWX0SWVgLPzYwF8lTzEy1egjhS4S4OEAHfsO8o65WOVsrnSRGaSiUaa9e0ggGlkMTtBlmOpEXiie9RUcBO86qg==", + "requires": {} } } }, @@ -70524,7 +70556,8 @@ "graphql-ws": { "version": "5.7.0", "resolved": "https://registry.npmjs.org/graphql-ws/-/graphql-ws-5.7.0.tgz", - "integrity": "sha512-8yYuvnyqIjlJ/WfebOyu2GSOQeFauRxnfuTveY9yvrDGs2g3kR9Nv4gu40AKvRHbXlSJwTbMJ6dVxAtEyKwVRA==" + "integrity": "sha512-8yYuvnyqIjlJ/WfebOyu2GSOQeFauRxnfuTveY9yvrDGs2g3kR9Nv4gu40AKvRHbXlSJwTbMJ6dVxAtEyKwVRA==", + "requires": {} }, "minimatch": { "version": "4.2.1", @@ -70607,7 +70640,8 @@ "graphql-sse": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/graphql-sse/-/graphql-sse-1.1.0.tgz", - "integrity": "sha512-xE8AGPJa5X+g7iFmRQw/8H+7lXIDJvSkW6lou/XSSq17opPQl+dbKOMiqraHMx52VrDgS061ZVx90OSuqS6ykA==" + "integrity": "sha512-xE8AGPJa5X+g7iFmRQw/8H+7lXIDJvSkW6lou/XSSq17opPQl+dbKOMiqraHMx52VrDgS061ZVx90OSuqS6ykA==", + "requires": {} }, "graphql-subscriptions": { "version": "1.1.0", @@ -70668,7 +70702,8 @@ "graphql-type-json": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/graphql-type-json/-/graphql-type-json-0.3.2.tgz", - "integrity": "sha512-J+vjof74oMlCWXSvt0DOf2APEdZOCdubEvGDUAlqH//VBYcOYsGgRW7Xzorr44LvkjiuvecWc8fChxuZZbChtg==" + "integrity": "sha512-J+vjof74oMlCWXSvt0DOf2APEdZOCdubEvGDUAlqH//VBYcOYsGgRW7Xzorr44LvkjiuvecWc8fChxuZZbChtg==", + "requires": {} }, "graphql-upload": { "version": "11.0.0", @@ -70692,7 +70727,8 @@ "graphql-ws": { "version": "4.9.0", "resolved": "https://registry.npmjs.org/graphql-ws/-/graphql-ws-4.9.0.tgz", - "integrity": "sha512-sHkK9+lUm20/BGawNEWNtVAeJzhZeBg21VmvmLoT5NdGVeZWv5PdIhkcayQIAgjSyyQ17WMKmbDijIPG2On+Ag==" + "integrity": "sha512-sHkK9+lUm20/BGawNEWNtVAeJzhZeBg21VmvmLoT5NdGVeZWv5PdIhkcayQIAgjSyyQ17WMKmbDijIPG2On+Ag==", + "requires": {} }, "gray-percentage": { "version": "2.0.0", @@ -71295,11 +71331,11 @@ } }, "http-proxy-middleware": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-1.1.2.tgz", - "integrity": "sha512-YRFUeOG3q85FJjAaYVJUoNRW9a73SDlOtAyQOS5PHLr18QeZ/vEhxywNoOPiEO8BxCegz4RXzTHcvyLEGB78UA==", + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-2.0.6.tgz", + "integrity": "sha512-ya/UeJ6HVBYxrgYotAZo1KvPWlgB48kUJLDePFeneHsVujFaW5WNj2NgWCAE//B1Dl02BIfYlpNgBy8Kf8Rjmw==", "requires": { - "@types/http-proxy": "^1.17.5", + "@types/http-proxy": "^1.17.8", "http-proxy": "^1.18.1", "is-glob": "^4.0.1", "is-plain-obj": "^3.0.0", @@ -72202,7 +72238,8 @@ "isomorphic-ws": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/isomorphic-ws/-/isomorphic-ws-4.0.1.tgz", - "integrity": "sha512-BhBvN2MBpWTaSHdWRb/bwdZJ1WaehQ2L1KngkCkfLUGF0mAWAT1sQUQacEmQ0jXkFw/czDXPNQSL5u2/Krsz1w==" + "integrity": "sha512-BhBvN2MBpWTaSHdWRb/bwdZJ1WaehQ2L1KngkCkfLUGF0mAWAT1sQUQacEmQ0jXkFw/czDXPNQSL5u2/Krsz1w==", + "requires": {} }, "istanbul-lib-coverage": { "version": "3.2.0", @@ -73616,7 +73653,8 @@ "version": "1.2.2", "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.2.tgz", "integrity": "sha512-olV41bKSMm8BdnuMsewT4jqlZ8+3TCARAXjZGT9jcoSnrfUnRCqnMoF9XEeoWjbzObpqF9dRhHQj0Xb9QdF6/w==", - "dev": true + "dev": true, + "requires": {} }, "jest-regex-util": { "version": "26.0.0", @@ -74800,7 +74838,8 @@ "version": "7.5.7", "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.7.tgz", "integrity": "sha512-KMvVuFzpKBuiIXW3E4u3mySRO2/mCHSyZDJQM5NQ9Q9KHWHWh0NHgfbRMLLrceUK5qAL4ytALJbpRMjixFZh8A==", - "dev": true + "dev": true, + "requires": {} } } }, @@ -76025,7 +76064,8 @@ "version": "7.1.7", "resolved": "https://registry.npmjs.org/markdown-to-jsx/-/markdown-to-jsx-7.1.7.tgz", "integrity": "sha512-VI3TyyHlGkO8uFle0IOibzpO1c1iJDcXcS/zBrQrXQQvJ2tpdwVzVZ7XdKsyRz1NdRmre4dqQkMZzUHaKIG/1w==", - "dev": true + "dev": true, + "requires": {} }, "match-sorter": { "version": "6.1.0", @@ -76334,7 +76374,8 @@ "meros": { "version": "1.1.4", "resolved": "https://registry.npmjs.org/meros/-/meros-1.1.4.tgz", - "integrity": "sha512-E9ZXfK9iQfG9s73ars9qvvvbSIkJZF5yOo9j4tcwM5tN8mUKfj/EKN5PzOr3ZH0y5wL7dLAHw3RVEfpQV9Q7VQ==" + "integrity": "sha512-E9ZXfK9iQfG9s73ars9qvvvbSIkJZF5yOo9j4tcwM5tN8mUKfj/EKN5PzOr3ZH0y5wL7dLAHw3RVEfpQV9Q7VQ==", + "requires": {} }, "mersenne-twister": { "version": "1.1.0", @@ -76762,7 +76803,8 @@ "mongoose-legacy-pluralize": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz", - "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==" + "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==", + "requires": {} }, "move-concurrently": { "version": "1.0.1", @@ -77338,7 +77380,8 @@ "@next/react-refresh-utils": { "version": "9.5.5", "resolved": "https://registry.npmjs.org/@next/react-refresh-utils/-/react-refresh-utils-9.5.5.tgz", - "integrity": "sha512-Gz5z0+ID+KAGto6Tkgv1a340damEw3HG6ANLKwNi5/QSHqQ3JUAVxMuhz3qnL54505I777evpzL89ofWEMIWKw==" + "integrity": "sha512-Gz5z0+ID+KAGto6Tkgv1a340damEw3HG6ANLKwNi5/QSHqQ3JUAVxMuhz3qnL54505I777evpzL89ofWEMIWKw==", + "requires": {} }, "acorn": { "version": "6.4.2", @@ -79693,7 +79736,8 @@ "pg-pool": { "version": "3.5.1", "resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.5.1.tgz", - "integrity": "sha512-6iCR0wVrro6OOHFsyavV+i6KYL4lVNyYAB9RD18w66xSzN+d8b66HiwuP30Gp1SH5O9T82fckkzsRjlrhD0ioQ==" + "integrity": "sha512-6iCR0wVrro6OOHFsyavV+i6KYL4lVNyYAB9RD18w66xSzN+d8b66HiwuP30Gp1SH5O9T82fckkzsRjlrhD0ioQ==", + "requires": {} }, "pg-protocol": { "version": "1.5.0", @@ -80583,7 +80627,8 @@ "version": "5.5.1", "resolved": "https://registry.npmjs.org/react-colorful/-/react-colorful-5.5.1.tgz", "integrity": "sha512-M1TJH2X3RXEt12sWkpa6hLc/bbYS0H6F4rIqjQZ+RxNBstpY67d9TrFXtqdZwhpmBXcCwEi7stKqFue3ZRkiOg==", - "dev": true + "dev": true, + "requires": {} }, "react-copy-to-clipboard": { "version": "5.0.4", @@ -80597,7 +80642,8 @@ "react-day-picker": { "version": "8.0.0-beta.3", "resolved": "https://registry.npmjs.org/react-day-picker/-/react-day-picker-8.0.0-beta.3.tgz", - "integrity": "sha512-qvasG5iH6M+Ix3HAtCiSum6xbngG5re5F2bZN3VRof7FAl3LJbNcq+5geEITSLRt5uo3IsXF5eFvG6Ggpzw8og==" + "integrity": "sha512-qvasG5iH6M+Ix3HAtCiSum6xbngG5re5F2bZN3VRof7FAl3LJbNcq+5geEITSLRt5uo3IsXF5eFvG6Ggpzw8og==", + "requires": {} }, "react-display-name": { "version": "0.2.5", @@ -80729,12 +80775,14 @@ "react-icons": { "version": "4.3.1", "resolved": "https://registry.npmjs.org/react-icons/-/react-icons-4.3.1.tgz", - "integrity": "sha512-cB10MXLTs3gVuXimblAdI71jrJx8njrJZmNMEMC+sQu5B/BIOmlsAjskdqpn81y8UBVEGuHODd7/ci5DvoSzTQ==" + "integrity": "sha512-cB10MXLTs3gVuXimblAdI71jrJx8njrJZmNMEMC+sQu5B/BIOmlsAjskdqpn81y8UBVEGuHODd7/ci5DvoSzTQ==", + "requires": {} }, "react-image": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/react-image/-/react-image-4.0.3.tgz", - "integrity": "sha512-19MUK9u1qaw9xys8XEsVkSpVhHctEBUeYFvrLTe1PN+4w5Co13AN2WA7xtBshPM6SthsOj77SlDrEAeOaJpf7g==" + "integrity": "sha512-19MUK9u1qaw9xys8XEsVkSpVhHctEBUeYFvrLTe1PN+4w5Co13AN2WA7xtBshPM6SthsOj77SlDrEAeOaJpf7g==", + "requires": {} }, "react-immutable-proptypes": { "version": "2.2.0", @@ -80766,7 +80814,8 @@ "react-intersection-observer": { "version": "8.31.0", "resolved": "https://registry.npmjs.org/react-intersection-observer/-/react-intersection-observer-8.31.0.tgz", - "integrity": "sha512-XraIC/tkrD9JtrmVA7ypEN1QIpKc52mXBH1u/bz/aicRLo8QQEJQAMUTb8mz4B6dqpPwyzgjrr7Ljv/2ACDtqw==" + "integrity": "sha512-XraIC/tkrD9JtrmVA7ypEN1QIpKc52mXBH1u/bz/aicRLo8QQEJQAMUTb8mz4B6dqpPwyzgjrr7Ljv/2ACDtqw==", + "requires": {} }, "react-is": { "version": "17.0.2", @@ -80852,12 +80901,14 @@ "react-prop-toggle": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/react-prop-toggle/-/react-prop-toggle-1.0.2.tgz", - "integrity": "sha512-JmerjAXs7qJ959+d0Ygt7Cb2+4fG+n3I2VXO6JO0AcAY1vkRN/JpZKAN67CMXY889xEJcfylmMPhzvf6nWO68Q==" + "integrity": "sha512-JmerjAXs7qJ959+d0Ygt7Cb2+4fG+n3I2VXO6JO0AcAY1vkRN/JpZKAN67CMXY889xEJcfylmMPhzvf6nWO68Q==", + "requires": {} }, "react-pseudo-state": { "version": "2.2.2", "resolved": "https://registry.npmjs.org/react-pseudo-state/-/react-pseudo-state-2.2.2.tgz", - "integrity": "sha512-czJNP0MpcqJISnFwxYz8IlJklsUbWyuSqWJWLbXi/MUQWIEu2t6PbOGHIVvtpw9i0N3hHZ50nSNCDMLp4fpenQ==" + "integrity": "sha512-czJNP0MpcqJISnFwxYz8IlJklsUbWyuSqWJWLbXi/MUQWIEu2t6PbOGHIVvtpw9i0N3hHZ50nSNCDMLp4fpenQ==", + "requires": {} }, "react-query": { "version": "3.5.11", @@ -81423,7 +81474,8 @@ "redux-immutable": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/redux-immutable/-/redux-immutable-4.0.0.tgz", - "integrity": "sha1-Ohoy32Y2ZGK2NpHw4dw15HK7yfM= sha512-SchSn/DWfGb3oAejd+1hhHx01xUoxY+V7TeK0BKqpkLKiQPVFf7DYzEaKmrEVxsWxielKfSK9/Xq66YyxgR1cg==" + "integrity": "sha1-Ohoy32Y2ZGK2NpHw4dw15HK7yfM= sha512-SchSn/DWfGb3oAejd+1hhHx01xUoxY+V7TeK0BKqpkLKiQPVFf7DYzEaKmrEVxsWxielKfSK9/Xq66YyxgR1cg==", + "requires": {} }, "redux-localstorage": { "version": "1.0.0-rc5", @@ -83077,12 +83129,14 @@ "slate-plain-serializer": { "version": "0.7.13", "resolved": "https://registry.npmjs.org/slate-plain-serializer/-/slate-plain-serializer-0.7.13.tgz", - "integrity": "sha512-TtrlaslxQBEMV0LYdf3s7VAbTxRPe1xaW10WNNGAzGA855/0RhkaHjKkQiRjHv5rvbRleVf7Nxr9fH+4uErfxQ==" + "integrity": "sha512-TtrlaslxQBEMV0LYdf3s7VAbTxRPe1xaW10WNNGAzGA855/0RhkaHjKkQiRjHv5rvbRleVf7Nxr9fH+4uErfxQ==", + "requires": {} }, "slate-prop-types": { "version": "0.5.44", "resolved": "https://registry.npmjs.org/slate-prop-types/-/slate-prop-types-0.5.44.tgz", - "integrity": "sha512-JS0iW7uaciE/W3ADuzeN1HOnSjncQhHPXJ65nZNQzB0DF7mXVmbwQKI6cmCo/xKni7XRJT0JbWSpXFhEdPiBUA==" + "integrity": "sha512-JS0iW7uaciE/W3ADuzeN1HOnSjncQhHPXJ65nZNQzB0DF7mXVmbwQKI6cmCo/xKni7XRJT0JbWSpXFhEdPiBUA==", + "requires": {} }, "slice-ansi": { "version": "3.0.0", @@ -83940,7 +83994,8 @@ "stylis-rule-sheet": { "version": "0.0.10", "resolved": "https://registry.npmjs.org/stylis-rule-sheet/-/stylis-rule-sheet-0.0.10.tgz", - "integrity": "sha512-nTbZoaqoBnmK+ptANthb10ZRZOGC+EmTLLUxeYIuHNkEKcmKgXX1XWKkUBT2Ac4es3NybooPe0SmvKdhKJZAuw==" + "integrity": "sha512-nTbZoaqoBnmK+ptANthb10ZRZOGC+EmTLLUxeYIuHNkEKcmKgXX1XWKkUBT2Ac4es3NybooPe0SmvKdhKJZAuw==", + "requires": {} } } }, @@ -84005,7 +84060,8 @@ "stylis-rule-sheet": { "version": "0.0.10", "resolved": "https://registry.npmjs.org/stylis-rule-sheet/-/stylis-rule-sheet-0.0.10.tgz", - "integrity": "sha512-nTbZoaqoBnmK+ptANthb10ZRZOGC+EmTLLUxeYIuHNkEKcmKgXX1XWKkUBT2Ac4es3NybooPe0SmvKdhKJZAuw==" + "integrity": "sha512-nTbZoaqoBnmK+ptANthb10ZRZOGC+EmTLLUxeYIuHNkEKcmKgXX1XWKkUBT2Ac4es3NybooPe0SmvKdhKJZAuw==", + "requires": {} } } }, @@ -85724,19 +85780,22 @@ "use-callback-ref": { "version": "1.2.4", "resolved": "https://registry.npmjs.org/use-callback-ref/-/use-callback-ref-1.2.4.tgz", - "integrity": "sha512-rXpsyvOnqdScyied4Uglsp14qzag1JIemLeTWGKbwpotWht57hbP78aNT+Q4wdFKQfQibbUX4fb6Qb4y11aVOQ==" + "integrity": "sha512-rXpsyvOnqdScyied4Uglsp14qzag1JIemLeTWGKbwpotWht57hbP78aNT+Q4wdFKQfQibbUX4fb6Qb4y11aVOQ==", + "requires": {} }, "use-composed-ref": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/use-composed-ref/-/use-composed-ref-1.2.1.tgz", "integrity": "sha512-6+X1FLlIcjvFMAeAD/hcxDT8tmyrWnbSPMU0EnxQuDLIxokuFzWliXBiYZuGIx+mrAMLBw0WFfCkaPw8ebzAhw==", - "dev": true + "dev": true, + "requires": {} }, "use-isomorphic-layout-effect": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/use-isomorphic-layout-effect/-/use-isomorphic-layout-effect-1.1.2.tgz", "integrity": "sha512-49L8yCO3iGT/ZF9QttjwLF/ZD9Iwto5LnH5LmEdk/6cFmXddqi2ulF0edxTwjj+7mqvpVVGQWvbXZdn32wRSHA==", - "dev": true + "dev": true, + "requires": {} }, "use-latest": { "version": "1.2.0", @@ -86715,7 +86774,8 @@ "version": "1.2.1", "resolved": "https://registry.npmjs.org/webpack-filter-warnings-plugin/-/webpack-filter-warnings-plugin-1.2.1.tgz", "integrity": "sha512-Ez6ytc9IseDMLPo0qCuNNYzgtUl8NovOqjIq4uAU8LTD4uoa1w1KpZyyzFtLTEMZpkkOkLfL9eN+KGYdk1Qtwg==", - "dev": true + "dev": true, + "requires": {} }, "webpack-hot-middleware": { "version": "2.25.1", @@ -86974,7 +87034,8 @@ "ws": { "version": "7.4.5", "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.5.tgz", - "integrity": "sha512-xzyu3hFvomRfXKH8vOFMU3OguG6oOvhXMo3xsGy3xWExqaM2dxBbVxuD99O7m3ZUFMvvscsZDqxfgMaRr/Nr1g==" + "integrity": "sha512-xzyu3hFvomRfXKH8vOFMU3OguG6oOvhXMo3xsGy3xWExqaM2dxBbVxuD99O7m3ZUFMvvscsZDqxfgMaRr/Nr1g==", + "requires": {} }, "xdg-basedir": { "version": "4.0.0", diff --git a/src/package.json b/src/package.json index a27c830bb..d8f6d2ba5 100644 --- a/src/package.json +++ b/src/package.json @@ -96,10 +96,11 @@ "graphql": "^15.4.0", "graphql-request": "^3.4.0", "graphql-tools": "^7.0.2", - "http-proxy-middleware": "^1.1.2", + "http-proxy-middleware": "^2.0.6", "isomorphic-unfetch": "^3.1.0", - "json-stable-stringify": "^1.0.2", "js-yaml": "^4.1.0", + "json-stable-stringify": "^1.0.2", + "just-kebab-case": "^4.2.0", "jwks-rsa": "^2.0.5", "jwt-decode": "^3.1.2", "keycloak-connect": "^17.0.1", diff --git a/src/server.ts b/src/server.ts index fa44b6009..5ad25937c 100644 --- a/src/server.ts +++ b/src/server.ts @@ -242,6 +242,8 @@ const apps = [ new ApiHealthApp(state), new ApiOpenapiApp(), new MaintenanceApp(), + new ApiDSProxyApp({ url: process.env.SSR_API_ROOT }), + new ApiProxyApp({ gwaApiUrl: process.env.GWA_API_URL }), new ApiGraphqlWhitelistApp({ apiPath, apollo: { @@ -270,8 +272,6 @@ const apps = [ pages: pages, enableDefaultRoute: false, }), - new ApiDSProxyApp({ url: process.env.SSR_API_ROOT }), - new ApiProxyApp({ gwaApiUrl: process.env.GWA_API_URL }), new NextApp({ dir: 'nextapp' }), ]; diff --git a/src/services/identifiers.ts b/src/services/identifiers.ts index ab3f01f97..24af1f39b 100644 --- a/src/services/identifiers.ts +++ b/src/services/identifiers.ts @@ -11,11 +11,15 @@ export function isEnvironmentID(id: string): boolean { } export function newProductID(): string { - return uuidv4().replace(/-/g, '').toUpperCase().substr(0, 12); + return uuidv4().replace(/-/g, '').toUpperCase().substring(0, 12); } export function newApplicationID(): string { - return uuidv4().replace(/-/g, '').toUpperCase().substr(0, 11); + return uuidv4().replace(/-/g, '').toUpperCase().substring(0, 11); } export function newEnvironmentID(): string { - return uuidv4().replace(/-/g, '').toUpperCase().substr(0, 8); + return uuidv4().replace(/-/g, '').toUpperCase().substring(0, 8); +} + +export function newNamespaceID(): string { + return 'gw-' + uuidv4().replace(/-/g, '').toLowerCase().substring(0, 5); } diff --git a/src/services/keycloak/client-service.ts b/src/services/keycloak/client-service.ts index 0c8f1133b..353b44d0e 100644 --- a/src/services/keycloak/client-service.ts +++ b/src/services/keycloak/client-service.ts @@ -103,6 +103,11 @@ export class KeycloakClientService { return lkup[0]; } + public async deleteClient(id: string): Promise { + await this.kcAdminClient.clients.del({ id }); + logger.debug('[deleteClient] CID=%s SUCCESS', id); + } + public async regenerateSecret(id: string): Promise { const cred = await this.kcAdminClient.clients.generateNewClientSecret({ id, diff --git a/src/services/keycloak/group-service.ts b/src/services/keycloak/group-service.ts index db8b45b17..3e9127369 100644 --- a/src/services/keycloak/group-service.ts +++ b/src/services/keycloak/group-service.ts @@ -175,7 +175,7 @@ export class KeycloakGroupService { (group: GroupRepresentation) => group.name == groupName ).length == 0 ) { - logger.debug('[getGroup] MISSING %s', groupName); + logger.error('[getGroup] MISSING %s', groupName); return null; } else { logger.debug('[getGroup] FOUND %s', groupName); diff --git a/src/services/keycloak/namespace-details.ts b/src/services/keycloak/namespace-details.ts index 7823d4ad2..d7afeafd6 100644 --- a/src/services/keycloak/namespace-details.ts +++ b/src/services/keycloak/namespace-details.ts @@ -12,6 +12,7 @@ import { getOrgPoliciesForResource, } from '../../lists/extensions/Common'; import { GWAService } from '../gwaapi'; +import { strict as assert } from 'assert'; const logger = Logger('kc.nsdetails'); @@ -35,16 +36,20 @@ export async function getAllNamespaces(envCtx: EnvironmentContext) { const client = new GWAService(process.env.GWA_API_URL); const defaultSettings = await client.getDefaultNamespaceSettings(); - return await Promise.all( - nsList.map(async (nsdata: any) => { - return backfillGroupAttributes( - nsdata.name, - nsdata, - defaultSettings, - kcGroupService - ); + return ( + await Promise.all( + nsList.map(async (nsdata: any) => { + return backfillGroupAttributes( + nsdata.name, + nsdata, + defaultSettings, + kcGroupService + ); + }) + ).catch((err: any) => { + throw err; }) - ); + ).filter((x) => Boolean(x)); } export async function getKeycloakGroupApi( @@ -67,7 +72,10 @@ export async function backfillGroupAttributes( ): Promise { const nsPermissions = await kcGroupService.getGroup('ns', ns); + assert.strictEqual(Boolean(nsPermissions), true, 'Invalid namespace'); + transformSingleValueAttributes(nsPermissions.attributes, [ + 'description', 'perm-data-plane', 'perm-protected-ns', 'org', @@ -161,6 +169,7 @@ export async function getResource( .map((ns: ResourceSet) => ({ id: ns.id, name: ns.name, + displayName: ns.displayName, scopes: ns.resource_scopes, })) .pop(); diff --git a/src/services/keystone/types.ts b/src/services/keystone/types.ts index d34a32302..5d5de8201 100644 --- a/src/services/keystone/types.ts +++ b/src/services/keystone/types.ts @@ -5326,7 +5326,8 @@ export type MutationUpdateCurrentNamespaceArgs = { export type MutationCreateNamespaceArgs = { - namespace: Scalars['String']; + name?: Maybe; + displayName?: Maybe; }; @@ -5415,6 +5416,7 @@ export type Namespace = { __typename?: 'Namespace'; id?: Maybe; name: Scalars['String']; + displayName?: Maybe; scopes?: Maybe>>; prodEnvId?: Maybe; permDomains?: Maybe>>; @@ -5429,7 +5431,8 @@ export type Namespace = { }; export type NamespaceInput = { - name: Scalars['String']; + name?: Maybe; + displayName?: Maybe; }; /** A keystone list */ diff --git a/src/services/report/data/namespaces.ts b/src/services/report/data/namespaces.ts index c5d10c914..ab0e9b1fa 100644 --- a/src/services/report/data/namespaces.ts +++ b/src/services/report/data/namespaces.ts @@ -13,6 +13,7 @@ import { GWAService } from '../../gwaapi'; export interface ReportOfNamespaces { resource_id: string; name: string; + displayName?: string; permProtectedNs?: string; permDomains?: string[]; permDataPlane?: string; diff --git a/src/services/report/output/structure.ts b/src/services/report/output/structure.ts index bc289503c..1fa93e723 100644 --- a/src/services/report/output/structure.ts +++ b/src/services/report/output/structure.ts @@ -18,6 +18,11 @@ export const reportStructure: any = { key: 'name', width: 32, }, + { + header: 'Display Name', + key: 'displayName', + width: 32, + }, { header: 'Privileged', key: 'permProtectedNs', diff --git a/src/services/uma2/resource-registration-service.ts b/src/services/uma2/resource-registration-service.ts index 4fdb76b6a..eb7b20ae8 100644 --- a/src/services/uma2/resource-registration-service.ts +++ b/src/services/uma2/resource-registration-service.ts @@ -14,6 +14,7 @@ export interface ResourceSetQuery { scope?: string; first?: number; max?: number; + deep?: boolean; } export interface ResourceScope { @@ -27,6 +28,7 @@ export interface ResourceOwner { export interface ResourceSet { id: string; name: string; + displayName?: string; type: string; uris?: string[]; icon_uri?: string; @@ -37,6 +39,7 @@ export interface ResourceSet { export interface ResourceSetInput { name: string; + displayName?: string; type: string; uris?: string[]; icon_uri?: string; diff --git a/src/services/utils.ts b/src/services/utils.ts index 278f42367..ee8b8e195 100644 --- a/src/services/utils.ts +++ b/src/services/utils.ts @@ -57,7 +57,6 @@ export function regExprValidation( value: string, errorMessage: string ) { - const namespaceValidationRule = '^[a-z][a-z0-9-]{4,14}$'; const re = new RegExp(rule); assert.strictEqual(re.test(value), true, errorMessage); } diff --git a/src/services/workflow/client-shared-idp.ts b/src/services/workflow/client-shared-idp.ts index ac3c97138..984ee1b1e 100644 --- a/src/services/workflow/client-shared-idp.ts +++ b/src/services/workflow/client-shared-idp.ts @@ -96,10 +96,7 @@ async function addClientToSharedIdP( const environment = issuerEnvConfig.environment; - const clientId = - environment == 'prod' - ? `ap-${profileClientId}` - : `ap-${profileClientId}-${environment}`; + const clientId = genClientId(environment, profileClientId); // If there are any custom client Mappers, then include them const clientMappers: any[] = []; @@ -137,6 +134,77 @@ async function addClientToSharedIdP( }; } +/** + * When a CredentialIssuer is deleted where the inheritFrom is set + * then call this to delete the client ID on the shared IdP + * + * @param context + * @param credentialIssuerPK + */ +export async function DeleteClientsFromSharedIdP( + context: any, + profileClientId: string, + inheritFromIssuerPK: string +) { + // Find the credential issuer and based on its type, go do the appropriate action + const inheritFromIssuer: CredentialIssuer = await lookupCredentialIssuerById( + context, + inheritFromIssuerPK + ); + + assert.strictEqual( + inheritFromIssuer.isShared, + true, + 'Invalid IdP for Sharing' + ); + + const envConfigs = getAllIssuerEnvironmentConfigs(inheritFromIssuer); + + for (const issuerEnvConfig of envConfigs) { + await deleteClientFromSharedIdP(profileClientId, issuerEnvConfig); + } +} + +async function deleteClientFromSharedIdP( + profileClientId: string, + issuerEnvConfig: IssuerEnvironmentConfig +): Promise { + const openid = await getOpenidFromIssuer(issuerEnvConfig.issuerUrl); + + // token is NULL if 'iat' + // token is retrieved from doing a /token login using the provided client ID and secret if 'managed' + // issuer.initialAccessToken if 'iat' + const kctoksvc = new KeycloakTokenService(openid.token_endpoint); + + const token = + issuerEnvConfig.clientRegistration == 'anonymous' + ? null + : issuerEnvConfig.clientRegistration == 'managed' + ? await kctoksvc.getKeycloakSession( + issuerEnvConfig.clientId, + issuerEnvConfig.clientSecret + ) + : issuerEnvConfig.initialAccessToken; + + const environment = issuerEnvConfig.environment; + + const clientId = genClientId(environment, profileClientId); + + // Find the Client ID for the ProductEnvironment - that will be used to associated the clientRoles + + const cliApi = await new KeycloakClientService(issuerEnvConfig.issuerUrl); + await cliApi.login(issuerEnvConfig.clientId, issuerEnvConfig.clientSecret); + + const exists = await cliApi.isClient(clientId); + + if (exists) { + const client = await cliApi.findByClientId(clientId); + assert.strictEqual(client.clientId, clientId); + + await cliApi.deleteClient(client.id); + } +} + /** * Whenever the CredentialIssuer is updated, and inheritFrom is set, * then sync the Roles and Authz settings in the Client Registration diff --git a/src/services/workflow/delete-namespace.ts b/src/services/workflow/delete-namespace.ts index 0bfc895fd..0d63d9704 100644 --- a/src/services/workflow/delete-namespace.ts +++ b/src/services/workflow/delete-namespace.ts @@ -146,5 +146,9 @@ export const DeleteNamespace = async ( await deleteRecords(context, 'Product', { namespace: ns }, true, ['id']); + await deleteRecords(context, 'CredentialIssuer', { namespace: ns }, true, [ + 'id', + ]); + await updateActivity(context.sudo(), activity.id, 'success', undefined); }; diff --git a/src/services/workflow/get-namespaces.ts b/src/services/workflow/get-namespaces.ts index d43a83ac6..4132462b9 100644 --- a/src/services/workflow/get-namespaces.ts +++ b/src/services/workflow/get-namespaces.ts @@ -59,6 +59,7 @@ export async function getMyNamespaces( return namespaces.map((ns: ResourceSet) => ({ id: ns.id, name: ns.name, + displayName: ns.displayName, scopes: ns.resource_scopes, prodEnvId: envCtx.prodEnv.id, })); @@ -269,6 +270,7 @@ export interface ResourceServerContext { export interface NamespaceSummary { id: string; name: string; + displayName: string; scopes: ResourceScope[]; prodEnvId: string; } diff --git a/src/services/workflow/index.ts b/src/services/workflow/index.ts index fb2335ac4..bb5b8627b 100644 --- a/src/services/workflow/index.ts +++ b/src/services/workflow/index.ts @@ -2,7 +2,11 @@ export { Apply } from './apply'; export { CreateServiceAccount } from './create-service-account'; -export { addClientsToSharedIdP, syncSharedIdp } from './client-shared-idp'; +export { + addClientsToSharedIdP, + DeleteClientsFromSharedIdP, + syncSharedIdp, +} from './client-shared-idp'; export { allConsumerGroupLabels, diff --git a/src/test/integrated/uma2/resource.ts b/src/test/integrated/uma2/resource.ts index 170dc1bda..f7f81d3bc 100644 --- a/src/test/integrated/uma2/resource.ts +++ b/src/test/integrated/uma2/resource.ts @@ -32,15 +32,18 @@ import { token ); - if (false) { + if (true) { await svc.createResourceSet({ name: 'sample2', - type: 'organization', + displayName: 'Sample Number 2', + type: 'integration_test', resource_scopes: ['Organization.Manage'], ownerManagedAccess: true, }); } - console.log(await svc.listResources({ type: 'organization' })); + console.log( + await svc.listResources({ type: 'integration_test', deep: true }) + ); - console.log(await svc.findResourceByName('sample33')); + console.log(await svc.findResourceByName('sample2')); })(); diff --git a/src/test/services/identifiers.test.js b/src/test/services/identifiers.test.js new file mode 100644 index 000000000..c288d9c84 --- /dev/null +++ b/src/test/services/identifiers.test.js @@ -0,0 +1,11 @@ +import fetch from 'node-fetch'; +import { newNamespaceID } from '../../services/identifiers'; + +describe('Identifiers', function () { + it('it should be a valid namespace', async function () { + const result = newNamespaceID(); + expect(result).toHaveLength(8); + expect(result.startsWith('gw-')).toBeTruthy(); + console.log(result); + }); +}); diff --git a/src/tsoa-v2.json b/src/tsoa-v2.json index f121dff89..5046af3f3 100644 --- a/src/tsoa-v2.json +++ b/src/tsoa-v2.json @@ -24,6 +24,11 @@ "description": "Authz Portal Login", "scheme": "bearer", "bearerFormat": "JWT" + }, + "openid": { + "type": "openIdConnect", + "description": "OIDC Login", + "openIdConnectUrl": "https://well_known_endpoint" } }, "tags": [ From 90c0d92193bf1560e9faf1fa47a41d9dfaa1567d Mon Sep 17 00:00:00 2001 From: ikethecoder Date: Tue, 12 Sep 2023 13:35:57 -0700 Subject: [PATCH 3/7] add jira sync --- .github/workflows/jira.yaml | 11 +++++++++++ 1 file changed, 11 insertions(+) create mode 100644 .github/workflows/jira.yaml diff --git a/.github/workflows/jira.yaml b/.github/workflows/jira.yaml new file mode 100644 index 000000000..acc3fc84c --- /dev/null +++ b/.github/workflows/jira.yaml @@ -0,0 +1,11 @@ +name: Sync GitHub issues to Jira +on: [issues, issue_comment] + +jobs: + sync-issues: + name: Sync issues to Jira + runs-on: ubuntu-latest + steps: + - uses: canonical/sync-issues-github-jira@v1 + with: + webhook-url: ${{ secrets.JIRA_WEBHOOK_URL }} From a56aa5eb44550ba1b52ffdbc10a6ff329b772490 Mon Sep 17 00:00:00 2001 From: ikethecoder Date: Wed, 13 Sep 2023 15:04:18 -0700 Subject: [PATCH 4/7] use ikethecoder jira sync --- .github/workflows/jira.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/jira.yaml b/.github/workflows/jira.yaml index acc3fc84c..1bb78e43e 100644 --- a/.github/workflows/jira.yaml +++ b/.github/workflows/jira.yaml @@ -6,6 +6,6 @@ jobs: name: Sync issues to Jira runs-on: ubuntu-latest steps: - - uses: canonical/sync-issues-github-jira@v1 + - uses: ikethecoder/sync-issues-github-jira@latest with: webhook-url: ${{ secrets.JIRA_WEBHOOK_URL }} From f025fbd25b084e03417e3c705b479f60e8857592 Mon Sep 17 00:00:00 2001 From: ikethecoder Date: Wed, 13 Sep 2023 15:06:36 -0700 Subject: [PATCH 5/7] use ikethecoder jira sync --- .github/workflows/jira.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/jira.yaml b/.github/workflows/jira.yaml index 1bb78e43e..ecec47d80 100644 --- a/.github/workflows/jira.yaml +++ b/.github/workflows/jira.yaml @@ -6,6 +6,6 @@ jobs: name: Sync issues to Jira runs-on: ubuntu-latest steps: - - uses: ikethecoder/sync-issues-github-jira@latest + - uses: ikethecoder/sync-issues-github-jira@dev with: webhook-url: ${{ secrets.JIRA_WEBHOOK_URL }} From a836efd57639f8c7c451d4037e998a2bb07e54a5 Mon Sep 17 00:00:00 2001 From: nirajCITZ <94716060+nirajCITZ@users.noreply.github.com> Date: Fri, 15 Sep 2023 10:49:00 -0700 Subject: [PATCH 6/7] Cypress gwa cli (#896) * upd controllers * add openid as api auth * add namespace creation * fix build error * fix required field for create namespace * upd roles for tsoa user * filter out inactive environments from drop down (#825) * Fix BCDC Datasets out of sync (#837) --------- Co-authored-by: James Elson Co-authored-by: Elson9 * add description to namespace * use name for namespace * gwa-cli for namespace, login and config * upg http-proxy * move around proxy * Update oauth2-proxy config * use displayName for namespace * have create namespace using application json * change from lodash to just-kebab-case for shared idp client id * change from lodash to just-kebab-case for shared idp client id * service account failing with cannot read properties of null * order the namespaces in the api * upd feed worker put bubble up child results * upd feed worker put bubble up child results * Added scenarios for GWA-CLI and update existing scenarios as per new gateway layout * make issuer env details optional * make issuer env details optional * upd the batch default array * upds for handling delete namespace * upd delete ns for issuers * chg delete ns put back gw deletion * Updated/added scenarios for GWA command and as per new namespace creation workflow * Updated Cypress Tests as per GWA CLI and new namespace creation logic * Added workflow to install binary file * Update step in the workflow file * update steps in the workflow * Update steps to Build Executable Binary * Update in the workflow file * Update step of Checkout souce-repo in the workflow file * Update step in the workflow file * Update step to install go in the workflow file * Code cleanup, update test name, added test for keycloak role and invalid JWK Public key format * Added scenarios for gwa generate config and update github action to build, install gwa cli * Update the workflow file to build gwa binary * Add cypress as branch regex to trigger the execution on the feature branch * update branch for GWA API step * Update steps in the workflow to build and install gwa cli * Corrected steps for access permission * Add GWA help step in workflow and update api.json test file * Check for gwa command through cypress * Update workflow and e2e dockerfile * Update workflow file * Modified step to build go binary file * Corrected error in the workflow file * Updated step to build go binary file * Update step to build go binary * Commented the GWA help step * Commented step to set up just and create issue in github * update command to build go binary file in the workflow file * Update steps to download and build go * Added go tidy command * Updated the build file in the workflow file * Downgrade go to 1.16 * Update action go setup * Updated Github action workflow to install GWA CLI executable binary and run cypress test * Update the workflow file * Update in the workflow step and removed unused code * Uncomment test in cypress config file * Change assertion type from equals to contains * Increased time out in cy exec command * Remove wait statement inside cy exec command * Re order the spec files for execution in CI/CD * Update Test scenarios/specs order and added test for zenhub#887 * Update steps for API Key request access and gwa generate confin * update workflow file to check cypress test on remote VM * 1)Added scenarios to verify invalid namespace name 2) Replace 'gwa publish'to 'gwa apply' 3)Made changes as per PR Review comments * Update API directory test data in the file * Remove commented code and debugger statements * remove branch name --------- Co-authored-by: ikethecoder Co-authored-by: Elson9 Co-authored-by: James Elson Co-authored-by: Niraj Patel --- .github/workflows/aps-cypress-e2e.yaml | 38 +- e2e/Dockerfile | 3 +- e2e/cypress.config.ts | 5 +- e2e/cypress/fixtures/api.json | 150 ++++- e2e/cypress/fixtures/apiowner.json | 516 ++++++++++-------- e2e/cypress/fixtures/cc-service-gwa.yml | 19 + .../fixtures/service-clear-resources-gwa.yml | 19 + e2e/cypress/fixtures/service-gwa.yml | 38 ++ .../fixtures/service-permission-gwa.yml | 19 + .../fixtures/service-plugin-key-auth-only.yml | 2 +- e2e/cypress/fixtures/test_data/gwa-cli.json | 12 + .../fixtures/test_data/gwa-dataset.yaml | 15 + .../fixtures/test_data/gwa-issuer.yaml | 12 + .../fixtures/test_data/gwa-product.yaml | 8 + e2e/cypress/pageObjects/activity.ts | 18 +- e2e/cypress/pageObjects/apiDirectory.ts | 33 +- e2e/cypress/pageObjects/authProfile.ts | 15 + e2e/cypress/pageObjects/consumers.ts | 40 +- .../pageObjects/keycloakClientScopes.ts | 23 + e2e/cypress/pageObjects/myAccess.ts | 7 + e2e/cypress/pageObjects/products.ts | 19 + e2e/cypress/support/auth-commands.ts | 63 ++- e2e/cypress/support/global.d.ts | 56 +- e2e/cypress/support/util-commands.ts | 85 ++- .../tests/01-api-key/01-create-api.cy.ts | 78 ++- .../tests/01-api-key/02-team-access.cy.ts | 9 - .../03-request-access-inactive-env.cy.ts | 143 +++++ ...ess-with-out-collecting-credentials.cy.ts} | 7 - ...uest-without-collecting-credentials.cy.ts} | 12 +- ...als.cy.ts => 06-collect-credentials.cy.ts} | 0 ...st.cy.ts => 07-approve-pending-rqst.cy.ts} | 10 +- ...ant-access.cy.ts => 08-grant-access.cy.ts} | 10 +- e2e/cypress/tests/01-api-key/09-gwa-get.ts | 85 +++ .../01-client-cred-team-access.cy.ts | 35 +- ...client-cred-create-api-prod-auth-pro.cy.ts | 43 +- .../05-cids-access-approve-api-rqst.cy.ts | 176 +++++- ...07-jwt-genkp-access-approve-api-rqst.cy.ts | 9 +- ...09-jwks-url-access-approval-api-rqst.cy.ts | 9 +- .../10-jwks-publicKey-access-rqst.cy.ts | 15 +- ...t-publlicKey-access-approve-api-rqst.cy.ts | 9 +- ...approve-pending-rqst-for-labels.spec.cy.ts | 10 +- .../03-manage-labels/03-filter-labels.cy.ts | 10 +- .../03-manage-labels/04-manage-labels.cy.ts | 12 +- .../03-manage-labels/05-link-consumers.ts | 21 +- .../01-gateway-service-details.cy.ts | 11 +- .../01-migrate-user-access.cy.ts | 28 +- .../02-client-credentials.cy.ts | 4 + .../07-manage-control/01-ip-restriction.cy.ts | 24 +- .../07-manage-control/02-rate-limiting.cy.ts | 11 +- ...03-kong-api-only-apply-rate-limiting.cy.ts | 22 +- .../02-add-roles-authorization-profile.ts | 2 +- .../08-client-role/03-read-client-role.ts | 18 +- .../08-client-role/04-write-client-role.ts | 16 +- .../08-client-role/05-check-without-role.ts | 9 +- ...01-client-credential-to-kong-acl-api.cy.ts | 20 +- ...02-kong-acl-api-to-client-credential.cy.ts | 65 ++- .../03-apply-multiple-services.cy.ts | 90 +-- .../05-keycloak-shared-IDP-config.cy.ts | 2 +- .../07-kong-public-auth.ts | 7 +- .../10-clear-resources/01-create-api.cy.ts | 68 ++- .../11-activity-feed/01-activity-feed.cy.ts | 15 +- .../02-activity-feed-failure.cy.ts | 4 + .../12-access-permission/01-create-api.cy.ts | 67 ++- .../04-access-manager.cy.ts | 24 +- .../05-namespace-manage.cy.ts | 14 +- .../06-credential-issuer.cy.ts | 15 +- .../07-namespace-view.cy.ts | 15 +- .../08-gateway-config.cy.ts | 24 +- .../01-create-api.cy.ts | 62 +-- .../02-namespace-preview-mode.cy.ts | 0 .../01-client-cred-team-access.ts | 71 ++- .../02-multiple-org-admin.ts | 38 +- .../03-multiple-org-admin-org-unit.ts | 37 +- .../tests/15-aps-api/01-create-api.cy.ts | 73 +++ .../02-organization.cy.ts | 21 +- .../03-documentation.cy.ts | 50 +- .../04-keycloak-shared-IDP-config.cy.ts | 0 .../15-aps-api/05-authorizationProfiles.cy.ts | 300 ++++++++++ .../06-products.cy.ts | 30 +- .../07-api-directory.cy.ts | 40 +- .../tests/15-aps-api/08-namespaces.cy.ts | 294 ++++++++++ .../tests/16-aps-api/01-create-api.cy.ts | 44 -- .../16-aps-api/05-authorizationProfiles.cy.ts | 161 ------ .../tests/16-aps-api/08-namespaces.cy.ts | 207 ------- .../tests/16-gwa-cli/01-cli-commands.ts | 102 ++++ .../16-gwa-cli/02-cli-generate-config.ts | 101 ++++ e2e/cypress/tests/17-CORS/01-create-api.cy.ts | 126 ----- ...01-delete-application-without-access.cy.ts | 0 ...ete-application-with-pending-request.cy.ts | 0 ...te-application-with-approved-request.cy.ts | 12 +- .../04-delete-namespace-gwa.ts | 82 +++ e2e/gw-config.yml | 36 ++ e2e/package-lock.json | 181 ++++-- e2e/package.json | 1 + local/feeder-init/init.sh | 2 +- local/oauth2-proxy/oauth2-proxy-local.cfg | 2 +- 96 files changed, 3130 insertions(+), 1436 deletions(-) create mode 100644 e2e/cypress/fixtures/cc-service-gwa.yml create mode 100644 e2e/cypress/fixtures/service-clear-resources-gwa.yml create mode 100644 e2e/cypress/fixtures/service-gwa.yml create mode 100644 e2e/cypress/fixtures/service-permission-gwa.yml create mode 100644 e2e/cypress/fixtures/test_data/gwa-cli.json create mode 100644 e2e/cypress/fixtures/test_data/gwa-dataset.yaml create mode 100644 e2e/cypress/fixtures/test_data/gwa-issuer.yaml create mode 100644 e2e/cypress/fixtures/test_data/gwa-product.yaml create mode 100644 e2e/cypress/pageObjects/keycloakClientScopes.ts create mode 100644 e2e/cypress/tests/01-api-key/03-request-access-inactive-env.cy.ts rename e2e/cypress/tests/01-api-key/{03-request-access-with-out-collecting-credentials.cy.ts => 04-request-access-with-out-collecting-credentials.cy.ts} (87%) rename e2e/cypress/tests/01-api-key/{04-review-request-without-collecting-credentials.cy.ts => 05-review-request-without-collecting-credentials.cy.ts} (81%) rename e2e/cypress/tests/01-api-key/{05-collect-credentials.cy.ts => 06-collect-credentials.cy.ts} (100%) rename e2e/cypress/tests/01-api-key/{06-approve-pending-rqst.cy.ts => 07-approve-pending-rqst.cy.ts} (93%) rename e2e/cypress/tests/01-api-key/{07-grant-access.cy.ts => 08-grant-access.cy.ts} (88%) create mode 100644 e2e/cypress/tests/01-api-key/09-gwa-get.ts rename e2e/cypress/tests/{14-namespace-preview-mode => 13-namespace-preview-mode}/01-create-api.cy.ts (54%) rename e2e/cypress/tests/{14-namespace-preview-mode => 13-namespace-preview-mode}/02-namespace-preview-mode.cy.ts (100%) rename e2e/cypress/tests/{15-org-assignment => 14-org-assignment}/01-client-cred-team-access.ts (80%) rename e2e/cypress/tests/{15-org-assignment => 14-org-assignment}/02-multiple-org-admin.ts (71%) rename e2e/cypress/tests/{15-org-assignment => 14-org-assignment}/03-multiple-org-admin-org-unit.ts (74%) create mode 100644 e2e/cypress/tests/15-aps-api/01-create-api.cy.ts rename e2e/cypress/tests/{16-aps-api => 15-aps-api}/02-organization.cy.ts (92%) rename e2e/cypress/tests/{16-aps-api => 15-aps-api}/03-documentation.cy.ts (81%) rename e2e/cypress/tests/{16-aps-api => 15-aps-api}/04-keycloak-shared-IDP-config.cy.ts (100%) create mode 100644 e2e/cypress/tests/15-aps-api/05-authorizationProfiles.cy.ts rename e2e/cypress/tests/{16-aps-api => 15-aps-api}/06-products.cy.ts (80%) rename e2e/cypress/tests/{16-aps-api => 15-aps-api}/07-api-directory.cy.ts (85%) create mode 100644 e2e/cypress/tests/15-aps-api/08-namespaces.cy.ts delete mode 100644 e2e/cypress/tests/16-aps-api/01-create-api.cy.ts delete mode 100644 e2e/cypress/tests/16-aps-api/05-authorizationProfiles.cy.ts delete mode 100644 e2e/cypress/tests/16-aps-api/08-namespaces.cy.ts create mode 100644 e2e/cypress/tests/16-gwa-cli/01-cli-commands.ts create mode 100644 e2e/cypress/tests/16-gwa-cli/02-cli-generate-config.ts delete mode 100644 e2e/cypress/tests/17-CORS/01-create-api.cy.ts rename e2e/cypress/tests/{13-delete-application => 17-delete-application}/01-delete-application-without-access.cy.ts (100%) rename e2e/cypress/tests/{13-delete-application => 17-delete-application}/02-delete-application-with-pending-request.cy.ts (100%) rename e2e/cypress/tests/{13-delete-application => 17-delete-application}/03-delete-application-with-approved-request.cy.ts (92%) create mode 100644 e2e/cypress/tests/17-delete-application/04-delete-namespace-gwa.ts create mode 100644 e2e/gw-config.yml diff --git a/.github/workflows/aps-cypress-e2e.yaml b/.github/workflows/aps-cypress-e2e.yaml index ed3e232ed..918a69a7d 100644 --- a/.github/workflows/aps-cypress-e2e.yaml +++ b/.github/workflows/aps-cypress-e2e.yaml @@ -2,7 +2,7 @@ name: Build and Deploy Cypress and Execute Tests on: push: - branches: ['test', 'cypress/*'] + branches: ['test', 'cypress*'] env: DASHBOARD_PROJECT_ID: ${{ secrets.CY_DASHBOARD_PRJ_ID }} @@ -20,14 +20,47 @@ jobs: steps: - name: Build GWA API Image run: | - git clone https://github.com/bcgov/gwa-api.git -b master + git clone https://github.com/bcgov/gwa-api.git cd gwa-api/microservices/gatewayApi docker build -t gwa-api:e2e . + - name: Checkout Portal uses: actions/checkout@v2 + + - name: Checkout Source Repository + run: | + git clone https://github.com/bcgov/gwa-cli.git + cd gwa-cli + + - name: Setup Go + uses: actions/setup-go@v2 + with: + go-version: '1.20.x' + + - name: create .env file + run: | + cd gwa-cli + echo "GWA_API_HOST=api-services-portal-feature-gwa-cli-v2.apps.silver.devops.gov.bc.ca" >> .env + echo "GWA_CLIENT_ID=gwa-cli" >> .env + + - name: Debug + run: echo ${{ github.workspace }} + + - name: Build + run: | + cd gwa-cli + go clean + go build -o ${{ github.workspace }}/e2e/gwa + + - uses: actions/upload-artifact@v2 + with: + name: binary-artifact + path: ${{ github.workspace }}/e2e + - name: Build Docker Images run: | docker-compose build + - name: Spin up API Services Portal and Run E2E Tests run: | export CY_EXECUTION_ENV=${{ env.EXECUTION_ENV }} @@ -40,6 +73,7 @@ jobs: export CY_REPO_URL=${{ env.GIT_REPO_URL }} export CY_COMMIT_AUTHOR_EMAIL=$(git --no-pager show -s --format='%ae' ${{ env.GIT_COMMIT_SHA }}) docker-compose up -d + - name: Execute Tests & Clean Up run: | while true; do diff --git a/e2e/Dockerfile b/e2e/Dockerfile index 46e1ec3bc..a77289f4a 100644 --- a/e2e/Dockerfile +++ b/e2e/Dockerfile @@ -3,14 +3,15 @@ FROM cypress/included:12.4.0 WORKDIR /e2e RUN apt-get -y update; apt-get -y install curl - COPY cypress.config.ts /e2e COPY tsconfig.json /e2e COPY package.json /e2e COPY package-lock.json /e2e COPY entrypoint.sh /tmp +COPY gwa /usr/local/bin ADD cypress /e2e/cypress RUN npm install + ENTRYPOINT ["npm", "run", "cy:run:html"] \ No newline at end of file diff --git a/e2e/cypress.config.ts b/e2e/cypress.config.ts index 150628ffc..526b4dd58 100644 --- a/e2e/cypress.config.ts +++ b/e2e/cypress.config.ts @@ -23,10 +23,11 @@ export default defineConfig({ './cypress/tests/10-*/*.ts', './cypress/tests/11-*/*.ts', './cypress/tests/12-*/*.ts', - './cypress/tests/13-*/*.ts', + './cypress/tests/13-*/*.ts', './cypress/tests/14-*/*.ts', './cypress/tests/15-*/*.ts', - './cypress/tests/16-*/*.ts' + './cypress/tests/16-*/*.ts', + './cypress/tests/17-*/*.ts' ] return config }, diff --git a/e2e/cypress/fixtures/api.json b/e2e/cypress/fixtures/api.json index 5ae336805..41435f225 100644 --- a/e2e/cypress/fixtures/api.json +++ b/e2e/cypress/fixtures/api.json @@ -10,10 +10,36 @@ "title": "Planning and Innovation Division" }, "orgName": "ministry-of-health", - "expectedScope": ["Dataset.Manage", "GroupAccess.Manage", "Namespace.Assign"], + "expectedScope": [ + "Dataset.Manage", + "GroupAccess.Manage", + "Namespace.Assign" + ], "expectedNamespace": { - "name": "newplatform", - "orgUnit": "planning-and-innovation-division" + "name": "gw-3a443", + "orgUnit": "planning-and-innovation-division", + "enabled": true + }, + "expectedRoles": { + "organization-admin": { + "label": "Organization Administrator", + "permissions": [ + { + "resourceType": "organization", + "scopes": [ + "GroupAccess.Manage", + "Namespace.Assign", + "Dataset.Manage" + ] + }, + { + "resourceType": "namespace", + "scopes": [ + "Namespace.View" + ] + } + ] + } }, "body": { "name": "ministry-of-health", @@ -24,7 +50,9 @@ "id": "janis@idir", "email": "janis@testmail.com" }, - "roles": ["organization-admin"] + "roles": [ + "organization-admin" + ] } ] } @@ -44,7 +72,10 @@ "order": 0, "isPublic": true, "isComplete": true, - "tags": ["tag1", "tag2"] + "tags": [ + "tag1", + "tag2" + ] } }, "apiDirectory": { @@ -65,18 +96,25 @@ "title": "Dataset for Test API", "isInCatalog": "false", "isDraft": "false", - "tags": ["gateway", "kong"], + "tags": [ + "gateway", + "kong" + ], "organization": "ministry-of-health", "organizationUnit": "planning-and-innovation-division" }, "directory": { - "name": "api-auto-test-product", - "title": "API Auto Test Product", - "notes": "API Gateway Services provides a way to configure services on the API Gateway, manage access to APIs and get insight into the use of them.", - "license_title": "Access Only", - "view_audience": "Government", - "security_class": "LOW-PUBLIC", - "tags": ["gateway", "kong", "openapi"], + "name": "gwa-auto-test-product", + "title": "GWA Auto Test Product", + "notes": "For Test Purpose", + "license_title": "Open Government Licence - British Columbia", + "view_audience": "Public", + "security_class": "PUBLIC", + "record_publish_date": "2017-09-05", + "tags": [ + "tag1", + "tag2" + ], "organization": { "name": "ministry-of-health", "title": "Ministry of Health" @@ -87,12 +125,18 @@ }, "products": [ { - "name": "API Auto Test Product", + "id": "2", + "name": "GWA Auto Test Product", "environments": [ { "name": "dev", "active": true, - "flow": "kong-api-key-acl" + "flow": "kong-api-key-only" + }, + { + "name": "test", + "active": true, + "flow": "client-credentials" } ] } @@ -105,7 +149,11 @@ "license_title": "Access Only", "view_audience": "Government", "security_class": "LOW-PUBLIC", - "tags": ["gateway", "kong", "openapi"], + "tags": [ + "gateway", + "kong", + "openapi" + ], "organization": { "name": "ministry-of-health", "title": "Ministry of Health" @@ -174,6 +222,46 @@ ], "isShared": true }, + "shared_gwa": { + "name": "Gold Shared IdP", + "description": "A Shared IdP for Teams to use", + "flow": "client-credentials", + "clientAuthenticator": "client-secret", + "mode": "auto", + "environmentDetails": [ + { + "environment": "test", + "issuerUrl": "http://keycloak.localtest.me:9080/auth/realms/master", + "clientRegistration": "managed", + "clientId": "gwa-api", + "clientSecret": "18900468-3db1-43f7-a8af-e75f079eb742" + } + ], + "isShared": true + }, + "shared_gwa_publish": { + "name": "Gold Shared IdP", + "description": "A Shared IdP for Teams to use", + "flow": "client-credentials", + "mode": "auto", + "clientAuthenticator": "client-secret", + "environmentDetails": [ + { + "clientId": "gwa-api", + "clientRegistration": "managed", + "clientSecret": "****", + "environment": "test", + "issuerUrl": "http://keycloak.localtest.me:9080/auth/realms/master", + "exists": true + } + ], + "clientRoles": [], + "clientMappers": [], + "availableScopes": [], + "resourceScopes": [], + "isShared": true, + "apiKeyName": "X-API-KEY" + }, "shared_IDP_body": { "name": "Sample Shared IdP new", "description": "A Shared IdP for Teams to use", @@ -194,8 +282,12 @@ "shared_IDP_inheritFrom": { "environmentDetails": [], "mode": "auto", - "clientRoles": ["administrator"], - "clientMappers": ["test-audience"], + "clientRoles": [ + "administrator" + ], + "clientMappers": [ + "test-audience" + ], "flow": "client-credentials", "clientAuthenticator": "client-secret", "name": "my-auth-client-secret-1", @@ -218,8 +310,12 @@ "clientId": "ap-my-auth-client-secret-1-test" } ], - "clientRoles": ["administrator"], - "clientMappers": ["test-audience"], + "clientRoles": [ + "administrator" + ], + "clientMappers": [ + "test-audience" + ], "isShared": false, "apiKeyName": "X-API-KEY", "inheritFrom": { @@ -282,11 +378,21 @@ "name": "CredentialIssuer.Admin" } ], - "permDomains": [".api.gov.bc.ca"], + "permDomains": [ + ".api.gov.bc.ca" + ], "permDataPlane": "local.dataplane", "permProtectedNs": "deny", "org": "ministry-of-health", "orgUnit": "planning-and-innovation-division" + }, + "userDefinedNamespace": { + "name": "gwanelatform", + "displayName": "Test for GWA test" + }, + "inValidNamespace": { + "name": "gwa", + "displayName": "Test for GWA test" } } -} +} \ No newline at end of file diff --git a/e2e/cypress/fixtures/apiowner.json b/e2e/cypress/fixtures/apiowner.json index c7b0844b3..fdd261d0e 100644 --- a/e2e/cypress/fixtures/apiowner.json +++ b/e2e/cypress/fixtures/apiowner.json @@ -2,17 +2,25 @@ "user": { "credentials": { "username": "janis@idir", - "email":"janis@testmail.com", + "email": "janis@testmail.com", "password": "awsummer" } }, "namespace": "newplatform", - "invalid_namespace": ["test_1234", "New_napespace", "Government"], + "invalid_namespace": [ + "test_1234", + "New_napespace", + "Government" + ], "serviceAccount": { - "scopes": ["GatewayConfig.Publish", "Namespace.Manage", "Content.Publish"] + "scopes": [ + "GatewayConfig.Publish", + "Namespace.Manage", + "Content.Publish" + ] }, "deleteResources": { - "namespace": "deleteplatform", + "namespace": "gw-2a3ef", "product": { "name": "Delete-Auto Test Product", "environment": { @@ -26,9 +34,11 @@ } } }, - "namespaceAccessPermissions": ["CredentialIssuer.Admin"], + "namespaceAccessPermissions": [ + "CredentialIssuer.Admin" + ], "product": { - "name": "Auto Test Product", + "name": "GWA Auto Test Product", "orgName": "Ministry of Health", "orgUnitName": "Planning and Innovation Division", "environment": { @@ -51,17 +61,21 @@ } }, "clientCredentials": { - "namespace": "ccplatform", + "namespace": "gw-e9cd4", "serviceName": "cc-service-for-platform", "Mark": { "userName": "mark", - "email":"mark@gmail.com", - "accessRole": ["Access.Manage"] + "email": "mark@gmail.com", + "accessRole": [ + "Access.Manage" + ] }, "Wendy": { "userName": "wendy", - "email":"wendy@test.com", - "accessRole": ["CredentialIssuer.Admin"] + "email": "wendy@test.com", + "accessRole": [ + "CredentialIssuer.Admin" + ] }, "jwtKeyPair": { "authProfile": { @@ -161,7 +175,9 @@ "name": "cy-client-id-secret-auth", "flow": "Client Credential Flow", "element": "cc-id-secret", - "scopes":["System.Write"], + "scopes": [ + "System.Write" + ], "clientAuthenticator": "Client ID and Secret", "environmentConfig": { "environment": "Test", @@ -206,7 +222,7 @@ }, "KongApiOnly": { "product": { - "name": "Auto Test Product", + "name": "GWA Auto Test Product", "orgName": "Ministry of Health", "orgUnitName": "Planning and Innovation Division", "environment": { @@ -222,7 +238,7 @@ }, "clientIdSecret_KongKeyToCC": { "product": { - "name": "Auto Test Product", + "name": "GWA Auto Test Product", "orgName": "Ministry of Health", "orgUnitName": "Planning and Innovation Division", "environment": { @@ -238,7 +254,9 @@ "name": "cy-client-id-secret-auth-updated", "flow": "Client Credential Flow", "element": "cc-id-secret", - "scopes":["System.Write"], + "scopes": [ + "System.Write" + ], "clientAuthenticator": "Client ID and Secret", "environmentConfig": { "environment": "Test", @@ -256,15 +274,18 @@ "element": "cc-id-secret", "clientAuthenticator": "Client ID and Secret", "environmentConfig": { - "isShardIDP":true + "isShardIDP": true } } } }, "namespacePreview": { - "namespace": "previewmode", + "namespace": "gw-07034", "serviceAccount": { - "scopes": ["GatewayConfig.Publish", "Namespace.Manage"] + "scopes": [ + "GatewayConfig.Publish", + "Namespace.Manage" + ] }, "product": { "name": "Namespace Preview Test Product", @@ -280,15 +301,20 @@ } }, "orgAssignment": { - "namespace": "orgassignment", + "namespace": "gw-57421", "serviceName": "a-service-for-orgassignment", "Mark": { "userName": "mark", - "email":"mark@gmail.com", - "accessRole": ["Access.Manage"] + "email": "mark@gmail.com", + "accessRole": [ + "Access.Manage" + ] }, "serviceAccount": { - "scopes": ["GatewayConfig.Publish", "Namespace.Manage"] + "scopes": [ + "GatewayConfig.Publish", + "Namespace.Manage" + ] }, "product": { "name": "Org Assignment Product", @@ -302,9 +328,9 @@ } } }, - "orgAdminNotification":{ - "parent" : "Your Organization Administrator has been notified to enable API Publishing to the Directory for the orgassignment namespace.", - "child" : "New namespaces must be reviewed by your Organization Administrator before you can publish APIs to the Directory. Your APIs are still in preview mode. For status inquiries, contact your Organization Administrator benny@test.com." + "orgAdminNotification": { + "parent": "Your Organization Administrator has been notified to enable API Publishing to the Directory for the orgassignment namespace.", + "child": "New namespaces must be reviewed by your Organization Administrator before you can publish APIs to the Directory. Your APIs are still in preview mode. For status inquiries, contact your Organization Administrator benny@test.com." } }, "orgAssignmentMultipleAdmin": { @@ -312,13 +338,17 @@ "serviceName": "a-service-for-orgassignment1", "Mark": { "userName": "mark", - "email":"mark@gmail.com", - "accessRole": ["Access.Manage"] + "email": "mark@gmail.com", + "accessRole": [ + "Access.Manage" + ] }, "GroupAccess": { - "members": ["benny@test.com", "wendy@test.com"] + "members": [ + "benny@test.com", + "wendy@test.com" + ] }, - "product": { "name": "Org Assignment Product", "environment": { @@ -331,9 +361,9 @@ } } }, - "orgAdminNotification":{ - "parent" : "Your Organization Administrator has been notified to enable API Publishing to the Directory for the orgassignment namespace.", - "child" : "New namespaces must be reviewed by your Organization Administrator before you can publish APIs to the Directory. Your APIs are still in preview mode. For status inquiries, contact your Organization Administrator benny@test.com." + "orgAdminNotification": { + "parent": "Your Organization Administrator has been notified to enable API Publishing to the Directory for the orgassignment namespace.", + "child": "New namespaces must be reviewed by your Organization Administrator before you can publish APIs to the Directory. Your APIs are still in preview mode. For status inquiries, contact your Organization Administrator benny@test.com." } }, "orgAssignmentOrgUnit": { @@ -341,13 +371,16 @@ "serviceName": "a-service-for-orgassign-unit", "Mark": { "userName": "mark", - "email":"mark@gmail.com", - "accessRole": ["Access.Manage"] + "email": "mark@gmail.com", + "accessRole": [ + "Access.Manage" + ] }, "GroupAccess": { - "members": ["benny@test.com"] + "members": [ + "benny@test.com" + ] }, - "product": { "name": "Org Assignment Unit Product", "environment": { @@ -360,15 +393,18 @@ } } }, - "orgAdminNotification":{ - "parent" : "Your Organization Administrator has been notified to enable API Publishing to the Directory for the orgassignment namespace.", - "child" : "New namespaces must be reviewed by your Organization Administrator before you can publish APIs to the Directory. Your APIs are still in preview mode. For status inquiries, contact your Organization Administrator benny@test.com." + "orgAdminNotification": { + "parent": "Your Organization Administrator has been notified to enable API Publishing to the Directory for the orgassignment namespace.", + "child": "New namespaces must be reviewed by your Organization Administrator before you can publish APIs to the Directory. Your APIs are still in preview mode. For status inquiries, contact your Organization Administrator benny@test.com." } }, "checkPermission": { - "namespace": "permission", + "namespace": "gw-e9cca", "serviceAccount": { - "scopes": ["GatewayConfig.Publish", "Namespace.Manage"] + "scopes": [ + "GatewayConfig.Publish", + "Namespace.Manage" + ] }, "product": { "name": "New-Auto Test Product", @@ -385,59 +421,88 @@ "grantPermission": { "Mark": { "userName": "Mark F Mark L", - "email":"mark@gmail.com", - "accessRole": ["Access.Manage"] + "email": "mark@gmail.com", + "accessRole": [ + "Access.Manage" + ] }, "Mark_NV": { "userName": "Mark F Mark L", - "email":"mark@gmail.com", - "accessRole": ["Namespace.View"] + "email": "mark@gmail.com", + "accessRole": [ + "Namespace.View" + ] }, "Wendy": { "userName": "Wendy F Wendy L", - "email":"wendy@test.com", - "accessRole": ["Namespace.Manage", "CredentialIssuer.Admin"] + "email": "wendy@test.com", + "accessRole": [ + "Namespace.Manage", + "CredentialIssuer.Admin" + ] }, "Wendy_NM": { "userName": "Wendy F Wendy L", - "email":"wendy@test.com", - "accessRole": ["Namespace.Manage"] + "email": "wendy@test.com", + "accessRole": [ + "Namespace.Manage" + ] }, "Wendy_CA": { "userName": "Wendy F Wendy L", - "email":"wendy@test.com", - "accessRole": ["CredentialIssuer.Admin"] + "email": "wendy@test.com", + "accessRole": [ + "CredentialIssuer.Admin" + ] }, "Wendy_GC": { "userName": "Wendy F Wendy L", - "email":"wendy@test.com", - "accessRole": ["GatewayConfig.Publish", "Namespace.View"] + "email": "wendy@test.com", + "accessRole": [ + "GatewayConfig.Publish", + "Namespace.View" + ] }, "Janis": { "userName": "Janis Smith", - "email":"janis@testmail.com", - "accessRole": ["Namespace.Manage", "CredentialIssuer.Admin"] + "email": "janis@testmail.com", + "accessRole": [ + "Namespace.Manage", + "CredentialIssuer.Admin" + ] } }, "revokePermission": { "Mark": { "userName": "mark", - "accessRole": ["Namespace.View", "Access.Manage"] + "accessRole": [ + "Namespace.View", + "Access.Manage" + ] }, "Mark_AM": { "userName": "mark", - "accessRole": ["Access.Manage"] + "accessRole": [ + "Access.Manage" + ] }, "Wendy": { "userName": "wendy", - "accessRole": ["Namespace.Manage"] + "accessRole": [ + "Namespace.Manage" + ] }, "Wendy_ci": { "userName": "wendy", - "accessRole": ["CredentialIssuer.Admin"] + "accessRole": [ + "CredentialIssuer.Admin" + ] }, "serviceAccount": { - "scopes": ["Namespace.Manage", "Content.Publish"] + "scopes": [ + "Namespace.Manage", + "Content.Publish" + ] } } }, @@ -445,27 +510,40 @@ "Mark": { "userName": "mark", "email": "mark@gmail.com", - "accessRole": ["Access.Manage", "Namespace.View"] + "accessRole": [ + "Access.Manage", + "Namespace.View" + ] }, "Janis": { "userName": "Janis Smith", "email": "janis@testmail.com", - "accessRole": ["CredentialIssuer.Admin","Namespace.Manage"] + "accessRole": [ + "CredentialIssuer.Admin", + "Namespace.Manage" + ] }, "OldUser": { "userName": "olduser@idir", "email": "olduser@testmail.com", - "accessRole": ["Access.Manage", "Namespace.View","Namespace.Manage"] + "accessRole": [ + "Access.Manage", + "Namespace.View", + "Namespace.Manage" + ] } }, "permission": { "Mark": { "userName": "mark", - "accessRole": ["Access.Manage", "Namespace.View"] + "accessRole": [ + "Access.Manage", + "Namespace.View" + ] } }, "apiTest": { - "namespace": "apiplatform", + "namespace": "gw-65590", "delete_namespace": "testplatform", "product": { "name": "API Auto Test Product", @@ -480,197 +558,197 @@ } } }, - "expectedActivitiesResponse":[ + "expectedActivitiesResponse": [ { - "result": "success", - "message": "{actor} {action} {entity} for {application} ({consumer}) to access {product} {environment}", - "params": { - "actor": "Mark F Mark L", - "action": "approved", - "entity": "access request", - "accessRequest": "2", - "environment": "dev", - "product": "Auto Test Product", - "application": "Auto Test App to check labels", - "consumer": "9C3EFC92-8F3DE9D3B74" - }, - "activityAt": "2022-09-20T20:39:05.318Z" + "result": "success", + "message": "{actor} {action} {entity} for {application} ({consumer}) to access {product} {environment}", + "params": { + "actor": "Mark F Mark L", + "action": "approved", + "entity": "access request", + "accessRequest": "2", + "environment": "dev", + "product": "GWA Auto Test Product", + "application": "Auto Test App to check labels", + "consumer": "9C3EFC92-8F3DE9D3B74" + }, + "activityAt": "2022-09-20T20:39:05.318Z" }, { - "result": "success", - "message": "{actor} {action} for {application} ({consumer}) to access {product} {environment} ({note})", - "params": { - "actor": "Harley Jones", - "action": "received credentials", - "entity": "access", - "note": "access pending approval", - "accessRequest": "2", - "environment": "dev", - "product": "Auto Test Product", - "application": "Auto Test App to check labels", - "consumer": "9C3EFC92-8F3DE9D3B73" - }, - "activityAt": "2022-09-20T20:38:40.857Z" + "result": "success", + "message": "{actor} {action} for {application} ({consumer}) to access {product} {environment} ({note})", + "params": { + "actor": "Harley Jones", + "action": "received credentials", + "entity": "access", + "note": "access pending approval", + "accessRequest": "2", + "environment": "dev", + "product": "GWA Auto Test Product", + "application": "Auto Test App to check labels", + "consumer": "9C3EFC92-8F3DE9D3B73" + }, + "activityAt": "2022-09-20T20:38:40.857Z" }, { - "result": "success", - "message": "{actor} {action} {entity} for {consumer}", - "params": { - "actor": "Mark F Mark L", - "action": "update", - "entity": "consumer control" - }, - "activityAt": "2022-09-20T20:38:19.732Z" + "result": "success", + "message": "{actor} {action} {entity} for {consumer}", + "params": { + "actor": "Mark F Mark L", + "action": "update", + "entity": "consumer control" + }, + "activityAt": "2022-09-20T20:38:19.732Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:38:19.678Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:38:19.678Z" }, { - "result": "success", - "message": "{actor} {action} {entity} for {consumer}", - "params": { - "actor": "Mark F Mark L", - "action": "update", - "entity": "consumer control" - }, - "activityAt": "2022-09-20T20:38:14.706Z" + "result": "success", + "message": "{actor} {action} {entity} for {consumer}", + "params": { + "actor": "Mark F Mark L", + "action": "update", + "entity": "consumer control" + }, + "activityAt": "2022-09-20T20:38:14.706Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:38:12.664Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:38:12.664Z" }, { - "result": "success", - "message": "{actor} {action} {entity} for {consumer}", - "params": { - "actor": "Mark F Mark L", - "action": "update", - "entity": "consumer control" - }, - "activityAt": "2022-09-20T20:38:08.054Z" + "result": "success", + "message": "{actor} {action} {entity} for {consumer}", + "params": { + "actor": "Mark F Mark L", + "action": "update", + "entity": "consumer control" + }, + "activityAt": "2022-09-20T20:38:08.054Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:38:07.996Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:38:07.996Z" }, { - "result": "success", - "message": "{actor} {action} {entity} for {consumer}", - "params": { - "actor": "Mark F Mark L", - "action": "update", - "entity": "consumer control" - }, - "activityAt": "2022-09-20T20:38:00.947Z" + "result": "success", + "message": "{actor} {action} {entity} for {consumer}", + "params": { + "actor": "Mark F Mark L", + "action": "update", + "entity": "consumer control" + }, + "activityAt": "2022-09-20T20:38:00.947Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:38:00.899Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:38:00.899Z" }, { - "result": "success", - "message": "{actor} {action} {entity} for {consumer}", - "params": { - "actor": "Mark F Mark L", - "action": "update", - "entity": "consumer control" - }, - "activityAt": "2022-09-20T20:37:56.316Z" + "result": "success", + "message": "{actor} {action} {entity} for {consumer}", + "params": { + "actor": "Mark F Mark L", + "action": "update", + "entity": "consumer control" + }, + "activityAt": "2022-09-20T20:37:56.316Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:37:56.252Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:37:56.252Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:37:56.192Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:37:56.192Z" }, { - "result": "success", - "message": "{actor} {action} {entity} for {consumer}", - "params": { - "actor": "Mark F Mark L", - "action": "update", - "entity": "consumer control" - }, - "activityAt": "2022-09-20T20:37:47.630Z" + "result": "success", + "message": "{actor} {action} {entity} for {consumer}", + "params": { + "actor": "Mark F Mark L", + "action": "update", + "entity": "consumer control" + }, + "activityAt": "2022-09-20T20:37:47.630Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:37:47.574Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:37:47.574Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:37:47.516Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:37:47.516Z" }, { - "result": "success", - "message": "{actor} {action} {entity} for {consumer}", - "params": { - "actor": "Mark F Mark L", - "action": "update", - "entity": "consumer control" - }, - "activityAt": "2022-09-20T20:37:38.815Z" + "result": "success", + "message": "{actor} {action} {entity} for {consumer}", + "params": { + "actor": "Mark F Mark L", + "action": "update", + "entity": "consumer control" + }, + "activityAt": "2022-09-20T20:37:38.815Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:37:38.763Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:37:38.763Z" }, { - "result": "completed", - "message": "GatewayConsumerPlugin Update", - "params": { - "actor": "Unknown Actor" - }, - "activityAt": "2022-09-20T20:37:38.696Z" + "result": "completed", + "message": "GatewayConsumerPlugin Update", + "params": { + "actor": "Unknown Actor" + }, + "activityAt": "2022-09-20T20:37:38.696Z" }, { - "result": "success", - "message": "{actor} {action} {entity} for {consumer}", - "params": { - "actor": "Mark F Mark L", - "action": "update", - "entity": "consumer control" - }, - "activityAt": "2022-09-20T20:37:30.570Z" + "result": "success", + "message": "{actor} {action} {entity} for {consumer}", + "params": { + "actor": "Mark F Mark L", + "action": "update", + "entity": "consumer control" + }, + "activityAt": "2022-09-20T20:37:30.570Z" } -] -} + ] +} \ No newline at end of file diff --git a/e2e/cypress/fixtures/cc-service-gwa.yml b/e2e/cypress/fixtures/cc-service-gwa.yml new file mode 100644 index 000000000..77b815834 --- /dev/null +++ b/e2e/cypress/fixtures/cc-service-gwa.yml @@ -0,0 +1,19 @@ +services: +- name: cc-service-for-platform + host: httpbin.org + tags: [ns.ccplatform] + port: 443 + protocol: https + retries: 0 + routes: + - name: cc-service-for-platform-route + tags: [ns.ccplatform] + hosts: + - cc-service-for-platform.api.gov.bc.ca + paths: + - / + methods: + - GET + strip_path: false + https_redirect_status_code: 426 + path_handling: v0 \ No newline at end of file diff --git a/e2e/cypress/fixtures/service-clear-resources-gwa.yml b/e2e/cypress/fixtures/service-clear-resources-gwa.yml new file mode 100644 index 000000000..11a83ad3d --- /dev/null +++ b/e2e/cypress/fixtures/service-clear-resources-gwa.yml @@ -0,0 +1,19 @@ +services: +- name: service-for-deleteplatform + host: httpbin.org + tags: [ns.deleteplatform] + port: 443 + protocol: https + retries: 0 + routes: + - name: service-for-deleteplatform-route + tags: [ns.deleteplatform] + hosts: + - service-for-deleteplatform.api.gov.bc.ca + paths: + - / + methods: + - GET + strip_path: false + https_redirect_status_code: 426 + path_handling: v0 \ No newline at end of file diff --git a/e2e/cypress/fixtures/service-gwa.yml b/e2e/cypress/fixtures/service-gwa.yml new file mode 100644 index 000000000..77ddda761 --- /dev/null +++ b/e2e/cypress/fixtures/service-gwa.yml @@ -0,0 +1,38 @@ +services: +- name: a-service-for-newplatform + host: httpbin.org + tags: [ns.newplatform] + port: 443 + protocol: https + retries: 0 + routes: + - name: a-service-for-newplatform-route + tags: [ns.newplatform] + hosts: + - a-service-for-newplatform.api.gov.bc.ca + paths: + - / + methods: + - GET + strip_path: false + https_redirect_status_code: 426 + path_handling: v0 + +- name: a-service-for-newplatform-test + host: httpbin.org + tags: [ns.newplatform] + port: 443 + protocol: https + retries: 0 + routes: + - name: a-service-for-newplatform-test-route + tags: [ns.newplatform] + hosts: + - a-service-for-newplatform-test.api.gov.bc.ca + paths: + - / + methods: + - GET + strip_path: false + https_redirect_status_code: 426 + path_handling: v0 \ No newline at end of file diff --git a/e2e/cypress/fixtures/service-permission-gwa.yml b/e2e/cypress/fixtures/service-permission-gwa.yml new file mode 100644 index 000000000..cbcf1e7c7 --- /dev/null +++ b/e2e/cypress/fixtures/service-permission-gwa.yml @@ -0,0 +1,19 @@ +services: +- name: service-for-permission + host: httpbin.org + tags: [ns.permission] + port: 443 + protocol: https + retries: 0 + routes: + - name: service-for-permission-route + tags: [ns.permission] + hosts: + - service-for-permission.api.gov.bc.ca + paths: + - / + methods: + - GET + strip_path: false + https_redirect_status_code: 426 + path_handling: v0 diff --git a/e2e/cypress/fixtures/service-plugin-key-auth-only.yml b/e2e/cypress/fixtures/service-plugin-key-auth-only.yml index 897d275a7..69ad7b220 100644 --- a/e2e/cypress/fixtures/service-plugin-key-auth-only.yml +++ b/e2e/cypress/fixtures/service-plugin-key-auth-only.yml @@ -9,4 +9,4 @@ plugins: config: key_names: - x-api-key - anonymous: 1aaf2351-0622-4cc4-8d14-32e78771e620 + anonymous: bb2c1e04-96ce-4b1d-baf4-a7e681a862a6 diff --git a/e2e/cypress/fixtures/test_data/gwa-cli.json b/e2e/cypress/fixtures/test_data/gwa-cli.json new file mode 100644 index 000000000..a993bc254 --- /dev/null +++ b/e2e/cypress/fixtures/test_data/gwa-cli.json @@ -0,0 +1,12 @@ +{ + "content": { + "help": "You can login via device login or by using client credentials\n\nTo use device login, simply run the command like so:\n $ gwa login\n\nTo use your credentials you must supply both a client-id and client-secret:\n $ gwa login --client-id --client-secret \n\nUsage:\n gwa login [flags]\n\nFlags:\n --client-id string Your gateway's client ID\n --client-secret string Your gateway's client secret\n -h, --help help for login\n\nGlobal Flags:\n --host string Set the default host to use for the API\n --namespace string Assign the namespace you would like to use\n --scheme string Use to override default https", + "login_help": "Log in to your IDIR account\n\nUsage:\n gwa login [flags]\n\nFlags:\n --client-id string Your gateway's client ID\n --client-secret string Your gateway's client secret\n -h, --help help for login\n\nGlobal Flags:\n --host string Set the default host to use for the API\n --scheme string Use to override default https", + "namespace_help": "Longer explanation to come...\n\nUsage:\n gwa namespace [command]\n\nAvailable Commands:\n create Create a new namespace\n list List all your managed namespaces\n\nFlags:\n -h, --help help for namespace\n\nGlobal Flags:\n --host string Set the default host to use for the API\n --namespace string Assign the namespace you would like to use\n --scheme string Use to override default https\n\nUse \"gwa namespace [command] --help\" for more information about a command.", + "config_help": "Configuration commands\n\nUsage:\n gwa config [command]\n\nAvailable Commands:\n set Write a specific global setting\n\nFlags:\n -h, --help help for config\n\nGlobal Flags:\n --host string Set the default host to use for the API\n --namespace string Assign the namespace you would like to use\n --scheme string Use to override default https\n\nUse \"gwa config [command] --help\" for more information about a command." + }, + "credentials": { + "clientID": "gwa-api", + "clientSecret": "18900468-3db1-43f7-a8af-e75f079eb742" + } +} diff --git a/e2e/cypress/fixtures/test_data/gwa-dataset.yaml b/e2e/cypress/fixtures/test_data/gwa-dataset.yaml new file mode 100644 index 000000000..28eb9d201 --- /dev/null +++ b/e2e/cypress/fixtures/test_data/gwa-dataset.yaml @@ -0,0 +1,15 @@ +name: gwa-auto-test-product +license_title: Open Government Licence - British Columbia +security_class: PUBLIC +view_audience: Public +download_audience: Public +record_publish_date: '2017-09-05' +notes: For Test Purpose +title: GWA Auto Test Product +tags: + - tag1 + - tag2 +organization: ministry-of-health +organizationUnit: planning-and-innovation-division +isInCatalog: true +isDraft: false \ No newline at end of file diff --git a/e2e/cypress/fixtures/test_data/gwa-issuer.yaml b/e2e/cypress/fixtures/test_data/gwa-issuer.yaml new file mode 100644 index 000000000..fb1c7a992 --- /dev/null +++ b/e2e/cypress/fixtures/test_data/gwa-issuer.yaml @@ -0,0 +1,12 @@ +name: Gold Shared IdP +description: A Shared IdP for Teams to use +flow: client-credentials +clientAuthenticator: client-secret +mode: auto +environmentDetails: + - environment: test + issuerUrl: http://keycloak.localtest.me:9080/auth/realms/master + clientRegistration: managed + clientId: gwa-api + clientSecret: 18900468-3db1-43f7-a8af-e75f079eb742 +isShared: true \ No newline at end of file diff --git a/e2e/cypress/fixtures/test_data/gwa-product.yaml b/e2e/cypress/fixtures/test_data/gwa-product.yaml new file mode 100644 index 000000000..8b95400a3 --- /dev/null +++ b/e2e/cypress/fixtures/test_data/gwa-product.yaml @@ -0,0 +1,8 @@ +name: GWA Auto Test Product +appId: 'KFH78YU956RE' +environments: + - name: dev + active: false + approval: false + flow: public + appId: '406CB7CF' \ No newline at end of file diff --git a/e2e/cypress/pageObjects/activity.ts b/e2e/cypress/pageObjects/activity.ts index 2d1f17774..357ff8d3f 100644 --- a/e2e/cypress/pageObjects/activity.ts +++ b/e2e/cypress/pageObjects/activity.ts @@ -31,12 +31,11 @@ class ActivityPage { responseText = responseText.replaceAll("{", "${filteredResponse[index].params.") const regexp = /\${([^{]+)}/g; if (!(record.result === 'failed')) { - result = responseText.replace(regexp, function (ignore: any, key: any) { + result = responseText.replace(regexp, function (ignore: any, key: any) { return eval(key); }); } - else if(responseText.includes("Failed to Apply Workflow - IssuerMisconfigError")) - { + else if (responseText.includes("Failed to Apply Workflow - IssuerMisconfigError")) { result = 'Failed to Apply Workflow - IssuerMisconfigError: undefined' } if (result === activityText) { @@ -61,12 +60,25 @@ class ActivityPage { } loadMoreRecords() { + // cy.get(this.loadButton) + // .then(($button) => { + // cy.wrap($button).click() + // cy.wait(2000) + // debugger + // if ($button.is(':visible')) { + // // If the button is still visible, call the function again + // debugger + // this.loadMoreRecords(); + // } + // }); cy.get("body").then($body => { if ($body.find(this.loadButton).length > 0) { cy.get(this.loadButton).click({ force: true }) cy.wait(2000) + this.loadMoreRecords() } }) } + } export default ActivityPage \ No newline at end of file diff --git a/e2e/cypress/pageObjects/apiDirectory.ts b/e2e/cypress/pageObjects/apiDirectory.ts index 984129983..f0911e67e 100644 --- a/e2e/cypress/pageObjects/apiDirectory.ts +++ b/e2e/cypress/pageObjects/apiDirectory.ts @@ -17,6 +17,7 @@ class ApiDirectoryPage { orgDropDown: string = '[data-testid="orgDropDown"]' orgUnitDropDown: string = '[data-testid="orgUnitDropDown"]' addOrganizationBtn: string = '[data-testid="addOrganizationBtn"]' + jwksPublicKeyCheckBox: string = '[data-testid="access-rqst-app-env-public-key"]' createAccessRequest(product: any, app: any, accessRqst: any, elevatedAccess?: boolean) { @@ -57,6 +58,14 @@ class ApiDirectoryPage { cy.get(this.submitBtn).click() } + checkInactiveEnvironmentAccessReqOption(product: any, app: any){ + cy.contains('a', product.name, { timeout: 10000 }).should('be.visible'); + cy.contains(product.name).click() + cy.get(this.rqstAccessBtn).click() + cy.get(this.appSelect).select(app.name) + cy.get('[data-testid=access-rqst-app-env-' + product.environment + ']').should('not.exist'); + } + isProductDisplay(productName: string, expResult: boolean) { cy.get("button").then(($btn) => { var flag = true @@ -105,14 +114,14 @@ class ApiDirectoryPage { cy.get(this.addOrganizationBtn).click({ force: true }) } - checkOrgAdminNotificationBanner(notification: any) { + checkOrgAdminNotificationBanner(notification: any, childMessage: string) { cy.get('[data-testid="org-assignment-notification-parent"]').invoke('text').then((text) => { text = this.getPlainText(text) - assert.equal(text, notification.parent) + assert.equal(text, notification) cy.contains('button', 'Learn More').click() cy.get('[data-testid="org-assignment-notification-child"]').invoke('text').then((text) => { text = this.getPlainText(text) - assert.equal(text, notification.child) + assert.equal(text, childMessage) }) }) } @@ -120,6 +129,24 @@ class ApiDirectoryPage { getPlainText(text: string): string { return text.replace(/[\r\n]/g, '').replace(/\s+/g, " ") } + + enterInvalidJWTKey(product: any, app: any, accessRqst: any) { + cy.contains(product.name).click() + cy.get(this.rqstAccessBtn).click() + cy.get(this.appSelect).select(app.name) + cy.get('[data-testid=access-rqst-app-env-' + product.environment + ']').click() + cy.get('body', { log: false }).then(($body) => { + if ($body.find(this.legatTermCheckBox).length > 0) { + cy.get(this.legatTermCheckBox).first().click() + } + }) + cy.readFile('cypress/fixtures/state/jwtReGenPublicKey_new.pub').then((publicKeyKey) => { + cy.get(this.jwksPublicKeyCheckBox).click() + cy.get('[name="clientCertificate"]').click().type(publicKeyKey + "End of File") + }) + cy.get(this.additionalNotes).type(accessRqst.notes) + cy.get(this.submitBtn).click() + } } export default ApiDirectoryPage diff --git a/e2e/cypress/pageObjects/authProfile.ts b/e2e/cypress/pageObjects/authProfile.ts index 09ce67d09..cb69ccb7a 100644 --- a/e2e/cypress/pageObjects/authProfile.ts +++ b/e2e/cypress/pageObjects/authProfile.ts @@ -165,6 +165,21 @@ class AuthorizationProfile { cy.contains(issuerURL).should('exist') cy.visit(this.path) } + + deleteAuthProfile(authProfileName: string) { + cy.wait(2000) + let authProfileText + cy.get(this.profileTable).find('tr').each(($e1, index, $list) => { + authProfileText = $e1.find('td:nth-child(1)').text(); + if (authProfileText===authProfileName) { + cy.wrap($e1).find('button').eq(1).click() + cy.wait(2000) + cy.wrap($e1).find('button').last().click({force: true}) + cy.verifyToastMessage(authProfileName +' deleted') + return false + } + }) + } } export default AuthorizationProfile diff --git a/e2e/cypress/pageObjects/consumers.ts b/e2e/cypress/pageObjects/consumers.ts index 95e50f454..93887f5b6 100644 --- a/e2e/cypress/pageObjects/consumers.ts +++ b/e2e/cypress/pageObjects/consumers.ts @@ -156,8 +156,13 @@ export default class ConsumersPage { cy.get(this.consumerDialogSaveBtn, { timeout: 2000 }).should('not.exist') } - approvePendingRequest() { - cy.get(this.approveBtn).click({ force: true }) + approvePendingRequest(flag?: boolean) { + if (flag) { + cy.get(this.approveBtn).click({ force: true, multiple : true }) + } + else { + cy.get(this.approveBtn).click({ force: true }) + } cy.wait(1000) cy.verifyToastMessage("Access request approved") } @@ -318,12 +323,20 @@ export default class ConsumersPage { cy.wait(1000) } - selectAuthorizationScope(scopes: any) { + selectAuthorizationScope(scopes: any, isChecked = true) { cy.contains("Authorization").click() - scopes.forEach(function (scope: string) { - cy.get('[data-testid="client-scope-' + scope + '"]').click() - cy.wait(1000) - }) + if (isChecked) { + scopes.forEach(function (scope: string) { + cy.get('[data-testid="client-scope-' + scope + '"]').find('[type="checkbox"]').check({ force: true }); + cy.wait(1000) + }) + } + else { + scopes.forEach(function (scope: string) { + cy.get('[data-testid="client-scope-' + scope + '"]').find('[type="checkbox"]').uncheck({ force: true }); + cy.wait(1000) + }) + } } selectClientRole(roles: any) { @@ -339,21 +352,26 @@ export default class ConsumersPage { cy.log($row.find('td:nth-child(1)').text()) if ($row.find('td:nth-child(1)').text() == consumerID) { cy.wrap($row).find('button').first().click() - cy.get('[data-testid="consumer-delete-menuitem"]').last().click({force:true}) + cy.get('[data-testid="consumer-delete-menuitem"]').last().click({ force: true }) } }) } + + saveAppliedConfig(){ + cy.get(this.consumerDialogSaveBtn).click() + cy.wait(1000) + } clickOnLinkConsumerToNamespaceBtn() { - cy.get(this.linkConsumerToNamespaceBtn).click({force:true}) + cy.get(this.linkConsumerToNamespaceBtn).click({ force: true }) } linkTheConsumerToNamespace(consumerID: any) { cy.get(this.userNameTxt).type(consumerID) - cy.get(this.linkBtn).click({force:true}) + cy.get(this.linkBtn).click({ force: true }) } - getText(){ + getText() { cy.get('[data-testid="all-consumer-control-tbl"]').find('tr').last().find('td').first().find('a').as('inputValue') } } \ No newline at end of file diff --git a/e2e/cypress/pageObjects/keycloakClientScopes.ts b/e2e/cypress/pageObjects/keycloakClientScopes.ts new file mode 100644 index 000000000..cdb96bd90 --- /dev/null +++ b/e2e/cypress/pageObjects/keycloakClientScopes.ts @@ -0,0 +1,23 @@ +import { Assertion } from "chai" + +class keycloakClientScopesPage { + path: string = '/' + + clientTab: string = '[data-ng-controller="ClientTabCtrl"]' + + selectTab(tabName: string){ + cy.get(this.clientTab).contains('a',tabName).click() + } + + verifyAssignedScope(scope: string, expResult:boolean) + { + if(expResult){ + cy.get('[id="assigned"]').find('[title="'+scope+'"]').should('exist'); + } + else{ + cy.get('[id="assigned"]').find('[title="'+scope+'"]').should('not.exist'); + } + } +} + +export default keycloakClientScopesPage diff --git a/e2e/cypress/pageObjects/myAccess.ts b/e2e/cypress/pageObjects/myAccess.ts index 9865f08b4..65b5d8bd6 100644 --- a/e2e/cypress/pageObjects/myAccess.ts +++ b/e2e/cypress/pageObjects/myAccess.ts @@ -14,6 +14,7 @@ class myAccessPage { regenerateCredentialCloseBtn: string = '[data-testid=regenerate-credentials-done-button]' collectCredentialsBtn: string = '[data-testid="generate-credentials-button"]' clientIDValueTxt: string = '[data-testid="sa-new-creds-client-id"]' + cancelRequestAccessRequest: string = '[data-testid="access-request-cancel-button"]' path: string = '/devportal/access' @@ -37,6 +38,11 @@ class myAccessPage { cy.get(this.closeRequestAccesss).click() } + cancelRequestAccessPopUp() + { + cy.get(this.cancelRequestAccessRequest).click() + } + saveReGenAPIKeyValue(): void { cy.get(this.apiKyeValueTxt).invoke('val').then(($apiKey: any) => { cy.saveState('newApiKey', $apiKey) @@ -166,6 +172,7 @@ class myAccessPage { cy.saveState('clientID', $clientID) }) } + } export default myAccessPage diff --git a/e2e/cypress/pageObjects/products.ts b/e2e/cypress/pageObjects/products.ts index 31f030291..e1b1ff727 100644 --- a/e2e/cypress/pageObjects/products.ts +++ b/e2e/cypress/pageObjects/products.ts @@ -319,6 +319,25 @@ class Products { cy.get(this.messageForNotDataset).should('be.visible'); } + + verifyAuthScope(expAuthValue: string){ + cy.get(this.envCfgAuthzDropdown).find(':selected').invoke('text').then(($auth: any) => { + assert.equal($auth,expAuthValue ) + }) + } + + verifyIssuer(expIssuerValue: string){ + cy.get('[name="credentialIssuer"]').find(':selected').invoke('text').then(($issuer: any) => { + assert.equal($issuer,expIssuerValue ) + }) + } + + verifyDataset(expDatasetValue: string, product: string){ + this.editProduct(product) + cy.get(this.catelogueDropDown).invoke('val').then(($dataset: any) => { + assert.equal($dataset,expDatasetValue) + }) + } } export default Products diff --git a/e2e/cypress/support/auth-commands.ts b/e2e/cypress/support/auth-commands.ts index 23f266765..a031aed1c 100644 --- a/e2e/cypress/support/auth-commands.ts +++ b/e2e/cypress/support/auth-commands.ts @@ -1,9 +1,6 @@ import * as jwt from 'jsonwebtoken' import HomePage from '../pageObjects/home' import LoginPage from '../pageObjects/login' -import request = require('request') -import { method } from 'cypress/types/bluebird' -import { url } from 'inspector' import NamespaceAccessPage from '../pageObjects/namespaceAccess' import _ = require('cypress/types/lodash') import { checkElementExists } from './e2e' @@ -82,9 +79,9 @@ Cypress.Commands.add('keycloakLogin', (username: string, password: string) => { cy.get(login.loginSubmitButton).click() }) -Cypress.Commands.add('getLastConsumerID',() =>{ - let id : any - cy.get('[data-testid="all-consumer-control-tbl"]').find('tr').last().find('td').first().find('a').then(($text)=>{ +Cypress.Commands.add('getLastConsumerID', () => { + let id: any + cy.get('[data-testid="all-consumer-control-tbl"]').find('tr').last().find('td').first().find('a').then(($text) => { id = $text.text() return id }) @@ -112,7 +109,7 @@ Cypress.Commands.add('resetCredential', (accessRole: string) => { }) }) -Cypress.Commands.add('getUserSessionTokenValue', (namespace: string) => { +Cypress.Commands.add('getUserSessionTokenValue', (namespace: string, isNamespaceSelected?: true) => { const login = new LoginPage() const home = new HomePage() const na = new NamespaceAccessPage() @@ -128,7 +125,9 @@ Cypress.Commands.add('getUserSessionTokenValue', (namespace: string) => { cy.login(user.credentials.username, user.credentials.password) cy.log('Logged in!') // home.useNamespace(apiTest.namespace) - home.useNamespace(namespace) + if (isNamespaceSelected || undefined) { + home.useNamespace(namespace) + } cy.get('@login').then(function (xhr: any) { userSession = xhr.response.headers['x-auth-request-access-token'] return userSession @@ -234,7 +233,7 @@ Cypress.Commands.add('getAccessToken', (client_id: string, client_secret: string cy.log('> Get Token') }) -Cypress.Commands.add('getServiceOrRouteID', (configType: string) => { +Cypress.Commands.add('getServiceOrRouteID', (configType: string, host: string) => { const config = configType.toLowerCase() cy.request({ method: 'GET', @@ -242,10 +241,10 @@ Cypress.Commands.add('getServiceOrRouteID', (configType: string) => { }).then((res) => { expect(res.status).to.eq(200) if (config === 'routes') { - cy.saveState(config + 'ID', Cypress._.get((Cypress._.filter(res.body.data, ["hosts", ["a-service-for-newplatform.api.gov.bc.ca"]]))[0], 'id')) + cy.saveState(config + 'ID', Cypress._.get((Cypress._.filter(res.body.data, ["hosts", [host+".api.gov.bc.ca"]]))[0], 'id')) } else { - cy.saveState(config + 'ID', Cypress._.get((Cypress._.filter(res.body.data, ["name", "a-service-for-newplatform"]))[0], 'id')) + cy.saveState(config + 'ID', Cypress._.get((Cypress._.filter(res.body.data, ["name", host]))[0], 'id')) } }) }) @@ -260,17 +259,18 @@ Cypress.Commands.add('publishApi', (fileName: string, namespace: string, flag?: () => { cy.wait(3000) cy.get('@accessTokenResponse').then((res: any) => { - const options = { - method: 'PUT', - url: Cypress.env('GWA_API_URL') + '/namespaces/' + namespace + '/gateway', - } - formDataRequest(options, res.body.access_token, fileName, requestName) - cy.wait(`@${requestName}`).then((res: any) => { - cy.wrap(res.response).as('publishAPIResponse') + cy.executeCliCommand('gwa config set --namespace ' + namespace).then((response) => { + cy.executeCliCommand('gwa config set --token ' + res.body.access_token).then((response) => { + { + expect(response.stdout).to.contain("Config settings saved") + cy.executeCliCommand('gwa pg ./cypress/fixtures/' + fileName).then((response) => { + expect(response.stdout).to.contain("Gateway config published") + }) + } + }) }) }) - } - ) + }) }) }) @@ -295,8 +295,7 @@ Cypress.Commands.add('makeKongRequest', (serviceName: string, methodType: string cy.fixture('state/regen').then((creds: any) => { cy.wait(5000) let token = key - if (key==undefined) - { + if (key == undefined) { token = creds.apikey } const service = serviceName @@ -361,7 +360,7 @@ Cypress.Commands.add('updateKongPlugin', (pluginName: string, name: string, endP let endpoint if (pluginName == '') endpoint = 'plugins' - else if(id !== undefined) + else if (id !== undefined) endpoint = pluginName.toLowerCase() + '/' + id.toString() + '/' + 'plugins' endpoint = (typeof endPoint !== 'undefined') ? endPoint : endpoint body = config[name] @@ -379,7 +378,7 @@ Cypress.Commands.add('updateKongPlugin', (pluginName: string, name: string, endP Cypress.Commands.add('updateKongPluginForJSONRequest', (jsonBody: string, endPoint: string, verb = 'POST') => { cy.fixture('state/store').then((creds: any) => { let body = {} - let headers = {"content-type": "application/json", "accept": "application/json"} + let headers = { "content-type": "application/json", "accept": "application/json" } body = jsonBody return cy.request({ url: Cypress.env('KONG_CONFIG_URL') + '/' + endPoint, @@ -503,6 +502,9 @@ Cypress.Commands.add('updatePropertiesOfPluginFile', (filename: string, property if (propertyName === "config.anonymous") { obj.plugins[0].config.anonymous = propertyValue } + else if (propertyName === "tags"){ + obj.plugins[0][propertyName] = propertyValue + } else { Object.keys(obj.services).forEach(function (key, index) { if (propertyName == "methods") { @@ -562,14 +564,19 @@ Cypress.Commands.add("generateKeyPair", () => { cy.writeFile('cypress/fixtures/state/jwtReGenPrivateKey_new.pem', privateKeyPem) cy.writeFile('cypress/fixtures/state/jwtReGenPublicKey_new.pub', publicKeyPem) - + }) -Cypress.Commands.add('forceVisit', (url:string) => { +Cypress.Commands.add('forceVisit', (url: string) => { cy.window().then(win => { - return win.open(url, '_self'); + return win.open(url, '_self'); }); -}); +}); + +Cypress.Commands.add('updateJsonBoby', (json: any, key: string, newValue: string):any => { + json[key] = newValue + return json +}); const formDataRequest = ( options: formDataRequestOptions, diff --git a/e2e/cypress/support/global.d.ts b/e2e/cypress/support/global.d.ts index 53ece714d..71d1f2e93 100644 --- a/e2e/cypress/support/global.d.ts +++ b/e2e/cypress/support/global.d.ts @@ -19,7 +19,7 @@ declare namespace Cypress { preserveCookies(): void - makeKongRequest(serviceName : string, methodType : string, key?: string): Chainable + makeKongRequest(serviceName: string, methodType: string, key?: string): Chainable makeKongGatewayRequestUsingClientIDSecret(hostURL: string, methodType?: string): Chainable @@ -36,44 +36,44 @@ declare namespace Cypress { client_secret: string ): Chainable> - publishApi(fileName: string, namespace: string, flag?:boolean): Chainable> - - getServiceOrRouteID(configType: string + publishApi(fileName: string, namespace: string, flag?: boolean): Chainable> + + getServiceOrRouteID(configType: string, host: string ): Chainable> - updateKongPlugin(pluginName : string, name : string, endPoint?: string, verb?: string): Chainable> + updateKongPlugin(pluginName: string, name: string, endPoint?: string, verb?: string): Chainable> - makeKongGatewayRequest(endpoint: string, requestName:string, methodType: string): Chainable> + makeKongGatewayRequest(endpoint: string, requestName: string, methodType: string): Chainable> // generateKeystore() : Chainable - generateKeystore() : void + generateKeystore(): void - setHeaders(headerValues : any) : void + setHeaders(headerValues: any): void - setRequestBody(requestBody : any) : void + setRequestBody(requestBody: any): void - setAuthorizationToken (token : string) : void + setAuthorizationToken(token: string): void - makeAPIRequest(endPoint: string,methodType: string): Chainable> + makeAPIRequest(endPoint: string, methodType: string): Chainable> getUserSession(): Chainable> - compareJSONObjects(actualResponse: any, expectedResponse:any, indexFlag?: boolean) : Chainable> + compareJSONObjects(actualResponse: any, expectedResponse: any, indexFlag?: boolean): Chainable> - getUserSessionTokenValue(namespace: string): Chainable> + getUserSessionTokenValue(namespace: string, isNamespaceSelected?: boolean): Chainable> getUserSessionResponse(): Chainable> - getTokenUsingJWKCredentials(credential: any, privateKey: any):Chainable> + getTokenUsingJWKCredentials(credential: any, privateKey: any): Chainable> - verifyToastMessage(msg: string):Chainable> + verifyToastMessage(msg: string): Chainable> - updatePluginFile (filename: string, serviceName: string, pluginFileName: string):Chainable> + updatePluginFile(filename: string, serviceName: string, pluginFileName: string): Chainable> - updateElementsInPluginFile(filename: string, elementName: string, elementValue: string):Chainable> + updateElementsInPluginFile(filename: string, elementName: string, elementValue: string): Chainable> - updatePropertiesOfPluginFile(filename: string, propertyName: any, propertyValue: any):Chainable> + updatePropertiesOfPluginFile(filename: string, propertyName: any, propertyValue: any): Chainable> keycloakLogin(username: string, password: string): Chainable @@ -81,13 +81,25 @@ declare namespace Cypress { getLastConsumerID(): Chainable - generateKeyPair() : void + generateKeyPair(): void // isProductDisplay(productName: string, expResult : boolean) :Chainable> - updateJsonValue(jsonBody: any, jsonPath: string, newValue: string, index?: number): Chainable + updateJsonValue(filePath: string, jsonPath: string, newValue: string, index?: any): Chainable updateKongPluginForJSONRequest(jsonBody: string, endPoint: string, verb?: string): Chainable> - - forceVisit (url:string): Chainable + + forceVisit(url: string): Chainable + + executeCliCommand(command: string): Chainable + + replaceWordInJsonObject(targetWord: string, replacement: string, fileName: string): Chainable> + + gwaPublish(type: string, fileName: string): Chainable> + + replaceWord(originalString: string, wordToReplace: string, replacementWord: string): Chainable + + updateJsonBoby(json: any, key: string, newValue: string):Chainable + + deleteFileInE2EFolder(fileName: string):Chainable } } diff --git a/e2e/cypress/support/util-commands.ts b/e2e/cypress/support/util-commands.ts index ad61b4fb5..9d1653869 100644 --- a/e2e/cypress/support/util-commands.ts +++ b/e2e/cypress/support/util-commands.ts @@ -1,5 +1,8 @@ import 'cypress-v10-preserve-cookie' -import jsonpath = require('jsonpath'); +const YAML = require('yamljs'); +const path = require('path'); + + const listOfCookies = [ 'AUTH_SESSION_ID_LEGACY', 'KC_RESTART', @@ -107,10 +110,78 @@ Cypress.Commands.add('resetState', () => { cy.log('Test state was reset') }) -Cypress.Commands.add('updateJsonValue', (jsonBody: any, jsonPath: string, newValue: string, index?: any) => { - let updatedFileContent: any - const jsonContent = JSON.parse(jsonBody); - jsonpath.apply(jsonContent, jsonPath, () => newValue); - updatedFileContent = JSON.stringify(jsonContent, null, 2); - return updatedFileContent +Cypress.Commands.add('updateJsonValue', (filePath: string, jsonPath: string, newValue: string, index?: any) => { + debugger + cy.readFile('cypress/fixtures/' + filePath).then(currState => { + debugger + + const keys = jsonPath.split('.'); // Split the keyPath using dot notation + let currentObj = currState; + + for (let i = 0; i < keys.length - 1; i++) { + const key = keys[i]; + if (!currentObj.hasOwnProperty(key) || typeof currentObj[key] !== 'object') { + return; // If any intermediate key doesn't exist or is not an object, return without updating + } + currentObj = currentObj[key]; + } + + const lastKey = keys[keys.length - 1]; + currentObj[lastKey] = newValue; + + cy.writeFile('cypress/fixtures/' + filePath, currState) + }) + + +}) + +Cypress.Commands.add('executeCliCommand', (command: string) => { + cy.exec(command, { timeout: 9000, failOnNonZeroExit: false }).then((response) => { + return response + }); +}) + +Cypress.Commands.add('replaceWordInJsonObject', (targetWord: string, replacement: string, fileName: string) => { + debugger + cy.readFile('cypress/fixtures/' + fileName).then((content: any) => { + let regex = new RegExp(targetWord, 'g'); + let modifiedString = content.replace(regex, replacement); + + let obj = YAML.parse(modifiedString) + + const yamlString = YAML.stringify(obj, 'utf8'); + cy.writeFile('cypress/fixtures/' + fileName, yamlString) + }) + +}) + +Cypress.Commands.add('gwaPublish', (type: string, fileName: string) => { + cy.exec('gwa publish '+type+' --input ./cypress/fixtures/test_data/'+fileName, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + return response + }); +}) +// cypress/support/commands.js + +Cypress.Commands.add('deleteFileInE2EFolder', (fileName: string) => { + const currentDirectory = Cypress.config('fileServerFolder'); // Get the current working directory + const filePath = path.join(currentDirectory, fileName) + debugger + try { + cy.exec(`rm -f ${filePath}`); + cy.log(`File '${fileName}' has been deleted from the e2e folder.`); + } catch (error) { + cy.log(`Error deleting file '${fileName}' from the e2e folder`); + } +}); + + +Cypress.Commands.add('replaceWord', (originalString: string, wordToReplace: string, replacementWord: string)=> { + // Create a regular expression with the 'g' flag for global search + let replacedString : any + const regex = new RegExp(wordToReplace, 'g'); + + // Use the 'replace()' method to replace all occurrences of the word + replacedString = originalString.replace(regex, replacementWord); + + return replacedString; }) \ No newline at end of file diff --git a/e2e/cypress/tests/01-api-key/01-create-api.cy.ts b/e2e/cypress/tests/01-api-key/01-create-api.cy.ts index 5e1abba9c..20f5bd96f 100644 --- a/e2e/cypress/tests/01-api-key/01-create-api.cy.ts +++ b/e2e/cypress/tests/01-api-key/01-create-api.cy.ts @@ -10,7 +10,7 @@ describe('Create API Spec', () => { const sa = new ServiceAccountsPage() const pd = new Products() var nameSpace: string - let userSession: string + let userSession: any before(() => { cy.visit('/') @@ -23,31 +23,43 @@ describe('Create API Spec', () => { cy.preserveCookies() cy.fixture('apiowner').as('apiowner') cy.fixture('api').as('api') - // cy.visit(login.path) + cy.visit(login.path) }) - it('authenticates Janis (api owner)', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.getUserSessionTokenValue(namespace, false).then((value) => { + userSession = value + }) }) }) - it('creates and activates new namespace', () => { + it('Check gwa config command to set environment', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.executeCliCommand('gwa config set --host ' + cleanedUrl + ' --scheme http').then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('Check gwa config command to set token', () => { + cy.executeCliCommand('gwa config set --token ' + userSession).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('creates new namespace', () => { cy.getUserSession().then(() => { cy.get('@apiowner').then(({ namespace }: any) => { nameSpace = namespace - home.createNamespace(namespace) - cy.get('@login').then(function (xhr: any) { - userSession = xhr.response.headers['x-auth-request-access-token'] + cy.executeCliCommand('gwa namespace create -n ' + namespace).then((response) => { + assert.equal(response.stdout, namespace) }) }) }) }) - it('Verify for invalid namespace name', () => { - cy.get('@apiowner').then(({ invalid_namespace }: any) => { - home.validateNamespaceName(invalid_namespace) - }) + it('activates new namespace', () => { + home.useNamespace(nameSpace) }) it('creates a new service account', () => { @@ -61,18 +73,16 @@ describe('Create API Spec', () => { it('publishes a new API for Dev environment to Kong Gateway', () => { cy.get('@apiowner').then(({ namespace }: any) => { - cy.publishApi('service.yml', namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - }) + cy.publishApi('service.yml', namespace).then((response: any) => { + expect(response.stdout).to.contain('Sync successful'); }) }) }) - it('creates as new product in the directory', () => { - cy.visit(pd.path) - cy.get('@apiowner').then(({ product }: any) => { - pd.createNewProduct(product.name, product.environment.name) + it('Upload dataset and Product using GWA Apply command', () => { + cy.executeCliCommand('gwa apply').then((response) => { + let wordOccurrences = (response.stdout.match(/\bcreated\b/g) || []).length; + expect(wordOccurrences).to.equal(2) }) }) @@ -89,7 +99,7 @@ describe('Create API Spec', () => { it('Verify the message when no dataset is linked to BCDC', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ product }: any) => { - pd.checkMessageForNoDataset(product.name,"health") + pd.checkMessageForNoDataset(product.name, "health") }) }) @@ -116,17 +126,14 @@ describe('Create API Spec', () => { pd.addEnvToProduct(product.name, product.test_environment.name) pd.editProductEnvironment(product.name, product.test_environment.name) pd.editProductEnvironmentConfig(product.test_environment.config) - pd.generateKongPluginConfig(product.name, product.test_environment.name,'service.yml', true) + pd.generateKongPluginConfig(product.name, product.test_environment.name, 'service.yml', true) }) }) it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ namespace }: any) => { - cy.publishApi('service-plugin.yml', namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + cy.publishApi('service-plugin.yml', namespace).then((response: any) => { + expect(response.stdout).to.contain('Sync successful'); }) }) }) @@ -134,7 +141,7 @@ describe('Create API Spec', () => { it('activate the service for Test environment', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ product }: any) => { - pd.activateService(product.name, product.test_environment.name,product.test_environment.config) + pd.activateService(product.name, product.test_environment.name, product.test_environment.config) cy.wait(3000) }) }) @@ -143,11 +150,22 @@ describe('Create API Spec', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ product }: any) => { // pd.editProductEnvironment(product.name, product.environment.name) - pd.activateService(product.name, product.environment.name,product.environment.config) + pd.activateService(product.name, product.environment.name, product.environment.config) cy.wait(3000) }) }) + it('verify status of the services using "gwa status" command', () => { + cy.get('@apiowner').then(({ product }: any) => { + cy.executeCliCommand('gwa status').then((response) => { + expect(response.stdout).to.contain(product.environment.config.serviceName); + expect(response.stdout).to.contain(product.test_environment.config.serviceName); + const wordOccurrences = (response.stdout.match(/\b200 Response\b/g) || []).length; + expect(wordOccurrences).to.equal(2) + }) + }) + }) + after(() => { cy.logout() cy.clearLocalStorage({ log: true }) diff --git a/e2e/cypress/tests/01-api-key/02-team-access.cy.ts b/e2e/cypress/tests/01-api-key/02-team-access.cy.ts index c3dda8ffe..84161c65c 100644 --- a/e2e/cypress/tests/01-api-key/02-team-access.cy.ts +++ b/e2e/cypress/tests/01-api-key/02-team-access.cy.ts @@ -39,17 +39,8 @@ describe('Team Access Spec', () => { }) }) - // it('Grant CredentialIssuer.Admin permission to Janis (API Owner)', () => { - // cy.get('@apiowner').then(({ grantPermission }: any) => { - // na.clickGrantUserAccessButton() - // na.grantPermission(grantPermission.Janis) - // }) - // }) - it('Grant CredentialIssuer.Admin permission to Janis (API Owner)', () => { cy.get('@apiowner').then(({ grantPermission }: any) => { - // na.clickGrantUserAccessButton() - // na.grantPermission(grantPermission.Janis) na.editPermission(grantPermission.Janis) }) }) diff --git a/e2e/cypress/tests/01-api-key/03-request-access-inactive-env.cy.ts b/e2e/cypress/tests/01-api-key/03-request-access-inactive-env.cy.ts new file mode 100644 index 000000000..cf01b89e4 --- /dev/null +++ b/e2e/cypress/tests/01-api-key/03-request-access-inactive-env.cy.ts @@ -0,0 +1,143 @@ +import ApiDirectoryPage from '../../pageObjects/apiDirectory' +import ApplicationPage from '../../pageObjects/applications' +import HomePage from '../../pageObjects/home' +import LoginPage from '../../pageObjects/login' +import NamespaceAccessPage from '../../pageObjects/namespaceAccess' +import Products from '../../pageObjects/products' +import MyAccessPage from '../../pageObjects/myAccess' + +describe('Change an Active environment to Inactive', () => { + const login = new LoginPage() + const home = new HomePage() + const na = new NamespaceAccessPage() + const pd = new Products() + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + // cy.visit(login.path) + }) + + it('authenticates Janis (api owner)', () => { + cy.get('@apiowner').then(({ user, namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + cy.log('Logged in!') + home.useNamespace(namespace) + }) + }) + + it('Navigate to Products Page', () => { + cy.visit(pd.path) + }) + + it('Change the current active environment to inactive state', () => { + cy.get('@apiowner').then(({ product }: any) => { + pd.editProductEnvironment(product.name, product.environment.name) + pd.editProductEnvironmentConfig(product.environment.config, true) + }) + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) +}) + +describe('Verify enactive environment in rrequest access pop up', () => { + const apiDir = new ApiDirectoryPage() + const app = new ApplicationPage() + const myAccessPage = new MyAccessPage() + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('developer').as('developer') + // cy.visit(login.path) + }) + + it('authenticates Harley (developer)', () => { + cy.get('@developer').then(({ user }: any) => { + cy.login(user.credentials.username, user.credentials.password) + }) + }) + + it('creates an application', () => { + cy.visit(app.path) + cy.get('@developer').then(({ application }: any) => { + app.createApplication(application) + }) + }) + + it('Verify that inactive environment is not displayed', () => { + cy.visit(apiDir.path) + cy.get('@developer').then(({ product, application }: any) => { + apiDir.checkInactiveEnvironmentAccessReqOption(product, application) + }) + }) + + it('Close the popup by click on Cancel button', () => { + myAccessPage.cancelRequestAccessPopUp() + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) +}) + +describe('Change an the environment back to active', () => { + const login = new LoginPage() + const home = new HomePage() + const na = new NamespaceAccessPage() + const pd = new Products() + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + // cy.visit(login.path) + }) + + it('authenticates Janis (api owner)', () => { + cy.get('@apiowner').then(({ user, namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + cy.log('Logged in!') + home.useNamespace(namespace) + }) + }) + + it('Navigate to Products Page', () => { + cy.visit(pd.path) + }) + + it('Change the environment back to active state', () => { + cy.get('@apiowner').then(({ product }: any) => { + pd.editProductEnvironment(product.name, product.environment.name) + pd.editProductEnvironmentConfig(product.environment.config) + }) + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) +}) \ No newline at end of file diff --git a/e2e/cypress/tests/01-api-key/03-request-access-with-out-collecting-credentials.cy.ts b/e2e/cypress/tests/01-api-key/04-request-access-with-out-collecting-credentials.cy.ts similarity index 87% rename from e2e/cypress/tests/01-api-key/03-request-access-with-out-collecting-credentials.cy.ts rename to e2e/cypress/tests/01-api-key/04-request-access-with-out-collecting-credentials.cy.ts index 63747c8fa..4eb63eb4c 100644 --- a/e2e/cypress/tests/01-api-key/03-request-access-with-out-collecting-credentials.cy.ts +++ b/e2e/cypress/tests/01-api-key/04-request-access-with-out-collecting-credentials.cy.ts @@ -26,13 +26,6 @@ describe('Request Access without colleting credential Spec', () => { }) }) - it('creates an application', () => { - cy.visit(app.path) - cy.get('@developer').then(({ application }: any) => { - app.createApplication(application) - }) - }) - it('Collect the credentials', () => { cy.visit(apiDir.path) cy.get('@developer').then(({ product, application,accessRequest }: any) => { diff --git a/e2e/cypress/tests/01-api-key/04-review-request-without-collecting-credentials.cy.ts b/e2e/cypress/tests/01-api-key/05-review-request-without-collecting-credentials.cy.ts similarity index 81% rename from e2e/cypress/tests/01-api-key/04-review-request-without-collecting-credentials.cy.ts rename to e2e/cypress/tests/01-api-key/05-review-request-without-collecting-credentials.cy.ts index 06a133037..39f89c4c7 100644 --- a/e2e/cypress/tests/01-api-key/04-review-request-without-collecting-credentials.cy.ts +++ b/e2e/cypress/tests/01-api-key/05-review-request-without-collecting-credentials.cy.ts @@ -12,8 +12,6 @@ describe('Approve Pending Request without collecting credentials Spec', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -25,9 +23,11 @@ describe('Approve Pending Request without collecting credentials Spec', () => { }) it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@apiowner').then(({ namespace }: any) => { + cy.get('@access-manager').then(({ user }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) @@ -45,5 +45,5 @@ describe('Approve Pending Request without collecting credentials Spec', () => { cy.clearLocalStorage({ log: true }) cy.deleteAllCookies() }) - + }) \ No newline at end of file diff --git a/e2e/cypress/tests/01-api-key/05-collect-credentials.cy.ts b/e2e/cypress/tests/01-api-key/06-collect-credentials.cy.ts similarity index 100% rename from e2e/cypress/tests/01-api-key/05-collect-credentials.cy.ts rename to e2e/cypress/tests/01-api-key/06-collect-credentials.cy.ts diff --git a/e2e/cypress/tests/01-api-key/06-approve-pending-rqst.cy.ts b/e2e/cypress/tests/01-api-key/07-approve-pending-rqst.cy.ts similarity index 93% rename from e2e/cypress/tests/01-api-key/06-approve-pending-rqst.cy.ts rename to e2e/cypress/tests/01-api-key/07-approve-pending-rqst.cy.ts index e0a21c595..1dc6ca499 100644 --- a/e2e/cypress/tests/01-api-key/06-approve-pending-rqst.cy.ts +++ b/e2e/cypress/tests/01-api-key/07-approve-pending-rqst.cy.ts @@ -12,8 +12,6 @@ describe('Approve Pending Request Spec', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -35,9 +33,11 @@ describe('Approve Pending Request Spec', () => { }) it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@apiowner').then(({ namespace }: any) => { + cy.get('@access-manager').then(({ user }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) diff --git a/e2e/cypress/tests/01-api-key/07-grant-access.cy.ts b/e2e/cypress/tests/01-api-key/08-grant-access.cy.ts similarity index 88% rename from e2e/cypress/tests/01-api-key/07-grant-access.cy.ts rename to e2e/cypress/tests/01-api-key/08-grant-access.cy.ts index 5658cfebc..7f0ad2b15 100644 --- a/e2e/cypress/tests/01-api-key/07-grant-access.cy.ts +++ b/e2e/cypress/tests/01-api-key/08-grant-access.cy.ts @@ -12,8 +12,6 @@ describe('Grant Access Spec', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -25,9 +23,11 @@ describe('Grant Access Spec', () => { }) it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@apiowner').then(({ namespace }: any) => { + cy.get('@access-manager').then(({ user }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) diff --git a/e2e/cypress/tests/01-api-key/09-gwa-get.ts b/e2e/cypress/tests/01-api-key/09-gwa-get.ts new file mode 100644 index 000000000..f39e1f99a --- /dev/null +++ b/e2e/cypress/tests/01-api-key/09-gwa-get.ts @@ -0,0 +1,85 @@ +import ConsumersPage from '../../pageObjects/consumers' +import LoginPage from '../../pageObjects/login' +import HomePage from '../../pageObjects/home' +import ProductPage from '../../pageObjects/products' +const yaml = require('js-yaml'); + +describe('Verify GWA get commands', () => { + const login = new LoginPage() + const consumers = new ConsumersPage() + const home = new HomePage() + var _namespace: string + let userSession: any + let resObj: any + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('api').as('api') + cy.fixture('apiowner').as('apiowner') + }) + + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.getUserSessionTokenValue(namespace, false).then((value) => { + userSession = value + _namespace = namespace + }) + }) + }) + + it('Check gwa config command to set token', () => { + cy.executeCliCommand('gwa config set --token ' + userSession).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('Verify "gwa get" for dataset', () => { + cy.get('@apiowner').then(({ product }: any) => { + cy.executeCliCommand('gwa get datasets').then((response) => { + expect(response.stdout).not.to.contain(product); + }) + }) + }) + + it('Verify "gwa get" for dataset in JSON format', () => { + cy.executeCliCommand('gwa get datasets --json').then((response) => { + cy.get('@api').then(({ apiDirectory }: any) => { + cy.setHeaders(apiDirectory.headers) + cy.setAuthorizationToken(userSession) + cy.makeAPIRequest(apiDirectory.endPoint + '/' + _namespace + '/directory', 'GET').then((res) => { + resObj = res.body[0] + Cypress._.isEqual(resObj, JSON.parse(response.stdout)[0]) + }) + }) + }) + }) + + it('Verify "gwa get" for dataset in YAML format', () => { + cy.executeCliCommand('gwa get datasets --yaml').then((response) => { + const yamlObject = yaml.load(response.stdout) + Cypress._.isEqual(resObj, yamlObject) + }) + }) + + it('Verify "gwa get" for products', () => { + cy.get('@apiowner').then(({ product }: any) => { + cy.executeCliCommand('gwa get products').then((response) => { + expect(response.stdout).not.to.contain(product); + }) + }) + }) + + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) + +}) \ No newline at end of file diff --git a/e2e/cypress/tests/02-client-credential-flow/01-client-cred-team-access.cy.ts b/e2e/cypress/tests/02-client-credential-flow/01-client-cred-team-access.cy.ts index 8c87f4696..24e2c4b36 100644 --- a/e2e/cypress/tests/02-client-credential-flow/01-client-cred-team-access.cy.ts +++ b/e2e/cypress/tests/02-client-credential-flow/01-client-cred-team-access.cy.ts @@ -7,6 +7,8 @@ describe('Grant appropriate permissions to team members for client credential fl const login = new LoginPage() const home = new HomePage() const na = new NamespaceAccessPage() + let userSession: any + let namespace: string before(() => { cy.visit('/') @@ -20,18 +22,37 @@ describe('Grant appropriate permissions to team members for client credential fl cy.visit(login.path) }) - it('Authenticates api owner', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) }) }) - it('Creates and activates new namespace', () => { - cy.get('@apiowner').then(({ clientCredentials }: any) => { - home.createNamespace(clientCredentials.namespace) - }) + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('create namespace using gwa cli command', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.exec('gwa namespace create --host ' + cleanedUrl + ' --scheme http', { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + cy.replaceWordInJsonObject('ccplatform', namespace, 'cc-service-gwa.yml') + cy.updateJsonValue('apiowner.json', 'clientCredentials.namespace', namespace) + // cy.updateJsonValue('apiowner.json', 'clientCredentials.clientIdSecret.product.environment.name.config.serviceName', 'cc-service-for-' + namespace) + cy.executeCliCommand("gwa config set --namespace " + namespace) + }); }) + it('activates new namespace', () => { + home.useNamespace(namespace) + }) + + it('Grant namespace access to access manager(Mark)', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { cy.visit(na.path) diff --git a/e2e/cypress/tests/02-client-credential-flow/03-client-cred-create-api-prod-auth-pro.cy.ts b/e2e/cypress/tests/02-client-credential-flow/03-client-cred-create-api-prod-auth-pro.cy.ts index 84170a63d..7766994a2 100644 --- a/e2e/cypress/tests/02-client-credential-flow/03-client-cred-create-api-prod-auth-pro.cy.ts +++ b/e2e/cypress/tests/02-client-credential-flow/03-client-cred-create-api-prod-auth-pro.cy.ts @@ -13,7 +13,8 @@ describe('Create API, Product, and Authorization Profiles; Apply Auth Profiles t const pd = new Products() const authProfile = new AuthorizationProfile() var nameSpace: string - let userSession: string + let userSession: any + let namespace : string before(() => { cy.visit('/') @@ -27,11 +28,15 @@ describe('Create API, Product, and Authorization Profiles; Apply Auth Profiles t cy.fixture('api').as('api') cy.visit(login.path) }) - it('Authenticates api owner', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) + + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) }) }) + it('Activates namespace for client credential flow tests', () => { cy.getUserSession().then(() => { cy.get('@apiowner').then(({ clientCredentials }: any) => { @@ -54,11 +59,11 @@ describe('Create API, Product, and Authorization Profiles; Apply Auth Profiles t it('Publishes a new API to Kong Gateway', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { - cy.publishApi('cc-service.yml', clientCredentials.namespace, true).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + cy.publishApi('cc-service-gwa.yml', clientCredentials.namespace, true).then(() => { + // cy.get('@publishAPIResponse').then((res: any) => { + // // cy.log(JSON.stringify(res.body)) + // // expect(res.body.message).to.contains("Sync successful") + // }) }) }) }) @@ -107,11 +112,13 @@ describe('Create API, Product, and Authorization Profiles; Apply Auth Profiles t it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.replaceWordInJsonObject('ccplatform', clientCredentials.namespace, 'cc-service-plugin.yml') + cy.wait(2000) cy.publishApi('cc-service-plugin.yml', clientCredentials.namespace,true).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + // cy.get('@publishAPIResponse').then((res: any) => { + // // cy.log(JSON.stringify(res.body)) + // // expect(res.body.message).to.contains("Sync successful") + // }) }) }) }) @@ -147,11 +154,13 @@ describe('Create API, Product, and Authorization Profiles; Apply Auth Profiles t it('Applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.replaceWordInJsonObject('ccplatform', clientCredentials.namespace, 'cc-service-plugin.yml') + cy.wait(2000) cy.publishApi('cc-service-plugin.yml', clientCredentials.namespace, true).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + // cy.get('@publishAPIResponse').then((res: any) => { + // // cy.log(JSON.stringify(res.body)) + // // expect(res.body.message).to.contains("Sync successful") + // }) }) }) }) diff --git a/e2e/cypress/tests/02-client-credential-flow/05-cids-access-approve-api-rqst.cy.ts b/e2e/cypress/tests/02-client-credential-flow/05-cids-access-approve-api-rqst.cy.ts index cce000c6f..87693de85 100644 --- a/e2e/cypress/tests/02-client-credential-flow/05-cids-access-approve-api-rqst.cy.ts +++ b/e2e/cypress/tests/02-client-credential-flow/05-cids-access-approve-api-rqst.cy.ts @@ -1,6 +1,10 @@ import HomePage from '../../pageObjects/home' import LoginPage from '../../pageObjects/login' import ConsumersPage from '../../pageObjects/consumers' +import KeycloakUserGroupPage from '../../pageObjects/keycloakUserGroup' +import keycloakGroupPage from '../../pageObjects/keycloakGroup' +import AuthorizationProfile from '../../pageObjects/authProfile' +import keycloakClientScopesPage from '../../pageObjects/keycloakClientScopes' describe('Access manager approves developer access request for Client ID/Secret authenticator', () => { const home = new HomePage() @@ -21,17 +25,17 @@ describe('Access manager approves developer access request for Client ID/Secret }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, clientCredentials }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(clientCredentials.namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(clientCredentials.namespace) + }) }) }) it('Access Manager approves developer access request', () => { - cy.get('@access-manager').then(() => { - cy.visit(consumers.path) - consumers.reviewThePendingRequest() - }) + cy.visit(consumers.path) + consumers.reviewThePendingRequest() }) it('Select scopes in Authorization Tab', () => { @@ -56,7 +60,7 @@ describe('Make an API request using Client ID, Secret, and Access Token', () => cy.readFile('cypress/fixtures/state/store.json').then((store_res) => { let cc = JSON.parse(store_res.clientidsecret) - + cy.getAccessToken(cc.clientId, cc.clientSecret).then(() => { cy.get('@accessTokenResponse').then((token_res: any) => { let token = token_res.body.access_token @@ -75,4 +79,160 @@ describe('Make an API request using Client ID, Secret, and Access Token', () => }) }) }) +}) + +describe('Verify the selected client scoped is displayed in assigned default list', () => { + const clientScopes = new keycloakClientScopesPage() + const groups = new keycloakGroupPage() + var nameSpace: string + const home = new HomePage() + const authProfile = new AuthorizationProfile() + + before(() => { + cy.visit(Cypress.env('KEYCLOAK_URL')) + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('developer').as('developer') + cy.fixture('apiowner').as('apiowner') + cy.fixture('state/regen').as('regen') + cy.fixture('admin').as('admin') + }) + + it('Authenticates Admin owner', () => { + cy.get('@admin').then(({ user }: any) => { + cy.contains('Administration Console').click({ force: true }) + cy.keycloakLogin(user.credentials.username, user.credentials.password) + }) + }) + + it('Navigate to Clients page', () => { + cy.contains('Clients').click() + }) + + it('Select the consumer ID', () => { + cy.readFile('cypress/fixtures/state/store.json').then((store_res) => { + let cc = JSON.parse(store_res.clientidsecret) + cy.contains(cc.clientId).click() + }) + }) + + it('Navigate to client scope tab', () => { + clientScopes.selectTab('Client Scopes') + }) + + it('Verify that "System.Write" scope is in assigned default scope', () => { + clientScopes.verifyAssignedScope('System.Write', true) + }) + + after(() => { + cy.keycloakLogout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) + +}) + +describe('Deselect the scope from authorization tab', () => { + const login = new LoginPage() + const home = new HomePage() + const consumers = new ConsumersPage() + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('access-manager').as('access-manager') + cy.fixture('apiowner').as('apiowner') + cy.fixture('manage-control-config-setting').as('manage-control-config-setting') + // cy.visit(login.path) + }) + + it('authenticates Mark (Access Manager)', () => { + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.login(user.credentials.username, user.credentials.password).then(() => { + home.useNamespace(clientCredentials.namespace); + }) + }) + }) + }) + + it('Navigate to Consumer page ', () => { + cy.visit(consumers.path); + }) + + it('Select the consumer from the list ', () => { + consumers.clickOnTheFirstConsumerID() + }) + + it('Deselect scopes in Authorization Tab', () => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + consumers.editConsumerDialog() + consumers.selectAuthorizationScope(clientCredentials.clientIdSecret.authProfile.scopes, false) + consumers.saveAppliedConfig() + }) + }) +}) + +describe('Verify the selected client scoped is not displayed in assigned default list', () => { + const clientScopes = new keycloakClientScopesPage() + const groups = new keycloakGroupPage() + var nameSpace: string + const home = new HomePage() + const authProfile = new AuthorizationProfile() + + before(() => { + cy.visit(Cypress.env('KEYCLOAK_URL')) + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('developer').as('developer') + cy.fixture('apiowner').as('apiowner') + cy.fixture('state/regen').as('regen') + cy.fixture('admin').as('admin') + }) + + it('Authenticates Admin owner', () => { + cy.get('@admin').then(({ user }: any) => { + cy.contains('Administration Console').click({ force: true }) + cy.keycloakLogin(user.credentials.username, user.credentials.password) + }) + }) + + it('Navigate to Clients page', () => { + cy.contains('Clients').click() + }) + + it('Select the consumer ID', () => { + cy.readFile('cypress/fixtures/state/store.json').then((store_res) => { + let cc = JSON.parse(store_res.clientidsecret) + cy.contains(cc.clientId).click() + }) + }) + + it('Navigate to client scope tab', () => { + clientScopes.selectTab('Client Scopes') + }) + + it('Verify that "System.Write" scope is not in assigned default scope', () => { + clientScopes.verifyAssignedScope('System.Write', false) + }) + + after(() => { + cy.keycloakLogout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) + }) \ No newline at end of file diff --git a/e2e/cypress/tests/02-client-credential-flow/07-jwt-genkp-access-approve-api-rqst.cy.ts b/e2e/cypress/tests/02-client-credential-flow/07-jwt-genkp-access-approve-api-rqst.cy.ts index 24adf4d1a..6ee1c408a 100644 --- a/e2e/cypress/tests/02-client-credential-flow/07-jwt-genkp-access-approve-api-rqst.cy.ts +++ b/e2e/cypress/tests/02-client-credential-flow/07-jwt-genkp-access-approve-api-rqst.cy.ts @@ -18,13 +18,16 @@ describe('Access manager approves developer access request for JWT - Generated K beforeEach(() => { cy.preserveCookies() cy.fixture('access-manager').as('access-manager') + cy.fixture('apiowner').as('apiowner') // cy.visit(login.path) }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, clientCredentials }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(clientCredentials.namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(clientCredentials.namespace) + }) }) }) diff --git a/e2e/cypress/tests/02-client-credential-flow/09-jwks-url-access-approval-api-rqst.cy.ts b/e2e/cypress/tests/02-client-credential-flow/09-jwks-url-access-approval-api-rqst.cy.ts index bdbf8b860..e44f75b99 100644 --- a/e2e/cypress/tests/02-client-credential-flow/09-jwks-url-access-approval-api-rqst.cy.ts +++ b/e2e/cypress/tests/02-client-credential-flow/09-jwks-url-access-approval-api-rqst.cy.ts @@ -19,13 +19,16 @@ describe('Access manager approves developer access request for JWKS URL flow', ( beforeEach(() => { cy.preserveCookies() cy.fixture('access-manager').as('access-manager') + cy.fixture('apiowner').as('apiowner') // cy.visit(login.path) }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, clientCredentials }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(clientCredentials.namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(clientCredentials.namespace) + }) }) }) diff --git a/e2e/cypress/tests/02-client-credential-flow/10-jwks-publicKey-access-rqst.cy.ts b/e2e/cypress/tests/02-client-credential-flow/10-jwks-publicKey-access-rqst.cy.ts index c3587dc51..798fdaef3 100644 --- a/e2e/cypress/tests/02-client-credential-flow/10-jwks-publicKey-access-rqst.cy.ts +++ b/e2e/cypress/tests/02-client-credential-flow/10-jwks-publicKey-access-rqst.cy.ts @@ -20,7 +20,7 @@ describe('Generates public/private key and set public key to access request', () beforeEach(() => { cy.preserveCookies() cy.fixture('developer').as('developer') - cy.visit(login.path) + // cy.visit(login.path) }) it('Generate the RS256 key pair', () => { @@ -40,6 +40,19 @@ describe('Generates public/private key and set public key to access request', () }) }) + it('Enter JWT key with invalid format', () => { + cy.visit(apiDir.path) + cy.get('@developer').then(({ clientCredentials, accessRequest }: any) => { + let jwksPublicKey = clientCredentials.jwksPublicKey + apiDir.enterInvalidJWTKey(jwksPublicKey.product, jwksPublicKey.application, accessRequest) + }) + }) + + it('Verify the error popups for invalid JWT key', () => { + cy.wait(3000) + cy.verifyToastMessage("Certificate failed validation") + }) + it('Creates an access request', () => { cy.visit(apiDir.path) cy.get('@developer').then(({ clientCredentials, accessRequest }: any) => { diff --git a/e2e/cypress/tests/02-client-credential-flow/11-jwt-publlicKey-access-approve-api-rqst.cy.ts b/e2e/cypress/tests/02-client-credential-flow/11-jwt-publlicKey-access-approve-api-rqst.cy.ts index f4917dcbe..a0818d6f5 100644 --- a/e2e/cypress/tests/02-client-credential-flow/11-jwt-publlicKey-access-approve-api-rqst.cy.ts +++ b/e2e/cypress/tests/02-client-credential-flow/11-jwt-publlicKey-access-approve-api-rqst.cy.ts @@ -18,13 +18,16 @@ describe('Access manager approves developer access request for JWT - Generated K beforeEach(() => { cy.preserveCookies() cy.fixture('access-manager').as('access-manager') + cy.fixture('apiowner').as('apiowner') // cy.visit(login.path) }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, clientCredentials }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(clientCredentials.namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(clientCredentials.namespace) + }) }) }) diff --git a/e2e/cypress/tests/03-manage-labels/02-approve-pending-rqst-for-labels.spec.cy.ts b/e2e/cypress/tests/03-manage-labels/02-approve-pending-rqst-for-labels.spec.cy.ts index 576bc21f6..acd3830b3 100644 --- a/e2e/cypress/tests/03-manage-labels/02-approve-pending-rqst-for-labels.spec.cy.ts +++ b/e2e/cypress/tests/03-manage-labels/02-approve-pending-rqst-for-labels.spec.cy.ts @@ -12,8 +12,6 @@ describe('Approve Pending Request Spec', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -26,9 +24,11 @@ describe('Approve Pending Request Spec', () => { }) it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) diff --git a/e2e/cypress/tests/03-manage-labels/03-filter-labels.cy.ts b/e2e/cypress/tests/03-manage-labels/03-filter-labels.cy.ts index 15670cb68..d79910e41 100644 --- a/e2e/cypress/tests/03-manage-labels/03-filter-labels.cy.ts +++ b/e2e/cypress/tests/03-manage-labels/03-filter-labels.cy.ts @@ -12,8 +12,6 @@ describe('Filter Manage labels Spec', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -27,9 +25,11 @@ describe('Filter Manage labels Spec', () => { it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) diff --git a/e2e/cypress/tests/03-manage-labels/04-manage-labels.cy.ts b/e2e/cypress/tests/03-manage-labels/04-manage-labels.cy.ts index eacbab194..e6bee8d2f 100644 --- a/e2e/cypress/tests/03-manage-labels/04-manage-labels.cy.ts +++ b/e2e/cypress/tests/03-manage-labels/04-manage-labels.cy.ts @@ -12,8 +12,6 @@ describe('Manage/Edit labels spec', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -25,9 +23,11 @@ describe('Manage/Edit labels spec', () => { }) it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) @@ -57,5 +57,5 @@ describe('Manage/Edit labels spec', () => { cy.logout() cy.clearLocalStorage({ log: true }) cy.deleteAllCookies() -}) + }) }) \ No newline at end of file diff --git a/e2e/cypress/tests/03-manage-labels/05-link-consumers.ts b/e2e/cypress/tests/03-manage-labels/05-link-consumers.ts index 8bfb50601..97eea0365 100644 --- a/e2e/cypress/tests/03-manage-labels/05-link-consumers.ts +++ b/e2e/cypress/tests/03-manage-labels/05-link-consumers.ts @@ -13,8 +13,6 @@ describe('Link Consumers to Namespace', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -27,9 +25,11 @@ describe('Link Consumers to Namespace', () => { }) it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) @@ -39,12 +39,9 @@ describe('Link Consumers to Namespace', () => { }) it('Get the consumer ID from the list', () => { - cy.getLastConsumerID().then((title)=>{ - consumerID = title - }) - // cy.wrap(consumers).its('inputValue').then(inputValue => { - // consumerID = inputValue.text() - // }) + cy.getLastConsumerID().then((title) => { + consumerID = title + }) }) it('Delete the consumer ID from the list', () => { @@ -61,7 +58,7 @@ describe('Link Consumers to Namespace', () => { }) it('Verify that the consumer is linked to the namespace', () => { - cy.getLastConsumerID().then((title)=>{ + cy.getLastConsumerID().then((title) => { expect(title).to.equal(consumerID) }) }) diff --git a/e2e/cypress/tests/04-gateway-services/01-gateway-service-details.cy.ts b/e2e/cypress/tests/04-gateway-services/01-gateway-service-details.cy.ts index 07401a29f..37134dcb8 100644 --- a/e2e/cypress/tests/04-gateway-services/01-gateway-service-details.cy.ts +++ b/e2e/cypress/tests/04-gateway-services/01-gateway-service-details.cy.ts @@ -6,11 +6,8 @@ import MyAccessPage from '../../pageObjects/myAccess' import NamespaceAccessPage from '../../pageObjects/namespaceAccess' describe('Verify Gateway Service details', () => { - const login = new LoginPage() - const apiDir = new ApiDirectoryPage() - const myAccessPage = new MyAccessPage() + const home = new HomePage() - const na = new NamespaceAccessPage() const gs = new GatewayServicePage() before(() => { @@ -52,7 +49,7 @@ describe('Verify Gateway Service details', () => { it('Verify the routes details ', () => { cy.get('@apiowner').then(({ product }: any) => { - gs.verifyRouteName(product.environment.config.serviceName, 'https://a-service-for-newplatform.api.gov.bc.ca/') + gs.verifyRouteName(product.environment.config.serviceName, 'https://'+product.environment.config.serviceName+'.api.gov.bc.ca/') }) }) @@ -61,7 +58,9 @@ describe('Verify Gateway Service details', () => { }) it('Verify the Tags details ', () => { - gs.verifyTagsName('ns.newplatform') + cy.get('@apiowner').then(({ namespace }: any) => { + gs.verifyTagsName('ns.'+namespace) + }) }) after(() => { diff --git a/e2e/cypress/tests/05-migrate-user/01-migrate-user-access.cy.ts b/e2e/cypress/tests/05-migrate-user/01-migrate-user-access.cy.ts index d8b566044..029ee9fd8 100644 --- a/e2e/cypress/tests/05-migrate-user/01-migrate-user-access.cy.ts +++ b/e2e/cypress/tests/05-migrate-user/01-migrate-user-access.cy.ts @@ -61,14 +61,17 @@ describe('Authernticate with old user to initiate migration', () => { beforeEach(() => { cy.preserveCookies() cy.fixture('usermigration').as('usermigration') + cy.fixture('apiowner').as('apiowner') // cy.visit(login.path) }) it('authenticates with old user', () => { - cy.get('@usermigration').then(({ oldUser, namespace }: any) => { - cy.login(oldUser.credentials.username, oldUser.credentials.password) - cy.log('Logged in!') - home.useNamespace(namespace) + cy.get('@usermigration').then(({ oldUser }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(oldUser.credentials.username, oldUser.credentials.password) + cy.log('Logged in!') + home.useNamespace(namespace) + }) }) }) @@ -95,20 +98,25 @@ describe('Verify that permission of old user is migrated to new user', () => { cy.preserveCookies() cy.fixture('usermigration').as('usermigration') cy.fixture('apiowner').as('apiowner') - // cy.visit(login.path) + cy.visit(login.path) }) it('authenticates with new user', () => { - cy.get('@usermigration').then(({ newUser, namespace }: any) => { + cy.get('@usermigration').then(({ newUser }: any) => { cy.login(newUser.credentials.username, newUser.credentials.password) cy.log('Logged in!') - // home.useNamespace(namespace) }) }) + // it('activates new namespace', () => { + // cy.get('@apiowner').then(({ namespace }: any) => { + // home.useNamespace(namespace) + // }) + // }) + it('Get the permission of the user', () => { cy.getUserSession().then(() => { - cy.get('@usermigration').then(({ namespace }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { home.useNamespace(namespace) cy.get('@login').then(function (xhr: any) { userScopes = xhr.response.body.user.scopes @@ -150,8 +158,8 @@ describe('Verify that old user is no longer able to sign in', () => { }) it('authenticates with old user', () => { - cy.get('@usermigration').then(({ oldUser}: any) => { - cy.login(oldUser.credentials.username, oldUser.credentials.password,true) + cy.get('@usermigration').then(({ oldUser }: any) => { + cy.login(oldUser.credentials.username, oldUser.credentials.password, true) }) }) diff --git a/e2e/cypress/tests/06-refresh-credential/02-client-credentials.cy.ts b/e2e/cypress/tests/06-refresh-credential/02-client-credentials.cy.ts index ca841ee5f..327508156 100644 --- a/e2e/cypress/tests/06-refresh-credential/02-client-credentials.cy.ts +++ b/e2e/cypress/tests/06-refresh-credential/02-client-credentials.cy.ts @@ -127,6 +127,10 @@ describe('Regenerate Credential for Client Credentials- Client ID/Secret', () => }) after(() => { + cy.get('@apiowner').then(({ product }: any) => { + cy.getServiceOrRouteID('services', product.environment.config.serviceName) + cy.getServiceOrRouteID('routes', product.environment.config.serviceName) + }) cy.logout() cy.clearLocalStorage({ log: true }) cy.deleteAllCookies() diff --git a/e2e/cypress/tests/07-manage-control/01-ip-restriction.cy.ts b/e2e/cypress/tests/07-manage-control/01-ip-restriction.cy.ts index 94d7b1c94..aed31e1d2 100644 --- a/e2e/cypress/tests/07-manage-control/01-ip-restriction.cy.ts +++ b/e2e/cypress/tests/07-manage-control/01-ip-restriction.cy.ts @@ -6,7 +6,7 @@ describe('Manage Control-IP Restriction Spec', () => { const login = new LoginPage() const home = new HomePage() const consumers = new ConsumersPage() - + before(() => { cy.visit('/') cy.deleteAllCookies() @@ -21,9 +21,11 @@ describe('Manage Control-IP Restriction Spec', () => { }) it('authenticates Mark (Access Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password).then(() => { - home.useNamespace(namespace); + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password).then(() => { + home.useNamespace(namespace); + }) }) }) }) @@ -44,7 +46,7 @@ describe('Manage Control-IP Restriction Spec', () => { consumers.setAllowedIPAddress(ipRestriction.ipRange_inValid, 'Route') }) }) - + it('verify IP Restriction error when the API calls other than the allowed IP', () => { cy.get('@apiowner').then(({ product }: any) => { cy.makeKongRequest(product.environment.config.serviceName, 'GET').then((response) => { @@ -58,7 +60,7 @@ describe('Manage Control-IP Restriction Spec', () => { cy.get('@access-manager').then(() => { cy.visit(consumers.path); consumers.clickOnTheFirstConsumerID() - cy.get('@manage-control-config-setting').then(({ipRestriction} : any) => { + cy.get('@manage-control-config-setting').then(({ ipRestriction }: any) => { consumers.setAllowedIPAddress(ipRestriction.ipRange_inValid) }) }) @@ -77,7 +79,7 @@ describe('Manage Control-IP Restriction Spec', () => { cy.get('@access-manager').then(() => { cy.visit(consumers.path); consumers.clickOnTheFirstConsumerID() - cy.get('@manage-control-config-setting').then(({ipRestriction} : any) => { + cy.get('@manage-control-config-setting').then(({ ipRestriction }: any) => { consumers.setAllowedIPAddress(ipRestriction.ipRange_valid, "Route") }) }) @@ -95,7 +97,7 @@ describe('Manage Control-IP Restriction Spec', () => { cy.get('@access-manager').then(() => { cy.visit(consumers.path); consumers.clickOnTheFirstConsumerID() - cy.get('@manage-control-config-setting').then(({ipRestriction} : any) => { + cy.get('@manage-control-config-setting').then(({ ipRestriction }: any) => { consumers.setAllowedIPAddress(ipRestriction.ipRange_valid) }) }) @@ -151,7 +153,7 @@ describe('Manage Control -Apply IP Restriction to Global and Consumer at Service cy.get('@access-manager').then(({ user, namespace }: any) => { cy.visit(consumers.path); consumers.clickOnTheFirstConsumerID() - cy.get('@manage-control-config-setting').then(({ipRestriction} : any) => { + cy.get('@manage-control-config-setting').then(({ ipRestriction }: any) => { consumers.setAllowedIPAddress(ipRestriction.ipRange_inValid) }) }) @@ -186,7 +188,7 @@ describe('Manage Control -Apply IP Restriction to Global and Consumer at Route l consumers.filterConsumerByTypeAndValue('Products', product.name) }) }) - + it('set api ip-restriction to global service level', () => { cy.visit(consumers.path); consumers.clickOnTheFirstConsumerID() @@ -208,7 +210,7 @@ describe('Manage Control -Apply IP Restriction to Global and Consumer at Route l cy.get('@access-manager').then(({ user, namespace }: any) => { cy.visit(consumers.path); consumers.clickOnTheFirstConsumerID() - cy.get('@manage-control-config-setting').then(({ipRestriction} : any) => { + cy.get('@manage-control-config-setting').then(({ ipRestriction }: any) => { consumers.setAllowedIPAddress(ipRestriction.ipRange_inValid) }) }) diff --git a/e2e/cypress/tests/07-manage-control/02-rate-limiting.cy.ts b/e2e/cypress/tests/07-manage-control/02-rate-limiting.cy.ts index da7b5c1a4..7dd342b09 100644 --- a/e2e/cypress/tests/07-manage-control/02-rate-limiting.cy.ts +++ b/e2e/cypress/tests/07-manage-control/02-rate-limiting.cy.ts @@ -24,9 +24,11 @@ describe('Manage Control-Rate Limiting Spec for Service as Scope and Local Polic }) it('authenticates Mark (Access Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password).then(() => { - home.useNamespace(namespace); + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password).then(() => { + home.useNamespace(namespace); + }) }) }) }) @@ -79,6 +81,7 @@ describe('Manage Control-Rate Limiting Spec for Route as Scope and Local Policy' cy.get('@manage-control-config-setting').then(({ rateLimiting }: any) => { cy.visit(consumers.path); consumers.clickOnTheFirstConsumerID() + consumers.clearRateLimitControl() consumers.setRateLimiting(rateLimiting.requestPerHour_Consumer, "Route") }) }) @@ -188,7 +191,7 @@ describe('Manage Control-Apply Rate limiting to Global and Consumer at Service l cy.get('@apiowner').then(({ product }: any) => { cy.makeKongRequest(product.environment.config.serviceName, 'GET').then((response) => { expect(response.status).to.be.equal(200) - expect(parseInt(response.headers["x-ratelimit-remaining-hour"])).to.be.within(16,18) + expect(parseInt(response.headers["x-ratelimit-remaining-hour"])).to.be.within(16, 18) }) }) }) diff --git a/e2e/cypress/tests/07-manage-control/03-kong-api-only-apply-rate-limiting.cy.ts b/e2e/cypress/tests/07-manage-control/03-kong-api-only-apply-rate-limiting.cy.ts index f3d0f283b..4c0b63f0d 100644 --- a/e2e/cypress/tests/07-manage-control/03-kong-api-only-apply-rate-limiting.cy.ts +++ b/e2e/cypress/tests/07-manage-control/03-kong-api-only-apply-rate-limiting.cy.ts @@ -64,6 +64,8 @@ describe('Apply Kong API key only plugin', () => { }) it('Set Consumer ID to key auth anonymous config', () => { + const arr_namespace = ['ns.' + nameSpace]; + cy.updatePropertiesOfPluginFile('service-plugin-key-auth-only.yml', 'tags', arr_namespace) cy.updatePropertiesOfPluginFile('service-plugin-key-auth-only.yml', 'config.anonymous', consumerID) }) @@ -80,9 +82,9 @@ describe('Apply Kong API key only plugin', () => { cy.get('@apiowner').then(({ namespace, product }: any) => { cy.updatePluginFile('service-plugin.yml', product.environment.config.serviceName, 'service-plugin-key-auth-only.yml') cy.publishApi('service-plugin.yml', namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - }) + // cy.get('@publishAPIResponse').then((res: any) => { + // cy.log(JSON.stringify(res.body)) + // }) }) }) }) @@ -97,7 +99,7 @@ describe('Apply Kong API key only plugin', () => { }) it('Apply Rate limiting on free access', () => { - cy.updateKongPlugin('consumers', 'rateLimitingConsumer', 'consumers/'+consumerID+'/plugins').then((response) => { + cy.updateKongPlugin('consumers', 'rateLimitingConsumer', 'consumers/' + consumerID + '/plugins').then((response) => { expect(response.status).to.be.equal(201) }) }) @@ -115,7 +117,7 @@ describe('Check the API key for free access', () => { cy.get('@apiowner').then(async ({ product }: any) => { cy.fixture('state/store').then((creds: any) => { const key = creds.consumerKey - cy.makeKongRequest(product.environment.config.serviceName, 'GET','').then((response) => { + cy.makeKongRequest(product.environment.config.serviceName, 'GET', '').then((response) => { expect(response.status).to.be.equal(200) expect(parseInt(response.headers["x-ratelimit-remaining-hour"])).to.be.equal(99) }) @@ -185,8 +187,6 @@ describe('Approve Pending Request Spec', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -200,9 +200,11 @@ describe('Approve Pending Request Spec', () => { }) it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) diff --git a/e2e/cypress/tests/08-client-role/02-add-roles-authorization-profile.ts b/e2e/cypress/tests/08-client-role/02-add-roles-authorization-profile.ts index 4a24b1262..b3094ced3 100644 --- a/e2e/cypress/tests/08-client-role/02-add-roles-authorization-profile.ts +++ b/e2e/cypress/tests/08-client-role/02-add-roles-authorization-profile.ts @@ -31,7 +31,7 @@ describe('Apply client roles to the Authorization Profile', () => { }) it('Select the namespace created for client credential ', () => { - cy.get('@credential-issuer').then(({ clientCredentials }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { home.useNamespace(clientCredentials.namespace) }) }) diff --git a/e2e/cypress/tests/08-client-role/03-read-client-role.ts b/e2e/cypress/tests/08-client-role/03-read-client-role.ts index 04cb10b6c..44d50fd85 100644 --- a/e2e/cypress/tests/08-client-role/03-read-client-role.ts +++ b/e2e/cypress/tests/08-client-role/03-read-client-role.ts @@ -79,9 +79,11 @@ describe('Access manager apply "Read" role and approves developer access request }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, clientCredentials }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(clientCredentials.namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(clientCredentials.namespace) + }) }) }) @@ -99,7 +101,7 @@ describe('Access manager apply "Read" role and approves developer access request }) it('approves an access request', () => { - consumers.approvePendingRequest() + consumers.approvePendingRequest(true) }) after(() => { @@ -142,10 +144,10 @@ describe('Update Kong plugin and verify that only only GET method is allowed for it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { cy.publishApi('cc-service-plugin.yml', clientCredentials.namespace, true).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + // cy.get('@publishAPIResponse').then((res: any) => { + // cy.log(JSON.stringify(res.body)) + // expect(res.body.message).to.contains("Sync successful") + // }) }) }) }) diff --git a/e2e/cypress/tests/08-client-role/04-write-client-role.ts b/e2e/cypress/tests/08-client-role/04-write-client-role.ts index edbfcfcd4..060933ec4 100644 --- a/e2e/cypress/tests/08-client-role/04-write-client-role.ts +++ b/e2e/cypress/tests/08-client-role/04-write-client-role.ts @@ -79,9 +79,11 @@ describe('Access manager apply "Write" role and approves developer access reques }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, clientCredentials }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(clientCredentials.namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(clientCredentials.namespace) + }) }) }) @@ -133,10 +135,10 @@ describe('Update Kong plugin and verify that only only PUT and POST methods are it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { cy.publishApi('cc-service-plugin.yml', clientCredentials.namespace, true).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + // cy.get('@publishAPIResponse').then((res: any) => { + // cy.log(JSON.stringify(res.body)) + // expect(res.body.message).to.contains("Sync successful") + // }) }) }) }) diff --git a/e2e/cypress/tests/08-client-role/05-check-without-role.ts b/e2e/cypress/tests/08-client-role/05-check-without-role.ts index 0a2d04402..e159768ea 100644 --- a/e2e/cypress/tests/08-client-role/05-check-without-role.ts +++ b/e2e/cypress/tests/08-client-role/05-check-without-role.ts @@ -58,11 +58,12 @@ describe('Reset Authorization profile to default (without any role)', () => { it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.replaceWordInJsonObject('ccplatform', nameSpace, 'cc-service-plugin.yml') cy.publishApi('cc-service-plugin.yml', clientCredentials.namespace,true).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + // cy.get('@publishAPIResponse').then((res: any) => { + // cy.log(JSON.stringify(res.body)) + // expect(res.body.message).to.contains("Sync successful") + // }) }) }) }) diff --git a/e2e/cypress/tests/09-update-product-env/01-client-credential-to-kong-acl-api.cy.ts b/e2e/cypress/tests/09-update-product-env/01-client-credential-to-kong-acl-api.cy.ts index 0b8e547ba..3d42867b8 100644 --- a/e2e/cypress/tests/09-update-product-env/01-client-credential-to-kong-acl-api.cy.ts +++ b/e2e/cypress/tests/09-update-product-env/01-client-credential-to-kong-acl-api.cy.ts @@ -72,10 +72,10 @@ describe('Change Authorization profile', () => { it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { cy.publishApi('cc-service-plugin.yml', clientCredentials.namespace,true).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + // cy.get('@publishAPIResponse').then((res: any) => { + // cy.log(JSON.stringify(res.body)) + // expect(res.body.message).to.contains("Sync successful") + // }) }) }) }) @@ -105,7 +105,7 @@ describe('Change Authorization profile', () => { }) }) }) - + after(() => { cy.logout() cy.clearLocalStorage({ log: true }) @@ -185,9 +185,11 @@ describe('Access manager approves developer access request for Kong API ACL auth }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, clientCredentials }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(clientCredentials.namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ clientCredentials }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(clientCredentials.namespace) + }) }) }) @@ -206,7 +208,7 @@ describe('Access manager approves developer access request for Kong API ACL auth cy.readFile('cypress/fixtures/state/store.json').then((store_cred) => { cy.get('@apiowner').then(({ clientCredentials }: any) => { let product = clientCredentials.clientIdSecret_authProfile.product - cy.makeKongRequest(product.environment.config.serviceName, 'GET',store_cred.apikey).then((response) => { + cy.makeKongRequest(product.environment.config.serviceName, 'GET', store_cred.apikey).then((response) => { cy.log(response) expect(response.status).to.be.equal(200) }) diff --git a/e2e/cypress/tests/09-update-product-env/02-kong-acl-api-to-client-credential.cy.ts b/e2e/cypress/tests/09-update-product-env/02-kong-acl-api-to-client-credential.cy.ts index 64e5684f4..dd50eed3e 100644 --- a/e2e/cypress/tests/09-update-product-env/02-kong-acl-api-to-client-credential.cy.ts +++ b/e2e/cypress/tests/09-update-product-env/02-kong-acl-api-to-client-credential.cy.ts @@ -62,7 +62,7 @@ describe('Change Authorization profile from Kong ACL-API to Client Credential', it('Deactivate the service for Test environment', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ product }: any) => { - pd.deactivateService(product.name, product.test_environment.name,product.test_environment.config) + pd.deactivateService(product.name, product.test_environment.name, product.test_environment.config) cy.wait(3000) }) }) @@ -77,27 +77,25 @@ describe('Change Authorization profile from Kong ACL-API to Client Credential', prod.environment.config.authIssuer = authProfile.name prod.environment.config.authIssuerEnv = authProfile.environmentConfig.environment pd.editProductEnvironmentConfig(prod.environment.config) - pd.generateKongPluginConfigForAuthScope(product.name, product.test_environment.name, 'service-plugin.yml',product.test_environment.config.serviceName) + pd.generateKongPluginConfigForAuthScope(product.name, product.test_environment.name, 'service-plugin.yml', product.test_environment.config.serviceName) // pd.generateKongPluginConfig(product.name, product.test_environment.name,'service-test.yml') }) }) }) it('applies authorization plugin to service published to Kong Gateway', () => { - cy.get('@apiowner').then(({ namespace }: any) => { - cy.publishApi('service-plugin.yml', nameSpace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) - }) + cy.publishApi('service-plugin.yml', nameSpace).then(() => { + // cy.get('@publishAPIResponse').then((res: any) => { + // cy.log(JSON.stringify(res.body)) + // expect(res.body.message).to.contains("Sync successful") + // }) }) }) it('activate the service for Test environment', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ product }: any) => { - pd.activateService(product.name, product.test_environment.name,product.test_environment.config) + pd.activateService(product.name, product.test_environment.name, product.test_environment.config) cy.wait(3000) }) }) @@ -193,9 +191,11 @@ describe('Access manager approves developer access request for Client ID/Secret }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace) + }) }) }) @@ -218,25 +218,30 @@ describe('Access manager approves developer access request for Client ID/Secret }) describe('Make an API request using Client ID, Secret, and Access Token', () => { + beforeEach(() => { + cy.fixture('apiowner').as('apiowner') + }) it('Get access token using client ID and secret; make API request', () => { cy.readFile('cypress/fixtures/state/store.json').then((store_res) => { - - let cc = JSON.parse(store_res.clientidsecret) - // let cc = JSON.parse(Cypress.env('clientidsecret')) - cy.log('cc-->' + cc.clientSecret) - cy.getAccessToken(cc.clientId, cc.clientSecret).then(() => { - cy.get('@accessTokenResponse').then((token_res: any) => { - let token = token_res.body.access_token - cy.request({ - url: Cypress.env('KONG_URL'), - headers: { - Host: 'a-service-for-newplatform-test.api.gov.bc.ca', - }, - auth: { - bearer: token, - }, - }).then((res) => { - expect(res.status).to.eq(200) + cy.get('@apiowner').then(({ product }: any) => { + let host = product.test_environment.config.serviceName+'.api.gov.bc.ca' + let cc = JSON.parse(store_res.clientidsecret) + // let cc = JSON.parse(Cypress.env('clientidsecret')) + cy.log('cc-->' + cc.clientSecret) + cy.getAccessToken(cc.clientId, cc.clientSecret).then(() => { + cy.get('@accessTokenResponse').then((token_res: any) => { + let token = token_res.body.access_token + cy.request({ + url: Cypress.env('KONG_URL'), + headers: { + Host: host, + }, + auth: { + bearer: token, + }, + }).then((res) => { + expect(res.status).to.eq(200) + }) }) }) }) diff --git a/e2e/cypress/tests/09-update-product-env/03-apply-multiple-services.cy.ts b/e2e/cypress/tests/09-update-product-env/03-apply-multiple-services.cy.ts index eff4e4eaa..6de9001f0 100644 --- a/e2e/cypress/tests/09-update-product-env/03-apply-multiple-services.cy.ts +++ b/e2e/cypress/tests/09-update-product-env/03-apply-multiple-services.cy.ts @@ -6,7 +6,8 @@ import HomePage from '../../pageObjects/home' import LoginPage from '../../pageObjects/login' import MyAccessPage from '../../pageObjects/myAccess' import Products from '../../pageObjects/products' - +let host_test: string +let host: string describe('Apply multiple services to the product environment', () => { const login = new LoginPage() @@ -60,7 +61,7 @@ describe('Apply multiple services to the product environment', () => { let authProfile = clientCredentials.clientIdSecret_KongKeyToCC.authProfile prod.environment.config.authIssuer = authProfile.name prod.environment.config.authIssuerEnv = authProfile.environmentConfig.environment - pd.editProductEnvironmentConfig(prod.environment.config, false ,false) + pd.editProductEnvironmentConfig(prod.environment.config, false, false) pd.generateKongPluginConfigForAuthScope(product.name, product.test_environment.name, 'service-plugin.yml', product.environment.config.serviceName) // pd.generateKongPluginConfig(product.name, product.test_environment.name,'service-test.yml') }) @@ -70,10 +71,10 @@ describe('Apply multiple services to the product environment', () => { it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ namespace }: any) => { cy.publishApi('service-plugin.yml', namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + // cy.get('@publishAPIResponse').then((res: any) => { + // cy.log(JSON.stringify(res.body)) + // expect(res.body.message).to.contains("Sync successful") + // }) }) }) }) @@ -96,45 +97,50 @@ describe('Apply multiple services to the product environment', () => { describe('Verify that the service is accessible using existing Client ID, Secret, and Access Token', () => { let token: string + beforeEach(() => { + cy.fixture('apiowner').as('apiowner') + }) it('Get access token using client ID and secret; make API request for test', () => { cy.readFile('cypress/fixtures/state/store.json').then((store_res) => { - - let cc = JSON.parse(store_res.clientidsecret) - // let cc = JSON.parse(Cypress.env('clientidsecret')) - cy.log('cc-->' + cc.clientSecret) - cy.getAccessToken(cc.clientId, cc.clientSecret).then(() => { - cy.get('@accessTokenResponse').then((token_res: any) => { - token = token_res.body.access_token - cy.request({ - url: Cypress.env('KONG_URL'), - headers: { - Host: 'a-service-for-newplatform-test.api.gov.bc.ca', - }, - auth: { - bearer: token, - }, - }).then((res) => { - expect(res.status).to.eq(200) + cy.get('@apiowner').then(({ product }: any) => { + host_test = product.test_environment.config.serviceName + '.api.gov.bc.ca' + host = product.environment.config.serviceName + '.api.gov.bc.ca' + let cc = JSON.parse(store_res.clientidsecret) + // let cc = JSON.parse(Cypress.env('clientidsecret')) + cy.log('cc-->' + cc.clientSecret) + cy.getAccessToken(cc.clientId, cc.clientSecret).then(() => { + cy.get('@accessTokenResponse').then((token_res: any) => { + token = token_res.body.access_token + cy.request({ + url: Cypress.env('KONG_URL'), + headers: { + Host: host_test, + }, + auth: { + bearer: token, + }, + }).then((res) => { + expect(res.status).to.eq(200) + }) }) }) }) }) - }) - it('Get access token using client ID and secret; make API request for Dev', () => { - cy.request({ - url: Cypress.env('KONG_URL'), - headers: { - Host: 'a-service-for-newplatform.api.gov.bc.ca', - }, - auth: { - bearer: token, - }, - }).then((res) => { - expect(res.status).to.eq(200) + it('Get access token using client ID and secret; make API request for Dev', () => { + cy.request({ + url: Cypress.env('KONG_URL'), + headers: { + Host: host, + }, + auth: { + bearer: token, + }, + }).then((res) => { + expect(res.status).to.eq(200) + }) }) }) - }) @@ -212,9 +218,11 @@ describe('Access manager approves developer access request for Client ID/Secret }) it('Access Manager logs in', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace) + }) }) }) @@ -250,7 +258,7 @@ describe('Verify that the service is accessible using new Client ID, Secret, and cy.request({ url: Cypress.env('KONG_URL'), headers: { - Host: 'a-service-for-newplatform-test.api.gov.bc.ca', + Host: host_test, }, auth: { bearer: token, @@ -267,7 +275,7 @@ describe('Verify that the service is accessible using new Client ID, Secret, and cy.request({ url: Cypress.env('KONG_URL'), headers: { - Host: 'a-service-for-newplatform.api.gov.bc.ca', + Host: host, }, auth: { bearer: token, diff --git a/e2e/cypress/tests/09-update-product-env/05-keycloak-shared-IDP-config.cy.ts b/e2e/cypress/tests/09-update-product-env/05-keycloak-shared-IDP-config.cy.ts index 48e92d978..7b38cefae 100644 --- a/e2e/cypress/tests/09-update-product-env/05-keycloak-shared-IDP-config.cy.ts +++ b/e2e/cypress/tests/09-update-product-env/05-keycloak-shared-IDP-config.cy.ts @@ -42,7 +42,7 @@ describe('Apply Shared IDP config at Keycloak user group', () => { }) it('Edit the namespace from the tree view', () => { - cy.get('@admin').then(({ namespace }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { cy.contains(namespace).click() userGroups.clickOnEditButton() }) diff --git a/e2e/cypress/tests/09-update-product-env/07-kong-public-auth.ts b/e2e/cypress/tests/09-update-product-env/07-kong-public-auth.ts index 12a48b816..39115b439 100644 --- a/e2e/cypress/tests/09-update-product-env/07-kong-public-auth.ts +++ b/e2e/cypress/tests/09-update-product-env/07-kong-public-auth.ts @@ -70,11 +70,8 @@ describe('Verify for Kong Public Auth', () => { it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ clientCredentials }: any) => { - cy.publishApi('cc-service.yml', clientCredentials.namespace,true).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) + cy.publishApi('cc-service-gwa.yml', clientCredentials.namespace,true).then((response:any) => { + expect(response.stdout).to.contain('Sync successful'); }) }) }) diff --git a/e2e/cypress/tests/10-clear-resources/01-create-api.cy.ts b/e2e/cypress/tests/10-clear-resources/01-create-api.cy.ts index d531bc523..9ffeb309f 100644 --- a/e2e/cypress/tests/10-clear-resources/01-create-api.cy.ts +++ b/e2e/cypress/tests/10-clear-resources/01-create-api.cy.ts @@ -8,8 +8,8 @@ describe('Create API Spec for Delete Resources', () => { const home = new HomePage() const sa = new ServiceAccountsPage() const pd = new Products() - var nameSpace: string - let userSession: string + let userSession: any + let namespace: string before(() => { cy.visit('/') @@ -22,25 +22,37 @@ describe('Create API Spec for Delete Resources', () => { cy.preserveCookies() cy.fixture('apiowner').as('apiowner') cy.fixture('api').as('api') - // cy.visit(login.path) + cy.visit(login.path) }) - it('authenticates Janis (api owner)', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) }) }) - it('creates and activates new namespace', () => { - cy.getUserSession().then(() => { - cy.get('@apiowner').then(({ deleteResources }: any) => { - nameSpace = deleteResources.namespace - home.createNamespace(deleteResources.namespace) - cy.get('@login').then(function (xhr: any) { - userSession = xhr.response.headers['x-auth-request-access-token'] - }) - }) - }) + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('create namespace using gwa cli command', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.exec('gwa namespace create --host ' + cleanedUrl + ' --scheme http', { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + cy.replaceWordInJsonObject('ns.deleteplatform', 'ns.' + namespace, 'service-clear-resources-gwa.yml') + cy.updateJsonValue('apiowner.json', 'deleteResources.namespace', namespace) + // cy.updateJsonValue('apiowner.json', 'clientCredentials.clientIdSecret.product.environment.name.config.serviceName', 'cc-service-for-' + namespace) + cy.executeCliCommand("gwa config set --namespace " + namespace) + }); + }) + + it('activates new namespace', () => { + home.useNamespace(namespace) }) it('creates a new service account', () => { @@ -53,13 +65,12 @@ describe('Create API Spec for Delete Resources', () => { it('publishes a new API to Kong Gateway', () => { cy.get('@apiowner').then(({ deleteResources }: any) => { - cy.publishApi('service-clear-resources.yml', deleteResources.namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - }) + cy.publishApi('service-clear-resources-gwa.yml', namespace).then((response: any) => { + expect(response.stdout).to.contain('Sync successful'); }) }) }) + it('creates as new product in the directory', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ deleteResources }: any) => { @@ -71,7 +82,7 @@ describe('Create API Spec for Delete Resources', () => { cy.get('@api').then(({ organization }: any) => { cy.setHeaders(organization.headers) cy.setAuthorizationToken(userSession) - cy.makeAPIRequest(organization.endPoint + '/' + organization.orgName + '/' + organization.orgExpectedList.name + '/namespaces/' + nameSpace, 'PUT').then((response) => { + cy.makeAPIRequest(organization.endPoint + '/' + organization.orgName + '/' + organization.orgExpectedList.name + '/namespaces/' + namespace, 'PUT').then((response) => { expect(response.status).to.be.equal(200) }) }) @@ -84,23 +95,22 @@ describe('Create API Spec for Delete Resources', () => { pd.updateDatasetNameToCatelogue(deleteResources.product.name, deleteResources.product.environment.name) }) }) - + it('publish product to directory', () => { cy.visit(sa.path) cy.visit(pd.path) cy.get('@apiowner').then(({ deleteResources }: any) => { pd.editProductEnvironment(deleteResources.product.name, deleteResources.product.environment.name) pd.editProductEnvironmentConfig(deleteResources.product.environment.config) - pd.generateKongPluginConfig(deleteResources.product.name, deleteResources.product.environment.name,'service-clear-resources.yml') + pd.generateKongPluginConfig(deleteResources.product.name, deleteResources.product.environment.name, 'service-clear-resources.yml') }) }) it('applies authorization plugin to service published to Kong Gateway', () => { cy.get('@apiowner').then(({ deleteResources }: any) => { - cy.publishApi('service-clear-resources-plugin.yml', deleteResources.namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - }) + cy.replaceWordInJsonObject('ns.deleteplatform', 'ns.' + namespace, 'service-clear-resources-plugin.yml') + cy.publishApi('service-clear-resources-plugin.yml', namespace).then((response: any) => { + expect(response.stdout).to.contain('Sync successful'); }) }) }) @@ -108,14 +118,14 @@ describe('Create API Spec for Delete Resources', () => { it('activate the service for Dev environment', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ deleteResources }: any) => { - pd.activateService(deleteResources.product.name, deleteResources.product.environment.name,deleteResources.product.environment.config) + pd.activateService(deleteResources.product.name, deleteResources.product.environment.name, deleteResources.product.environment.config) cy.wait(3000) }) }) after(() => { cy.logout() - cy.clearLocalStorage({log:true}) + cy.clearLocalStorage({ log: true }) cy.deleteAllCookies() }) }) diff --git a/e2e/cypress/tests/11-activity-feed/01-activity-feed.cy.ts b/e2e/cypress/tests/11-activity-feed/01-activity-feed.cy.ts index 83c376ebb..58bbff630 100644 --- a/e2e/cypress/tests/11-activity-feed/01-activity-feed.cy.ts +++ b/e2e/cypress/tests/11-activity-feed/01-activity-feed.cy.ts @@ -106,26 +106,33 @@ describe('Verify the Activity filter for users', () => { // cy.visit(login.path) }) + it('activates new namespace', () => { + cy.get('@apiowner').then(({ namespace }: any) => { + home.useNamespace(namespace) + }) + }) + + it('Navigate to activity page', () => { cy.visit(activity.path) }) it('Verify Activity filter for "Janis Smith" user', () => { - activity.checkActivityFilter("User","Janis Smith", response) + activity.checkActivityFilter("User", "Janis Smith", response) }) it('Verify Activity filter for "Harley Jones" user', () => { - activity.checkActivityFilter("User","Harley Jones", response) + activity.checkActivityFilter("User", "Harley Jones", response) }) it('Verify Activity filter for "Mark F Mark L" user', () => { - activity.checkActivityFilter("User","Mark F Mark L", response) + activity.checkActivityFilter("User", "Mark F Mark L", response) }) it('Verify Activities filter for consumer', () => { cy.readFile('cypress/fixtures/state/regen.json').then((store) => { let consumerID = store.consumernumber - activity.checkActivityFilter("Consumer",consumerID, response) + activity.checkActivityFilter("Consumer", consumerID, response) }) }) }) \ No newline at end of file diff --git a/e2e/cypress/tests/11-activity-feed/02-activity-feed-failure.cy.ts b/e2e/cypress/tests/11-activity-feed/02-activity-feed-failure.cy.ts index a7c69dc4e..d1535a836 100644 --- a/e2e/cypress/tests/11-activity-feed/02-activity-feed-failure.cy.ts +++ b/e2e/cypress/tests/11-activity-feed/02-activity-feed-failure.cy.ts @@ -112,6 +112,10 @@ describe('Create API, Product, and Authorization Profiles; Apply Auth Profiles t cy.visit(activity.path) }) + it('Load all the records by click on "Load More" button', () => { + activity.loadMoreRecords() + }) + it('Verify Activity filter foe all the listed activities', () => { activity.checkActivityFilter("User", "", response) }) diff --git a/e2e/cypress/tests/12-access-permission/01-create-api.cy.ts b/e2e/cypress/tests/12-access-permission/01-create-api.cy.ts index ed7f07ab5..f30f59aa4 100644 --- a/e2e/cypress/tests/12-access-permission/01-create-api.cy.ts +++ b/e2e/cypress/tests/12-access-permission/01-create-api.cy.ts @@ -8,8 +8,8 @@ describe('Create API Spec', () => { const home = new HomePage() const sa = new ServiceAccountsPage() const pd = new Products() - var nameSpace: string - let userSession: string + let userSession: any + let namespace: string before(() => { cy.visit('/') @@ -25,22 +25,34 @@ describe('Create API Spec', () => { cy.visit(login.path) }) - it('authenticates Janis (api owner)', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) }) }) - it('creates and activates new namespace', () => { - cy.getUserSession().then(() => { - cy.get('@apiowner').then(({ checkPermission }: any) => { - nameSpace = checkPermission.namespace - home.createNamespace(checkPermission.namespace) - cy.get('@login').then(function (xhr: any) { - userSession = xhr.response.headers['x-auth-request-access-token'] - }) - }) - }) + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('create namespace using gwa cli command', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.exec('gwa namespace create --host ' + cleanedUrl + ' --scheme http', { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + cy.replaceWordInJsonObject('ns.permission', 'ns.' + namespace, 'service-permission-gwa.yml') + cy.updateJsonValue('apiowner.json', 'checkPermission.namespace', namespace) + // cy.updateJsonValue('apiowner.json', 'clientCredentials.clientIdSecret.product.environment.name.config.serviceName', 'cc-service-for-' + namespace) + cy.executeCliCommand("gwa config set --namespace " + namespace) + }); + }) + + it('activates new namespace', () => { + home.useNamespace(namespace) }) it('creates a new service account', () => { @@ -52,12 +64,8 @@ describe('Create API Spec', () => { }) it('publishes a new API to Kong Gateway', () => { - cy.get('@apiowner').then(({ checkPermission }: any) => { - cy.publishApi('service-permission.yml', checkPermission.namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - }) - }) + cy.publishApi('service-permission-gwa.yml', namespace).then((response: any) => { + expect(response.stdout).to.contain('Sync successful'); }) }) @@ -72,7 +80,7 @@ describe('Create API Spec', () => { cy.get('@api').then(({ organization }: any) => { cy.setHeaders(organization.headers) cy.setAuthorizationToken(userSession) - cy.makeAPIRequest(organization.endPoint + '/' + organization.orgName + '/' + organization.orgExpectedList.name + '/namespaces/' + nameSpace, 'PUT').then((response) => { + cy.makeAPIRequest(organization.endPoint + '/' + organization.orgName + '/' + organization.orgExpectedList.name + '/namespaces/' + namespace, 'PUT').then((response) => { expect(response.status).to.be.equal(200) }) }) @@ -84,28 +92,27 @@ describe('Create API Spec', () => { pd.updateDatasetNameToCatelogue(checkPermission.product.name, checkPermission.product.environment.name) }) }) - + it('publish product to directory', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ checkPermission }: any) => { pd.editProductEnvironment(checkPermission.product.name, checkPermission.product.environment.name) pd.editProductEnvironmentConfig(checkPermission.product.environment.config) - pd.generateKongPluginConfig(checkPermission.product.name, checkPermission.product.environment.name,'service-permission.yml') + pd.generateKongPluginConfig(checkPermission.product.name, checkPermission.product.environment.name, 'service-permission.yml') }) }) it('applies authorization plugin to service published to Kong Gateway', () => { - cy.get('@apiowner').then(({ checkPermission }: any) => { - cy.publishApi('service-permission.yml', checkPermission.namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - }) - }) + cy.replaceWordInJsonObject('ns.permission', 'ns.' + namespace, 'service-permission-plugin.yml') + cy.replaceWordInJsonObject('ns.permission', 'ns.' + namespace, 'service-permission.yml') + cy.publishApi('service-permission-plugin.yml', namespace).then((res: any) => { + expect(res.stdout).to.contain('Sync successful'); }) }) after(() => { cy.logout() - cy.clearLocalStorage({log:true}) + cy.clearLocalStorage({ log: true }) cy.deleteAllCookies() }) }) diff --git a/e2e/cypress/tests/12-access-permission/04-access-manager.cy.ts b/e2e/cypress/tests/12-access-permission/04-access-manager.cy.ts index cb8a1166e..47a0a6668 100644 --- a/e2e/cypress/tests/12-access-permission/04-access-manager.cy.ts +++ b/e2e/cypress/tests/12-access-permission/04-access-manager.cy.ts @@ -51,7 +51,7 @@ describe('Verify that Mark is able to view the pending request', () => { const consumers = new ConsumersPage() const mp = new MyProfilePage() const na = new NamespaceAccessPage() - + before(() => { cy.visit('/') @@ -62,24 +62,28 @@ describe('Verify that Mark is able to view the pending request', () => { beforeEach(() => { cy.preserveCookies() cy.fixture('access-manager').as('access-manager') + cy.fixture('apiowner').as('apiowner') }) it('Authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, checkPermission }: any) => { - cy.visit(login.path) - cy.login(user.credentials.username, user.credentials.password) - cy.log('Logged in!') - home.useNamespace(checkPermission.namespace) - cy.visit(mp.path) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({checkPermission}: any) => { + cy.visit(login.path) + cy.login(user.credentials.username, user.credentials.password) + cy.log('Logged in!') + debugger + home.useNamespace(checkPermission.namespace) + cy.visit(mp.path) + }) }) }) - it('Navigate to Consumer Page to see the Approve Request option', ()=> { + it('Navigate to Consumer Page to see the Approve Request option', () => { cy.visit(consumers.path) }) - it('Verify that the option to approve request is displayed', ()=> { - consumers.isApproveAccessEnabled(true) + it('Verify that the option to approve request is displayed', () => { + consumers.isApproveAccessEnabled(true) }) after(() => { diff --git a/e2e/cypress/tests/12-access-permission/05-namespace-manage.cy.ts b/e2e/cypress/tests/12-access-permission/05-namespace-manage.cy.ts index 6453ed1ed..cb489c8e8 100644 --- a/e2e/cypress/tests/12-access-permission/05-namespace-manage.cy.ts +++ b/e2e/cypress/tests/12-access-permission/05-namespace-manage.cy.ts @@ -72,12 +72,14 @@ describe('Verify that Wendy is able to see all the options for the Namespace', ( }) it('Authenticates Wendy (Credential-Issuer)', () => { - cy.get('@credential-issuer').then(({ user, checkPermission }: any) => { - cy.visit(login.path) - cy.login(user.credentials.username, user.credentials.password) - cy.log('Logged in!') - home.useNamespace(checkPermission.namespace) - cy.visit(mp.path) + cy.get('@credential-issuer').then(({ user }: any) => { + cy.get('@apiowner').then(({ checkPermission }: any) => { + cy.visit(login.path) + cy.login(user.credentials.username, user.credentials.password) + cy.log('Logged in!') + home.useNamespace(checkPermission.namespace) + cy.visit(mp.path) + }) }) }) diff --git a/e2e/cypress/tests/12-access-permission/06-credential-issuer.cy.ts b/e2e/cypress/tests/12-access-permission/06-credential-issuer.cy.ts index aae608930..24efcdd65 100644 --- a/e2e/cypress/tests/12-access-permission/06-credential-issuer.cy.ts +++ b/e2e/cypress/tests/12-access-permission/06-credential-issuer.cy.ts @@ -66,15 +66,18 @@ describe('Verify that Wendy is able to generate authorization profile', () => { beforeEach(() => { cy.preserveCookies() cy.fixture('credential-issuer').as('credential-issuer') + cy.fixture('apiowner').as('apiowner') }) it('Authenticates Wendy (Credential-Issuer)', () => { - cy.get('@credential-issuer').then(({ user, checkPermission }: any) => { - cy.visit(login.path) - cy.login(user.credentials.username, user.credentials.password) - cy.log('Logged in!') - home.useNamespace(checkPermission.namespace) - cy.visit(mp.path) + cy.get('@credential-issuer').then(({ user }: any) => { + cy.get('@apiowner').then(({ checkPermission }: any) => { + cy.visit(login.path) + cy.login(user.credentials.username, user.credentials.password) + cy.log('Logged in!') + home.useNamespace(checkPermission.namespace) + cy.visit(mp.path) + }) }) }) diff --git a/e2e/cypress/tests/12-access-permission/07-namespace-view.cy.ts b/e2e/cypress/tests/12-access-permission/07-namespace-view.cy.ts index 1155bd1ab..f97827918 100644 --- a/e2e/cypress/tests/12-access-permission/07-namespace-view.cy.ts +++ b/e2e/cypress/tests/12-access-permission/07-namespace-view.cy.ts @@ -65,15 +65,18 @@ describe('Verify that Mark is unable to create service account', () => { cy.preserveCookies() cy.fixture('credential-issuer').as('credential-issuer') cy.fixture('access-manager').as('access-manager') + cy.fixture('apiowner').as('apiowner') }) it('authenticates Mark', () => { - cy.get('@access-manager').then(({ user, checkPermission }: any) => { - cy.visit(login.path) - cy.login(user.credentials.username, user.credentials.password) - cy.log('Logged in!') - home.useNamespace(checkPermission.namespace) - cy.visit(mp.path) + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ checkPermission }: any) => { + cy.visit(login.path) + cy.login(user.credentials.username, user.credentials.password) + cy.log('Logged in!') + home.useNamespace(checkPermission.namespace) + cy.visit(mp.path) + }) }) }) diff --git a/e2e/cypress/tests/12-access-permission/08-gateway-config.cy.ts b/e2e/cypress/tests/12-access-permission/08-gateway-config.cy.ts index 7f41fefd0..c10b06cb7 100644 --- a/e2e/cypress/tests/12-access-permission/08-gateway-config.cy.ts +++ b/e2e/cypress/tests/12-access-permission/08-gateway-config.cy.ts @@ -68,25 +68,25 @@ describe('Verify that Wendy is able to generate authorization profile', () => { beforeEach(() => { cy.preserveCookies() cy.fixture('credential-issuer').as('credential-issuer') + cy.fixture('apiowner').as('apiowner') }) it('Authenticates Wendy (Credential-Issuer)', () => { - cy.get('@credential-issuer').then(({ user, checkPermission }: any) => { - cy.visit(login.path) - cy.login(user.credentials.username, user.credentials.password) - cy.log('Logged in!') - home.useNamespace(checkPermission.namespace) - cy.visit(mp.path) + cy.get('@credential-issuer').then(({ user }: any) => { + cy.get('@apiowner').then(({ checkPermission }: any) => { + cy.visit(login.path) + cy.login(user.credentials.username, user.credentials.password) + cy.log('Logged in!') + home.useNamespace(checkPermission.namespace) + cy.visit(mp.path) + }) }) }) it('Verify that GWA API allows user to publish the API to Kong gateway', () => { - cy.get('@credential-issuer').then(({ checkPermission }: any) => { - cy.publishApi('service-permission.yml', checkPermission.namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - expect(JSON.stringify(res.body.message)).to.be.contain('Sync successful.') - expect(res.statusCode).to.be.equal(200) - }) + cy.get('@apiowner').then(({ checkPermission }: any) => { + cy.publishApi('service-permission.yml', checkPermission.namespace).then((response: any) => { + expect(response.stdout).to.contain('Sync successful'); }) }) }) diff --git a/e2e/cypress/tests/14-namespace-preview-mode/01-create-api.cy.ts b/e2e/cypress/tests/13-namespace-preview-mode/01-create-api.cy.ts similarity index 54% rename from e2e/cypress/tests/14-namespace-preview-mode/01-create-api.cy.ts rename to e2e/cypress/tests/13-namespace-preview-mode/01-create-api.cy.ts index 7011be473..c1c977618 100644 --- a/e2e/cypress/tests/14-namespace-preview-mode/01-create-api.cy.ts +++ b/e2e/cypress/tests/13-namespace-preview-mode/01-create-api.cy.ts @@ -8,8 +8,8 @@ describe('Create API Spec', () => { const home = new HomePage() const sa = new ServiceAccountsPage() const pd = new Products() - var nameSpace: string - let userSession: string + let userSession: any + let namespace: string before(() => { cy.visit('/') @@ -25,22 +25,33 @@ describe('Create API Spec', () => { cy.visit(login.path) }) - it('authenticates Janis (api owner)', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) }) }) - it('creates and activates new namespace', () => { - cy.getUserSession().then(() => { - cy.get('@apiowner').then(({ namespacePreview }: any) => { - nameSpace = namespacePreview.namespace - home.createNamespace(namespacePreview.namespace) - cy.get('@login').then(function (xhr: any) { - userSession = xhr.response.headers['x-auth-request-access-token'] - }) - }) - }) + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('create namespace using gwa cli command', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.exec('gwa namespace create --host ' + cleanedUrl + ' --scheme http', { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + cy.updateJsonValue('apiowner.json', 'namespacePreview.namespace', namespace) + // cy.updateJsonValue('apiowner.json', 'clientCredentials.clientIdSecret.product.environment.name.config.serviceName', 'cc-service-for-' + namespace) + cy.executeCliCommand("gwa config set --namespace " + namespace) + }); + }) + + it('activates new namespace', () => { + home.useNamespace(namespace) }) it('creates a new service account', () => { @@ -51,16 +62,6 @@ describe('Create API Spec', () => { sa.saveServiceAcctCreds() }) - it('publishes a new API to Kong Gateway', () => { - cy.get('@apiowner').then(({ namespacePreview }: any) => { - cy.publishApi('service-permission.yml', namespacePreview.namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - }) - }) - }) - }) - it('creates as new product in the directory', () => { cy.visit(pd.path) cy.get('@apiowner').then(({ namespacePreview }: any) => { @@ -75,18 +76,9 @@ describe('Create API Spec', () => { }) }) - it('applies authorization plugin to service published to Kong Gateway', () => { - cy.get('@apiowner').then(({ namespacePreview }: any) => { - cy.publishApi('service-permission.yml', namespacePreview.namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - }) - }) - }) - }) - after(() => { cy.logout() - cy.clearLocalStorage({log:true}) + cy.clearLocalStorage({ log: true }) cy.deleteAllCookies() }) }) diff --git a/e2e/cypress/tests/14-namespace-preview-mode/02-namespace-preview-mode.cy.ts b/e2e/cypress/tests/13-namespace-preview-mode/02-namespace-preview-mode.cy.ts similarity index 100% rename from e2e/cypress/tests/14-namespace-preview-mode/02-namespace-preview-mode.cy.ts rename to e2e/cypress/tests/13-namespace-preview-mode/02-namespace-preview-mode.cy.ts diff --git a/e2e/cypress/tests/15-org-assignment/01-client-cred-team-access.ts b/e2e/cypress/tests/14-org-assignment/01-client-cred-team-access.ts similarity index 80% rename from e2e/cypress/tests/15-org-assignment/01-client-cred-team-access.ts rename to e2e/cypress/tests/14-org-assignment/01-client-cred-team-access.ts index 6ac09b0c5..3d491297b 100644 --- a/e2e/cypress/tests/15-org-assignment/01-client-cred-team-access.ts +++ b/e2e/cypress/tests/14-org-assignment/01-client-cred-team-access.ts @@ -8,6 +8,7 @@ import Products from '../../pageObjects/products' import ServiceAccountsPage from '../../pageObjects/serviceAccounts' import MyAccessPage from '../../pageObjects/myAccess' import ConsumersPage from '../../pageObjects/consumers' +let namespace: string describe('Add Organization to publish API', () => { const login = new LoginPage() @@ -16,6 +17,8 @@ describe('Add Organization to publish API', () => { const pd = new Products() const sa = new ServiceAccountsPage() const apiDir = new ApiDirectoryPage() + let userSession: any + before(() => { cy.visit('/') @@ -27,20 +30,36 @@ describe('Add Organization to publish API', () => { beforeEach(() => { cy.preserveCookies() cy.fixture('apiowner').as('apiowner') - // cy.visit(login.path) + cy.visit(login.path) }) - it('Authenticates api owner', () => { - cy.get('@apiowner').then(({ user, orgAssignment }: any) => { - cy.login(user.credentials.username, user.credentials.password) - // home.useNamespace(orgAssignment.namespace) + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) }) }) - it('Creates and activates new namespace', () => { - cy.get('@apiowner').then(({ orgAssignment }: any) => { - home.createNamespace(orgAssignment.namespace) - }) + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('create namespace using gwa cli command', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.exec('gwa namespace create --host ' + cleanedUrl + ' --scheme http', { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + cy.updateJsonValue('apiowner.json', 'orgAssignment.namespace', namespace) + // cy.updateJsonValue('apiowner.json', 'clientCredentials.clientIdSecret.product.environment.name.config.serviceName', 'cc-service-for-' + namespace) + cy.executeCliCommand("gwa config set --namespace " + namespace) + }); + }) + + it('activates new namespace', () => { + home.useNamespace(namespace) }) it('creates a new service account', () => { @@ -76,13 +95,9 @@ describe('Add Organization to publish API', () => { }) it('applies authorization plugin to service published to Kong Gateway', () => { - cy.get('@apiowner').then(({ orgAssignment }: any) => { - cy.publishApi('org-service-plugin.yml', orgAssignment.namespace).then(() => { - cy.get('@publishAPIResponse').then((res: any) => { - cy.log(JSON.stringify(res.body)) - expect(res.body.message).to.contains("Sync successful") - }) - }) + cy.replaceWordInJsonObject('ns.orgassignment', 'ns.' + namespace, 'org-service-plugin.yml') + cy.publishApi('org-service-plugin.yml', namespace).then((response: any) => { + expect(response.stdout).to.contain('Sync successful'); }) }) @@ -107,7 +122,9 @@ describe('Add Organization to publish API', () => { it('Verify Organization Administrator notification banner', () => { cy.visit(apiDir.path) cy.get('@apiowner').then(({ orgAssignment }: any) => { - apiDir.checkOrgAdminNotificationBanner(orgAssignment.orgAdminNotification) + cy.replaceWord(orgAssignment.orgAdminNotification.parent, 'orgassignment', namespace).then((updatedNotification: string) => { + apiDir.checkOrgAdminNotificationBanner(updatedNotification, orgAssignment.orgAdminNotification.child) + }) }) }) @@ -138,15 +155,13 @@ describe('Org Admin approves the request', () => { }) it('Authenticates Product Owner', () => { - cy.get('@product-owner').then(({ user, namespace }: any) => { + cy.get('@product-owner').then(({ user }: any) => { cy.login(user.credentials.username, user.credentials.password) }) }) it('Select the namespace', () => { - cy.get('@product-owner').then(({ namespace }: any) => { - home.useNamespace(namespace) - }) + home.useNamespace(namespace) }) it('Clik on Enable Publishing option from Namespace Page', () => { @@ -188,9 +203,9 @@ describe('Activate the API to make it visible in API Directory', () => { }) it('Authenticates api owner', () => { - cy.get('@apiowner').then(({ user, orgAssignment }: any) => { + cy.get('@apiowner').then(({ user }: any) => { cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(orgAssignment.namespace) + home.useNamespace(namespace) }) }) @@ -217,7 +232,7 @@ describe('Activate the API to make it visible in API Directory', () => { cy.get('@apiowner').then(({ orgAssignment }: any) => { let product = orgAssignment.product.name // cy.contains('button').click() - apiDir.isProductDisplay(product,true) + apiDir.isProductDisplay(product, true) }) }) @@ -299,11 +314,9 @@ describe('Access manager approves developer access request for Kong API ACL auth }) it('Access Manager logs in', () => { - cy.get('@apiowner').then(({ orgAssignment }: any) => { - cy.get('@access-manager').then(({ user}: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(orgAssignment.namespace) - }) + cy.get('@access-manager').then(({ user }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace) }) }) diff --git a/e2e/cypress/tests/15-org-assignment/02-multiple-org-admin.ts b/e2e/cypress/tests/14-org-assignment/02-multiple-org-admin.ts similarity index 71% rename from e2e/cypress/tests/15-org-assignment/02-multiple-org-admin.ts rename to e2e/cypress/tests/14-org-assignment/02-multiple-org-admin.ts index ecf53ccb7..4a04424c9 100644 --- a/e2e/cypress/tests/15-org-assignment/02-multiple-org-admin.ts +++ b/e2e/cypress/tests/14-org-assignment/02-multiple-org-admin.ts @@ -64,6 +64,9 @@ describe('Multiple Org Adming for the organization', () => { const pd = new Products() const sa = new ServiceAccountsPage() const apiDir = new ApiDirectoryPage() + const login = new LoginPage() + let userSession: any + let namespace: any before(() => { cy.visit('/') @@ -75,21 +78,40 @@ describe('Multiple Org Adming for the organization', () => { beforeEach(() => { cy.preserveCookies() cy.fixture('apiowner').as('apiowner') - // cy.visit(login.path) + cy.visit(login.path) }) - it('Authenticates api owner', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) + + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) }) }) - it('Creates and activates new namespace', () => { - cy.get('@apiowner').then(({ orgAssignmentMultipleAdmin }: any) => { - home.createNamespace(orgAssignmentMultipleAdmin.namespace) - }) + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('create namespace using gwa cli command', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.exec('gwa namespace create --host ' + cleanedUrl + ' --scheme http', { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + cy.updateJsonValue('apiowner.json', 'orgAssignment.namespace', namespace) + // cy.updateJsonValue('apiowner.json', 'clientCredentials.clientIdSecret.product.environment.name.config.serviceName', 'cc-service-for-' + namespace) + cy.executeCliCommand("gwa config set --namespace " + namespace) + }); + }) + + it('activates new namespace', () => { + home.useNamespace(namespace) }) + it('creates a new service account', () => { cy.visit(sa.path) cy.get('@apiowner').then(({ serviceAccount }: any) => { diff --git a/e2e/cypress/tests/15-org-assignment/03-multiple-org-admin-org-unit.ts b/e2e/cypress/tests/14-org-assignment/03-multiple-org-admin-org-unit.ts similarity index 74% rename from e2e/cypress/tests/15-org-assignment/03-multiple-org-admin-org-unit.ts rename to e2e/cypress/tests/14-org-assignment/03-multiple-org-admin-org-unit.ts index 512a0987a..2d10435b5 100644 --- a/e2e/cypress/tests/15-org-assignment/03-multiple-org-admin-org-unit.ts +++ b/e2e/cypress/tests/14-org-assignment/03-multiple-org-admin-org-unit.ts @@ -81,6 +81,9 @@ describe('Multiple Org Admin for the organization', () => { const pd = new Products() const sa = new ServiceAccountsPage() const apiDir = new ApiDirectoryPage() + const login = new LoginPage() + let userSession: any + let namespace: any before(() => { cy.visit('/') @@ -92,20 +95,36 @@ describe('Multiple Org Admin for the organization', () => { beforeEach(() => { cy.preserveCookies() cy.fixture('apiowner').as('apiowner') - // cy.visit(login.path) + cy.visit(login.path) }) - it('Authenticates api owner', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) }) }) - it('Creates and activates new namespace', () => { - cy.get('@apiowner').then(({ orgAssignmentOrgUnit }: any) => { - home.createNamespace(orgAssignmentOrgUnit.namespace) - // home.useNamespace(orgAssignmentMultipleAdmin.namespace) - }) + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('create namespace using gwa cli command', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.exec('gwa namespace create --host ' + cleanedUrl + ' --scheme http', { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + cy.updateJsonValue('apiowner.json', 'orgAssignment.namespace', namespace) + // cy.updateJsonValue('apiowner.json', 'clientCredentials.clientIdSecret.product.environment.name.config.serviceName', 'cc-service-for-' + namespace) + cy.executeCliCommand("gwa config set --namespace " + namespace) + }); + }) + + it('activates new namespace', () => { + home.useNamespace(namespace) }) it('creates a new service account', () => { diff --git a/e2e/cypress/tests/15-aps-api/01-create-api.cy.ts b/e2e/cypress/tests/15-aps-api/01-create-api.cy.ts new file mode 100644 index 000000000..d47a5fa60 --- /dev/null +++ b/e2e/cypress/tests/15-aps-api/01-create-api.cy.ts @@ -0,0 +1,73 @@ +import HomePage from '../../pageObjects/home' +import LoginPage from '../../pageObjects/login' +import Products from '../../pageObjects/products' +import ServiceAccountsPage from '../../pageObjects/serviceAccounts' + +describe('Create API Spec', () => { + const login = new LoginPage() + const home = new HomePage() + const sa = new ServiceAccountsPage() + const pd = new Products() + let userSession: any + let namespace: any + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + cy.resetState() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + cy.fixture('api').as('api') + cy.visit(login.path) + }) + + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) + }) + }) + + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('create namespace using gwa cli command', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.exec('gwa namespace create --host ' + cleanedUrl + ' --scheme http', { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + cy.updateJsonValue('apiowner.json', 'apiTest.namespace', namespace) + cy.updateJsonValue('api.json', 'organization.expectedNamespace.name', namespace) + // cy.updateJsonValue('apiowner.json', 'clientCredentials.clientIdSecret.product.environment.name.config.serviceName', 'cc-service-for-' + namespace) + cy.executeCliCommand("gwa config set --namespace " + namespace) + }); + }) + + it('activates new namespace', () => { + home.useNamespace(namespace) + }) + + it('Associate Namespace to the organization Unit', () => { + cy.get('@api').then(({ organization }: any) => { + cy.setHeaders(organization.headers) + cy.setAuthorizationToken(userSession) + cy.makeAPIRequest(organization.endPoint + '/' + organization.orgName + '/' + organization.orgExpectedList.name + '/namespaces/' + namespace, 'PUT').then((response) => { + expect(response.status).to.be.equal(200) + }) + }) + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({log:true}) + cy.deleteAllCookies() + }) +}) diff --git a/e2e/cypress/tests/16-aps-api/02-organization.cy.ts b/e2e/cypress/tests/15-aps-api/02-organization.cy.ts similarity index 92% rename from e2e/cypress/tests/16-aps-api/02-organization.cy.ts rename to e2e/cypress/tests/15-aps-api/02-organization.cy.ts index feb5a92a4..5b9649af9 100644 --- a/e2e/cypress/tests/16-aps-api/02-organization.cy.ts +++ b/e2e/cypress/tests/15-aps-api/02-organization.cy.ts @@ -54,7 +54,7 @@ describe('Verify /Organization/{Org} end point', () => { it('Verify the status code and response message for invalid organization name', () => { cy.get('@api').then(({ organization }: any) => { - cy.makeAPIRequest(organization.endPoint + '/health' , 'GET').then((response) => { + cy.makeAPIRequest(organization.endPoint + '/health', 'GET').then((response) => { expect(response.status).to.be.oneOf([404, 422]) expect(response.body.message).to.be.equal("Organization not found.") }) @@ -123,6 +123,20 @@ describe('Get the Organization Role', () => { }) }) + it('Get the list of roles and verify the success code in the response', () => { + cy.makeAPIRequest('ds/api/v2/roles', 'GET').then((res) => { + expect(res.status).to.be.equal(200) + response = res.body + }) + }) + + it('Compare the roles values in response against the expected values', () => { + cy.get('@api').then(({ organization }: any) => { + expectedResponse = organization.expectedRoles + cy.compareJSONObjects(response, expectedResponse) + }) + }) + }) describe('Get the Namespace associated with the organization', () => { @@ -156,6 +170,7 @@ describe('Get the Namespace associated with the organization', () => { cy.get('@api').then(({ organization }: any) => { expectedResponse = organization.expectedNamespace // assert.isTrue(Cypress._.isEqual(response, expectedResponse)) + debugger cy.compareJSONObjects(response, expectedResponse, true) }) }) @@ -246,7 +261,7 @@ describe('Add and Get Organization Access', () => { after(() => { cy.logout() - cy.clearLocalStorage({log:true}) + cy.clearLocalStorage({ log: true }) cy.deleteAllCookies() - }) + }) }) \ No newline at end of file diff --git a/e2e/cypress/tests/16-aps-api/03-documentation.cy.ts b/e2e/cypress/tests/15-aps-api/03-documentation.cy.ts similarity index 81% rename from e2e/cypress/tests/16-aps-api/03-documentation.cy.ts rename to e2e/cypress/tests/15-aps-api/03-documentation.cy.ts index b41c52f69..9f749ba12 100644 --- a/e2e/cypress/tests/16-aps-api/03-documentation.cy.ts +++ b/e2e/cypress/tests/15-aps-api/03-documentation.cy.ts @@ -2,7 +2,8 @@ import HomePage from "../../pageObjects/home" import LoginPage from "../../pageObjects/login" let userSession: any let slugValue: string - +let namespace: string +let updatedDocumentEndPoint: string describe('Get the user session token', () => { const login = new LoginPage() @@ -24,6 +25,7 @@ describe('Get the user session token', () => { cy.get('@apiowner').then(({ apiTest }: any) => { cy.getUserSessionTokenValue(apiTest.namespace).then((value) => { userSession = value + namespace = apiTest.namespace }) }) }) @@ -48,8 +50,11 @@ describe('API Tests for Updating documentation', () => { it('Put the resource and verify the success code in the response', () => { cy.get('@api').then(({ documentation }: any) => { - cy.makeAPIRequest(documentation.endPoint, 'PUT').then((response) => { - expect(response.status).to.be.equal(200) + cy.replaceWord(documentation.endPoint, 'apiplatform', namespace).then((updatedEndPoint: string) => { + updatedDocumentEndPoint = updatedEndPoint + cy.makeAPIRequest(updatedDocumentEndPoint, 'PUT').then((response) => { + expect(response.status).to.be.equal(200) + }) }) }) }) @@ -74,12 +79,10 @@ describe('API Tests for Fetching documentation', () => { }) it('Get the resource and verify the success code in the response', () => { - cy.get('@api').then(({ documentation }: any) => { - cy.makeAPIRequest(documentation.endPoint, 'GET').then((res) => { - expect(res.status).to.be.equal(200) - slugValue = res.body[0].slug - response = res.body[0] - }) + cy.makeAPIRequest(updatedDocumentEndPoint, 'GET').then((res) => { + expect(res.status).to.be.equal(200) + slugValue = res.body[0].slug + response = res.body[0] }) }) @@ -109,19 +112,16 @@ describe('API Tests for Deleting documentation', () => { }) it('Verify the status code and response message for invalid slugvalue', () => { - cy.get('@api').then(({ documentation }: any) => { - cy.makeAPIRequest(documentation.endPoint + '/platform_test', 'DELETE').then((response) => { - expect(response.status).to.be.oneOf([404, 422]) - expect(response.body.message).to.be.equal("Content not found") - }) + cy.makeAPIRequest(updatedDocumentEndPoint + '/platform_test', 'DELETE').then((response) => { + expect(response.status).to.be.oneOf([404, 422]) + expect(response.body.message).to.be.equal("Content not found") }) }) + it('Delete the documentation', () => { - cy.get('@api').then(({ documentation }: any) => { - cy.makeAPIRequest(documentation.endPoint + '/' + slugValue, 'DELETE').then((response) => { - expect(response.status).to.be.equal(200) - }) + cy.makeAPIRequest(updatedDocumentEndPoint + '/' + slugValue, 'DELETE').then((response) => { + expect(response.status).to.be.equal(200) }) }) }) @@ -144,11 +144,9 @@ describe('API Tests to verify no value in Get call after deleting document conte }) it('Delete the documentation', () => { - cy.get('@api').then(({ documentation }: any) => { - cy.makeAPIRequest(documentation.endPoint, 'GET').then((response) => { - expect(response.status).to.be.equal(200) - expect(response.body).to.be.empty - }) + cy.makeAPIRequest(updatedDocumentEndPoint, 'GET').then((response) => { + expect(response.status).to.be.equal(200) + expect(response.body).to.be.empty }) }) }) @@ -180,10 +178,8 @@ describe('API Tests to verify Get documentation content', () => { }) it('Put the resource and verify the success code in the response', () => { - cy.get('@api').then(({ documentation }: any) => { - cy.makeAPIRequest(documentation.endPoint, 'PUT').then((response) => { - expect(response.status).to.be.equal(200) - }) + cy.makeAPIRequest(updatedDocumentEndPoint, 'PUT').then((response) => { + expect(response.status).to.be.equal(200) }) }) diff --git a/e2e/cypress/tests/16-aps-api/04-keycloak-shared-IDP-config.cy.ts b/e2e/cypress/tests/15-aps-api/04-keycloak-shared-IDP-config.cy.ts similarity index 100% rename from e2e/cypress/tests/16-aps-api/04-keycloak-shared-IDP-config.cy.ts rename to e2e/cypress/tests/15-aps-api/04-keycloak-shared-IDP-config.cy.ts diff --git a/e2e/cypress/tests/15-aps-api/05-authorizationProfiles.cy.ts b/e2e/cypress/tests/15-aps-api/05-authorizationProfiles.cy.ts new file mode 100644 index 000000000..5417e4c65 --- /dev/null +++ b/e2e/cypress/tests/15-aps-api/05-authorizationProfiles.cy.ts @@ -0,0 +1,300 @@ +import AuthorizationProfile from "../../pageObjects/authProfile" +import HomePage from "../../pageObjects/home" +import login from "../../pageObjects/login" +import LoginPage from "../../pageObjects/login" +let userSession: any +let testData = require("../../fixtures/test_data/authorizationProfile.json") +let namespace: string +let updatedAuthProfileEndPoint: string + +describe('Get the user session token', () => { + + const login = new LoginPage() + const home = new HomePage() + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + cy.visit(login.path) + }) + + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace).then((value) => { + userSession = value + namespace = apiTest.namespace + }) + }) + }) + + it('Set token with gwa config command', () => { + cy.exec('gwa config set --token ' + userSession, { timeout: 3000, failOnNonZeroExit: false }).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) +}) + +testData.forEach((testCase: any) => { + describe('API Tests for Authorization Profiles', () => { + + var response: any + var actualResponse: any = {} + var expectedResponse: any = {} + + beforeEach(() => { + cy.fixture('api').as('api') + }) + + it('Prepare the Request Specification for the API', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + cy.setHeaders(authorizationProfiles.headers) + cy.setAuthorizationToken(userSession) + cy.setRequestBody(testCase.body) + }) + }) + + it('Put the resource and verify the success code in the response', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + cy.replaceWord(authorizationProfiles.endPoint, 'apiplatform', namespace).then((updatedEndPoint: string) => { + updatedAuthProfileEndPoint = updatedEndPoint + cy.makeAPIRequest(updatedAuthProfileEndPoint, 'PUT').then((response) => { + expect(response.status).to.be.equal(200) + }) + }) + }) + }) + + it('Get the resource and verify the success code in the response', () => { + cy.makeAPIRequest(updatedAuthProfileEndPoint, 'GET').then((res) => { + expect(res.status).to.be.equal(200) + response = res.body + }) + }) + + it('Compare the values in response against the values passed in the request', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + actualResponse = response + expectedResponse = testCase.body + cy.compareJSONObjects(actualResponse, expectedResponse, true) + }) + }) + + it('Delete the authorization profile', () => { + cy.makeAPIRequest(updatedAuthProfileEndPoint + '/' + testCase.body.name, 'DELETE').then((response) => { + expect(response.status).to.be.equal(200) + }) + }) + + + it('Verify that the authorization profile is deleted', () => { + cy.makeAPIRequest(updatedAuthProfileEndPoint, 'GET').then((response) => { + expect(response.status).to.be.equal(200) + expect(response.body.length).to.be.equal(0) + }) + }) + }) + + after(() => { + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) +}) + +describe('API Tests for Authorization Profiles created with inheritFrom attribute set to a valid shared Issuer', () => { + + let response: any + let actualResponse: any + let expectedResponse: any + + beforeEach(() => { + cy.fixture('api').as('api') + cy.fixture('apiowner').as('apiowner') + }) + + it('Prepare the Request Specification to create a shared IDP profile', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + cy.setHeaders(authorizationProfiles.headers) + cy.setAuthorizationToken(userSession) + cy.setRequestBody(authorizationProfiles.shared_IDP_body) + }) + }) + + it('Put the resource to create shared IDP profile and verify the success code in the response', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.makeAPIRequest('ds/api/v2/namespaces/' + apiTest.namespace + '/issuers', 'PUT').then((response) => { + expect(response.status).to.be.equal(200) + }) + }) + }) + + it('Prepare the Request Specification to create a shared IDP profile using inheritFrom attribute', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + cy.setHeaders(authorizationProfiles.headers) + cy.setAuthorizationToken(userSession) + cy.setRequestBody(authorizationProfiles.shared_IDP_inheritFrom) + }) + }) + + it('Create an authorization profile using inheritFrom attribute and verify the success code in the response', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.makeAPIRequest('ds/api/v2/namespaces/' + apiTest.namespace + '/issuers', 'PUT').then((response) => { + expect(response.status).to.be.equal(200) + expect(response.body.result).to.be.equal("created") + }) + }) + }) + + it('Get list of authorization profile and verify the success code in the response', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.makeAPIRequest('ds/api/v2/namespaces/' + apiTest.namespace + '/issuers', 'GET').then((res) => { + expect(res.status).to.be.equal(200) + response = res.body + }) + }) + }) + + it('Compare the values in response against the values passed in the request', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + actualResponse = response + expectedResponse = authorizationProfiles.shared_IDP_inheritFrom_expectedResponse + cy.compareJSONObjects(actualResponse, expectedResponse, true) + }) + }) + +}) + +describe('Published a shared authorization profile', () => { + + let response: any + let actualResponse: any + let expectedResponse: any + + beforeEach(() => { + cy.fixture('api').as('api') + cy.fixture('apiowner').as('apiowner') + }) + + it('Prepare the Request Specification to create a shared IDP profile', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + cy.setHeaders(authorizationProfiles.headers) + cy.setAuthorizationToken(userSession) + cy.setRequestBody(authorizationProfiles.shared_gwa) + }) + }) + + it('Create a shared credential issuer', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.makeAPIRequest('ds/api/v2/namespaces/' + apiTest.namespace + '/issuers', 'PUT').then((response) => { + expect(response.status).to.be.equal(200) + expect(response.body.result).to.be.equal("created") + }) + }) + }) + + it('Get list of authorization profile and verify the success code in the response', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.makeAPIRequest('ds/api/v2/namespaces/' + apiTest.namespace + '/issuers', 'GET').then((res) => { + expect(res.status).to.be.equal(200) + response = res.body + }) + }) + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) + +}) + +describe('Deleted shared auth profile', () => { + + const login = new LoginPage() + const home = new HomePage() + const authProfile = new AuthorizationProfile() + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + cy.fixture('api').as('api') + }) + + it('Authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace).then((value) => { + userSession = value + namespace = apiTest.namespace + home.useNamespace(namespace); + }) + }) + }) + + it('Navigate to authorization profile page', () => { + cy.visit(authProfile.path) + }) + + it('Delete the authorizarion profile inherited from shared IDP', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + authProfile.deleteAuthProfile(authorizationProfiles.shared_IDP_inheritFrom_expectedResponse.name) + }) + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) +}) + +describe('Verify that client ID of deleted shared auth profile in IDP', () => { + + var nameSpace: string + const home = new HomePage() + const authProfile = new AuthorizationProfile() + + before(() => { + cy.visit(Cypress.env('KEYCLOAK_URL')) + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('developer').as('developer') + cy.fixture('apiowner').as('apiowner') + cy.fixture('state/regen').as('regen') + cy.fixture('admin').as('admin') + cy.fixture('api').as('api') + }) + + it('Authenticates Admin owner', () => { + cy.get('@admin').then(({ user }: any) => { + cy.contains('Administration Console').click({ force: true }) + cy.keycloakLogin(user.credentials.username, user.credentials.password) + }) + }) + + it('Navigate to Clients', () => { + cy.contains('Clients').click() + }) + + it('Verify that the client id of deleted shared auth profile does not display', () => { + cy.get('@api').then(({ authorizationProfiles }: any) => { + cy.contains(authorizationProfiles.shared_IDP_inheritFrom_expectedResponse.name).should('not.exist') + }) + }) + +}) \ No newline at end of file diff --git a/e2e/cypress/tests/16-aps-api/06-products.cy.ts b/e2e/cypress/tests/15-aps-api/06-products.cy.ts similarity index 80% rename from e2e/cypress/tests/16-aps-api/06-products.cy.ts rename to e2e/cypress/tests/15-aps-api/06-products.cy.ts index f9e69a7f5..0b9053cb1 100644 --- a/e2e/cypress/tests/16-aps-api/06-products.cy.ts +++ b/e2e/cypress/tests/15-aps-api/06-products.cy.ts @@ -4,6 +4,8 @@ import Products from "../../pageObjects/products" let userSession: any let productID: string let envID: string +let updatedProductEndPoint: string +let namespace: string describe('Get the user session token to check ', () => { @@ -27,6 +29,7 @@ describe('Get the user session token to check ', () => { cy.get('@apiowner').then(({ user, apiTest }: any) => { cy.login(user.credentials.username, user.credentials.password) home.useNamespace(apiTest.namespace) + namespace = apiTest.namespace cy.get('@login').then(function (xhr: any) { userSession = xhr.response.headers['x-auth-request-access-token'] }) @@ -56,20 +59,24 @@ describe('API Tests for Updating Products', () => { it('Put the resource and verify the success code in the response', () => { cy.get('@api').then(({ products }: any) => { - cy.makeAPIRequest(products.endPoint, 'PUT').then((response) => { - expect(response.status).to.be.equal(200) + cy.replaceWord(products.endPoint, 'apiplatform', namespace).then((updatedEndPoint: string) => { + updatedProductEndPoint = updatedEndPoint + cy.makeAPIRequest(updatedProductEndPoint, 'PUT').then((response) => { + expect(response.status).to.be.equal(200) + }) }) }) }) it('Get the resource and verify the success code and product name in the response', () => { cy.get('@api').then(({ products }: any) => { - cy.makeAPIRequest(products.endPoint, 'GET').then((res) => { + cy.makeAPIRequest(updatedProductEndPoint, 'GET').then((res) => { expect(res.status).to.be.equal(200) let index = res.body.findIndex((x: { name: string }) => x.name === products.body.name) response = res.body[index] productID = res.body[index].appId envID = res.body[index].environments[0].appId + debugger }) }) }) @@ -102,6 +109,7 @@ describe('Verify that created Product is displayed in UI', () => { it('authenticates Janis (api owner) to get the user session token', () => { cy.get('@apiowner').then(({ apiTest }: any) => { cy.getUserSessionTokenValue(apiTest.namespace).then((value) => { + home.useNamespace(apiTest.namespace) userSession = value }) }) @@ -134,15 +142,17 @@ describe('API Tests for Delete Products', () => { it('Delete the product environment and verify the success code in the response', () => { cy.get('@api').then(({ products }: any) => { - cy.makeAPIRequest(products.deleteEnvironmentEndPoint + '/' + envID, 'Delete').then((response) => { - expect(response.status).to.be.equal(200) + cy.replaceWord(products.deleteEnvironmentEndPoint, 'apiplatform', namespace).then((updatedEndPoint: string) => { + cy.makeAPIRequest(updatedEndPoint + '/' + envID, 'Delete').then((response) => { + expect(response.status).to.be.equal(200) + }) }) }) }) it('Get the resource and verify that product environment is deleted', () => { cy.get('@api').then(({ products }: any) => { - cy.makeAPIRequest(products.endPoint, 'GET').then((res) => { + cy.makeAPIRequest(updatedProductEndPoint, 'GET').then((res) => { expect(res.status).to.be.equal(200) let index = res.body.findIndex((x: { name: string }) => x.name === products.body.name) expect(res.body[index].environments).to.be.empty @@ -151,16 +161,14 @@ describe('API Tests for Delete Products', () => { }) it('Delete the product and verify the success code in the response', () => { - cy.get('@api').then(({ products }: any) => { - cy.makeAPIRequest(products.endPoint + '/' + productID, 'Delete').then((response) => { - expect(response.status).to.be.equal(200) - }) + cy.makeAPIRequest(updatedProductEndPoint + '/' + productID, 'Delete').then((response) => { + expect(response.status).to.be.equal(200) }) }) it('Get the resource and verify that product is deleted', () => { cy.get('@api').then(({ products }: any) => { - cy.makeAPIRequest(products.endPoint, 'GET').then((res) => { + cy.makeAPIRequest(updatedProductEndPoint, 'GET').then((res) => { expect(res.status).to.be.equal(200) response = res.body assert.equal(response.findIndex((x: { name: string }) => x.name === products.body.name), -1) diff --git a/e2e/cypress/tests/16-aps-api/07-api-directory.cy.ts b/e2e/cypress/tests/15-aps-api/07-api-directory.cy.ts similarity index 85% rename from e2e/cypress/tests/16-aps-api/07-api-directory.cy.ts rename to e2e/cypress/tests/15-aps-api/07-api-directory.cy.ts index ce6c1311f..19309fcd2 100644 --- a/e2e/cypress/tests/16-aps-api/07-api-directory.cy.ts +++ b/e2e/cypress/tests/15-aps-api/07-api-directory.cy.ts @@ -104,12 +104,10 @@ describe('API Tests for Updating dataset', () => { }) it('Get the resource (/organizations/{org}/datasets/{name}) and verify the success code in the response', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - cy.get('@api').then(({ apiDirectory, organization }: any) => { - cy.makeAPIRequest(apiDirectory.orgEndPoint + '/' + organization.orgName + '/datasets/' + apiDirectory.body.name, 'GET').then((res) => { - expect(res.status).to.be.equal(200) - response = res.body - }) + cy.get('@api').then(({ apiDirectory, organization }: any) => { + cy.makeAPIRequest(apiDirectory.orgEndPoint + '/' + organization.orgName + '/datasets/' + apiDirectory.body.name, 'GET').then((res) => { + expect(res.status).to.be.equal(200) + response = res.body }) }) }) @@ -121,12 +119,10 @@ describe('API Tests for Updating dataset', () => { }) it('Get the resource (/organizations/{org}/datasets) and verify the success code in the response', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - cy.get('@api').then(({ apiDirectory, organization }: any) => { - cy.makeAPIRequest(apiDirectory.orgEndPoint + '/' + organization.orgName + '/datasets/', 'GET').then((res) => { - expect(res.status).to.be.equal(200) - response = res.body - }) + cy.get('@api').then(({ apiDirectory, organization }: any) => { + cy.makeAPIRequest(apiDirectory.orgEndPoint + '/' + organization.orgName + '/datasets/', 'GET').then((res) => { + expect(res.status).to.be.equal(200) + response = res.body }) }) }) @@ -150,7 +146,7 @@ describe('API Tests for Updating dataset', () => { it('Verify the expected directory details are display in the response', () => { cy.get('@api').then(({ apiDirectory }: any) => { - // cy.compareJSONObjects(response, apiDirectory.directory, true) + cy.compareJSONObjects(response, apiDirectory.directory, true) }) }) @@ -187,11 +183,11 @@ describe('API Tests for Updating dataset', () => { }) }) - // it('Verify the expected namespace directory details are display in the response', () => { - // cy.get('@api').then(({ apiDirectory }: any) => { - // cy.compareJSONObjects(response, apiDirectory.directory, true) - // }) - // }) + it('Verify the expected namespace directory details are display in the response', () => { + cy.get('@api').then(({ apiDirectory }: any) => { + cy.compareJSONObjects(response, apiDirectory.directory, false) + }) + }) it('Get the namespace directory details by its ID (/namespaces/{ns}/directory/{id}) and verify the success code in the response', () => { cy.get('@apiowner').then(({ namespace }: any) => { @@ -215,11 +211,9 @@ describe('API Tests for Updating dataset', () => { }) it('Delete the dataset (/organizations/{org}/datasets/{name}) and verify the success code in the response', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - cy.get('@api').then(({ apiDirectory, organization }: any) => { - cy.makeAPIRequest(apiDirectory.orgEndPoint + '/' + organization.orgName + '/datasets/' + apiDirectory.body.name, 'DELETE').then((res) => { - expect(res.status).to.be.equal(200) - }) + cy.get('@api').then(({ apiDirectory, organization }: any) => { + cy.makeAPIRequest(apiDirectory.orgEndPoint + '/' + organization.orgName + '/datasets/' + apiDirectory.body.name, 'DELETE').then((res) => { + expect(res.status).to.be.equal(200) }) }) }) diff --git a/e2e/cypress/tests/15-aps-api/08-namespaces.cy.ts b/e2e/cypress/tests/15-aps-api/08-namespaces.cy.ts new file mode 100644 index 000000000..8a050f7e4 --- /dev/null +++ b/e2e/cypress/tests/15-aps-api/08-namespaces.cy.ts @@ -0,0 +1,294 @@ +import HomePage from "../../pageObjects/home" +import LoginPage from "../../pageObjects/login" +let testData = require("../../fixtures/apiowner.json") +let userSession: any +let nameSpace: string + +describe('Get the user session token to pass it as authorization token to make the API call ', () => { + + const login = new LoginPage() + const home = new HomePage() + + before(() => { + cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + cy.visit(login.path) + }) + + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace).then((value) => { + userSession = value + }) + nameSpace = apiTest.namespace + }) + }) +}) + +describe('API Tests for Namespace Report', () => { + + const login = new LoginPage() + const home = new HomePage() + var response: any + + beforeEach(() => { + cy.fixture('api').as('api') + }) + + it('Prepare the Request Specification for the API', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.setHeaders(namespaces.headers) + cy.setAuthorizationToken(userSession) + }) + }) + + it('Get the resource and verify the success code in the response', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.makeAPIRequest(namespaces.endPoint + "/report", 'GET').then((response) => { + // expect(response.status).to.be.equal(200) + }) + }) + }) +}) + +describe('API Tests for Namespace List', () => { + + const login = new LoginPage() + const home = new HomePage() + var response: any + + beforeEach(() => { + cy.fixture('api').as('api') + }) + + it('Prepare the Request Specification for the API', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.setHeaders(namespaces.headers) + cy.setAuthorizationToken(userSession) + }) + }) + + it('Get the resource and verify the success code in the response', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.makeAPIRequest(namespaces.endPoint, 'GET').then((res) => { + expect(res.status).to.be.equal(200) + response = res.body + }) + }) + }) + + it('Verify that the selected Namespace is displayed in the Response list in the response', () => { + expect(response).to.be.contain(nameSpace) + }) +}) + +describe('API Tests for Namespace Activities', () => { + + const login = new LoginPage() + const home = new HomePage() + var response: any + + beforeEach(() => { + cy.fixture('api').as('api') + }) + + it('Prepare the Request Specification for the API', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.setHeaders(namespaces.headers) + cy.setAuthorizationToken(userSession) + }) + }) + + it('Get the resource and verify the success code in the response', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.makeAPIRequest(namespaces.endPoint + "/" + nameSpace + "/activity", 'GET').then((res) => { + expect(res.status).to.be.equal(200) + }) + }) + }) +}) + +describe('API Tests for Namespace Summary', () => { + + const login = new LoginPage() + const home = new HomePage() + var response: any + + beforeEach(() => { + cy.fixture('api').as('api') + cy.fixture('apiowner').as('apiowner') + }) + + it('Prepare the Request Specification for the API', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.setHeaders(namespaces.headers) + cy.setAuthorizationToken(userSession) + }) + }) + + it('Get the resource for namespace summary and verify the success code in the response', () => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.get('@api').then(({ namespaces }: any) => { + cy.makeAPIRequest(namespaces.endPoint + "/" + namespace, 'GET').then((res) => { + expect(res.status).to.be.equal(200) + response = res.body.name + }) + }) + }) + }) + + it('Verify that expected namespace summary details are display in the response', () => { + cy.get('@api').then(({ namespaces }: any) => { + // cy.compareJSONObjects(response, namespaces.activity) + }) + }) +}) + +describe('API Tests for Create Namespace', () => { + + const login = new LoginPage() + const home = new HomePage() + var response: any + + beforeEach(() => { + cy.fixture('api').as('api') + cy.fixture('apiowner').as('apiowner') + }) + + it('Prepare the Request Specification for the API', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.setHeaders(namespaces.headers) + cy.setAuthorizationToken(userSession) + }) + }) + + it('Create system generated namespace when user does not specify namespace name', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.makeAPIRequest(namespaces.endPoint, 'POST').then((res) => { + expect(res.status).to.be.equal(200) + expect(res.body.displayName).to.be.equal(null) + nameSpace = res.body.name + }) + }) + }) + + it('Verify that the generated namespace is displayed in the namespace list', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.makeAPIRequest(namespaces.endPoint, 'GET').then((res) => { + expect(res.status).to.be.equal(200) + expect(res.body).to.be.contain(nameSpace) + }) + }) + }) + + it('Create users own namespace with its description', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.setRequestBody(namespaces.userDefinedNamespace) + cy.makeAPIRequest(namespaces.endPoint, 'POST').then((res) => { + expect(res.status).to.be.equal(200) + expect(res.body.displayName).to.be.equal(namespaces.userDefinedNamespace.displayName) + nameSpace = res.body.name + }) + }) + }) + + it('Verify that the generated namespace is displayed in the namespace list', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.makeAPIRequest(namespaces.endPoint, 'GET').then((res) => { + expect(res.status).to.be.equal(200) + expect(res.body).to.be.contain(nameSpace) + }) + }) + }) +}) + +describe('API Tests for invalid namespace name', () => { + + beforeEach(() => { + cy.fixture('api').as('api') + cy.fixture('apiowner').as('apiowner') + }) + + it('Verify validation message in response when user creat namespace using invalid name', () => { + cy.get('@api').then(({ namespaces }: any) => { + cy.fixture('apiowner').then((testdata: any) => { + let namespaceName: Array = testdata.invalid_namespace + testdata + namespaceName.forEach((name: any) => { + cy.setHeaders(namespaces.headers) + // cy.setRequestBody('{"name": "' + name + '","displayName": "Test for GWA test"}') + cy.updateJsonBoby(namespaces.inValidNamespace, 'name', name).then((updatedBody) => { + cy.setRequestBody(updatedBody) + cy.makeAPIRequest(namespaces.endPoint, 'POST').then((res) => { + expect(res.status).to.be.equal(422) + expect(res.body.message).to.be.equal('Validation Failed') + }) + }) + }) + }) + }) + }) +}) + +// describe('API Tests for Deleting Namespace', () => { + +// const login = new LoginPage() +// const home = new HomePage() + + +// beforeEach(() => { +// cy.fixture('api').as('api') +// cy.fixture('apiowner').as('apiowner') +// }) + +// it('Prepare the Request Specification for the API', () => { +// cy.get('@api').then(({ namespaces }: any) => { +// cy.setHeaders(namespaces.headers) +// cy.setAuthorizationToken(userSession) +// }) +// }) + +// it('Delete the namespace and verify the Validation to prevent deleting the namespace', () => { +// cy.get('@apiowner').then(({ namespace }: any) => { +// cy.get('@api').then(({ namespaces }: any) => { +// cy.makeAPIRequest(namespaces.endPoint + "/" + namespace, 'DELETE').then((res) => { +// expect(res.status).to.be.equal(422) +// }) +// }) +// }) +// }) + +// it('Force delete the namespace and verify the success code in the response', () => { +// cy.get('@apiowner').then(({ namespace }: any) => { +// cy.get('@api').then(({ namespaces }: any) => { +// cy.makeAPIRequest(namespaces.endPoint + "/" + namespace + '?force=true', 'DELETE').then((res) => { +// expect(res.status).to.be.equal(200) +// }) +// }) +// }) +// }) + +// //need to confirm with Aidan - service returns 500 status code if there is no any namespaces +// // it('Verify that deleted namespace does not display in Get namespace list', () => { +// // let response: any +// // cy.get('@api').then(({ namespaces }: any) => { +// // cy.makeAPIRequest(namespaces.endPoint, 'GET').then((res) => { +// // // expect(res.status).to.be.equal(200) +// // response = res.body +// // expect(response).to.not.contain(nameSpace) +// // }) +// // }) +// // }) + +// after(() => { +// cy.logout() +// cy.clearLocalStorage({ log: true }) +// cy.deleteAllCookies() +// }) +// }) diff --git a/e2e/cypress/tests/16-aps-api/01-create-api.cy.ts b/e2e/cypress/tests/16-aps-api/01-create-api.cy.ts deleted file mode 100644 index 3f314bd82..000000000 --- a/e2e/cypress/tests/16-aps-api/01-create-api.cy.ts +++ /dev/null @@ -1,44 +0,0 @@ -import HomePage from '../../pageObjects/home' -import LoginPage from '../../pageObjects/login' -import Products from '../../pageObjects/products' -import ServiceAccountsPage from '../../pageObjects/serviceAccounts' - -describe('Create API Spec', () => { - const login = new LoginPage() - const home = new HomePage() - const sa = new ServiceAccountsPage() - const pd = new Products() - - before(() => { - cy.visit('/') - cy.deleteAllCookies() - cy.reload() - cy.resetState() - }) - - beforeEach(() => { - cy.preserveCookies() - cy.fixture('apiowner').as('apiowner') - cy.visit(login.path) - }) - - it('authenticates Janis (api owner)', () => { - cy.get('@apiowner').then(({ user }: any) => { - cy.login(user.credentials.username, user.credentials.password) - }) - }) - - it('creates and activates new namespace', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - home.createNamespace(apiTest.namespace) - // home.createNamespace(apiTest.delete_namespace) - }) - }) - - - after(() => { - cy.logout() - cy.clearLocalStorage({log:true}) - cy.deleteAllCookies() - }) -}) diff --git a/e2e/cypress/tests/16-aps-api/05-authorizationProfiles.cy.ts b/e2e/cypress/tests/16-aps-api/05-authorizationProfiles.cy.ts deleted file mode 100644 index a75bd2580..000000000 --- a/e2e/cypress/tests/16-aps-api/05-authorizationProfiles.cy.ts +++ /dev/null @@ -1,161 +0,0 @@ -import HomePage from "../../pageObjects/home" -import LoginPage from "../../pageObjects/login" -let userSession: any -let testData = require("../../fixtures/test_data/authorizationProfile.json") - -describe('Get the user session token', () => { - - const login = new LoginPage() - const home = new HomePage() - - before(() => { - cy.visit('/') - cy.deleteAllCookies() - cy.reload() - }) - - beforeEach(() => { - cy.preserveCookies() - cy.fixture('apiowner').as('apiowner') - cy.visit(login.path) - }) - - it('authenticates Janis (api owner) to get the user session token', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - cy.getUserSessionTokenValue(apiTest.namespace).then((value) => { - userSession = value - }) - }) - }) -}) - -testData.forEach((testCase: any) => { - describe('API Tests for Authorization Profiles', () => { - - var response: any - var actualResponse: any = {} - var expectedResponse: any = {} - - beforeEach(() => { - cy.fixture('api').as('api') - }) - - it('Prepare the Request Specification for the API', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - cy.setHeaders(authorizationProfiles.headers) - cy.setAuthorizationToken(userSession) - cy.setRequestBody(testCase.body) - }) - }) - - it('Put the resource and verify the success code in the response', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - cy.makeAPIRequest(authorizationProfiles.endPoint, 'PUT').then((response) => { - expect(response.status).to.be.equal(200) - }) - }) - }) - - it('Get the resource and verify the success code in the response', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - cy.makeAPIRequest(authorizationProfiles.endPoint, 'GET').then((res) => { - expect(res.status).to.be.equal(200) - response = res.body - }) - }) - }) - - it('Compare the values in response against the values passed in the request', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - actualResponse = response - expectedResponse = testCase.body - cy.compareJSONObjects(actualResponse, expectedResponse, true) - }) - }) - - it('Delete the authorization profile', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - cy.makeAPIRequest(authorizationProfiles.endPoint + '/' + testCase.body.name, 'DELETE').then((response) => { - expect(response.status).to.be.equal(200) - }) - }) - }) - - it('Verify that the authorization profile is deleted', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - cy.makeAPIRequest(authorizationProfiles.endPoint, 'GET').then((response) => { - expect(response.status).to.be.equal(200) - expect(response.body.length).to.be.equal(0) - }) - }) - }) - }) - - after(() => { - cy.clearLocalStorage({ log: true }) - cy.deleteAllCookies() - }) -}) - -describe('API Tests for Authorization Profiles created with inheritFrom attribute set to a valid shared Issuer', () => { - - let response: any - let actualResponse: any - let expectedResponse: any - - beforeEach(() => { - cy.fixture('api').as('api') - cy.fixture('apiowner').as('apiowner') - }) - - it('Prepare the Request Specification to create a shared IDP profile', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - cy.setHeaders(authorizationProfiles.headers) - cy.setAuthorizationToken(userSession) - cy.setRequestBody(authorizationProfiles.shared_IDP_body) - }) - }) - - it('Put the resource to create shared IDP profile and verify the success code in the response', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - cy.makeAPIRequest('ds/api/v2/namespaces/' + apiTest.namespace + '/issuers', 'PUT').then((response) => { - expect(response.status).to.be.equal(200) - }) - }) - }) - - it('Prepare the Request Specification to create a shared IDP profile using inheritFrom attribute', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - cy.setHeaders(authorizationProfiles.headers) - cy.setAuthorizationToken(userSession) - cy.setRequestBody(authorizationProfiles.shared_IDP_inheritFrom) - }) - }) - - it('Create an authorization profile using inheritFrom attribute and verify the success code in the response', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - cy.makeAPIRequest('ds/api/v2/namespaces/' + apiTest.namespace + '/issuers', 'PUT').then((response) => { - expect(response.status).to.be.equal(200) - expect(response.body.result).to.be.equal("created") - }) - }) - }) - - it('Get list of authorization profile and verify the success code in the response', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - cy.makeAPIRequest('ds/api/v2/namespaces/' + apiTest.namespace + '/issuers', 'GET').then((res) => { - expect(res.status).to.be.equal(200) - response = res.body - }) - }) - }) - - it('Compare the values in response against the values passed in the request', () => { - cy.get('@api').then(({ authorizationProfiles }: any) => { - actualResponse = response - expectedResponse = authorizationProfiles.shared_IDP_inheritFrom_expectedResponse - cy.compareJSONObjects(actualResponse, expectedResponse, true) - }) - }) - -}) \ No newline at end of file diff --git a/e2e/cypress/tests/16-aps-api/08-namespaces.cy.ts b/e2e/cypress/tests/16-aps-api/08-namespaces.cy.ts deleted file mode 100644 index 81bac4693..000000000 --- a/e2e/cypress/tests/16-aps-api/08-namespaces.cy.ts +++ /dev/null @@ -1,207 +0,0 @@ -import HomePage from "../../pageObjects/home" -import LoginPage from "../../pageObjects/login" -let userSession: any -let nameSpace: string - -describe('Get the user session token to pass it as authorization token to make the API call ', () => { - - const login = new LoginPage() - const home = new HomePage() - - before(() => { - cy.visit('/') - cy.deleteAllCookies() - cy.reload() - }) - - beforeEach(() => { - cy.preserveCookies() - cy.fixture('apiowner').as('apiowner') - cy.visit(login.path) - }) - - it('authenticates Janis (api owner) to get the user session token', () => { - cy.get('@apiowner').then(({ apiTest }: any) => { - cy.getUserSessionTokenValue(apiTest.namespace).then((value) => { - userSession = value - }) - nameSpace = apiTest.namespace - }) - }) -}) - -describe('API Tests for Namespace Report', () => { - - const login = new LoginPage() - const home = new HomePage() - var response: any - - beforeEach(() => { - cy.fixture('api').as('api') - }) - - it('Prepare the Request Specification for the API', () => { - cy.get('@api').then(({ namespaces }: any) => { - cy.setHeaders(namespaces.headers) - cy.setAuthorizationToken(userSession) - }) - }) - - it('Get the resource and verify the success code in the response', () => { - cy.get('@api').then(({ namespaces }: any) => { - cy.makeAPIRequest(namespaces.endPoint + "/report", 'GET').then((response) => { - // expect(response.status).to.be.equal(200) - }) - }) - }) -}) - -describe('API Tests for Namespace List', () => { - - const login = new LoginPage() - const home = new HomePage() - var response: any - - beforeEach(() => { - cy.fixture('api').as('api') - }) - - it('Prepare the Request Specification for the API', () => { - cy.get('@api').then(({ namespaces }: any) => { - cy.setHeaders(namespaces.headers) - cy.setAuthorizationToken(userSession) - }) - }) - - it('Get the resource and verify the success code in the response', () => { - cy.get('@api').then(({ namespaces }: any) => { - cy.makeAPIRequest(namespaces.endPoint, 'GET').then((res) => { - expect(res.status).to.be.equal(200) - response = res.body - }) - }) - }) - - it('Verify that the selected Namespace is displayed in the Response list in the response', () => { - expect(response).to.be.contain(nameSpace) - }) -}) - -describe('API Tests for Namespace Activities', () => { - - const login = new LoginPage() - const home = new HomePage() - var response: any - - beforeEach(() => { - cy.fixture('api').as('api') - }) - - it('Prepare the Request Specification for the API', () => { - cy.get('@api').then(({ namespaces }: any) => { - cy.setHeaders(namespaces.headers) - cy.setAuthorizationToken(userSession) - }) - }) - - it('Get the resource and verify the success code in the response', () => { - cy.get('@api').then(({ namespaces }: any) => { - cy.makeAPIRequest(namespaces.endPoint + "/" + nameSpace + "/activity", 'GET').then((res) => { - expect(res.status).to.be.equal(200) - }) - }) - }) -}) - -describe('API Tests for Namespace Summary', () => { - - const login = new LoginPage() - const home = new HomePage() - var response: any - - beforeEach(() => { - cy.fixture('api').as('api') - cy.fixture('apiowner').as('apiowner') - }) - - it('Prepare the Request Specification for the API', () => { - cy.get('@api').then(({ namespaces }: any) => { - cy.setHeaders(namespaces.headers) - cy.setAuthorizationToken(userSession) - }) - }) - - it('Get the resource for namespace summary and verify the success code in the response', () => { - cy.get('@apiowner').then(({ namespace }: any) => { - cy.get('@api').then(({ namespaces }: any) => { - cy.makeAPIRequest(namespaces.endPoint + "/" + namespace, 'GET').then((res) => { - expect(res.status).to.be.equal(200) - response = res.body - }) - }) - }) - }) - - it('Verify that expected namespace summary details are display in the response', () => { - cy.get('@api').then(({ namespaces }: any) => { - // cy.compareJSONObjects(response, namespaces.activity) - }) - }) -}) - -describe('API Tests for Deleting Namespace', () => { - - const login = new LoginPage() - const home = new HomePage() - - - beforeEach(() => { - cy.fixture('api').as('api') - cy.fixture('apiowner').as('apiowner') - }) - - it('Prepare the Request Specification for the API', () => { - cy.get('@api').then(({ namespaces }: any) => { - cy.setHeaders(namespaces.headers) - cy.setAuthorizationToken(userSession) - }) - }) - - it('Delete the namespace and verify the Validation to prevent deleting the namespace', () => { - cy.get('@apiowner').then(({ namespace }: any) => { - cy.get('@api').then(({ namespaces }: any) => { - cy.makeAPIRequest(namespaces.endPoint + "/" + namespace, 'DELETE').then((res) => { - expect(res.status).to.be.equal(422) - }) - }) - }) - }) - - it('Force delete the namespace and verify the success code in the response', () => { - cy.get('@apiowner').then(({ namespace }: any) => { - cy.get('@api').then(({ namespaces }: any) => { - cy.makeAPIRequest(namespaces.endPoint + "/" + namespace + '?force=true', 'DELETE').then((res) => { - expect(res.status).to.be.equal(200) - }) - }) - }) - }) - - //need to confirm with Aidan - service returns 500 status code if there is no any namespaces - // it('Verify that deleted namespace does not display in Get namespace list', () => { - // let response: any - // cy.get('@api').then(({ namespaces }: any) => { - // cy.makeAPIRequest(namespaces.endPoint, 'GET').then((res) => { - // // expect(res.status).to.be.equal(200) - // response = res.body - // expect(response).to.not.contain(nameSpace) - // }) - // }) - // }) - - after(() => { - cy.logout() - cy.clearLocalStorage({ log: true }) - cy.deleteAllCookies() - }) -}) diff --git a/e2e/cypress/tests/16-gwa-cli/01-cli-commands.ts b/e2e/cypress/tests/16-gwa-cli/01-cli-commands.ts new file mode 100644 index 000000000..fb4bd7390 --- /dev/null +++ b/e2e/cypress/tests/16-gwa-cli/01-cli-commands.ts @@ -0,0 +1,102 @@ +import LoginPage from '../../pageObjects/login' +import ApplicationPage from '../../pageObjects/applications' +import ApiDirectoryPage from '../../pageObjects/apiDirectory' +import MyAccessPage from '../../pageObjects/myAccess' +const YAML = require('yamljs'); +let userSession: any +let cli = require("../../fixtures/test_data/gwa-cli.json") +var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); +const jose = require('node-jose') + +describe('Verify CLI commands', () => { + const login = new LoginPage() + const apiDir = new ApiDirectoryPage() + const app = new ApplicationPage() + const ma = new MyAccessPage() + let namespace: string + + before(() => { + // cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + // cy.visit(login.path) + }) + + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) + }) + }) + + + it('Check gwa command to login with client ID and secret', () => { + let clientID = cli.credentials.clientID + let clientSecret = cli.credentials.clientSecret + cy.log('gwa login --host ${url} --scheme http') + cy.executeCliCommand('gwa login --client-id ' + clientID + ' --client-secret ' + clientSecret + ' --host ' + cleanedUrl + ' --scheme http').then((response) => { + expect(response.stdout).to.contain('Successfully logged in'); + }); + }) + + it('Check gwa command for login with invalid client id', () => { + let clientID = "dummy-client" + let clientSecret = cli.credentials.clientSecret + cy.executeCliCommand('gwa login --client-id ' + clientID + ' --client-secret ' + clientSecret + ' --host ' + cleanedUrl + ' --scheme http').then((response) => { + assert.equal(response.stderr, "Error: unauthorized_client\nINVALID_CREDENTIALS: Invalid client credentials") + }); + }) + + it('Check gwa command for login with invalid client secret', () => { + let clientID = cli.credentials.clientID + let clientSecret = "dummy-client-secret" + cy.executeCliCommand('gwa login --client-id ' + clientID + ' --client-secret ' + clientSecret + ' --host ' + cleanedUrl + ' --scheme http').then((response) => { + assert.equal(response.stderr, "Error: unauthorized_client\nINVALID_CREDENTIALS: Invalid client credentials") + }); + }) + + + it('Check gwa config command to set environment', () => { + var cleanedUrl = Cypress.env('BASE_URL').replace(/^http?:\/\//i, ""); + cy.executeCliCommand('gwa config set --host ' + cleanedUrl + ' --scheme http').then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('Check gwa config command to set token', () => { + cy.executeCliCommand('gwa config set --token ' + userSession).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('Check gwa command to create namespace', () => { + cy.executeCliCommand('gwa namespace create --host ' + cleanedUrl + ' --scheme http').then((response) => { + assert.isNotNaN(response.stdout) + namespace = response.stdout + }); + }) + + + it('Check gwa namespace list command and verify the created namespace in the list', () => { + cy.executeCliCommand('gwa namespace list --host ' + cleanedUrl + ' --scheme http').then((response) => { + expect(response.stdout).to.contain(namespace); + }); + }) + + it('Delete Exsting config file', () => { + cy.deleteFileInE2EFolder('gw-config.yml') + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) + +}) \ No newline at end of file diff --git a/e2e/cypress/tests/16-gwa-cli/02-cli-generate-config.ts b/e2e/cypress/tests/16-gwa-cli/02-cli-generate-config.ts new file mode 100644 index 000000000..7ced10b71 --- /dev/null +++ b/e2e/cypress/tests/16-gwa-cli/02-cli-generate-config.ts @@ -0,0 +1,101 @@ +import LoginPage from '../../pageObjects/login' +import ApplicationPage from '../../pageObjects/applications' +import ApiDirectoryPage from '../../pageObjects/apiDirectory' +import MyAccessPage from '../../pageObjects/myAccess' +import HomePage from '../../pageObjects/home'; +import Products from '../../pageObjects/products'; +const YAML = require('yamljs'); +let userSession: any +let cli = require("../../fixtures/test_data/gwa-cli.json") + +const jose = require('node-jose') + +describe('Verify CLI commands for generate/apply config', () => { + const login = new LoginPage() + const apiDir = new ApiDirectoryPage() + const app = new ApplicationPage() + const ma = new MyAccessPage() + const pd = new Products() + let namespace: string + const home = new HomePage() + + before(() => { + // cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + // cy.visit(login.path) + }) + + it('authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) + }) + }) + + it('Check gwa config command to set token', () => { + cy.executeCliCommand('gwa config set --token ' + userSession).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('Check gwa command to generate config for client credential template', () => { + cy.executeCliCommand('gwa generate-config --template client-credentials-shared-idp --service my-service --upstream https://httpbin.org --org ministry-of-health --org-unit planning-and-innovation-division').then((response) => { + assert.equal(response.stdout, "File gw-config.yml created") + }); + }) + + it('Check gwa command to apply generated config', () => { + cy.executeCliCommand('gwa apply').then((response) => { + let wordOccurrences = (response.stdout.match(/\bcreated\b/g) || []).length; + expect(wordOccurrences).to.equal(3) + namespace = response.stdout.split('\n')[0] + namespace = namespace.replace('-', '').trim() + }); + }) + + it('Check gwa command to generate config for kong httpbin template', () => { + cy.executeCliCommand('gwa generate-config --template kong-httpbin --service my-service --upstream https://httpbin.org --org ministry-of-health --org-unit planning-and-innovation-division').then((response) => { + assert.equal(response.stdout, "File gw-config.yml created") + }); + }) + + it('activates new namespace', () => { + home.useNamespace(namespace) + }) + + it('Verify that the product created through gwa command is displayed in the portal', () => { + cy.visit(pd.path) + pd.editProductEnvironment('my-service API', 'dev') + }) + + it('Verify the Authorization scope and issuer details for the product', () => { + pd.verifyAuthScope('Oauth2 Client Credentials Flow') + }) + + it('Verify the issuer details for the product', () => { + pd.verifyIssuer(namespace + ' default (test)') + }) + + it('Verify that the dataset created through GWA comand is assocuated with the product', () => { + cy.visit(pd.path) + pd.verifyDataset('my-service', 'my-service API') + }) + + it('Navigate to home path', () => { + cy.visit(login.path) + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() +}) + +}) \ No newline at end of file diff --git a/e2e/cypress/tests/17-CORS/01-create-api.cy.ts b/e2e/cypress/tests/17-CORS/01-create-api.cy.ts deleted file mode 100644 index e7a84a363..000000000 --- a/e2e/cypress/tests/17-CORS/01-create-api.cy.ts +++ /dev/null @@ -1,126 +0,0 @@ -import WebAppPage from '../../pageObjects/webApp' - -describe('Create API Spec', () => { - const webApp = new WebAppPage() - let updatedBody: any - let pluginID: string - let serviceID: string - before(() => { - cy.visit(Cypress.env('WEBAPP_URL')) - // cy.visit(Cypress.env('WEBAPP_URL')) - cy.deleteAllCookies() - cy.reload() - }) - - beforeEach(() => { - cy.fixture('apiowner').as('apiowner') - cy.fixture('cors/kong-cors-plugin-config.json').as('kong-cors-plugin-config') - cy.preserveCookies() - }) - serviceID = 'ce5d3e76-1d35-47f0-8ddf-fad087b1f969' - pluginID = '73a51e86-23ec-43c2-97fc-b96ed9f5bd90' - - // it('Create a route for the service in Kong', () => { - // cy.readFile('cypress/fixtures/state/store.json').then((store_cred) => { - // serviceID = store_cred.servicesid - // cy.get('@kong-cors-plugin-config').then(({ createRoute }: any) => { - // cy.updateJsonValue(JSON.stringify(createRoute), '$.service.id', serviceID).then((updatedValue) => { - // cy.updateKongPluginForJSONRequest(updatedValue, 'routes').then((response) => { - // expect(response.status).to.be.equal(201) - // expect(response.statusText).to.be.equal('Created') - // }) - // }) - // }) - // }) - // }) - - // it('Publish CORS plugin to kong with a valid origins value', () => { - // cy.get('@kong-cors-plugin-config').then(({ uploadCORSPlugin }: any) => { - // debugger - // cy.updateKongPluginForJSONRequest(uploadCORSPlugin, 'services/'+serviceID+'/plugins').then((response) => { - // debugger - // expect(response.status).to.be.equal(201) - // expect(response.statusText).to.be.equal('Created') - // pluginID=response.body.id - // }) - // }) - // }) - - // it('Verify for successful CORS call for valid origin value', () => { - // webApp.getStatusAfterClickOnCORS().then((statusText)=>{ - // expect(statusText).to.be.equal('Response Code: 200') - // }) - // }) - - it('Set incorrrect origin name in CORS plugin', () => { - cy.get('@kong-cors-plugin-config').then(({ uploadCORSPlugin }: any) => { - debugger - cy.updateJsonValue(JSON.stringify(uploadCORSPlugin), '$.config.origins[0]', 'https://google.com').then((updatedValue) => { - updatedBody = updatedValue - cy.updateKongPluginForJSONRequest(updatedBody, 'services/' + serviceID + '/plugins/' + pluginID, 'PUT').then((response) => { - debugger - expect(response.status).to.be.equal(200) - }) - }) - }) - }) - - it('Verify for successful CORS call for valid origin value', () => { - // cy.origin(Cypress.env('WEBAPP_URL'), () => { - // cy.wait(5000) - // cy.get('[id="corsButton"]').click({ force: true }) - // cy.wait(8000) - // cy.visit('/') - - webApp.getStatusAfterClickOnCORS().then((statusText) => { - expect(statusText).to.be.equal('Error: Failed to fetch') - }) - }) - - - // it('Allow all origin name in CORS plugin', () => { - // cy.get('@kong-cors-plugin-config').then(({ uploadCORSPlugin }: any) => { - // debugger - // cy.updateJsonValue(JSON.stringify(uploadCORSPlugin), '$.config.origins[0]', '*').then((updatedValue) => { - // updatedBody = updatedValue - // cy.updateKongPluginForJSONRequest(updatedBody, 'services/'+serviceID+'/plugins/'+pluginID, 'PUT').then((response) => { - // debugger - // expect(response.status).to.be.equal(200) - // }) - // }) - // }) - // }) - - // it('Verify for successful CORS call for valid origin value', () => { - // webApp.getStatusAfterClickOnCORS().then((statusText)=>{ - // expect(statusText).to.be.equal('Response Code: 200') - // }) - // }) - - // it('Verify for successful CORS call for invalid header', () => { - // webApp.getStatusAfterClickOnCORSForHeaders().then((statusText)=>{ - // expect(statusText).to.be.equal('Error: Failed to fetch') - // }) - // }) - - // it('Set the header name as Access-Control-Allow-Headers in CORS plugin', () => { - // cy.get('@kong-cors-plugin-config').then(({ uploadCORSPlugin }: any) => { - // debugger - // cy.updateJsonValue(JSON.stringify(uploadCORSPlugin), '$.config.headers[4]', 'X-PINGOTHER').then((updatedValue) => { - // updatedBody = updatedValue - // cy.updateKongPluginForJSONRequest(updatedBody, 'services/'+serviceID+'/plugins/'+pluginID, 'PUT').then((response) => { - // debugger - // expect(response.status).to.be.equal(200) - // }) - // }) - // }) - // }) - - // it('Verify for successful CORS call after setting the header in Access-Control-Allow-Headers list', () => { - // webApp.getStatusAfterClickOnCORSForHeaders().then((statusText)=>{ - // expect(statusText).to.be.equal('Response Code: 200') - // }) - // }) - -}) - diff --git a/e2e/cypress/tests/13-delete-application/01-delete-application-without-access.cy.ts b/e2e/cypress/tests/17-delete-application/01-delete-application-without-access.cy.ts similarity index 100% rename from e2e/cypress/tests/13-delete-application/01-delete-application-without-access.cy.ts rename to e2e/cypress/tests/17-delete-application/01-delete-application-without-access.cy.ts diff --git a/e2e/cypress/tests/13-delete-application/02-delete-application-with-pending-request.cy.ts b/e2e/cypress/tests/17-delete-application/02-delete-application-with-pending-request.cy.ts similarity index 100% rename from e2e/cypress/tests/13-delete-application/02-delete-application-with-pending-request.cy.ts rename to e2e/cypress/tests/17-delete-application/02-delete-application-with-pending-request.cy.ts diff --git a/e2e/cypress/tests/13-delete-application/03-delete-application-with-approved-request.cy.ts b/e2e/cypress/tests/17-delete-application/03-delete-application-with-approved-request.cy.ts similarity index 92% rename from e2e/cypress/tests/13-delete-application/03-delete-application-with-approved-request.cy.ts rename to e2e/cypress/tests/17-delete-application/03-delete-application-with-approved-request.cy.ts index b1c91207c..ad189ce02 100644 --- a/e2e/cypress/tests/13-delete-application/03-delete-application-with-approved-request.cy.ts +++ b/e2e/cypress/tests/17-delete-application/03-delete-application-with-approved-request.cy.ts @@ -62,8 +62,8 @@ describe('Approve Pending Request Spec', () => { cy.visit('/') cy.deleteAllCookies() cy.reload() - cy.getServiceOrRouteID('services') - cy.getServiceOrRouteID('routes') + // cy.getServiceOrRouteID('services') + // cy.getServiceOrRouteID('routes') }) beforeEach(() => { @@ -76,9 +76,11 @@ describe('Approve Pending Request Spec', () => { }) it('authenticates Mark (Access-Manager)', () => { - cy.get('@access-manager').then(({ user, namespace }: any) => { - cy.login(user.credentials.username, user.credentials.password) - home.useNamespace(namespace); + cy.get('@access-manager').then(({ user }: any) => { + cy.get('@apiowner').then(({ namespace }: any) => { + cy.login(user.credentials.username, user.credentials.password) + home.useNamespace(namespace); + }) }) }) diff --git a/e2e/cypress/tests/17-delete-application/04-delete-namespace-gwa.ts b/e2e/cypress/tests/17-delete-application/04-delete-namespace-gwa.ts new file mode 100644 index 000000000..219282ac6 --- /dev/null +++ b/e2e/cypress/tests/17-delete-application/04-delete-namespace-gwa.ts @@ -0,0 +1,82 @@ +import LoginPage from "../../pageObjects/login" + +describe('Verify namespace delete using gwa command', () => { + const login = new LoginPage() + let _namespace: string + let userSession: any + + before(() => { + // cy.visit('/') + cy.deleteAllCookies() + cy.reload() + }) + + beforeEach(() => { + cy.preserveCookies() + cy.fixture('apiowner').as('apiowner') + cy.visit(login.path) + }) + + it('Authenticates Janis (api owner) to get the user session token', () => { + cy.get('@apiowner').then(({ apiTest }: any) => { + cy.getUserSessionTokenValue(apiTest.namespace, false).then((value) => { + userSession = value + }) + }) + }) + + it('Set token using gwa config command', () => { + cy.executeCliCommand('gwa config set --token ' + userSession).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + }); + }) + + it('Create namespace using gwa cli command', () => { + cy.executeCliCommand('gwa namespace create').then((response) => { + assert.isNotNaN(response.stdout) + _namespace = response.stdout + }); + }) + + it('Check gwa namespace destroy command for soft deleting namespace', () => { + cy.executeCliCommand('gwa namespace destroy ' + _namespace).then((response) => { + expect(response.stdout).to.contain('Namespace destroyed: ' + _namespace); + }); + }) + + it('Check that deleted namespace does not display in gwa namespace list command', () => { + cy.executeCliCommand('gwa namespace list').then((response) => { + expect(response.stdout).not.to.contain(_namespace); + }); + }) + + it('Check gwa namespace destroy command for the namespace associated with services', () => { + cy.get('@apiowner').then(({ namespace }: any) => { + _namespace = namespace + cy.executeCliCommand('gwa config set --namespace ' + namespace).then((response) => { + expect(response.stdout).to.contain("Config settings saved") + cy.executeCliCommand('gwa namespace destroy').then((response) => { + expect(response.stderr).to.contain('services have been configured in this namespace'); + }); + }) + }) + }) + + it('Check gwa namespace destroy command for hard deleting namespace', () => { + cy.executeCliCommand('gwa namespace destroy --force').then((response) => { + expect(response.stdout).to.contain('Namespace destroyed: ' + _namespace); + }); + }) + + it('Check that deleted namespace does not display in gwa namespace list command', () => { + cy.executeCliCommand('gwa namespace list').then((response) => { + expect(response.stdout).not.to.contain(_namespace); + }); + }) + + after(() => { + cy.logout() + cy.clearLocalStorage({ log: true }) + cy.deleteAllCookies() + }) +}) \ No newline at end of file diff --git a/e2e/gw-config.yml b/e2e/gw-config.yml new file mode 100644 index 000000000..3b0034ccf --- /dev/null +++ b/e2e/gw-config.yml @@ -0,0 +1,36 @@ +kind: DraftDataset +name: gwa-auto-test-product +license_title: Open Government Licence - British Columbia +security_class: PUBLIC +view_audience: Public +download_audience: Public +record_publish_date: '2017-09-05' +notes: For Test Purpose +title: GWA Auto Test Product +tags: + - tag1 + - tag2 +organization: ministry-of-health +organizationUnit: planning-and-innovation-division +isInCatalog: true +isDraft: false +--- +kind: Product +name: GWA Auto Test Product +appId: 'KFH78YU956RE' +environments: + - name: dev + active: false + approval: false + flow: public + appId: '406CB7CF' +# --- +# kind: ProductEnvironment +# name: dev +# product: my-service API +# appId: '5F194B28' +# active: false +# approval: true +# flow: client-credentials +# credentialIssuer: gw-1977c default +# services: [my-service-dev] diff --git a/e2e/package-lock.json b/e2e/package-lock.json index 01b3db0b0..5df9ae714 100644 --- a/e2e/package-lock.json +++ b/e2e/package-lock.json @@ -14,6 +14,7 @@ "cypress-ntlm-auth": "^4.1.2", "cypress-xpath": "^1.6.2", "dotenv": "^10.0.0", + "js-yaml": "^4.1.0", "jsonwebtoken": "^8.5.1", "lodash": "^4.17.21", "mochawesome": "^7.1.3", @@ -83,6 +84,18 @@ "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", "integrity": "sha512-bCK/2Z4zLidyB4ReuIsvALH6w31YfAQDmXMqMx6FyfHqvBxtjC0eRumeSu4Bs3XtXwpyIywtSTrVT99BxY1f9w==" }, + "node_modules/@atomist/yaml-updater/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, "node_modules/@babel/code-frame": { "version": "7.18.6", "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.18.6.tgz", @@ -1892,6 +1905,19 @@ } } }, + "node_modules/@cypress/code-coverage/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, "node_modules/@cypress/code-coverage/node_modules/ms": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", @@ -2085,6 +2111,20 @@ "node": ">= 4" } }, + "node_modules/@eslint/eslintrc/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "peer": true, + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, "node_modules/@eslint/eslintrc/node_modules/type-fest": { "version": "0.20.2", "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", @@ -2140,6 +2180,18 @@ "node": ">=8" } }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, "node_modules/@istanbuljs/schema": { "version": "0.1.3", "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", @@ -4976,6 +5028,20 @@ "node": ">= 4" } }, + "node_modules/eslint/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "peer": true, + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, "node_modules/eslint/node_modules/semver": { "version": "7.3.7", "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", @@ -6662,17 +6728,21 @@ "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" }, "node_modules/js-yaml": { - "version": "3.14.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", - "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", "dependencies": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" + "argparse": "^2.0.1" }, "bin": { "js-yaml": "bin/js-yaml.js" } }, + "node_modules/js-yaml/node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" + }, "node_modules/jsbn": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", @@ -7404,12 +7474,6 @@ "url": "https://opencollective.com/mochajs" } }, - "node_modules/mocha/node_modules/argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "peer": true - }, "node_modules/mocha/node_modules/find-up": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", @@ -7426,18 +7490,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/mocha/node_modules/js-yaml": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", - "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", - "peer": true, - "dependencies": { - "argparse": "^2.0.1" - }, - "bin": { - "js-yaml": "bin/js-yaml.js" - } - }, "node_modules/mocha/node_modules/locate-path": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", @@ -10714,6 +10766,15 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", "integrity": "sha512-bCK/2Z4zLidyB4ReuIsvALH6w31YfAQDmXMqMx6FyfHqvBxtjC0eRumeSu4Bs3XtXwpyIywtSTrVT99BxY1f9w==" + }, + "js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } } } }, @@ -12007,6 +12068,16 @@ "ms": "2.1.2" } }, + "js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, "ms": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", @@ -12169,6 +12240,17 @@ "dev": true, "peer": true }, + "js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "peer": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, "type-fest": { "version": "0.20.2", "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", @@ -12212,6 +12294,17 @@ "get-package-type": "^0.1.0", "js-yaml": "^3.13.1", "resolve-from": "^5.0.0" + }, + "dependencies": { + "js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + } } }, "@istanbuljs/schema": { @@ -14200,6 +14293,17 @@ "dev": true, "peer": true }, + "js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "peer": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, "semver": { "version": "7.3.7", "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", @@ -15629,12 +15733,18 @@ "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" }, "js-yaml": { - "version": "3.14.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", - "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", "requires": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" + "argparse": "^2.0.1" + }, + "dependencies": { + "argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" + } } }, "jsbn": { @@ -16203,12 +16313,6 @@ "yargs-unparser": "2.0.0" }, "dependencies": { - "argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "peer": true - }, "find-up": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", @@ -16219,15 +16323,6 @@ "path-exists": "^4.0.0" } }, - "js-yaml": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", - "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", - "peer": true, - "requires": { - "argparse": "^2.0.1" - } - }, "locate-path": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", diff --git a/e2e/package.json b/e2e/package.json index 6e1eca52b..1726f71fc 100644 --- a/e2e/package.json +++ b/e2e/package.json @@ -50,6 +50,7 @@ "cypress-ntlm-auth": "^4.1.2", "cypress-xpath": "^1.6.2", "dotenv": "^10.0.0", + "js-yaml": "^4.1.0", "jsonwebtoken": "^8.5.1", "lodash": "^4.17.21", "mochawesome": "^7.1.3", diff --git a/local/feeder-init/init.sh b/local/feeder-init/init.sh index 8d3536eb5..45c62582b 100755 --- a/local/feeder-init/init.sh +++ b/local/feeder-init/init.sh @@ -15,7 +15,7 @@ while true; do curl http://feeder.localtest.me:6000/push -F yaml=@shared-idp.yaml curl http://feeder.localtest.me:6000/push -F yaml=@platform-gwa-api.yaml curl http://feeder.localtest.me:6000/push -F yaml=@organization-unit.yaml - curl http://feeder.localtest.me:6000/push -F yaml=@dataset-gwa.yaml + # curl http://feeder.localtest.me:6000/push -F yaml=@dataset-gwa.yaml curl http://feeder.localtest.me:6000/push -F yaml=@org-dataset.yaml # curl http://feeder.localtest.me:6000/push -F yaml=@product-initializer.yaml # curl http://feeder.localtest.me:6000/push -F yaml=@product-initializer-permission.yaml diff --git a/local/oauth2-proxy/oauth2-proxy-local.cfg b/local/oauth2-proxy/oauth2-proxy-local.cfg index 749890067..2c7281267 100644 --- a/local/oauth2-proxy/oauth2-proxy-local.cfg +++ b/local/oauth2-proxy/oauth2-proxy-local.cfg @@ -21,7 +21,7 @@ set_xauthrequest="true" skip_jwt_bearer_tokens="false" set_authorization_header="false" pass_authorization_header="false" -skip_auth_regex="/login|/health|/public|/docs|/redirect|/_next|/images|/devportal|/manager|/about|/maintenance|/admin/session|/ds/api|/feed/|/signout|^[/]$" +skip_auth_regex="/login|/health|/public|/docs|/redirect|/_next|/images|/devportal|/manager|/about|/maintenance|/admin/session|/ds/api|/gw/api|/feed/|/signout|^[/]$" whitelist_domains="keycloak.localtest.me:9080" upstreams=["http://apsportal.localtest.me:3000"] skip_provider_button='true' From 112518dd0ada1e0939b32fd086ec7b492a7453e3 Mon Sep 17 00:00:00 2001 From: nirajCITZ <94716060+nirajCITZ@users.noreply.github.com> Date: Mon, 18 Sep 2023 09:25:09 -0700 Subject: [PATCH 7/7] Cypress gwa cli (#912) --- .github/workflows/aps-cypress-e2e.yaml | 46 ++++++++++++-------------- e2e/Dockerfile | 1 + 2 files changed, 22 insertions(+), 25 deletions(-) diff --git a/.github/workflows/aps-cypress-e2e.yaml b/.github/workflows/aps-cypress-e2e.yaml index 918a69a7d..5b5eea26e 100644 --- a/.github/workflows/aps-cypress-e2e.yaml +++ b/.github/workflows/aps-cypress-e2e.yaml @@ -26,36 +26,32 @@ jobs: - name: Checkout Portal uses: actions/checkout@v2 - - - name: Checkout Source Repository - run: | - git clone https://github.com/bcgov/gwa-cli.git - cd gwa-cli - - name: Setup Go - uses: actions/setup-go@v2 - with: - go-version: '1.20.x' - - name: create .env file + - name: Determine Download file name + id: set_variable run: | - cd gwa-cli - echo "GWA_API_HOST=api-services-portal-feature-gwa-cli-v2.apps.silver.devops.gov.bc.ca" >> .env - echo "GWA_CLIENT_ID=gwa-cli" >> .env - - - name: Debug - run: echo ${{ github.workspace }} + echo ${{ runner.arch }} + if [ "${{ runner.arch }}" == "X64" ]; then + echo "::set-output name=my_variable::gwa_Linux_x86_64.tgz" + elif [ "${{ runner.arch }}" == "ARM64" ]; then + echo "::set-output name=my_variable::gwa_Linux_arm64.tgz" + else + echo "unsupported architecture" + fi + + - name: Download Binar + uses: robinraju/release-downloader@v1.8 + with: + repository: "bcgov/gwa-cli" + latest: true + fileName: ${{ steps.set_variable.outputs.my_variable }} + out-file-path: "${{ github.workspace }}/e2e" - - name: Build + - name: Unzip file run: | - cd gwa-cli - go clean - go build -o ${{ github.workspace }}/e2e/gwa - - - uses: actions/upload-artifact@v2 - with: - name: binary-artifact - path: ${{ github.workspace }}/e2e + cd ${{ github.workspace }}/e2e + tar xvzf ${{ steps.set_variable.outputs.my_variable }} - name: Build Docker Images run: | diff --git a/e2e/Dockerfile b/e2e/Dockerfile index a77289f4a..b253160d1 100644 --- a/e2e/Dockerfile +++ b/e2e/Dockerfile @@ -7,6 +7,7 @@ COPY cypress.config.ts /e2e COPY tsconfig.json /e2e COPY package.json /e2e COPY package-lock.json /e2e +COPY *.yml /e2e COPY entrypoint.sh /tmp COPY gwa /usr/local/bin ADD cypress /e2e/cypress