index.js 23.9 KB
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
/**
 * @license
 * Copyright Google Inc. All Rights Reserved.
 *
 * Use of this source code is governed by an MIT-style license that can be
 * found in the LICENSE file at https://angular.io/license
 */
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const ts = require("typescript");
const stringUtils = require("../strings");
const ast_utils_1 = require("../utility/ast-utils");
const change_1 = require("../utility/change");
const config_1 = require("../utility/config");
const ng_ast_utils_1 = require("../utility/ng-ast-utils");
function updateConfigFile(options) {
    return (host) => {
        const config = config_1.getConfig(host);
        const clientApp = config_1.getAppFromConfig(config, options.clientApp || '0');
        if (clientApp === null) {
            throw new schematics_1.SchematicsException('Client app not found.');
        }
        options.test = options.test || clientApp.test;
        const tsCfg = options.tsconfigFileName && options.tsconfigFileName.endsWith('.json')
            ? options.tsconfigFileName : `${options.tsconfigFileName}.json`;
        const testTsCfg = options.testTsconfigFileName && options.testTsconfigFileName.endsWith('.json')
            ? options.testTsconfigFileName : `${options.testTsconfigFileName}.json`;
        const serverApp = Object.assign({}, clientApp, { platform: 'server', root: options.root, outDir: options.outDir, index: options.index, main: options.main, test: options.test, tsconfig: tsCfg, testTsconfig: testTsCfg, polyfills: undefined });
        if (options.name) {
            serverApp.name = options.name;
        }
        if (!config.apps) {
            config.apps = [];
        }
        config.apps.push(serverApp);
        host.overwrite('/.angular-cli.json', JSON.stringify(config, null, 2));
        return host;
    };
}
function findBrowserModuleImport(host, modulePath) {
    const moduleBuffer = host.read(modulePath);
    if (!moduleBuffer) {
        throw new schematics_1.SchematicsException(`Module file (${modulePath}) not found`);
    }
    const moduleFileText = moduleBuffer.toString('utf-8');
    const source = ts.createSourceFile(modulePath, moduleFileText, ts.ScriptTarget.Latest, true);
    const decoratorMetadata = ast_utils_1.getDecoratorMetadata(source, 'NgModule', '@angular/core')[0];
    const browserModuleNode = ast_utils_1.findNode(decoratorMetadata, ts.SyntaxKind.Identifier, 'BrowserModule');
    if (browserModuleNode === null) {
        throw new schematics_1.SchematicsException(`Cannot find BrowserModule import in ${modulePath}`);
    }
    return browserModuleNode;
}
function wrapBootstrapCall(options) {
    return (host) => {
        const config = config_1.getConfig(host);
        const clientApp = config_1.getAppFromConfig(config, options.clientApp || '0');
        if (clientApp === null) {
            throw new schematics_1.SchematicsException('Client app not found.');
        }
        const mainPath = core_1.normalize(`/${clientApp.root}/${clientApp.main}`);
        let bootstrapCall = ng_ast_utils_1.findBootstrapModuleCall(host, mainPath);
        if (bootstrapCall === null) {
            throw new schematics_1.SchematicsException('Bootstrap module not found.');
        }
        let bootstrapCallExpression = null;
        let currentCall = bootstrapCall;
        while (bootstrapCallExpression === null && currentCall.parent) {
            currentCall = currentCall.parent;
            if (currentCall.kind === ts.SyntaxKind.ExpressionStatement) {
                bootstrapCallExpression = currentCall;
            }
        }
        bootstrapCall = currentCall;
        const recorder = host.beginUpdate(mainPath);
        const beforeText = `document.addEventListener('DOMContentLoaded', () => {\n  `;
        const afterText = `\n});`;
        recorder.insertLeft(bootstrapCall.getStart(), beforeText);
        recorder.insertRight(bootstrapCall.getEnd(), afterText);
        host.commitUpdate(recorder);
    };
}
function addServerTransition(options) {
    return (host) => {
        const config = config_1.getConfig(host);
        const clientApp = config_1.getAppFromConfig(config, options.clientApp || '0');
        if (clientApp === null) {
            throw new schematics_1.SchematicsException('Client app not found.');
        }
        const mainPath = core_1.normalize(`/${clientApp.root}/${clientApp.main}`);
        const bootstrapModuleRelativePath = ng_ast_utils_1.findBootstrapModulePath(host, mainPath);
        const bootstrapModulePath = core_1.normalize(`/${clientApp.root}/${bootstrapModuleRelativePath}.ts`);
        const browserModuleImport = findBrowserModuleImport(host, bootstrapModulePath);
        const appId = options.appId;
        const transitionCall = `.withServerTransition({ appId: '${appId}' })`;
        const position = browserModuleImport.pos + browserModuleImport.getFullText().length;
        const transitionCallChange = new change_1.InsertChange(bootstrapModulePath, position, transitionCall);
        const transitionCallRecorder = host.beginUpdate(bootstrapModulePath);
        transitionCallRecorder.insertLeft(transitionCallChange.pos, transitionCallChange.toAdd);
        host.commitUpdate(transitionCallRecorder);
    };
}
function addDependencies() {
    return (host) => {
        const pkgPath = '/package.json';
        const buffer = host.read(pkgPath);
        if (buffer === null) {
            throw new schematics_1.SchematicsException('Could not find package.json');
        }
        const pkg = JSON.parse(buffer.toString());
        const ngCoreVersion = pkg.dependencies['@angular/core'];
        pkg.dependencies['@angular/platform-server'] = ngCoreVersion;
        host.overwrite(pkgPath, JSON.stringify(pkg, null, 2));
        return host;
    };
}
function updateGitignore(options) {
    return (host) => {
        const ignorePath = core_1.normalize('/.gitignore');
        const buffer = host.read(ignorePath);
        if (buffer === null) {
            // Assumption is made that there is no git repository.
            return host;
        }
        else {
            const content = buffer.toString();
            host.overwrite(ignorePath, `${content}\n${options.outDir}`);
        }
        return host;
    };
}
function default_1(options) {
    return (host, context) => {
        const templateSource = schematics_1.apply(schematics_1.url('./files'), [
            schematics_1.template(Object.assign({}, stringUtils, options, { stripTsExtension: (s) => { return s.replace(/\.ts$/, ''); } })),
        ]);
        return schematics_1.chain([
            schematics_1.mergeWith(templateSource),
            addDependencies(),
            updateConfigFile(options),
            wrapBootstrapCall(options),
            addServerTransition(options),
            updateGitignore(options),
        ])(host, context);
    };
}
exports.default = default_1;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"/Users/hansl/Sources/hansl/devkit/","sources":["packages/schematics/angular/universal/index.ts"],"names":[],"mappings":";;AAAA;;;;;;GAMG;AACH,+CAAiD;AACjD,2DAUoC;AACpC,iCAAiC;AACjC,0CAA0C;AAC1C,oDAAsE;AACtE,8CAAiD;AACjD,8CAA2E;AAC3E,0DAA2F;AAI3F,0BAA0B,OAAyB;IACjD,MAAM,CAAC,CAAC,IAAU,EAAE,EAAE;QACpB,MAAM,MAAM,GAAG,kBAAS,CAAC,IAAI,CAAC,CAAC;QAC/B,MAAM,SAAS,GAAG,yBAAgB,CAAC,MAAM,EAAE,OAAO,CAAC,SAAS,IAAI,GAAG,CAAC,CAAC;QACrE,EAAE,CAAC,CAAC,SAAS,KAAK,IAAI,CAAC,CAAC,CAAC;YACvB,MAAM,IAAI,gCAAmB,CAAC,uBAAuB,CAAC,CAAC;QACzD,CAAC;QACD,OAAO,CAAC,IAAI,GAAG,OAAO,CAAC,IAAI,IAAI,SAAS,CAAC,IAAI,CAAC;QAE9C,MAAM,KAAK,GAAG,OAAO,CAAC,gBAAgB,IAAI,OAAO,CAAC,gBAAgB,CAAC,QAAQ,CAAC,OAAO,CAAC;YAClF,CAAC,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,gBAAgB,OAAO,CAAC;QAClE,MAAM,SAAS,GAAG,OAAO,CAAC,oBAAoB,IAAI,OAAO,CAAC,oBAAoB,CAAC,QAAQ,CAAC,OAAO,CAAC;YAC9F,CAAC,CAAC,OAAO,CAAC,oBAAoB,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,oBAAoB,OAAO,CAAC;QAE1E,MAAM,SAAS,qBACV,SAAS,IACZ,QAAQ,EAAE,QAAQ,EAClB,IAAI,EAAE,OAAO,CAAC,IAAI,EAClB,MAAM,EAAE,OAAO,CAAC,MAAM,EACtB,KAAK,EAAE,OAAO,CAAC,KAAK,EACpB,IAAI,EAAE,OAAO,CAAC,IAAI,EAClB,IAAI,EAAE,OAAO,CAAC,IAAI,EAClB,QAAQ,EAAE,KAAK,EACf,YAAY,EAAE,SAAS,EACvB,SAAS,EAAE,SAAS,GACrB,CAAC;QACF,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;YACjB,SAAS,CAAC,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;QAChC,CAAC;QACD,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC;YACjB,MAAM,CAAC,IAAI,GAAG,EAAE,CAAC;QACnB,CAAC;QACD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAE5B,IAAI,CAAC,SAAS,CAAC,oBAAoB,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAEtE,MAAM,CAAC,IAAI,CAAC;IACd,CAAC,CAAC;AACJ,CAAC;AAED,iCAAiC,IAAU,EAAE,UAAkB;IAC7D,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC3C,EAAE,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC;QAClB,MAAM,IAAI,gCAAmB,CAAC,gBAAgB,UAAU,aAAa,CAAC,CAAC;IACzE,CAAC;IACD,MAAM,cAAc,GAAG,YAAY,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;IAEtD,MAAM,MAAM,GAAG,EAAE,CAAC,gBAAgB,CAAC,UAAU,EAAE,cAAc,EAAE,EAAE,CAAC,YAAY,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;IAE7F,MAAM,iBAAiB,GAAG,gCAAoB,CAAC,MAAM,EAAE,UAAU,EAAE,eAAe,CAAC,CAAC,CAAC,CAAC,CAAC;IACvF,MAAM,iBAAiB,GAAG,oBAAQ,CAAC,iBAAiB,EAAE,EAAE,CAAC,UAAU,CAAC,UAAU,EAAE,eAAe,CAAC,CAAC;IAEjG,EAAE,CAAC,CAAC,iBAAiB,KAAK,IAAI,CAAC,CAAC,CAAC;QAC/B,MAAM,IAAI,gCAAmB,CAAC,uCAAuC,UAAU,EAAE,CAAC,CAAC;IACrF,CAAC;IAED,MAAM,CAAC,iBAAiB,CAAC;AAC3B,CAAC;AAED,2BAA2B,OAAyB;IAClD,MAAM,CAAC,CAAC,IAAU,EAAE,EAAE;QACpB,MAAM,MAAM,GAAG,kBAAS,CAAC,IAAI,CAAC,CAAC;QAC/B,MAAM,SAAS,GAAG,yBAAgB,CAAC,MAAM,EAAE,OAAO,CAAC,SAAS,IAAI,GAAG,CAAC,CAAC;QACrE,EAAE,CAAC,CAAC,SAAS,KAAK,IAAI,CAAC,CAAC,CAAC;YACvB,MAAM,IAAI,gCAAmB,CAAC,uBAAuB,CAAC,CAAC;QACzD,CAAC;QACD,MAAM,QAAQ,GAAG,gBAAS,CAAC,IAAI,SAAS,CAAC,IAAI,IAAI,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC;QACnE,IAAI,aAAa,GAAmB,sCAAuB,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QAC5E,EAAE,CAAC,CAAC,aAAa,KAAK,IAAI,CAAC,CAAC,CAAC;YAC3B,MAAM,IAAI,gCAAmB,CAAC,6BAA6B,CAAC,CAAC;QAC/D,CAAC;QAED,IAAI,uBAAuB,GAAmB,IAAI,CAAC;QACnD,IAAI,WAAW,GAAG,aAAa,CAAC;QAChC,OAAO,uBAAuB,KAAK,IAAI,IAAI,WAAW,CAAC,MAAM,EAAE,CAAC;YAC9D,WAAW,GAAG,WAAW,CAAC,MAAM,CAAC;YACjC,EAAE,CAAC,CAAC,WAAW,CAAC,IAAI,KAAK,EAAE,CAAC,UAAU,CAAC,mBAAmB,CAAC,CAAC,CAAC;gBAC3D,uBAAuB,GAAG,WAAW,CAAC;YACxC,CAAC;QACH,CAAC;QACD,aAAa,GAAG,WAAW,CAAC;QAE5B,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QAC5C,MAAM,UAAU,GAAG,2DAA2D,CAAC;QAC/E,MAAM,SAAS,GAAG,OAAO,CAAC;QAC1B,QAAQ,CAAC,UAAU,CAAC,aAAa,CAAC,QAAQ,EAAE,EAAE,UAAU,CAAC,CAAC;QAC1D,QAAQ,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,EAAE,SAAS,CAAC,CAAC;QACxD,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;IAC9B,CAAC,CAAC;AACJ,CAAC;AAED,6BAA6B,OAAyB;IACpD,MAAM,CAAC,CAAC,IAAU,EAAE,EAAE;QACpB,MAAM,MAAM,GAAG,kBAAS,CAAC,IAAI,CAAC,CAAC;QAC/B,MAAM,SAAS,GAAG,yBAAgB,CAAC,MAAM,EAAE,OAAO,CAAC,SAAS,IAAI,GAAG,CAAC,CAAC;QACrE,EAAE,CAAC,CAAC,SAAS,KAAK,IAAI,CAAC,CAAC,CAAC;YACvB,MAAM,IAAI,gCAAmB,CAAC,uBAAuB,CAAC,CAAC;QACzD,CAAC;QACD,MAAM,QAAQ,GAAG,gBAAS,CAAC,IAAI,SAAS,CAAC,IAAI,IAAI,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC;QAEnE,MAAM,2BAA2B,GAAG,sCAAuB,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QAC5E,MAAM,mBAAmB,GAAG,gBAAS,CAAC,IAAI,SAAS,CAAC,IAAI,IAAI,2BAA2B,KAAK,CAAC,CAAC;QAE9F,MAAM,mBAAmB,GAAG,uBAAuB,CAAC,IAAI,EAAE,mBAAmB,CAAC,CAAC;QAC/E,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;QAC5B,MAAM,cAAc,GAAG,mCAAmC,KAAK,MAAM,CAAC;QACtE,MAAM,QAAQ,GAAG,mBAAmB,CAAC,GAAG,GAAG,mBAAmB,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC;QACpF,MAAM,oBAAoB,GAAG,IAAI,qBAAY,CAC3C,mBAAmB,EAAE,QAAQ,EAAE,cAAc,CAAC,CAAC;QAEjD,MAAM,sBAAsB,GAAG,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;QACrE,sBAAsB,CAAC,UAAU,CAAC,oBAAoB,CAAC,GAAG,EAAE,oBAAoB,CAAC,KAAK,CAAC,CAAC;QACxF,IAAI,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;IAC5C,CAAC,CAAC;AACJ,CAAC;AAED;IACE,MAAM,CAAC,CAAC,IAAU,EAAE,EAAE;QACpB,MAAM,OAAO,GAAG,eAAe,CAAC;QAChC,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAClC,EAAE,CAAC,CAAC,MAAM,KAAK,IAAI,CAAC,CAAC,CAAC;YACpB,MAAM,IAAI,gCAAmB,CAAC,6BAA6B,CAAC,CAAC;QAC/D,CAAC;QAED,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;QAE1C,MAAM,aAAa,GAAG,GAAG,CAAC,YAAY,CAAC,eAAe,CAAC,CAAC;QACxD,GAAG,CAAC,YAAY,CAAC,0BAA0B,CAAC,GAAG,aAAa,CAAC;QAE7D,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAEtD,MAAM,CAAC,IAAI,CAAC;IACd,CAAC,CAAC;AACJ,CAAC;AAED,yBAAyB,OAAyB;IAChD,MAAM,CAAC,CAAC,IAAU,EAAE,EAAE;QACpB,MAAM,UAAU,GAAG,gBAAS,CAAC,aAAa,CAAC,CAAC;QAC5C,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACrC,EAAE,CAAC,CAAC,MAAM,KAAK,IAAI,CAAC,CAAC,CAAC;YACpB,sDAAsD;YACtD,MAAM,CAAC,IAAI,CAAC;QACd,CAAC;QAAC,IAAI,CAAC,CAAC;YACN,MAAM,OAAO,GAAG,MAAM,CAAC,QAAQ,EAAE,CAAC;YAClC,IAAI,CAAC,SAAS,CAAC,UAAU,EAAE,GAAG,OAAO,KAAK,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;QAC9D,CAAC;QAED,MAAM,CAAC,IAAI,CAAC;IACd,CAAC,CAAC;AACJ,CAAC;AAED,mBAAyB,OAAyB;IAChD,MAAM,CAAC,CAAC,IAAU,EAAE,OAAyB,EAAE,EAAE;QAC/C,MAAM,cAAc,GAAG,kBAAK,CAAC,gBAAG,CAAC,SAAS,CAAC,EAAE;YAC3C,qBAAQ,mBACH,WAAW,EACX,OAAiB,IACpB,gBAAgB,EAAE,CAAC,CAAS,EAAE,EAAE,GAAG,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,IACnE;SACH,CAAC,CAAC;QAEH,MAAM,CAAC,kBAAK,CAAC;YACX,sBAAS,CAAC,cAAc,CAAC;YACzB,eAAe,EAAE;YACjB,gBAAgB,CAAC,OAAO,CAAC;YACzB,iBAAiB,CAAC,OAAO,CAAC;YAC1B,mBAAmB,CAAC,OAAO,CAAC;YAC5B,eAAe,CAAC,OAAO,CAAC;SACzB,CAAC,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;IACpB,CAAC,CAAC;AACJ,CAAC;AAnBD,4BAmBC","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 { normalize } from '@angular-devkit/core';\nimport {\n  Rule,\n  SchematicContext,\n  SchematicsException,\n  Tree,\n  apply,\n  chain,\n  mergeWith,\n  template,\n  url,\n} from '@angular-devkit/schematics';\nimport * as ts from 'typescript';\nimport * as stringUtils from '../strings';\nimport { findNode, getDecoratorMetadata } from '../utility/ast-utils';\nimport { InsertChange } from '../utility/change';\nimport { AppConfig, getAppFromConfig, getConfig } from '../utility/config';\nimport { findBootstrapModuleCall, findBootstrapModulePath } from '../utility/ng-ast-utils';\nimport { Schema as UniversalOptions } from './schema';\n\n\nfunction updateConfigFile(options: UniversalOptions): Rule {\n  return (host: Tree) => {\n    const config = getConfig(host);\n    const clientApp = getAppFromConfig(config, options.clientApp || '0');\n    if (clientApp === null) {\n      throw new SchematicsException('Client app not found.');\n    }\n    options.test = options.test || clientApp.test;\n\n    const tsCfg = options.tsconfigFileName && options.tsconfigFileName.endsWith('.json')\n      ? options.tsconfigFileName : `${options.tsconfigFileName}.json`;\n    const testTsCfg = options.testTsconfigFileName && options.testTsconfigFileName.endsWith('.json')\n      ? options.testTsconfigFileName : `${options.testTsconfigFileName}.json`;\n\n    const serverApp: AppConfig = {\n      ...clientApp,\n      platform: 'server',\n      root: options.root,\n      outDir: options.outDir,\n      index: options.index,\n      main: options.main,\n      test: options.test,\n      tsconfig: tsCfg,\n      testTsconfig: testTsCfg,\n      polyfills: undefined,\n    };\n    if (options.name) {\n      serverApp.name = options.name;\n    }\n    if (!config.apps) {\n      config.apps = [];\n    }\n    config.apps.push(serverApp);\n\n    host.overwrite('/.angular-cli.json', JSON.stringify(config, null, 2));\n\n    return host;\n  };\n}\n\nfunction findBrowserModuleImport(host: Tree, modulePath: string): ts.Node {\n  const moduleBuffer = host.read(modulePath);\n  if (!moduleBuffer) {\n    throw new SchematicsException(`Module file (${modulePath}) not found`);\n  }\n  const moduleFileText = moduleBuffer.toString('utf-8');\n\n  const source = ts.createSourceFile(modulePath, moduleFileText, ts.ScriptTarget.Latest, true);\n\n  const decoratorMetadata = getDecoratorMetadata(source, 'NgModule', '@angular/core')[0];\n  const browserModuleNode = findNode(decoratorMetadata, ts.SyntaxKind.Identifier, 'BrowserModule');\n\n  if (browserModuleNode === null) {\n    throw new SchematicsException(`Cannot find BrowserModule import in ${modulePath}`);\n  }\n\n  return browserModuleNode;\n}\n\nfunction wrapBootstrapCall(options: UniversalOptions): Rule {\n  return (host: Tree) => {\n    const config = getConfig(host);\n    const clientApp = getAppFromConfig(config, options.clientApp || '0');\n    if (clientApp === null) {\n      throw new SchematicsException('Client app not found.');\n    }\n    const mainPath = normalize(`/${clientApp.root}/${clientApp.main}`);\n    let bootstrapCall: ts.Node | null = findBootstrapModuleCall(host, mainPath);\n    if (bootstrapCall === null) {\n      throw new SchematicsException('Bootstrap module not found.');\n    }\n\n    let bootstrapCallExpression: ts.Node | null = null;\n    let currentCall = bootstrapCall;\n    while (bootstrapCallExpression === null && currentCall.parent) {\n      currentCall = currentCall.parent;\n      if (currentCall.kind === ts.SyntaxKind.ExpressionStatement) {\n        bootstrapCallExpression = currentCall;\n      }\n    }\n    bootstrapCall = currentCall;\n\n    const recorder = host.beginUpdate(mainPath);\n    const beforeText = `document.addEventListener('DOMContentLoaded', () => {\\n  `;\n    const afterText = `\\n});`;\n    recorder.insertLeft(bootstrapCall.getStart(), beforeText);\n    recorder.insertRight(bootstrapCall.getEnd(), afterText);\n    host.commitUpdate(recorder);\n  };\n}\n\nfunction addServerTransition(options: UniversalOptions): Rule {\n  return (host: Tree) => {\n    const config = getConfig(host);\n    const clientApp = getAppFromConfig(config, options.clientApp || '0');\n    if (clientApp === null) {\n      throw new SchematicsException('Client app not found.');\n    }\n    const mainPath = normalize(`/${clientApp.root}/${clientApp.main}`);\n\n    const bootstrapModuleRelativePath = findBootstrapModulePath(host, mainPath);\n    const bootstrapModulePath = normalize(`/${clientApp.root}/${bootstrapModuleRelativePath}.ts`);\n\n    const browserModuleImport = findBrowserModuleImport(host, bootstrapModulePath);\n    const appId = options.appId;\n    const transitionCall = `.withServerTransition({ appId: '${appId}' })`;\n    const position = browserModuleImport.pos + browserModuleImport.getFullText().length;\n    const transitionCallChange = new InsertChange(\n      bootstrapModulePath, position, transitionCall);\n\n    const transitionCallRecorder = host.beginUpdate(bootstrapModulePath);\n    transitionCallRecorder.insertLeft(transitionCallChange.pos, transitionCallChange.toAdd);\n    host.commitUpdate(transitionCallRecorder);\n  };\n}\n\nfunction addDependencies(): Rule {\n  return (host: Tree) => {\n    const pkgPath = '/package.json';\n    const buffer = host.read(pkgPath);\n    if (buffer === null) {\n      throw new SchematicsException('Could not find package.json');\n    }\n\n    const pkg = JSON.parse(buffer.toString());\n\n    const ngCoreVersion = pkg.dependencies['@angular/core'];\n    pkg.dependencies['@angular/platform-server'] = ngCoreVersion;\n\n    host.overwrite(pkgPath, JSON.stringify(pkg, null, 2));\n\n    return host;\n  };\n}\n\nfunction updateGitignore(options: UniversalOptions): Rule {\n  return (host: Tree) => {\n    const ignorePath = normalize('/.gitignore');\n    const buffer = host.read(ignorePath);\n    if (buffer === null) {\n      // Assumption is made that there is no git repository.\n      return host;\n    } else {\n      const content = buffer.toString();\n      host.overwrite(ignorePath, `${content}\\n${options.outDir}`);\n    }\n\n    return host;\n  };\n}\n\nexport default function (options: UniversalOptions): Rule {\n  return (host: Tree, context: SchematicContext) => {\n    const templateSource = apply(url('./files'), [\n      template({\n        ...stringUtils,\n        ...options as object,\n        stripTsExtension: (s: string) => { return s.replace(/\\.ts$/, ''); },\n      }),\n    ]);\n\n    return chain([\n      mergeWith(templateSource),\n      addDependencies(),\n      updateConfigFile(options),\n      wrapBootstrapCall(options),\n      addServerTransition(options),\n      updateGitignore(options),\n    ])(host, context);\n  };\n}\n"]}