@cdklabs/cdk-ecs-codedeploy
Version:
CDK Constructs for performing ECS Deployments with CodeDeploy
109 lines • 18.6 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.DeploymentStatus = void 0;
exports.handler = handler;
const client_codedeploy_1 = require("@aws-sdk/client-codedeploy");
const logger_1 = require("./logger");
var DeploymentStatus;
(function (DeploymentStatus) {
DeploymentStatus["CREATED"] = "Created";
DeploymentStatus["QUEUED"] = "Queued";
DeploymentStatus["IN_PROGRESS"] = "InProgress";
DeploymentStatus["BAKING"] = "Baking";
DeploymentStatus["SUCCEEDED"] = "Succeeded";
DeploymentStatus["FAILED"] = "Failed";
DeploymentStatus["STOPPED"] = "Stopped";
DeploymentStatus["READY"] = "Ready";
})(DeploymentStatus || (exports.DeploymentStatus = DeploymentStatus = {}));
/**
* The lambda function called from CloudFormation for this custom resource.
*
* @param event
* @returns whether the deployment is complete
*/
async function handler(event) {
const logger = new logger_1.Logger();
const codedeployClient = new client_codedeploy_1.CodeDeployClient({});
try {
const resp = await codedeployClient.send(new client_codedeploy_1.GetDeploymentCommand({ deploymentId: event.PhysicalResourceId }));
let rollbackResp = {};
if (resp.deploymentInfo?.rollbackInfo?.rollbackDeploymentId) {
rollbackResp = await codedeployClient.send(new client_codedeploy_1.GetDeploymentCommand({ deploymentId: resp.deploymentInfo?.rollbackInfo?.rollbackDeploymentId }));
}
logger.appendKeys({
stackEvent: event.RequestType,
deploymentId: event.PhysicalResourceId,
deploymentStatus: resp.deploymentInfo?.status,
rollbackStatus: rollbackResp?.deploymentInfo?.status,
});
logger.info('Checking deployment');
// check if deployment id is complete
switch (event.RequestType) {
case 'Create':
case 'Update':
switch (resp.deploymentInfo?.status) {
case DeploymentStatus.SUCCEEDED:
logger.info('Deployment finished successfully', { complete: true });
return { IsComplete: true };
case DeploymentStatus.FAILED:
case DeploymentStatus.STOPPED:
if (rollbackResp.deploymentInfo?.status) {
if (rollbackResp.deploymentInfo?.status == DeploymentStatus.SUCCEEDED ||
rollbackResp.deploymentInfo?.status == DeploymentStatus.FAILED ||
rollbackResp.deploymentInfo?.status == DeploymentStatus.STOPPED) {
const errInfo = resp.deploymentInfo.errorInformation;
const error = new Error(`Deployment ${resp.deploymentInfo.status}: [${errInfo?.code}] ${errInfo?.message}`);
logger.error('Deployment failed', { complete: true, error });
throw error;
}
logger.info('Waiting for final status from a rollback', { complete: false });
return { IsComplete: false }; // waiting for final status from rollback
}
else {
const errInfo = resp.deploymentInfo.errorInformation;
const error = new Error(`Deployment ${resp.deploymentInfo.status}: [${errInfo?.code}] ${errInfo?.message}`);
logger.error('No rollback to wait for', { complete: true, error });
throw error;
}
default:
logger.info('Waiting for final status from deployment', { complete: false });
return { IsComplete: false };
}
case 'Delete':
switch (resp.deploymentInfo?.status) {
case DeploymentStatus.SUCCEEDED:
logger.info('Deployment finished successfully - nothing to delete', { complete: true });
return { IsComplete: true };
case DeploymentStatus.FAILED:
case DeploymentStatus.STOPPED:
if (rollbackResp.deploymentInfo?.status) {
if (rollbackResp.deploymentInfo?.status == DeploymentStatus.SUCCEEDED ||
rollbackResp.deploymentInfo?.status == DeploymentStatus.FAILED ||
rollbackResp.deploymentInfo?.status == DeploymentStatus.STOPPED) {
logger.info('Rollback in final status', { complete: true });
return { IsComplete: true }; // rollback finished, we're deleted
}
logger.info('Waiting for final status from a rollback', { complete: false });
return { IsComplete: false }; // waiting for rollback
}
logger.info('No rollback to wait for', { complete: true });
return { IsComplete: true };
default:
logger.info('Waiting for final status from deployment', { complete: false });
return { IsComplete: false };
}
default:
logger.error('Unknown request type');
throw new Error(`Unknown request type: ${event.RequestType}`);
}
}
catch (e) {
logger.error('Unable to determine deployment status', { error: e });
if (event.RequestType === 'Delete') {
logger.warn('Ignoring error - nothing to do', { complete: true });
return { IsComplete: true };
}
throw e;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"is-complete.lambda.js","sourceRoot":"","sources":["../../src/ecs-deployment-provider/is-complete.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { CodeDeployClient, GetDeploymentCommand, GetDeploymentOutput } from '@aws-sdk/client-codedeploy';\nimport { Logger } from './logger';\n\nexport enum DeploymentStatus {\n  CREATED = 'Created',\n  QUEUED = 'Queued',\n  IN_PROGRESS = 'InProgress',\n  BAKING = 'Baking',\n  SUCCEEDED = 'Succeeded',\n  FAILED = 'Failed',\n  STOPPED = 'Stopped',\n  READY = 'Ready',\n}\n\n/**\n * The request object that the custom resource lamba function receives from CloudFormation.\n */\nexport interface IsCompleteRequest {\n  /**\n   * The type of CloudFormation request (e.g. 'Create', 'Update', or 'Delete')\n   */\n  RequestType: string;\n\n  /**\n   * The physical resource id.\n   */\n  PhysicalResourceId: string;\n}\n\n/**\n * The response object that the custom resource lambda function returns to CloudFormation.\n */\nexport interface IsCompleteResponse {\n  /**\n   * True if and only if the deployment is in a final state.\n   */\n  IsComplete: boolean;\n}\n\n/**\n * The lambda function called from CloudFormation for this custom resource.\n *\n * @param event\n * @returns whether the deployment is complete\n */\nexport async function handler(event: IsCompleteRequest): Promise<IsCompleteResponse> {\n  const logger = new Logger();\n  const codedeployClient = new CodeDeployClient({});\n  try {\n    const resp = await codedeployClient.send(new GetDeploymentCommand({ deploymentId: event.PhysicalResourceId }));\n    let rollbackResp: GetDeploymentOutput = {};\n    if (resp.deploymentInfo?.rollbackInfo?.rollbackDeploymentId) {\n      rollbackResp = await codedeployClient.send(new GetDeploymentCommand({ deploymentId: resp.deploymentInfo?.rollbackInfo?.rollbackDeploymentId }));\n    }\n    logger.appendKeys({\n      stackEvent: event.RequestType,\n      deploymentId: event.PhysicalResourceId,\n      deploymentStatus: resp.deploymentInfo?.status,\n      rollbackStatus: rollbackResp?.deploymentInfo?.status,\n    });\n    logger.info('Checking deployment');\n\n    // check if deployment id is complete\n    switch (event.RequestType) {\n      case 'Create':\n      case 'Update':\n        switch (resp.deploymentInfo?.status) {\n          case DeploymentStatus.SUCCEEDED:\n            logger.info('Deployment finished successfully', { complete: true });\n\n            return { IsComplete: true };\n          case DeploymentStatus.FAILED:\n          case DeploymentStatus.STOPPED:\n            if (rollbackResp.deploymentInfo?.status) {\n              if (rollbackResp.deploymentInfo?.status == DeploymentStatus.SUCCEEDED ||\n                rollbackResp.deploymentInfo?.status == DeploymentStatus.FAILED ||\n                rollbackResp.deploymentInfo?.status == DeploymentStatus.STOPPED) {\n                const errInfo = resp.deploymentInfo.errorInformation;\n                const error = new Error(`Deployment ${resp.deploymentInfo.status}: [${errInfo?.code}] ${errInfo?.message}`);\n                logger.error('Deployment failed', { complete: true, error });\n                throw error;\n              }\n              logger.info('Waiting for final status from a rollback', { complete: false });\n\n              return { IsComplete: false }; // waiting for final status from rollback\n            } else {\n              const errInfo = resp.deploymentInfo.errorInformation;\n              const error = new Error(`Deployment ${resp.deploymentInfo.status}: [${errInfo?.code}] ${errInfo?.message}`);\n              logger.error('No rollback to wait for', { complete: true, error });\n              throw error;\n            }\n          default:\n            logger.info('Waiting for final status from deployment', { complete: false });\n\n            return { IsComplete: false };\n        }\n      case 'Delete':\n        switch (resp.deploymentInfo?.status) {\n          case DeploymentStatus.SUCCEEDED:\n            logger.info('Deployment finished successfully - nothing to delete', { complete: true });\n\n            return { IsComplete: true };\n          case DeploymentStatus.FAILED:\n          case DeploymentStatus.STOPPED:\n            if (rollbackResp.deploymentInfo?.status) {\n              if (rollbackResp.deploymentInfo?.status == DeploymentStatus.SUCCEEDED ||\n                rollbackResp.deploymentInfo?.status == DeploymentStatus.FAILED ||\n                rollbackResp.deploymentInfo?.status == DeploymentStatus.STOPPED) {\n                logger.info('Rollback in final status', { complete: true });\n\n                return { IsComplete: true }; // rollback finished, we're deleted\n              }\n              logger.info('Waiting for final status from a rollback', { complete: false });\n\n              return { IsComplete: false }; // waiting for rollback\n            }\n            logger.info('No rollback to wait for', { complete: true });\n\n            return { IsComplete: true };\n          default:\n            logger.info('Waiting for final status from deployment', { complete: false });\n\n            return { IsComplete: false };\n        }\n      default:\n        logger.error('Unknown request type');\n        throw new Error(`Unknown request type: ${event.RequestType}`);\n    }\n  } catch (e) {\n    logger.error('Unable to determine deployment status', { error: e as Error });\n    if (event.RequestType === 'Delete') {\n      logger.warn('Ignoring error - nothing to do', { complete: true });\n\n      return { IsComplete: true };\n    }\n    throw e;\n  }\n}"]}