@nikovirtala/projen-bundle-lambda-function-code 0.0.14 → 0.0.16

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/LICENSE CHANGED
@@ -1,4 +1,4 @@
1
- Copyright (c) 2023 Niko Virtala
1
+ Copyright (c) 2025 Niko Virtala
2
2
 
3
3
  Permission is hereby granted, free of charge, to any person obtaining a copy
4
4
  of this software and associated documentation files (the "Software"), to deal
@@ -0,0 +1,189 @@
1
+ import { Component, type Project, type Task } from "projen";
2
+ /**
3
+ * Options for `Bundler`.
4
+ */
5
+ export interface BundlerOptions {
6
+ /**
7
+ * The semantic version requirement for `esbuild`.
8
+ *
9
+ * @default - no specific version (implies latest)
10
+ */
11
+ readonly esbuildVersion?: string;
12
+ /**
13
+ * Output directory for all bundles.
14
+ * @default "assets"
15
+ */
16
+ readonly assetsDir?: string;
17
+ /**
18
+ * Install the `bundle` command as a pre-compile phase.
19
+ *
20
+ * @default true
21
+ */
22
+ readonly addToPreCompile?: boolean;
23
+ /**
24
+ * Map of file extensions (without dot) and loaders to use for this file type.
25
+ * Loaders are appended to the esbuild command by `--loader:.extension=loader`
26
+ */
27
+ readonly loaders?: {
28
+ [key: string]: string;
29
+ };
30
+ }
31
+ /**
32
+ * Adds support for bundling JavaScript applications and dependencies into a
33
+ * single file. In the future, this will also supports bundling websites.
34
+ */
35
+ export declare class Bundler extends Component {
36
+ /**
37
+ * Returns the `Bundler` instance associated with a project or `undefined` if
38
+ * there is no Bundler.
39
+ * @param project The project
40
+ * @returns A bundler
41
+ */
42
+ static of(project: Project): Bundler | undefined;
43
+ /**
44
+ * The semantic version requirement for `esbuild` (if defined).
45
+ */
46
+ readonly esbuildVersion: string | undefined;
47
+ /**
48
+ * Root bundle directory.
49
+ */
50
+ readonly bundledir: string;
51
+ private _task;
52
+ private readonly addToPreCompile;
53
+ private readonly loaders?;
54
+ /**
55
+ * Creates a `Bundler`.
56
+ */
57
+ constructor(project: Project, options?: BundlerOptions);
58
+ /**
59
+ * Gets or creates the singleton "bundle" task of the project.
60
+ *
61
+ * If the project doesn't have a "bundle" task, it will be created and spawned
62
+ * during the pre-compile phase.
63
+ */
64
+ get bundleTask(): Task;
65
+ /**
66
+ * Adds a task to the project which bundles a specific entrypoint and all of
67
+ * its dependencies into a single javascript output file.
68
+ *
69
+ * @param entrypoint The relative path of the artifact within the project
70
+ * @param options Bundling options
71
+ */
72
+ addBundle(entrypoint: string, options: BundlingOptions): Bundle;
73
+ /**
74
+ * Add bundling support to a project. This is called implicitly when
75
+ * `bundleTask` is referenced first. It adds the dependency on `esbuild`,
76
+ * gitignore/npmignore, etc.
77
+ */
78
+ private addBundlingSupport;
79
+ }
80
+ export interface Bundle {
81
+ /**
82
+ * The task that produces this bundle.
83
+ */
84
+ readonly bundleTask: Task;
85
+ /**
86
+ * Location of the output file (relative to project root).
87
+ */
88
+ readonly outfile: string;
89
+ /**
90
+ * Base directory containing the output file (relative to project root).
91
+ */
92
+ readonly outdir: string;
93
+ }
94
+ /**
95
+ * Options for bundling.
96
+ */
97
+ export interface BundlingOptions {
98
+ /**
99
+ * You can mark a file or a package as external to exclude it from your build.
100
+ * Instead of being bundled, the import will be preserved (using require for
101
+ * the iife and cjs formats and using import for the esm format) and will be
102
+ * evaluated at run time instead.
103
+ *
104
+ * This has several uses. First of all, it can be used to trim unnecessary
105
+ * code from your bundle for a code path that you know will never be executed.
106
+ * For example, a package may contain code that only runs in node but you will
107
+ * only be using that package in the browser. It can also be used to import
108
+ * code in node at run time from a package that cannot be bundled. For
109
+ * example, the fsevents package contains a native extension, which esbuild
110
+ * doesn't support.
111
+ *
112
+ * @default []
113
+ */
114
+ readonly externals?: string[];
115
+ /**
116
+ * Include a source map in the bundle.
117
+ *
118
+ * @default true
119
+ */
120
+ readonly sourcemap?: boolean;
121
+ /**
122
+ * esbuild target.
123
+ *
124
+ * @default "esnext"
125
+ */
126
+ readonly target?: string;
127
+ /**
128
+ * esbuild platform.
129
+ *
130
+ * @default "node"
131
+ */
132
+ readonly platform?: string;
133
+ /**
134
+ * Bundler output path relative to the asset's output directory.
135
+ * @default "index.mjs"
136
+ */
137
+ readonly outfile?: string;
138
+ /**
139
+ * The path of the tsconfig.json file to use for bundling
140
+ * @default "tsconfig.json"
141
+ */
142
+ readonly tsconfigPath?: string;
143
+ /**
144
+ * Map of file extensions (without dot) and loaders to use for this file type.
145
+ * Loaders are appended to the esbuild command by `--loader:.extension=loader`
146
+ */
147
+ readonly loaders?: {
148
+ [key: string]: string;
149
+ };
150
+ /**
151
+ * Output format for the generated JavaScript files. There are currently three possible values that can be configured: `"iife"`, `"cjs"`, and `"esm"`.
152
+ *
153
+ * If not set (`undefined`), esbuild picks an output format for you based on `platform`:
154
+ * - `"cjs"` if `platform` is `"node"`
155
+ * - `"iife"` if `platform` is `"browser"`
156
+ * - `"esm"` if `platform` is `"neutral"`
157
+ *
158
+ * Note: If making a bundle to run under node with ESM, set `format` to `"esm"` instead of setting `platform` to `"neutral"`.
159
+ *
160
+ * @default undefined
161
+ *
162
+ * @see https://esbuild.github.io/api/#format
163
+ */
164
+ readonly format?: string;
165
+ /**
166
+ *
167
+ *
168
+ * @default true
169
+ */
170
+ readonly minify?: boolean;
171
+ /**
172
+ *
173
+ *
174
+ * @default false
175
+ */
176
+ readonly sourcesContent?: boolean;
177
+ /**
178
+ *
179
+ *
180
+ * @default "module,main"
181
+ */
182
+ readonly mainFields?: string;
183
+ /**
184
+ *
185
+ *
186
+ * @default ":js='import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)'"
187
+ */
188
+ readonly banner?: string;
189
+ }
package/lib/bundler.js ADDED
@@ -0,0 +1,169 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.Bundler = void 0;
37
+ const path = __importStar(require("node:path"));
38
+ const projen_1 = require("projen");
39
+ const utils_1 = require("./utils");
40
+ /**
41
+ * Adds support for bundling JavaScript applications and dependencies into a
42
+ * single file. In the future, this will also supports bundling websites.
43
+ */
44
+ class Bundler extends projen_1.Component {
45
+ /**
46
+ * Returns the `Bundler` instance associated with a project or `undefined` if
47
+ * there is no Bundler.
48
+ * @param project The project
49
+ * @returns A bundler
50
+ */
51
+ static of(project) {
52
+ const isBundler = (o) => o instanceof Bundler;
53
+ return project.components.find(isBundler);
54
+ }
55
+ /**
56
+ * Creates a `Bundler`.
57
+ */
58
+ constructor(project, options = {}) {
59
+ super(project);
60
+ this.esbuildVersion = options.esbuildVersion;
61
+ this.bundledir = options.assetsDir ?? "assets";
62
+ this.addToPreCompile = options.addToPreCompile ?? true;
63
+ this.loaders = options.loaders;
64
+ }
65
+ /**
66
+ * Gets or creates the singleton "bundle" task of the project.
67
+ *
68
+ * If the project doesn't have a "bundle" task, it will be created and spawned
69
+ * during the pre-compile phase.
70
+ */
71
+ get bundleTask() {
72
+ if (!this._task) {
73
+ this.addBundlingSupport();
74
+ this._task = this.project.tasks.addTask("bundle", {
75
+ description: "Prepare assets",
76
+ });
77
+ // install the bundle task into the pre-compile phase.
78
+ if (this.addToPreCompile) {
79
+ this.project.preCompileTask.spawn(this._task);
80
+ }
81
+ }
82
+ return this._task;
83
+ }
84
+ /**
85
+ * Adds a task to the project which bundles a specific entrypoint and all of
86
+ * its dependencies into a single javascript output file.
87
+ *
88
+ * @param entrypoint The relative path of the artifact within the project
89
+ * @param options Bundling options
90
+ */
91
+ addBundle(entrypoint, options) {
92
+ const name = (0, utils_1.renderBundleName)(entrypoint);
93
+ const outdir = path.posix.join(this.bundledir, name);
94
+ const outfile = path.posix.join(outdir, options.outfile ?? "index.mjs");
95
+ const target = options.target ?? "esnext";
96
+ const platform = options.platform ?? "node";
97
+ const format = options.format ?? "esm";
98
+ const sourcemap = options.sourcemap ?? true;
99
+ const sourcesContent = options.sourcesContent ?? false;
100
+ const minify = options.minify ?? true;
101
+ const mainFields = options.mainFields ?? "module,main";
102
+ const banner = options.banner ??
103
+ ":js=import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)";
104
+ const args = [
105
+ "esbuild",
106
+ "--bundle",
107
+ entrypoint,
108
+ `--target="${target}"`,
109
+ `--platform="${platform}"`,
110
+ `--outfile="${outfile}"`,
111
+ ];
112
+ const tsconfig = options.tsconfigPath ?? false;
113
+ if (tsconfig) {
114
+ args.push(`--tsconfig="${tsconfig}"`);
115
+ }
116
+ for (const x of options.externals ?? []) {
117
+ args.push(`--external:${x}`);
118
+ }
119
+ if (sourcemap) {
120
+ args.push("--sourcemap");
121
+ }
122
+ if (sourcesContent) {
123
+ args.push(`--sources-content=${sourcesContent}`);
124
+ }
125
+ if (minify) {
126
+ args.push("--minify");
127
+ }
128
+ if (format) {
129
+ args.push(`--format=${format}`);
130
+ }
131
+ if (mainFields) {
132
+ args.push(`--main-fields=${mainFields}`);
133
+ }
134
+ if (banner) {
135
+ args.push(`--banner${banner}`);
136
+ }
137
+ const loaders = (options.loaders ?? false) ? options.loaders : (this.loaders ?? false);
138
+ if (loaders) {
139
+ for (const [extension, loader] of Object.entries(loaders)) {
140
+ args.push(`--loader:.${extension}=${loader}`);
141
+ }
142
+ }
143
+ console.log({ args: args });
144
+ const bundleTask = this.project.addTask(`bundle:${name}`, {
145
+ description: `Create a JavaScript bundle from ${entrypoint}`,
146
+ exec: args.join(" "),
147
+ });
148
+ this.bundleTask.spawn(bundleTask);
149
+ return {
150
+ bundleTask: bundleTask,
151
+ outdir: outdir,
152
+ outfile: outfile,
153
+ };
154
+ }
155
+ /**
156
+ * Add bundling support to a project. This is called implicitly when
157
+ * `bundleTask` is referenced first. It adds the dependency on `esbuild`,
158
+ * gitignore/npmignore, etc.
159
+ */
160
+ addBundlingSupport() {
161
+ const ignoreEntry = `/${this.bundledir}/`;
162
+ this.project.addGitIgnore(ignoreEntry);
163
+ this.project.addPackageIgnore(`!${ignoreEntry}`); // include in tarball
164
+ const dep = this.esbuildVersion ? `esbuild@${this.esbuildVersion}` : "esbuild";
165
+ this.project.deps.addDependency(dep, projen_1.DependencyType.BUILD);
166
+ }
167
+ }
168
+ exports.Bundler = Bundler;
169
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bundler.js","sourceRoot":"","sources":["../src/bundler.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,gDAAkC;AAClC,mCAA4E;AAC5E,mCAA2C;AAiC3C;;;GAGG;AACH,MAAa,OAAQ,SAAQ,kBAAS;IAClC;;;;;OAKG;IACI,MAAM,CAAC,EAAE,CAAC,OAAgB;QAC7B,MAAM,SAAS,GAAG,CAAC,CAAY,EAAgB,EAAE,CAAC,CAAC,YAAY,OAAO,CAAC;QACvE,OAAO,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAC9C,CAAC;IAgBD;;OAEG;IACH,YAAY,OAAgB,EAAE,UAA0B,EAAE;QACtD,KAAK,CAAC,OAAO,CAAC,CAAC;QAEf,IAAI,CAAC,cAAc,GAAG,OAAO,CAAC,cAAc,CAAC;QAC7C,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,SAAS,IAAI,QAAQ,CAAC;QAC/C,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC,eAAe,IAAI,IAAI,CAAC;QACvD,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;IACnC,CAAC;IAED;;;;;OAKG;IACH,IAAW,UAAU;QACjB,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;YACd,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAC1B,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,EAAE;gBAC9C,WAAW,EAAE,gBAAgB;aAChC,CAAC,CAAC;YAEH,sDAAsD;YACtD,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAClD,CAAC;QACL,CAAC;QAED,OAAO,IAAI,CAAC,KAAK,CAAC;IACtB,CAAC;IAED;;;;;;OAMG;IACI,SAAS,CAAC,UAAkB,EAAE,OAAwB;QACzD,MAAM,IAAI,GAAG,IAAA,wBAAgB,EAAC,UAAU,CAAC,CAAC;QAE1C,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QACrD,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,OAAO,CAAC,OAAO,IAAI,WAAW,CAAC,CAAC;QACxE,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,QAAQ,CAAC;QAC1C,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,IAAI,MAAM,CAAC;QAC5C,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,KAAK,CAAC;QACvC,MAAM,SAAS,GAAG,OAAO,CAAC,SAAS,IAAI,IAAI,CAAC;QAC5C,MAAM,cAAc,GAAG,OAAO,CAAC,cAAc,IAAI,KAAK,CAAC;QACvD,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,IAAI,CAAC;QACtC,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,IAAI,aAAa,CAAC;QACvD,MAAM,MAAM,GACR,OAAO,CAAC,MAAM;YACd,4HAA4H,CAAC;QAEjI,MAAM,IAAI,GAAG;YACT,SAAS;YACT,UAAU;YACV,UAAU;YACV,aAAa,MAAM,GAAG;YACtB,eAAe,QAAQ,GAAG;YAC1B,cAAc,OAAO,GAAG;SAC3B,CAAC;QAEF,MAAM,QAAQ,GAAG,OAAO,CAAC,YAAY,IAAI,KAAK,CAAC;QAC/C,IAAI,QAAQ,EAAE,CAAC;YACX,IAAI,CAAC,IAAI,CAAC,eAAe,QAAQ,GAAG,CAAC,CAAC;QAC1C,CAAC;QAED,KAAK,MAAM,CAAC,IAAI,OAAO,CAAC,SAAS,IAAI,EAAE,EAAE,CAAC;YACtC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,EAAE,CAAC,CAAC;QACjC,CAAC;QAED,IAAI,SAAS,EAAE,CAAC;YACZ,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QAC7B,CAAC;QAED,IAAI,cAAc,EAAE,CAAC;YACjB,IAAI,CAAC,IAAI,CAAC,qBAAqB,cAAc,EAAE,CAAC,CAAC;QACrD,CAAC;QAED,IAAI,MAAM,EAAE,CAAC;YACT,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC1B,CAAC;QAED,IAAI,MAAM,EAAE,CAAC;YACT,IAAI,CAAC,IAAI,CAAC,YAAY,MAAM,EAAE,CAAC,CAAC;QACpC,CAAC;QAED,IAAI,UAAU,EAAE,CAAC;YACb,IAAI,CAAC,IAAI,CAAC,iBAAiB,UAAU,EAAE,CAAC,CAAC;QAC7C,CAAC;QAED,IAAI,MAAM,EAAE,CAAC;YACT,IAAI,CAAC,IAAI,CAAC,WAAW,MAAM,EAAE,CAAC,CAAC;QACnC,CAAC;QAED,MAAM,OAAO,GAAG,CAAC,OAAO,CAAC,OAAO,IAAI,KAAK,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,IAAI,KAAK,CAAC,CAAC;QACvF,IAAI,OAAO,EAAE,CAAC;YACV,KAAK,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;gBACxD,IAAI,CAAC,IAAI,CAAC,aAAa,SAAS,IAAI,MAAM,EAAE,CAAC,CAAC;YAClD,CAAC;QACL,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;QAE5B,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,UAAU,IAAI,EAAE,EAAE;YACtD,WAAW,EAAE,mCAAmC,UAAU,EAAE;YAC5D,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC;SACvB,CAAC,CAAC;QAEH,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAElC,OAAO;YACH,UAAU,EAAE,UAAU;YACtB,MAAM,EAAE,MAAM;YACd,OAAO,EAAE,OAAO;SACnB,CAAC;IACN,CAAC;IAED;;;;OAIG;IACK,kBAAkB;QACtB,MAAM,WAAW,GAAG,IAAI,IAAI,CAAC,SAAS,GAAG,CAAC;QAC1C,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC;QACvC,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,IAAI,WAAW,EAAE,CAAC,CAAC,CAAC,qBAAqB;QACvE,MAAM,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,WAAW,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;QAC/E,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG,EAAE,uBAAc,CAAC,KAAK,CAAC,CAAC;IAC/D,CAAC;CACJ;AAhKD,0BAgKC","sourcesContent":["import * as path from \"node:path\";\nimport { Component, DependencyType, type Project, type Task } from \"projen\";\nimport { renderBundleName } from \"./utils\";\n\n/**\n * Options for `Bundler`.\n */\nexport interface BundlerOptions {\n    /**\n     * The semantic version requirement for `esbuild`.\n     *\n     * @default - no specific version (implies latest)\n     */\n    readonly esbuildVersion?: string;\n\n    /**\n     * Output directory for all bundles.\n     * @default \"assets\"\n     */\n    readonly assetsDir?: string;\n\n    /**\n     * Install the `bundle` command as a pre-compile phase.\n     *\n     * @default true\n     */\n    readonly addToPreCompile?: boolean;\n\n    /**\n     * Map of file extensions (without dot) and loaders to use for this file type.\n     * Loaders are appended to the esbuild command by `--loader:.extension=loader`\n     */\n    readonly loaders?: { [key: string]: string };\n}\n\n/**\n * Adds support for bundling JavaScript applications and dependencies into a\n * single file. In the future, this will also supports bundling websites.\n */\nexport class Bundler extends Component {\n    /**\n     * Returns the `Bundler` instance associated with a project or `undefined` if\n     * there is no Bundler.\n     * @param project The project\n     * @returns A bundler\n     */\n    public static of(project: Project): Bundler | undefined {\n        const isBundler = (o: Component): o is Bundler => o instanceof Bundler;\n        return project.components.find(isBundler);\n    }\n\n    /**\n     * The semantic version requirement for `esbuild` (if defined).\n     */\n    public readonly esbuildVersion: string | undefined;\n\n    /**\n     * Root bundle directory.\n     */\n    public readonly bundledir: string;\n\n    private _task: Task | undefined;\n    private readonly addToPreCompile: boolean;\n    private readonly loaders?: { [key: string]: string };\n\n    /**\n     * Creates a `Bundler`.\n     */\n    constructor(project: Project, options: BundlerOptions = {}) {\n        super(project);\n\n        this.esbuildVersion = options.esbuildVersion;\n        this.bundledir = options.assetsDir ?? \"assets\";\n        this.addToPreCompile = options.addToPreCompile ?? true;\n        this.loaders = options.loaders;\n    }\n\n    /**\n     * Gets or creates the singleton \"bundle\" task of the project.\n     *\n     * If the project doesn't have a \"bundle\" task, it will be created and spawned\n     * during the pre-compile phase.\n     */\n    public get bundleTask(): Task {\n        if (!this._task) {\n            this.addBundlingSupport();\n            this._task = this.project.tasks.addTask(\"bundle\", {\n                description: \"Prepare assets\",\n            });\n\n            // install the bundle task into the pre-compile phase.\n            if (this.addToPreCompile) {\n                this.project.preCompileTask.spawn(this._task);\n            }\n        }\n\n        return this._task;\n    }\n\n    /**\n     * Adds a task to the project which bundles a specific entrypoint and all of\n     * its dependencies into a single javascript output file.\n     *\n     * @param entrypoint The relative path of the artifact within the project\n     * @param options Bundling options\n     */\n    public addBundle(entrypoint: string, options: BundlingOptions): Bundle {\n        const name = renderBundleName(entrypoint);\n\n        const outdir = path.posix.join(this.bundledir, name);\n        const outfile = path.posix.join(outdir, options.outfile ?? \"index.mjs\");\n        const target = options.target ?? \"esnext\";\n        const platform = options.platform ?? \"node\";\n        const format = options.format ?? \"esm\";\n        const sourcemap = options.sourcemap ?? true;\n        const sourcesContent = options.sourcesContent ?? false;\n        const minify = options.minify ?? true;\n        const mainFields = options.mainFields ?? \"module,main\";\n        const banner =\n            options.banner ??\n            \":js=import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)\";\n\n        const args = [\n            \"esbuild\",\n            \"--bundle\",\n            entrypoint,\n            `--target=\"${target}\"`,\n            `--platform=\"${platform}\"`,\n            `--outfile=\"${outfile}\"`,\n        ];\n\n        const tsconfig = options.tsconfigPath ?? false;\n        if (tsconfig) {\n            args.push(`--tsconfig=\"${tsconfig}\"`);\n        }\n\n        for (const x of options.externals ?? []) {\n            args.push(`--external:${x}`);\n        }\n\n        if (sourcemap) {\n            args.push(\"--sourcemap\");\n        }\n\n        if (sourcesContent) {\n            args.push(`--sources-content=${sourcesContent}`);\n        }\n\n        if (minify) {\n            args.push(\"--minify\");\n        }\n\n        if (format) {\n            args.push(`--format=${format}`);\n        }\n\n        if (mainFields) {\n            args.push(`--main-fields=${mainFields}`);\n        }\n\n        if (banner) {\n            args.push(`--banner${banner}`);\n        }\n\n        const loaders = (options.loaders ?? false) ? options.loaders : (this.loaders ?? false);\n        if (loaders) {\n            for (const [extension, loader] of Object.entries(loaders)) {\n                args.push(`--loader:.${extension}=${loader}`);\n            }\n        }\n\n        console.log({ args: args });\n\n        const bundleTask = this.project.addTask(`bundle:${name}`, {\n            description: `Create a JavaScript bundle from ${entrypoint}`,\n            exec: args.join(\" \"),\n        });\n\n        this.bundleTask.spawn(bundleTask);\n\n        return {\n            bundleTask: bundleTask,\n            outdir: outdir,\n            outfile: outfile,\n        };\n    }\n\n    /**\n     * Add bundling support to a project. This is called implicitly when\n     * `bundleTask` is referenced first. It adds the dependency on `esbuild`,\n     * gitignore/npmignore, etc.\n     */\n    private addBundlingSupport() {\n        const ignoreEntry = `/${this.bundledir}/`;\n        this.project.addGitIgnore(ignoreEntry);\n        this.project.addPackageIgnore(`!${ignoreEntry}`); // include in tarball\n        const dep = this.esbuildVersion ? `esbuild@${this.esbuildVersion}` : \"esbuild\";\n        this.project.deps.addDependency(dep, DependencyType.BUILD);\n    }\n}\n\nexport interface Bundle {\n    /**\n     * The task that produces this bundle.\n     */\n    readonly bundleTask: Task;\n\n    /**\n     * Location of the output file (relative to project root).\n     */\n    readonly outfile: string;\n\n    /**\n     * Base directory containing the output file (relative to project root).\n     */\n    readonly outdir: string;\n}\n\n/**\n * Options for bundling.\n */\nexport interface BundlingOptions {\n    /**\n     * You can mark a file or a package as external to exclude it from your build.\n     * Instead of being bundled, the import will be preserved (using require for\n     * the iife and cjs formats and using import for the esm format) and will be\n     * evaluated at run time instead.\n     *\n     * This has several uses. First of all, it can be used to trim unnecessary\n     * code from your bundle for a code path that you know will never be executed.\n     * For example, a package may contain code that only runs in node but you will\n     * only be using that package in the browser. It can also be used to import\n     * code in node at run time from a package that cannot be bundled. For\n     * example, the fsevents package contains a native extension, which esbuild\n     * doesn't support.\n     *\n     * @default []\n     */\n    readonly externals?: string[];\n\n    /**\n     * Include a source map in the bundle.\n     *\n     * @default true\n     */\n    readonly sourcemap?: boolean;\n\n    /**\n     * esbuild target.\n     *\n     * @default \"esnext\"\n     */\n    readonly target?: string;\n\n    /**\n     * esbuild platform.\n     *\n     * @default \"node\"\n     */\n    readonly platform?: string;\n\n    /**\n     * Bundler output path relative to the asset's output directory.\n     * @default \"index.mjs\"\n     */\n    readonly outfile?: string;\n\n    /**\n     * The path of the tsconfig.json file to use for bundling\n     * @default \"tsconfig.json\"\n     */\n    readonly tsconfigPath?: string;\n\n    /**\n     * Map of file extensions (without dot) and loaders to use for this file type.\n     * Loaders are appended to the esbuild command by `--loader:.extension=loader`\n     */\n    readonly loaders?: { [key: string]: string };\n\n    /**\n     * Output format for the generated JavaScript files. There are currently three possible values that can be configured: `\"iife\"`, `\"cjs\"`, and `\"esm\"`.\n     *\n     * If not set (`undefined`), esbuild picks an output format for you based on `platform`:\n     * - `\"cjs\"` if `platform` is `\"node\"`\n     * - `\"iife\"` if `platform` is `\"browser\"`\n     * - `\"esm\"` if `platform` is `\"neutral\"`\n     *\n     * Note: If making a bundle to run under node with ESM, set `format` to `\"esm\"` instead of setting `platform` to `\"neutral\"`.\n     *\n     * @default undefined\n     *\n     * @see https://esbuild.github.io/api/#format\n     */\n    readonly format?: string;\n\n    /**\n     *\n     *\n     * @default true\n     */\n    readonly minify?: boolean;\n\n    /**\n     *\n     *\n     * @default false\n     */\n    readonly sourcesContent?: boolean;\n\n    /**\n     *\n     *\n     * @default \"module,main\"\n     */\n    readonly mainFields?: string;\n\n    /**\n     *\n     *\n     * @default \":js='import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)'\"\n     */\n    readonly banner?: string;\n}\n"]}
package/lib/index.d.ts CHANGED
@@ -1,5 +1,5 @@
1
- import { Project, cdk } from "projen";
2
- import { LambdaFunctionCodeBundleOptions } from "./lambda-function-code";
1
+ import { cdk, type Project } from "projen";
2
+ import { type LambdaFunctionCodeBundleOptions } from "./lambda-function-code";
3
3
  /**
4
4
  * Options for `LambdaFunctionCodeBundler`
5
5
  */
