UNPKG

@angular-devkit/build-angular

Version:
129 lines • 23.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const core_1 = require("@angular-devkit/core"); const rxjs_1 = require("rxjs"); const operators_1 = require("rxjs/operators"); const require_project_module_1 = require("../angular-cli-files/utilities/require-project-module"); const service_worker_1 = require("../angular-cli-files/utilities/service-worker"); class AppShellBuilder { constructor(context) { this.context = context; } run(builderConfig) { const options = builderConfig.options; return new rxjs_1.Observable(obs => { let success = true; const subscription = rxjs_1.merge(this.build(options.serverTarget, {}), // Never run the browser target in watch mode. // If service worker is needed, it will be added in this.renderUniversal(); this.build(options.browserTarget, { watch: false, serviceWorker: false })).subscribe((event) => { // TODO: once we support a better build event, add support for merging two event streams // together. success = success && event.success; }, error => { obs.error(error); }, () => { obs.next({ success }); obs.complete(); }); // Allow subscriptions to us to unsubscribe from each builds at the same time. return () => subscription.unsubscribe(); }).pipe(operators_1.switchMap(event => { if (!event.success) { return rxjs_1.of(event); } return this.renderUniversal(options); })); } build(targetString, overrides) { const architect = this.context.architect; const [project, target, configuration] = targetString.split(':'); // Override browser build watch setting. const builderConfig = architect.getBuilderConfiguration({ project, target, configuration, overrides, }); return architect.run(builderConfig, this.context); } getServerModuleBundlePath(options) { const architect = this.context.architect; return new rxjs_1.Observable(obs => { if (options.appModuleBundle) { obs.next(core_1.join(this.context.workspace.root, options.appModuleBundle)); return obs.complete(); } else { const [project, target, configuration] = options.serverTarget.split(':'); const builderConfig = architect.getBuilderConfiguration({ project, target, configuration, }); return architect.getBuilderDescription(builderConfig).pipe(operators_1.concatMap(description => architect.validateBuilderOptions(builderConfig, description)), operators_1.switchMap(config => { const outputPath = core_1.join(this.context.workspace.root, config.options.outputPath); return this.context.host.list(outputPath).pipe(operators_1.switchMap(files => { const re = /^main\.(?:[a-zA-Z0-9]{20}\.)?(?:bundle\.)?js$/; const maybeMain = files.filter(x => re.test(x))[0]; if (!maybeMain) { return rxjs_1.throwError(new Error('Could not find the main bundle.')); } else { return rxjs_1.of(core_1.join(outputPath, maybeMain)); } })); })).subscribe(obs); } }); } getBrowserBuilderConfig(options) { const architect = this.context.architect; const [project, target, configuration] = options.browserTarget.split(':'); const builderConfig = architect.getBuilderConfiguration({ project, target, configuration, }); return architect.getBuilderDescription(builderConfig).pipe(operators_1.concatMap(description => architect.validateBuilderOptions(builderConfig, description))); } renderUniversal(options) { let browserOptions; let projectRoot; return rxjs_1.forkJoin(this.getBrowserBuilderConfig(options).pipe(operators_1.switchMap(config => { browserOptions = config.options; projectRoot = core_1.resolve(this.context.workspace.root, config.root); const browserIndexOutputPath = core_1.join(core_1.normalize(browserOptions.outputPath), 'index.html'); const path = core_1.join(this.context.workspace.root, browserIndexOutputPath); return this.context.host.read(path).pipe(operators_1.map(x => { return [browserIndexOutputPath, x]; })); })), this.getServerModuleBundlePath(options)).pipe(operators_1.switchMap(([[browserIndexOutputPath, indexContent], serverBundlePath]) => { const root = this.context.workspace.root; require_project_module_1.requireProjectModule(core_1.getSystemPath(root), 'zone.js/dist/zone-node'); const renderModuleFactory = require_project_module_1.requireProjectModule(core_1.getSystemPath(root), '@angular/platform-server').renderModuleFactory; const AppServerModuleNgFactory = require(core_1.getSystemPath(serverBundlePath)).AppServerModuleNgFactory; const indexHtml = core_1.virtualFs.fileBufferToString(indexContent); const outputIndexPath = core_1.join(root, options.outputIndexPath || browserIndexOutputPath); // Render to HTML and overwrite the client index file. return rxjs_1.from(renderModuleFactory(AppServerModuleNgFactory, { document: indexHtml, url: options.route, }) .then((html) => { return this.context.host .write(outputIndexPath, core_1.virtualFs.stringToFileBuffer(html)) .toPromise(); }) .then(() => { if (browserOptions.serviceWorker) { return service_worker_1.augmentAppWithServiceWorker(this.context.host, root, projectRoot, core_1.join(root, browserOptions.outputPath), browserOptions.baseHref || '/', browserOptions.ngswConfigPath); } }) .then(() => ({ success: true }))); })); } } exports.AppShellBuilder = AppShellBuilder; exports.default = AppShellBuilder; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"./","sources":["packages/angular_devkit/build_angular/src/app-shell/index.ts"],"names":[],"mappings":";;AAaA,+CAAgG;AAChG,+BAAyE;AACzE,8CAA2D;AAC3D,kGAA6F;AAC7F,kFAA4F;AAM5F;IAEE,YAAmB,OAAuB;QAAvB,YAAO,GAAP,OAAO,CAAgB;IAAI,CAAC;IAE/C,GAAG,CAAC,aAA+D;QACjE,MAAM,OAAO,GAAG,aAAa,CAAC,OAAO,CAAC;QAEtC,OAAO,IAAI,iBAAU,CAAa,GAAG,CAAC,EAAE;YACtC,IAAI,OAAO,GAAG,IAAI,CAAC;YACnB,MAAM,YAAY,GAAG,YAAK,CACxB,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC;YACpC,8CAA8C;YAC9C,2EAA2E;YAC3E,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,aAAa,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,CAAC,CAC1E,CAAC,SAAS,CAAC,CAAC,KAAiB,EAAE,EAAE;gBAChC,wFAAwF;gBACxF,YAAY;gBACZ,OAAO,GAAG,OAAO,IAAI,KAAK,CAAC,OAAO,CAAC;YACrC,CAAC,EAAE,KAAK,CAAC,EAAE;gBACT,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACnB,CAAC,EAAE,GAAG,EAAE;gBACN,GAAG,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC;gBACtB,GAAG,CAAC,QAAQ,EAAE,CAAC;YACjB,CAAC,CAAC,CAAC;YAEH,8EAA8E;YAC9E,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;QAC1C,CAAC,CAAC,CAAC,IAAI,CACL,qBAAS,CAAC,KAAK,CAAC,EAAE;YAChB,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE;gBAClB,OAAO,SAAE,CAAC,KAAK,CAAC,CAAC;aAClB;YAED,OAAO,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC;QACvC,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,YAAoB,EAAE,SAAa;QACvC,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC;QACzC,MAAM,CAAC,OAAO,EAAE,MAAM,EAAE,aAAa,CAAC,GAAG,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAEjE,wCAAwC;QACxC,MAAM,aAAa,GAAG,SAAS,CAAC,uBAAuB,CAAK;YAC1D,OAAO;YACP,MAAM;YACN,aAAa;YACb,SAAS;SACV,CAAC,CAAC;QAEH,OAAO,SAAS,CAAC,GAAG,CAAC,aAAa,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;IACpD,CAAC;IAED,yBAAyB,CAAC,OAAmC;QAC3D,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC;QAEzC,OAAO,IAAI,iBAAU,CAAO,GAAG,CAAC,EAAE;YAChC,IAAI,OAAO,CAAC,eAAe,EAAE;gBAC3B,GAAG,CAAC,IAAI,CAAC,WAAI,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC,CAAC;gBAErE,OAAO,GAAG,CAAC,QAAQ,EAAE,CAAC;aACvB;iBAAM;gBACL,MAAM,CAAC,OAAO,EAAE,MAAM,EAAE,aAAa,CAAC,GAAG,OAAO,CAAC,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBACzE,MAAM,aAAa,GAAG,SAAS,CAAC,uBAAuB,CAA2B;oBAChF,OAAO;oBACP,MAAM;oBACN,aAAa;iBACd,CAAC,CAAC;gBAEH,OAAO,SAAS,CAAC,qBAAqB,CAAC,aAAa,CAAC,CAAC,IAAI,CACxD,qBAAS,CAAC,WAAW,CAAC,EAAE,CAAC,SAAS,CAAC,sBAAsB,CAAC,aAAa,EAAE,WAAW,CAAC,CAAC,EACtF,qBAAS,CAAC,MAAM,CAAC,EAAE;oBACjB,MAAM,UAAU,GAAG,WAAI,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,EAAE,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;oBAEhF,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,IAAI,CAC5C,qBAAS,CAAC,KAAK,CAAC,EAAE;wBAChB,MAAM,EAAE,GAAG,+CAA+C,CAAC;wBAC3D,MAAM,SAAS,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;wBAEnD,IAAI,CAAC,SAAS,EAAE;4BACd,OAAO,iBAAU,CAAC,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC,CAAC;yBACjE;6BAAM;4BACL,OAAO,SAAE,CAAC,WAAI,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC,CAAC;yBACxC;oBACH,CAAC,CAAC,CACH,CAAC;gBACJ,CAAC,CAAC,CACH,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;aAClB;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,uBAAuB,CAAC,OAAmC;QACzD,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC;QACzC,MAAM,CAAC,OAAO,EAAE,MAAM,EAAE,aAAa,CAAC,GAAG,OAAO,CAAC,aAAa,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC1E,MAAM,aAAa,GAAG,SAAS,CAAC,uBAAuB,CAAuB;YAC5E,OAAO;YACP,MAAM;YACN,aAAa;SACd,CAAC,CAAC;QAEH,OAAO,SAAS,CAAC,qBAAqB,CAAC,aAAa,CAAC,CAAC,IAAI,CACxD,qBAAS,CAAC,WAAW,CAAC,EAAE,CAAC,SAAS,CAAC,sBAAsB,CAAC,aAAa,EAAE,WAAW,CAAC,CAAC,CACvF,CAAC;IACJ,CAAC;IAED,eAAe,CAAC,OAAmC;QACjD,IAAI,cAAoC,CAAC;QACzC,IAAI,WAAiB,CAAC;QAEtB,OAAO,eAAQ,CACb,IAAI,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC,IAAI,CACxC,qBAAS,CAAC,MAAM,CAAC,EAAE;YACjB,cAAc,GAAG,MAAM,CAAC,OAAO,CAAC;YAChC,WAAW,GAAG,cAAO,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC;YAChE,MAAM,sBAAsB,GAAG,WAAI,CAAC,gBAAS,CAAC,cAAc,CAAC,UAAU,CAAC,EAAE,YAAY,CAAC,CAAC;YACxF,MAAM,IAAI,GAAG,WAAI,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,EAAE,sBAAsB,CAAC,CAAC;YAEvE,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,CACtC,eAAG,CAAqD,CAAC,CAAC,EAAE;gBAC1D,OAAO,CAAC,sBAAsB,EAAE,CAAC,CAAC,CAAC;YACrC,CAAC,CAAC,CACH,CAAC;QACJ,CAAC,CAAC,CACH,EACD,IAAI,CAAC,yBAAyB,CAAC,OAAO,CAAC,CACxC,CAAC,IAAI,CACJ,qBAAS,CAAC,CAAC,CAAC,CAAC,sBAAsB,EAAE,YAAY,CAAC,EAAE,gBAAgB,CAAC,EAAE,EAAE;YACvE,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC;YACzC,6CAAoB,CAAC,oBAAa,CAAC,IAAI,CAAC,EAAE,wBAAwB,CAAC,CAAC;YAEpE,MAAM,mBAAmB,GAAG,6CAAoB,CAC9C,oBAAa,CAAC,IAAI,CAAC,EACnB,0BAA0B,CAC3B,CAAC,mBAAmB,CAAC;YACtB,MAAM,wBAAwB,GAAG,OAAO,CACtC,oBAAa,CAAC,gBAAgB,CAAC,CAChC,CAAC,wBAAwB,CAAC;YAC3B,MAAM,SAAS,GAAG,gBAAS,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;YAC7D,MAAM,eAAe,GAAG,WAAI,CAAC,IAAI,EAAE,OAAO,CAAC,eAAe,IAAI,sBAAsB,CAAC,CAAC;YAEtF,sDAAsD;YACtD,OAAO,WAAI,CACT,mBAAmB,CAAC,wBAAwB,EAAE;gBAC5C,QAAQ,EAAE,SAAS;gBACnB,GAAG,EAAE,OAAO,CAAC,KAAK;aACnB,CAAC;iBACD,IAAI,CAAC,CAAC,IAAY,EAAE,EAAE;gBACrB,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI;qBACrB,KAAK,CAAC,eAAe,EAAE,gBAAS,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC;qBAC1D,SAAS,EAAE,CAAC;YACjB,CAAC,CAAC;iBACD,IAAI,CAAC,GAAG,EAAE;gBACT,IAAI,cAAc,CAAC,aAAa,EAAE;oBAChC,OAAO,4CAA2B,CAChC,IAAI,CAAC,OAAO,CAAC,IAAI,EACjB,IAAI,EACJ,WAAW,EACX,WAAI,CAAC,IAAI,EAAE,cAAc,CAAC,UAAU,CAAC,EACrC,cAAc,CAAC,QAAQ,IAAI,GAAG,EAC9B,cAAc,CAAC,cAAc,CAC9B,CAAC;iBACH;YACH,CAAC,CAAC;iBACD,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CACjC,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;CACF;AAzKD,0CAyKC;AAED,kBAAe,eAAe,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 { Path, getSystemPath, join, normalize, resolve, virtualFs } from '@angular-devkit/core';\nimport { Observable, forkJoin, from, merge, of, throwError } from 'rxjs';\nimport { concatMap, map, switchMap } from 'rxjs/operators';\nimport { requireProjectModule } from '../angular-cli-files/utilities/require-project-module';\nimport { augmentAppWithServiceWorker } from '../angular-cli-files/utilities/service-worker';\nimport { BrowserBuilderSchema } from '../browser/schema';\nimport { BuildWebpackServerSchema } from '../server/schema';\nimport { BuildWebpackAppShellSchema } from './schema';\n\n\nexport class AppShellBuilder implements Builder<BuildWebpackAppShellSchema> {\n\n  constructor(public context: BuilderContext) { }\n\n  run(builderConfig: BuilderConfiguration<BuildWebpackAppShellSchema>): Observable<BuildEvent> {\n    const options = builderConfig.options;\n\n    return new Observable<BuildEvent>(obs => {\n      let success = true;\n      const subscription = merge(\n        this.build(options.serverTarget, {}),\n        // Never run the browser target in watch mode.\n        // If service worker is needed, it will be added in this.renderUniversal();\n        this.build(options.browserTarget, { watch: false, serviceWorker: false }),\n      ).subscribe((event: BuildEvent) => {\n        // TODO: once we support a better build event, add support for merging two event streams\n        // together.\n        success = success && event.success;\n      }, error => {\n        obs.error(error);\n      }, () => {\n        obs.next({ success });\n        obs.complete();\n      });\n\n      // Allow subscriptions to us to unsubscribe from each builds at the same time.\n      return () => subscription.unsubscribe();\n    }).pipe(\n      switchMap(event => {\n        if (!event.success) {\n          return of(event);\n        }\n\n        return this.renderUniversal(options);\n      }),\n    );\n  }\n\n  build(targetString: string, overrides: {}) {\n    const architect = this.context.architect;\n    const [project, target, configuration] = targetString.split(':');\n\n    // Override browser build watch setting.\n    const builderConfig = architect.getBuilderConfiguration<{}>({\n      project,\n      target,\n      configuration,\n      overrides,\n    });\n\n    return architect.run(builderConfig, this.context);\n  }\n\n  getServerModuleBundlePath(options: BuildWebpackAppShellSchema) {\n    const architect = this.context.architect;\n\n    return new Observable<Path>(obs => {\n      if (options.appModuleBundle) {\n        obs.next(join(this.context.workspace.root, options.appModuleBundle));\n\n        return obs.complete();\n      } else {\n        const [project, target, configuration] = options.serverTarget.split(':');\n        const builderConfig = architect.getBuilderConfiguration<BuildWebpackServerSchema>({\n          project,\n          target,\n          configuration,\n        });\n\n        return architect.getBuilderDescription(builderConfig).pipe(\n          concatMap(description => architect.validateBuilderOptions(builderConfig, description)),\n          switchMap(config => {\n            const outputPath = join(this.context.workspace.root, config.options.outputPath);\n\n            return this.context.host.list(outputPath).pipe(\n              switchMap(files => {\n                const re = /^main\\.(?:[a-zA-Z0-9]{20}\\.)?(?:bundle\\.)?js$/;\n                const maybeMain = files.filter(x => re.test(x))[0];\n\n                if (!maybeMain) {\n                  return throwError(new Error('Could not find the main bundle.'));\n                } else {\n                  return of(join(outputPath, maybeMain));\n                }\n              }),\n            );\n          }),\n        ).subscribe(obs);\n      }\n    });\n  }\n\n  getBrowserBuilderConfig(options: BuildWebpackAppShellSchema) {\n    const architect = this.context.architect;\n    const [project, target, configuration] = options.browserTarget.split(':');\n    const builderConfig = architect.getBuilderConfiguration<BrowserBuilderSchema>({\n      project,\n      target,\n      configuration,\n    });\n\n    return architect.getBuilderDescription(builderConfig).pipe(\n      concatMap(description => architect.validateBuilderOptions(builderConfig, description)),\n    );\n  }\n\n  renderUniversal(options: BuildWebpackAppShellSchema): Observable<BuildEvent> {\n    let browserOptions: BrowserBuilderSchema;\n    let projectRoot: Path;\n\n    return forkJoin(\n      this.getBrowserBuilderConfig(options).pipe(\n        switchMap(config => {\n          browserOptions = config.options;\n          projectRoot = resolve(this.context.workspace.root, config.root);\n          const browserIndexOutputPath = join(normalize(browserOptions.outputPath), 'index.html');\n          const path = join(this.context.workspace.root, browserIndexOutputPath);\n\n          return this.context.host.read(path).pipe(\n            map<virtualFs.FileBuffer, [Path, virtualFs.FileBuffer]>(x => {\n              return [browserIndexOutputPath, x];\n            }),\n          );\n        }),\n      ),\n      this.getServerModuleBundlePath(options),\n    ).pipe(\n      switchMap(([[browserIndexOutputPath, indexContent], serverBundlePath]) => {\n        const root = this.context.workspace.root;\n        requireProjectModule(getSystemPath(root), 'zone.js/dist/zone-node');\n\n        const renderModuleFactory = requireProjectModule(\n          getSystemPath(root),\n          '@angular/platform-server',\n        ).renderModuleFactory;\n        const AppServerModuleNgFactory = require(\n          getSystemPath(serverBundlePath),\n        ).AppServerModuleNgFactory;\n        const indexHtml = virtualFs.fileBufferToString(indexContent);\n        const outputIndexPath = join(root, options.outputIndexPath || browserIndexOutputPath);\n\n        // Render to HTML and overwrite the client index file.\n        return from(\n          renderModuleFactory(AppServerModuleNgFactory, {\n            document: indexHtml,\n            url: options.route,\n          })\n          .then((html: string) => {\n            return this.context.host\n              .write(outputIndexPath, virtualFs.stringToFileBuffer(html))\n              .toPromise();\n          })\n          .then(() => {\n            if (browserOptions.serviceWorker) {\n              return augmentAppWithServiceWorker(\n                this.context.host,\n                root,\n                projectRoot,\n                join(root, browserOptions.outputPath),\n                browserOptions.baseHref || '/',\n                browserOptions.ngswConfigPath,\n              );\n            }\n          })\n          .then(() => ({ success: true })),\n        );\n      }),\n    );\n  }\n}\n\nexport default AppShellBuilder;\n"]}