@sylvesterllc/aws-constructs 1.1.25 → 1.1.26

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -9,7 +9,7 @@ exports.config = {
9
9
  name: `${process.env.APP_NAME}`,
10
10
  accountNumber: process.env.CDK_DEFAULT_ACCOUNT || "",
11
11
  region: process.env.CDK_DEFAULT_REGION || "us-east-1",
12
- stackRuntime: aws_lambda_1.Runtime.NODEJS_22_X,
12
+ stackRuntime: aws_lambda_1.Runtime.NODEJS_LATEST,
13
13
  },
14
14
  API: {
15
15
  Name: `${process.env.APP_NAME}-auth-api`,
@@ -79,4 +79,4 @@ exports.config = {
79
79
  FQDN: ''
80
80
  }
81
81
  };
82
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWljcm9TZXJ2aWNlQ29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbmZpZy9taWNyb1NlcnZpY2VDb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsMkRBQXNGO0FBQ3RGLHVEQUFpRDtBQUNqRCwyQ0FBd0M7QUFHM0IsUUFBQSxNQUFNLEdBQWU7SUFDOUIsT0FBTyxFQUFFO1FBQ0wsSUFBSSxFQUFFLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxRQUFRLEVBQUU7UUFDL0IsYUFBYSxFQUFFLE9BQU8sQ0FBQyxHQUFHLENBQUMsbUJBQW1CLElBQUksRUFBRTtRQUNwRCxNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0IsSUFBSSxXQUFXO1FBQ3JELFlBQVksRUFBRSxvQkFBTyxDQUFDLFdBQVc7S0FDcEM7SUFDRCxHQUFHLEVBQUU7UUFDRCxJQUFJLEVBQUUsR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLFFBQVEsV0FBVztRQUN4QyxXQUFXLEVBQUUsb0JBQW9CO1FBQ2pDLFlBQVksRUFBRSxlQUFlO0tBQ2hDO0lBQ0QsU0FBUyxFQUFFO1FBQ1AsTUFBTSxFQUFFO1lBQ0o7Z0JBQ0ksSUFBSSxFQUFFLGdCQUFnQjtnQkFDdEIsUUFBUSxFQUFFLDBDQUEwQztnQkFDcEQsT0FBTyxFQUFFLE1BQU07Z0JBQ2YsVUFBVSxFQUFFO29CQUNSLEtBQUssRUFBRSx5QkFBeUI7b0JBQ2hDLE1BQU0sRUFBRSxNQUFNO2lCQUNqQjthQUVKO1lBQ0Q7Z0JBQ0ksSUFBSSxFQUFFLGlCQUFpQjtnQkFDdkIsUUFBUSxFQUFFLDJDQUEyQztnQkFDckQsT0FBTyxFQUFFLE1BQU07Z0JBQ2YsVUFBVSxFQUFFO29CQUNSLEtBQUssRUFBRSwwQkFBMEI7b0JBQ2pDLE1BQU0sRUFBRSxNQUFNO29CQUNkLE9BQU8sRUFBRSxDQUFDO2lCQUNiO2FBQ0o7U0FDSjtRQUVELE1BQU0sRUFBRTtZQUNKLE1BQU0sRUFBRTtnQkFDSjtvQkFDSSxTQUFTLEVBQUUsR0FBRyxxQkFBUyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsa0JBQWtCLENBQUMsSUFBSSxFQUFFO29CQUNqRSxVQUFVLEVBQUU7d0JBQ1IsSUFBSSxFQUFFLElBQUk7d0JBQ1YsSUFBSSxFQUFFLDRCQUFhLENBQUMsTUFBTTtxQkFDN0I7b0JBQ0QsV0FBVyxFQUFFLDBCQUFXLENBQUMsZUFBZTtvQkFDeEMsT0FBTyxFQUFFO3dCQUNMOzRCQUNJLFNBQVMsRUFBRSxxQkFBUyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsa0JBQWtCLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxJQUFJOzRCQUNsRixZQUFZLEVBQUU7Z0NBQ1YsSUFBSSxFQUFFLFdBQVc7Z0NBQ2pCLElBQUksRUFBRSw0QkFBYSxDQUFDLE1BQU07NkJBQzdCOzRCQUNELGNBQWMsRUFBRSw2QkFBYyxDQUFDLEdBQUc7eUJBQ3JDO3dCQUNEOzRCQUNJLFNBQVMsRUFBRSxxQkFBUyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsa0JBQWtCLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxJQUFJOzRCQUM3RSxZQUFZLEVBQUU7Z0NBQ1YsSUFBSSxFQUFFLFVBQVU7Z0NBQ2hCLElBQUksRUFBRSw0QkFBYSxDQUFDLE1BQU07NkJBQzdCOzRCQUNELGNBQWMsRUFBRSw2QkFBYyxDQUFDLEdBQUc7eUJBQ3JDO3FCQUNKO2lCQUNKO2FBQ0o7U0FDSjtLQUVKO0lBQ0QsR0FBRyxFQUFFO1FBQ0QsUUFBUSxFQUFFLEVBQUU7UUFDWixNQUFNLEVBQUUsRUFBRTtRQUNWLHFCQUFxQixFQUFFLFVBQVU7UUFDakMscUJBQXFCLEVBQUUsVUFBVTtRQUNqQyxTQUFTLEVBQUUsSUFBSTtRQUNmLFFBQVEsRUFBRSxFQUFFO1FBQ1osSUFBSSxFQUFFLEVBQUU7S0FFWDtDQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBdHRyaWJ1dGVUeXBlLCBCaWxsaW5nTW9kZSwgUHJvamVjdGlvblR5cGUgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWR5bmFtb2RiXCI7XHJcbmltcG9ydCB7IFJ1bnRpbWUgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWxhbWJkYVwiO1xyXG5pbXBvcnQgeyBDT05TVEFOVFMgfSBmcm9tIFwiLi9Db25zdGFudHNcIjtcclxuaW1wb3J0IHsgSUFwcENvbmZpZyB9IGZyb20gXCIuL2N1c3RvbUNvbmZpZ3MvSUFwcENvbmZpZ1wiO1xyXG5cclxuZXhwb3J0IGNvbnN0IGNvbmZpZzogSUFwcENvbmZpZyA9IHtcclxuICAgIEdMT0JBTFM6IHtcclxuICAgICAgICBuYW1lOiBgJHtwcm9jZXNzLmVudi5BUFBfTkFNRX1gLFxyXG4gICAgICAgIGFjY291bnROdW1iZXI6IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX0FDQ09VTlQgfHwgXCJcIixcclxuICAgICAgICByZWdpb246IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX1JFR0lPTiB8fCBcInVzLWVhc3QtMVwiLFxyXG4gICAgICAgIHN0YWNrUnVudGltZTogUnVudGltZS5OT0RFSlNfMjJfWCxcclxuICAgIH0sXHJcbiAgICBBUEk6IHtcclxuICAgICAgICBOYW1lOiBgJHtwcm9jZXNzLmVudi5BUFBfTkFNRX0tYXV0aC1hcGlgLFxyXG4gICAgICAgIERlc2NyaXB0aW9uOiAnVGhpcyBpcyBteSBuZXcgQVBJJyxcclxuICAgICAgICBEb21haW5QcmVmaXg6ICdteS1jdXN0b20tYXBpJ1xyXG4gICAgfSxcclxuICAgIFJFU09VUkNFUzoge1xyXG4gICAgICAgIExBTUJEQTogW1xyXG4gICAgICAgICAgICB7XHJcbiAgICAgICAgICAgICAgICBuYW1lOiBgY3JlYXRlLWFjY291bnRgLFxyXG4gICAgICAgICAgICAgICAgY29kZVBhdGg6ICcuL2xhbWJkYS1mdW5jdGlvbnMvYXV0aC9jcmVhdGVBY2NvdW50LnRzJyxcclxuICAgICAgICAgICAgICAgIGhhbmRsZXI6ICdtYWluJyxcclxuICAgICAgICAgICAgICAgIGFwaUdhdGV3YXk6IHtcclxuICAgICAgICAgICAgICAgICAgICByb3V0ZTogJy9hY2NvdW50L2NyZWF0ZS1hY2NvdW50JyxcclxuICAgICAgICAgICAgICAgICAgICBtZXRob2Q6ICdwb3N0JywgICAgICAgICAgICAgICAgICAgIFxyXG4gICAgICAgICAgICAgICAgfSxcclxuXHJcbiAgICAgICAgICAgIH0sXHJcbiAgICAgICAgICAgIHtcclxuICAgICAgICAgICAgICAgIG5hbWU6IGBjaGFuZ2UtcGFzc3dvcmRgLFxyXG4gICAgICAgICAgICAgICAgY29kZVBhdGg6ICcuL2xhbWJkYS1mdW5jdGlvbnMvYXV0aC9jaGFuZ2VQYXNzd29yZC50cycsXHJcbiAgICAgICAgICAgICAgICBoYW5kbGVyOiAnbWFpbicsXHJcbiAgICAgICAgICAgICAgICBhcGlHYXRld2F5OiB7XHJcbiAgICAgICAgICAgICAgICAgICAgcm91dGU6ICcvYWNjb3VudC9jaGFuZ2UtcGFzc3dvcmQnLFxyXG4gICAgICAgICAgICAgICAgICAgIG1ldGhvZDogJ3Bvc3QnLFxyXG4gICAgICAgICAgICAgICAgICAgIHZlcnNpb246IDJcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgfSxcclxuICAgICAgICBdLFxyXG4gICAgICAgIFxyXG4gICAgICAgIERZTkFNTzoge1xyXG4gICAgICAgICAgICBUQUJMRVM6IFtcclxuICAgICAgICAgICAgICAgIHtcclxuICAgICAgICAgICAgICAgICAgICB0YWJsZU5hbWU6IGAke0NPTlNUQU5UUy5EWU5BTU9EQi5UQUJMRVMuQVVUSF9ISVNUT1JZX1RBQkxFLm5hbWV9YCxcclxuICAgICAgICAgICAgICAgICAgICBwcmltYXJ5S2V5OiB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIG5hbWU6ICdpZCcsXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU6IEF0dHJpYnV0ZVR5cGUuU1RSSU5HLFxyXG4gICAgICAgICAgICAgICAgICAgIH0sXHJcbiAgICAgICAgICAgICAgICAgICAgYmlsbGluZ01vZGU6IEJpbGxpbmdNb2RlLlBBWV9QRVJfUkVRVUVTVCxcclxuICAgICAgICAgICAgICAgICAgICBpbmRleGVzOiBbXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGluZGV4TmFtZTogQ09OU1RBTlRTLkRZTkFNT0RCLlRBQkxFUy5BVVRIX0hJU1RPUllfVEFCTEUuaW5kZXhlcy5BdXRoSGlzdG9yeVRTLm5hbWUsXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBwYXJ0aXRpb25LZXk6IHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYW1lOiAnY3JlYXRlZFRTJyxcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0eXBlOiBBdHRyaWJ1dGVUeXBlLk5VTUJFUlxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHByb2plY3Rpb25UeXBlOiBQcm9qZWN0aW9uVHlwZS5BTExcclxuICAgICAgICAgICAgICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgICAgICAgICAgICAge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW5kZXhOYW1lOiBDT05TVEFOVFMuRFlOQU1PREIuVEFCTEVTLkFVVEhfSElTVE9SWV9UQUJMRS5pbmRleGVzLlVzZXJuYW1lLm5hbWUsXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBwYXJ0aXRpb25LZXk6IHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYW1lOiAndXNlcm5hbWUnLFxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU6IEF0dHJpYnV0ZVR5cGUuU1RSSU5HXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LFxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgcHJvamVjdGlvblR5cGU6IFByb2plY3Rpb25UeXBlLkFMTFxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9LFxyXG4gICAgICAgICAgICAgICAgICAgIF1cclxuICAgICAgICAgICAgICAgIH0sXHJcbiAgICAgICAgICAgIF0sXHJcbiAgICAgICAgfSxcclxuXHJcbiAgICB9LFxyXG4gICAgRE5TOiB7XHJcbiAgICAgICAgWm9uZU5hbWU6ICcnLFxyXG4gICAgICAgIFpvbmVJZDogJycsXHJcbiAgICAgICAgWm9uZU5hbWVXaXRob3V0UGVyaW9kOiAnbm90LXVzZWQnLFxyXG4gICAgICAgIFpvbmVOYW1lV2l0aG91dFN1ZmZpeDogJ25vdC11c2VkJyxcclxuICAgICAgICBab25lRXhpc3Q6IHRydWUsXHJcbiAgICAgICAgSG9zdE5hbWU6ICcnLFxyXG4gICAgICAgIEZRRE46ICcnXHJcblxyXG4gICAgfVxyXG59OyAiXX0=
82
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWljcm9TZXJ2aWNlQ29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbmZpZy9taWNyb1NlcnZpY2VDb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsMkRBQXNGO0FBQ3RGLHVEQUFpRDtBQUNqRCwyQ0FBd0M7QUFHM0IsUUFBQSxNQUFNLEdBQWU7SUFDOUIsT0FBTyxFQUFFO1FBQ0wsSUFBSSxFQUFFLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxRQUFRLEVBQUU7UUFDL0IsYUFBYSxFQUFFLE9BQU8sQ0FBQyxHQUFHLENBQUMsbUJBQW1CLElBQUksRUFBRTtRQUNwRCxNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0IsSUFBSSxXQUFXO1FBQ3JELFlBQVksRUFBRSxvQkFBTyxDQUFDLGFBQWE7S0FDdEM7SUFDRCxHQUFHLEVBQUU7UUFDRCxJQUFJLEVBQUUsR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLFFBQVEsV0FBVztRQUN4QyxXQUFXLEVBQUUsb0JBQW9CO1FBQ2pDLFlBQVksRUFBRSxlQUFlO0tBQ2hDO0lBQ0QsU0FBUyxFQUFFO1FBQ1AsTUFBTSxFQUFFO1lBQ0o7Z0JBQ0ksSUFBSSxFQUFFLGdCQUFnQjtnQkFDdEIsUUFBUSxFQUFFLDBDQUEwQztnQkFDcEQsT0FBTyxFQUFFLE1BQU07Z0JBQ2YsVUFBVSxFQUFFO29CQUNSLEtBQUssRUFBRSx5QkFBeUI7b0JBQ2hDLE1BQU0sRUFBRSxNQUFNO2lCQUNqQjthQUVKO1lBQ0Q7Z0JBQ0ksSUFBSSxFQUFFLGlCQUFpQjtnQkFDdkIsUUFBUSxFQUFFLDJDQUEyQztnQkFDckQsT0FBTyxFQUFFLE1BQU07Z0JBQ2YsVUFBVSxFQUFFO29CQUNSLEtBQUssRUFBRSwwQkFBMEI7b0JBQ2pDLE1BQU0sRUFBRSxNQUFNO29CQUNkLE9BQU8sRUFBRSxDQUFDO2lCQUNiO2FBQ0o7U0FDSjtRQUVELE1BQU0sRUFBRTtZQUNKLE1BQU0sRUFBRTtnQkFDSjtvQkFDSSxTQUFTLEVBQUUsR0FBRyxxQkFBUyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsa0JBQWtCLENBQUMsSUFBSSxFQUFFO29CQUNqRSxVQUFVLEVBQUU7d0JBQ1IsSUFBSSxFQUFFLElBQUk7d0JBQ1YsSUFBSSxFQUFFLDRCQUFhLENBQUMsTUFBTTtxQkFDN0I7b0JBQ0QsV0FBVyxFQUFFLDBCQUFXLENBQUMsZUFBZTtvQkFDeEMsT0FBTyxFQUFFO3dCQUNMOzRCQUNJLFNBQVMsRUFBRSxxQkFBUyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsa0JBQWtCLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxJQUFJOzRCQUNsRixZQUFZLEVBQUU7Z0NBQ1YsSUFBSSxFQUFFLFdBQVc7Z0NBQ2pCLElBQUksRUFBRSw0QkFBYSxDQUFDLE1BQU07NkJBQzdCOzRCQUNELGNBQWMsRUFBRSw2QkFBYyxDQUFDLEdBQUc7eUJBQ3JDO3dCQUNEOzRCQUNJLFNBQVMsRUFBRSxxQkFBUyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsa0JBQWtCLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxJQUFJOzRCQUM3RSxZQUFZLEVBQUU7Z0NBQ1YsSUFBSSxFQUFFLFVBQVU7Z0NBQ2hCLElBQUksRUFBRSw0QkFBYSxDQUFDLE1BQU07NkJBQzdCOzRCQUNELGNBQWMsRUFBRSw2QkFBYyxDQUFDLEdBQUc7eUJBQ3JDO3FCQUNKO2lCQUNKO2FBQ0o7U0FDSjtLQUVKO0lBQ0QsR0FBRyxFQUFFO1FBQ0QsUUFBUSxFQUFFLEVBQUU7UUFDWixNQUFNLEVBQUUsRUFBRTtRQUNWLHFCQUFxQixFQUFFLFVBQVU7UUFDakMscUJBQXFCLEVBQUUsVUFBVTtRQUNqQyxTQUFTLEVBQUUsSUFBSTtRQUNmLFFBQVEsRUFBRSxFQUFFO1FBQ1osSUFBSSxFQUFFLEVBQUU7S0FFWDtDQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBdHRyaWJ1dGVUeXBlLCBCaWxsaW5nTW9kZSwgUHJvamVjdGlvblR5cGUgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWR5bmFtb2RiXCI7XHJcbmltcG9ydCB7IFJ1bnRpbWUgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWxhbWJkYVwiO1xyXG5pbXBvcnQgeyBDT05TVEFOVFMgfSBmcm9tIFwiLi9Db25zdGFudHNcIjtcclxuaW1wb3J0IHsgSUFwcENvbmZpZyB9IGZyb20gXCIuL2N1c3RvbUNvbmZpZ3MvSUFwcENvbmZpZ1wiO1xyXG5cclxuZXhwb3J0IGNvbnN0IGNvbmZpZzogSUFwcENvbmZpZyA9IHtcclxuICAgIEdMT0JBTFM6IHtcclxuICAgICAgICBuYW1lOiBgJHtwcm9jZXNzLmVudi5BUFBfTkFNRX1gLFxyXG4gICAgICAgIGFjY291bnROdW1iZXI6IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX0FDQ09VTlQgfHwgXCJcIixcclxuICAgICAgICByZWdpb246IHByb2Nlc3MuZW52LkNES19ERUZBVUxUX1JFR0lPTiB8fCBcInVzLWVhc3QtMVwiLFxyXG4gICAgICAgIHN0YWNrUnVudGltZTogUnVudGltZS5OT0RFSlNfTEFURVNULFxyXG4gICAgfSxcclxuICAgIEFQSToge1xyXG4gICAgICAgIE5hbWU6IGAke3Byb2Nlc3MuZW52LkFQUF9OQU1FfS1hdXRoLWFwaWAsXHJcbiAgICAgICAgRGVzY3JpcHRpb246ICdUaGlzIGlzIG15IG5ldyBBUEknLFxyXG4gICAgICAgIERvbWFpblByZWZpeDogJ215LWN1c3RvbS1hcGknXHJcbiAgICB9LFxyXG4gICAgUkVTT1VSQ0VTOiB7XHJcbiAgICAgICAgTEFNQkRBOiBbXHJcbiAgICAgICAgICAgIHtcclxuICAgICAgICAgICAgICAgIG5hbWU6IGBjcmVhdGUtYWNjb3VudGAsXHJcbiAgICAgICAgICAgICAgICBjb2RlUGF0aDogJy4vbGFtYmRhLWZ1bmN0aW9ucy9hdXRoL2NyZWF0ZUFjY291bnQudHMnLFxyXG4gICAgICAgICAgICAgICAgaGFuZGxlcjogJ21haW4nLFxyXG4gICAgICAgICAgICAgICAgYXBpR2F0ZXdheToge1xyXG4gICAgICAgICAgICAgICAgICAgIHJvdXRlOiAnL2FjY291bnQvY3JlYXRlLWFjY291bnQnLFxyXG4gICAgICAgICAgICAgICAgICAgIG1ldGhvZDogJ3Bvc3QnLCAgICAgICAgICAgICAgICAgICAgXHJcbiAgICAgICAgICAgICAgICB9LFxyXG5cclxuICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAge1xyXG4gICAgICAgICAgICAgICAgbmFtZTogYGNoYW5nZS1wYXNzd29yZGAsXHJcbiAgICAgICAgICAgICAgICBjb2RlUGF0aDogJy4vbGFtYmRhLWZ1bmN0aW9ucy9hdXRoL2NoYW5nZVBhc3N3b3JkLnRzJyxcclxuICAgICAgICAgICAgICAgIGhhbmRsZXI6ICdtYWluJyxcclxuICAgICAgICAgICAgICAgIGFwaUdhdGV3YXk6IHtcclxuICAgICAgICAgICAgICAgICAgICByb3V0ZTogJy9hY2NvdW50L2NoYW5nZS1wYXNzd29yZCcsXHJcbiAgICAgICAgICAgICAgICAgICAgbWV0aG9kOiAncG9zdCcsXHJcbiAgICAgICAgICAgICAgICAgICAgdmVyc2lvbjogMlxyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICB9LFxyXG4gICAgICAgIF0sXHJcbiAgICAgICAgXHJcbiAgICAgICAgRFlOQU1POiB7XHJcbiAgICAgICAgICAgIFRBQkxFUzogW1xyXG4gICAgICAgICAgICAgICAge1xyXG4gICAgICAgICAgICAgICAgICAgIHRhYmxlTmFtZTogYCR7Q09OU1RBTlRTLkRZTkFNT0RCLlRBQkxFUy5BVVRIX0hJU1RPUllfVEFCTEUubmFtZX1gLFxyXG4gICAgICAgICAgICAgICAgICAgIHByaW1hcnlLZXk6IHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgbmFtZTogJ2lkJyxcclxuICAgICAgICAgICAgICAgICAgICAgICAgdHlwZTogQXR0cmlidXRlVHlwZS5TVFJJTkcsXHJcbiAgICAgICAgICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgICAgICAgICBiaWxsaW5nTW9kZTogQmlsbGluZ01vZGUuUEFZX1BFUl9SRVFVRVNULFxyXG4gICAgICAgICAgICAgICAgICAgIGluZGV4ZXM6IFtcclxuICAgICAgICAgICAgICAgICAgICAgICAge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgaW5kZXhOYW1lOiBDT05TVEFOVFMuRFlOQU1PREIuVEFCTEVTLkFVVEhfSElTVE9SWV9UQUJMRS5pbmRleGVzLkF1dGhIaXN0b3J5VFMubmFtZSxcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBhcnRpdGlvbktleToge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5hbWU6ICdjcmVhdGVkVFMnLFxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU6IEF0dHJpYnV0ZVR5cGUuTlVNQkVSXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LFxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgcHJvamVjdGlvblR5cGU6IFByb2plY3Rpb25UeXBlLkFMTFxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9LFxyXG4gICAgICAgICAgICAgICAgICAgICAgICB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbmRleE5hbWU6IENPTlNUQU5UUy5EWU5BTU9EQi5UQUJMRVMuQVVUSF9ISVNUT1JZX1RBQkxFLmluZGV4ZXMuVXNlcm5hbWUubmFtZSxcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBhcnRpdGlvbktleToge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5hbWU6ICd1c2VybmFtZScsXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdHlwZTogQXR0cmlidXRlVHlwZS5TVFJJTkdcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBwcm9qZWN0aW9uVHlwZTogUHJvamVjdGlvblR5cGUuQUxMXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIH0sXHJcbiAgICAgICAgICAgICAgICAgICAgXVxyXG4gICAgICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgXSxcclxuICAgICAgICB9LFxyXG5cclxuICAgIH0sXHJcbiAgICBETlM6IHtcclxuICAgICAgICBab25lTmFtZTogJycsXHJcbiAgICAgICAgWm9uZUlkOiAnJyxcclxuICAgICAgICBab25lTmFtZVdpdGhvdXRQZXJpb2Q6ICdub3QtdXNlZCcsXHJcbiAgICAgICAgWm9uZU5hbWVXaXRob3V0U3VmZml4OiAnbm90LXVzZWQnLFxyXG4gICAgICAgIFpvbmVFeGlzdDogdHJ1ZSxcclxuICAgICAgICBIb3N0TmFtZTogJycsXHJcbiAgICAgICAgRlFETjogJydcclxuXHJcbiAgICB9XHJcbn07ICJdfQ==
@@ -5,8 +5,73 @@ const aws_cdk_lib_1 = require("aws-cdk-lib");
5
5
  const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
