@aws-cdk/core
Version:
AWS Cloud Development Kit Core Library
129 lines • 18.7 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.CustomResourceProvider = exports.CustomResourceProviderRuntime = void 0;
const fs = require("fs");
const path = require("path");
const asset_staging_1 = require("../asset-staging");
const assets_1 = require("../assets");
const cfn_resource_1 = require("../cfn-resource");
const duration_1 = require("../duration");
const size_1 = require("../size");
const stack_1 = require("../stack");
const token_1 = require("../token");
const ENTRYPOINT_FILENAME = '__entrypoint__';
const ENTRYPOINT_NODEJS_SOURCE = path.join(__dirname, 'nodejs-entrypoint.js');
// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.
// eslint-disable-next-line
const construct_compat_1 = require("../construct-compat");
/**
* The lambda runtime to use for the resource provider. This also indicates
* which language is used for the handler.
* @experimental
*/
var CustomResourceProviderRuntime;
(function (CustomResourceProviderRuntime) {
CustomResourceProviderRuntime["NODEJS_12"] = "nodejs12";
})(CustomResourceProviderRuntime = exports.CustomResourceProviderRuntime || (exports.CustomResourceProviderRuntime = {}));
/**
* (experimental) An AWS-Lambda backed custom resource provider.
*
* @experimental
*/
class CustomResourceProvider extends construct_compat_1.Construct {
/**
* @experimental
*/
constructor(scope, id, props) {
var _a, _b;
super(scope, id);
const stack = stack_1.Stack.of(scope);
// copy the entry point to the code directory
fs.copyFileSync(ENTRYPOINT_NODEJS_SOURCE, path.join(props.codeDirectory, `${ENTRYPOINT_FILENAME}.js`));
// verify we have an index file there
if (!fs.existsSync(path.join(props.codeDirectory, 'index.js'))) {
throw new Error(`cannot find ${props.codeDirectory}/index.js`);
}
const staging = new asset_staging_1.AssetStaging(this, 'Staging', {
sourcePath: props.codeDirectory,
});
const asset = stack.addFileAsset({
fileName: staging.stagedPath,
sourceHash: staging.sourceHash,
packaging: assets_1.FileAssetPackaging.ZIP_DIRECTORY,
});
const policies = !props.policyStatements ? undefined : [
{
PolicyName: 'Inline',
PolicyDocument: {
Version: '2012-10-17',
Statement: props.policyStatements,
},
},
];
const role = new cfn_resource_1.CfnResource(this, 'Role', {
type: 'AWS::IAM::Role',
properties: {
AssumeRolePolicyDocument: {
Version: '2012-10-17',
Statement: [{ Action: 'sts:AssumeRole', Effect: 'Allow', Principal: { Service: 'lambda.amazonaws.com' } }],
},
ManagedPolicyArns: [
{ 'Fn::Sub': 'arn:${AWS::Partition}:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole' },
],
Policies: policies,
},
});
const timeout = (_a = props.timeout) !== null && _a !== void 0 ? _a : duration_1.Duration.minutes(15);
const memory = (_b = props.memorySize) !== null && _b !== void 0 ? _b : size_1.Size.mebibytes(128);
const handler = new cfn_resource_1.CfnResource(this, 'Handler', {
type: 'AWS::Lambda::Function',
properties: {
Code: {
S3Bucket: asset.bucketName,
S3Key: asset.objectKey,
},
Timeout: timeout.toSeconds(),
MemorySize: memory.toMebibytes(),
Handler: `${ENTRYPOINT_FILENAME}.handler`,
Role: role.getAtt('Arn'),
Runtime: 'nodejs12.x',
Environment: this.renderEnvironmentVariables(props.environment),
},
});
handler.addDependsOn(role);
this.serviceToken = token_1.Token.asString(handler.getAtt('Arn'));
}
/**
* (experimental) Returns a stack-level singleton ARN (service token) for the custom resource provider.
*
* @param scope Construct scope.
* @param uniqueid A globally unique id that will be used for the stack-level construct.
* @param props Provider properties which will only be applied when the provider is first created.
* @returns the service token of the custom resource provider, which should be
* used when defining a `CustomResource`.
* @experimental
*/
static getOrCreate(scope, uniqueid, props) {
var _a;
const id = `${uniqueid}CustomResourceProvider`;
const stack = stack_1.Stack.of(scope);
const provider = (_a = stack.node.tryFindChild(id)) !== null && _a !== void 0 ? _a : new CustomResourceProvider(stack, id, props);
return provider.serviceToken;
}
renderEnvironmentVariables(env) {
if (!env || Object.keys(env).length === 0) {
return undefined;
}
// Sort environment so the hash of the function used to create
// `currentVersion` is not affected by key order (this is how lambda does
// it)
const variables = {};
const keys = Object.keys(env).sort();
for (const key of keys) {
variables[key] = env[key];
}
return { Variables: variables };
}
}
exports.CustomResourceProvider = CustomResourceProvider;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-resource-provider.js","sourceRoot":"","sources":["custom-resource-provider.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport { Construct } from 'constructs';\nimport { AssetStaging } from '../asset-staging';\nimport { FileAssetPackaging } from '../assets';\nimport { CfnResource } from '../cfn-resource';\nimport { Duration } from '../duration';\nimport { Size } from '../size';\nimport { Stack } from '../stack';\nimport { Token } from '../token';\n\nconst ENTRYPOINT_FILENAME = '__entrypoint__';\nconst ENTRYPOINT_NODEJS_SOURCE = path.join(__dirname, 'nodejs-entrypoint.js');\n\n// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.\n// eslint-disable-next-line\nimport { Construct as CoreConstruct } from '../construct-compat';\n\n                                                                                      \nexport interface CustomResourceProviderProps {\n                                                                                                                                                                        \n  readonly codeDirectory: string;\n\n                                                                            \n  readonly runtime: CustomResourceProviderRuntime;\n\n                                                                                                                                                                                                                                                                                              \n  readonly policyStatements?: any[];\n\n                                                                                             \n  readonly timeout?: Duration;\n\n                                                                                                                                                                                       \n  readonly memorySize?: Size;\n\n                                                                                                                        \n  readonly environment?: { [key: string]: string };\n}\n\n/**\n * The lambda runtime to use for the resource provider. This also indicates\n * which language is used for the handler.\n * @experimental\n */\nexport enum CustomResourceProviderRuntime {\n                             \n  NODEJS_12 = 'nodejs12'\n}\n\n                                                                             \nexport class CustomResourceProvider extends CoreConstruct {\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         \n  public static getOrCreate(scope: Construct, uniqueid: string, props: CustomResourceProviderProps) {\n    const id = `${uniqueid}CustomResourceProvider`;\n    const stack = Stack.of(scope);\n    const provider = stack.node.tryFindChild(id) as CustomResourceProvider\n      ?? new CustomResourceProvider(stack, id, props);\n\n    return provider.serviceToken;\n  }\n\n                                                                                                                                                                                                                                                                                                                      \n  public readonly serviceToken: string;\n\n  protected constructor(scope: Construct, id: string, props: CustomResourceProviderProps) {\n    super(scope, id);\n\n    const stack = Stack.of(scope);\n\n    // copy the entry point to the code directory\n    fs.copyFileSync(ENTRYPOINT_NODEJS_SOURCE, path.join(props.codeDirectory, `${ENTRYPOINT_FILENAME}.js`));\n\n    // verify we have an index file there\n    if (!fs.existsSync(path.join(props.codeDirectory, 'index.js'))) {\n      throw new Error(`cannot find ${props.codeDirectory}/index.js`);\n    }\n\n    const staging = new AssetStaging(this, 'Staging', {\n      sourcePath: props.codeDirectory,\n    });\n\n    const asset = stack.addFileAsset({\n      fileName: staging.stagedPath,\n      sourceHash: staging.sourceHash,\n      packaging: FileAssetPackaging.ZIP_DIRECTORY,\n    });\n\n    const policies = !props.policyStatements ? undefined : [\n      {\n        PolicyName: 'Inline',\n        PolicyDocument: {\n          Version: '2012-10-17',\n          Statement: props.policyStatements,\n        },\n      },\n    ];\n\n    const role = new CfnResource(this, 'Role', {\n      type: 'AWS::IAM::Role',\n      properties: {\n        AssumeRolePolicyDocument: {\n          Version: '2012-10-17',\n          Statement: [{ Action: 'sts:AssumeRole', Effect: 'Allow', Principal: { Service: 'lambda.amazonaws.com' } }],\n        },\n        ManagedPolicyArns: [\n          { 'Fn::Sub': 'arn:${AWS::Partition}:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole' },\n        ],\n        Policies: policies,\n      },\n    });\n\n    const timeout = props.timeout ?? Duration.minutes(15);\n    const memory = props.memorySize ?? Size.mebibytes(128);\n\n    const handler = new CfnResource(this, 'Handler', {\n      type: 'AWS::Lambda::Function',\n      properties: {\n        Code: {\n          S3Bucket: asset.bucketName,\n          S3Key: asset.objectKey,\n        },\n        Timeout: timeout.toSeconds(),\n        MemorySize: memory.toMebibytes(),\n        Handler: `${ENTRYPOINT_FILENAME}.handler`,\n        Role: role.getAtt('Arn'),\n        Runtime: 'nodejs12.x',\n        Environment: this.renderEnvironmentVariables(props.environment),\n      },\n    });\n\n    handler.addDependsOn(role);\n\n    this.serviceToken = Token.asString(handler.getAtt('Arn'));\n  }\n\n  private renderEnvironmentVariables(env?: { [key: string]: string }) {\n    if (!env || Object.keys(env).length === 0) {\n      return undefined;\n    }\n\n    // Sort environment so the hash of the function used to create\n    // `currentVersion` is not affected by key order (this is how lambda does\n    // it)\n    const variables: { [key: string]: string } = {};\n    const keys = Object.keys(env).sort();\n\n    for (const key of keys) {\n      variables[key] = env[key];\n    }\n\n    return { Variables: variables };\n  }\n}\n"]}