UNPKG

16.1 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3exports.VERSION_LOCKED = exports.trimFromStart = exports.calculateFunctionHash = void 0;
4const crypto = require("crypto");
5const core_1 = require("@aws-cdk/core");
6const cx_api_1 = require("@aws-cdk/cx-api");
7const function_1 = require("./function");
8function calculateFunctionHash(fn) {
9 const stack = core_1.Stack.of(fn);
10 const functionResource = fn.node.defaultChild;
11 // render the cloudformation resource from this function
12 const config = stack.resolve(functionResource._toCloudFormation());
13 // config is of the shape: { Resources: { LogicalId: { Type: 'Function', Properties: { ... } }}}
14 const resources = config.Resources;
15 const resourceKeys = Object.keys(resources);
16 if (resourceKeys.length !== 1) {
17 throw new Error(`Expected one rendered CloudFormation resource but found ${resourceKeys.length}`);
18 }
19 const logicalId = resourceKeys[0];
20 const properties = resources[logicalId].Properties;
21 let stringifiedConfig;
22 if (core_1.FeatureFlags.of(fn).isEnabled(cx_api_1.LAMBDA_RECOGNIZE_VERSION_PROPS)) {
23 const updatedProps = sortProperties(filterUsefulKeys(properties));
24 stringifiedConfig = JSON.stringify(updatedProps);
25 }
26 else {
27 const sorted = sortProperties(properties);
28 config.Resources[logicalId].Properties = sorted;
29 stringifiedConfig = JSON.stringify(config);
30 }
31 const hash = crypto.createHash('md5');
32 hash.update(stringifiedConfig);
33 return hash.digest('hex');
34}
35exports.calculateFunctionHash = calculateFunctionHash;
36function trimFromStart(s, maxLength) {
37 const desiredLength = Math.min(maxLength, s.length);
38 const newStart = s.length - desiredLength;
39 return s.substring(newStart);
40}
41exports.trimFromStart = trimFromStart;
42/*
43 * The list of properties found in CfnFunction (or AWS::Lambda::Function).
44 * They are classified as "locked" to a Function Version or not.
45 * When a property is locked, any change to that property will not take effect on previously created Versions.
46 * Instead, a new Version must be generated for the change to take effect.
47 * Similarly, if a property that's not locked to a Version is modified, a new Version
48 * must not be generated.
49 *
50 * Adding a new property to this list - If the property is part of the UpdateFunctionConfiguration
51 * API or UpdateFunctionCode API, then it must be classified as true, otherwise false.
52 * See https://docs.aws.amazon.com/lambda/latest/dg/API_UpdateFunctionConfiguration.html and
53 * https://docs.aws.amazon.com/lambda/latest/dg/API_UpdateFunctionConfiguration.html
54 */
55exports.VERSION_LOCKED = {
56 // locked to the version
57 Architectures: true,
58 Code: true,
59 DeadLetterConfig: true,
60 Description: true,
61 Environment: true,
62 EphemeralStorage: true,
63 FileSystemConfigs: true,
64 FunctionName: true,
65 Handler: true,
66 ImageConfig: true,
67 KmsKeyArn: true,
68 Layers: true,
69 MemorySize: true,
70 PackageType: true,
71 Role: true,
72 Runtime: true,
73 Timeout: true,
74 TracingConfig: true,
75 VpcConfig: true,
76 // not locked to the version
77 CodeSigningConfigArn: false,
78 ReservedConcurrentExecutions: false,
79 Tags: false,
80};
81function filterUsefulKeys(properties) {
82 const versionProps = { ...exports.VERSION_LOCKED, ...function_1.Function._VER_PROPS };
83 const unclassified = Object.entries(properties)
84 .filter(([k, v]) => v != null && !Object.keys(versionProps).includes(k))
85 .map(([k, _]) => k);
86 if (unclassified.length > 0) {
87 throw new Error(`The following properties are not recognized as version properties: [${unclassified}].`
88 + ' See the README of the aws-lambda module to learn more about this and to fix it.');
89 }
90 const notLocked = Object.entries(versionProps).filter(([_, v]) => !v).map(([k, _]) => k);
91 notLocked.forEach(p => delete properties[p]);
92 const ret = {};
93 Object.entries(properties).filter(([k, _]) => versionProps[k]).forEach(([k, v]) => ret[k] = v);
94 return ret;
95}
96function sortProperties(properties) {
97 const ret = {};
98 // We take all required properties in the order that they were historically,
99 // to make sure the hash we calculate is stable.
100 // There cannot be more required properties added in the future,
101 // as that would be a backwards-incompatible change.
102 const requiredProperties = ['Code', 'Handler', 'Role', 'Runtime'];
103 for (const requiredProperty of requiredProperties) {
104 ret[requiredProperty] = properties[requiredProperty];
105 }
106 // then, add all of the non-required properties,
107 // in the original order
108 for (const property of Object.keys(properties)) {
109 if (requiredProperties.indexOf(property) === -1) {
110 ret[property] = properties[property];
111 }
112 }
113 return ret;
114}
115//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"function-hash.js","sourceRoot":"","sources":["function-hash.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport { CfnResource, FeatureFlags, Stack } from '@aws-cdk/core';\nimport { LAMBDA_RECOGNIZE_VERSION_PROPS } from '@aws-cdk/cx-api';\nimport { Function as LambdaFunction } from './function';\n\nexport function calculateFunctionHash(fn: LambdaFunction) {\n  const stack = Stack.of(fn);\n\n  const functionResource = fn.node.defaultChild as CfnResource;\n\n  // render the cloudformation resource from this function\n  const config = stack.resolve((functionResource as any)._toCloudFormation());\n  // config is of the shape: { Resources: { LogicalId: { Type: 'Function', Properties: { ... } }}}\n  const resources = config.Resources;\n  const resourceKeys = Object.keys(resources);\n  if (resourceKeys.length !== 1) {\n    throw new Error(`Expected one rendered CloudFormation resource but found ${resourceKeys.length}`);\n  }\n  const logicalId = resourceKeys[0];\n  const properties = resources[logicalId].Properties;\n\n  let stringifiedConfig;\n  if (FeatureFlags.of(fn).isEnabled(LAMBDA_RECOGNIZE_VERSION_PROPS)) {\n    const updatedProps = sortProperties(filterUsefulKeys(properties));\n    stringifiedConfig = JSON.stringify(updatedProps);\n  } else {\n    const sorted = sortProperties(properties);\n    config.Resources[logicalId].Properties = sorted;\n    stringifiedConfig = JSON.stringify(config);\n  }\n\n  const hash = crypto.createHash('md5');\n  hash.update(stringifiedConfig);\n  return hash.digest('hex');\n}\n\nexport function trimFromStart(s: string, maxLength: number) {\n  const desiredLength = Math.min(maxLength, s.length);\n  const newStart = s.length - desiredLength;\n  return s.substring(newStart);\n}\n\n/*\n * The list of properties found in CfnFunction (or AWS::Lambda::Function).\n * They are classified as \"locked\" to a Function Version or not.\n * When a property is locked, any change to that property will not take effect on previously created Versions.\n * Instead, a new Version must be generated for the change to take effect.\n * Similarly, if a property that's not locked to a Version is modified, a new Version\n * must not be generated.\n *\n * Adding a new property to this list - If the property is part of the UpdateFunctionConfiguration\n * API or UpdateFunctionCode API, then it must be classified as true, otherwise false.\n * See https://docs.aws.amazon.com/lambda/latest/dg/API_UpdateFunctionConfiguration.html and\n * https://docs.aws.amazon.com/lambda/latest/dg/API_UpdateFunctionConfiguration.html\n */\nexport const VERSION_LOCKED: { [key: string]: boolean } = {\n  // locked to the version\n  Architectures: true,\n  Code: true,\n  DeadLetterConfig: true,\n  Description: true,\n  Environment: true,\n  EphemeralStorage: true,\n  FileSystemConfigs: true,\n  FunctionName: true,\n  Handler: true,\n  ImageConfig: true,\n  KmsKeyArn: true,\n  Layers: true,\n  MemorySize: true,\n  PackageType: true,\n  Role: true,\n  Runtime: true,\n  Timeout: true,\n  TracingConfig: true,\n  VpcConfig: true,\n\n  // not locked to the version\n  CodeSigningConfigArn: false,\n  ReservedConcurrentExecutions: false,\n  Tags: false,\n};\n\nfunction filterUsefulKeys(properties: any) {\n  const versionProps = { ...VERSION_LOCKED, ...LambdaFunction._VER_PROPS };\n  const unclassified = Object.entries(properties)\n    .filter(([k, v]) => v != null && !Object.keys(versionProps).includes(k))\n    .map(([k, _]) => k);\n  if (unclassified.length > 0) {\n    throw new Error(`The following properties are not recognized as version properties: [${unclassified}].`\n      + ' See the README of the aws-lambda module to learn more about this and to fix it.');\n  }\n  const notLocked = Object.entries(versionProps).filter(([_, v]) => !v).map(([k, _]) => k);\n  notLocked.forEach(p => delete properties[p]);\n\n  const ret: { [key: string]: any } = {};\n  Object.entries(properties).filter(([k, _]) => versionProps[k]).forEach(([k, v]) => ret[k] = v);\n  return ret;\n}\n\nfunction sortProperties(properties: any) {\n  const ret: any = {};\n  // We take all required properties in the order that they were historically,\n  // to make sure the hash we calculate is stable.\n  // There cannot be more required properties added in the future,\n  // as that would be a backwards-incompatible change.\n  const requiredProperties = ['Code', 'Handler', 'Role', 'Runtime'];\n  for (const requiredProperty of requiredProperties) {\n    ret[requiredProperty] = properties[requiredProperty];\n  }\n  // then, add all of the non-required properties,\n  // in the original order\n  for (const property of Object.keys(properties)) {\n    if (requiredProperties.indexOf(property) === -1) {\n      ret[property] = properties[property];\n    }\n  }\n  return ret;\n}\n"]}
\No newline at end of file