projen
Version:
CDK for software projects
301 lines • 46.1 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.BundleLogLevel = exports.Charset = exports.SourceMapMode = exports.RunBundleTask = exports.Bundler = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const path = require("path");
const path_1 = require("path");
const util_1 = require("./util");
const component_1 = require("../component");
const dependencies_1 = require("../dependencies");
/**
* Adds support for bundling JavaScript applications and dependencies into a
* single file. In the future, this will also supports bundling websites.
*/
class Bundler extends component_1.Component {
/**
* Returns the `Bundler` instance associated with a project or `undefined` if
* there is no Bundler.
* @param project The project
* @returns A bundler
*/
static of(project) {
const isBundler = (o) => o instanceof Bundler;
return project.components.find(isBundler);
}
/**
* Creates a `Bundler`.
*/
constructor(project, options = {}) {
super(project);
this.esbuildVersion = options.esbuildVersion;
this.bundledir = options.assetsDir ?? "assets";
this.loaders = options.loaders;
this.runBundleTask =
options.runBundleTask ??
(options.addToPreCompile === false
? RunBundleTask.MANUAL
: RunBundleTask.PRE_COMPILE);
}
/**
* Gets or creates the singleton "bundle" task of the project.
*
* If the project doesn't have a "bundle" task, it will be created and spawned
* during the pre-compile phase.
*/
get bundleTask() {
if (!this._task) {
this.addBundlingSupport();
this._task = this.project.tasks.addTask("bundle", {
description: "Prepare assets",
});
// install the bundle task into the pre-compile phase.
if (this.runBundleTask === RunBundleTask.PRE_COMPILE) {
this.project.preCompileTask.spawn(this._task);
}
else if (this.runBundleTask === RunBundleTask.POST_COMPILE) {
this.project.postCompileTask.spawn(this._task);
}
}
return this._task;
}
/**
* Adds a task to the project which bundles a specific entrypoint and all of
* its dependencies into a single javascript output file.
*
* @param entrypoint The relative path of the artifact within the project
* @param options Bundling options
*/
addBundle(entrypoint, options) {
const name = (0, util_1.renderBundleName)(entrypoint);
const outdir = path.posix.join(this.bundledir, name);
const outfile = path.posix.join(outdir, options.outfile ?? "index.js");
const args = [
"esbuild",
"--bundle",
entrypoint,
`--target="${options.target}"`,
`--platform="${options.platform}"`,
`--outfile="${outfile}"`,
];
if (options.tsconfigPath) {
args.push(`--tsconfig="${options.tsconfigPath}"`);
}
for (const x of options.externals ?? []) {
args.push(`--external:${x}`);
}
if (options.sourcemap || options.sourceMapMode) {
const sourceMapMode = options.sourceMapMode ?? SourceMapMode.DEFAULT;
const sourceMapValue = sourceMapMode === SourceMapMode.DEFAULT
? ""
: `=${options.sourceMapMode}`;
args.push(`--sourcemap${sourceMapValue}`);
if (options.sourcesContent === true) {
args.push(`--sources-content=${options.sourcesContent}`);
}
}
const format = options.format;
if (format) {
args.push(`--format=${format}`);
}
const loaders = options.loaders ?? false ? options.loaders : this.loaders ?? false;
if (loaders) {
for (let [extension, loader] of Object.entries(loaders)) {
args.push(`--loader:.${extension}=${loader}`);
}
}
const defines = Object.entries(options.define ?? {});
for (const [key, value] of defines) {
args.push(`--define:${key}=${JSON.stringify(value)}`);
}
if (options.minify) {
args.push("--minify");
}
if (options.logLevel) {
args.push(`--log-level=${options.logLevel}`);
}
if (options.keepNames) {
args.push("--keep-names");
}
if (options.metafile) {
args.push(`--metafile=${(0, path_1.join)(outdir, "index.meta.json")}`);
}
if (options.banner) {
args.push(`--banner:js=${JSON.stringify(options.banner)}`);
}
if (options.footer) {
args.push(`--footer:js=${JSON.stringify(options.footer)}`);
}
if (options.mainFields) {
args.push(`--main-fields=${options.mainFields.join(",")}`);
}
if (options.inject) {
args.push(...options.inject.map((i) => `--inject:${i}`));
}
if (options.esbuildArgs) {
const subArgs = new Array();
for (const [key, value] of Object.entries(options.esbuildArgs)) {
if (value === true || value === "") {
subArgs.push(key);
}
else if (value) {
subArgs.push(`${key}="${value}"`);
}
}
args.push(subArgs.join(" "));
}
const bundleTask = this.project.addTask(`bundle:${name}`, {
description: `Create a JavaScript bundle from ${entrypoint}`,
exec: args.join(" "),
});
this.bundleTask.spawn(bundleTask);
if (options.executable ?? false) {
bundleTask.exec(`chmod +x ${outfile}`);
}
let watchTask;
const watch = options.watchTask ?? true;
if (watch) {
watchTask = this.project.addTask(`bundle:${name}:watch`, {
description: `Continuously update the JavaScript bundle from ${entrypoint}`,
exec: `${args.join(" ")} --watch`,
});
}
return {
bundleTask: bundleTask,
watchTask: watchTask,
outdir: outdir,
outfile: outfile,
};
}
/**
* Add bundling support to a project. This is called implicitly when
* `bundleTask` is referenced first. It adds the dependency on `esbuild`,
* gitignore/npmignore, etc.
*/
addBundlingSupport() {
const ignoreEntry = `/${this.bundledir}/`;
this.project.addGitIgnore(ignoreEntry);
this.project.addPackageIgnore(`!${ignoreEntry}`); // include in tarball
const dep = this.esbuildVersion
? `esbuild@${this.esbuildVersion}`
: "esbuild";
this.project.deps.addDependency(dep, dependencies_1.DependencyType.BUILD);
}
}
exports.Bundler = Bundler;
_a = JSII_RTTI_SYMBOL_1;
Bundler[_a] = { fqn: "projen.javascript.Bundler", version: "0.95.2" };
/**
* Options for BundlerOptions.runBundleTask
*/
var RunBundleTask;
(function (RunBundleTask) {
/**
* Don't bundle automatically as part of the build.
*/
RunBundleTask["MANUAL"] = "manual";
/**
* Bundle automatically before compilation.
*/
RunBundleTask["PRE_COMPILE"] = "pre_compile";
/**
* Bundle automatically after compilation. This is useful if you want to
* bundle the compiled results.
*
* Thus will run compilation tasks (using tsc, etc.) before running file
* through bundling step.
*
* This is only required unless you are using new experimental features that
* are not supported by `esbuild` but are supported by typescript's `tsc`
* compiler. One example of such feature is `emitDecoratorMetadata`.
*
* ```typescript
* // In a TypeScript project with output configured
* // to go to the "lib" directory:
* const project = new TypeScriptProject({
* name: "test",
* defaultReleaseBranch: "main",
* tsconfig: {
* compilerOptions: {
* outDir: "lib",
* },
* },
* bundlerOptions: {
* // ensure we compile with `tsc` before bundling
* runBundleTask: RunBundleTask.POST_COMPILE,
* },
* });
*
* // Tell the bundler to bundle the compiled results (from the "lib" directory)
* project.bundler.addBundle("./lib/index.js", {
* platform: "node",
* target: "node18",
* sourcemap: false,
* format: "esm",
* });
* ```
**/
RunBundleTask["POST_COMPILE"] = "post_compile";
})(RunBundleTask || (exports.RunBundleTask = RunBundleTask = {}));
/**
* SourceMap mode for esbuild
* @see https://esbuild.github.io/api/#sourcemap
*/
var SourceMapMode;
(function (SourceMapMode) {
/**
* Default sourceMap mode - will generate a .js.map file alongside any generated .js file and add a special //# sourceMappingURL=
* comment to the bottom of the .js file pointing to the .js.map file
*/
SourceMapMode["DEFAULT"] = "default";
/**
* External sourceMap mode - If you want to omit the special //# sourceMappingURL= comment from the generated .js file but you still
* want to generate the .js.map files
*/
SourceMapMode["EXTERNAL"] = "external";
/**
* Inline sourceMap mode - If you want to insert the entire source map into the .js file instead of generating a separate .js.map file
*/
SourceMapMode["INLINE"] = "inline";
/**
* Both sourceMap mode - If you want to have the effect of both inline and external simultaneously
*/
SourceMapMode["BOTH"] = "both";
})(SourceMapMode || (exports.SourceMapMode = SourceMapMode = {}));
/**
* Charset for esbuild's output
*/
var Charset;
(function (Charset) {
/**
* ASCII
*
* Any non-ASCII characters are escaped using backslash escape sequences
*/
Charset["ASCII"] = "ascii";
/**
* UTF-8
*
* Keep original characters without using escape sequences
*/
Charset["UTF8"] = "utf8";
})(Charset || (exports.Charset = Charset = {}));
/**
* Log levels for esbuild and package managers' install commands.
*/
var BundleLogLevel;
(function (BundleLogLevel) {
/** Show everything */
BundleLogLevel["VERBOSE"] = "verbose";
/** Show everything from info and some additional messages for debugging */
BundleLogLevel["DEBUG"] = "debug";
/** Show warnings, errors, and an output file summary */
BundleLogLevel["INFO"] = "info";
/** Show warnings and errors */
BundleLogLevel["WARNING"] = "warning";
/** Show errors only */
BundleLogLevel["ERROR"] = "error";
/** Show nothing */
BundleLogLevel["SILENT"] = "silent";
})(BundleLogLevel || (exports.BundleLogLevel = BundleLogLevel = {}));
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bundler.js","sourceRoot":"","sources":["../../src/javascript/bundler.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"path\";\nimport { join as pathJoin } from \"path\";\nimport { renderBundleName } from \"./util\";\nimport { Component } from \"../component\";\nimport { DependencyType } from \"../dependencies\";\nimport { Project } from \"../project\";\nimport { Task } from \"../task\";\n\n// Parts of this file inspired by @aws-cdk-lib/aws-lambda-nodejs\n//   https://github.com/aws/aws-cdk/blob/c3c771c6f6f6790f2298a85a549bded640d2e35b/packages/aws-cdk-lib/aws-lambda-nodejs/lib/bundling.ts#L195\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   * @deprecated Use `runBundleTask` instead.\n   */\n  readonly addToPreCompile?: boolean;\n\n  /**\n   * Choose which phase (if any) to add the `bundle` command to.\n   *\n   * Note: If using `addBundle()` with the `bundleCompiledResults`, this option\n   * must be set to `RunBundleTask.POST_COMPILE` or `RunBundleTask.MANUAL`.\n   *\n   * @see AddBundleOptions.bundleCompiledResults\n   *\n   * @default RunBundleTask.PRE_COMPILE\n   */\n  readonly runBundleTask?: RunBundleTask;\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 runBundleTask?: RunBundleTask;\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.loaders = options.loaders;\n\n    this.runBundleTask =\n      options.runBundleTask ??\n      (options.addToPreCompile === false\n        ? RunBundleTask.MANUAL\n        : RunBundleTask.PRE_COMPILE);\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.runBundleTask === RunBundleTask.PRE_COMPILE) {\n        this.project.preCompileTask.spawn(this._task);\n      } else if (this.runBundleTask === RunBundleTask.POST_COMPILE) {\n        this.project.postCompileTask.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: AddBundleOptions): 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.js\");\n    const args = [\n      \"esbuild\",\n      \"--bundle\",\n      entrypoint,\n      `--target=\"${options.target}\"`,\n      `--platform=\"${options.platform}\"`,\n      `--outfile=\"${outfile}\"`,\n    ];\n\n    if (options.tsconfigPath) {\n      args.push(`--tsconfig=\"${options.tsconfigPath}\"`);\n    }\n\n    for (const x of options.externals ?? []) {\n      args.push(`--external:${x}`);\n    }\n\n    if (options.sourcemap || options.sourceMapMode) {\n      const sourceMapMode = options.sourceMapMode ?? SourceMapMode.DEFAULT;\n      const sourceMapValue =\n        sourceMapMode === SourceMapMode.DEFAULT\n          ? \"\"\n          : `=${options.sourceMapMode}`;\n      args.push(`--sourcemap${sourceMapValue}`);\n\n      if (options.sourcesContent === true) {\n        args.push(`--sources-content=${options.sourcesContent}`);\n      }\n    }\n\n    const format = options.format;\n    if (format) {\n      args.push(`--format=${format}`);\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    const defines = Object.entries(options.define ?? {});\n    for (const [key, value] of defines) {\n      args.push(`--define:${key}=${JSON.stringify(value)}`);\n    }\n\n    if (options.minify) {\n      args.push(\"--minify\");\n    }\n\n    if (options.logLevel) {\n      args.push(`--log-level=${options.logLevel}`);\n    }\n    if (options.keepNames) {\n      args.push(\"--keep-names\");\n    }\n    if (options.metafile) {\n      args.push(`--metafile=${pathJoin(outdir, \"index.meta.json\")}`);\n    }\n    if (options.banner) {\n      args.push(`--banner:js=${JSON.stringify(options.banner)}`);\n    }\n    if (options.footer) {\n      args.push(`--footer:js=${JSON.stringify(options.footer)}`);\n    }\n    if (options.mainFields) {\n      args.push(`--main-fields=${options.mainFields.join(\",\")}`);\n    }\n    if (options.inject) {\n      args.push(...options.inject.map((i) => `--inject:${i}`));\n    }\n    if (options.esbuildArgs) {\n      const subArgs = new Array<string>();\n\n      for (const [key, value] of Object.entries(options.esbuildArgs)) {\n        if (value === true || value === \"\") {\n          subArgs.push(key);\n        } else if (value) {\n          subArgs.push(`${key}=\"${value}\"`);\n        }\n      }\n\n      args.push(subArgs.join(\" \"));\n    }\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    if (options.executable ?? false) {\n      bundleTask.exec(`chmod +x ${outfile}`);\n    }\n\n    let watchTask;\n    const watch = options.watchTask ?? true;\n    if (watch) {\n      watchTask = this.project.addTask(`bundle:${name}:watch`, {\n        description: `Continuously update the JavaScript bundle from ${entrypoint}`,\n        exec: `${args.join(\" \")} --watch`,\n      });\n    }\n\n    return {\n      bundleTask: bundleTask,\n      watchTask: watchTask,\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   * The \"watch\" task for this bundle.\n   */\n  readonly watchTask?: 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 false\n   */\n  readonly sourcemap?: boolean;\n\n  /**\n   * In addition to the `bundle:xyz` task, creates `bundle:xyz:watch` task which will\n   * invoke the same esbuild command with the `--watch` flag. This can be used\n   * to continusouly watch for changes.\n   *\n   * @default true\n   */\n  readonly watchTask?: boolean;\n}\n\n/**\n * Options for `addBundle()`.\n */\nexport interface AddBundleOptions extends BundlingOptions {\n  /**\n   * esbuild target.\n   *\n   * @example \"node12\"\n   */\n  readonly target: string;\n\n  /**\n   * esbuild platform.\n   *\n   * @example \"node\"\n   */\n  readonly platform: string;\n\n  /**\n   * Bundler output path relative to the asset's output directory.\n   * @default \"index.js\"\n   */\n  readonly outfile?: string;\n\n  /**\n   * Mark the output file as executable.\n   * @default false\n   */\n  readonly executable?: boolean;\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\n  readonly format?: string;\n\n  /**\n   * Whether to minify files when bundling.\n   *\n   * @default false\n   */\n  readonly minify?: boolean;\n\n  /**\n   * Source map mode to be used when bundling.\n   * @see https://esbuild.github.io/api/#sourcemap\n   *\n   * @default SourceMapMode.DEFAULT\n   */\n  readonly sourceMapMode?: SourceMapMode;\n\n  /**\n   * Whether to include original source code in source maps when bundling.\n   *\n   * @see https://esbuild.github.io/api/#sources-content\n   *\n   * @default true\n   */\n  readonly sourcesContent?: boolean;\n\n  /**\n   * Log level for esbuild. This is also propagated to the package manager and\n   * applies to its specific install command.\n   *\n   * @default LogLevel.WARNING\n   */\n  readonly logLevel?: BundleLogLevel;\n\n  /**\n   * Whether to preserve the original `name` values even in minified code.\n   *\n   * In JavaScript the `name` property on functions and classes defaults to a\n   * nearby identifier in the source code.\n   *\n   * However, minification renames symbols to reduce code size and bundling\n   * sometimes need to rename symbols to avoid collisions. That changes value of\n   * the `name` property for many of these cases. This is usually fine because\n   * the `name` property is normally only used for debugging. However, some\n   * frameworks rely on the `name` property for registration and binding purposes.\n   * If this is the case, you can enable this option to preserve the original\n   * `name` values even in minified code.\n   *\n   * @default false\n   */\n  readonly keepNames?: boolean;\n\n  /**\n   * This option tells esbuild to write out a JSON file relative to output directory with metadata about the build.\n   *\n   * The metadata in this JSON file follows this schema (specified using TypeScript syntax):\n   *\n   * ```text\n   * {\n   *   outputs: {\n   *     [path: string]: {\n   *       bytes: number\n   *       inputs: {\n   *         [path: string]: { bytesInOutput: number }\n   *       }\n   *       imports: { path: string }[]\n   *       exports: string[]\n   *     }\n   *   }\n   * }\n   * ```\n   * This data can then be analyzed by other tools. For example,\n   * bundle buddy can consume esbuild's metadata format and generates a treemap visualization\n   * of the modules in your bundle and how much space each one takes up.\n   * @see https://esbuild.github.io/api/#metafile\n   * @default false\n   */\n  readonly metafile?: boolean;\n\n  /**\n   * Use this to insert an arbitrary string at the beginning of generated JavaScript files.\n   *\n   * This is similar to footer which inserts at the end instead of the beginning.\n   *\n   * This is commonly used to insert comments:\n   *\n   * @default - no comments are passed\n   */\n  readonly banner?: string;\n\n  /**\n   * Use this to insert an arbitrary string at the end of generated JavaScript files.\n   *\n   * This is similar to banner which inserts at the beginning instead of the end.\n   *\n   * This is commonly used to insert comments\n   *\n   * @default - no comments are passed\n   */\n  readonly footer?: string;\n\n  /**\n   * The charset to use for esbuild's output.\n   *\n   * By default esbuild's output is ASCII-only. Any non-ASCII characters are escaped\n   * using backslash escape sequences. Using escape sequences makes the generated output\n   * slightly bigger, and also makes it harder to read. If you would like for esbuild to print\n   * the original characters without using escape sequences, use `Charset.UTF8`.\n   *\n   * @see https://esbuild.github.io/api/#charset\n   * @default Charset.ASCII\n   */\n  readonly charset?: Charset;\n\n  /**\n   * Replace global identifiers with constant expressions.\n   *\n   * For example, `{ 'process.env.DEBUG': 'true' }`.\n   *\n   * Another example, `{ 'process.env.API_KEY': JSON.stringify('xxx-xxxx-xxx') }`.\n   *\n   * @default - no replacements are made\n   */\n  readonly define?: { [key: string]: string };\n\n  /**\n   * Build arguments to pass into esbuild.\n   *\n   * For example, to add the [--log-limit](https://esbuild.github.io/api/#log-limit) flag:\n   *\n   * ```text\n   * project.bundler.addBundle(\"./src/hello.ts\", {\n   *   platform: \"node\",\n   *   target: \"node18\",\n   *   sourcemap: true,\n   *   format: \"esm\",\n   *   esbuildArgs: {\n   *     \"--log-limit\": \"0\",\n   *   },\n   * });\n   * ```\n   *\n   * @default - no additional esbuild arguments are passed\n   */\n  readonly esbuildArgs?: { [key: string]: string | boolean };\n\n  /**\n   * How to determine the entry point for modules.\n   * Try ['module', 'main'] to default to ES module versions.\n   *\n   * @default []\n   */\n  readonly mainFields?: string[];\n\n  /**\n   * This option allows you to automatically replace a global variable with an\n   * import from another file.\n   *\n   * @see https://esbuild.github.io/api/#inject\n   * @default - no code is injected\n   */\n  readonly inject?: string[];\n}\n\n/**\n * Options for BundlerOptions.runBundleTask\n */\nexport enum RunBundleTask {\n  /**\n   * Don't bundle automatically as part of the build.\n   */\n  MANUAL = \"manual\",\n  /**\n   * Bundle automatically before compilation.\n   */\n  PRE_COMPILE = \"pre_compile\",\n  /**\n   * Bundle automatically after compilation. This is useful if you want to\n   * bundle the compiled results.\n   *\n   * Thus will run compilation tasks (using tsc, etc.) before running file\n   * through bundling step.\n   *\n   * This is only required unless you are using new experimental features that\n   * are not supported by `esbuild` but are supported by typescript's `tsc`\n   * compiler. One example of such feature is `emitDecoratorMetadata`.\n   *\n   * ```typescript\n   * // In a TypeScript project with output configured\n   * // to go to the \"lib\" directory:\n   * const project = new TypeScriptProject({\n   *   name: \"test\",\n   *   defaultReleaseBranch: \"main\",\n   *   tsconfig: {\n   *     compilerOptions: {\n   *       outDir: \"lib\",\n   *     },\n   *   },\n   *   bundlerOptions: {\n   *     // ensure we compile with `tsc` before bundling\n   *     runBundleTask: RunBundleTask.POST_COMPILE,\n   *   },\n   * });\n   *\n   * // Tell the bundler to bundle the compiled results (from the \"lib\" directory)\n   * project.bundler.addBundle(\"./lib/index.js\", {\n   *   platform: \"node\",\n   *   target: \"node18\",\n   *   sourcemap: false,\n   *   format: \"esm\",\n   * });\n   * ```\n   **/\n  POST_COMPILE = \"post_compile\",\n}\n\n/**\n * SourceMap mode for esbuild\n * @see https://esbuild.github.io/api/#sourcemap\n */\nexport enum SourceMapMode {\n  /**\n   * Default sourceMap mode - will generate a .js.map file alongside any generated .js file and add a special //# sourceMappingURL=\n   * comment to the bottom of the .js file pointing to the .js.map file\n   */\n  DEFAULT = \"default\",\n  /**\n   *  External sourceMap mode - If you want to omit the special //# sourceMappingURL= comment from the generated .js file but you still\n   *  want to generate the .js.map files\n   */\n  EXTERNAL = \"external\",\n  /**\n   * Inline sourceMap mode - If you want to insert the entire source map into the .js file instead of generating a separate .js.map file\n   */\n  INLINE = \"inline\",\n  /**\n   * Both sourceMap mode - If you want to have the effect of both inline and external simultaneously\n   */\n  BOTH = \"both\",\n}\n\n/**\n * Charset for esbuild's output\n */\nexport enum Charset {\n  /**\n   * ASCII\n   *\n   * Any non-ASCII characters are escaped using backslash escape sequences\n   */\n  ASCII = \"ascii\",\n\n  /**\n   * UTF-8\n   *\n   * Keep original characters without using escape sequences\n   */\n  UTF8 = \"utf8\",\n}\n\n/**\n * Log levels for esbuild and package managers' install commands.\n */\nexport enum BundleLogLevel {\n  /** Show everything */\n  VERBOSE = \"verbose\",\n  /** Show everything from info and some additional messages for debugging */\n  DEBUG = \"debug\",\n  /** Show warnings, errors, and an output file summary */\n  INFO = \"info\",\n  /** Show warnings and errors */\n  WARNING = \"warning\",\n  /** Show errors only */\n  ERROR = \"error\",\n  /** Show nothing */\n  SILENT = \"silent\",\n}\n"]}