@farris/build-angular
Version:
Angular Webpack Build Facade
116 lines • 22.2 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
const build_webpack_1 = require("@angular-devkit/build-webpack");
const core_1 = require("@angular-devkit/core");
const rxjs_1 = require("rxjs");
const operators_1 = require("rxjs/operators");
const webpack_configs_1 = require("../angular-cli-files/models/webpack-configs");
const read_tsconfig_1 = require("../angular-cli-files/utilities/read-tsconfig");
const require_project_module_1 = require("../angular-cli-files/utilities/require-project-module");
const service_worker_1 = require("../angular-cli-files/utilities/service-worker");
const stats_1 = require("../angular-cli-files/utilities/stats");
const utils_1 = require("../utils");
const webpackMerge = require('webpack-merge');
class BrowserBuilder {
constructor(context) {
this.context = context;
}
run(builderConfig) {
const options = builderConfig.options;
const root = this.context.workspace.root;
const projectRoot = core_1.resolve(root, builderConfig.root);
const host = new core_1.virtualFs.AliasHost(this.context.host);
const webpackBuilder = new build_webpack_1.WebpackBuilder(Object.assign({}, this.context, { host }));
return rxjs_1.of(null).pipe(operators_1.concatMap(() => options.deleteOutputPath
? this._deleteOutputDir(root, core_1.normalize(options.outputPath), this.context.host)
: rxjs_1.of(null)), operators_1.concatMap(() => utils_1.normalizeFileReplacements(options.fileReplacements, host, root)), operators_1.tap(fileReplacements => options.fileReplacements = fileReplacements), operators_1.concatMap(() => utils_1.normalizeAssetPatterns(options.assets, host, root, projectRoot, builderConfig.sourceRoot)),
// Replace the assets in options with the normalized version.
operators_1.tap((assetPatternObjects => options.assets = assetPatternObjects)), operators_1.concatMap(() => {
// Ensure Build Optimizer is only used with AOT.
if (options.buildOptimizer && !options.aot) {
throw new Error('The `--build-optimizer` option cannot be used without `--aot`.');
}
let webpackConfig;
try {
webpackConfig = this.buildWebpackConfig(root, projectRoot, host, options);
}
catch (e) {
return rxjs_1.throwError(e);
}
return webpackBuilder.runWebpack(webpackConfig, exports.getBrowserLoggingCb(options.verbose));
}), operators_1.concatMap(buildEvent => {
if (buildEvent.success && !options.watch && options.serviceWorker) {
return new rxjs_1.Observable(obs => {
service_worker_1.augmentAppWithServiceWorker(this.context.host, root, projectRoot, core_1.resolve(root, core_1.normalize(options.outputPath)), options.baseHref || '/', options.ngswConfigPath).then(() => {
obs.next({ success: true });
obs.complete();
}, (err) => {
obs.error(err);
});
});
}
else {
return rxjs_1.of(buildEvent);
}
}));
}
buildWebpackConfig(root, projectRoot, host, options) {
let wco;
const tsConfigPath = core_1.getSystemPath(core_1.normalize(core_1.resolve(root, core_1.normalize(options.tsConfig))));
const tsConfig = read_tsconfig_1.readTsconfig(tsConfigPath);
const projectTs = require_project_module_1.requireProjectModule(core_1.getSystemPath(projectRoot), 'typescript');
const supportES2015 = tsConfig.options.target !== projectTs.ScriptTarget.ES3
&& tsConfig.options.target !== projectTs.ScriptTarget.ES5;
wco = {
root: core_1.getSystemPath(root),
projectRoot: core_1.getSystemPath(projectRoot),
buildOptions: options,
tsConfig,
tsConfigPath,
supportES2015,
};
const webpackConfigs = [
webpack_configs_1.getCommonConfig(wco),
webpack_configs_1.getBrowserConfig(wco),
webpack_configs_1.getStylesConfig(wco),
webpack_configs_1.getStatsConfig(wco),
];
if (wco.buildOptions.main || wco.buildOptions.polyfills) {
const typescriptConfigPartial = wco.buildOptions.aot
? webpack_configs_1.getAotConfig(wco, host)
: webpack_configs_1.getNonAotConfig(wco, host);
webpackConfigs.push(typescriptConfigPartial);
}
return webpackMerge(webpackConfigs);
}
_deleteOutputDir(root, outputPath, host) {
const resolvedOutputPath = core_1.resolve(root, outputPath);
if (resolvedOutputPath === root) {
throw new Error('Output path MUST not be project root directory!');
}
return host.exists(resolvedOutputPath).pipe(operators_1.concatMap(exists => exists
// TODO: remove this concat once host ops emit an event.
? rxjs_1.concat(host.delete(resolvedOutputPath), rxjs_1.of(null)).pipe(operators_1.last())
// ? of(null)
: rxjs_1.of(null)));
}
}
exports.BrowserBuilder = BrowserBuilder;
exports.getBrowserLoggingCb = (verbose) => (stats, config, logger) => {
// config.stats contains our own stats settings, added during buildWebpackConfig().
const json = stats.toJson(config.stats);
if (verbose) {
logger.info(stats.toString(config.stats));
}
else {
logger.info(stats_1.statsToString(json, config.stats));
}
if (stats.hasWarnings()) {
logger.warn(stats_1.statsWarningsToString(json, config.stats));
}
if (stats.hasErrors()) {
logger.error(stats_1.statsErrorsToString(json, config.stats));
}
};
exports.default = BrowserBuilder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"./","sources":["packages/farris_devkit/build_angular/src/browser/index.ts"],"names":[],"mappings":";;AAaA,iEAAgF;AAChF,+CAA0F;AAE1F,+BAA0D;AAC1D,8CAAsD;AAGtD,iFAOqD;AACrD,gFAA4E;AAC5E,kGAA6F;AAC7F,kFAA4F;AAC5F,gEAI8C;AAC9C,oCAA6E;AAE7E,MAAM,YAAY,GAAG,OAAO,CAAC,eAAe,CAAC,CAAC;AAa9C;IAEE,YAAmB,OAAuB;QAAvB,YAAO,GAAP,OAAO,CAAgB;IAAI,CAAC;IAE/C,GAAG,CAAC,aAAyD;QAC3D,MAAM,OAAO,GAAG,aAAa,CAAC,OAAO,CAAC;QACtC,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC;QACzC,MAAM,WAAW,GAAG,cAAO,CAAC,IAAI,EAAE,aAAa,CAAC,IAAI,CAAC,CAAC;QACtD,MAAM,IAAI,GAAG,IAAI,gBAAS,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,IAAgC,CAAC,CAAC;QACpF,MAAM,cAAc,GAAG,IAAI,8BAAc,mBAAM,IAAI,CAAC,OAAO,IAAE,IAAI,IAAG,CAAC;QAErE,MAAM,CAAC,SAAE,CAAC,IAAI,CAAC,CAAC,IAAI,CAClB,qBAAS,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,gBAAgB;YACtC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE,gBAAS,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC;YAC/E,CAAC,CAAC,SAAE,CAAC,IAAI,CAAC,CAAC,EACb,qBAAS,CAAC,GAAG,EAAE,CAAC,iCAAyB,CAAC,OAAO,CAAC,gBAAgB,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC,EAChF,eAAG,CAAC,gBAAgB,CAAC,EAAE,CAAC,OAAO,CAAC,gBAAgB,GAAG,gBAAgB,CAAC,EACpE,qBAAS,CAAC,GAAG,EAAE,CAAC,8BAAsB,CACpC,OAAO,CAAC,MAAM,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,aAAa,CAAC,UAAU,CAAC,CAAC;QACrE,6DAA6D;QAC7D,eAAG,CAAC,CAAC,mBAAmB,CAAC,EAAE,CAAC,OAAO,CAAC,MAAM,GAAG,mBAAmB,CAAC,CAAC,EAClE,qBAAS,CAAC,GAAG,EAAE;YACb,gDAAgD;YAChD,EAAE,CAAC,CAAC,OAAO,CAAC,cAAc,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC;gBAC3C,MAAM,IAAI,KAAK,CAAC,gEAAgE,CAAC,CAAC;YACpF,CAAC;YAED,IAAI,aAAa,CAAC;YAClB,IAAI,CAAC;gBACH,aAAa,GAAG,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,WAAW,EAAE,IAAI,EAC7D,OAAyC,CAAC,CAAC;YAC/C,CAAC;YAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBACX,MAAM,CAAC,iBAAU,CAAC,CAAC,CAAC,CAAC;YACvB,CAAC;YAED,MAAM,CAAC,cAAc,CAAC,UAAU,CAAC,aAAa,EAAE,2BAAmB,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC;QACxF,CAAC,CAAC,EACF,qBAAS,CAAC,UAAU,CAAC,EAAE;YACrB,EAAE,CAAC,CAAC,UAAU,CAAC,OAAO,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC;gBAClE,MAAM,CAAC,IAAI,iBAAU,CAAC,GAAG,CAAC,EAAE;oBAC1B,4CAA2B,CACzB,IAAI,CAAC,OAAO,CAAC,IAAI,EACjB,IAAI,EACJ,WAAW,EACX,cAAO,CAAC,IAAI,EAAE,gBAAS,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,EAC5C,OAAO,CAAC,QAAQ,IAAI,GAAG,EACvB,OAAO,CAAC,cAAc,CACvB,CAAC,IAAI,CACJ,GAAG,EAAE;wBACH,GAAG,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;wBAC5B,GAAG,CAAC,QAAQ,EAAE,CAAC;oBACjB,CAAC,EACD,CAAC,GAAU,EAAE,EAAE;wBACb,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;oBACjB,CAAC,CACF,CAAC;gBACJ,CAAC,CAAC,CAAC;YACL,CAAC;YAAC,IAAI,CAAC,CAAC;gBACN,MAAM,CAAC,SAAE,CAAC,UAAU,CAAC,CAAC;YACxB,CAAC;QACH,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,kBAAkB,CAChB,IAAU,EACV,WAAiB,EACjB,IAA8B,EAC9B,OAAuC;QAEvC,IAAI,GAAyD,CAAC;QAE9D,MAAM,YAAY,GAAG,oBAAa,CAAC,gBAAS,CAAC,cAAO,CAAC,IAAI,EAAE,gBAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1F,MAAM,QAAQ,GAAG,4BAAY,CAAC,YAAY,CAAC,CAAC;QAE5C,MAAM,SAAS,GAAG,6CAAoB,CAAC,oBAAa,CAAC,WAAW,CAAC,EAAE,YAAY,CAAc,CAAC;QAE9F,MAAM,aAAa,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,KAAK,SAAS,CAAC,YAAY,CAAC,GAAG;eACvE,QAAQ,CAAC,OAAO,CAAC,MAAM,KAAK,SAAS,CAAC,YAAY,CAAC,GAAG,CAAC;QAE5D,GAAG,GAAG;YACJ,IAAI,EAAE,oBAAa,CAAC,IAAI,CAAC;YACzB,WAAW,EAAE,oBAAa,CAAC,WAAW,CAAC;YACvC,YAAY,EAAE,OAAO;YACrB,QAAQ;YACR,YAAY;YACZ,aAAa;SACd,CAAC;QAEF,MAAM,cAAc,GAAS;YAC3B,iCAAe,CAAC,GAAG,CAAC;YACpB,kCAAgB,CAAC,GAAG,CAAC;YACrB,iCAAe,CAAC,GAAG,CAAC;YACpB,gCAAc,CAAC,GAAG,CAAC;SACpB,CAAC;QAEF,EAAE,CAAC,CAAC,GAAG,CAAC,YAAY,CAAC,IAAI,IAAI,GAAG,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC,CAAC;YACxD,MAAM,uBAAuB,GAAG,GAAG,CAAC,YAAY,CAAC,GAAG;gBAClD,CAAC,CAAC,8BAAY,CAAC,GAAG,EAAE,IAAI,CAAC;gBACzB,CAAC,CAAC,iCAAe,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;YAC/B,cAAc,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC;QAC/C,CAAC;QAED,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC;IACtC,CAAC;IAEO,gBAAgB,CAAC,IAAU,EAAE,UAAgB,EAAE,IAAoB;QACzE,MAAM,kBAAkB,GAAG,cAAO,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;QACrD,EAAE,CAAC,CAAC,kBAAkB,KAAK,IAAI,CAAC,CAAC,CAAC;YAChC,MAAM,IAAI,KAAK,CAAC,iDAAiD,CAAC,CAAC;QACrE,CAAC;QAED,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,IAAI,CACzC,qBAAS,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM;YACxB,wDAAwD;YACxD,CAAC,CAAC,aAAM,CAAC,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,EAAE,SAAE,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAI,EAAE,CAAC;YAChE,aAAa;YACb,CAAC,CAAC,SAAE,CAAC,IAAI,CAAC,CAAC,CACd,CAAC;IACJ,CAAC;CACF;AAxHD,wCAwHC;AAEY,QAAA,mBAAmB,GAAG,CAAC,OAAgB,EAAmB,EAAE,CACvE,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE;IACxB,mFAAmF;IACnF,MAAM,IAAI,GAAG,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACxC,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;QACZ,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;IAC5C,CAAC;IAAC,IAAI,CAAC,CAAC;QACN,MAAM,CAAC,IAAI,CAAC,qBAAa,CAAC,IAAI,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;IACjD,CAAC;IAED,EAAE,CAAC,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;QACxB,MAAM,CAAC,IAAI,CAAC,6BAAqB,CAAC,IAAI,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;IACzD,CAAC;IACD,EAAE,CAAC,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;QACtB,MAAM,CAAC,KAAK,CAAC,2BAAmB,CAAC,IAAI,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;IACxD,CAAC;AACH,CAAC,CAAC;AAEJ,kBAAe,cAAc,CAAC","sourcesContent":["/**\n * @license\n * Copyright Google Inc. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nimport {\n  BuildEvent,\n  Builder,\n  BuilderConfiguration,\n  BuilderContext,\n} from '@angular-devkit/architect';\nimport { LoggingCallback, WebpackBuilder } from '@angular-devkit/build-webpack';\nimport { Path, getSystemPath, normalize, resolve, virtualFs } from '@angular-devkit/core';\nimport * as fs from 'fs';\nimport { Observable, concat, of, throwError } from 'rxjs';\nimport { concatMap, last, tap } from 'rxjs/operators';\nimport * as ts from 'typescript'; // tslint:disable-line:no-implicit-dependencies\nimport { WebpackConfigOptions } from '../angular-cli-files/models/build-options';\nimport {\n  getAotConfig,\n  getBrowserConfig,\n  getCommonConfig,\n  getNonAotConfig,\n  getStatsConfig,\n  getStylesConfig,\n} from '../angular-cli-files/models/webpack-configs';\nimport { readTsconfig } from '../angular-cli-files/utilities/read-tsconfig';\nimport { requireProjectModule } from '../angular-cli-files/utilities/require-project-module';\nimport { augmentAppWithServiceWorker } from '../angular-cli-files/utilities/service-worker';\nimport {\n  statsErrorsToString,\n  statsToString,\n  statsWarningsToString,\n} from '../angular-cli-files/utilities/stats';\nimport { normalizeAssetPatterns, normalizeFileReplacements } from '../utils';\nimport { AssetPatternObject, BrowserBuilderSchema, CurrentFileReplacement } from './schema';\nconst webpackMerge = require('webpack-merge');\n\n\n// TODO: figure out a better way to normalize assets, extra entry points, file replacements,\n// and whatever else needs to be normalized, while keeping type safety.\n// Right now this normalization has to be done in all other builders that make use of the\n// BrowserBuildSchema and BrowserBuilder.buildWebpackConfig.\n// It would really help if it happens during architect.validateBuilderOptions, or similar.\nexport interface NormalizedBrowserBuilderSchema extends BrowserBuilderSchema {\n  assets: AssetPatternObject[];\n  fileReplacements: CurrentFileReplacement[];\n}\n\nexport class BrowserBuilder implements Builder<BrowserBuilderSchema> {\n\n  constructor(public context: BuilderContext) { }\n\n  run(builderConfig: BuilderConfiguration<BrowserBuilderSchema>): Observable<BuildEvent> {\n    const options = builderConfig.options;\n    const root = this.context.workspace.root;\n    const projectRoot = resolve(root, builderConfig.root);\n    const host = new virtualFs.AliasHost(this.context.host as virtualFs.Host<fs.Stats>);\n    const webpackBuilder = new WebpackBuilder({ ...this.context, host });\n\n    return of(null).pipe(\n      concatMap(() => options.deleteOutputPath\n        ? this._deleteOutputDir(root, normalize(options.outputPath), this.context.host)\n        : of(null)),\n      concatMap(() => normalizeFileReplacements(options.fileReplacements, host, root)),\n      tap(fileReplacements => options.fileReplacements = fileReplacements),\n      concatMap(() => normalizeAssetPatterns(\n        options.assets, host, root, projectRoot, builderConfig.sourceRoot)),\n      // Replace the assets in options with the normalized version.\n      tap((assetPatternObjects => options.assets = assetPatternObjects)),\n      concatMap(() => {\n        // Ensure Build Optimizer is only used with AOT.\n        if (options.buildOptimizer && !options.aot) {\n          throw new Error('The `--build-optimizer` option cannot be used without `--aot`.');\n        }\n\n        let webpackConfig;\n        try {\n          webpackConfig = this.buildWebpackConfig(root, projectRoot, host,\n            options as NormalizedBrowserBuilderSchema);\n        } catch (e) {\n          return throwError(e);\n        }\n\n        return webpackBuilder.runWebpack(webpackConfig, getBrowserLoggingCb(options.verbose));\n      }),\n      concatMap(buildEvent => {\n        if (buildEvent.success && !options.watch && options.serviceWorker) {\n          return new Observable(obs => {\n            augmentAppWithServiceWorker(\n              this.context.host,\n              root,\n              projectRoot,\n              resolve(root, normalize(options.outputPath)),\n              options.baseHref || '/',\n              options.ngswConfigPath,\n            ).then(\n              () => {\n                obs.next({ success: true });\n                obs.complete();\n              },\n              (err: Error) => {\n                obs.error(err);\n              },\n            );\n          });\n        } else {\n          return of(buildEvent);\n        }\n      }),\n    );\n  }\n\n  buildWebpackConfig(\n    root: Path,\n    projectRoot: Path,\n    host: virtualFs.Host<fs.Stats>,\n    options: NormalizedBrowserBuilderSchema,\n  ) {\n    let wco: WebpackConfigOptions<NormalizedBrowserBuilderSchema>;\n\n    const tsConfigPath = getSystemPath(normalize(resolve(root, normalize(options.tsConfig))));\n    const tsConfig = readTsconfig(tsConfigPath);\n\n    const projectTs = requireProjectModule(getSystemPath(projectRoot), 'typescript') as typeof ts;\n\n    const supportES2015 = tsConfig.options.target !== projectTs.ScriptTarget.ES3\n      && tsConfig.options.target !== projectTs.ScriptTarget.ES5;\n\n    wco = {\n      root: getSystemPath(root),\n      projectRoot: getSystemPath(projectRoot),\n      buildOptions: options,\n      tsConfig,\n      tsConfigPath,\n      supportES2015,\n    };\n\n    const webpackConfigs: {}[] = [\n      getCommonConfig(wco),\n      getBrowserConfig(wco),\n      getStylesConfig(wco),\n      getStatsConfig(wco),\n    ];\n\n    if (wco.buildOptions.main || wco.buildOptions.polyfills) {\n      const typescriptConfigPartial = wco.buildOptions.aot\n        ? getAotConfig(wco, host)\n        : getNonAotConfig(wco, host);\n      webpackConfigs.push(typescriptConfigPartial);\n    }\n\n    return webpackMerge(webpackConfigs);\n  }\n\n  private _deleteOutputDir(root: Path, outputPath: Path, host: virtualFs.Host) {\n    const resolvedOutputPath = resolve(root, outputPath);\n    if (resolvedOutputPath === root) {\n      throw new Error('Output path MUST not be project root directory!');\n    }\n\n    return host.exists(resolvedOutputPath).pipe(\n      concatMap(exists => exists\n        // TODO: remove this concat once host ops emit an event.\n        ? concat(host.delete(resolvedOutputPath), of(null)).pipe(last())\n        // ? of(null)\n        : of(null)),\n    );\n  }\n}\n\nexport const getBrowserLoggingCb = (verbose: boolean): LoggingCallback =>\n  (stats, config, logger) => {\n    // config.stats contains our own stats settings, added during buildWebpackConfig().\n    const json = stats.toJson(config.stats);\n    if (verbose) {\n      logger.info(stats.toString(config.stats));\n    } else {\n      logger.info(statsToString(json, config.stats));\n    }\n\n    if (stats.hasWarnings()) {\n      logger.warn(statsWarningsToString(json, config.stats));\n    }\n    if (stats.hasErrors()) {\n      logger.error(statsErrorsToString(json, config.stats));\n    }\n  };\n\nexport default BrowserBuilder;\n"]}