UNPKG

@cdklabs/cdk-ecs-codedeploy

Version:

CDK Constructs for performing ECS Deployments with CodeDeploy

91 lines 15.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.handler = handler; const client_codedeploy_1 = require("@aws-sdk/client-codedeploy"); const logger_1 = require("./logger"); /** * The lambda function called from CloudFormation for this custom resource. * * @param event * @returns attribues of the deployment that was created */ async function handler(event) { const logger = new logger_1.Logger(); logger.appendKeys({ stackEvent: event.RequestType, }); switch (event.RequestType) { case 'Create': case 'Update': { // create deployment const props = event.ResourceProperties; let autoRollbackConfiguration; if (props.autoRollbackConfigurationEnabled === 'true') { autoRollbackConfiguration = { enabled: true, events: props.autoRollbackConfigurationEvents.split(','), }; } else if (props.autoRollbackConfigurationEnabled === 'false') { autoRollbackConfiguration = { enabled: false, }; } const resp = await codedeployClient().send(new client_codedeploy_1.CreateDeploymentCommand({ applicationName: props.applicationName, deploymentConfigName: props.deploymentConfigName, deploymentGroupName: props.deploymentGroupName, autoRollbackConfiguration, description: props.description, revision: { revisionType: 'AppSpecContent', appSpecContent: { content: props.revisionAppSpecContent, }, }, })); if (!resp.deploymentId) { throw new Error('No deploymentId received from call to CreateDeployment'); } logger.appendKeys({ deploymentId: resp.deploymentId, }); logger.info('Created new deployment'); return { PhysicalResourceId: resp.deploymentId, Data: { deploymentId: resp.deploymentId, }, }; } case 'Delete': logger.appendKeys({ deploymentId: event.PhysicalResourceId, }); // cancel deployment and rollback try { const resp = await codedeployClient().send(new client_codedeploy_1.StopDeploymentCommand({ deploymentId: event.PhysicalResourceId, autoRollbackEnabled: true, })); logger.info(`Stopped deployment: ${resp.status} ${resp.statusMessage}`); } catch (e) { logger.warn('Ignoring error', { error: e }); } return { PhysicalResourceId: event.PhysicalResourceId, Data: { deploymentId: event.PhysicalResourceId, }, }; default: logger.error('Unknown stack event'); throw new Error(`Unknown request type: ${event.RequestType}`); } } function codedeployClient() { return new client_codedeploy_1.CodeDeployClient({}); } ; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"on-event.lambda.js","sourceRoot":"","sources":["../../src/ecs-deployment-provider/on-event.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { AutoRollbackConfiguration, AutoRollbackEvent, CodeDeployClient, CreateDeploymentCommand, StopDeploymentCommand } from '@aws-sdk/client-codedeploy';\nimport { Logger } from './logger';\n\n/**\n * The properties of the CodeDeploy Deployment to create\n */\ninterface DeploymentProperties {\n  description: string;\n  /**\n   * The name of an AWS CodeDeploy application.\n   */\n  applicationName: string;\n  /**\n   * The name of a deployment configuration.  If not specified, the value\n   * configured in the deployment group is used as the default.\n   * If the deployment group does not have a deployment configuration associated with it,\n   * CodeDeployDefault.OneAtATime is used by default.\n   */\n  deploymentConfigName: string;\n  /**\n   * The name of the deployment group.\n   */\n  deploymentGroupName: string;\n  /**\n   * Indicates whether a defined automatic rollback configuration is currently enabled.\n   */\n  autoRollbackConfigurationEnabled: string;\n  /**\n   * The event type or types that trigger a rollback.\n   * Valid Values: DEPLOYMENT_FAILURE | DEPLOYMENT_STOP_ON_ALARM | DEPLOYMENT_STOP_ON_REQUEST\n   */\n  autoRollbackConfigurationEvents: string;\n  /**\n   * The content of an AppSpec file for an Amazon ECS deployment.\n   * The content is formatted as JSON or YAML and stored as a RawString.\n   */\n  revisionAppSpecContent: string;\n}\n\n/**\n * The properties in the Data object returned to CloudFormation\n */\nexport interface DataAttributes {\n  /**\n   * ID of the CodeDeploy deployment\n   */\n  deploymentId: string;\n}\n\n/**\n * The request object that the custom resource lamba function receives from CloudFormation.\n */\nexport interface OnEventRequest {\n  /**\n   * The type of lifecycle event: Create, Update or Delete.\n   */\n  RequestType: string;\n  /**\n   * The template developer-chosen name (logical ID) of the custom resource in the AWS CloudFormation template.\n   */\n  LogicalResourceId: string;\n  /**\n   * This field will only be present for Update and Delete events and includes the value\n   * returned in PhysicalResourceId of the previous operation.\n   */\n  PhysicalResourceId: string;\n  /**\n   * This field contains the properties defined in the template for this custom resource.\n   */\n  ResourceProperties: DeploymentProperties;\n  /**\n   * This field will only be present for Update events and contains the resource properties\n   * that were declared previous to the update request.\n   */\n  OldResourceProperties: DeploymentProperties;\n  /**\n   * The resource type defined for this custom resource in the template.\n   * A provider may handle any number of custom resource types.\n   */\n  ResourceType: string;\n  /**\n   * A unique ID for the request.\n   */\n  RequestId: string;\n  /**\n   * The ARN that identifies the stack that contains the custom resource.\n   */\n  StackId: string;\n}\n/**\n * The response object that the custom resource lambda function returns to CloudFormation.\n */\nexport interface OnEventResponse {\n  /**\n   * The allocated/assigned physical ID of the resource. If omitted for Create events,\n   * the event's RequestId will be used. For Update, the current physical ID will be used.\n   * If a different value is returned, CloudFormation will follow with a subsequent Delete\n   * for the previous ID (resource replacement). For Delete, it will always return the current\n   * physical resource ID, and if the user returns a different one, an error will occur.\n   */\n  PhysicalResourceId?: string;\n  /**\n   * Resource attributes, which can later be retrieved through Fn::GetAtt on the custom resource object.\n   */\n  Data?: DataAttributes;\n  /**\n   * Whether to mask the output of the custom resource when retrieved by using the Fn::GetAtt function.\n   */\n  NoEcho?: boolean;\n}\n\n/**\n * The lambda function called from CloudFormation for this custom resource.\n *\n * @param event\n * @returns attribues of the deployment that was created\n */\nexport async function handler(event: OnEventRequest): Promise<OnEventResponse> {\n  const logger = new Logger();\n  logger.appendKeys({\n    stackEvent: event.RequestType,\n  });\n  switch (event.RequestType) {\n    case 'Create':\n    case 'Update': {\n      // create deployment\n      const props = event.ResourceProperties;\n      let autoRollbackConfiguration : AutoRollbackConfiguration | undefined;\n      if (props.autoRollbackConfigurationEnabled === 'true') {\n        autoRollbackConfiguration = {\n          enabled: true,\n          events: props.autoRollbackConfigurationEvents.split(',') as AutoRollbackEvent[],\n        };\n      } else if (props.autoRollbackConfigurationEnabled === 'false') {\n        autoRollbackConfiguration = {\n          enabled: false,\n        };\n      }\n      const resp = await codedeployClient().send(new CreateDeploymentCommand({\n        applicationName: props.applicationName,\n        deploymentConfigName: props.deploymentConfigName,\n        deploymentGroupName: props.deploymentGroupName,\n        autoRollbackConfiguration,\n        description: props.description,\n        revision: {\n          revisionType: 'AppSpecContent',\n          appSpecContent: {\n            content: props.revisionAppSpecContent,\n          },\n        },\n      }));\n      if (!resp.deploymentId) {\n        throw new Error('No deploymentId received from call to CreateDeployment');\n      }\n      logger.appendKeys({\n        deploymentId: resp.deploymentId,\n      });\n      logger.info('Created new deployment');\n\n      return {\n        PhysicalResourceId: resp.deploymentId,\n        Data: {\n          deploymentId: resp.deploymentId,\n        },\n      };\n    }\n    case 'Delete':\n      logger.appendKeys({\n        deploymentId: event.PhysicalResourceId,\n      });\n      // cancel deployment and rollback\n      try {\n        const resp = await codedeployClient().send(new StopDeploymentCommand({\n          deploymentId: event.PhysicalResourceId,\n          autoRollbackEnabled: true,\n        }));\n        logger.info(`Stopped deployment: ${resp.status} ${resp.statusMessage}`);\n      } catch (e) {\n        logger.warn('Ignoring error', { error: e as Error });\n      }\n\n      return {\n        PhysicalResourceId: event.PhysicalResourceId,\n        Data: {\n          deploymentId: event.PhysicalResourceId,\n        },\n      };\n    default:\n      logger.error('Unknown stack event');\n      throw new Error(`Unknown request type: ${event.RequestType}`);\n  }\n}\n\n\nfunction codedeployClient(): CodeDeployClient {\n  return new CodeDeployClient({});\n};"]}