UNPKG

20.2 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.Provider = void 0;
5const jsiiDeprecationWarnings = require("../../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const path = require("path");
8const lambda = require("@aws-cdk/aws-lambda");
9const core_1 = require("@aws-cdk/core");
10const consts = require("./runtime/consts");
11const util_1 = require("./util");
12const waiter_state_machine_1 = require("./waiter-state-machine");
13// keep this import separate from other imports to reduce chance for merge conflicts with v2-main",
14// eslint-disable-next-line
15const core_2 = require("@aws-cdk/core");
16const RUNTIME_HANDLER_PATH = path.join(__dirname, 'runtime');
17const FRAMEWORK_HANDLER_TIMEOUT = core_1.Duration.minutes(15); // keep it simple for now
18/**
19 * Defines an AWS CloudFormation custom resource provider.
20 */
21class Provider extends core_2.Construct {
22 constructor(scope, id, props) {
23 super(scope, id);
24 try {
25 jsiiDeprecationWarnings._aws_cdk_custom_resources_ProviderProps(props);
26 }
27 catch (error) {
28 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
29 Error.captureStackTrace(error, Provider);
30 }
31 throw error;
32 }
33 if (!props.isCompleteHandler && (props.queryInterval || props.totalTimeout)) {
34 throw new Error('"queryInterval" and "totalTimeout" can only be configured if "isCompleteHandler" is specified. '
35 + 'Otherwise, they have no meaning');
36 }
37 this.onEventHandler = props.onEventHandler;
38 this.isCompleteHandler = props.isCompleteHandler;
39 this.logRetention = props.logRetention;
40 this.vpc = props.vpc;
41 this.vpcSubnets = props.vpcSubnets;
42 this.securityGroups = props.securityGroups;
43 this.role = props.role;
44 const onEventFunction = this.createFunction(consts.FRAMEWORK_ON_EVENT_HANDLER_NAME, props.providerFunctionName);
45 if (this.isCompleteHandler) {
46 const isCompleteFunction = this.createFunction(consts.FRAMEWORK_IS_COMPLETE_HANDLER_NAME);
47 const timeoutFunction = this.createFunction(consts.FRAMEWORK_ON_TIMEOUT_HANDLER_NAME);
48 const retry = util_1.calculateRetryPolicy(props);
49 const waiterStateMachine = new waiter_state_machine_1.WaiterStateMachine(this, 'waiter-state-machine', {
50 isCompleteHandler: isCompleteFunction,
51 timeoutHandler: timeoutFunction,
52 backoffRate: retry.backoffRate,
53 interval: retry.interval,
54 maxAttempts: retry.maxAttempts,
55 });
56 // the on-event entrypoint is going to start the execution of the waiter
57 onEventFunction.addEnvironment(consts.WAITER_STATE_MACHINE_ARN_ENV, waiterStateMachine.stateMachineArn);
58 waiterStateMachine.grantStartExecution(onEventFunction);
59 }
60 this.entrypoint = onEventFunction;
61 this.serviceToken = this.entrypoint.functionArn;
62 }
63 /**
64 * Called by `CustomResource` which uses this provider.
65 * @deprecated use `provider.serviceToken` instead
66 */
67 bind(_scope) {
68 try {
69 jsiiDeprecationWarnings.print("@aws-cdk/custom-resources.Provider#bind", "use `provider.serviceToken` instead");
70 }
71 catch (error) {
72 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
73 Error.captureStackTrace(error, this.bind);
74 }
75 throw error;
76 }
77 return {
78 serviceToken: this.entrypoint.functionArn,
79 };
80 }
81 createFunction(entrypoint, name) {
82 const fn = new lambda.Function(this, `framework-${entrypoint}`, {
83 code: lambda.Code.fromAsset(RUNTIME_HANDLER_PATH, {
84 exclude: ['*.ts'],
85 }),
86 description: `AWS CDK resource provider framework - ${entrypoint} (${this.node.path})`.slice(0, 256),
87 runtime: lambda.Runtime.NODEJS_14_X,
88 handler: `framework.${entrypoint}`,
89 timeout: FRAMEWORK_HANDLER_TIMEOUT,
90 logRetention: this.logRetention,
91 vpc: this.vpc,
92 vpcSubnets: this.vpcSubnets,
93 securityGroups: this.securityGroups,
94 role: this.role,
95 functionName: name,
96 });
97 fn.addEnvironment(consts.USER_ON_EVENT_FUNCTION_ARN_ENV, this.onEventHandler.functionArn);
98 this.onEventHandler.grantInvoke(fn);
99 if (this.isCompleteHandler) {
100 fn.addEnvironment(consts.USER_IS_COMPLETE_FUNCTION_ARN_ENV, this.isCompleteHandler.functionArn);
101 this.isCompleteHandler.grantInvoke(fn);
102 }
103 return fn;
104 }
105}
106exports.Provider = Provider;
107_a = JSII_RTTI_SYMBOL_1;
108Provider[_a] = { fqn: "@aws-cdk/custom-resources.Provider", version: "1.203.0" };
109//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"provider.js","sourceRoot":"","sources":["provider.ts"],"names":[],"mappings":";;;;;;AAAA,6BAA6B;AAG7B,8CAA8C;AAE9C,wCAAyC;AAEzC,2CAA2C;AAC3C,iCAA8C;AAC9C,iEAA4D;AAK5D,mGAAmG;AACnG,2BAA2B;AAC3B,wCAA2D;AAE3D,MAAM,oBAAoB,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;AAC7D,MAAM,yBAAyB,GAAG,eAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC,yBAAyB;AA6GjF;;GAEG;AACH,MAAa,QAAS,SAAQ,gBAAa;IA2BzC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAoB;QAC5D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;;;;;;+CA5BR,QAAQ;;;;QA8BjB,IAAI,CAAC,KAAK,CAAC,iBAAiB,IAAI,CAAC,KAAK,CAAC,aAAa,IAAI,KAAK,CAAC,YAAY,CAAC,EAAE;YAC3E,MAAM,IAAI,KAAK,CAAC,iGAAiG;kBAC7G,iCAAiC,CAAC,CAAC;SACxC;QAED,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC,cAAc,CAAC;QAC3C,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC,iBAAiB,CAAC;QAEjD,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,YAAY,CAAC;QACvC,IAAI,CAAC,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC;QACrB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,UAAU,CAAC;QACnC,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC,cAAc,CAAC;QAE3C,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC;QAEvB,MAAM,eAAe,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,+BAA+B,EAAE,KAAK,CAAC,oBAAoB,CAAC,CAAC;QAEhH,IAAI,IAAI,CAAC,iBAAiB,EAAE;YAC1B,MAAM,kBAAkB,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,kCAAkC,CAAC,CAAC;YAC1F,MAAM,eAAe,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,iCAAiC,CAAC,CAAC;YAEtF,MAAM,KAAK,GAAG,2BAAoB,CAAC,KAAK,CAAC,CAAC;YAC1C,MAAM,kBAAkB,GAAG,IAAI,yCAAkB,CAAC,IAAI,EAAE,sBAAsB,EAAE;gBAC9E,iBAAiB,EAAE,kBAAkB;gBACrC,cAAc,EAAE,eAAe;gBAC/B,WAAW,EAAE,KAAK,CAAC,WAAW;gBAC9B,QAAQ,EAAE,KAAK,CAAC,QAAQ;gBACxB,WAAW,EAAE,KAAK,CAAC,WAAW;aAC/B,CAAC,CAAC;YACH,wEAAwE;YACxE,eAAe,CAAC,cAAc,CAAC,MAAM,CAAC,4BAA4B,EAAE,kBAAkB,CAAC,eAAe,CAAC,CAAC;YACxG,kBAAkB,CAAC,mBAAmB,CAAC,eAAe,CAAC,CAAC;SACzD;QAED,IAAI,CAAC,UAAU,GAAG,eAAe,CAAC;QAClC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC;KACjD;IAED;;;OAGG;IACI,IAAI,CAAC,MAAqB;;;;;;;;;;QAC/B,OAAO;YACL,YAAY,EAAE,IAAI,CAAC,UAAU,CAAC,WAAW;SAC1C,CAAC;KACH;IAEO,cAAc,CAAC,UAAkB,EAAE,IAAa;QACtD,MAAM,EAAE,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,aAAa,UAAU,EAAE,EAAE;YAC9D,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,oBAAoB,EAAE;gBAChD,OAAO,EAAE,CAAC,MAAM,CAAC;aAClB,CAAC;YACF,WAAW,EAAE,yCAAyC,UAAU,KAAK,IAAI,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC;YACpG,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,OAAO,EAAE,aAAa,UAAU,EAAE;YAClC,OAAO,EAAE,yBAAyB;YAClC,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,UAAU,EAAE,IAAI,CAAC,UAAU;YAC3B,cAAc,EAAE,IAAI,CAAC,cAAc;YACnC,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,YAAY,EAAE,IAAI;SACnB,CAAC,CAAC;QAEH,EAAE,CAAC,cAAc,CAAC,MAAM,CAAC,8BAA8B,EAAE,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;QAC1F,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;QAEpC,IAAI,IAAI,CAAC,iBAAiB,EAAE;YAC1B,EAAE,CAAC,cAAc,CAAC,MAAM,CAAC,iCAAiC,EAAE,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;YAChG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;SACxC;QAED,OAAO,EAAE,CAAC;KACX;;AAxGH,4BAyGC","sourcesContent":["import * as path from 'path';\nimport * as ec2 from '@aws-cdk/aws-ec2';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as lambda from '@aws-cdk/aws-lambda';\nimport * as logs from '@aws-cdk/aws-logs';\nimport { Duration } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport * as consts from './runtime/consts';\nimport { calculateRetryPolicy } from './util';\nimport { WaiterStateMachine } from './waiter-state-machine';\n\n// keep this import separate from other imports to reduce chance for merge conflicts with v2-main\n// eslint-disable-next-line no-duplicate-imports, import/order\nimport { CustomResourceProviderConfig, ICustomResourceProvider } from '@aws-cdk/aws-cloudformation';\n// keep this import separate from other imports to reduce chance for merge conflicts with v2-main\",\n// eslint-disable-next-line\nimport { Construct as CoreConstruct } from '@aws-cdk/core';\n\nconst RUNTIME_HANDLER_PATH = path.join(__dirname, 'runtime');\nconst FRAMEWORK_HANDLER_TIMEOUT = Duration.minutes(15); // keep it simple for now\n\n/**\n * Initialization properties for the `Provider` construct.\n */\nexport interface ProviderProps {\n\n  /**\n   * The AWS Lambda function to invoke for all resource lifecycle operations\n   * (CREATE/UPDATE/DELETE).\n   *\n   * This function is responsible to begin the requested resource operation\n   * (CREATE/UPDATE/DELETE) and return any additional properties to add to the\n   * event, which will later be passed to `isComplete`. The `PhysicalResourceId`\n   * property must be included in the response.\n   */\n  readonly onEventHandler: lambda.IFunction;\n\n  /**\n   * The AWS Lambda function to invoke in order to determine if the operation is\n   * complete.\n   *\n   * This function will be called immediately after `onEvent` and then\n   * periodically based on the configured query interval as long as it returns\n   * `false`. If the function still returns `false` and the alloted timeout has\n   * passed, the operation will fail.\n   *\n   * @default - provider is synchronous. This means that the `onEvent` handler\n   * is expected to finish all lifecycle operations within the initial invocation.\n   */\n  readonly isCompleteHandler?: lambda.IFunction;\n\n  /**\n   * Time between calls to the `isComplete` handler which determines if the\n   * resource has been stabilized.\n   *\n   * The first `isComplete` will be called immediately after `handler` and then\n   * every `queryInterval` seconds, and until `timeout` has been reached or until\n   * `isComplete` returns `true`.\n   *\n   * @default Duration.seconds(5)\n   */\n  readonly queryInterval?: Duration;\n\n  /**\n   * Total timeout for the entire operation.\n   *\n   * The maximum timeout is 2 hours (yes, it can exceed the AWS Lambda 15 minutes)\n   *\n   * @default Duration.minutes(30)\n   */\n  readonly totalTimeout?: Duration;\n\n  /**\n   * The number of days framework log events are kept in CloudWatch Logs. When\n   * updating this property, unsetting it doesn't remove the log retention policy.\n   * To remove the retention policy, set the value to `INFINITE`.\n   *\n   * @default logs.RetentionDays.INFINITE\n   */\n  readonly logRetention?: logs.RetentionDays;\n\n  /**\n   * The vpc to provision the lambda functions in.\n   *\n   * @default - functions are not provisioned inside a vpc.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Which subnets from the VPC to place the lambda functions in.\n   *\n   * Only used if 'vpc' is supplied. Note: internet access for Lambdas\n   * requires a NAT gateway, so picking Public subnets is not allowed.\n   *\n   * @default - the Vpc default strategy if not specified\n   */\n  readonly vpcSubnets?: ec2.SubnetSelection;\n\n  /**\n   * Security groups to attach to the provider functions.\n   *\n   * Only used if 'vpc' is supplied\n   *\n   * @default - If `vpc` is not supplied, no security groups are attached. Otherwise, a dedicated security\n   * group is created for each function.\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * AWS Lambda execution role.\n   *\n   * The role that will be assumed by the AWS Lambda.\n   * Must be assumable by the 'lambda.amazonaws.com' service principal.\n   *\n   * @default - A default role will be created.\n   */\n  readonly role?: iam.IRole;\n\n  /**\n   * Provider Lambda name.\n   *\n   * The provider lambda function name.\n   *\n   * @default -  CloudFormation default name from unique physical ID\n   */\n  readonly providerFunctionName?: string;\n}\n\n/**\n * Defines an AWS CloudFormation custom resource provider.\n */\nexport class Provider extends CoreConstruct implements ICustomResourceProvider {\n\n  /**\n   * The user-defined AWS Lambda function which is invoked for all resource\n   * lifecycle operations (CREATE/UPDATE/DELETE).\n   */\n  public readonly onEventHandler: lambda.IFunction;\n\n  /**\n   * The user-defined AWS Lambda function which is invoked asynchronously in\n   * order to determine if the operation is complete.\n   */\n  public readonly isCompleteHandler?: lambda.IFunction;\n\n  /**\n   * The service token to use in order to define custom resources that are\n   * backed by this provider.\n   */\n  public readonly serviceToken: string;\n\n  private readonly entrypoint: lambda.Function;\n  private readonly logRetention?: logs.RetentionDays;\n  private readonly vpc?: ec2.IVpc;\n  private readonly vpcSubnets?: ec2.SubnetSelection;\n  private readonly securityGroups?: ec2.ISecurityGroup[];\n  private readonly role?: iam.IRole;\n\n  constructor(scope: Construct, id: string, props: ProviderProps) {\n    super(scope, id);\n\n    if (!props.isCompleteHandler && (props.queryInterval || props.totalTimeout)) {\n      throw new Error('\"queryInterval\" and \"totalTimeout\" can only be configured if \"isCompleteHandler\" is specified. '\n        + 'Otherwise, they have no meaning');\n    }\n\n    this.onEventHandler = props.onEventHandler;\n    this.isCompleteHandler = props.isCompleteHandler;\n\n    this.logRetention = props.logRetention;\n    this.vpc = props.vpc;\n    this.vpcSubnets = props.vpcSubnets;\n    this.securityGroups = props.securityGroups;\n\n    this.role = props.role;\n\n    const onEventFunction = this.createFunction(consts.FRAMEWORK_ON_EVENT_HANDLER_NAME, props.providerFunctionName);\n\n    if (this.isCompleteHandler) {\n      const isCompleteFunction = this.createFunction(consts.FRAMEWORK_IS_COMPLETE_HANDLER_NAME);\n      const timeoutFunction = this.createFunction(consts.FRAMEWORK_ON_TIMEOUT_HANDLER_NAME);\n\n      const retry = calculateRetryPolicy(props);\n      const waiterStateMachine = new WaiterStateMachine(this, 'waiter-state-machine', {\n        isCompleteHandler: isCompleteFunction,\n        timeoutHandler: timeoutFunction,\n        backoffRate: retry.backoffRate,\n        interval: retry.interval,\n        maxAttempts: retry.maxAttempts,\n      });\n      // the on-event entrypoint is going to start the execution of the waiter\n      onEventFunction.addEnvironment(consts.WAITER_STATE_MACHINE_ARN_ENV, waiterStateMachine.stateMachineArn);\n      waiterStateMachine.grantStartExecution(onEventFunction);\n    }\n\n    this.entrypoint = onEventFunction;\n    this.serviceToken = this.entrypoint.functionArn;\n  }\n\n  /**\n   * Called by `CustomResource` which uses this provider.\n   * @deprecated use `provider.serviceToken` instead\n   */\n  public bind(_scope: CoreConstruct): CustomResourceProviderConfig {\n    return {\n      serviceToken: this.entrypoint.functionArn,\n    };\n  }\n\n  private createFunction(entrypoint: string, name?: string) {\n    const fn = new lambda.Function(this, `framework-${entrypoint}`, {\n      code: lambda.Code.fromAsset(RUNTIME_HANDLER_PATH, {\n        exclude: ['*.ts'],\n      }),\n      description: `AWS CDK resource provider framework - ${entrypoint} (${this.node.path})`.slice(0, 256),\n      runtime: lambda.Runtime.NODEJS_14_X,\n      handler: `framework.${entrypoint}`,\n      timeout: FRAMEWORK_HANDLER_TIMEOUT,\n      logRetention: this.logRetention,\n      vpc: this.vpc,\n      vpcSubnets: this.vpcSubnets,\n      securityGroups: this.securityGroups,\n      role: this.role,\n      functionName: name,\n    });\n\n    fn.addEnvironment(consts.USER_ON_EVENT_FUNCTION_ARN_ENV, this.onEventHandler.functionArn);\n    this.onEventHandler.grantInvoke(fn);\n\n    if (this.isCompleteHandler) {\n      fn.addEnvironment(consts.USER_IS_COMPLETE_FUNCTION_ARN_ENV, this.isCompleteHandler.functionArn);\n      this.isCompleteHandler.grantInvoke(fn);\n    }\n\n    return fn;\n  }\n}\n"]}
\No newline at end of file