@@ -17,7 +17,7 @@ export interface LambdaFunctionCodeBundlerOptions {
17
17
  /**
18
18
  * Options for AWS Lambda Function bundling.
19
19
  */
20
- readonly buildOptions?: LambdaFunctionCodeBundleOptions;
20
+ readonly bundleOptions?: LambdaFunctionCodeBundleOptions;
21
21
  }
22
22
  /**
23
23
  * Creates Lambda Function code bundles from entrypoints discovered in the project's source tree.
package/lib/index.js CHANGED
@@ -16,10 +16,10 @@ class LambdaFunctionCodeBundler extends projen_1.cdk.AutoDiscoverBase {
16
16
  new lambda_function_code_1.LambdaFunctionCodeBundle(this.project, {
17
17
  entrypoint: entrypoint,
18
18
  extension: options.extension,
19
- ...options.buildOptions,
19
+ ...options.bundleOptions,
20
20
  });
21
21
  }
22
22
  }
23
23
  }
24
24
  exports.LambdaFunctionCodeBundler = LambdaFunctionCodeBundler;
25
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsbUNBQXNDO0FBQ3RDLGlFQUdnQztBQXdCaEM7O0dBRUc7QUFDSCxNQUFhLHlCQUEwQixTQUFRLFlBQUcsQ0FBQyxnQkFBZ0I7SUFDakUsWUFBWSxPQUFnQixFQUFFLE9BQXlDO1FBQ3JFLEtBQUssQ0FBQyxPQUFPLEVBQUU7WUFDYixVQUFVLEVBQUUsT0FBTyxDQUFDLE1BQU07WUFDMUIsU0FBUyxFQUFFLE9BQU8sQ0FBQyxTQUFTO1NBQzdCLENBQUMsQ0FBQztRQUVILEtBQUssTUFBTSxVQUFVLElBQUksSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQzFDLElBQUksK0NBQXdCLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRTtnQkFDekMsVUFBVSxFQUFFLFVBQVU7Z0JBQ3RCLFNBQVMsRUFBRSxPQUFPLENBQUMsU0FBUztnQkFDNUIsR0FBRyxPQUFPLENBQUMsWUFBWTthQUN4QixDQUFDLENBQUM7UUFDTCxDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBZkQsOERBZUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQcm9qZWN0LCBjZGsgfSBmcm9tIFwicHJvamVuXCI7XG5pbXBvcnQge1xuICBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGUsXG4gIExhbWJkYUZ1bmN0aW9uQ29kZUJ1bmRsZU9wdGlvbnMsXG59IGZyb20gXCIuL2xhbWJkYS1mdW5jdGlvbi1jb2RlXCI7XG5cbi8qKlxuICogT3B0aW9ucyBmb3IgYExhbWJkYUZ1bmN0aW9uQ29kZUJ1bmRsZXJgXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTGFtYmRhRnVuY3Rpb25Db2RlQnVuZGxlck9wdGlvbnMge1xuICAvKipcbiAgICogTG9jYXRlIGZpbGVzIHdpdGggdGhlIGdpdmVuIGV4dGVuc2lvbi5cbiAgICpcbiAgICogQGV4YW1wbGUgXCIubGFtYmRhLnRzXCJcbiAgICovXG4gIHJlYWRvbmx5IGV4dGVuc2lvbjogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBQcm9qZWN0IHNvdXJjZSB0cmVlIChyZWxhdGl2ZSB0byBwcm9qZWN0IG91dHB1dCBkaXJlY3RvcnkpLlxuICAgKi9cbiAgcmVhZG9ubHkgc3JjZGlyOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIE9wdGlvbnMgZm9yIEFXUyBMYW1iZGEgRnVuY3Rpb24gYnVuZGxpbmcuXG4gICAqL1xuICByZWFkb25seSBidWlsZE9wdGlvbnM/OiBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGVPcHRpb25zO1xufVxuXG4vKipcbiAqIENyZWF0ZXMgTGFtYmRhIEZ1bmN0aW9uIGNvZGUgYnVuZGxlcyBmcm9tIGVudHJ5cG9pbnRzIGRpc2NvdmVyZWQgaW4gdGhlIHByb2plY3QncyBzb3VyY2UgdHJlZS5cbiAqL1xuZXhwb3J0IGNsYXNzIExhbWJkYUZ1bmN0aW9uQ29kZUJ1bmRsZXIgZXh0ZW5kcyBjZGsuQXV0b0Rpc2NvdmVyQmFzZSB7XG4gIGNvbnN0cnVjdG9yKHByb2plY3Q6IFByb2plY3QsIG9wdGlvbnM6IExhbWJkYUZ1bmN0aW9uQ29kZUJ1bmRsZXJPcHRpb25zKSB7XG4gICAgc3VwZXIocHJvamVjdCwge1xuICAgICAgcHJvamVjdGRpcjogb3B0aW9ucy5zcmNkaXIsXG4gICAgICBleHRlbnNpb246IG9wdGlvbnMuZXh0ZW5zaW9uLFxuICAgIH0pO1xuXG4gICAgZm9yIChjb25zdCBlbnRyeXBvaW50IG9mIHRoaXMuZW50cnlwb2ludHMpIHtcbiAgICAgIG5ldyBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGUodGhpcy5wcm9qZWN0LCB7XG4gICAgICAgIGVudHJ5cG9pbnQ6IGVudHJ5cG9pbnQsXG4gICAgICAgIGV4dGVuc2lvbjogb3B0aW9ucy5leHRlbnNpb24sXG4gICAgICAgIC4uLm9wdGlvbnMuYnVpbGRPcHRpb25zLFxuICAgICAgfSk7XG4gICAgfVxuICB9XG59XG4iXX0=
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsbUNBQTJDO0FBQzNDLGlFQUF3RztBQXdCeEc7O0dBRUc7QUFDSCxNQUFhLHlCQUEwQixTQUFRLFlBQUcsQ0FBQyxnQkFBZ0I7SUFDL0QsWUFBWSxPQUFnQixFQUFFLE9BQXlDO1FBQ25FLEtBQUssQ0FBQyxPQUFPLEVBQUU7WUFDWCxVQUFVLEVBQUUsT0FBTyxDQUFDLE1BQU07WUFDMUIsU0FBUyxFQUFFLE9BQU8sQ0FBQyxTQUFTO1NBQy9CLENBQUMsQ0FBQztRQUVILEtBQUssTUFBTSxVQUFVLElBQUksSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQ3hDLElBQUksK0NBQXdCLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRTtnQkFDdkMsVUFBVSxFQUFFLFVBQVU7Z0JBQ3RCLFNBQVMsRUFBRSxPQUFPLENBQUMsU0FBUztnQkFDNUIsR0FBRyxPQUFPLENBQUMsYUFBYTthQUMzQixDQUFDLENBQUM7UUFDUCxDQUFDO0lBQ0wsQ0FBQztDQUNKO0FBZkQsOERBZUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjZGssIHR5cGUgUHJvamVjdCB9IGZyb20gXCJwcm9qZW5cIjtcbmltcG9ydCB7IExhbWJkYUZ1bmN0aW9uQ29kZUJ1bmRsZSwgdHlwZSBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGVPcHRpb25zIH0gZnJvbSBcIi4vbGFtYmRhLWZ1bmN0aW9uLWNvZGVcIjtcblxuLyoqXG4gKiBPcHRpb25zIGZvciBgTGFtYmRhRnVuY3Rpb25Db2RlQnVuZGxlcmBcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGVyT3B0aW9ucyB7XG4gICAgLyoqXG4gICAgICogTG9jYXRlIGZpbGVzIHdpdGggdGhlIGdpdmVuIGV4dGVuc2lvbi5cbiAgICAgKlxuICAgICAqIEBleGFtcGxlIFwiLmxhbWJkYS50c1wiXG4gICAgICovXG4gICAgcmVhZG9ubHkgZXh0ZW5zaW9uOiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiBQcm9qZWN0IHNvdXJjZSB0cmVlIChyZWxhdGl2ZSB0byBwcm9qZWN0IG91dHB1dCBkaXJlY3RvcnkpLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IHNyY2Rpcjogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogT3B0aW9ucyBmb3IgQVdTIExhbWJkYSBGdW5jdGlvbiBidW5kbGluZy5cbiAgICAgKi9cbiAgICByZWFkb25seSBidW5kbGVPcHRpb25zPzogTGFtYmRhRnVuY3Rpb25Db2RlQnVuZGxlT3B0aW9ucztcbn1cblxuLyoqXG4gKiBDcmVhdGVzIExhbWJkYSBGdW5jdGlvbiBjb2RlIGJ1bmRsZXMgZnJvbSBlbnRyeXBvaW50cyBkaXNjb3ZlcmVkIGluIHRoZSBwcm9qZWN0J3Mgc291cmNlIHRyZWUuXG4gKi9cbmV4cG9ydCBjbGFzcyBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGVyIGV4dGVuZHMgY2RrLkF1dG9EaXNjb3ZlckJhc2Uge1xuICAgIGNvbnN0cnVjdG9yKHByb2plY3Q6IFByb2plY3QsIG9wdGlvbnM6IExhbWJkYUZ1bmN0aW9uQ29kZUJ1bmRsZXJPcHRpb25zKSB7XG4gICAgICAgIHN1cGVyKHByb2plY3QsIHtcbiAgICAgICAgICAgIHByb2plY3RkaXI6IG9wdGlvbnMuc3JjZGlyLFxuICAgICAgICAgICAgZXh0ZW5zaW9uOiBvcHRpb25zLmV4dGVuc2lvbixcbiAgICAgICAgfSk7XG5cbiAgICAgICAgZm9yIChjb25zdCBlbnRyeXBvaW50IG9mIHRoaXMuZW50cnlwb2ludHMpIHtcbiAgICAgICAgICAgIG5ldyBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGUodGhpcy5wcm9qZWN0LCB7XG4gICAgICAgICAgICAgICAgZW50cnlwb2ludDogZW50cnlwb2ludCxcbiAgICAgICAgICAgICAgICBleHRlbnNpb246IG9wdGlvbnMuZXh0ZW5zaW9uLFxuICAgICAgICAgICAgICAgIC4uLm9wdGlvbnMuYnVuZGxlT3B0aW9ucyxcbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgfVxufVxuIl19
@@ -1,7 +1,7 @@
1
- import { BuildOptions } from "@mrgrain/cdk-esbuild";
2
- import { Component, Project } from "projen";
1
+ import { Component, type Project } from "projen";
2
+ import { type BundlingOptions } from "./bundler";
3
3
  /**
4
- * Options for `Function`.
4
+ * Options for `LambdaFunctionCodeBundle`.
5
5
  */
