UNPKG

@aws-cdk/integ-tests-alpha

Version:

CDK Integration Testing Constructs

99 lines 12.6 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.DeployAssert = void 0; const core_1 = require("aws-cdk-lib/core"); const constructs_1 = require("constructs"); const assertions_1 = require("../assertions"); const http_call_1 = require("../http-call"); const hash_1 = require("../private/hash"); const sdk_1 = require("../sdk"); const DEPLOY_ASSERT_SYMBOL = Symbol.for('@aws-cdk/integ-tests.DeployAssert'); /** * Construct that allows for registering a list of assertions * that should be performed on a construct */ class DeployAssert extends constructs_1.Construct { /** * Returns whether the construct is a DeployAssert construct */ static isDeployAssert(x) { return x !== null && typeof (x) === 'object' && DEPLOY_ASSERT_SYMBOL in x; } /** * Finds a DeployAssert construct in the given scope */ static of(construct) { const scopes = constructs_1.Node.of(constructs_1.Node.of(construct).root).findAll(); const deployAssert = scopes.find(s => DeployAssert.isDeployAssert(s)); if (!deployAssert) { throw new Error('No DeployAssert construct found in scopes'); } return deployAssert; } scope; assertionIdCounts = new Map(); constructor(scope, props) { super(scope, 'Default'); this.scope = props?.stack ?? new core_1.Stack(scope, 'DeployAssert'); Object.defineProperty(this, DEPLOY_ASSERT_SYMBOL, { value: true }); } awsApiCall(service, api, parameters, outputPaths) { let hash = ''; try { hash = (0, hash_1.md5hash)(this.scope.resolve(parameters)); } catch { } return new sdk_1.AwsApiCall(this.scope, this.uniqueAssertionId(`AwsApiCall${service}${api}${hash}`), { api, service, parameters, outputPaths, }); } httpApiCall(url, options) { let hash = ''; try { hash = (0, hash_1.md5hash)(this.scope.resolve({ url, options, })); } catch { } let append = ''; if (!core_1.Token.isUnresolved(url)) { const parsedUrl = new URL(url); append = `${parsedUrl.hostname}${parsedUrl.pathname}`; } return new http_call_1.HttpApiCall(this.scope, this.uniqueAssertionId(`HttpApiCall${append}${hash}`), { url, fetchOptions: options, }); } invokeFunction(props) { const hash = (0, hash_1.md5hash)(this.scope.resolve(props)); return new sdk_1.LambdaInvokeFunction(this.scope, this.uniqueAssertionId(`LambdaInvoke${hash}`), props); } expect(id, expected, actual) { new assertions_1.EqualsAssertion(this.scope, `EqualsAssertion${id}`, { expected, actual, }); } /** * Gets a unique logical id based on a proposed assertion id. */ uniqueAssertionId(id) { const count = this.assertionIdCounts.get(id); if (count === undefined) { // If we've never seen this id before, we'll return the id unchanged // to maintain backward compatibility. this.assertionIdCounts.set(id, 1); return id; } // Otherwise, we'll increment the counter and return a unique id. this.assertionIdCounts.set(id, count + 1); return `${id}${count}`; } } exports.DeployAssert = DeployAssert; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-assert.js","sourceRoot":"","sources":["deploy-assert.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack, Token } from 'aws-cdk-lib/core';\nimport { Construct, IConstruct, Node } from 'constructs';\nimport { IApiCall } from '../api-call-base';\nimport { EqualsAssertion } from '../assertions';\nimport { ActualResult, ExpectedResult } from '../common';\nimport { HttpApiCall as HttpApiCall } from '../http-call';\nimport { md5hash } from '../private/hash';\nimport { FetchOptions } from '../providers';\nimport { AwsApiCall, LambdaInvokeFunction, LambdaInvokeFunctionProps } from '../sdk';\nimport { IDeployAssert } from '../types';\n\nconst DEPLOY_ASSERT_SYMBOL = Symbol.for('@aws-cdk/integ-tests.DeployAssert');\n\n/**\n * Options for DeployAssert\n */\nexport interface DeployAssertProps {\n\n  /**\n   * A stack to use for assertions\n   *\n   * @default - a stack is created for you\n   */\n  readonly stack?: Stack;\n}\n\n/**\n * Construct that allows for registering a list of assertions\n * that should be performed on a construct\n */\nexport class DeployAssert extends Construct implements IDeployAssert {\n  /**\n   * Returns whether the construct is a DeployAssert construct\n   */\n  public static isDeployAssert(x: any): x is DeployAssert {\n    return x !== null && typeof (x) === 'object' && DEPLOY_ASSERT_SYMBOL in x;\n  }\n\n  /**\n   * Finds a DeployAssert construct in the given scope\n   */\n  public static of(construct: IConstruct): DeployAssert {\n    const scopes = Node.of(Node.of(construct).root).findAll();\n    const deployAssert = scopes.find(s => DeployAssert.isDeployAssert(s));\n    if (!deployAssert) {\n      throw new Error('No DeployAssert construct found in scopes');\n    }\n    return deployAssert as DeployAssert;\n  }\n\n  public scope: Stack;\n  private assertionIdCounts = new Map<string, number>();\n\n  constructor(scope: Construct, props?: DeployAssertProps) {\n    super(scope, 'Default');\n\n    this.scope = props?.stack ?? new Stack(scope, 'DeployAssert');\n\n    Object.defineProperty(this, DEPLOY_ASSERT_SYMBOL, { value: true });\n  }\n\n  public awsApiCall(service: string, api: string, parameters?: any, outputPaths?: string[]): IApiCall {\n    let hash = '';\n    try {\n      hash = md5hash(this.scope.resolve(parameters));\n    } catch {}\n\n    return new AwsApiCall(this.scope, this.uniqueAssertionId(`AwsApiCall${service}${api}${hash}`), {\n      api,\n      service,\n      parameters,\n      outputPaths,\n    });\n  }\n\n  public httpApiCall(url: string, options?: FetchOptions): IApiCall {\n    let hash = '';\n    try {\n      hash = md5hash(this.scope.resolve({\n        url,\n        options,\n      }));\n    } catch {}\n\n    let append = '';\n    if (!Token.isUnresolved(url)) {\n      const parsedUrl = new URL(url);\n      append = `${parsedUrl.hostname}${parsedUrl.pathname}`;\n    }\n    return new HttpApiCall(this.scope, this.uniqueAssertionId(`HttpApiCall${append}${hash}`), {\n      url,\n      fetchOptions: options,\n    });\n  }\n\n  public invokeFunction(props: LambdaInvokeFunctionProps): IApiCall {\n    const hash = md5hash(this.scope.resolve(props));\n    return new LambdaInvokeFunction(this.scope, this.uniqueAssertionId(`LambdaInvoke${hash}`), props);\n  }\n\n  public expect(id: string, expected: ExpectedResult, actual: ActualResult): void {\n    new EqualsAssertion(this.scope, `EqualsAssertion${id}`, {\n      expected,\n      actual,\n    });\n  }\n\n  /**\n   * Gets a unique logical id based on a proposed assertion id.\n   */\n  private uniqueAssertionId(id: string): string {\n    const count = this.assertionIdCounts.get(id);\n\n    if (count === undefined) {\n      // If we've never seen this id before, we'll return the id unchanged\n      // to maintain backward compatibility.\n      this.assertionIdCounts.set(id, 1);\n      return id;\n    }\n\n    // Otherwise, we'll increment the counter and return a unique id.\n    this.assertionIdCounts.set(id, count + 1);\n    return `${id}${count}`;\n  }\n}\n"]}