1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.Alias = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const appscaling = require("@aws-cdk/aws-applicationautoscaling");
|
8 | const iam = require("@aws-cdk/aws-iam");
|
9 | const core_1 = require("@aws-cdk/core");
|
10 | const function_base_1 = require("./function-base");
|
11 | const lambda_version_1 = require("./lambda-version");
|
12 | const lambda_generated_1 = require("./lambda.generated");
|
13 | const scalable_function_attribute_1 = require("./private/scalable-function-attribute");
|
14 | /**
|
15 | * A new alias to a particular version of a Lambda function.
|
16 | */
|
17 | class Alias extends function_base_1.QualifiedFunctionBase {
|
18 | constructor(scope, id, props) {
|
19 | super(scope, id, {
|
20 | physicalName: props.aliasName,
|
21 | });
|
22 | this.canCreatePermissions = true;
|
23 | try {
|
24 | jsiiDeprecationWarnings._aws_cdk_aws_lambda_AliasProps(props);
|
25 | }
|
26 | catch (error) {
|
27 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
28 | Error.captureStackTrace(error, this.constructor);
|
29 | }
|
30 | throw error;
|
31 | }
|
32 | this.lambda = props.version.lambda;
|
33 | this.aliasName = this.physicalName;
|
34 | this.version = props.version;
|
35 | this.architecture = this.lambda.architecture;
|
36 | const alias = new lambda_generated_1.CfnAlias(this, 'Resource', {
|
37 | name: this.aliasName,
|
38 | description: props.description,
|
39 | functionName: this.version.lambda.functionName,
|
40 | functionVersion: props.version.version,
|
41 | routingConfig: this.determineRoutingConfig(props),
|
42 | provisionedConcurrencyConfig: this.determineProvisionedConcurrency(props),
|
43 | });
|
44 | // Use a Service Linked Role
|
45 | // https://docs.aws.amazon.com/autoscaling/application/userguide/application-auto-scaling-service-linked-roles.html
|
46 | this.scalingRole = iam.Role.fromRoleArn(this, 'ScalingRole', this.stack.formatArn({
|
47 | service: 'iam',
|
48 | region: '',
|
49 | resource: 'role/aws-service-role/lambda.application-autoscaling.amazonaws.com',
|
50 | resourceName: 'AWSServiceRoleForApplicationAutoScaling_LambdaConcurrency',
|
51 | }));
|
52 | this.functionArn = this.getResourceArnAttribute(alias.ref, {
|
53 | service: 'lambda',
|
54 | resource: 'function',
|
55 | resourceName: `${this.lambda.functionName}:${this.physicalName}`,
|
56 | arnFormat: core_1.ArnFormat.COLON_RESOURCE_NAME,
|
57 | });
|
58 | this.qualifier = lambda_version_1.extractQualifierFromArn(alias.ref);
|
59 | if (props.onFailure || props.onSuccess || props.maxEventAge || props.retryAttempts !== undefined) {
|
60 | this.configureAsyncInvoke({
|
61 | onFailure: props.onFailure,
|
62 | onSuccess: props.onSuccess,
|
63 | maxEventAge: props.maxEventAge,
|
64 | retryAttempts: props.retryAttempts,
|
65 | });
|
66 | }
|
67 | // ARN parsing splits on `:`, so we can only get the function's name from the ARN as resourceName...
|
68 | // And we're parsing it out (instead of using the underlying function directly) in order to have use of it incur
|
69 | // an implicit dependency on the resource.
|
70 | this.functionName = `${this.stack.splitArn(this.functionArn, core_1.ArnFormat.COLON_RESOURCE_NAME).resourceName}:${this.aliasName}`;
|
71 | }
|
72 | static fromAliasAttributes(scope, id, attrs) {
|
73 | try {
|
74 | jsiiDeprecationWarnings._aws_cdk_aws_lambda_AliasAttributes(attrs);
|
75 | }
|
76 | catch (error) {
|
77 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
78 | Error.captureStackTrace(error, this.fromAliasAttributes);
|
79 | }
|
80 | throw error;
|
81 | }
|
82 | class Imported extends function_base_1.QualifiedFunctionBase {
|
83 | constructor() {
|
84 | super(...arguments);
|
85 | this.aliasName = attrs.aliasName;
|
86 | this.version = attrs.aliasVersion;
|
87 | this.lambda = attrs.aliasVersion.lambda;
|
88 | this.functionArn = `${attrs.aliasVersion.lambda.functionArn}:${attrs.aliasName}`;
|
89 | this.functionName = `${attrs.aliasVersion.lambda.functionName}:${attrs.aliasName}`;
|
90 | this.grantPrincipal = attrs.aliasVersion.grantPrincipal;
|
91 | this.role = attrs.aliasVersion.role;
|
92 | this.architecture = attrs.aliasVersion.lambda.architecture;
|
93 | this.canCreatePermissions = this._isStackAccount();
|
94 | this.qualifier = attrs.aliasName;
|
95 | }
|
96 | }
|
97 | return new Imported(scope, id);
|
98 | }
|
99 | get grantPrincipal() {
|
100 | return this.version.grantPrincipal;
|
101 | }
|
102 | get role() {
|
103 | return this.version.role;
|
104 | }
|
105 | metric(metricName, props = {}) {
|
106 | // Metrics on Aliases need the "bare" function name, and the alias' ARN, this differs from the base behavior.
|
107 | return super.metric(metricName, {
|
108 | dimensionsMap: {
|
109 | FunctionName: this.lambda.functionName,
|
110 | // construct the name from the underlying lambda so that alarms on an alias
|
111 | // don't cause a circular dependency with CodeDeploy
|
112 | // see: https://github.com/aws/aws-cdk/issues/2231
|
113 | Resource: `${this.lambda.functionName}:${this.aliasName}`,
|
114 | },
|
115 | ...props,
|
116 | });
|
117 | }
|
118 | /**
|
119 | * Configure provisioned concurrency autoscaling on a function alias. Returns a scalable attribute that can call
|
120 | * `scaleOnUtilization()` and `scaleOnSchedule()`.
|
121 | *
|
122 | * @param options Autoscaling options
|
123 | */
|
124 | addAutoScaling(options) {
|
125 | var _b;
|
126 | try {
|
127 | jsiiDeprecationWarnings._aws_cdk_aws_lambda_AutoScalingOptions(options);
|
128 | }
|
129 | catch (error) {
|
130 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
131 | Error.captureStackTrace(error, this.addAutoScaling);
|
132 | }
|
133 | throw error;
|
134 | }
|
135 | if (this.scalableAlias) {
|
136 | throw new Error('AutoScaling already enabled for this alias');
|
137 | }
|
138 | return this.scalableAlias = new scalable_function_attribute_1.ScalableFunctionAttribute(this, 'AliasScaling', {
|
139 | minCapacity: (_b = options.minCapacity) !== null && _b !== void 0 ? _b : 1,
|
140 | maxCapacity: options.maxCapacity,
|
141 | resourceId: `function:${this.functionName}`,
|
142 | dimension: 'lambda:function:ProvisionedConcurrency',
|
143 | serviceNamespace: appscaling.ServiceNamespace.LAMBDA,
|
144 | role: this.scalingRole,
|
145 | });
|
146 | }
|
147 | /**
|
148 | * Calculate the routingConfig parameter from the input props
|
149 | */
|
150 | determineRoutingConfig(props) {
|
151 | if (!props.additionalVersions || props.additionalVersions.length === 0) {
|
152 | return undefined;
|
153 | }
|
154 | this.validateAdditionalWeights(props.additionalVersions);
|
155 | return {
|
156 | additionalVersionWeights: props.additionalVersions.map(vw => {
|
157 | return {
|
158 | functionVersion: vw.version.version,
|
159 | functionWeight: vw.weight,
|
160 | };
|
161 | }),
|
162 | };
|
163 | }
|
164 | /**
|
165 | * Validate that the additional version weights make sense
|
166 | *
|
167 | * We validate that they are positive and add up to something <= 1.
|
168 | */
|
169 | validateAdditionalWeights(weights) {
|
170 | const total = weights.map(w => {
|
171 | if (w.weight < 0 || w.weight > 1) {
|
172 | throw new Error(`Additional version weight must be between 0 and 1, got: ${w.weight}`);
|
173 | }
|
174 | return w.weight;
|
175 | }).reduce((a, x) => a + x);
|
176 | if (total > 1) {
|
177 | throw new Error(`Sum of additional version weights must not exceed 1, got: ${total}`);
|
178 | }
|
179 | }
|
180 | /**
|
181 | * Validate that the provisionedConcurrentExecutions makes sense
|
182 | *
|
183 | * Member must have value greater than or equal to 1
|
184 | */
|
185 | determineProvisionedConcurrency(props) {
|
186 | if (!props.provisionedConcurrentExecutions) {
|
187 | return undefined;
|
188 | }
|
189 | if (props.provisionedConcurrentExecutions <= 0) {
|
190 | throw new Error('provisionedConcurrentExecutions must have value greater than or equal to 1');
|
191 | }
|
192 | return { provisionedConcurrentExecutions: props.provisionedConcurrentExecutions };
|
193 | }
|
194 | }
|
195 | exports.Alias = Alias;
|
196 | _a = JSII_RTTI_SYMBOL_1;
|
197 | Alias[_a] = { fqn: "@aws-cdk/aws-lambda.Alias", version: "1.154.0" };
|
198 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alias.js","sourceRoot":"","sources":["alias.ts"],"names":[],"mappings":"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","sourcesContent":["import * as appscaling from '@aws-cdk/aws-applicationautoscaling';\nimport * as cloudwatch from '@aws-cdk/aws-cloudwatch';\nimport * as iam from '@aws-cdk/aws-iam';\nimport { ArnFormat } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { Architecture } from './architecture';\nimport { EventInvokeConfigOptions } from './event-invoke-config';\nimport { IFunction, QualifiedFunctionBase } from './function-base';\nimport { extractQualifierFromArn, IVersion } from './lambda-version';\nimport { CfnAlias } from './lambda.generated';\nimport { ScalableFunctionAttribute } from './private/scalable-function-attribute';\nimport { AutoScalingOptions, IScalableFunctionAttribute } from './scalable-attribute-api';\n\nexport interface IAlias extends IFunction {\n  /**\n   * Name of this alias.\n   *\n   * @attribute\n   */\n  readonly aliasName: string;\n\n  /**\n   * The underlying Lambda function version.\n   */\n  readonly version: IVersion;\n}\n\n/**\n * Options for `lambda.Alias`.\n */\nexport interface AliasOptions extends EventInvokeConfigOptions {\n  /**\n   * Description for the alias\n   *\n   * @default No description\n   */\n  readonly description?: string;\n\n  /**\n   * Additional versions with individual weights this alias points to\n   *\n   * Individual additional version weights specified here should add up to\n   * (less than) one. All remaining weight is routed to the default\n   * version.\n   *\n   * For example, the config is\n   *\n   *    version: \"1\"\n   *    additionalVersions: [{ version: \"2\", weight: 0.05 }]\n   *\n   * Then 5% of traffic will be routed to function version 2, while\n   * the remaining 95% of traffic will be routed to function version 1.\n   *\n   * @default No additional versions\n   */\n  readonly additionalVersions?: VersionWeight[];\n\n  /**\n   * Specifies a provisioned concurrency configuration for a function's alias.\n   *\n   * @default No provisioned concurrency\n   */\n  readonly provisionedConcurrentExecutions?: number;\n}\n\n/**\n * Properties for a new Lambda alias\n */\nexport interface AliasProps extends AliasOptions {\n  /**\n   * Name of this alias\n   */\n  readonly aliasName: string;\n\n  /**\n   * Function version this alias refers to\n   *\n   * Use lambda.currentVersion to reference a version with your latest changes.\n   */\n  readonly version: IVersion;\n}\n\nexport interface AliasAttributes {\n  readonly aliasName: string;\n  readonly aliasVersion: IVersion;\n}\n\n/**\n * A new alias to a particular version of a Lambda function.\n */\nexport class Alias extends QualifiedFunctionBase implements IAlias {\n  public static fromAliasAttributes(scope: Construct, id: string, attrs: AliasAttributes): IAlias {\n    class Imported extends QualifiedFunctionBase implements IAlias {\n      public readonly aliasName = attrs.aliasName;\n      public readonly version = attrs.aliasVersion;\n      public readonly lambda = attrs.aliasVersion.lambda;\n      public readonly functionArn = `${attrs.aliasVersion.lambda.functionArn}:${attrs.aliasName}`;\n      public readonly functionName = `${attrs.aliasVersion.lambda.functionName}:${attrs.aliasName}`;\n      public readonly grantPrincipal = attrs.aliasVersion.grantPrincipal;\n      public readonly role = attrs.aliasVersion.role;\n      public readonly architecture = attrs.aliasVersion.lambda.architecture;\n\n      protected readonly canCreatePermissions = this._isStackAccount();\n      protected readonly qualifier = attrs.aliasName;\n    }\n    return new Imported(scope, id);\n  }\n\n  /**\n   * Name of this alias.\n   *\n   * @attribute\n   */\n  public readonly aliasName: string;\n  /**\n   * ARN of this alias\n   *\n   * Used to be able to use Alias in place of a regular Lambda. Lambda accepts\n   * ARNs everywhere it accepts function names.\n   */\n  public readonly functionName: string;\n\n  public readonly lambda: IFunction;\n\n  public readonly architecture: Architecture;\n\n  public readonly version: IVersion;\n\n  /**\n   * ARN of this alias\n   *\n   * Used to be able to use Alias in place of a regular Lambda. Lambda accepts\n   * ARNs everywhere it accepts function names.\n   */\n  public readonly functionArn: string;\n\n  protected readonly qualifier: string;\n\n  protected readonly canCreatePermissions: boolean = true;\n\n  private scalableAlias?: ScalableFunctionAttribute;\n  private readonly scalingRole: iam.IRole;\n\n  constructor(scope: Construct, id: string, props: AliasProps) {\n    super(scope, id, {\n      physicalName: props.aliasName,\n    });\n\n    this.lambda = props.version.lambda;\n    this.aliasName = this.physicalName;\n    this.version = props.version;\n    this.architecture = this.lambda.architecture;\n\n    const alias = new CfnAlias(this, 'Resource', {\n      name: this.aliasName,\n      description: props.description,\n      functionName: this.version.lambda.functionName,\n      functionVersion: props.version.version,\n      routingConfig: this.determineRoutingConfig(props),\n      provisionedConcurrencyConfig: this.determineProvisionedConcurrency(props),\n    });\n\n    // Use a Service Linked Role\n    // https://docs.aws.amazon.com/autoscaling/application/userguide/application-auto-scaling-service-linked-roles.html\n    this.scalingRole = iam.Role.fromRoleArn(this, 'ScalingRole', this.stack.formatArn({\n      service: 'iam',\n      region: '',\n      resource: 'role/aws-service-role/lambda.application-autoscaling.amazonaws.com',\n      resourceName: 'AWSServiceRoleForApplicationAutoScaling_LambdaConcurrency',\n    }));\n\n    this.functionArn = this.getResourceArnAttribute(alias.ref, {\n      service: 'lambda',\n      resource: 'function',\n      resourceName: `${this.lambda.functionName}:${this.physicalName}`,\n      arnFormat: ArnFormat.COLON_RESOURCE_NAME,\n    });\n\n    this.qualifier = extractQualifierFromArn(alias.ref);\n\n    if (props.onFailure || props.onSuccess || props.maxEventAge || props.retryAttempts !== undefined) {\n      this.configureAsyncInvoke({\n        onFailure: props.onFailure,\n        onSuccess: props.onSuccess,\n        maxEventAge: props.maxEventAge,\n        retryAttempts: props.retryAttempts,\n      });\n    }\n\n    // ARN parsing splits on `:`, so we can only get the function's name from the ARN as resourceName...\n    // And we're parsing it out (instead of using the underlying function directly) in order to have use of it incur\n    // an implicit dependency on the resource.\n    this.functionName = `${this.stack.splitArn(this.functionArn, ArnFormat.COLON_RESOURCE_NAME).resourceName!}:${this.aliasName}`;\n  }\n\n  public get grantPrincipal() {\n    return this.version.grantPrincipal;\n  }\n\n  public get role() {\n    return this.version.role;\n  }\n\n  public metric(metricName: string, props: cloudwatch.MetricOptions = {}): cloudwatch.Metric {\n    // Metrics on Aliases need the \"bare\" function name, and the alias' ARN, this differs from the base behavior.\n    return super.metric(metricName, {\n      dimensionsMap: {\n        FunctionName: this.lambda.functionName,\n        // construct the name from the underlying lambda so that alarms on an alias\n        // don't cause a circular dependency with CodeDeploy\n        // see: https://github.com/aws/aws-cdk/issues/2231\n        Resource: `${this.lambda.functionName}:${this.aliasName}`,\n      },\n      ...props,\n    });\n  }\n\n  /**\n   * Configure provisioned concurrency autoscaling on a function alias. Returns a scalable attribute that can call\n   * `scaleOnUtilization()` and `scaleOnSchedule()`.\n   *\n   * @param options Autoscaling options\n   */\n  public addAutoScaling(options: AutoScalingOptions): IScalableFunctionAttribute {\n    if (this.scalableAlias) {\n      throw new Error('AutoScaling already enabled for this alias');\n    }\n    return this.scalableAlias = new ScalableFunctionAttribute(this, 'AliasScaling', {\n      minCapacity: options.minCapacity ?? 1,\n      maxCapacity: options.maxCapacity,\n      resourceId: `function:${this.functionName}`,\n      dimension: 'lambda:function:ProvisionedConcurrency',\n      serviceNamespace: appscaling.ServiceNamespace.LAMBDA,\n      role: this.scalingRole,\n    });\n  }\n\n  /**\n   * Calculate the routingConfig parameter from the input props\n   */\n  private determineRoutingConfig(props: AliasProps) {\n    if (!props.additionalVersions || props.additionalVersions.length === 0) {\n      return undefined;\n    }\n\n    this.validateAdditionalWeights(props.additionalVersions);\n\n    return {\n      additionalVersionWeights: props.additionalVersions.map(vw => {\n        return {\n          functionVersion: vw.version.version,\n          functionWeight: vw.weight,\n        };\n      }),\n    };\n  }\n\n  /**\n   * Validate that the additional version weights make sense\n   *\n   * We validate that they are positive and add up to something <= 1.\n   */\n  private validateAdditionalWeights(weights: VersionWeight[]) {\n    const total = weights.map(w => {\n      if (w.weight < 0 || w.weight > 1) { throw new Error(`Additional version weight must be between 0 and 1, got: ${w.weight}`); }\n      return w.weight;\n    }).reduce((a, x) => a + x);\n\n    if (total > 1) {\n      throw new Error(`Sum of additional version weights must not exceed 1, got: ${total}`);\n    }\n  }\n\n  /**\n   * Validate that the provisionedConcurrentExecutions makes sense\n   *\n   * Member must have value greater than or equal to 1\n   */\n  private determineProvisionedConcurrency(props: AliasProps): CfnAlias.ProvisionedConcurrencyConfigurationProperty | undefined {\n    if (!props.provisionedConcurrentExecutions) {\n      return undefined;\n    }\n\n    if (props.provisionedConcurrentExecutions <= 0) {\n      throw new Error('provisionedConcurrentExecutions must have value greater than or equal to 1');\n    }\n\n    return { provisionedConcurrentExecutions: props.provisionedConcurrentExecutions };\n  }\n}\n\n/**\n * A version/weight pair for routing traffic to Lambda functions\n */\nexport interface VersionWeight {\n  /**\n   * The version to route traffic to\n   */\n  readonly version: IVersion;\n\n  /**\n   * How much weight to assign to this version (0..1)\n   */\n  readonly weight: number;\n}\n"]} |
\ | No newline at end of file |