@nikovirtala/projen-lambda-function-construct-generator 0.0.0

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.
package/lib/index.d.ts ADDED
@@ -0,0 +1,99 @@
1
+ import { BuildOptions } from "@mrgrain/cdk-esbuild";
2
+ import { Component } from "projen";
3
+ import { NodeProject } from "projen/lib/javascript";
4
+ /**
5
+ * Options for the LambdaFunctionConstructGenerator
6
+ */
7
+ export interface LambdaFunctionConstructGeneratorOptions {
8
+ /**
9
+ * Source directory where Lambda Function handlers are located
10
+ *
11
+ * @default "src/handlers"
12
+ */
13
+ readonly sourceDir?: string;
14
+ /**
15
+ * Output directory where Lambda Function constructs will be generated
16
+ *
17
+ * @default "src/constructs/lambda"
18
+ */
19
+ readonly outputDir?: string;
20
+ /**
21
+ * File pattern to identify Lambda Function handlers
22
+ *
23
+ * @default "*.lambda.ts"
24
+ */
25
+ readonly filePattern?: string;
26
+ /**
27
+ * esbuild options to customize the bundling process
28
+ *
29
+ * @default {}
30
+ */
31
+ readonly esbuildOptions?: BuildOptions;
32
+ /**
33
+ * Whether to automatically add the required dependencies
34
+ *
35
+ * @default true
36
+ */
37
+ readonly addDependencies?: boolean;
38
+ /**
39
+ * Import statement for the base construct
40
+ *
41
+ * @example "import { NodejsFunction } from 'aws-cdk-lib/aws-lambda-nodejs';"
42
+ *
43
+ * @default "import { aws_lambda } from 'aws-cdk-lib';"
44
+ */
45
+ readonly baseConstructImport?: string;
46
+ /**
47
+ * Name of the construct class to extend
48
+ *
49
+ * @example "NodejsFunction"
50
+ *
51
+ * @default "aws_lambda.Function"
52
+ */
53
+ readonly baseConstructClass?: string;
54
+ /**
55
+ * Package name to add as dependency for the base construct
56
+ *
57
+ * @example "aws-cdk-lib"
58
+ *
59
+ * @default "aws-cdk-lib"
60
+ */
61
+ readonly baseConstructPackage?: string;
62
+ }
63
+ /**
64
+ * A projen component that generates AWS CDK Lambda Function constructs and bundles their code assets using esbuild.
65
+ *
66
+ * The bundling happens during projen execution, not during CDK synth, enabling a "build once, deploy many" pattern.
67
+ */
68
+ export declare class LambdaFunctionConstructGenerator extends Component {
69
+ readonly sourceDir: string;
70
+ readonly outputDir: string;
71
+ readonly filePattern: string;
72
+ readonly esbuildOptions: BuildOptions;
73
+ readonly baseConstructImport?: string;
74
+ readonly baseConstructClass?: string;
75
+ readonly baseConstructPackage?: string;
76
+ private readonly nodeProject;
77
+ private readonly bundlerScriptPath;
78
+ constructor(project: NodeProject, options?: LambdaFunctionConstructGeneratorOptions);
79
+ /**
80
+ * Add required dependencies for the component
81
+ */
82
+ private addDependencies;
83
+ /**
84
+ * Create a unique ID based on sourceDir and filePattern
85
+ */
86
+ private createUniqueId;
87
+ /**
88
+ * Create the bundle task that will be executed during projen build
89
+ */
90
+ private createBundleTask;
91
+ /**
92
+ * Create the bundler script that will be executed by the bundle task
93
+ */
94
+ private createBundlerScript;
95
+ /**
96
+ * Add the bundle task to the build workflow
97
+ */
98
+ private addBundleTaskToWorkflow;
99
+ }
package/lib/index.js ADDED
@@ -0,0 +1,245 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.LambdaFunctionConstructGenerator = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const path = require("path");
7
+ const projen_1 = require("projen");
8
+ /**
9
+ * A projen component that generates AWS CDK Lambda Function constructs and bundles their code assets using esbuild.
10
+ *
11
+ * The bundling happens during projen execution, not during CDK synth, enabling a "build once, deploy many" pattern.
12
+ */
13
+ class LambdaFunctionConstructGenerator extends projen_1.Component {
14
+ constructor(project, options) {
15
+ super(project);
16
+ this.nodeProject = project;
17
+ this.sourceDir = options?.sourceDir ?? "src/handlers";
18
+ this.outputDir = options?.outputDir ?? "src/constructs/lambda";
19
+ this.filePattern = options?.filePattern ?? "*.lambda.ts";
20
+ this.esbuildOptions = options?.esbuildOptions ?? {};
21
+ this.baseConstructImport = options?.baseConstructImport;
22
+ this.baseConstructClass = options?.baseConstructClass;
23
+ this.baseConstructPackage = options?.baseConstructPackage;
24
+ // Create unique script name based on sourceDir and filePattern
25
+ const uniqueId = this.createUniqueId(this.sourceDir, this.filePattern);
26
+ this.bundlerScriptPath = path.join(".projen", `generate-and-bundle-${uniqueId}.ts`);
27
+ // Add required dependencies
28
+ if (options?.addDependencies ?? true) {
29
+ this.addDependencies(options?.baseConstructPackage);
30
+ }
31
+ // Create the bundle task
32
+ this.createBundleTask();
33
+ // Add the bundle task to the build workflow
34
+ this.addBundleTaskToWorkflow();
35
+ }
36
+ /**
37
+ * Add required dependencies for the component
38
+ */
39
+ addDependencies(additionalPackage) {
40
+ this.nodeProject.addDeps("aws-cdk-lib", "constructs");
41
+ if (additionalPackage && additionalPackage !== "aws-cdk-lib" && additionalPackage !== "constructs") {
42
+ this.nodeProject.addDeps(additionalPackage);
43
+ }
44
+ }
45
+ /**
46
+ * Create a unique ID based on sourceDir and filePattern
47
+ */
48
+ createUniqueId(sourceDir, filePattern) {
49
+ // Remove special characters and convert to kebab case
50
+ const dirPart = sourceDir.replace(/\//g, "-").replace(/[^\w-]/g, "");
51
+ const patternPart = filePattern
52
+ .replace(/\*/g, "")
53
+ .replace(/\./g, "-")
54
+ .replace(/[^\w-]/g, "");
55
+ return `${dirPart}-${patternPart}`.replace(/--+/g, "-").replace(/^-|-$/g, "");
56
+ }
57
+ /**
58
+ * Create the bundle task that will be executed during projen build
59
+ */
60
+ createBundleTask() {
61
+ const uniqueId = this.createUniqueId(this.sourceDir, this.filePattern);
62
+ const taskName = `generate-and-bundle-${uniqueId}`;
63
+ let baseConstructArgs = "";
64
+ if (this.baseConstructImport) {
65
+ baseConstructArgs += ` --base-construct-import '${this.baseConstructImport}'`;
66
+ }
67
+ if (this.baseConstructClass) {
68
+ baseConstructArgs += ` --base-construct-class '${this.baseConstructClass}'`;
69
+ }
70
+ const bundleTask = this.nodeProject.addTask(taskName, {
71
+ description: `Generate Lambda Function Constructs from ${this.sourceDir}/${this.filePattern} and bundle their handlers`,
72
+ exec: `tsx --tsconfig tsconfig.dev.json ${this.bundlerScriptPath} --source-dir ${this.sourceDir} --output-dir ${this.outputDir} --file-pattern "${this.filePattern}" --esbuild-options '${JSON.stringify(this.esbuildOptions)}'${baseConstructArgs}`,
73
+ });
74
+ // Create the bundler script
75
+ this.createBundlerScript();
76
+ return bundleTask;
77
+ }
78
+ /**
79
+ * Create the bundler script that will be executed by the bundle task
80
+ */
81
+ createBundlerScript() {
82
+ const bundlerScript = `// ~~ Generated by projen. To modify, edit .projenrc.ts and run "npx projen".
83
+
84
+ import * as path from 'path';
85
+ import * as fs from 'fs';
86
+ import * as esbuild from 'esbuild';
87
+ import * as glob from 'glob';
88
+ import { pascalCase } from 'change-case';
89
+ import yargs from 'yargs/yargs';
90
+ import { hideBin } from 'yargs/helpers';
91
+
92
+ async function main() {
93
+ const argv = await yargs(hideBin(process.argv))
94
+ .option('source-dir', { type: 'string', default: 'src/handlers', description: 'Source directory where Lambda Function handlers are located' })
95
+ .option('output-dir', { type: 'string', default: 'src/constructs/lambda', description: 'Output directory where Lambda Function constructs will be generated' })
96
+ .option('file-pattern', { type: 'string', default: '*.lambda.ts', description: 'File pattern to identify Lambda Function handlers' })
97
+ .option('esbuild-options', { type: 'string', default: '{}', description: 'esbuild options as JSON string' })
98
+ .option('base-construct-import', { type: 'string', description: 'Import statement for the base construct' })
99
+ .option('base-construct-class', { type: 'string', description: 'Name of the construct class to extend' })
100
+ .help()
101
+ .parse();
102
+
103
+ const sourceDir = argv['source-dir'];
104
+ const outputDir = argv['output-dir'];
105
+ const filePattern = argv['file-pattern'];
106
+ const esbuildOptions = JSON.parse(argv['esbuild-options'] as string);
107
+ const baseConstructImport = argv['base-construct-import'] as string | undefined;
108
+ const baseConstructClass = argv['base-construct-class'] as string | undefined;
109
+
110
+ // Ensure output directory exists
111
+ fs.mkdirSync(path.join(process.cwd(), outputDir), { recursive: true });
112
+
113
+ // Ensure assets directory exists
114
+ const assetsDir = path.join(process.cwd(), 'assets', 'handlers');
115
+ fs.mkdirSync(assetsDir, { recursive: true });
116
+
117
+ // Find all Lambda Function handler files
118
+ const handlerFiles = glob.sync(path.join(process.cwd(), sourceDir, filePattern));
119
+
120
+ console.log(\`Found \${handlerFiles.length} Lambda Function handler files\`);
121
+
122
+ // Process each handler file
123
+ for (const handlerFile of handlerFiles) {
124
+ const relativePath = path.relative(path.join(process.cwd(), sourceDir), handlerFile);
125
+ const fileName = path.basename(relativePath, path.extname(relativePath));
126
+ const functionName = fileName.replace('.lambda', '');
127
+
128
+ console.log(\`Processing Lambda Function handler: \${functionName}\`);
129
+
130
+ // Create function-specific directory
131
+ const functionDir = path.join(assetsDir, functionName);
132
+ fs.mkdirSync(functionDir, { recursive: true });
133
+
134
+ // Bundle the handler code to index.js in the function directory
135
+ const outfile = path.join(functionDir, 'index.js');
136
+
137
+ try {
138
+ await esbuild.build({
139
+ entryPoints: [handlerFile],
140
+ bundle: true,
141
+ minify: true,
142
+ platform: 'node',
143
+ target: 'node18',
144
+ outfile,
145
+ ...esbuildOptions,
146
+ });
147
+
148
+ console.log(\`Successfully bundled \${functionName} to \${outfile}\`);
149
+
150
+ // Generate the CDK construct
151
+ const constructFilePath = path.join(process.cwd(), outputDir, \`\${functionName}.ts\`);
152
+ const constructCode = generateConstructCode(functionName, relativePath, baseConstructImport, baseConstructClass);
153
+
154
+ fs.writeFileSync(constructFilePath, constructCode);
155
+ console.log(\`Generated construct at \${constructFilePath}\`);
156
+ } catch (error) {
157
+ console.error(\`Error processing \${functionName}:\`, error);
158
+ }
159
+ }
160
+ }
161
+
162
+ function generateConstructCode(functionName: string, handlerPath: string, baseConstructImport?: string, baseConstructClass?: string) {
163
+ const constructName = \`\${pascalCase(functionName)}Function\`;
164
+
165
+ // Default values if no baseConstruct is provided
166
+ const importStatement = baseConstructImport ?? "";
167
+ const baseClassName = baseConstructClass ?? "aws_lambda.Function";
168
+
169
+ // Determine if we need to use aws_lambda.Runtime or not based on the base class
170
+ const useAwsLambdaRuntime = !baseConstructClass || baseClassName.includes("Function");
171
+
172
+ return \`// ~~ Generated by projen. To modify, edit .projenrc.ts and run "npx projen".
173
+
174
+ import * as path from 'path';
175
+ import { fileURLToPath } from 'url';
176
+ \${importStatement}
177
+ import { aws_lambda } from 'aws-cdk-lib';
178
+ import { Construct } from 'constructs';
179
+
180
+ // ES Module compatibility
181
+ const __filename = fileURLToPath(import.meta.url);
182
+ const __dirname = path.dirname(__filename);
183
+
184
+ /**
185
+ * Properties for \${constructName}
186
+ */
187
+ export interface \${constructName}Props extends Omit<\${baseClassName === "aws_lambda.Function" ? "aws_lambda.FunctionProps" : \`\${baseClassName}Props\`}, 'code'\${useAwsLambdaRuntime ? " | 'runtime'" : ""} | 'handler'> {
188
+ \${useAwsLambdaRuntime ? \`/**
189
+ * Override the default runtime
190
+ * @default nodejs22.x
191
+ */
192
+ readonly runtime?: aws_lambda.Runtime;\` : ""}
193
+ }
194
+
195
+ /**
196
+ * \${constructName} - Lambda Function Construct for \${handlerPath}
197
+ */
198
+ export class \${constructName} extends \${baseClassName} {
199
+ constructor(scope: Construct, id: string, props: \${constructName}Props = {}) {
200
+ super(scope, id, {
201
+ ...props,
202
+ \${useAwsLambdaRuntime ? "runtime: props.runtime ?? aws_lambda.Runtime.NODEJS_22_X," : ""}
203
+ handler: 'index.handler',
204
+ code: aws_lambda.Code.fromAsset(path.join(__dirname, '../../../assets/handlers/\${functionName}')),
205
+ });
206
+ }
207
+ }
208
+ \`;
209
+ }
210
+
211
+ main().catch(error => {
212
+ console.error('Error:', error);
213
+ process.exit(1);
214
+ });
215
+ `;
216
+ // Create source code file and add code lines to it
217
+ const src = new projen_1.SourceCode(this.project, this.bundlerScriptPath);
218
+ const lines = bundlerScript.split("\n");
219
+ for (const line of lines) {
220
+ src.line(line);
221
+ }
222
+ // Add the dependencies needed for the bundler script
223
+ this.nodeProject.addDevDeps("esbuild", "glob", "yargs", "@types/glob", "@types/yargs", "tsx", "change-case");
224
+ }
225
+ /**
226
+ * Add the bundle task to the build workflow
227
+ */
228
+ addBundleTaskToWorkflow() {
229
+ // Get the compile task
230
+ const compileTask = this.nodeProject.tasks.tryFind("compile");
231
+ if (compileTask) {
232
+ // Add the bundle task as a dependency of the compile task
233
+ const uniqueId = this.createUniqueId(this.sourceDir, this.filePattern);
234
+ const taskName = `generate-and-bundle-${uniqueId}`;
235
+ const bundleTask = this.nodeProject.tasks.tryFind(taskName);
236
+ if (bundleTask) {
237
+ compileTask.prependSpawn(bundleTask);
238
+ }
239
+ }
240
+ }
241
+ }
242
+ exports.LambdaFunctionConstructGenerator = LambdaFunctionConstructGenerator;
243
+ _a = JSII_RTTI_SYMBOL_1;
244
+ LambdaFunctionConstructGenerator[_a] = { fqn: "@nikovirtala/projen-lambda-function-construct-generator.LambdaFunctionConstructGenerator", version: "0.0.0" };
245
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"path\";\nimport { BuildOptions } from \"@mrgrain/cdk-esbuild\";\nimport { Component, SourceCode } from \"projen\";\nimport { NodeProject } from \"projen/lib/javascript\";\n\n/**\n * Options for the LambdaFunctionConstructGenerator\n */\nexport interface LambdaFunctionConstructGeneratorOptions {\n    /**\n     * Source directory where Lambda Function handlers are located\n     *\n     * @default \"src/handlers\"\n     */\n    readonly sourceDir?: string;\n\n    /**\n     * Output directory where Lambda Function constructs will be generated\n     *\n     * @default \"src/constructs/lambda\"\n     */\n    readonly outputDir?: string;\n\n    /**\n     * File pattern to identify Lambda Function handlers\n     *\n     * @default \"*.lambda.ts\"\n     */\n    readonly filePattern?: string;\n\n    /**\n     * esbuild options to customize the bundling process\n     *\n     * @default {}\n     */\n    readonly esbuildOptions?: BuildOptions;\n\n    /**\n     * Whether to automatically add the required dependencies\n     *\n     * @default true\n     */\n    readonly addDependencies?: boolean;\n\n    /**\n     * Import statement for the base construct\n     *\n     * @example \"import { NodejsFunction } from 'aws-cdk-lib/aws-lambda-nodejs';\"\n     *\n     * @default \"import { aws_lambda } from 'aws-cdk-lib';\"\n     */\n    readonly baseConstructImport?: string;\n\n    /**\n     * Name of the construct class to extend\n     *\n     * @example \"NodejsFunction\"\n     *\n     * @default \"aws_lambda.Function\"\n     */\n    readonly baseConstructClass?: string;\n\n    /**\n     * Package name to add as dependency for the base construct\n     *\n     * @example \"aws-cdk-lib\"\n     *\n     * @default \"aws-cdk-lib\"\n     */\n    readonly baseConstructPackage?: string;\n}\n\n/**\n * A projen component that generates AWS CDK Lambda Function constructs and bundles their code assets using esbuild.\n *\n * The bundling happens during projen execution, not during CDK synth, enabling a \"build once, deploy many\" pattern.\n */\nexport class LambdaFunctionConstructGenerator extends Component {\n    public readonly sourceDir: string;\n    public readonly outputDir: string;\n    public readonly filePattern: string;\n    public readonly esbuildOptions: BuildOptions;\n    public readonly baseConstructImport?: string;\n    public readonly baseConstructClass?: string;\n    public readonly baseConstructPackage?: string;\n    private readonly nodeProject: NodeProject;\n    private readonly bundlerScriptPath: string;\n\n    constructor(project: NodeProject, options?: LambdaFunctionConstructGeneratorOptions) {\n        super(project);\n        this.nodeProject = project;\n        this.sourceDir = options?.sourceDir ?? \"src/handlers\";\n        this.outputDir = options?.outputDir ?? \"src/constructs/lambda\";\n        this.filePattern = options?.filePattern ?? \"*.lambda.ts\";\n        this.esbuildOptions = options?.esbuildOptions ?? {};\n        this.baseConstructImport = options?.baseConstructImport;\n        this.baseConstructClass = options?.baseConstructClass;\n        this.baseConstructPackage = options?.baseConstructPackage;\n\n        // Create unique script name based on sourceDir and filePattern\n        const uniqueId = this.createUniqueId(this.sourceDir, this.filePattern);\n        this.bundlerScriptPath = path.join(\".projen\", `generate-and-bundle-${uniqueId}.ts`);\n\n        // Add required dependencies\n        if (options?.addDependencies ?? true) {\n            this.addDependencies(options?.baseConstructPackage);\n        }\n\n        // Create the bundle task\n        this.createBundleTask();\n\n        // Add the bundle task to the build workflow\n        this.addBundleTaskToWorkflow();\n    }\n\n    /**\n     * Add required dependencies for the component\n     */\n    private addDependencies(additionalPackage?: string) {\n        this.nodeProject.addDeps(\"aws-cdk-lib\", \"constructs\");\n\n        if (additionalPackage && additionalPackage !== \"aws-cdk-lib\" && additionalPackage !== \"constructs\") {\n            this.nodeProject.addDeps(additionalPackage);\n        }\n    }\n\n    /**\n     * Create a unique ID based on sourceDir and filePattern\n     */\n    private createUniqueId(sourceDir: string, filePattern: string): string {\n        // Remove special characters and convert to kebab case\n        const dirPart = sourceDir.replace(/\\//g, \"-\").replace(/[^\\w-]/g, \"\");\n        const patternPart = filePattern\n            .replace(/\\*/g, \"\")\n            .replace(/\\./g, \"-\")\n            .replace(/[^\\w-]/g, \"\");\n        return `${dirPart}-${patternPart}`.replace(/--+/g, \"-\").replace(/^-|-$/g, \"\");\n    }\n\n    /**\n     * Create the bundle task that will be executed during projen build\n     */\n    private createBundleTask() {\n        const uniqueId = this.createUniqueId(this.sourceDir, this.filePattern);\n        const taskName = `generate-and-bundle-${uniqueId}`;\n\n        let baseConstructArgs = \"\";\n        if (this.baseConstructImport) {\n            baseConstructArgs += ` --base-construct-import '${this.baseConstructImport}'`;\n        }\n        if (this.baseConstructClass) {\n            baseConstructArgs += ` --base-construct-class '${this.baseConstructClass}'`;\n        }\n\n        const bundleTask = this.nodeProject.addTask(taskName, {\n            description: `Generate Lambda Function Constructs from ${this.sourceDir}/${this.filePattern} and bundle their handlers`,\n            exec: `tsx --tsconfig tsconfig.dev.json ${this.bundlerScriptPath} --source-dir ${this.sourceDir} --output-dir ${this.outputDir} --file-pattern \"${this.filePattern}\" --esbuild-options '${JSON.stringify(this.esbuildOptions)}'${baseConstructArgs}`,\n        });\n\n        // Create the bundler script\n        this.createBundlerScript();\n\n        return bundleTask;\n    }\n\n    /**\n     * Create the bundler script that will be executed by the bundle task\n     */\n    private createBundlerScript() {\n        const bundlerScript = `// ~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\".\n\nimport * as path from 'path';\nimport * as fs from 'fs';\nimport * as esbuild from 'esbuild';\nimport * as glob from 'glob';\nimport { pascalCase } from 'change-case';\nimport yargs from 'yargs/yargs';\nimport { hideBin } from 'yargs/helpers';\n\nasync function main() {\n  const argv = await yargs(hideBin(process.argv))\n    .option('source-dir', { type: 'string', default: 'src/handlers', description: 'Source directory where Lambda Function handlers are located' })\n    .option('output-dir', { type: 'string', default: 'src/constructs/lambda', description: 'Output directory where Lambda Function constructs will be generated' })\n    .option('file-pattern', { type: 'string', default: '*.lambda.ts', description: 'File pattern to identify Lambda Function handlers' })\n    .option('esbuild-options', { type: 'string', default: '{}', description: 'esbuild options as JSON string' })\n    .option('base-construct-import', { type: 'string', description: 'Import statement for the base construct' })\n    .option('base-construct-class', { type: 'string', description: 'Name of the construct class to extend' })\n    .help()\n    .parse();\n\n  const sourceDir = argv['source-dir'];\n  const outputDir = argv['output-dir'];\n  const filePattern = argv['file-pattern'];\n  const esbuildOptions = JSON.parse(argv['esbuild-options'] as string);\n  const baseConstructImport = argv['base-construct-import'] as string | undefined;\n  const baseConstructClass = argv['base-construct-class'] as string | undefined;\n\n  // Ensure output directory exists\n  fs.mkdirSync(path.join(process.cwd(), outputDir), { recursive: true });\n\n  // Ensure assets directory exists\n  const assetsDir = path.join(process.cwd(), 'assets', 'handlers');\n  fs.mkdirSync(assetsDir, { recursive: true });\n\n  // Find all Lambda Function handler files\n  const handlerFiles = glob.sync(path.join(process.cwd(), sourceDir, filePattern));\n\n  console.log(\\`Found \\${handlerFiles.length} Lambda Function handler files\\`);\n\n  // Process each handler file\n  for (const handlerFile of handlerFiles) {\n    const relativePath = path.relative(path.join(process.cwd(), sourceDir), handlerFile);\n    const fileName = path.basename(relativePath, path.extname(relativePath));\n    const functionName = fileName.replace('.lambda', '');\n\n    console.log(\\`Processing Lambda Function handler: \\${functionName}\\`);\n\n    // Create function-specific directory\n    const functionDir = path.join(assetsDir, functionName);\n    fs.mkdirSync(functionDir, { recursive: true });\n\n    // Bundle the handler code to index.js in the function directory\n    const outfile = path.join(functionDir, 'index.js');\n\n    try {\n      await esbuild.build({\n        entryPoints: [handlerFile],\n        bundle: true,\n        minify: true,\n        platform: 'node',\n        target: 'node18',\n        outfile,\n        ...esbuildOptions,\n      });\n\n      console.log(\\`Successfully bundled \\${functionName} to \\${outfile}\\`);\n\n      // Generate the CDK construct\n      const constructFilePath = path.join(process.cwd(), outputDir, \\`\\${functionName}.ts\\`);\n      const constructCode = generateConstructCode(functionName, relativePath, baseConstructImport, baseConstructClass);\n\n      fs.writeFileSync(constructFilePath, constructCode);\n      console.log(\\`Generated construct at \\${constructFilePath}\\`);\n    } catch (error) {\n      console.error(\\`Error processing \\${functionName}:\\`, error);\n    }\n  }\n}\n\nfunction generateConstructCode(functionName: string, handlerPath: string, baseConstructImport?: string, baseConstructClass?: string) {\n  const constructName = \\`\\${pascalCase(functionName)}Function\\`;\n\n  // Default values if no baseConstruct is provided\n  const importStatement = baseConstructImport ?? \"\";\n  const baseClassName = baseConstructClass ?? \"aws_lambda.Function\";\n\n  // Determine if we need to use aws_lambda.Runtime or not based on the base class\n  const useAwsLambdaRuntime = !baseConstructClass || baseClassName.includes(\"Function\");\n\n  return \\`// ~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\".\n\nimport * as path from 'path';\nimport { fileURLToPath } from 'url';\n\\${importStatement}\nimport { aws_lambda } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\n// ES Module compatibility\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = path.dirname(__filename);\n\n/**\n * Properties for \\${constructName}\n */\nexport interface \\${constructName}Props extends Omit<\\${baseClassName === \"aws_lambda.Function\" ? \"aws_lambda.FunctionProps\" : \\`\\${baseClassName}Props\\`}, 'code'\\${useAwsLambdaRuntime ? \" | 'runtime'\" : \"\"} | 'handler'> {\n  \\${useAwsLambdaRuntime ? \\`/**\n   * Override the default runtime\n   * @default nodejs22.x\n   */\n  readonly runtime?: aws_lambda.Runtime;\\` : \"\"}\n}\n\n/**\n * \\${constructName} - Lambda Function Construct for \\${handlerPath}\n */\nexport class \\${constructName} extends \\${baseClassName} {\n  constructor(scope: Construct, id: string, props: \\${constructName}Props = {}) {\n    super(scope, id, {\n      ...props,\n      \\${useAwsLambdaRuntime ? \"runtime: props.runtime ?? aws_lambda.Runtime.NODEJS_22_X,\" : \"\"}\n      handler: 'index.handler',\n      code: aws_lambda.Code.fromAsset(path.join(__dirname, '../../../assets/handlers/\\${functionName}')),\n    });\n  }\n}\n\\`;\n}\n\nmain().catch(error => {\n  console.error('Error:', error);\n  process.exit(1);\n});\n`;\n\n        // Create source code file and add code lines to it\n        const src = new SourceCode(this.project, this.bundlerScriptPath);\n        const lines = bundlerScript.split(\"\\n\");\n        for (const line of lines) {\n            src.line(line);\n        }\n\n        // Add the dependencies needed for the bundler script\n        this.nodeProject.addDevDeps(\"esbuild\", \"glob\", \"yargs\", \"@types/glob\", \"@types/yargs\", \"tsx\", \"change-case\");\n    }\n\n    /**\n     * Add the bundle task to the build workflow\n     */\n    private addBundleTaskToWorkflow() {\n        // Get the compile task\n        const compileTask = this.nodeProject.tasks.tryFind(\"compile\");\n\n        if (compileTask) {\n            // Add the bundle task as a dependency of the compile task\n            const uniqueId = this.createUniqueId(this.sourceDir, this.filePattern);\n            const taskName = `generate-and-bundle-${uniqueId}`;\n            const bundleTask = this.nodeProject.tasks.tryFind(taskName);\n            if (bundleTask) {\n                compileTask.prependSpawn(bundleTask);\n            }\n        }\n    }\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,99 @@
1
+ {
2
+ "name": "@nikovirtala/projen-lambda-function-construct-generator",
3
+ "description": "Projen component to generate AWS CDK Lambda Function Constructs and bundle their Code Assets",
4
+ "repository": {
5
+ "type": "git",
6
+ "url": "https://github.com/nikovirtala/projen-lambda-function-construct-generator.git"
7
+ },
8
+ "author": {
9
+ "name": "Niko Virtala",
10
+ "email": "niko.virtala@hey.com",
11
+ "organization": false
12
+ },
13
+ "devDependencies": {
14
+ "@mrgrain/cdk-esbuild": "^5.7.3",
15
+ "@nikovirtala/projen-vitest": "^2.1.1",
16
+ "@types/glob": "^8.1.0",
17
+ "@types/node": "ts5.9",
18
+ "@types/yargs": "^17.0.34",
19
+ "@typescript-eslint/eslint-plugin": "^8",
20
+ "@typescript-eslint/parser": "^8",
21
+ "@vitest/coverage-v8": "^4",
22
+ "change-case": "^5.4.4",
23
+ "commit-and-tag-version": "^12",
24
+ "constructs": "^10.4.3",
25
+ "esbuild": "^0.27.0",
26
+ "eslint": "^9",
27
+ "eslint-config-prettier": "^10.1.8",
28
+ "eslint-import-resolver-typescript": "^4.4.4",
29
+ "eslint-plugin-import": "^2.32.0",
30
+ "eslint-plugin-prettier": "^5.5.4",
31
+ "glob": "^10.4.5",
32
+ "jsii": "~5.9.3",
33
+ "jsii-diff": "^1.118.0",
34
+ "jsii-docgen": "^10.5.0",
35
+ "jsii-pacmak": "^1.118.0",
36
+ "jsii-rosetta": "~5.9.3",
37
+ "prettier": "^3.6.2",
38
+ "projen": "^0.98.10",
39
+ "ts-node": "^10.9.2",
40
+ "tsx": "^4.20.6",
41
+ "typescript": "5.9.3",
42
+ "vitest": "^4",
43
+ "yargs": "^17.7.2"
44
+ },
45
+ "peerDependencies": {
46
+ "@mrgrain/cdk-esbuild": "^5.7.3",
47
+ "aws-cdk-lib": "^2.222.0",
48
+ "constructs": "^10.4.3",
49
+ "projen": "^0.98.10"
50
+ },
51
+ "dependencies": {
52
+ "@mrgrain/cdk-esbuild": "^5.7.3",
53
+ "aws-cdk-lib": "^2.222.0",
54
+ "constructs": "^10.4.3",
55
+ "projen": "^0.98.10"
56
+ },
57
+ "main": "lib/index.js",
58
+ "license": "MIT",
59
+ "publishConfig": {
60
+ "access": "public"
61
+ },
62
+ "version": "0.0.0",
63
+ "types": "lib/index.d.ts",
64
+ "stability": "stable",
65
+ "jsii": {
66
+ "outdir": "dist",
67
+ "targets": {},
68
+ "tsc": {
69
+ "outDir": "lib",
70
+ "rootDir": "src"
71
+ }
72
+ },
73
+ "//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\".",
74
+ "scripts": {
75
+ "build": "npx projen build",
76
+ "bump": "npx projen bump",
77
+ "clobber": "npx projen clobber",
78
+ "compat": "npx projen compat",
79
+ "compile": "npx projen compile",
80
+ "default": "npx projen default",
81
+ "docgen": "npx projen docgen",
82
+ "eject": "npx projen eject",
83
+ "eslint": "npx projen eslint",
84
+ "package": "npx projen package",
85
+ "package-all": "npx projen package-all",
86
+ "package:js": "npx projen package:js",
87
+ "post-compile": "npx projen post-compile",
88
+ "post-upgrade": "npx projen post-upgrade",
89
+ "pre-compile": "npx projen pre-compile",
90
+ "release": "npx projen release",
91
+ "test": "npx projen test",
92
+ "test:update": "npx projen test:update",
93
+ "test:watch": "npx projen test:watch",
94
+ "unbump": "npx projen unbump",
95
+ "upgrade": "npx projen upgrade",
96
+ "watch": "npx projen watch",
97
+ "projen": "npx projen"
98
+ }
99
+ }