6
6
  const aws_lambda_nodejs_1 = require("aws-cdk-lib/aws-lambda-nodejs");
7
7
  const aws_logs_1 = require("aws-cdk-lib/aws-logs");
8
+ const fs = require("fs");
8
9
  const path = require("path");
9
10
  const aws_dynamodb_1 = require("aws-cdk-lib/aws-dynamodb");
11
+ const isSubPath = (root, target) => {
12
+ const relative = path.relative(path.resolve(root), path.resolve(target));
13
+ return relative === "" || (!relative.startsWith("..") && !path.isAbsolute(relative));
14
+ };
15
+ const LOCK_FILE_NAMES = ["pnpm-lock.yaml", "yarn.lock", "package-lock.json"];
16
+ const resolveProjectRoot = (projectRoot) => {
17
+ if (!projectRoot) {
18
+ return undefined;
19
+ }
20
+ return path.isAbsolute(projectRoot)
21
+ ? path.resolve(projectRoot)
22
+ : path.resolve(process.cwd(), projectRoot);
23
+ };
24
+ const findNearestLockFile = (startDir) => {
25
+ let current = path.resolve(startDir);
26
+ const { root } = path.parse(current);
27
+ while (true) {
28
+ for (const candidate of LOCK_FILE_NAMES) {
29
+ const resolvedCandidate = path.join(current, candidate);
30
+ if (fs.existsSync(resolvedCandidate) && fs.statSync(resolvedCandidate).isFile()) {
31
+ return resolvedCandidate;
32
+ }
33
+ }
34
+ if (current === root) {
35
+ return undefined;
36
+ }
37
+ current = path.dirname(current);
38
+ }
39
+ };
40
+ const findCommonAncestor = (paths) => {
41
+ if (paths.length === 0) {
42
+ return undefined;
43
+ }
44
+ let ancestor = path.resolve(paths[0]);
45
+ for (const currentPath of paths.slice(1)) {
46
+ const target = path.resolve(currentPath);
47
+ const ancestorRoot = path.parse(ancestor).root;
48
+ if (path.parse(ancestor).root.toLowerCase() !== path.parse(target).root.toLowerCase()) {
49
+ return undefined;
50
+ }
51
+ let candidate = ancestor;
52
+ while (!isSubPath(candidate, target)) {
53
+ if (candidate === ancestorRoot) {
54
+ candidate = ancestorRoot;
55
+ break;
56
+ }
57
+ candidate = path.dirname(candidate);
58
+ }
59
+ ancestor = candidate;
60
+ }
61
+ return ancestor;
62
+ };
63
+ const resolvePath = (inputPath, projectRoot) => {
64
+ if (path.isAbsolute(inputPath)) {
65
+ return path.resolve(inputPath);
66
+ }
67
+ if (projectRoot) {
68
+ const candidateFromProjectRoot = path.resolve(projectRoot, inputPath);
69
+ if (isSubPath(projectRoot, candidateFromProjectRoot)) {
70
+ return candidateFromProjectRoot;
71
+ }
72
+ }
73
+ return path.resolve(process.cwd(), inputPath);
74
+ };
10
75
  const createBasicLambda = (scope, props) => {
11
76
  const lambdaProps = createBasicLambdaProps(props);
12
77
  let lambdaFunction = new aws_lambda_nodejs_1.NodejsFunction(scope, `${props.appPrefix || ""}${props.functionName}`, lambdaProps);
@@ -16,26 +81,44 @@ const createBasicLambda = (scope, props) => {
16
81
  };
17
82
  exports.createBasicLambda = createBasicLambda;
18
83
  const createBasicLambdaProps = (props) => {
19
- let resolvedEntry;
20
- if (props.codePath && path.isAbsolute(props.codePath)) {
21
- resolvedEntry = props.codePath;
22
- }
23
- else if (props.codePath && props.projectRoot) {
24
- // codePath is relative, resolve from current directory + projectRoot
25
- resolvedEntry = path.resolve(process.cwd(), props.codePath);
84
+ const resolvedProjectRoot = resolveProjectRoot(props.projectRoot);
85
+ const defaultRelativeEntry = path.join("resources", "lambdas", props.functionName, "main.mts");
86
+ const entryPath = props.codePath ?? defaultRelativeEntry;
87
+ const resolvedEntry = resolvePath(entryPath, resolvedProjectRoot);
88
+ const cwd = path.resolve(process.cwd());
89
+ const entryDir = path.dirname(resolvedEntry);
90
+ let effectiveProjectRoot = resolvedProjectRoot ?? cwd;
91
+ if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {
92
+ const ancestor = findCommonAncestor([effectiveProjectRoot, entryDir]);
93
+ if (!ancestor) {
94
+ throw new Error(`Unable to determine a projectRoot that contains both ${effectiveProjectRoot} and ${resolvedEntry}.`);
95
+ }
96
+ effectiveProjectRoot = ancestor;
26
97
  }
27
- else if (props.projectRoot) {
28
- // No codePath, use default path with projectRoot
29
- resolvedEntry = path.resolve(process.cwd(), props.projectRoot, `resources/lambdas/${props.functionName}/main.mts`);
98
+ if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {
99
+ throw new Error(`Resolved lambda entry ${resolvedEntry} must be located within projectRoot ${effectiveProjectRoot}.`);
30
100
  }
31
- else if (props.codePath) {
32
- // codePath without projectRoot
33
- resolvedEntry = path.resolve(process.cwd(), props.codePath);
101
+ let resolvedDepsLockFilePath = props.depsLockFilePath
102
+ ? resolvePath(props.depsLockFilePath, resolvedProjectRoot)
103
+ : undefined;
104
+ if (!resolvedDepsLockFilePath) {
105
+ const potentialLockFiles = [
106
+ findNearestLockFile(entryDir),
107
+ findNearestLockFile(cwd),
108
+ effectiveProjectRoot !== cwd ? findNearestLockFile(effectiveProjectRoot) : undefined,
109
+ ];
110
+ for (const lockFile of potentialLockFiles) {
111
+ if (lockFile && isSubPath(effectiveProjectRoot, lockFile)) {
112
+ resolvedDepsLockFilePath = lockFile;
113
+ break;
114
+ }
115
+ }
34
116
  }
35
- else {
36
- // Default path without projectRoot
37
- resolvedEntry = path.join(`./resources/lambdas/${props.functionName}/main.mts`);
117
+ if (resolvedDepsLockFilePath &&
118
+ !isSubPath(effectiveProjectRoot, resolvedDepsLockFilePath)) {
119
+ throw new Error(`Resolved depsLockFilePath ${resolvedDepsLockFilePath} must be located within projectRoot ${effectiveProjectRoot}.`);
38
120
  }
121
+ const environmentVars = props.envs ?? {};
39
122
  const lambdaProp = {
40
123
  entry: resolvedEntry,
41
124
  functionName: `${props.appPrefix ? `${props.appPrefix}-` : ""}${props.functionName}`,
@@ -45,7 +128,7 @@ const createBasicLambdaProps = (props) => {
45
128
  timeout: aws_cdk_lib_1.Duration.minutes(props.timeoutInMinutes ? props.timeoutInMinutes : 1),
46
129
  memorySize: props.memory,
47
130
  environment: {
48
- ...props.envs,
131
+ ...environmentVars,
49
132
  },
50
133
  bundling: {
51
134
  minify: true,
@@ -53,15 +136,15 @@ const createBasicLambdaProps = (props) => {
53
136
  sourceMap: true,
54
137
  sourceMapMode: aws_lambda_nodejs_1.SourceMapMode.EXTERNAL,
55
138
  environment: {
56
- ...props.envs,
139
+ ...environmentVars,
57
140
  },
58
- ...(props.projectRoot && { projectRoot: props.projectRoot }),
59
- ...(props.depsLockFilePath && {
60
- depsLockFilePath: props.depsLockFilePath,
61
- }),
62
141
  },
63
142
  role: props.role,
64
143
  layers: undefined,
144
+ ...(effectiveProjectRoot && { projectRoot: effectiveProjectRoot }),
145
+ ...(resolvedDepsLockFilePath && {
146
+ depsLockFilePath: resolvedDepsLockFilePath,
147
+ }),
65
148
  };
66
149
  return lambdaProp;
67
150
  };
@@ -81,4 +164,4 @@ const addLambdaLayers = (scope, lambda, layerArns) => {
81
164
  });
82
165
  }
83
166
  };
84
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-basic-lambda-helper.js","sourceRoot":"","sources":["../../../src/resources/lambda/create-basic-lambda-helper.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from \"aws-cdk-lib\";\r\nimport { ServicePrincipal } from \"aws-cdk-lib/aws-iam\";\r\nimport { LayerVersion, Runtime } from \"aws-cdk-lib/aws-lambda\";\r\nimport {\r\n  NodejsFunction,\r\n  NodejsFunctionProps,\r\n  SourceMapMode,\r\n} from \"aws-cdk-lib/aws-lambda-nodejs\";\r\nimport { RetentionDays } from \"aws-cdk-lib/aws-logs\";\r\nimport path = require(\"path\");\r\nimport { LambdaProps } from \"../../interfaces/lambda\";\r\nimport { Construct } from \"constructs\";\r\nimport { Table } from \"aws-cdk-lib/aws-dynamodb\";\r\n\r\nexport const createBasicLambda = (\r\n  scope: Construct,\r\n  props: LambdaProps\r\n): NodejsFunction => {\r\n  const lambdaProps = createBasicLambdaProps(props);\r\n\r\n  let lambdaFunction = new NodejsFunction(\r\n    scope,\r\n    `${props.appPrefix || \"\"}${props.functionName}`,\r\n    lambdaProps\r\n  );\r\n\r\n  grantAccessToDynamoTables(scope, lambdaFunction, props.dynamoTableNames);\r\n\r\n  addLambdaLayers(scope, lambdaFunction, props.lambdaLayerArn);\r\n\r\n  return lambdaFunction;\r\n};\r\n\r\nconst createBasicLambdaProps = (props: LambdaProps): NodejsFunctionProps => {\r\n  let resolvedEntry: string;\r\n\r\n  if (props.codePath && path.isAbsolute(props.codePath)) {\r\n    resolvedEntry = props.codePath;\r\n  } else if (props.codePath && props.projectRoot) {\r\n    // codePath is relative, resolve from current directory + projectRoot\r\n    resolvedEntry = path.resolve(process.cwd(), props.codePath);\r\n  } else if (props.projectRoot) {\r\n    // No codePath, use default path with projectRoot\r\n    resolvedEntry = path.resolve(\r\n      process.cwd(),\r\n      props.projectRoot,\r\n      `resources/lambdas/${props.functionName}/main.mts`\r\n    );\r\n  } else if (props.codePath) {\r\n    // codePath without projectRoot\r\n    resolvedEntry = path.resolve(process.cwd(), props.codePath);\r\n  } else {\r\n    // Default path without projectRoot\r\n    resolvedEntry = path.join(\r\n      `./resources/lambdas/${props.functionName}/main.mts`\r\n    );\r\n  }\r\n\r\n  const lambdaProp: NodejsFunctionProps = {\r\n    entry: resolvedEntry,\r\n    functionName: `${props.appPrefix ? `${props.appPrefix}-` : \"\"}${\r\n      props.functionName\r\n    }`,\r\n    handler: \"main.ts\",\r\n    logRetention: RetentionDays.TWO_WEEKS,\r\n    runtime: Runtime.NODEJS_LATEST,\r\n    timeout: Duration.minutes(\r\n      props.timeoutInMinutes ? props.timeoutInMinutes : 1\r\n    ),\r\n    memorySize: props.memory,\r\n    environment: {\r\n      ...props.envs,\r\n    },\r\n    bundling: {\r\n      minify: true,\r\n      target: `esnext`,\r\n      sourceMap: true,\r\n      sourceMapMode: SourceMapMode.EXTERNAL,\r\n      environment: {\r\n        ...props.envs,\r\n      },\r\n      ...(props.projectRoot && { projectRoot: props.projectRoot }),\r\n      ...(props.depsLockFilePath && {\r\n        depsLockFilePath: props.depsLockFilePath,\r\n      }),\r\n    },\r\n    role: props.role,\r\n    layers: undefined,\r\n  };\r\n\r\n  return lambdaProp;\r\n};\r\n\r\nconst grantAccessToDynamoTables = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  tableNames?: string[]\r\n) => {\r\n  if (tableNames && tableNames.length > 0) {\r\n    tableNames.forEach((tableName) => {\r\n      const table = Table.fromTableName(scope, `${tableName}-table`, tableName);\r\n\r\n      table.grantReadWriteData(lambda);\r\n    });\r\n  }\r\n};\r\n\r\nconst addLambdaLayers = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  layerArns?: string[]\r\n) => {\r\n  if (layerArns && layerArns.length > 0) {\r\n    layerArns.forEach((arn: string, idx: number) => {\r\n      const layer = LayerVersion.fromLayerVersionArn(\r\n        scope,\r\n        `common-layer-${idx}`,\r\n        arn\r\n      );\r\n\r\n      lambda.addLayers(layer);\r\n    });\r\n  }\r\n};\r\n"]}
167
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-basic-lambda-helper.js","sourceRoot":"","sources":["../../../src/resources/lambda/create-basic-lambda-helper.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from \"aws-cdk-lib\";\r\nimport { ServicePrincipal } from \"aws-cdk-lib/aws-iam\";\r\nimport { LayerVersion, Runtime } from \"aws-cdk-lib/aws-lambda\";\r\nimport {\r\n  NodejsFunction,\r\n  NodejsFunctionProps,\r\n  SourceMapMode,\r\n} from \"aws-cdk-lib/aws-lambda-nodejs\";\r\nimport { RetentionDays } from \"aws-cdk-lib/aws-logs\";\r\nimport fs = require(\"fs\");\r\nimport path = require(\"path\");\r\nimport { LambdaProps } from \"../../interfaces/lambda\";\r\nimport { Construct } from \"constructs\";\r\nimport { Table } from \"aws-cdk-lib/aws-dynamodb\";\r\n\r\nconst isSubPath = (root: string, target: string): boolean => {\r\n  const relative = path.relative(path.resolve(root), path.resolve(target));\r\n  return relative === \"\" || (!relative.startsWith(\"..\") && !path.isAbsolute(relative));\r\n};\r\n\r\nconst LOCK_FILE_NAMES = [\"pnpm-lock.yaml\", \"yarn.lock\", \"package-lock.json\"];\r\n\r\nconst resolveProjectRoot = (projectRoot?: string): string | undefined => {\r\n  if (!projectRoot) {\r\n    return undefined;\r\n  }\r\n\r\n  return path.isAbsolute(projectRoot)\r\n    ? path.resolve(projectRoot)\r\n    : path.resolve(process.cwd(), projectRoot);\r\n};\r\n\r\nconst findNearestLockFile = (startDir: string): string | undefined => {\r\n  let current = path.resolve(startDir);\r\n  const { root } = path.parse(current);\r\n\r\n  while (true) {\r\n    for (const candidate of LOCK_FILE_NAMES) {\r\n      const resolvedCandidate = path.join(current, candidate);\r\n      if (fs.existsSync(resolvedCandidate) && fs.statSync(resolvedCandidate).isFile()) {\r\n        return resolvedCandidate;\r\n      }\r\n    }\r\n\r\n    if (current === root) {\r\n      return undefined;\r\n    }\r\n\r\n    current = path.dirname(current);\r\n  }\r\n};\r\n\r\nconst findCommonAncestor = (paths: string[]): string | undefined => {\r\n  if (paths.length === 0) {\r\n    return undefined;\r\n  }\r\n\r\n  let ancestor = path.resolve(paths[0]);\r\n\r\n  for (const currentPath of paths.slice(1)) {\r\n    const target = path.resolve(currentPath);\r\n    const ancestorRoot = path.parse(ancestor).root;\r\n\r\n    if (path.parse(ancestor).root.toLowerCase() !== path.parse(target).root.toLowerCase()) {\r\n      return undefined;\r\n    }\r\n\r\n    let candidate = ancestor;\r\n    while (!isSubPath(candidate, target)) {\r\n      if (candidate === ancestorRoot) {\r\n        candidate = ancestorRoot;\r\n        break;\r\n      }\r\n      candidate = path.dirname(candidate);\r\n    }\r\n\r\n    ancestor = candidate;\r\n  }\r\n\r\n  return ancestor;\r\n};\r\n\r\nconst resolvePath = (inputPath: string, projectRoot?: string): string => {\r\n  if (path.isAbsolute(inputPath)) {\r\n    return path.resolve(inputPath);\r\n  }\r\n\r\n  if (projectRoot) {\r\n    const candidateFromProjectRoot = path.resolve(projectRoot, inputPath);\r\n    if (isSubPath(projectRoot, candidateFromProjectRoot)) {\r\n      return candidateFromProjectRoot;\r\n    }\r\n  }\r\n\r\n  return path.resolve(process.cwd(), inputPath);\r\n};\r\n\r\nexport const createBasicLambda = (\r\n  scope: Construct,\r\n  props: LambdaProps\r\n): NodejsFunction => {\r\n  const lambdaProps = createBasicLambdaProps(props);\r\n\r\n  let lambdaFunction = new NodejsFunction(\r\n    scope,\r\n    `${props.appPrefix || \"\"}${props.functionName}`,\r\n    lambdaProps\r\n  );\r\n\r\n  grantAccessToDynamoTables(scope, lambdaFunction, props.dynamoTableNames);\r\n\r\n  addLambdaLayers(scope, lambdaFunction, props.lambdaLayerArn);\r\n\r\n  return lambdaFunction;\r\n};\r\n\r\nconst createBasicLambdaProps = (props: LambdaProps): NodejsFunctionProps => {\r\n  const resolvedProjectRoot = resolveProjectRoot(props.projectRoot);\r\n  const defaultRelativeEntry = path.join(\r\n    \"resources\",\r\n    \"lambdas\",\r\n    props.functionName,\r\n    \"main.mts\"\r\n  );\r\n  const entryPath = props.codePath ?? defaultRelativeEntry;\r\n  const resolvedEntry = resolvePath(entryPath, resolvedProjectRoot);\r\n\r\n  const cwd = path.resolve(process.cwd());\r\n  const entryDir = path.dirname(resolvedEntry);\r\n\r\n  let effectiveProjectRoot = resolvedProjectRoot ?? cwd;\r\n  if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {\r\n    const ancestor = findCommonAncestor([effectiveProjectRoot, entryDir]);\r\n    if (!ancestor) {\r\n      throw new Error(\r\n        `Unable to determine a projectRoot that contains both ${effectiveProjectRoot} and ${resolvedEntry}.`\r\n      );\r\n    }\r\n    effectiveProjectRoot = ancestor;\r\n  }\r\n\r\n  if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {\r\n    throw new Error(\r\n      `Resolved lambda entry ${resolvedEntry} must be located within projectRoot ${effectiveProjectRoot}.`\r\n    );\r\n  }\r\n\r\n  let resolvedDepsLockFilePath = props.depsLockFilePath\r\n    ? resolvePath(props.depsLockFilePath, resolvedProjectRoot)\r\n    : undefined;\r\n\r\n  if (!resolvedDepsLockFilePath) {\r\n    const potentialLockFiles = [\r\n      findNearestLockFile(entryDir),\r\n      findNearestLockFile(cwd),\r\n      effectiveProjectRoot !== cwd ? findNearestLockFile(effectiveProjectRoot) : undefined,\r\n    ];\r\n\r\n    for (const lockFile of potentialLockFiles) {\r\n      if (lockFile && isSubPath(effectiveProjectRoot, lockFile)) {\r\n        resolvedDepsLockFilePath = lockFile;\r\n        break;\r\n      }\r\n    }\r\n  }\r\n\r\n  if (\r\n    resolvedDepsLockFilePath &&\r\n    !isSubPath(effectiveProjectRoot, resolvedDepsLockFilePath)\r\n  ) {\r\n    throw new Error(\r\n      `Resolved depsLockFilePath ${resolvedDepsLockFilePath} must be located within projectRoot ${effectiveProjectRoot}.`\r\n    );\r\n  }\r\n\r\n  const environmentVars = props.envs ?? {};\r\n\r\n  const lambdaProp: NodejsFunctionProps = {\r\n    entry: resolvedEntry,\r\n    functionName: `${props.appPrefix ? `${props.appPrefix}-` : \"\"}${\r\n      props.functionName\r\n    }`,\r\n    handler: \"main.ts\",\r\n    logRetention: RetentionDays.TWO_WEEKS,\r\n    runtime: Runtime.NODEJS_LATEST,\r\n    timeout: Duration.minutes(\r\n      props.timeoutInMinutes ? props.timeoutInMinutes : 1\r\n    ),\r\n    memorySize: props.memory,\r\n    environment: {\r\n      ...environmentVars,\r\n    },\r\n    bundling: {\r\n      minify: true,\r\n      target: `esnext`,\r\n      sourceMap: true,\r\n      sourceMapMode: SourceMapMode.EXTERNAL,\r\n      environment: {\r\n        ...environmentVars,\r\n      },\r\n    },\r\n    role: props.role,\r\n    layers: undefined,\r\n    ...(effectiveProjectRoot && { projectRoot: effectiveProjectRoot }),\r\n    ...(resolvedDepsLockFilePath && {\r\n      depsLockFilePath: resolvedDepsLockFilePath,\r\n    }),\r\n  };\r\n\r\n  return lambdaProp;\r\n};\r\n\r\nconst grantAccessToDynamoTables = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  tableNames?: string[]\r\n) => {\r\n  if (tableNames && tableNames.length > 0) {\r\n    tableNames.forEach((tableName) => {\r\n      const table = Table.fromTableName(scope, `${tableName}-table`, tableName);\r\n\r\n      table.grantReadWriteData(lambda);\r\n    });\r\n  }\r\n};\r\n\r\nconst addLambdaLayers = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  layerArns?: string[]\r\n) => {\r\n  if (layerArns && layerArns.length > 0) {\r\n    layerArns.forEach((arn: string, idx: number) => {\r\n      const layer = LayerVersion.fromLayerVersionArn(\r\n        scope,\r\n        `common-layer-${idx}`,\r\n        arn\r\n      );\r\n\r\n      lambda.addLayers(layer);\r\n    });\r\n  }\r\n};\r\n"]}
@@ -6,11 +6,76 @@ const aws_iam_1 = require("aws-cdk-lib/aws-iam");
6
6
  const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