6
6
  export interface LambdaFunctionCodeBundleOptions {
7
7
  /**
@@ -35,10 +35,14 @@ export interface LambdaFunctionCodeBundleOptions {
35
35
  */
36
36
  readonly constructName?: string;
37
37
  /**
38
+ * Bundling options for this AWS Lambda Function.
38
39
  *
40
+ * If not specified the default bundling options specified for the project
41
+ * `Bundler` instance will be used.
39
42
  *
43
+ * @default - defaults
40
44
  */
41
- readonly buildOptions?: BuildOptions;
45
+ readonly bundlingOptions?: BundlingOptions;
42
46
  }
43
47
  /**
44
48
  * Generates a pre-bundled AWS Lambda Function code bundle construct from handler code.
@@ -15,20 +15,30 @@ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (
15
15
  }) : function(o, v) {
16
16
  o["default"] = v;
17
17
  });
18
- var __importStar = (this && this.__importStar) || function (mod) {
19
- if (mod && mod.__esModule) return mod;
20
- var result = {};
21
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
- __setModuleDefault(result, mod);
23
- return result;
24
- };
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
25
35
  Object.defineProperty(exports, "__esModule", { value: true });
26
36
  exports.LambdaFunctionCodeBundle = void 0;
27
- const path = __importStar(require("path"));
28
- const cdk_esbuild_1 = require("@mrgrain/cdk-esbuild");
37
+ const path = __importStar(require("node:path"));
29
38
  const case_1 = require("case");
30
39
  const projen_1 = require("projen");
31
- const internal_1 = require("./internal");
40
+ const bundler_1 = require("./bundler");
41
+ const utils_1 = require("./utils");
32
42
  /**
33
43
  * Generates a pre-bundled AWS Lambda Function code bundle construct from handler code.
34
44
  *
@@ -57,57 +67,38 @@ class LambdaFunctionCodeBundle extends projen_1.Component {
57
67
  * @param options Options
58
68
  */
