@aws-cdk/integ-tests-alpha
Version:
CDK Integration Testing Constructs
137 lines • 14.4 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.CustomResourceHandler = void 0;
/* eslint-disable no-console */
const https = require("https");
const url = require("url");
const client_sfn_1 = require("@aws-sdk/client-sfn");
class CustomResourceHandler {
event;
context;
physicalResourceId;
timeout;
timedOut = false;
constructor(event, context) {
this.event = event;
this.context = context;
this.timeout = setTimeout(async () => {
await this.respond({
status: 'FAILED',
reason: 'Lambda Function Timeout',
data: this.context.logStreamName,
});
this.timedOut = true;
}, context.getRemainingTimeInMillis() - 1200);
this.event = event;
this.physicalResourceId = extractPhysicalResourceId(event);
}
/**
* Handles executing the custom resource event. If `stateMachineArn` is present
* in the props then trigger the waiter statemachine
*/
async handle() {
try {
if ('stateMachineArn' in this.event.ResourceProperties) {
const req = {
stateMachineArn: this.event.ResourceProperties.stateMachineArn,
name: this.event.RequestId,
input: JSON.stringify(this.event),
};
await this.startExecution(req);
return;
}
else {
const response = await this.processEvent(this.event.ResourceProperties);
return response;
}
}
catch (e) {
console.log(e);
throw e;
}
finally {
clearTimeout(this.timeout);
}
}
/**
* Handle async requests from the waiter state machine
*/
async handleIsComplete() {
try {
const result = await this.processEvent(this.event.ResourceProperties);
return result;
}
catch (e) {
console.log(e);
return;
}
finally {
clearTimeout(this.timeout);
}
}
/**
* Start a step function state machine which will wait for the request
* to be successful.
*/
async startExecution(req) {
try {
const sfn = new client_sfn_1.SFN({});
await sfn.startExecution(req);
}
finally {
clearTimeout(this.timeout);
}
}
respond(response) {
if (this.timedOut) {
return;
}
const cfResponse = {
Status: response.status,
Reason: response.reason,
PhysicalResourceId: this.physicalResourceId,
StackId: this.event.StackId,
RequestId: this.event.RequestId,
LogicalResourceId: this.event.LogicalResourceId,
NoEcho: false,
Data: response.data,
};
const responseBody = JSON.stringify(cfResponse);
console.log('Responding to CloudFormation', responseBody);
const parsedUrl = url.parse(this.event.ResponseURL);
const requestOptions = {
hostname: parsedUrl.hostname,
path: parsedUrl.path,
method: 'PUT',
headers: {
'content-type': '',
'content-length': Buffer.byteLength(responseBody, 'utf8'),
},
};
return new Promise((resolve, reject) => {
try {
const request = https.request(requestOptions, resolve);
request.on('error', reject);
request.write(responseBody);
request.end();
}
catch (e) {
reject(e);
}
finally {
clearTimeout(this.timeout);
}
});
}
}
exports.CustomResourceHandler = CustomResourceHandler;
function extractPhysicalResourceId(event) {
switch (event.RequestType) {
case 'Create':
return event.LogicalResourceId;
case 'Update':
case 'Delete':
return event.PhysicalResourceId;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["base.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-console */\nimport * as https from 'https';\nimport * as url from 'url';\nimport { SFN, StartExecutionInput } from '@aws-sdk/client-sfn';\n\ninterface HandlerResponse {\n  readonly status: 'SUCCESS' | 'FAILED';\n  readonly reason: 'OK' | string;\n  readonly data?: any;\n}\n\nexport abstract class CustomResourceHandler<Request extends object, Response extends object> {\n  public readonly physicalResourceId: string;\n  private readonly timeout: NodeJS.Timeout;\n  private timedOut = false;\n\n  constructor(protected readonly event: AWSLambda.CloudFormationCustomResourceEvent, protected readonly context: AWSLambda.Context) {\n    this.timeout = setTimeout(async () => {\n      await this.respond({\n        status: 'FAILED',\n        reason: 'Lambda Function Timeout',\n        data: this.context.logStreamName,\n      });\n      this.timedOut = true;\n    }, context.getRemainingTimeInMillis() - 1200);\n    this.event = event;\n    this.physicalResourceId = extractPhysicalResourceId(event);\n  }\n\n  /**\n   * Handles executing the custom resource event. If `stateMachineArn` is present\n   * in the props then trigger the waiter statemachine\n   */\n  public async handle(): Promise<Response | undefined> {\n    try {\n      if ('stateMachineArn' in this.event.ResourceProperties) {\n        const req: StartExecutionInput = {\n          stateMachineArn: this.event.ResourceProperties.stateMachineArn,\n          name: this.event.RequestId,\n          input: JSON.stringify(this.event),\n        };\n        await this.startExecution(req);\n        return;\n      } else {\n        const response = await this.processEvent(this.event.ResourceProperties as unknown as Request);\n        return response;\n      }\n    } catch (e) {\n      console.log(e);\n      throw e;\n    } finally {\n      clearTimeout(this.timeout);\n    }\n  }\n\n  /**\n   * Handle async requests from the waiter state machine\n   */\n  public async handleIsComplete(): Promise<Response | undefined> {\n    try {\n      const result = await this.processEvent(this.event.ResourceProperties as unknown as Request);\n      return result;\n    } catch (e) {\n      console.log(e);\n      return;\n    } finally {\n      clearTimeout(this.timeout);\n    }\n  }\n\n  /**\n   * Start a step function state machine which will wait for the request\n   * to be successful.\n   */\n  private async startExecution(req: StartExecutionInput): Promise<void> {\n    try {\n      const sfn = new SFN({});\n      await sfn.startExecution(req);\n    } finally {\n      clearTimeout(this.timeout);\n    }\n  }\n\n  protected abstract processEvent(request: Request): Promise<Response | undefined>;\n\n  public respond(response: HandlerResponse) {\n    if (this.timedOut) {\n      return;\n    }\n    const cfResponse: AWSLambda.CloudFormationCustomResourceResponse = {\n      Status: response.status,\n      Reason: response.reason,\n      PhysicalResourceId: this.physicalResourceId,\n      StackId: this.event.StackId,\n      RequestId: this.event.RequestId,\n      LogicalResourceId: this.event.LogicalResourceId,\n      NoEcho: false,\n      Data: response.data,\n    };\n    const responseBody = JSON.stringify(cfResponse);\n\n    console.log('Responding to CloudFormation', responseBody);\n\n    const parsedUrl = url.parse(this.event.ResponseURL);\n    const requestOptions = {\n      hostname: parsedUrl.hostname,\n      path: parsedUrl.path,\n      method: 'PUT',\n      headers: {\n        'content-type': '',\n        'content-length': Buffer.byteLength(responseBody, 'utf8'),\n      },\n    };\n\n    return new Promise((resolve, reject) => {\n      try {\n        const request = https.request(requestOptions, resolve);\n        request.on('error', reject);\n        request.write(responseBody);\n        request.end();\n      } catch (e) {\n        reject(e);\n      } finally {\n        clearTimeout(this.timeout);\n      }\n    });\n  }\n}\n\nfunction extractPhysicalResourceId(event: AWSLambda.CloudFormationCustomResourceEvent): string {\n  switch (event.RequestType) {\n    case 'Create':\n      return event.LogicalResourceId;\n    case 'Update':\n    case 'Delete':\n      return event.PhysicalResourceId;\n  }\n}\n"]}