@aws-cdk/aws-cloudfront
Version:
The CDK Construct Library for AWS::CloudFront
216 lines • 31.6 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.EdgeFunction = void 0;
const jsiiDeprecationWarnings = require("../../.warnings.jsii.js");
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const path = require("path");
const iam = require("@aws-cdk/aws-iam");
const lambda = require("@aws-cdk/aws-lambda");
const ssm = require("@aws-cdk/aws-ssm");
const core_1 = require("@aws-cdk/core");
/**
* A Lambda@Edge function.
*
* Convenience resource for requesting a Lambda function in the 'us-east-1' region for use with Lambda@Edge.
* Implements several restrictions enforced by Lambda@Edge.
*
* Note that this construct requires that the 'us-east-1' region has been bootstrapped.
* See https://docs.aws.amazon.com/cdk/latest/guide/bootstrapping.html or 'cdk bootstrap --help' for options.
*
* @resource AWS::Lambda::Function
*/
class EdgeFunction extends core_1.Resource {
constructor(scope, id, props) {
super(scope, id);
this.isBoundToVpc = false;
try {
jsiiDeprecationWarnings._aws_cdk_aws_cloudfront_experimental_EdgeFunctionProps(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.constructor);
}
throw error;
}
// Create a simple Function if we're already in us-east-1; otherwise create a cross-region stack.
const regionIsUsEast1 = !core_1.Token.isUnresolved(this.env.region) && this.env.region === 'us-east-1';
const { edgeFunction, edgeArn } = regionIsUsEast1
? this.createInRegionFunction(props)
: this.createCrossRegionFunction(id, props);
this.edgeArn = edgeArn;
this.functionArn = edgeArn;
this._edgeFunction = edgeFunction;
this.functionName = this._edgeFunction.functionName;
this.grantPrincipal = this._edgeFunction.role;
this.permissionsNode = this._edgeFunction.permissionsNode;
this.version = lambda.extractQualifierFromArn(this.functionArn);
this.architecture = this._edgeFunction.architecture;
this.resourceArnsForGrantInvoke = this._edgeFunction.resourceArnsForGrantInvoke;
this.node.defaultChild = this._edgeFunction;
}
get lambda() {
return this._edgeFunction;
}
/**
* Convenience method to make `EdgeFunction` conform to the same interface as `Function`.
*/
get currentVersion() {
return this;
}
addAlias(aliasName, options = {}) {
return new lambda.Alias(this._edgeFunction, `Alias${aliasName}`, {
aliasName,
version: this._edgeFunction.currentVersion,
...options,
});
}
/**
* Not supported. Connections are only applicable to VPC-enabled functions.
*/
get connections() {
throw new Error('Lambda@Edge does not support connections');
}
get latestVersion() {
throw new Error('$LATEST function version cannot be used for Lambda@Edge');
}
addEventSourceMapping(id, options) {
return this.lambda.addEventSourceMapping(id, options);
}
addPermission(id, permission) {
return this.lambda.addPermission(id, permission);
}
addToRolePolicy(statement) {
return this.lambda.addToRolePolicy(statement);
}
grantInvoke(identity) {
return this.lambda.grantInvoke(identity);
}
grantInvokeUrl(identity) {
return this.lambda.grantInvokeUrl(identity);
}
metric(metricName, props) {
return this.lambda.metric(metricName, { ...props, region: EdgeFunction.EDGE_REGION });
}
metricDuration(props) {
return this.lambda.metricDuration({ ...props, region: EdgeFunction.EDGE_REGION });
}
metricErrors(props) {
return this.lambda.metricErrors({ ...props, region: EdgeFunction.EDGE_REGION });
}
metricInvocations(props) {
return this.lambda.metricInvocations({ ...props, region: EdgeFunction.EDGE_REGION });
}
metricThrottles(props) {
return this.lambda.metricThrottles({ ...props, region: EdgeFunction.EDGE_REGION });
}
/** Adds an event source to this function. */
addEventSource(source) {
return this.lambda.addEventSource(source);
}
configureAsyncInvoke(options) {
return this.lambda.configureAsyncInvoke(options);
}
addFunctionUrl(options) {
return this.lambda.addFunctionUrl(options);
}
/** Create a function in-region */
createInRegionFunction(props) {
const edgeFunction = new lambda.Function(this, 'Fn', props);
addEdgeLambdaToRoleTrustStatement(edgeFunction.role);
return { edgeFunction, edgeArn: edgeFunction.currentVersion.edgeArn };
}
/** Create a support stack and function in us-east-1, and a SSM reader in-region */
createCrossRegionFunction(id, props) {
const parameterNamePrefix = 'cdk/EdgeFunctionArn';
if (core_1.Token.isUnresolved(this.env.region)) {
throw new Error('stacks which use EdgeFunctions must have an explicitly set region');
}
// SSM parameter names must only contain letters, numbers, ., _, -, or /.
const sanitizedPath = this.node.path.replace(/[^\/\w.-]/g, '_');
const parameterName = `/${parameterNamePrefix}/${this.env.region}/${sanitizedPath}`;
const functionStack = this.edgeStack(props.stackId);
const edgeFunction = new lambda.Function(functionStack, id, props);
addEdgeLambdaToRoleTrustStatement(edgeFunction.role);
// Store the current version's ARN to be retrieved by the cross region reader below.
const version = edgeFunction.currentVersion;
new ssm.StringParameter(edgeFunction, 'Parameter', {
parameterName,
stringValue: version.edgeArn,
});
const edgeArn = this.createCrossRegionArnReader(parameterNamePrefix, parameterName, version);
return { edgeFunction, edgeArn };
}
createCrossRegionArnReader(parameterNamePrefix, parameterName, version) {
// Prefix of the parameter ARN that applies to all EdgeFunctions.
// This is necessary because the `CustomResourceProvider` is a singleton, and the `policyStatement`
// must work for multiple EdgeFunctions.
const parameterArnPrefix = this.stack.formatArn({
service: 'ssm',
region: EdgeFunction.EDGE_REGION,
resource: 'parameter',
resourceName: parameterNamePrefix + '/*',
});
const resourceType = 'Custom::CrossRegionStringParameterReader';
const serviceToken = core_1.CustomResourceProvider.getOrCreate(this, resourceType, {
codeDirectory: path.join(__dirname, 'edge-function'),
runtime: core_1.CustomResourceProviderRuntime.NODEJS_12_X,
policyStatements: [{
Effect: 'Allow',
Resource: parameterArnPrefix,
Action: ['ssm:GetParameter'],
}],
});
const resource = new core_1.CustomResource(this, 'ArnReader', {
resourceType: resourceType,
serviceToken,
properties: {
Region: EdgeFunction.EDGE_REGION,
ParameterName: parameterName,
// This is used to determine when the function has changed, to refresh the ARN from the custom resource.
//
// Use the logical id of the function version. Whenever a function version changes, the logical id must be
// changed for it to take effect - a good candidate for RefreshToken.
RefreshToken: core_1.Lazy.uncachedString({
produce: () => {
const cfn = version.node.defaultChild;
return this.stack.resolve(cfn.logicalId);
},
}),
},
});
return resource.getAttString('FunctionArn');
}
edgeStack(stackId) {
const stage = core_1.Stage.of(this);
if (!stage) {
throw new Error('stacks which use EdgeFunctions must be part of a CDK app or stage');
}
const edgeStackId = stackId !== null && stackId !== void 0 ? stackId : `edge-lambda-stack-${this.stack.node.addr}`;
let edgeStack = stage.node.tryFindChild(edgeStackId);
if (!edgeStack) {
edgeStack = new core_1.Stack(stage, edgeStackId, {
env: {
region: EdgeFunction.EDGE_REGION,
account: core_1.Stack.of(this).account,
},
});
}
this.stack.addDependency(edgeStack);
return edgeStack;
}
}
exports.EdgeFunction = EdgeFunction;
_a = JSII_RTTI_SYMBOL_1;
EdgeFunction[_a] = { fqn: "@aws-cdk/aws-cloudfront.experimental.EdgeFunction", version: "1.157.0" };
EdgeFunction.EDGE_REGION = 'us-east-1';
function addEdgeLambdaToRoleTrustStatement(role) {
if (role instanceof iam.Role && role.assumeRolePolicy) {
const statement = new iam.PolicyStatement();
const edgeLambdaServicePrincipal = new iam.ServicePrincipal('edgelambda.amazonaws.com');
statement.addPrincipals(edgeLambdaServicePrincipal);
statement.addActions(edgeLambdaServicePrincipal.assumeRoleAction);
role.assumeRolePolicy.addStatements(statement);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"edge-function.js","sourceRoot":"","sources":["edge-function.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as cloudwatch from '@aws-cdk/aws-cloudwatch';\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 ssm from '@aws-cdk/aws-ssm';\nimport {\n  CfnResource, ConstructNode,\n  CustomResource, CustomResourceProvider, CustomResourceProviderRuntime,\n  Lazy, Resource, Stack, Stage, Token,\n} from '@aws-cdk/core';\nimport { Construct } from 'constructs';\n\n/**\n * Properties for creating a Lambda@Edge function\n */\nexport interface EdgeFunctionProps extends lambda.FunctionProps {\n  /**\n   * The stack ID of Lambda@Edge function.\n   *\n   * @default - `edge-lambda-stack-${region}`\n   */\n  readonly stackId?: string;\n}\n\n/**\n * A Lambda@Edge function.\n *\n * Convenience resource for requesting a Lambda function in the 'us-east-1' region for use with Lambda@Edge.\n * Implements several restrictions enforced by Lambda@Edge.\n *\n * Note that this construct requires that the 'us-east-1' region has been bootstrapped.\n * See https://docs.aws.amazon.com/cdk/latest/guide/bootstrapping.html or 'cdk bootstrap --help' for options.\n *\n * @resource AWS::Lambda::Function\n */\nexport class EdgeFunction extends Resource implements lambda.IVersion {\n\n  private static readonly EDGE_REGION: string = 'us-east-1';\n\n  public readonly edgeArn: string;\n  public readonly functionName: string;\n  public readonly functionArn: string;\n  public readonly grantPrincipal: iam.IPrincipal;\n  public readonly isBoundToVpc = false;\n  public readonly permissionsNode: ConstructNode;\n  public readonly role?: iam.IRole;\n  public readonly version: string;\n  public readonly architecture: lambda.Architecture;\n  public readonly resourceArnsForGrantInvoke: string[];\n\n  private readonly _edgeFunction: lambda.Function;\n\n  constructor(scope: Construct, id: string, props: EdgeFunctionProps) {\n    super(scope, id);\n\n    // Create a simple Function if we're already in us-east-1; otherwise create a cross-region stack.\n    const regionIsUsEast1 = !Token.isUnresolved(this.env.region) && this.env.region === 'us-east-1';\n    const { edgeFunction, edgeArn } = regionIsUsEast1\n      ? this.createInRegionFunction(props)\n      : this.createCrossRegionFunction(id, props);\n\n    this.edgeArn = edgeArn;\n\n    this.functionArn = edgeArn;\n    this._edgeFunction = edgeFunction;\n    this.functionName = this._edgeFunction.functionName;\n    this.grantPrincipal = this._edgeFunction.role!;\n    this.permissionsNode = this._edgeFunction.permissionsNode;\n    this.version = lambda.extractQualifierFromArn(this.functionArn);\n    this.architecture = this._edgeFunction.architecture;\n    this.resourceArnsForGrantInvoke = this._edgeFunction.resourceArnsForGrantInvoke;\n\n    this.node.defaultChild = this._edgeFunction;\n  }\n\n  public get lambda(): lambda.IFunction {\n    return this._edgeFunction;\n  }\n\n  /**\n   * Convenience method to make `EdgeFunction` conform to the same interface as `Function`.\n   */\n  public get currentVersion(): lambda.IVersion {\n    return this;\n  }\n\n  public addAlias(aliasName: string, options: lambda.AliasOptions = {}): lambda.Alias {\n    return new lambda.Alias(this._edgeFunction, `Alias${aliasName}`, {\n      aliasName,\n      version: this._edgeFunction.currentVersion,\n      ...options,\n    });\n  }\n\n  /**\n   * Not supported. Connections are only applicable to VPC-enabled functions.\n   */\n  public get connections(): ec2.Connections {\n    throw new Error('Lambda@Edge does not support connections');\n  }\n  public get latestVersion(): lambda.IVersion {\n    throw new Error('$LATEST function version cannot be used for Lambda@Edge');\n  }\n\n  public addEventSourceMapping(id: string, options: lambda.EventSourceMappingOptions): lambda.EventSourceMapping {\n    return this.lambda.addEventSourceMapping(id, options);\n  }\n  public addPermission(id: string, permission: lambda.Permission): void {\n    return this.lambda.addPermission(id, permission);\n  }\n  public addToRolePolicy(statement: iam.PolicyStatement): void {\n    return this.lambda.addToRolePolicy(statement);\n  }\n  public grantInvoke(identity: iam.IGrantable): iam.Grant {\n    return this.lambda.grantInvoke(identity);\n  }\n  public grantInvokeUrl(identity: iam.IGrantable): iam.Grant {\n    return this.lambda.grantInvokeUrl(identity);\n  }\n  public metric(metricName: string, props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.lambda.metric(metricName, { ...props, region: EdgeFunction.EDGE_REGION });\n  }\n  public metricDuration(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.lambda.metricDuration({ ...props, region: EdgeFunction.EDGE_REGION });\n  }\n  public metricErrors(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.lambda.metricErrors({ ...props, region: EdgeFunction.EDGE_REGION });\n  }\n  public metricInvocations(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.lambda.metricInvocations({ ...props, region: EdgeFunction.EDGE_REGION });\n  }\n  public metricThrottles(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.lambda.metricThrottles({ ...props, region: EdgeFunction.EDGE_REGION });\n  }\n  /** Adds an event source to this function. */\n  public addEventSource(source: lambda.IEventSource): void {\n    return this.lambda.addEventSource(source);\n  }\n  public configureAsyncInvoke(options: lambda.EventInvokeConfigOptions): void {\n    return this.lambda.configureAsyncInvoke(options);\n  }\n  public addFunctionUrl(options?: lambda.FunctionUrlOptions): lambda.FunctionUrl {\n    return this.lambda.addFunctionUrl(options);\n  }\n\n  /** Create a function in-region */\n  private createInRegionFunction(props: lambda.FunctionProps): FunctionConfig {\n    const edgeFunction = new lambda.Function(this, 'Fn', props);\n    addEdgeLambdaToRoleTrustStatement(edgeFunction.role!);\n\n    return { edgeFunction, edgeArn: edgeFunction.currentVersion.edgeArn };\n  }\n\n  /** Create a support stack and function in us-east-1, and a SSM reader in-region */\n  private createCrossRegionFunction(id: string, props: EdgeFunctionProps): FunctionConfig {\n    const parameterNamePrefix = 'cdk/EdgeFunctionArn';\n    if (Token.isUnresolved(this.env.region)) {\n      throw new Error('stacks which use EdgeFunctions must have an explicitly set region');\n    }\n    // SSM parameter names must only contain letters, numbers, ., _, -, or /.\n    const sanitizedPath = this.node.path.replace(/[^\\/\\w.-]/g, '_');\n    const parameterName = `/${parameterNamePrefix}/${this.env.region}/${sanitizedPath}`;\n    const functionStack = this.edgeStack(props.stackId);\n\n    const edgeFunction = new lambda.Function(functionStack, id, props);\n    addEdgeLambdaToRoleTrustStatement(edgeFunction.role!);\n\n    // Store the current version's ARN to be retrieved by the cross region reader below.\n    const version = edgeFunction.currentVersion;\n    new ssm.StringParameter(edgeFunction, 'Parameter', {\n      parameterName,\n      stringValue: version.edgeArn,\n    });\n\n    const edgeArn = this.createCrossRegionArnReader(parameterNamePrefix, parameterName, version);\n\n    return { edgeFunction, edgeArn };\n  }\n\n  private createCrossRegionArnReader(parameterNamePrefix: string, parameterName: string, version: lambda.Version): string {\n    // Prefix of the parameter ARN that applies to all EdgeFunctions.\n    // This is necessary because the `CustomResourceProvider` is a singleton, and the `policyStatement`\n    // must work for multiple EdgeFunctions.\n    const parameterArnPrefix = this.stack.formatArn({\n      service: 'ssm',\n      region: EdgeFunction.EDGE_REGION,\n      resource: 'parameter',\n      resourceName: parameterNamePrefix + '/*',\n    });\n\n    const resourceType = 'Custom::CrossRegionStringParameterReader';\n    const serviceToken = CustomResourceProvider.getOrCreate(this, resourceType, {\n      codeDirectory: path.join(__dirname, 'edge-function'),\n      runtime: CustomResourceProviderRuntime.NODEJS_12_X,\n      policyStatements: [{\n        Effect: 'Allow',\n        Resource: parameterArnPrefix,\n        Action: ['ssm:GetParameter'],\n      }],\n    });\n    const resource = new CustomResource(this, 'ArnReader', {\n      resourceType: resourceType,\n      serviceToken,\n      properties: {\n        Region: EdgeFunction.EDGE_REGION,\n        ParameterName: parameterName,\n        // This is used to determine when the function has changed, to refresh the ARN from the custom resource.\n        //\n        // Use the logical id of the function version. Whenever a function version changes, the logical id must be\n        // changed for it to take effect - a good candidate for RefreshToken.\n        RefreshToken: Lazy.uncachedString({\n          produce: () => {\n            const cfn = version.node.defaultChild as CfnResource;\n            return this.stack.resolve(cfn.logicalId);\n          },\n        }),\n      },\n    });\n\n    return resource.getAttString('FunctionArn');\n  }\n\n  private edgeStack(stackId?: string): Stack {\n    const stage = Stage.of(this);\n    if (!stage) {\n      throw new Error('stacks which use EdgeFunctions must be part of a CDK app or stage');\n    }\n\n    const edgeStackId = stackId ?? `edge-lambda-stack-${this.stack.node.addr}`;\n    let edgeStack = stage.node.tryFindChild(edgeStackId) as Stack;\n    if (!edgeStack) {\n      edgeStack = new Stack(stage, edgeStackId, {\n        env: {\n          region: EdgeFunction.EDGE_REGION,\n          account: Stack.of(this).account,\n        },\n      });\n    }\n    this.stack.addDependency(edgeStack);\n    return edgeStack;\n  }\n}\n\n/** Result of creating an in-region or cross-region function */\ninterface FunctionConfig {\n  readonly edgeFunction: lambda.Function;\n  readonly edgeArn: string;\n}\n\nfunction addEdgeLambdaToRoleTrustStatement(role: iam.IRole) {\n  if (role instanceof iam.Role && role.assumeRolePolicy) {\n    const statement = new iam.PolicyStatement();\n    const edgeLambdaServicePrincipal = new iam.ServicePrincipal('edgelambda.amazonaws.com');\n    statement.addPrincipals(edgeLambdaServicePrincipal);\n    statement.addActions(edgeLambdaServicePrincipal.assumeRoleAction);\n    role.assumeRolePolicy.addStatements(statement);\n  }\n}\n"]}