UNPKG

cdkdx

Version:

Zero-config CLI for aws cdk development

197 lines 19.7 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.Bundler = exports.Lambdas = void 0; const path_1 = __importDefault(require("path")); const fs_extra_1 = __importDefault(require("fs-extra")); const webpack_1 = __importStar(require("webpack")); const fork_ts_checker_webpack_plugin_1 = __importDefault(require("fork-ts-checker-webpack-plugin")); const friendly_errors_webpack_plugin_1 = __importDefault(require("friendly-errors-webpack-plugin")); const size_plugin_1 = __importDefault(require("size-plugin")); const terser_webpack_plugin_1 = __importDefault(require("terser-webpack-plugin")); const webpack_license_plugin_1 = __importDefault(require("webpack-license-plugin")); const cdkdx_config_1 = require("./cdkdx-config"); const plugins_1 = require("./plugins"); class Lambdas { constructor(srcPath) { this.entries = {}; this.warnings = new Array(); if (fs_extra_1.default.existsSync(srcPath)) { fs_extra_1.default.readdirSync(srcPath).forEach((name) => { if (name === Lambdas.SHARED_FOLDER) return; const lambdaPath = path_1.default.join(srcPath, name); if (!fs_extra_1.default.statSync(lambdaPath).isDirectory()) return; let entry = path_1.default.join(lambdaPath, 'index.ts'); if (fs_extra_1.default.existsSync(entry)) { this.entries[name] = entry; } else { entry = path_1.default.join(lambdaPath, 'index.tsx'); if (fs_extra_1.default.existsSync(entry)) { this.entries[name] = entry; } else { this.warnings.push(entry); } } }); } } hasEntries() { return Object.keys(this.entries).length > 0; } hasWarnings() { return this.warnings.length > 0; } } exports.Lambdas = Lambdas; Lambdas.SHARED_FOLDER = 'shared'; class Bundler { constructor(props) { var _a; const compilerOptions = { importsNotUsedAsValues: 'preserve', noEmit: false, declaration: false, inlineSourceMap: false, sourceMap: true, composite: false, }; const cdkdxConfig = new cdkdx_config_1.CdkdxConfig(props.projectInfo); const config = cdkdxConfig.webpack({ target: 'node', mode: 'production', devtool: 'source-map', optimization: { minimize: props.minify, minimizer: [ new terser_webpack_plugin_1.default({ cache: true, parallel: true, extractComments: true, sourceMap: true, }), ], }, entry: { ...props.lambdas.entries, }, resolve: { extensions: ['.ts', '.tsx', '.js'], }, module: { rules: [ { test: /\.tsx?$/, include: props.projectInfo.lambdasSrcPath, exclude: /node_modules/, use: { loader: 'ts-loader', options: { configFile: props.tsConfigFile, transpileOnly: true, compilerOptions, }, }, }, { test: /\.html$/i, include: props.projectInfo.lambdasSrcPath, loader: 'html-loader', options: { minimize: props.minify, }, }, ], }, plugins: [ // https://github.com/jmblog/how-to-optimize-momentjs-with-webpack new webpack_1.IgnorePlugin(/^\.\/locale$/, /moment$/), new size_plugin_1.default({ writeFile: true, publish: false, filename: 'lambda-file-sizes.json', }), new fork_ts_checker_webpack_plugin_1.default({ typescript: { enabled: true, configFile: props.tsConfigFile, configOverwrite: { compilerOptions, }, }, }), new plugins_1.NodeModulesPlugin({ nodeModules: props.projectInfo.nodeModules, }), //new LambdaFileSizePlugin({}), new friendly_errors_webpack_plugin_1.default({ clearConsole: false, }), ], output: { path: props.projectInfo.lambdasOutPath, filename: '[name]/index.js', libraryTarget: 'commonjs2', }, externals: [ ...props.projectInfo.externals, ...props.projectInfo.nodeModules, ], }); if (props.projectInfo.isConstructLib) { (_a = config.plugins) === null || _a === void 0 ? void 0 : _a.push(new webpack_license_plugin_1.default({ outputFilename: 'thirdPartyNotice.json' })); } this.compiler = (0, webpack_1.default)(config); } async run() { return new Promise((resolve, reject) => { this.compiler.run((err, stats) => { if (err) { return reject(err); } if (stats.hasErrors()) { return reject('Webpack compilation error, see above'); } resolve(); }); }); } watch() { return this.compiler.watch({ aggregateTimeout: 300, poll: undefined, }, (_err, _stats) => { return; }); } } exports.Bundler = Bundler; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bundler.js","sourceRoot":"","sources":["../src/bundler.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path';\nimport fs from 'fs-extra';\nimport webpack, { IgnorePlugin } from 'webpack';\nimport ForkTsCheckerWebpackPlugin from 'fork-ts-checker-webpack-plugin';\nimport FriendlyErrorsWebpackPlugin from 'friendly-errors-webpack-plugin';\nimport SizePlugin from 'size-plugin';\nimport TerserPlugin from 'terser-webpack-plugin';\nimport LicensePlugin from 'webpack-license-plugin';\n\nimport { ProjectInfo } from './project-info';\nimport { CdkdxConfig } from './cdkdx-config';\nimport { NodeModulesPlugin } from './plugins';\n\nexport class Lambdas {\n  public static readonly SHARED_FOLDER = 'shared';\n\n  public readonly entries: Record<string, string> = {};\n  public readonly warnings = new Array<string>();\n\n  constructor(srcPath: string) {\n    if (fs.existsSync(srcPath)) {\n      fs.readdirSync(srcPath).forEach((name) => {\n        if (name === Lambdas.SHARED_FOLDER) return;\n\n        const lambdaPath = path.join(srcPath, name);\n\n        if (!fs.statSync(lambdaPath).isDirectory()) return;\n\n        let entry = path.join(lambdaPath, 'index.ts');\n\n        if (fs.existsSync(entry)) {\n          this.entries[name] = entry;\n        } else {\n          entry = path.join(lambdaPath, 'index.tsx');\n\n          if (fs.existsSync(entry)) {\n            this.entries[name] = entry;\n          } else {\n            this.warnings.push(entry);\n          }\n        }\n      });\n    }\n  }\n\n  public hasEntries(): boolean {\n    return Object.keys(this.entries).length > 0;\n  }\n\n  public hasWarnings(): boolean {\n    return this.warnings.length > 0;\n  }\n}\n\nexport interface BundlerProps {\n  projectInfo: ProjectInfo;\n  tsConfigFile: string;\n  lambdas: Lambdas;\n  minify: boolean;\n}\n\nexport class Bundler {\n  private readonly compiler: webpack.Compiler;\n\n  constructor(props: BundlerProps) {\n    const compilerOptions = {\n      importsNotUsedAsValues: 'preserve',\n      noEmit: false,\n      declaration: false,\n      inlineSourceMap: false,\n      sourceMap: true,\n      composite: false,\n    };\n\n    const cdkdxConfig = new CdkdxConfig(props.projectInfo);\n\n    const config: webpack.Configuration = cdkdxConfig.webpack({\n      target: 'node',\n      mode: 'production',\n      devtool: 'source-map',\n      optimization: {\n        minimize: props.minify,\n        minimizer: [\n          new TerserPlugin({\n            cache: true,\n            parallel: true,\n            extractComments: true,\n            sourceMap: true,\n          }),\n        ],\n      },\n      entry: {\n        ...props.lambdas.entries,\n      },\n      resolve: {\n        extensions: ['.ts', '.tsx', '.js'],\n      },\n      module: {\n        rules: [\n          {\n            test: /\\.tsx?$/,\n            include: props.projectInfo.lambdasSrcPath,\n            exclude: /node_modules/,\n            use: {\n              loader: 'ts-loader',\n              options: {\n                configFile: props.tsConfigFile,\n                transpileOnly: true,\n                compilerOptions,\n              },\n            },\n          },\n          {\n            test: /\\.html$/i,\n            include: props.projectInfo.lambdasSrcPath,\n            loader: 'html-loader',\n            options: {\n              minimize: props.minify,\n            },\n          },\n        ],\n      },\n      plugins: [\n        // https://github.com/jmblog/how-to-optimize-momentjs-with-webpack\n        new IgnorePlugin(/^\\.\\/locale$/, /moment$/),\n        new SizePlugin({\n          writeFile: true,\n          publish: false,\n          filename: 'lambda-file-sizes.json',\n        }),\n        new ForkTsCheckerWebpackPlugin({\n          typescript: {\n            enabled: true,\n            configFile: props.tsConfigFile,\n            configOverwrite: {\n              compilerOptions,\n            },\n          },\n        }),\n        new NodeModulesPlugin({\n          nodeModules: props.projectInfo.nodeModules,\n        }),\n        //new LambdaFileSizePlugin({}),\n        new FriendlyErrorsWebpackPlugin({\n          clearConsole: false,\n        }),\n      ],\n      output: {\n        path: props.projectInfo.lambdasOutPath,\n        filename: '[name]/index.js',\n        libraryTarget: 'commonjs2',\n      },\n      externals: [\n        ...props.projectInfo.externals,\n        ...props.projectInfo.nodeModules,\n      ],\n    });\n\n    if (props.projectInfo.isConstructLib) {\n      config.plugins?.push(\n        new LicensePlugin({ outputFilename: 'thirdPartyNotice.json' }),\n      );\n    }\n\n    this.compiler = webpack(config);\n  }\n\n  public async run(): Promise<void> {\n    return new Promise((resolve, reject) => {\n      this.compiler.run((err, stats) => {\n        if (err) {\n          return reject(err);\n        }\n\n        if (stats.hasErrors()) {\n          return reject('Webpack compilation error, see above');\n        }\n\n        resolve();\n      });\n    });\n  }\n\n  public watch(): webpack.Watching {\n    return this.compiler.watch(\n      {\n        aggregateTimeout: 300,\n        poll: undefined,\n      },\n      (_err, _stats) => {\n        return;\n      },\n    );\n  }\n}\n"]}