UNPKG

@aws-cdk/integ-tests-alpha

Version:

CDK Integration Testing Constructs

145 lines 13.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.AssertionHandler = void 0; /* eslint-disable no-console */ // eslint-disable-next-line import/no-extraneous-dependencies const helpers_internal_1 = require("aws-cdk-lib/assertions/lib/helpers-internal"); const base_1 = require("./base"); class AssertionHandler extends base_1.CustomResourceHandler { async processEvent(request) { let actual = decodeCall(request.actual); const expected = decodeCall(request.expected); let result; const matcher = new MatchCreator(expected).getMatcher(); console.log(`Testing equality between ${JSON.stringify(request.actual)} and ${JSON.stringify(request.expected)}`); const matchResult = matcher.test(actual); matchResult.finished(); if (matchResult.hasFailed()) { result = { failed: true, assertion: JSON.stringify({ status: 'fail', message: matchResult.renderMismatch(), }), }; if (request.failDeployment) { throw new Error(result.assertion); } } else { result = { assertion: JSON.stringify({ status: 'success', }), }; } return result; } } exports.AssertionHandler = AssertionHandler; class MatchCreator { parsedObj; constructor(obj) { this.parsedObj = { matcher: obj, }; } /** * Return a Matcher that can be tested against the actual results. * This will convert the encoded matchers into their corresponding * assertions matcher. * * For example: * * ExpectedResult.objectLike({ * Messages: [{ * Body: Match.objectLike({ * Elements: Match.arrayWith([{ Asdf: 3 }]), * Payload: Match.serializedJson({ key: 'value' }), * }), * }], * }); * * Will be encoded as: * { * $ObjectLike: { * Messages: [{ * Body: { * $ObjectLike: { * Elements: { * $ArrayWith: [{ Asdf: 3 }], * }, * Payload: { * $SerializedJson: { key: 'value' } * } * }, * }, * }], * }, * } * * Which can then be parsed by this function. For each key (recursively) * the parser will check if the value has one of the encoded matchers as a key * and if so, it will set the value as the Matcher. So, * * { * Body: { * $ObjectLike: { * Elements: { * $ArrayWith: [{ Asdf: 3 }], * }, * Payload: { * $SerializedJson: { key: 'value' } * } * }, * }, * } * * Will be converted to * { * Body: Match.objectLike({ * Elements: Match.arrayWith([{ Asdf: 3 }]), * Payload: Match.serializedJson({ key: 'value' }), * }), * } */ getMatcher() { try { const final = JSON.parse(JSON.stringify(this.parsedObj), function (_k, v) { const nested = Object.keys(v)[0]; switch (nested) { case '$ArrayWith': return helpers_internal_1.Match.arrayWith(v[nested]); case '$ObjectLike': return helpers_internal_1.Match.objectLike(v[nested]); case '$StringLike': return helpers_internal_1.Match.stringLikeRegexp(v[nested]); case '$SerializedJson': return helpers_internal_1.Match.serializedJson(v[nested]); default: return v; } }); if (helpers_internal_1.Matcher.isMatcher(final.matcher)) { return final.matcher; } return helpers_internal_1.Match.exact(final.matcher); } catch { return helpers_internal_1.Match.exact(this.parsedObj.matcher); } } } function decodeCall(call) { if (!call) { return undefined; } try { const parsed = JSON.parse(call); return parsed; } catch { return call; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assertion.js","sourceRoot":"","sources":["assertion.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-console */\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { Match, Matcher } from 'aws-cdk-lib/assertions/lib/helpers-internal';\nimport { CustomResourceHandler } from './base';\nimport { AssertionResult, AssertionRequest } from './types';\n\nexport class AssertionHandler extends CustomResourceHandler<AssertionRequest, AssertionResult> {\n  protected async processEvent(request: AssertionRequest): Promise<AssertionResult | undefined> {\n    let actual = decodeCall(request.actual);\n    const expected = decodeCall(request.expected);\n    let result: AssertionResult;\n    const matcher = new MatchCreator(expected).getMatcher();\n    console.log(`Testing equality between ${JSON.stringify(request.actual)} and ${JSON.stringify(request.expected)}`);\n\n    const matchResult = matcher.test(actual);\n    matchResult.finished();\n    if (matchResult.hasFailed()) {\n      result = {\n        failed: true,\n        assertion: JSON.stringify({\n          status: 'fail',\n          message: matchResult.renderMismatch(),\n        }),\n      };\n      if (request.failDeployment) {\n        throw new Error(result.assertion);\n      }\n    } else {\n      result = {\n        assertion: JSON.stringify({\n          status: 'success',\n        }),\n      };\n    }\n\n    return result;\n  }\n}\n\nclass MatchCreator {\n  private readonly parsedObj: { [key: string]: any };\n  constructor(obj: { [key: string]: any }) {\n    this.parsedObj = {\n      matcher: obj,\n    };\n  }\n\n  /**\n   * Return a Matcher that can be tested against the actual results.\n   * This will convert the encoded matchers into their corresponding\n   * assertions matcher.\n   *\n   * For example:\n   *\n   * ExpectedResult.objectLike({\n   *   Messages: [{\n   *     Body: Match.objectLike({\n   *       Elements: Match.arrayWith([{ Asdf: 3 }]),\n   *       Payload: Match.serializedJson({ key: 'value' }),\n   *     }),\n   *   }],\n   * });\n   *\n   * Will be encoded as:\n   * {\n   *   $ObjectLike: {\n   *     Messages: [{\n   *       Body: {\n   *         $ObjectLike: {\n   *           Elements: {\n   *             $ArrayWith: [{ Asdf: 3 }],\n   *           },\n   *           Payload: {\n   *             $SerializedJson: { key: 'value' }\n   *           }\n   *         },\n   *       },\n   *     }],\n   *   },\n   * }\n   *\n   * Which can then be parsed by this function. For each key (recursively)\n   * the parser will check if the value has one of the encoded matchers as a key\n   * and if so, it will set the value as the Matcher. So,\n   *\n   * {\n   *   Body: {\n   *     $ObjectLike: {\n   *       Elements: {\n   *         $ArrayWith: [{ Asdf: 3 }],\n   *       },\n   *       Payload: {\n   *         $SerializedJson: { key: 'value' }\n   *       }\n   *     },\n   *   },\n   * }\n   *\n   * Will be converted to\n   * {\n   *   Body: Match.objectLike({\n   *     Elements: Match.arrayWith([{ Asdf: 3 }]),\n   *     Payload: Match.serializedJson({ key: 'value' }),\n   *   }),\n   * }\n   */\n  public getMatcher(): Matcher {\n    try {\n      const final = JSON.parse(JSON.stringify(this.parsedObj), function(_k, v) {\n        const nested = Object.keys(v)[0];\n        switch (nested) {\n          case '$ArrayWith':\n            return Match.arrayWith(v[nested]);\n          case '$ObjectLike':\n            return Match.objectLike(v[nested]);\n          case '$StringLike':\n            return Match.stringLikeRegexp(v[nested]);\n          case '$SerializedJson':\n            return Match.serializedJson(v[nested]);\n          default:\n            return v;\n        }\n      });\n      if (Matcher.isMatcher(final.matcher)) {\n        return final.matcher;\n      }\n      return Match.exact(final.matcher);\n    } catch {\n      return Match.exact(this.parsedObj.matcher);\n    }\n  }\n}\n\nfunction decodeCall(call?: string) {\n  if (!call) { return undefined; }\n  try {\n    const parsed = JSON.parse(call);\n    return parsed;\n  } catch {\n    return call;\n  }\n}\n"]}