@cdklabs/cdk-ecs-codedeploy
Version:
CDK Constructs for performing ECS Deployments with CodeDeploy
91 lines • 15.3 kB
JavaScript
;
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};"]}