@aws-cdk/integ-tests-alpha
Version:
CDK Integration Testing Constructs
145 lines • 13.1 kB
JavaScript
;
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"]}