@nikovirtala/projen-bundle-lambda-function-code 0.0.14 → 0.0.15
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/.projenrc.ts +1 -2
- package/lib/bundler.d.ts +189 -0
- package/lib/bundler.js +162 -0
- package/lib/index.d.ts +1 -1
- package/lib/index.js +2 -2
- package/lib/lambda-function-code.d.ts +7 -3
- package/lib/lambda-function-code.js +21 -38
- package/lib/{internal.d.ts → utils.d.ts} +1 -0
- package/lib/utils.js +23 -0
- package/package.json +1 -6
- package/lib/internal.js +0 -12
package/.projenrc.ts
CHANGED
|
@@ -2,12 +2,11 @@ import { typescript, javascript } from "projen";
|
|
|
2
2
|
const project = new typescript.TypeScriptProject({
|
|
3
3
|
copyrightOwner: "Niko Virtala",
|
|
4
4
|
defaultReleaseBranch: "main",
|
|
5
|
-
deps: ["case", "projen"
|
|
5
|
+
deps: ["case", "projen"],
|
|
6
6
|
jest: false,
|
|
7
7
|
license: "MIT",
|
|
8
8
|
name: "@nikovirtala/projen-bundle-lambda-function-code",
|
|
9
9
|
npmAccess: javascript.NpmAccess.PUBLIC,
|
|
10
|
-
peerDeps: ["aws-cdk-lib@^2.51.0"],
|
|
11
10
|
prettier: true,
|
|
12
11
|
projenrcTs: true,
|
|
13
12
|
releaseToNpm: true,
|
package/lib/bundler.d.ts
ADDED
|
@@ -0,0 +1,189 @@
|
|
|
1
|
+
import { Component, Project, 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,162 @@
|
|
|
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 (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
|
+
};
|
|
25
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
|
+
exports.Bundler = void 0;
|
|
27
|
+
const path = __importStar(require("path"));
|
|
28
|
+
const projen_1 = require("projen");
|
|
29
|
+
const utils_1 = require("./utils");
|
|
30
|
+
/**
|
|
31
|
+
* Adds support for bundling JavaScript applications and dependencies into a
|
|
32
|
+
* single file. In the future, this will also supports bundling websites.
|
|
33
|
+
*/
|
|
34
|
+
class Bundler extends projen_1.Component {
|
|
35
|
+
/**
|
|
36
|
+
* Returns the `Bundler` instance associated with a project or `undefined` if
|
|
37
|
+
* there is no Bundler.
|
|
38
|
+
* @param project The project
|
|
39
|
+
* @returns A bundler
|
|
40
|
+
*/
|
|
41
|
+
static of(project) {
|
|
42
|
+
const isBundler = (o) => o instanceof Bundler;
|
|
43
|
+
return project.components.find(isBundler);
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Creates a `Bundler`.
|
|
47
|
+
*/
|
|
48
|
+
constructor(project, options = {}) {
|
|
49
|
+
var _a, _b;
|
|
50
|
+
super(project);
|
|
51
|
+
this.esbuildVersion = options.esbuildVersion;
|
|
52
|
+
this.bundledir = (_a = options.assetsDir) !== null && _a !== void 0 ? _a : "assets";
|
|
53
|
+
this.addToPreCompile = (_b = options.addToPreCompile) !== null && _b !== void 0 ? _b : true;
|
|
54
|
+
this.loaders = options.loaders;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Gets or creates the singleton "bundle" task of the project.
|
|
58
|
+
*
|
|
59
|
+
* If the project doesn't have a "bundle" task, it will be created and spawned
|
|
60
|
+
* during the pre-compile phase.
|
|
61
|
+
*/
|
|
62
|
+
get bundleTask() {
|
|
63
|
+
if (!this._task) {
|
|
64
|
+
this.addBundlingSupport();
|
|
65
|
+
this._task = this.project.tasks.addTask("bundle", {
|
|
66
|
+
description: "Prepare assets",
|
|
67
|
+
});
|
|
68
|
+
// install the bundle task into the pre-compile phase.
|
|
69
|
+
if (this.addToPreCompile) {
|
|
70
|
+
this.project.preCompileTask.spawn(this._task);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
return this._task;
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Adds a task to the project which bundles a specific entrypoint and all of
|
|
77
|
+
* its dependencies into a single javascript output file.
|
|
78
|
+
*
|
|
79
|
+
* @param entrypoint The relative path of the artifact within the project
|
|
80
|
+
* @param options Bundling options
|
|
81
|
+
*/
|
|
82
|
+
addBundle(entrypoint, options) {
|
|
83
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o;
|
|
84
|
+
const name = (0, utils_1.renderBundleName)(entrypoint);
|
|
85
|
+
const outdir = path.posix.join(this.bundledir, name);
|
|
86
|
+
const outfile = path.posix.join(outdir, (_a = options.outfile) !== null && _a !== void 0 ? _a : "index.mjs");
|
|
87
|
+
const target = (_b = options.target) !== null && _b !== void 0 ? _b : "esnext";
|
|
88
|
+
const platform = (_c = options.platform) !== null && _c !== void 0 ? _c : "node";
|
|
89
|
+
const format = (_d = options.format) !== null && _d !== void 0 ? _d : "esm";
|
|
90
|
+
const sourcemap = (_e = options.sourcemap) !== null && _e !== void 0 ? _e : true;
|
|
91
|
+
const sourcesContent = (_f = options.sourcesContent) !== null && _f !== void 0 ? _f : false;
|
|
92
|
+
const minify = (_g = options.minify) !== null && _g !== void 0 ? _g : true;
|
|
93
|
+
const mainFields = (_h = options.mainFields) !== null && _h !== void 0 ? _h : "module,main";
|
|
94
|
+
const banner = (_j = options.banner) !== null && _j !== void 0 ? _j : ":js=import { createRequire as topLevelCreateRequire } from 'module';const require = topLevelCreateRequire(import.meta.url)";
|
|
95
|
+
const args = [
|
|
96
|
+
"esbuild",
|
|
97
|
+
"--bundle",
|
|
98
|
+
entrypoint,
|
|
99
|
+
`--target="${target}"`,
|
|
100
|
+
`--platform="${platform}"`,
|
|
101
|
+
`--outfile="${outfile}"`,
|
|
102
|
+
];
|
|
103
|
+
const tsconfig = (_k = options.tsconfigPath) !== null && _k !== void 0 ? _k : false;
|
|
104
|
+
if (tsconfig) {
|
|
105
|
+
args.push(`--tsconfig="${tsconfig}"`);
|
|
106
|
+
}
|
|
107
|
+
for (const x of (_l = options.externals) !== null && _l !== void 0 ? _l : []) {
|
|
108
|
+
args.push(`--external:${x}`);
|
|
109
|
+
}
|
|
110
|
+
if (sourcemap) {
|
|
111
|
+
args.push("--sourcemap");
|
|
112
|
+
}
|
|
113
|
+
if (sourcesContent) {
|
|
114
|
+
args.push(`--sources-content=${sourcesContent}`);
|
|
115
|
+
}
|
|
116
|
+
if (minify) {
|
|
117
|
+
args.push("--minify");
|
|
118
|
+
}
|
|
119
|
+
if (format) {
|
|
120
|
+
args.push(`--format=${format}`);
|
|
121
|
+
}
|
|
122
|
+
if (mainFields) {
|
|
123
|
+
args.push(`--main-fields=${mainFields}`);
|
|
124
|
+
}
|
|
125
|
+
if (banner) {
|
|
126
|
+
args.push(`--banner${banner}`);
|
|
127
|
+
}
|
|
128
|
+
const loaders = ((_m = options.loaders) !== null && _m !== void 0 ? _m : false) ? options.loaders : (_o = this.loaders) !== null && _o !== void 0 ? _o : false;
|
|
129
|
+
if (loaders) {
|
|
130
|
+
for (let [extension, loader] of Object.entries(loaders)) {
|
|
131
|
+
args.push(`--loader:.${extension}=${loader}`);
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
console.log({ args: args });
|
|
135
|
+
const bundleTask = this.project.addTask(`bundle:${name}`, {
|
|
136
|
+
description: `Create a JavaScript bundle from ${entrypoint}`,
|
|
137
|
+
exec: args.join(" "),
|
|
138
|
+
});
|
|
139
|
+
this.bundleTask.spawn(bundleTask);
|
|
140
|
+
return {
|
|
141
|
+
bundleTask: bundleTask,
|
|
142
|
+
outdir: outdir,
|
|
143
|
+
outfile: outfile,
|
|
144
|
+
};
|
|
145
|
+
}
|
|
146
|
+
/**
|
|
147
|
+
* Add bundling support to a project. This is called implicitly when
|
|
148
|
+
* `bundleTask` is referenced first. It adds the dependency on `esbuild`,
|
|
149
|
+
* gitignore/npmignore, etc.
|
|
150
|
+
*/
|
|
151
|
+
addBundlingSupport() {
|
|
152
|
+
const ignoreEntry = `/${this.bundledir}/`;
|
|
153
|
+
this.project.addGitIgnore(ignoreEntry);
|
|
154
|
+
this.project.addPackageIgnore(`!${ignoreEntry}`); // include in tarball
|
|
155
|
+
const dep = this.esbuildVersion
|
|
156
|
+
? `esbuild@${this.esbuildVersion}`
|
|
157
|
+
: "esbuild";
|
|
158
|
+
this.project.deps.addDependency(dep, projen_1.DependencyType.BUILD);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
exports.Bundler = Bundler;
|
|
162
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bundler.js","sourceRoot":"","sources":["../src/bundler.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"path\";\nimport { Component, DependencyType, Project, 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 =\n      options.loaders ?? false ? options.loaders : this.loaders ?? false;\n    if (loaders) {\n      for (let [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\n      ? `esbuild@${this.esbuildVersion}`\n      : \"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
|
@@ -17,7 +17,7 @@ export interface LambdaFunctionCodeBundlerOptions {
|
|
|
17
17
|
/**
|
|
18
18
|
* Options for AWS Lambda Function bundling.
|
|
19
19
|
*/
|
|
20
|
-
readonly
|
|
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.
|
|
19
|
+
...options.bundleOptions,
|
|
20
20
|
});
|
|
21
21
|
}
|
|
22
22
|
}
|
|
23
23
|
}
|
|
24
24
|
exports.LambdaFunctionCodeBundler = LambdaFunctionCodeBundler;
|
|
25
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
25
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsbUNBQXNDO0FBQ3RDLGlFQUdnQztBQXdCaEM7O0dBRUc7QUFDSCxNQUFhLHlCQUEwQixTQUFRLFlBQUcsQ0FBQyxnQkFBZ0I7SUFDakUsWUFBWSxPQUFnQixFQUFFLE9BQXlDO1FBQ3JFLEtBQUssQ0FBQyxPQUFPLEVBQUU7WUFDYixVQUFVLEVBQUUsT0FBTyxDQUFDLE1BQU07WUFDMUIsU0FBUyxFQUFFLE9BQU8sQ0FBQyxTQUFTO1NBQzdCLENBQUMsQ0FBQztRQUVILEtBQUssTUFBTSxVQUFVLElBQUksSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQzFDLElBQUksK0NBQXdCLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRTtnQkFDekMsVUFBVSxFQUFFLFVBQVU7Z0JBQ3RCLFNBQVMsRUFBRSxPQUFPLENBQUMsU0FBUztnQkFDNUIsR0FBRyxPQUFPLENBQUMsYUFBYTthQUN6QixDQUFDLENBQUM7UUFDTCxDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBZkQsOERBZUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQcm9qZWN0LCBjZGsgfSBmcm9tIFwicHJvamVuXCI7XG5pbXBvcnQge1xuICBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGUsXG4gIExhbWJkYUZ1bmN0aW9uQ29kZUJ1bmRsZU9wdGlvbnMsXG59IGZyb20gXCIuL2xhbWJkYS1mdW5jdGlvbi1jb2RlXCI7XG5cbi8qKlxuICogT3B0aW9ucyBmb3IgYExhbWJkYUZ1bmN0aW9uQ29kZUJ1bmRsZXJgXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTGFtYmRhRnVuY3Rpb25Db2RlQnVuZGxlck9wdGlvbnMge1xuICAvKipcbiAgICogTG9jYXRlIGZpbGVzIHdpdGggdGhlIGdpdmVuIGV4dGVuc2lvbi5cbiAgICpcbiAgICogQGV4YW1wbGUgXCIubGFtYmRhLnRzXCJcbiAgICovXG4gIHJlYWRvbmx5IGV4dGVuc2lvbjogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBQcm9qZWN0IHNvdXJjZSB0cmVlIChyZWxhdGl2ZSB0byBwcm9qZWN0IG91dHB1dCBkaXJlY3RvcnkpLlxuICAgKi9cbiAgcmVhZG9ubHkgc3JjZGlyOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIE9wdGlvbnMgZm9yIEFXUyBMYW1iZGEgRnVuY3Rpb24gYnVuZGxpbmcuXG4gICAqL1xuICByZWFkb25seSBidW5kbGVPcHRpb25zPzogTGFtYmRhRnVuY3Rpb25Db2RlQnVuZGxlT3B0aW9ucztcbn1cblxuLyoqXG4gKiBDcmVhdGVzIExhbWJkYSBGdW5jdGlvbiBjb2RlIGJ1bmRsZXMgZnJvbSBlbnRyeXBvaW50cyBkaXNjb3ZlcmVkIGluIHRoZSBwcm9qZWN0J3Mgc291cmNlIHRyZWUuXG4gKi9cbmV4cG9ydCBjbGFzcyBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGVyIGV4dGVuZHMgY2RrLkF1dG9EaXNjb3ZlckJhc2Uge1xuICBjb25zdHJ1Y3Rvcihwcm9qZWN0OiBQcm9qZWN0LCBvcHRpb25zOiBMYW1iZGFGdW5jdGlvbkNvZGVCdW5kbGVyT3B0aW9ucykge1xuICAgIHN1cGVyKHByb2plY3QsIHtcbiAgICAgIHByb2plY3RkaXI6IG9wdGlvbnMuc3JjZGlyLFxuICAgICAgZXh0ZW5zaW9uOiBvcHRpb25zLmV4dGVuc2lvbixcbiAgICB9KTtcblxuICAgIGZvciAoY29uc3QgZW50cnlwb2ludCBvZiB0aGlzLmVudHJ5cG9pbnRzKSB7XG4gICAgICBuZXcgTGFtYmRhRnVuY3Rpb25Db2RlQnVuZGxlKHRoaXMucHJvamVjdCwge1xuICAgICAgICBlbnRyeXBvaW50OiBlbnRyeXBvaW50LFxuICAgICAgICBleHRlbnNpb246IG9wdGlvbnMuZXh0ZW5zaW9uLFxuICAgICAgICAuLi5vcHRpb25zLmJ1bmRsZU9wdGlvbnMsXG4gICAgICB9KTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
|
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
import { BuildOptions } from "@mrgrain/cdk-esbuild";
|
|
2
1
|
import { Component, Project } from "projen";
|
|
2
|
+
import { BundlingOptions } from "./bundler";
|
|
3
3
|
/**
|
|
4
|
-
* Options for `
|
|
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
|
|
45
|
+
readonly bundlingOptions?: BundlingOptions;
|
|
42
46
|
}
|
|
43
47
|
/**
|
|
44
48
|
* Generates a pre-bundled AWS Lambda Function code bundle construct from handler code.
|
|
@@ -25,10 +25,10 @@ var __importStar = (this && this.__importStar) || function (mod) {
|
|
|
25
25
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
26
|
exports.LambdaFunctionCodeBundle = void 0;
|
|
27
27
|
const path = __importStar(require("path"));
|
|
28
|
-
const cdk_esbuild_1 = require("@mrgrain/cdk-esbuild");
|
|
29
28
|
const case_1 = require("case");
|
|
30
29
|
const projen_1 = require("projen");
|
|
31
|
-
const
|
|
30
|
+
const bundler_1 = require("./bundler");
|
|
31
|
+
const utils_1 = require("./utils");
|
|
32
32
|
/**
|
|
33
33
|
* Generates a pre-bundled AWS Lambda Function code bundle construct from handler code.
|
|
34
34
|
*
|
|
@@ -59,55 +59,37 @@ class LambdaFunctionCodeBundle extends projen_1.Component {
|
|
|
59
59
|
constructor(project, options) {
|
|
60
60
|
var _a, _b, _c, _d;
|
|
61
61
|
super(project);
|
|
62
|
+
const bundler = bundler_1.Bundler.of(project);
|
|
63
|
+
if (!bundler) {
|
|
64
|
+
throw new Error("No bundler found. Please add a Bundler component to your project.");
|
|
65
|
+
}
|
|
66
|
+
// allow Lambda handler code to import dev-deps since they are only needed
|
|
67
|
+
// during bundling
|
|
68
|
+
const eslint = projen_1.javascript.Eslint.of(project);
|
|
69
|
+
eslint === null || eslint === void 0 ? void 0 : eslint.allowDevDeps(options.entrypoint);
|
|
62
70
|
const entrypoint = options.entrypoint;
|
|
63
71
|
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);
|
|
66
72
|
if (!entrypoint.endsWith(extension)) {
|
|
67
73
|
throw new Error(`${entrypoint} must have a ${extension} extension`);
|
|
68
74
|
}
|
|
69
|
-
const
|
|
70
|
-
const
|
|
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`;
|
|
75
|
+
const basePath = path.posix.join(path.dirname(entrypoint), path.basename(entrypoint, extension));
|
|
76
|
+
const constructFile = (_b = options.constructFile) !== null && _b !== void 0 ? _b : `${basePath}-code.ts`;
|
|
99
77
|
if (path.extname(constructFile) !== ".ts") {
|
|
100
78
|
throw new Error(`Construct file name "${constructFile}" must have a .ts extension`);
|
|
101
79
|
}
|
|
102
80
|
// type names
|
|
103
|
-
const constructName = (
|
|
81
|
+
const constructName = (_c = options.constructName) !== null && _c !== void 0 ? _c : (0, case_1.pascal)(path.basename(basePath)) + "FunctionCode";
|
|
82
|
+
const bundle = bundler.addBundle(entrypoint, {
|
|
83
|
+
...options.bundlingOptions,
|
|
84
|
+
tsconfigPath: (_d = project === null || project === void 0 ? void 0 : project.tsconfigDev) === null || _d === void 0 ? void 0 : _d.fileName,
|
|
85
|
+
});
|
|
104
86
|
// calculate the relative path between the directory containing the
|
|
105
87
|
// generated construct source file to the directory containing the bundle
|
|
106
88
|
// index.js by resolving them as absolute paths first.
|
|
107
89
|
// e.g:
|
|
108
90
|
// - outfileAbs => `/project-outdir/assets/foo/bar/baz/foo-function/index.js`
|
|
109
91
|
// - constructAbs => `/project-outdir/src/foo/bar/baz/foo-function.ts`
|
|
110
|
-
const outfileAbs = path.join(outdir, outfile);
|
|
92
|
+
const outfileAbs = path.join(project.outdir, bundle.outfile);
|
|
111
93
|
const constructAbs = path.join(project.outdir, constructFile);
|
|
112
94
|
const relativeOutfile = path.relative(path.dirname(constructAbs), path.dirname(outfileAbs));
|
|
113
95
|
const src = new projen_1.SourceCode(project, constructFile);
|
|
@@ -118,10 +100,11 @@ class LambdaFunctionCodeBundle extends projen_1.Component {
|
|
|
118
100
|
src.line("import { aws_lambda } from 'aws-cdk-lib';");
|
|
119
101
|
src.line();
|
|
120
102
|
src.open(`export const ${constructName} = aws_lambda.Code.fromAsset(`);
|
|
121
|
-
src.line(`path.join(__dirname, '${(0,
|
|
103
|
+
src.line(`path.join(__dirname, '${(0, utils_1.convertToPosixPath)(relativeOutfile)}'),`);
|
|
122
104
|
src.close(");");
|
|
123
105
|
this.project.logger.verbose(`${basePath}: construct "${constructName}" generated under "${constructFile}"`);
|
|
106
|
+
this.project.logger.verbose(`${basePath}: bundle task "${bundle.bundleTask.name}"`);
|
|
124
107
|
}
|
|
125
108
|
}
|
|
126
109
|
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":"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","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"]}
|
|
110
|
+
//# 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 \"path\";\nimport { pascal } from \"case\";\nimport { Component, Project, SourceCode, javascript, typescript } from \"projen\";\nimport { Bundler, 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(\n        \"No bundler found. Please add a Bundler component to your project.\",\n      );\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(\n      path.dirname(entrypoint),\n      path.basename(entrypoint, extension),\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    const bundle = bundler.addBundle(entrypoint, {\n      ...options.bundlingOptions,\n      tsconfigPath: (project as typescript.TypeScriptProject)?.tsconfigDev\n        ?.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(\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    this.project.logger.verbose(\n      `${basePath}: bundle task \"${bundle.bundleTask.name}\"`,\n    );\n  }\n}\n"]}
|
package/lib/utils.js
ADDED
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.convertToPosixPath = exports.renderBundleName = void 0;
|
|
4
|
+
const path_1 = require("path");
|
|
5
|
+
function renderBundleName(entrypoint) {
|
|
6
|
+
const parts = (0, path_1.join)(entrypoint).split(path_1.sep);
|
|
7
|
+
if (parts[0] === "src") {
|
|
8
|
+
parts.shift(); // just remove 'src' if its the first element for ergonomics
|
|
9
|
+
}
|
|
10
|
+
const p = parts.join(path_1.sep);
|
|
11
|
+
const dir = (0, path_1.dirname)(p);
|
|
12
|
+
const base = (0, path_1.basename)(p, (0, path_1.extname)(p));
|
|
13
|
+
return (0, path_1.join)(dir, base);
|
|
14
|
+
}
|
|
15
|
+
exports.renderBundleName = renderBundleName;
|
|
16
|
+
/**
|
|
17
|
+
* Converts the given path string to posix if it wasn't already.
|
|
18
|
+
*/
|
|
19
|
+
function convertToPosixPath(p) {
|
|
20
|
+
return p.split(path_1.sep).join(path_1.posix.sep);
|
|
21
|
+
}
|
|
22
|
+
exports.convertToPosixPath = convertToPosixPath;
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsK0JBQW9FO0FBRXBFLFNBQWdCLGdCQUFnQixDQUFDLFVBQWtCO0lBQ2pELE1BQU0sS0FBSyxHQUFHLElBQUEsV0FBSSxFQUFDLFVBQVUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxVQUFHLENBQUMsQ0FBQztJQUMxQyxJQUFJLEtBQUssQ0FBQyxDQUFDLENBQUMsS0FBSyxLQUFLLEVBQUUsQ0FBQztRQUN2QixLQUFLLENBQUMsS0FBSyxFQUFFLENBQUMsQ0FBQyw0REFBNEQ7SUFDN0UsQ0FBQztJQUVELE1BQU0sQ0FBQyxHQUFHLEtBQUssQ0FBQyxJQUFJLENBQUMsVUFBRyxDQUFDLENBQUM7SUFDMUIsTUFBTSxHQUFHLEdBQUcsSUFBQSxjQUFPLEVBQUMsQ0FBQyxDQUFDLENBQUM7SUFDdkIsTUFBTSxJQUFJLEdBQUcsSUFBQSxlQUFRLEVBQUMsQ0FBQyxFQUFFLElBQUEsY0FBTyxFQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDckMsT0FBTyxJQUFBLFdBQUksRUFBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7QUFDekIsQ0FBQztBQVZELDRDQVVDO0FBRUQ7O0dBRUc7QUFDSCxTQUFnQixrQkFBa0IsQ0FBQyxDQUFTO0lBQzFDLE9BQU8sQ0FBQyxDQUFDLEtBQUssQ0FBQyxVQUFHLENBQUMsQ0FBQyxJQUFJLENBQUMsWUFBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBQ3RDLENBQUM7QUFGRCxnREFFQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGJhc2VuYW1lLCBkaXJuYW1lLCBleHRuYW1lLCBqb2luLCBwb3NpeCwgc2VwIH0gZnJvbSBcInBhdGhcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIHJlbmRlckJ1bmRsZU5hbWUoZW50cnlwb2ludDogc3RyaW5nKSB7XG4gIGNvbnN0IHBhcnRzID0gam9pbihlbnRyeXBvaW50KS5zcGxpdChzZXApO1xuICBpZiAocGFydHNbMF0gPT09IFwic3JjXCIpIHtcbiAgICBwYXJ0cy5zaGlmdCgpOyAvLyBqdXN0IHJlbW92ZSAnc3JjJyBpZiBpdHMgdGhlIGZpcnN0IGVsZW1lbnQgZm9yIGVyZ29ub21pY3NcbiAgfVxuXG4gIGNvbnN0IHAgPSBwYXJ0cy5qb2luKHNlcCk7XG4gIGNvbnN0IGRpciA9IGRpcm5hbWUocCk7XG4gIGNvbnN0IGJhc2UgPSBiYXNlbmFtZShwLCBleHRuYW1lKHApKTtcbiAgcmV0dXJuIGpvaW4oZGlyLCBiYXNlKTtcbn1cblxuLyoqXG4gKiBDb252ZXJ0cyB0aGUgZ2l2ZW4gcGF0aCBzdHJpbmcgdG8gcG9zaXggaWYgaXQgd2Fzbid0IGFscmVhZHkuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb252ZXJ0VG9Qb3NpeFBhdGgocDogc3RyaW5nKSB7XG4gIHJldHVybiBwLnNwbGl0KHNlcCkuam9pbihwb3NpeC5zZXApO1xufVxuIl19
|
package/package.json
CHANGED
|
@@ -23,7 +23,6 @@
|
|
|
23
23
|
"@types/node": "^18",
|
|
24
24
|
"@typescript-eslint/eslint-plugin": "^6",
|
|
25
25
|
"@typescript-eslint/parser": "^6",
|
|
26
|
-
"aws-cdk-lib": "2.51.0",
|
|
27
26
|
"constructs": "^10.0.0",
|
|
28
27
|
"eslint": "^8",
|
|
29
28
|
"eslint-config-prettier": "^9.0.0",
|
|
@@ -37,11 +36,7 @@
|
|
|
37
36
|
"ts-node": "^10.9.1",
|
|
38
37
|
"typescript": "^5.3.2"
|
|
39
38
|
},
|
|
40
|
-
"peerDependencies": {
|
|
41
|
-
"aws-cdk-lib": "^2.51.0"
|
|
42
|
-
},
|
|
43
39
|
"dependencies": {
|
|
44
|
-
"@mrgrain/cdk-esbuild": "^5.0.5",
|
|
45
40
|
"case": "^1.6.3",
|
|
46
41
|
"projen": "^0.77.1"
|
|
47
42
|
},
|
|
@@ -50,7 +45,7 @@
|
|
|
50
45
|
"publishConfig": {
|
|
51
46
|
"access": "public"
|
|
52
47
|
},
|
|
53
|
-
"version": "0.0.
|
|
48
|
+
"version": "0.0.15",
|
|
54
49
|
"types": "lib/index.d.ts",
|
|
55
50
|
"//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\"."
|
|
56
51
|
}
|
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
|