59
69
  constructor(project, options) {
60
- var _a, _b, _c, _d;
61
70
  super(project);
71
+ const bundler = bundler_1.Bundler.of(project);
72
+ if (!bundler) {
73
+ throw new Error("No bundler found. Please add a Bundler component to your project.");
74
+ }
75
+ // allow Lambda handler code to import dev-deps since they are only needed
76
+ // during bundling
77
+ const eslint = projen_1.javascript.Eslint.of(project);
78
+ eslint?.allowDevDeps(options.entrypoint);
62
79
  const entrypoint = options.entrypoint;
63
- const extension = (_a = options.extension) !== null && _a !== void 0 ? _a : ".lambda.ts";
64
- const dirname = path.dirname(entrypoint);
65
- const basename = path.basename(entrypoint, extension);
80
+ const extension = options.extension ?? ".lambda.ts";
66
81
  if (!entrypoint.endsWith(extension)) {
67
82
  throw new Error(`${entrypoint} must have a ${extension} extension`);
68
83
  }
69
- const outdir = `${project.outdir}/assets`;
70
- const outfile = `${basename}.mjs`;
71
- const buildOptions = {
72
- banner: {
73
- ":js": "import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)",
74
- ":ts": "import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)",
75
- },
76
- bundle: true,
77
- drop: ["debugger"],
78
- external: [],
79
- format: "esm",
80
- legalComments: "none",
81
- minify: true,
82
- outExtension: { ".js": ".mjs", ".ts": ".mjs" },
83
- outdir: "bundled",
84
- platform: "node",
85
- sourcemap: "linked",
86
- sourcesContent: false,
87
- target: "esnext",
88
- treeShaking: true,
89
- tsconfig: (_b = project === null || project === void 0 ? void 0 : project.tsconfigDev) === null || _b === void 0 ? void 0 : _b.fileName,
90
- ...options.buildOptions,
91
- };
92
- new cdk_esbuild_1.EsbuildBundler(entrypoint, {
93
- buildOptions: {
94
- ...buildOptions,
95
- },
96
- });
97
- const basePath = path.posix.join(dirname, basename);
98
- const constructFile = (_c = options.constructFile) !== null && _c !== void 0 ? _c : `${basePath}-code.ts`;
84
+ const basePath = path.posix.join(path.dirname(entrypoint), path.basename(entrypoint, extension));
85
+ const constructFile = options.constructFile ?? `${basePath}-code.ts`;
99
86
  if (path.extname(constructFile) !== ".ts") {
100
87
  throw new Error(`Construct file name "${constructFile}" must have a .ts extension`);
101
88
  }
102
89
  // type names
103
- const constructName = (_d = options.constructName) !== null && _d !== void 0 ? _d : (0, case_1.pascal)(path.basename(basePath)) + "FunctionCode";
90
+ const constructName = options.constructName ?? `${(0, case_1.pascal)(path.basename(basePath))}FunctionCode`;
91
+ const bundle = bundler.addBundle(entrypoint, {
92
+ ...options.bundlingOptions,
93
+ tsconfigPath: project?.tsconfigDev?.fileName,
94
+ });
104
95
  // calculate the relative path between the directory containing the
105
96
  // generated construct source file to the directory containing the bundle
106
97
  // index.js by resolving them as absolute paths first.
107
98
  // e.g:
108
99
  // - outfileAbs => `/project-outdir/assets/foo/bar/baz/foo-function/index.js`
109
100
  // - constructAbs => `/project-outdir/src/foo/bar/baz/foo-function.ts`
110
- const outfileAbs = path.join(outdir, outfile);
101
+ const outfileAbs = path.join(project.outdir, bundle.outfile);
111
102
  const constructAbs = path.join(project.outdir, constructFile);
112
103
  const relativeOutfile = path.relative(path.dirname(constructAbs), path.dirname(outfileAbs));
113
104
  const src = new projen_1.SourceCode(project, constructFile);
@@ -118,10 +109,11 @@ class LambdaFunctionCodeBundle extends projen_1.Component {
118
109
  src.line("import { aws_lambda } from 'aws-cdk-lib';");
119
110
  src.line();
120
111
  src.open(`export const ${constructName} = aws_lambda.Code.fromAsset(`);
121
- src.line(`path.join(__dirname, '${(0, internal_1.convertToPosixPath)(relativeOutfile)}'),`);
112
+ src.line(`path.join(__dirname, '${(0, utils_1.convertToPosixPath)(relativeOutfile)}'),`);
122
113
  src.close(");");
123
114
  this.project.logger.verbose(`${basePath}: construct "${constructName}" generated under "${constructFile}"`);
115
+ this.project.logger.verbose(`${basePath}: bundle task "${bundle.bundleTask.name}"`);
124
116
  }
125
117
  }
