UNPKG

@cdklabs/cdk-ecs-codedeploy

Version:

CDK Constructs for performing ECS Deployments with CodeDeploy

109 lines 18.6 kB
"use strict"; 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}"]}