@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/.jsii +4490 -0
- package/API.md +404 -0
- package/LICENSE +19 -0
- package/README.md +260 -0
- package/lib/index.d.ts +99 -0
- package/lib/index.js +245 -0
- package/package.json +99 -0
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
|
+
}
|