126
118
  exports.LambdaFunctionCodeBundle = LambdaFunctionCodeBundle;
127
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-function-code.js","sourceRoot":"","sources":["../src/lambda-function-code.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,2CAA6B;AAC7B,sDAAoE;AACpE,+BAA8B;AAC9B,mCAAoE;AACpE,yCAAgD;AA+ChD;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAa,wBAAyB,SAAQ,kBAAS;IACrD;;;;;OAKG;IACH,YAAY,OAAgB,EAAE,OAAwC;;QACpE,KAAK,CAAC,OAAO,CAAC,CAAC;QAEf,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,CAAC;QACtC,MAAM,SAAS,GAAG,MAAA,OAAO,CAAC,SAAS,mCAAI,YAAY,CAAC;QACpD,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QACzC,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;QAEtD,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;YACpC,MAAM,IAAI,KAAK,CAAC,GAAG,UAAU,gBAAgB,SAAS,YAAY,CAAC,CAAC;QACtE,CAAC;QAED,MAAM,MAAM,GAAG,GAAG,OAAO,CAAC,MAAM,SAAS,CAAC;QAC1C,MAAM,OAAO,GAAG,GAAG,QAAQ,MAAM,CAAC;QAElC,MAAM,YAAY,GAAiB;YACjC,MAAM,EAAE;gBACN,KAAK,EACH,wHAAwH;gBAC1H,KAAK,EACH,wHAAwH;aAC3H;YACD,MAAM,EAAE,IAAI;YACZ,IAAI,EAAE,CAAC,UAAU,CAAC;YAClB,QAAQ,EAAE,EAAE;YACZ,MAAM,EAAE,KAAK;YACb,aAAa,EAAE,MAAM;YACrB,MAAM,EAAE,IAAI;YACZ,YAAY,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE;YAC9C,MAAM,EAAE,SAAS;YACjB,QAAQ,EAAE,MAAM;YAChB,SAAS,EAAE,QAAQ;YACnB,cAAc,EAAE,KAAK;YACrB,MAAM,EAAE,QAAQ;YAChB,WAAW,EAAE,IAAI;YACjB,QAAQ,EAAE,MAAC,OAAwC,aAAxC,OAAO,uBAAP,OAAO,CAAmC,WAAW,0CAC5D,QAAQ;YACZ,GAAG,OAAO,CAAC,YAAY;SACxB,CAAC;QAEF,IAAI,4BAAc,CAAC,UAAU,EAAE;YAC7B,YAAY,EAAE;gBACZ,GAAG,YAAY;aAChB;SACF,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;QAEpD,MAAM,aAAa,GAAG,MAAA,OAAO,CAAC,aAAa,mCAAI,GAAG,QAAQ,UAAU,CAAC;QAErE,IAAI,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,KAAK,KAAK,EAAE,CAAC;YAC1C,MAAM,IAAI,KAAK,CACb,wBAAwB,aAAa,6BAA6B,CACnE,CAAC;QACJ,CAAC;QAED,aAAa;QACb,MAAM,aAAa,GACjB,MAAA,OAAO,CAAC,aAAa,mCAAI,IAAA,aAAM,EAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,GAAG,cAAc,CAAC;QAE5E,mEAAmE;QACnE,yEAAyE;QACzE,sDAAsD;QACtD,OAAO;QACP,8EAA8E;QAC9E,uEAAuE;QACvE,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QAC9C,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;QAC9D,MAAM,eAAe,GAAG,IAAI,CAAC,QAAQ,CACnC,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,EAC1B,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CACzB,CAAC;QAEF,MAAM,GAAG,GAAG,IAAI,mBAAU,CAAC,OAAO,EAAE,aAAa,CAAC,CAAC;QACnD,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;YACf,GAAG,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;QAC/B,CAAC;QACD,GAAG,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC;QAC1C,GAAG,CAAC,IAAI,CAAC,2CAA2C,CAAC,CAAC;QACtD,GAAG,CAAC,IAAI,EAAE,CAAC;QACX,GAAG,CAAC,IAAI,CAAC,gBAAgB,aAAa,+BAA+B,CAAC,CAAC;QACvE,GAAG,CAAC,IAAI,CAAC,yBAAyB,IAAA,6BAAkB,EAAC,eAAe,CAAC,KAAK,CAAC,CAAC;QAC5E,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAEhB,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CACzB,GAAG,QAAQ,gBAAgB,aAAa,sBAAsB,aAAa,GAAG,CAC/E,CAAC;IACJ,CAAC;CACF;AA/FD,4DA+FC","sourcesContent":["import * as path from \"path\";\nimport { BuildOptions, EsbuildBundler } from \"@mrgrain/cdk-esbuild\";\nimport { pascal } from \"case\";\nimport { Component, Project, SourceCode, typescript } from \"projen\";\nimport { convertToPosixPath } from \"./internal\";\n\n/**\n * Options for `Function`.\n */\nexport interface LambdaFunctionCodeBundleOptions {\n  /**\n   * A path from the project root directory to a TypeScript file which contains\n   * the AWS Lambda handler entrypoint (exports a `handler` function).\n   *\n   * This is relative to the root directory of the project.\n   *\n   * @example \"src/subdir/foo.lambda.ts\"\n   */\n  readonly entrypoint: string;\n\n  /**\n   * Suffix for AWS Lambda handlers.\n   *\n   * @example \".lambda.ts\"\n   */\n  readonly extension: string;\n\n  /**\n   * The name of the generated TypeScript source file. This file should also be\n   * under the source tree.\n   *\n   * @default - The name of the entrypoint file, with the `-function.ts` suffix\n   * instead of `.lambda.ts`.\n   */\n  readonly constructFile?: string;\n\n  /**\n   * The name of the generated `lambda.Function` subclass.\n   *\n   * @default - A pascal cased version of the name of the entrypoint file, with\n   * the extension `Function` (e.g. `ResizeImageFunction`).\n   */\n  readonly constructName?: string;\n\n  /**\n   *\n   *\n   */\n  readonly buildOptions?: BuildOptions;\n}\n\n/**\n * Generates a pre-bundled AWS Lambda Function code bundle construct from handler code.\n *\n * To use this, create an AWS Lambda handler file under your source tree with\n * the `.lambda.ts` extension and add a `LambdaFunctionCodeBundle` component to your\n * typescript project pointing to this entrypoint.\n *\n * This will add a task to your \"compile\" step which will use `esbuild` to\n * bundle the handler code into the build directory. It will also generate a\n * file `src/foo-function.ts` with a custom AWS construct called `FooFunction`\n * which extends `@aws-cdk/aws-lambda.Function` which is bound to the bundled\n * handle through an asset.\n *\n * @example\n *\n * new LambdaFunctionCodeBundle(myProject, {\n *   srcdir: myProject.srcdir,\n *   entrypoint: 'src/foo.lambda.ts',\n * });\n */\nexport class LambdaFunctionCodeBundle extends Component {\n  /**\n   * Defines a pre-bundled AWS Lambda Function construct from handler code.\n   *\n   * @param project The project to use\n   * @param options Options\n   */\n  constructor(project: Project, options: LambdaFunctionCodeBundleOptions) {\n    super(project);\n\n    const entrypoint = options.entrypoint;\n    const extension = options.extension ?? \".lambda.ts\";\n    const dirname = path.dirname(entrypoint);\n    const basename = path.basename(entrypoint, extension);\n\n    if (!entrypoint.endsWith(extension)) {\n      throw new Error(`${entrypoint} must have a ${extension} extension`);\n    }\n\n    const outdir = `${project.outdir}/assets`;\n    const outfile = `${basename}.mjs`;\n\n    const buildOptions: BuildOptions = {\n      banner: {\n        \":js\":\n          \"import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)\",\n        \":ts\":\n          \"import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)\",\n      },\n      bundle: true,\n      drop: [\"debugger\"],\n      external: [],\n      format: \"esm\",\n      legalComments: \"none\",\n      minify: true,\n      outExtension: { \".js\": \".mjs\", \".ts\": \".mjs\" },\n      outdir: \"bundled\",\n      platform: \"node\",\n      sourcemap: \"linked\",\n      sourcesContent: false,\n      target: \"esnext\",\n      treeShaking: true,\n      tsconfig: (project as typescript.TypeScriptProject)?.tsconfigDev\n        ?.fileName,\n      ...options.buildOptions,\n    };\n\n    new EsbuildBundler(entrypoint, {\n      buildOptions: {\n        ...buildOptions,\n      },\n    });\n\n    const basePath = path.posix.join(dirname, basename);\n\n    const constructFile = options.constructFile ?? `${basePath}-code.ts`;\n\n    if (path.extname(constructFile) !== \".ts\") {\n      throw new Error(\n        `Construct file name \"${constructFile}\" must have a .ts extension`,\n      );\n    }\n\n    // type names\n    const constructName =\n      options.constructName ?? pascal(path.basename(basePath)) + \"FunctionCode\";\n\n    // calculate the relative path between the directory containing the\n    // generated construct source file to the directory containing the bundle\n    // index.js by resolving them as absolute paths first.\n    // e.g:\n    //  - outfileAbs => `/project-outdir/assets/foo/bar/baz/foo-function/index.js`\n    //  - constructAbs => `/project-outdir/src/foo/bar/baz/foo-function.ts`\n    const outfileAbs = path.join(outdir, outfile);\n    const constructAbs = path.join(project.outdir, constructFile);\n    const relativeOutfile = path.relative(\n      path.dirname(constructAbs),\n      path.dirname(outfileAbs),\n    );\n\n    const src = new SourceCode(project, constructFile);\n    if (src.marker) {\n      src.line(`// ${src.marker}`);\n    }\n    src.line(\"import * as path from 'path';\");\n    src.line(\"import { aws_lambda } from 'aws-cdk-lib';\");\n    src.line();\n    src.open(`export const ${constructName} = aws_lambda.Code.fromAsset(`);\n    src.line(`path.join(__dirname, '${convertToPosixPath(relativeOutfile)}'),`);\n    src.close(\");\");\n\n    this.project.logger.verbose(\n      `${basePath}: construct \"${constructName}\" generated under \"${constructFile}\"`,\n    );\n  }\n}\n"]}
119
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-function-code.js","sourceRoot":"","sources":["../src/lambda-function-code.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"node:path\";\nimport { pascal } from \"case\";\nimport { Component, javascript, type Project, SourceCode, type typescript } from \"projen\";\nimport { Bundler, type BundlingOptions } from \"./bundler\";\nimport { convertToPosixPath } from \"./utils\";\n\n/**\n * Options for `LambdaFunctionCodeBundle`.\n */\nexport interface LambdaFunctionCodeBundleOptions {\n    /**\n     * A path from the project root directory to a TypeScript file which contains\n     * the AWS Lambda handler entrypoint (exports a `handler` function).\n     *\n     * This is relative to the root directory of the project.\n     *\n     * @example \"src/subdir/foo.lambda.ts\"\n     */\n    readonly entrypoint: string;\n\n    /**\n     * Suffix for AWS Lambda handlers.\n     *\n     * @example \".lambda.ts\"\n     */\n    readonly extension: string;\n\n    /**\n     * The name of the generated TypeScript source file. This file should also be\n     * under the source tree.\n     *\n     * @default - The name of the entrypoint file, with the `-function.ts` suffix\n     * instead of `.lambda.ts`.\n     */\n    readonly constructFile?: string;\n\n    /**\n     * The name of the generated `lambda.Function` subclass.\n     *\n     * @default - A pascal cased version of the name of the entrypoint file, with\n     * the extension `Function` (e.g. `ResizeImageFunction`).\n     */\n    readonly constructName?: string;\n\n    /**\n     * Bundling options for this AWS Lambda Function.\n     *\n     * If not specified the default bundling options specified for the project\n     * `Bundler` instance will be used.\n     *\n     * @default - defaults\n     */\n    readonly bundlingOptions?: BundlingOptions;\n}\n\n/**\n * Generates a pre-bundled AWS Lambda Function code bundle construct from handler code.\n *\n * To use this, create an AWS Lambda handler file under your source tree with\n * the `.lambda.ts` extension and add a `LambdaFunctionCodeBundle` component to your\n * typescript project pointing to this entrypoint.\n *\n * This will add a task to your \"compile\" step which will use `esbuild` to\n * bundle the handler code into the build directory. It will also generate a\n * file `src/foo-function.ts` with a custom AWS construct called `FooFunction`\n * which extends `@aws-cdk/aws-lambda.Function` which is bound to the bundled\n * handle through an asset.\n *\n * @example\n *\n * new LambdaFunctionCodeBundle(myProject, {\n *   srcdir: myProject.srcdir,\n *   entrypoint: 'src/foo.lambda.ts',\n * });\n */\nexport class LambdaFunctionCodeBundle extends Component {\n    /**\n     * Defines a pre-bundled AWS Lambda Function construct from handler code.\n     *\n     * @param project The project to use\n     * @param options Options\n     */\n    constructor(project: Project, options: LambdaFunctionCodeBundleOptions) {\n        super(project);\n\n        const bundler = Bundler.of(project);\n        if (!bundler) {\n            throw new Error(\"No bundler found. Please add a Bundler component to your project.\");\n        }\n\n        // allow Lambda handler code to import dev-deps since they are only needed\n        // during bundling\n        const eslint = javascript.Eslint.of(project);\n        eslint?.allowDevDeps(options.entrypoint);\n\n        const entrypoint = options.entrypoint;\n\n        const extension = options.extension ?? \".lambda.ts\";\n\n        if (!entrypoint.endsWith(extension)) {\n            throw new Error(`${entrypoint} must have a ${extension} extension`);\n        }\n\n        const basePath = path.posix.join(path.dirname(entrypoint), path.basename(entrypoint, extension));\n        const constructFile = options.constructFile ?? `${basePath}-code.ts`;\n\n        if (path.extname(constructFile) !== \".ts\") {\n            throw new Error(`Construct file name \"${constructFile}\" must have a .ts extension`);\n        }\n\n        // type names\n        const constructName = options.constructName ?? `${pascal(path.basename(basePath))}FunctionCode`;\n\n        const bundle = bundler.addBundle(entrypoint, {\n            ...options.bundlingOptions,\n            tsconfigPath: (project as typescript.TypeScriptProject)?.tsconfigDev?.fileName,\n        });\n\n        // calculate the relative path between the directory containing the\n        // generated construct source file to the directory containing the bundle\n        // index.js by resolving them as absolute paths first.\n        // e.g:\n        //  - outfileAbs => `/project-outdir/assets/foo/bar/baz/foo-function/index.js`\n        //  - constructAbs => `/project-outdir/src/foo/bar/baz/foo-function.ts`\n        const outfileAbs = path.join(project.outdir, bundle.outfile);\n        const constructAbs = path.join(project.outdir, constructFile);\n        const relativeOutfile = path.relative(path.dirname(constructAbs), path.dirname(outfileAbs));\n\n        const src = new SourceCode(project, constructFile);\n        if (src.marker) {\n            src.line(`// ${src.marker}`);\n        }\n        src.line(\"import * as path from 'path';\");\n        src.line(\"import { aws_lambda } from 'aws-cdk-lib';\");\n        src.line();\n        src.open(`export const ${constructName} = aws_lambda.Code.fromAsset(`);\n        src.line(`path.join(__dirname, '${convertToPosixPath(relativeOutfile)}'),`);\n        src.close(\");\");\n\n        this.project.logger.verbose(`${basePath}: construct \"${constructName}\" generated under \"${constructFile}\"`);\n        this.project.logger.verbose(`${basePath}: bundle task \"${bundle.bundleTask.name}\"`);\n    }\n}\n"]}
@@ -1,3 +1,4 @@
1
+ export declare function renderBundleName(entrypoint: string): string;
1
2
  /**
2
3
  * Converts the given path string to posix if it wasn't already.
3
4
  */