7
7
  const aws_lambda_nodejs_1 = require("aws-cdk-lib/aws-lambda-nodejs");
8
8
  const aws_logs_1 = require("aws-cdk-lib/aws-logs");
9
+ const fs = require("fs");
9
10
  const path = require("path");
10
11
  const aws_service_principal_constants_1 = require("../../constants/aws-service-principal-constants");
11
12
  const aws_events_1 = require("aws-cdk-lib/aws-events");
12
13
  const aws_events_targets_1 = require("aws-cdk-lib/aws-events-targets");
13
14
  const aws_dynamodb_1 = require("aws-cdk-lib/aws-dynamodb");
15
+ const isSubPath = (root, target) => {
16
+ const relative = path.relative(path.resolve(root), path.resolve(target));
17
+ return relative === "" || (!relative.startsWith("..") && !path.isAbsolute(relative));
18
+ };
19
+ const LOCK_FILE_NAMES = ["pnpm-lock.yaml", "yarn.lock", "package-lock.json"];
20
+ const resolveProjectRoot = (projectRoot) => {
21
+ if (!projectRoot) {
22
+ return undefined;
23
+ }
24
+ return path.isAbsolute(projectRoot)
25
+ ? path.resolve(projectRoot)
26
+ : path.resolve(process.cwd(), projectRoot);
27
+ };
28
+ const findNearestLockFile = (startDir) => {
29
+ let current = path.resolve(startDir);
30
+ const { root } = path.parse(current);
31
+ while (true) {
32
+ for (const candidate of LOCK_FILE_NAMES) {
33
+ const resolvedCandidate = path.join(current, candidate);
34
+ if (fs.existsSync(resolvedCandidate) && fs.statSync(resolvedCandidate).isFile()) {
35
+ return resolvedCandidate;
36
+ }
37
+ }
38
+ if (current === root) {
39
+ return undefined;
40
+ }
41
+ current = path.dirname(current);
42
+ }
43
+ };
44
+ const findCommonAncestor = (paths) => {
45
+ if (paths.length === 0) {
46
+ return undefined;
47
+ }
48
+ let ancestor = path.resolve(paths[0]);
49
+ for (const currentPath of paths.slice(1)) {
50
+ const target = path.resolve(currentPath);
51
+ const ancestorRoot = path.parse(ancestor).root;
52
+ if (path.parse(ancestor).root.toLowerCase() !== path.parse(target).root.toLowerCase()) {
53
+ return undefined;
54
+ }
55
+ let candidate = ancestor;
56
+ while (!isSubPath(candidate, target)) {
57
+ if (candidate === ancestorRoot) {
58
+ candidate = ancestorRoot;
59
+ break;
60
+ }
61
+ candidate = path.dirname(candidate);
62
+ }
63
+ ancestor = candidate;
64
+ }
65
+ return ancestor;
66
+ };
67
+ const resolvePath = (inputPath, projectRoot) => {
68
+ if (path.isAbsolute(inputPath)) {
69
+ return path.resolve(inputPath);
70
+ }
71
+ if (projectRoot) {
72
+ const candidateFromProjectRoot = path.resolve(projectRoot, inputPath);
73
+ if (isSubPath(projectRoot, candidateFromProjectRoot)) {
74
+ return candidateFromProjectRoot;
75
+ }
76
+ }
77
+ return path.resolve(process.cwd(), inputPath);
78
+ };
14
79
  const createBasicLambdaTimerJob = (scope, props) => {
15
80
  const lambdaProps = createBasicLambdaProps(props);
16
81
  let lambdaFunction = new aws_lambda_nodejs_1.NodejsFunction(scope, `${props.appPrefix}${props.functionName}`, lambdaProps);
@@ -23,26 +88,44 @@ const createBasicLambdaTimerJob = (scope, props) => {
23
88
  };
24
89
  exports.createBasicLambdaTimerJob = createBasicLambdaTimerJob;
25
90
  const createBasicLambdaProps = (props) => {
26
- let resolvedEntry;
27
- if (props.codePath && path.isAbsolute(props.codePath)) {
28
- resolvedEntry = props.codePath;
29
- }
30
- else if (props.codePath && props.projectRoot) {
31
- // codePath is relative, resolve from current directory + projectRoot
32
- resolvedEntry = path.resolve(process.cwd(), props.codePath);
91
+ const resolvedProjectRoot = resolveProjectRoot(props.projectRoot);
92
+ const defaultRelativeEntry = path.join("resources", "lambdas", "timer-jobs", props.functionName, "main.mts");
93
+ const entryPath = props.codePath ?? defaultRelativeEntry;
94
+ const resolvedEntry = resolvePath(entryPath, resolvedProjectRoot);
95
+ const cwd = path.resolve(process.cwd());
96
+ const entryDir = path.dirname(resolvedEntry);
97
+ let effectiveProjectRoot = resolvedProjectRoot ?? cwd;
98
+ if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {
99
+ const ancestor = findCommonAncestor([effectiveProjectRoot, entryDir]);
100
+ if (!ancestor) {
101
+ throw new Error(`Unable to determine a projectRoot that contains both ${effectiveProjectRoot} and ${resolvedEntry}.`);
102
+ }
103
+ effectiveProjectRoot = ancestor;
33
104
  }
34
- else if (props.projectRoot) {
35
- // No codePath, use default path with projectRoot
36
- resolvedEntry = path.resolve(process.cwd(), props.projectRoot, `resources/lambdas/timer-jobs/${props.functionName}/main.mts`);
105
+ if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {
106
+ throw new Error(`Resolved lambda entry ${resolvedEntry} must be located within projectRoot ${effectiveProjectRoot}.`);
37
107
  }
38
- else if (props.codePath) {
39
- // codePath without projectRoot
40
- resolvedEntry = path.resolve(process.cwd(), props.codePath);
108
+ let resolvedDepsLockFilePath = props.depsLockFilePath
109
+ ? resolvePath(props.depsLockFilePath, resolvedProjectRoot)
110
+ : undefined;
111
+ if (!resolvedDepsLockFilePath) {
112
+ const potentialLockFiles = [
113
+ findNearestLockFile(entryDir),
114
+ findNearestLockFile(cwd),
115
+ effectiveProjectRoot !== cwd ? findNearestLockFile(effectiveProjectRoot) : undefined,
116
+ ];
117
+ for (const lockFile of potentialLockFiles) {
118
+ if (lockFile && isSubPath(effectiveProjectRoot, lockFile)) {
119
+ resolvedDepsLockFilePath = lockFile;
120
+ break;
121
+ }
122
+ }
41
123
  }
42
- else {
43
- // Default path without projectRoot
44
- resolvedEntry = path.join(`./resources/lambdas/timer-jobs/${props.functionName}/main.mts`);
124
+ if (resolvedDepsLockFilePath &&
125
+ !isSubPath(effectiveProjectRoot, resolvedDepsLockFilePath)) {
126
+ throw new Error(`Resolved depsLockFilePath ${resolvedDepsLockFilePath} must be located within projectRoot ${effectiveProjectRoot}.`);
45
127
  }
128
+ const environmentVars = props.envs ?? {};
46
129
  const lambdaProp = {
47
130
  entry: resolvedEntry,
48
131
  functionName: `${props.appPrefix ? `${props.appPrefix}-` : ""}${props.functionName}`,
@@ -52,7 +135,7 @@ const createBasicLambdaProps = (props) => {
52
135
  timeout: aws_cdk_lib_1.Duration.minutes(props.timeoutInMinutes ? props.timeoutInMinutes : 1),
53
136
  memorySize: props.memory,
54
137
  environment: {
55
- ...props.envs,
138
+ ...environmentVars,
56
139
  },
57
140
  bundling: {
58
141
  minify: true,
@@ -60,15 +143,15 @@ const createBasicLambdaProps = (props) => {
60
143
  sourceMap: true,
61
144
  sourceMapMode: aws_lambda_nodejs_1.SourceMapMode.EXTERNAL,
62
145
  environment: {
63
- ...props.envs,
146
+ ...environmentVars,
64
147
  },
65
- ...(props.projectRoot && { projectRoot: props.projectRoot }),
66
- ...(props.depsLockFilePath && {
67
- depsLockFilePath: props.depsLockFilePath,
68
- }),
69
148
  },
70
149
  role: props.role,
71
150
  layers: undefined,
151
+ ...(effectiveProjectRoot && { projectRoot: effectiveProjectRoot }),
152
+ ...(resolvedDepsLockFilePath && {
153
+ depsLockFilePath: resolvedDepsLockFilePath,
154
+ }),
72
155
  };
73
156
  return lambdaProp;
74
157
  };
@@ -99,4 +182,4 @@ const addLambdaLayers = (scope, lambda, layerArns) => {
99
182
  });
100
183
  }
101
184
  };
102
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-basic-lambda.js","sourceRoot":"","sources":["../../../src/resources/lambda/create-basic-lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from \"aws-cdk-lib\";\r\nimport { ServicePrincipal } from \"aws-cdk-lib/aws-iam\";\r\nimport { LayerVersion, Runtime } from \"aws-cdk-lib/aws-lambda\";\r\nimport {\r\n  NodejsFunction,\r\n  NodejsFunctionProps,\r\n  SourceMapMode,\r\n} from \"aws-cdk-lib/aws-lambda-nodejs\";\r\nimport { RetentionDays } from \"aws-cdk-lib/aws-logs\";\r\nimport path = require(\"path\");\r\nimport { TimerJobProps } from \"../../interfaces/timer-job\";\r\nimport { Construct } from \"constructs\";\r\nimport { SERVICE_PRINCIPAL } from \"../../constants/aws-service-principal-constants\";\r\nimport { CronOptions, Rule, Schedule } from \"aws-cdk-lib/aws-events\";\r\nimport { LambdaFunction } from \"aws-cdk-lib/aws-events-targets\";\r\nimport { Table } from \"aws-cdk-lib/aws-dynamodb\";\r\n\r\nexport const createBasicLambdaTimerJob = (\r\n  scope: Construct,\r\n  props: TimerJobProps\r\n): NodejsFunction => {\r\n  const lambdaProps = createBasicLambdaProps(props);\r\n\r\n  let lambdaFunction = new NodejsFunction(\r\n    scope,\r\n    `${props.appPrefix}${props.functionName}`,\r\n    lambdaProps\r\n  );\r\n\r\n  addInvokePermissionToLambdaForEvents(lambdaFunction);\r\n\r\n  const eventRule = createEventRuleForLambda(\r\n    scope,\r\n    lambdaFunction,\r\n    props.cronOptions\r\n  );\r\n\r\n  eventRule.addTarget(new LambdaFunction(lambdaFunction));\r\n\r\n  grantAccessToDynamoTables(scope, lambdaFunction, props.dynamoTableNames);\r\n\r\n  addLambdaLayers(scope, lambdaFunction, props.lambdaLayerArn);\r\n\r\n  return lambdaFunction;\r\n};\r\n\r\nconst createBasicLambdaProps = (props: TimerJobProps): NodejsFunctionProps => {\r\n  let resolvedEntry: string;\r\n\r\n  if (props.codePath && path.isAbsolute(props.codePath)) {\r\n    resolvedEntry = props.codePath;\r\n  } else if (props.codePath && props.projectRoot) {\r\n    // codePath is relative, resolve from current directory + projectRoot\r\n    resolvedEntry = path.resolve(process.cwd(), props.codePath);\r\n  } else if (props.projectRoot) {\r\n    // No codePath, use default path with projectRoot\r\n    resolvedEntry = path.resolve(\r\n      process.cwd(),\r\n      props.projectRoot,\r\n      `resources/lambdas/timer-jobs/${props.functionName}/main.mts`\r\n    );\r\n  } else if (props.codePath) {\r\n    // codePath without projectRoot\r\n    resolvedEntry = path.resolve(process.cwd(), props.codePath);\r\n  } else {\r\n    // Default path without projectRoot\r\n    resolvedEntry = path.join(\r\n      `./resources/lambdas/timer-jobs/${props.functionName}/main.mts`\r\n    );\r\n  }\r\n\r\n  const lambdaProp: NodejsFunctionProps = {\r\n    entry: resolvedEntry,\r\n    functionName: `${props.appPrefix ? `${props.appPrefix}-` : \"\"}${\r\n      props.functionName\r\n    }`,\r\n    handler: \"main.ts\",\r\n    logRetention: RetentionDays.TWO_WEEKS,\r\n    runtime: Runtime.NODEJS_LATEST,\r\n    timeout: Duration.minutes(\r\n      props.timeoutInMinutes ? props.timeoutInMinutes : 1\r\n    ),\r\n    memorySize: props.memory,\r\n    environment: {\r\n      ...props.envs,\r\n    },\r\n    bundling: {\r\n      minify: true,\r\n      target: `esnext`,\r\n      sourceMap: true,\r\n      sourceMapMode: SourceMapMode.EXTERNAL,\r\n      environment: {\r\n        ...props.envs,\r\n      },\r\n      ...(props.projectRoot && { projectRoot: props.projectRoot }),\r\n      ...(props.depsLockFilePath && {\r\n        depsLockFilePath: props.depsLockFilePath,\r\n      }),\r\n    },\r\n    role: props.role,\r\n    layers: undefined,\r\n  };\r\n\r\n  return lambdaProp;\r\n};\r\n\r\nconst addInvokePermissionToLambdaForEvents = (lambda: NodejsFunction) => {\r\n  lambda.addPermission(`InvokePermission-${lambda.functionName}`, {\r\n    principal: new ServicePrincipal(SERVICE_PRINCIPAL.EVENTS),\r\n  });\r\n};\r\n\r\nconst createEventRuleForLambda = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  options: CronOptions\r\n) => {\r\n  const eventRule = new Rule(\r\n    scope,\r\n    `scheduleRule-${lambda?.node.id || \"010\"}`,\r\n    {\r\n      schedule: Schedule.cron(options),\r\n    }\r\n  );\r\n\r\n  return eventRule;\r\n};\r\n\r\nconst grantAccessToDynamoTables = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  tableNames?: string[]\r\n) => {\r\n  if (tableNames && tableNames.length > 0) {\r\n    tableNames.forEach((tableName) => {\r\n      const table = Table.fromTableName(scope, `${tableName}-table`, tableName);\r\n\r\n      table.grantReadWriteData(lambda);\r\n    });\r\n  }\r\n};\r\n\r\nconst addLambdaLayers = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  layerArns?: string[]\r\n) => {\r\n  if (layerArns && layerArns.length > 0) {\r\n    layerArns.forEach((arn: string, idx: number) => {\r\n      const layer = LayerVersion.fromLayerVersionArn(\r\n        scope,\r\n        `common-layer-${idx}`,\r\n        arn\r\n      );\r\n\r\n      lambda.addLayers(layer);\r\n    });\r\n  }\r\n};\r\n"]}
185
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-basic-lambda.js","sourceRoot":"","sources":["../../../src/resources/lambda/create-basic-lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from \"aws-cdk-lib\";\r\nimport { ServicePrincipal } from \"aws-cdk-lib/aws-iam\";\r\nimport { LayerVersion, Runtime } from \"aws-cdk-lib/aws-lambda\";\r\nimport {\r\n  NodejsFunction,\r\n  NodejsFunctionProps,\r\n  SourceMapMode,\r\n} from \"aws-cdk-lib/aws-lambda-nodejs\";\r\nimport { RetentionDays } from \"aws-cdk-lib/aws-logs\";\r\nimport fs = require(\"fs\");\r\nimport path = require(\"path\");\r\nimport { TimerJobProps } from \"../../interfaces/timer-job\";\r\nimport { Construct } from \"constructs\";\r\nimport { SERVICE_PRINCIPAL } from \"../../constants/aws-service-principal-constants\";\r\nimport { CronOptions, Rule, Schedule } from \"aws-cdk-lib/aws-events\";\r\nimport { LambdaFunction } from \"aws-cdk-lib/aws-events-targets\";\r\nimport { Table } from \"aws-cdk-lib/aws-dynamodb\";\r\n\r\nconst isSubPath = (root: string, target: string): boolean => {\r\n  const relative = path.relative(path.resolve(root), path.resolve(target));\r\n  return relative === \"\" || (!relative.startsWith(\"..\") && !path.isAbsolute(relative));\r\n};\r\n\r\nconst LOCK_FILE_NAMES = [\"pnpm-lock.yaml\", \"yarn.lock\", \"package-lock.json\"];\r\n\r\nconst resolveProjectRoot = (projectRoot?: string): string | undefined => {\r\n  if (!projectRoot) {\r\n    return undefined;\r\n  }\r\n\r\n  return path.isAbsolute(projectRoot)\r\n    ? path.resolve(projectRoot)\r\n    : path.resolve(process.cwd(), projectRoot);\r\n};\r\n\r\nconst findNearestLockFile = (startDir: string): string | undefined => {\r\n  let current = path.resolve(startDir);\r\n  const { root } = path.parse(current);\r\n\r\n  while (true) {\r\n    for (const candidate of LOCK_FILE_NAMES) {\r\n      const resolvedCandidate = path.join(current, candidate);\r\n      if (fs.existsSync(resolvedCandidate) && fs.statSync(resolvedCandidate).isFile()) {\r\n        return resolvedCandidate;\r\n      }\r\n    }\r\n\r\n    if (current === root) {\r\n      return undefined;\r\n    }\r\n\r\n    current = path.dirname(current);\r\n  }\r\n};\r\n\r\nconst findCommonAncestor = (paths: string[]): string | undefined => {\r\n  if (paths.length === 0) {\r\n    return undefined;\r\n  }\r\n\r\n  let ancestor = path.resolve(paths[0]);\r\n\r\n  for (const currentPath of paths.slice(1)) {\r\n    const target = path.resolve(currentPath);\r\n    const ancestorRoot = path.parse(ancestor).root;\r\n\r\n    if (path.parse(ancestor).root.toLowerCase() !== path.parse(target).root.toLowerCase()) {\r\n      return undefined;\r\n    }\r\n\r\n    let candidate = ancestor;\r\n    while (!isSubPath(candidate, target)) {\r\n      if (candidate === ancestorRoot) {\r\n        candidate = ancestorRoot;\r\n        break;\r\n      }\r\n      candidate = path.dirname(candidate);\r\n    }\r\n\r\n    ancestor = candidate;\r\n  }\r\n\r\n  return ancestor;\r\n};\r\n\r\nconst resolvePath = (inputPath: string, projectRoot?: string): string => {\r\n  if (path.isAbsolute(inputPath)) {\r\n    return path.resolve(inputPath);\r\n  }\r\n\r\n  if (projectRoot) {\r\n    const candidateFromProjectRoot = path.resolve(projectRoot, inputPath);\r\n    if (isSubPath(projectRoot, candidateFromProjectRoot)) {\r\n      return candidateFromProjectRoot;\r\n    }\r\n  }\r\n\r\n  return path.resolve(process.cwd(), inputPath);\r\n};\r\n\r\nexport const createBasicLambdaTimerJob = (\r\n  scope: Construct,\r\n  props: TimerJobProps\r\n): NodejsFunction => {\r\n  const lambdaProps = createBasicLambdaProps(props);\r\n\r\n  let lambdaFunction = new NodejsFunction(\r\n    scope,\r\n    `${props.appPrefix}${props.functionName}`,\r\n    lambdaProps\r\n  );\r\n\r\n  addInvokePermissionToLambdaForEvents(lambdaFunction);\r\n\r\n  const eventRule = createEventRuleForLambda(\r\n    scope,\r\n    lambdaFunction,\r\n    props.cronOptions\r\n  );\r\n\r\n  eventRule.addTarget(new LambdaFunction(lambdaFunction));\r\n\r\n  grantAccessToDynamoTables(scope, lambdaFunction, props.dynamoTableNames);\r\n\r\n  addLambdaLayers(scope, lambdaFunction, props.lambdaLayerArn);\r\n\r\n  return lambdaFunction;\r\n};\r\n\r\nconst createBasicLambdaProps = (props: TimerJobProps): NodejsFunctionProps => {\r\n  const resolvedProjectRoot = resolveProjectRoot(props.projectRoot);\r\n  const defaultRelativeEntry = path.join(\r\n    \"resources\",\r\n    \"lambdas\",\r\n    \"timer-jobs\",\r\n    props.functionName,\r\n    \"main.mts\"\r\n  );\r\n  const entryPath = props.codePath ?? defaultRelativeEntry;\r\n  const resolvedEntry = resolvePath(entryPath, resolvedProjectRoot);\r\n\r\n  const cwd = path.resolve(process.cwd());\r\n  const entryDir = path.dirname(resolvedEntry);\r\n\r\n  let effectiveProjectRoot = resolvedProjectRoot ?? cwd;\r\n  if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {\r\n    const ancestor = findCommonAncestor([effectiveProjectRoot, entryDir]);\r\n    if (!ancestor) {\r\n      throw new Error(\r\n        `Unable to determine a projectRoot that contains both ${effectiveProjectRoot} and ${resolvedEntry}.`\r\n      );\r\n    }\r\n    effectiveProjectRoot = ancestor;\r\n  }\r\n\r\n  if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {\r\n    throw new Error(\r\n      `Resolved lambda entry ${resolvedEntry} must be located within projectRoot ${effectiveProjectRoot}.`\r\n    );\r\n  }\r\n\r\n  let resolvedDepsLockFilePath = props.depsLockFilePath\r\n    ? resolvePath(props.depsLockFilePath, resolvedProjectRoot)\r\n    : undefined;\r\n\r\n  if (!resolvedDepsLockFilePath) {\r\n    const potentialLockFiles = [\r\n      findNearestLockFile(entryDir),\r\n      findNearestLockFile(cwd),\r\n      effectiveProjectRoot !== cwd ? findNearestLockFile(effectiveProjectRoot) : undefined,\r\n    ];\r\n\r\n    for (const lockFile of potentialLockFiles) {\r\n      if (lockFile && isSubPath(effectiveProjectRoot, lockFile)) {\r\n        resolvedDepsLockFilePath = lockFile;\r\n        break;\r\n      }\r\n    }\r\n  }\r\n\r\n  if (\r\n    resolvedDepsLockFilePath &&\r\n    !isSubPath(effectiveProjectRoot, resolvedDepsLockFilePath)\r\n  ) {\r\n    throw new Error(\r\n      `Resolved depsLockFilePath ${resolvedDepsLockFilePath} must be located within projectRoot ${effectiveProjectRoot}.`\r\n    );\r\n  }\r\n\r\n  const environmentVars = props.envs ?? {};\r\n\r\n  const lambdaProp: NodejsFunctionProps = {\r\n    entry: resolvedEntry,\r\n    functionName: `${props.appPrefix ? `${props.appPrefix}-` : \"\"}${\r\n      props.functionName\r\n    }`,\r\n    handler: \"main.ts\",\r\n    logRetention: RetentionDays.TWO_WEEKS,\r\n    runtime: Runtime.NODEJS_LATEST,\r\n    timeout: Duration.minutes(\r\n      props.timeoutInMinutes ? props.timeoutInMinutes : 1\r\n    ),\r\n    memorySize: props.memory,\r\n    environment: {\r\n      ...environmentVars,\r\n    },\r\n    bundling: {\r\n      minify: true,\r\n      target: `esnext`,\r\n      sourceMap: true,\r\n      sourceMapMode: SourceMapMode.EXTERNAL,\r\n      environment: {\r\n        ...environmentVars,\r\n      },\r\n    },\r\n    role: props.role,\r\n    layers: undefined,\r\n    ...(effectiveProjectRoot && { projectRoot: effectiveProjectRoot }),\r\n    ...(resolvedDepsLockFilePath && {\r\n      depsLockFilePath: resolvedDepsLockFilePath,\r\n    }),\r\n  };\r\n\r\n  return lambdaProp;\r\n};\r\n\r\nconst addInvokePermissionToLambdaForEvents = (lambda: NodejsFunction) => {\r\n  lambda.addPermission(`InvokePermission-${lambda.functionName}`, {\r\n    principal: new ServicePrincipal(SERVICE_PRINCIPAL.EVENTS),\r\n  });\r\n};\r\n\r\nconst createEventRuleForLambda = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  options: CronOptions\r\n) => {\r\n  const eventRule = new Rule(\r\n    scope,\r\n    `scheduleRule-${lambda?.node.id || \"010\"}`,\r\n    {\r\n      schedule: Schedule.cron(options),\r\n    }\r\n  );\r\n\r\n  return eventRule;\r\n};\r\n\r\nconst grantAccessToDynamoTables = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  tableNames?: string[]\r\n) => {\r\n  if (tableNames && tableNames.length > 0) {\r\n    tableNames.forEach((tableName) => {\r\n      const table = Table.fromTableName(scope, `${tableName}-table`, tableName);\r\n\r\n      table.grantReadWriteData(lambda);\r\n    });\r\n  }\r\n};\r\n\r\nconst addLambdaLayers = (\r\n  scope: Construct,\r\n  lambda: NodejsFunction,\r\n  layerArns?: string[]\r\n) => {\r\n  if (layerArns && layerArns.length > 0) {\r\n    layerArns.forEach((arn: string, idx: number) => {\r\n      const layer = LayerVersion.fromLayerVersionArn(\r\n        scope,\r\n        `common-layer-${idx}`,\r\n        arn\r\n      );\r\n\r\n      lambda.addLayers(layer);\r\n    });\r\n  }\r\n};\r\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sylvesterllc/aws-constructs",
3
- "version": "1.1.25",
3
+ "version": "1.1.26",
4
4
  "description": "AWS Constructs",
5
5
  "main": "dist/index.js",
6
6
  "keywords": [
@@ -8,7 +8,7 @@ export const config: IAppConfig = {
8
8
  name: `${process.env.APP_NAME}`,
9
9
  accountNumber: process.env.CDK_DEFAULT_ACCOUNT || "",
10
10
  region: process.env.CDK_DEFAULT_REGION || "us-east-1",
11
- stackRuntime: Runtime.NODEJS_22_X,
11
+ stackRuntime: Runtime.NODEJS_LATEST,
12
12
  },
13
13
  API: {
14
14
  Name: `${process.env.APP_NAME}-auth-api`,
@@ -7,11 +7,94 @@ import {
7
7
  SourceMapMode,
8
8
  } from "aws-cdk-lib/aws-lambda-nodejs";
9
9
  import { RetentionDays } from "aws-cdk-lib/aws-logs";
10
+ import fs = require("fs");
10
11
  import path = require("path");
11
12
  import { LambdaProps } from "../../interfaces/lambda";
12
13
  import { Construct } from "constructs";
13
14
  import { Table } from "aws-cdk-lib/aws-dynamodb";
14
15
 
16
+ const isSubPath = (root: string, target: string): boolean => {
17
+ const relative = path.relative(path.resolve(root), path.resolve(target));
18
+ return relative === "" || (!relative.startsWith("..") && !path.isAbsolute(relative));
19
+ };
20
+
21
+ const LOCK_FILE_NAMES = ["pnpm-lock.yaml", "yarn.lock", "package-lock.json"];
22
+
23
+ const resolveProjectRoot = (projectRoot?: string): string | undefined => {
24
+ if (!projectRoot) {
25
+ return undefined;
26
+ }
27
+
28
+ return path.isAbsolute(projectRoot)
29
+ ? path.resolve(projectRoot)
30
+ : path.resolve(process.cwd(), projectRoot);
31
+ };
32
+
33
+ const findNearestLockFile = (startDir: string): string | undefined => {
34
+ let current = path.resolve(startDir);
35
+ const { root } = path.parse(current);
36
+
37
+ while (true) {
38
+ for (const candidate of LOCK_FILE_NAMES) {
39
+ const resolvedCandidate = path.join(current, candidate);
40
+ if (fs.existsSync(resolvedCandidate) && fs.statSync(resolvedCandidate).isFile()) {
41
+ return resolvedCandidate;
42
+ }
43
+ }
44
+
45
+ if (current === root) {
46
+ return undefined;
47
+ }
48
+
49
+ current = path.dirname(current);
50
+ }
51
+ };
52
+
53
+ const findCommonAncestor = (paths: string[]): string | undefined => {
54
+ if (paths.length === 0) {
55
+ return undefined;
56
+ }
57
+
58
+ let ancestor = path.resolve(paths[0]);
59
+
60
+ for (const currentPath of paths.slice(1)) {
61
+ const target = path.resolve(currentPath);
62
+ const ancestorRoot = path.parse(ancestor).root;
63
+
64
+ if (path.parse(ancestor).root.toLowerCase() !== path.parse(target).root.toLowerCase()) {
65
+ return undefined;
66
+ }
67
+
68
+ let candidate = ancestor;
69
+ while (!isSubPath(candidate, target)) {
70
+ if (candidate === ancestorRoot) {
71
+ candidate = ancestorRoot;
72
+ break;
73
+ }
74
+ candidate = path.dirname(candidate);
75
+ }
76
+
77
+ ancestor = candidate;
78
+ }
79
+
80
+ return ancestor;
81
+ };
82
+
83
+ const resolvePath = (inputPath: string, projectRoot?: string): string => {
84
+ if (path.isAbsolute(inputPath)) {
85
+ return path.resolve(inputPath);
86
+ }
87
+
88
+ if (projectRoot) {
89
+ const candidateFromProjectRoot = path.resolve(projectRoot, inputPath);
90
+ if (isSubPath(projectRoot, candidateFromProjectRoot)) {
91
+ return candidateFromProjectRoot;
92
+ }
93
+ }
94
+
95
+ return path.resolve(process.cwd(), inputPath);
96
+ };
97
+
15
98
  export const createBasicLambda = (
16
99
  scope: Construct,
17
100
  props: LambdaProps
@@ -32,30 +115,66 @@ export const createBasicLambda = (
32
115
  };
33
116
 
34
117
  const createBasicLambdaProps = (props: LambdaProps): NodejsFunctionProps => {
35
- let resolvedEntry: string;
36
-
37
- if (props.codePath && path.isAbsolute(props.codePath)) {
38
- resolvedEntry = props.codePath;
39
- } else if (props.codePath && props.projectRoot) {
40
- // codePath is relative, resolve from current directory + projectRoot
41
- resolvedEntry = path.resolve(process.cwd(), props.codePath);
42
- } else if (props.projectRoot) {
43
- // No codePath, use default path with projectRoot
44
- resolvedEntry = path.resolve(
45
- process.cwd(),
46
- props.projectRoot,
47
- `resources/lambdas/${props.functionName}/main.mts`
118
+ const resolvedProjectRoot = resolveProjectRoot(props.projectRoot);
119
+ const defaultRelativeEntry = path.join(
120
+ "resources",
121
+ "lambdas",
122
+ props.functionName,
123
+ "main.mts"
124
+ );
125
+ const entryPath = props.codePath ?? defaultRelativeEntry;
126
+ const resolvedEntry = resolvePath(entryPath, resolvedProjectRoot);
127
+
128
+ const cwd = path.resolve(process.cwd());
129
+ const entryDir = path.dirname(resolvedEntry);
130
+
131
+ let effectiveProjectRoot = resolvedProjectRoot ?? cwd;
132
+ if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {
133
+ const ancestor = findCommonAncestor([effectiveProjectRoot, entryDir]);
134
+ if (!ancestor) {
135
+ throw new Error(
136
+ `Unable to determine a projectRoot that contains both ${effectiveProjectRoot} and ${resolvedEntry}.`
137
+ );
138
+ }
139
+ effectiveProjectRoot = ancestor;
140
+ }
141
+
142
+ if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {
143
+ throw new Error(
144
+ `Resolved lambda entry ${resolvedEntry} must be located within projectRoot ${effectiveProjectRoot}.`
48
145
  );
49
- } else if (props.codePath) {
50
- // codePath without projectRoot
51
- resolvedEntry = path.resolve(process.cwd(), props.codePath);
52
- } else {
53
- // Default path without projectRoot
54
- resolvedEntry = path.join(
55
- `./resources/lambdas/${props.functionName}/main.mts`
146
+ }
147
+
148
+ let resolvedDepsLockFilePath = props.depsLockFilePath
149
+ ? resolvePath(props.depsLockFilePath, resolvedProjectRoot)
150
+ : undefined;
151
+
152
+ if (!resolvedDepsLockFilePath) {
153
+ const potentialLockFiles = [
154
+ findNearestLockFile(entryDir),
155
+ findNearestLockFile(cwd),
156
+ effectiveProjectRoot !== cwd ? findNearestLockFile(effectiveProjectRoot) : undefined,
157
+ ];
158
+
159
+ for (const lockFile of potentialLockFiles) {
160
+ if (lockFile && isSubPath(effectiveProjectRoot, lockFile)) {
161
+ resolvedDepsLockFilePath = lockFile;
162
+ break;
163
+ }
164
+ }
165
+ }
166
+
167
+ if (
168
+ resolvedDepsLockFilePath &&
169
+ !isSubPath(effectiveProjectRoot, resolvedDepsLockFilePath)
170
+ ) {
171
+ throw new Error(
172
+ `Resolved depsLockFilePath ${resolvedDepsLockFilePath} must be located within projectRoot ${effectiveProjectRoot}.`
56
173
  );
57
174
  }
58
175
 
176
+ const environmentVars = props.envs ?? {};
177
+
59
178
  const lambdaProp: NodejsFunctionProps = {
60
179
  entry: resolvedEntry,
61
180
  functionName: `${props.appPrefix ? `${props.appPrefix}-` : ""}${
@@ -69,7 +188,7 @@ const createBasicLambdaProps = (props: LambdaProps): NodejsFunctionProps => {
69
188
  ),
70
189
  memorySize: props.memory,
71
190
  environment: {
72
- ...props.envs,
191
+ ...environmentVars,
73
192
  },
74
193
  bundling: {
75
194
  minify: true,
@@ -77,15 +196,15 @@ const createBasicLambdaProps = (props: LambdaProps): NodejsFunctionProps => {
77
196
  sourceMap: true,
78
197
  sourceMapMode: SourceMapMode.EXTERNAL,
79
198
  environment: {
80
- ...props.envs,
199
+ ...environmentVars,
81
200
  },
82
- ...(props.projectRoot && { projectRoot: props.projectRoot }),
83
- ...(props.depsLockFilePath && {
84
- depsLockFilePath: props.depsLockFilePath,
85
- }),
86
201
  },
87
202
  role: props.role,
88
203
  layers: undefined,
204
+ ...(effectiveProjectRoot && { projectRoot: effectiveProjectRoot }),
205
+ ...(resolvedDepsLockFilePath && {
206
+ depsLockFilePath: resolvedDepsLockFilePath,
207
+ }),
89
208
  };
90
209
 
91
210
  return lambdaProp;
@@ -7,6 +7,7 @@ import {
7
7
  SourceMapMode,
8
8
  } from "aws-cdk-lib/aws-lambda-nodejs";
9
9
  import { RetentionDays } from "aws-cdk-lib/aws-logs";
10
+ import fs = require("fs");
10
11
  import path = require("path");
11
12
  import { TimerJobProps } from "../../interfaces/timer-job";
12
13
  import { Construct } from "constructs";
@@ -15,6 +16,88 @@ import { CronOptions, Rule, Schedule } from "aws-cdk-lib/aws-events";
15
16
  import { LambdaFunction } from "aws-cdk-lib/aws-events-targets";
16
17
  import { Table } from "aws-cdk-lib/aws-dynamodb";
17
18
 
19
+ const isSubPath = (root: string, target: string): boolean => {
20
+ const relative = path.relative(path.resolve(root), path.resolve(target));
21
+ return relative === "" || (!relative.startsWith("..") && !path.isAbsolute(relative));
22
+ };
23
+
24
+ const LOCK_FILE_NAMES = ["pnpm-lock.yaml", "yarn.lock", "package-lock.json"];
25
+
26
+ const resolveProjectRoot = (projectRoot?: string): string | undefined => {
27
+ if (!projectRoot) {
28
+ return undefined;
29
+ }
30
+
31
+ return path.isAbsolute(projectRoot)
32
+ ? path.resolve(projectRoot)
33
+ : path.resolve(process.cwd(), projectRoot);
34
+ };
35
+
36
+ const findNearestLockFile = (startDir: string): string | undefined => {
37
+ let current = path.resolve(startDir);
38
+ const { root } = path.parse(current);
39
+
40
+ while (true) {
41
+ for (const candidate of LOCK_FILE_NAMES) {
42
+ const resolvedCandidate = path.join(current, candidate);
43
+ if (fs.existsSync(resolvedCandidate) && fs.statSync(resolvedCandidate).isFile()) {
44
+ return resolvedCandidate;
45
+ }
46
+ }
47
+
48
+ if (current === root) {
49
+ return undefined;
50
+ }
51
+
52
+ current = path.dirname(current);
53
+ }
54
+ };
55
+
56
+ const findCommonAncestor = (paths: string[]): string | undefined => {
57
+ if (paths.length === 0) {
58
+ return undefined;
59
+ }
60
+
61
+ let ancestor = path.resolve(paths[0]);
62
+
63
+ for (const currentPath of paths.slice(1)) {
64
+ const target = path.resolve(currentPath);
65
+ const ancestorRoot = path.parse(ancestor).root;
66
+
67
+ if (path.parse(ancestor).root.toLowerCase() !== path.parse(target).root.toLowerCase()) {
68
+ return undefined;
69
+ }
70
+
71
+ let candidate = ancestor;
72
+ while (!isSubPath(candidate, target)) {
73
+ if (candidate === ancestorRoot) {
74
+ candidate = ancestorRoot;
75
+ break;
76
+ }
77
+ candidate = path.dirname(candidate);
78
+ }
79
+
80
+ ancestor = candidate;
81
+ }
82
+
83
+ return ancestor;
84
+ };
85
+
86
+ const resolvePath = (inputPath: string, projectRoot?: string): string => {
87
+ if (path.isAbsolute(inputPath)) {
88
+ return path.resolve(inputPath);
89
+ }
90
+
91
+ if (projectRoot) {
92
+ const candidateFromProjectRoot = path.resolve(projectRoot, inputPath);
93
+ if (isSubPath(projectRoot, candidateFromProjectRoot)) {
94
+ return candidateFromProjectRoot;
95
+ }
96
+ }
97
+
98
+ return path.resolve(process.cwd(), inputPath);
99
+ };
100
+
18
101
  export const createBasicLambdaTimerJob = (
19
102
  scope: Construct,
20
103
  props: TimerJobProps
@@ -45,30 +128,67 @@ export const createBasicLambdaTimerJob = (
45
128
  };
46
129
 
47
130
  const createBasicLambdaProps = (props: TimerJobProps): NodejsFunctionProps => {
48
- let resolvedEntry: string;
49
-
50
- if (props.codePath && path.isAbsolute(props.codePath)) {
51
- resolvedEntry = props.codePath;
52
- } else if (props.codePath && props.projectRoot) {
53
- // codePath is relative, resolve from current directory + projectRoot
54
- resolvedEntry = path.resolve(process.cwd(), props.codePath);
55
- } else if (props.projectRoot) {
56
- // No codePath, use default path with projectRoot
57
- resolvedEntry = path.resolve(
58
- process.cwd(),
59
- props.projectRoot,
60
- `resources/lambdas/timer-jobs/${props.functionName}/main.mts`
131
+ const resolvedProjectRoot = resolveProjectRoot(props.projectRoot);
132
+ const defaultRelativeEntry = path.join(
133
+ "resources",
134
+ "lambdas",
135
+ "timer-jobs",
136
+ props.functionName,
137
+ "main.mts"
138
+ );
139
+ const entryPath = props.codePath ?? defaultRelativeEntry;
140
+ const resolvedEntry = resolvePath(entryPath, resolvedProjectRoot);
141
+
142
+ const cwd = path.resolve(process.cwd());
143
+ const entryDir = path.dirname(resolvedEntry);
144
+
145
+ let effectiveProjectRoot = resolvedProjectRoot ?? cwd;
146
+ if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {
147
+ const ancestor = findCommonAncestor([effectiveProjectRoot, entryDir]);
148
+ if (!ancestor) {
149
+ throw new Error(
150
+ `Unable to determine a projectRoot that contains both ${effectiveProjectRoot} and ${resolvedEntry}.`
151
+ );
152
+ }
153
+ effectiveProjectRoot = ancestor;
154
+ }
155
+
156
+ if (!isSubPath(effectiveProjectRoot, resolvedEntry)) {
157
+ throw new Error(
158
+ `Resolved lambda entry ${resolvedEntry} must be located within projectRoot ${effectiveProjectRoot}.`
61
159
  );
62
- } else if (props.codePath) {
63
- // codePath without projectRoot
64
- resolvedEntry = path.resolve(process.cwd(), props.codePath);
65
- } else {
66
- // Default path without projectRoot
67
- resolvedEntry = path.join(
68
- `./resources/lambdas/timer-jobs/${props.functionName}/main.mts`
160
+ }
161
+
162
+ let resolvedDepsLockFilePath = props.depsLockFilePath
163
+ ? resolvePath(props.depsLockFilePath, resolvedProjectRoot)
164
+ : undefined;
165
+
166
+ if (!resolvedDepsLockFilePath) {
167
+ const potentialLockFiles = [
168
+ findNearestLockFile(entryDir),
169
+ findNearestLockFile(cwd),
170
+ effectiveProjectRoot !== cwd ? findNearestLockFile(effectiveProjectRoot) : undefined,
171
+ ];
172
+
173
+ for (const lockFile of potentialLockFiles) {
174
+ if (lockFile && isSubPath(effectiveProjectRoot, lockFile)) {
175
+ resolvedDepsLockFilePath = lockFile;
176
+ break;
177
+ }
178
+ }
179
+ }
180
+
181
+ if (
182
+ resolvedDepsLockFilePath &&
183
+ !isSubPath(effectiveProjectRoot, resolvedDepsLockFilePath)
184
+ ) {
185
+ throw new Error(
186
+ `Resolved depsLockFilePath ${resolvedDepsLockFilePath} must be located within projectRoot ${effectiveProjectRoot}.`
69
187
  );
70
188
  }
71
189
 
190
+ const environmentVars = props.envs ?? {};
191
+
72
192
  const lambdaProp: NodejsFunctionProps = {
73
193
  entry: resolvedEntry,
74
194
  functionName: `${props.appPrefix ? `${props.appPrefix}-` : ""}${
@@ -82,7 +202,7 @@ const createBasicLambdaProps = (props: TimerJobProps): NodejsFunctionProps => {
82
202
  ),
83
203
  memorySize: props.memory,
84
204
  environment: {
85
- ...props.envs,
205
+ ...environmentVars,
86
206
  },
87
207
  bundling: {
88
208
  minify: true,
@@ -90,15 +210,15 @@ const createBasicLambdaProps = (props: TimerJobProps): NodejsFunctionProps => {
90
210
  sourceMap: true,
91
211
  sourceMapMode: SourceMapMode.EXTERNAL,
92
212
  environment: {
93
- ...props.envs,
213
+ ...environmentVars,
94
214
  },
95
- ...(props.projectRoot && { projectRoot: props.projectRoot }),
96
- ...(props.depsLockFilePath && {
97
- depsLockFilePath: props.depsLockFilePath,
98
- }),
99
215
  },
100
216
  role: props.role,
101
217
  layers: undefined,
218
+ ...(effectiveProjectRoot && { projectRoot: effectiveProjectRoot }),
219
+ ...(resolvedDepsLockFilePath && {
220
+ depsLockFilePath: resolvedDepsLockFilePath,
221
+ }),
102
222
  };
103
223
 
104
224
  return lambdaProp;