@aws-cdk/integ-tests-alpha
Version:
CDK Integration Testing Constructs
170 lines • 20.3 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.handler = handler;
exports.onTimeout = onTimeout;
exports.isComplete = isComplete;
/* eslint-disable no-console */
const assertion_1 = require("./assertion");
const http_1 = require("./http");
const sdk_1 = require("./sdk");
const types = require("./types");
async function handler(event, context) {
console.log(`Event: ${JSON.stringify({ ...event, ResponseURL: '...' })}`);
const provider = createResourceHandler(event, context);
try {
// if we are deleting the custom resource, just respond
// with 'SUCCESS' since there is nothing to do.
if (event.RequestType === 'Delete') {
await provider.respond({
status: 'SUCCESS',
reason: 'OK',
});
return;
}
const result = await provider.handle();
// if there is a `stateMachineArn` then we have already started a state machine
// execution and the rest will be handled there
if ('stateMachineArn' in event.ResourceProperties) {
console.info('Found "stateMachineArn", waiter statemachine started');
return;
}
else if ('expected' in event.ResourceProperties) {
console.info('Found "expected", testing assertions');
const actualPath = event.ResourceProperties.actualPath;
// if we are providing a path to make the assertion at, that means that we have
// flattened the response, otherwise the path to assert against in the entire response
const actual = actualPath ? result[`apiCallResponse.${actualPath}`] : result.apiCallResponse;
const assertion = new assertion_1.AssertionHandler({
...event,
ResourceProperties: {
ServiceToken: event.ServiceToken,
actual,
expected: event.ResourceProperties.expected,
},
}, context);
try {
const assertionResult = await assertion.handle();
await provider.respond({
status: 'SUCCESS',
reason: 'OK',
// return both the result of the API call _and_ the assertion results
data: {
...assertionResult,
...result,
},
});
return;
}
catch (e) {
await provider.respond({
status: 'FAILED',
reason: e.message ?? 'Internal Error',
});
return;
}
}
await provider.respond({
status: 'SUCCESS',
reason: 'OK',
data: result,
});
}
catch (e) {
await provider.respond({
status: 'FAILED',
reason: e.message ?? 'Internal Error',
});
return;
}
return;
}
/**
* Invoked by the waiter statemachine when the retry
* attempts are exhausted
*/
async function onTimeout(timeoutEvent) {
// the event payload is passed through the `errorMessage` in the state machine
// timeout event
const isCompleteRequest = JSON.parse(JSON.parse(timeoutEvent.Cause).errorMessage);
const provider = createResourceHandler(isCompleteRequest, standardContext);
await provider.respond({
status: 'FAILED',
reason: 'Operation timed out: ' + JSON.stringify(isCompleteRequest),
});
}
/**
* Invoked by the waiter statemachine when the user is waiting for a specific
* result.
*
* If the result of the assertion is not successful then it will throw an error
* which will cause the state machine to try again
*/
async function isComplete(event, context) {
console.log(`Event: ${JSON.stringify({ ...event, ResponseURL: '...' })}`);
const provider = createResourceHandler(event, context);
try {
const result = await provider.handleIsComplete();
const actualPath = event.ResourceProperties.actualPath;
if (result) {
const actual = actualPath ? result[`apiCallResponse.${actualPath}`] : result.apiCallResponse;
if ('expected' in event.ResourceProperties) {
const assertion = new assertion_1.AssertionHandler({
...event,
ResourceProperties: {
ServiceToken: event.ServiceToken,
actual,
expected: event.ResourceProperties.expected,
},
}, context);
const assertionResult = await assertion.handleIsComplete();
if (!(assertionResult?.failed)) {
await provider.respond({
status: 'SUCCESS',
reason: 'OK',
data: {
...assertionResult,
...result,
},
});
return;
}
else {
console.log(`Assertion Failed: ${JSON.stringify(assertionResult)}`);
throw new Error(JSON.stringify(event));
}
}
await provider.respond({
status: 'SUCCESS',
reason: 'OK',
data: result,
});
}
else {
console.log('No result');
throw new Error(JSON.stringify(event));
}
return;
}
catch (e) {
console.log(e);
throw new Error(JSON.stringify(event));
}
}
function createResourceHandler(event, context) {
if (event.ResourceType.startsWith(types.SDK_RESOURCE_TYPE_PREFIX)) {
return new sdk_1.AwsApiCallHandler(event, context);
}
else if (event.ResourceType.startsWith(types.ASSERT_RESOURCE_TYPE)) {
return new assertion_1.AssertionHandler(event, context);
}
else if (event.ResourceType.startsWith(types.HTTP_RESOURCE_TYPE)) {
return new http_1.HttpHandler(event, context);
}
else {
throw new Error(`Unsupported resource type "${event.ResourceType}`);
}
}
const standardContext = {
getRemainingTimeInMillis: () => 90000,
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-console */\nimport { AssertionHandler } from './assertion';\nimport { HttpHandler } from './http';\nimport { AwsApiCallHandler } from './sdk';\nimport * as types from './types';\n\nexport async function handler(event: AWSLambda.CloudFormationCustomResourceEvent, context: AWSLambda.Context) {\n  console.log(`Event: ${JSON.stringify({ ...event, ResponseURL: '...' })}`);\n  const provider = createResourceHandler(event, context);\n  try {\n    // if we are deleting the custom resource, just respond\n    // with 'SUCCESS' since there is nothing to do.\n    if (event.RequestType === 'Delete') {\n      await provider.respond({\n        status: 'SUCCESS',\n        reason: 'OK',\n      });\n      return;\n    }\n    const result = await provider.handle();\n    // if there is a `stateMachineArn` then we have already started a state machine\n    // execution and the rest will be handled there\n    if ('stateMachineArn' in event.ResourceProperties) {\n      console.info('Found \"stateMachineArn\", waiter statemachine started');\n      return;\n    } else if ('expected' in event.ResourceProperties) {\n      console.info('Found \"expected\", testing assertions');\n      const actualPath = event.ResourceProperties.actualPath;\n      // if we are providing a path to make the assertion at, that means that we have\n      // flattened the response, otherwise the path to assert against in the entire response\n      const actual = actualPath ? (result as { [key: string]: string })[`apiCallResponse.${actualPath}`] : (result as types.AwsApiCallResult).apiCallResponse;\n      const assertion = new AssertionHandler({\n        ...event,\n        ResourceProperties: {\n          ServiceToken: event.ServiceToken,\n          actual,\n          expected: event.ResourceProperties.expected,\n        },\n      }, context);\n      try {\n        const assertionResult = await assertion.handle();\n        await provider.respond({\n          status: 'SUCCESS',\n          reason: 'OK',\n          // return both the result of the API call _and_ the assertion results\n          data: {\n            ...assertionResult,\n            ...result,\n          },\n        });\n        return;\n      } catch (e: any) {\n        await provider.respond({\n          status: 'FAILED',\n          reason: e.message ?? 'Internal Error',\n        });\n        return;\n      }\n    }\n    await provider.respond({\n      status: 'SUCCESS',\n      reason: 'OK',\n      data: result,\n    });\n  } catch (e: any) {\n    await provider.respond({\n      status: 'FAILED',\n      reason: e.message ?? 'Internal Error',\n    });\n    return;\n  }\n  return;\n}\n\n/**\n * Invoked by the waiter statemachine when the retry\n * attempts are exhausted\n */\nexport async function onTimeout(timeoutEvent: any) {\n  // the event payload is passed through the `errorMessage` in the state machine\n  // timeout event\n  const isCompleteRequest = JSON.parse(JSON.parse(timeoutEvent.Cause).errorMessage);\n  const provider = createResourceHandler(isCompleteRequest, standardContext);\n  await provider.respond({\n    status: 'FAILED',\n    reason: 'Operation timed out: ' + JSON.stringify(isCompleteRequest),\n  });\n}\n\n/**\n * Invoked by the waiter statemachine when the user is waiting for a specific\n * result.\n *\n * If the result of the assertion is not successful then it will throw an error\n * which will cause the state machine to try again\n */\nexport async function isComplete(event: AWSLambda.CloudFormationCustomResourceEvent, context: AWSLambda.Context) {\n  console.log(`Event: ${JSON.stringify({ ...event, ResponseURL: '...' })}`);\n  const provider = createResourceHandler(event, context);\n  try {\n    const result = await provider.handleIsComplete();\n    const actualPath = event.ResourceProperties.actualPath;\n    if (result) {\n      const actual = actualPath ? (result as { [key: string]: string })[`apiCallResponse.${actualPath}`] : (result as types.AwsApiCallResult).apiCallResponse;\n      if ('expected' in event.ResourceProperties) {\n        const assertion = new AssertionHandler({\n          ...event,\n          ResourceProperties: {\n            ServiceToken: event.ServiceToken,\n            actual,\n            expected: event.ResourceProperties.expected,\n          },\n        }, context);\n        const assertionResult = await assertion.handleIsComplete();\n        if (!(assertionResult?.failed)) {\n          await provider.respond({\n            status: 'SUCCESS',\n            reason: 'OK',\n            data: {\n              ...assertionResult,\n              ...result,\n            },\n          });\n          return;\n        } else {\n          console.log(`Assertion Failed: ${JSON.stringify(assertionResult)}`);\n          throw new Error(JSON.stringify(event));\n        }\n      }\n      await provider.respond({\n        status: 'SUCCESS',\n        reason: 'OK',\n        data: result,\n      });\n    } else {\n      console.log('No result');\n      throw new Error(JSON.stringify(event));\n    }\n    return;\n  } catch (e) {\n    console.log(e);\n    throw new Error(JSON.stringify(event));\n  }\n}\n\nfunction createResourceHandler(event: AWSLambda.CloudFormationCustomResourceEvent, context: AWSLambda.Context) {\n  if (event.ResourceType.startsWith(types.SDK_RESOURCE_TYPE_PREFIX)) {\n    return new AwsApiCallHandler(event, context);\n  } else if (event.ResourceType.startsWith(types.ASSERT_RESOURCE_TYPE)) {\n    return new AssertionHandler(event, context);\n  } else if (event.ResourceType.startsWith(types.HTTP_RESOURCE_TYPE)) {\n    return new HttpHandler(event, context);\n  } else {\n    throw new Error(`Unsupported resource type \"${event.ResourceType}`);\n  }\n}\n\nconst standardContext: any = {\n  getRemainingTimeInMillis: () => 90000,\n};\n"]}