package/lib/utils.js ADDED
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.renderBundleName = renderBundleName;
4
+ exports.convertToPosixPath = convertToPosixPath;
5
+ const node_path_1 = require("node:path");
6
+ function renderBundleName(entrypoint) {
7
+ const parts = (0, node_path_1.join)(entrypoint).split(node_path_1.sep);
8
+ if (parts[0] === "src") {
9
+ parts.shift(); // just remove 'src' if its the first element for ergonomics
10
+ }
11
+ const p = parts.join(node_path_1.sep);
12
+ const dir = (0, node_path_1.dirname)(p);
13
+ const base = (0, node_path_1.basename)(p, (0, node_path_1.extname)(p));
14
+ return (0, node_path_1.join)(dir, base);
15
+ }
16
+ /**
17
+ * Converts the given path string to posix if it wasn't already.
18
+ */
19
+ function convertToPosixPath(p) {
20
+ return p.split(node_path_1.sep).join(node_path_1.posix.sep);
21
+ }
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFFQSw0Q0FVQztBQUtELGdEQUVDO0FBbkJELHlDQUF5RTtBQUV6RSxTQUFnQixnQkFBZ0IsQ0FBQyxVQUFrQjtJQUMvQyxNQUFNLEtBQUssR0FBRyxJQUFBLGdCQUFJLEVBQUMsVUFBVSxDQUFDLENBQUMsS0FBSyxDQUFDLGVBQUcsQ0FBQyxDQUFDO0lBQzFDLElBQUksS0FBSyxDQUFDLENBQUMsQ0FBQyxLQUFLLEtBQUssRUFBRSxDQUFDO1FBQ3JCLEtBQUssQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDLDREQUE0RDtJQUMvRSxDQUFDO0lBRUQsTUFBTSxDQUFDLEdBQUcsS0FBSyxDQUFDLElBQUksQ0FBQyxlQUFHLENBQUMsQ0FBQztJQUMxQixNQUFNLEdBQUcsR0FBRyxJQUFBLG1CQUFPLEVBQUMsQ0FBQyxDQUFDLENBQUM7SUFDdkIsTUFBTSxJQUFJLEdBQUcsSUFBQSxvQkFBUSxFQUFDLENBQUMsRUFBRSxJQUFBLG1CQUFPLEVBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNyQyxPQUFPLElBQUEsZ0JBQUksRUFBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7QUFDM0IsQ0FBQztBQUVEOztHQUVHO0FBQ0gsU0FBZ0Isa0JBQWtCLENBQUMsQ0FBUztJQUN4QyxPQUFPLENBQUMsQ0FBQyxLQUFLLENBQUMsZUFBRyxDQUFDLENBQUMsSUFBSSxDQUFDLGlCQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7QUFDeEMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGJhc2VuYW1lLCBkaXJuYW1lLCBleHRuYW1lLCBqb2luLCBwb3NpeCwgc2VwIH0gZnJvbSBcIm5vZGU6cGF0aFwiO1xuXG5leHBvcnQgZnVuY3Rpb24gcmVuZGVyQnVuZGxlTmFtZShlbnRyeXBvaW50OiBzdHJpbmcpIHtcbiAgICBjb25zdCBwYXJ0cyA9IGpvaW4oZW50cnlwb2ludCkuc3BsaXQoc2VwKTtcbiAgICBpZiAocGFydHNbMF0gPT09IFwic3JjXCIpIHtcbiAgICAgICAgcGFydHMuc2hpZnQoKTsgLy8ganVzdCByZW1vdmUgJ3NyYycgaWYgaXRzIHRoZSBmaXJzdCBlbGVtZW50IGZvciBlcmdvbm9taWNzXG4gICAgfVxuXG4gICAgY29uc3QgcCA9IHBhcnRzLmpvaW4oc2VwKTtcbiAgICBjb25zdCBkaXIgPSBkaXJuYW1lKHApO1xuICAgIGNvbnN0IGJhc2UgPSBiYXNlbmFtZShwLCBleHRuYW1lKHApKTtcbiAgICByZXR1cm4gam9pbihkaXIsIGJhc2UpO1xufVxuXG4vKipcbiAqIENvbnZlcnRzIHRoZSBnaXZlbiBwYXRoIHN0cmluZyB0byBwb3NpeCBpZiBpdCB3YXNuJ3QgYWxyZWFkeS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNvbnZlcnRUb1Bvc2l4UGF0aChwOiBzdHJpbmcpIHtcbiAgICByZXR1cm4gcC5zcGxpdChzZXApLmpvaW4ocG9zaXguc2VwKTtcbn1cbiJdfQ==
package/package.json CHANGED
@@ -1,13 +1,38 @@
1
1
  {
2
2
  "name": "@nikovirtala/projen-bundle-lambda-function-code",
3
+ "repository": {
4
+ "type": "git",
5
+ "url": "https://github.com/nikovirtala/projen-bundle-lambda-function-code.git"
6
+ },
7
+ "devDependencies": {
8
+ "@biomejs/biome": "^2",
9
+ "@types/node": "^18.19.130",
10
+ "commit-and-tag-version": "^12",
11
+ "constructs": "^10.0.0",
12
+ "projen": "^0.98.10",
13
+ "ts-node": "^10.9.2",
14
+ "typescript": "^5.9.3"
15
+ },
16
+ "dependencies": {
17
+ "case": "^1.6.3",
18
+ "projen": "^0.98.10"
19
+ },
20
+ "main": "lib/index.js",
21
+ "license": "MIT",
22
+ "publishConfig": {
23
+ "access": "public"
24
+ },
25
+ "version": "0.0.16",
26
+ "types": "lib/index.d.ts",
27
+ "//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\".",
3
28
  "scripts": {
29
+ "biome": "npx projen biome",
4
30
  "build": "npx projen build",
5
31
  "bump": "npx projen bump",
6
32
  "clobber": "npx projen clobber",
7
33
  "compile": "npx projen compile",
8
34
  "default": "npx projen default",
9
35
  "eject": "npx projen eject",
10
- "eslint": "npx projen eslint",
11
36
  "package": "npx projen package",
12
37
  "post-compile": "npx projen post-compile",
13
38
  "post-upgrade": "npx projen post-upgrade",
@@ -18,39 +43,5 @@
18
43
  "upgrade": "npx projen upgrade",
19
44
  "watch": "npx projen watch",
20
45
  "projen": "npx projen"
21
- },
22
- "devDependencies": {
23
- "@types/node": "^18",
24
- "@typescript-eslint/eslint-plugin": "^6",
25
- "@typescript-eslint/parser": "^6",
26
- "aws-cdk-lib": "2.51.0",
27
- "constructs": "^10.0.0",
28
- "eslint": "^8",
29
- "eslint-config-prettier": "^9.0.0",
30
- "eslint-import-resolver-typescript": "^3.6.1",
31
- "eslint-plugin-import": "^2.29.0",
32
- "eslint-plugin-prettier": "^5.0.1",
33
- "npm-check-updates": "^16",
34
- "prettier": "^3.1.0",
35
- "projen": "^0.77.1",
36
- "standard-version": "^9",
37
- "ts-node": "^10.9.1",
38
- "typescript": "^5.3.2"
39
- },
40
- "peerDependencies": {
41
- "aws-cdk-lib": "^2.51.0"
42
- },
43
- "dependencies": {
44
- "@mrgrain/cdk-esbuild": "^5.0.5",
45
- "case": "^1.6.3",
46
- "projen": "^0.77.1"
47
- },
48
- "main": "lib/index.js",
49
- "license": "MIT",
50
- "publishConfig": {
51
- "access": "public"
52
- },
53
- "version": "0.0.14",
54
- "types": "lib/index.d.ts",
55
- "//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\"."
56
- }
46
+ }
47
+ }
package/.gitattributes DELETED
@@ -1,23 +0,0 @@
1
- # ~~ Generated by projen. To modify, edit .projenrc.ts and run "npx projen".
2
-
3
- /.eslintrc.json linguist-generated
4
- /.gitattributes linguist-generated
5
- /.github/pull_request_template.md linguist-generated
6
- /.github/workflows/build.yml linguist-generated
7
- /.github/workflows/pull-request-lint.yml linguist-generated
8
- /.github/workflows/release.yml linguist-generated
9
- /.github/workflows/upgrade-main.yml linguist-generated
10
- /.gitignore linguist-generated
11
- /.mergify.yml linguist-generated
12
- /.npmignore linguist-generated
13
- /.prettierignore linguist-generated
14
- /.prettierrc.json linguist-generated
15
- /.projen/** linguist-generated
16
- /.projen/deps.json linguist-generated
17
- /.projen/files.json linguist-generated
18
- /.projen/tasks.json linguist-generated
19
- /LICENSE linguist-generated
20
- /package.json linguist-generated
21
- /tsconfig.dev.json linguist-generated
22
- /tsconfig.json linguist-generated
23
- /yarn.lock linguist-generated
package/.prettierignore DELETED
@@ -1 +0,0 @@
1
- # ~~ Generated by projen. To modify, edit .projenrc.ts and run "npx projen".
package/.prettierrc.json DELETED
@@ -1,3 +0,0 @@
1
- {
2
- "overrides": []
3
- }
package/.projenrc.ts DELETED
@@ -1,15 +0,0 @@
1
- import { typescript, javascript } from "projen";
2
- const project = new typescript.TypeScriptProject({
3
- copyrightOwner: "Niko Virtala",
4
- defaultReleaseBranch: "main",
5
- deps: ["case", "projen", "@mrgrain/cdk-esbuild"],
6
- jest: false,
7
- license: "MIT",
8
- name: "@nikovirtala/projen-bundle-lambda-function-code",
9
- npmAccess: javascript.NpmAccess.PUBLIC,
10
- peerDeps: ["aws-cdk-lib@^2.51.0"],
11
- prettier: true,
12
- projenrcTs: true,
13
- releaseToNpm: true,
14
- });
15
- project.synth();
package/lib/internal.js DELETED
@@ -1,12 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.convertToPosixPath = void 0;
4
- const path_1 = require("path");
5
- /**
6
- * Converts the given path string to posix if it wasn't already.
7
- */
8
- function convertToPosixPath(p) {
9
- return p.split(path_1.sep).join(path_1.posix.sep);
10
- }
11
- exports.convertToPosixPath = convertToPosixPath;
12
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJuYWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW50ZXJuYWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsK0JBQWtDO0FBRWxDOztHQUVHO0FBQ0gsU0FBZ0Isa0JBQWtCLENBQUMsQ0FBUztJQUMxQyxPQUFPLENBQUMsQ0FBQyxLQUFLLENBQUMsVUFBRyxDQUFDLENBQUMsSUFBSSxDQUFDLFlBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQztBQUN0QyxDQUFDO0FBRkQsZ0RBRUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBzZXAsIHBvc2l4IH0gZnJvbSBcInBhdGhcIjtcblxuLyoqXG4gKiBDb252ZXJ0cyB0aGUgZ2l2ZW4gcGF0aCBzdHJpbmcgdG8gcG9zaXggaWYgaXQgd2Fzbid0IGFscmVhZHkuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb252ZXJ0VG9Qb3NpeFBhdGgocDogc3RyaW5nKSB7XG4gIHJldHVybiBwLnNwbGl0KHNlcCkuam9pbihwb3NpeC5zZXApO1xufVxuIl19
@@ -1,3 +0,0 @@
1
- <?xml version="1.0" encoding="UTF-8"?>
2
- <testsuites name="jest tests" tests="0" failures="0" errors="0" time="0.001">
3
- </testsuites>