1 | ;
|
2 | /**
|
3 | * @license
|
4 | * Copyright Google LLC All Rights Reserved.
|
5 | *
|
6 | * Use of this source code is governed by an MIT-style license that can be
|
7 | * found in the LICENSE file at https://angular.io/license
|
8 | */
|
9 | Object.defineProperty(exports, "__esModule", { value: true });
|
10 | const core_1 = require("@angular-devkit/core");
|
11 | const schematics_1 = require("@angular-devkit/schematics");
|
12 | const utility_1 = require("@schematics/angular/utility");
|
13 | const json_file_1 = require("@schematics/angular/utility/json-file");
|
14 | const ng_ast_utils_1 = require("@schematics/angular/utility/ng-ast-utils");
|
15 | const project_targets_1 = require("@schematics/angular/utility/project-targets");
|
16 | const ts = require("typescript");
|
17 | const utils_1 = require("../utils");
|
18 | const SERVE_SSR_TARGET_NAME = 'serve-ssr';
|
19 | const PRERENDER_TARGET_NAME = 'prerender';
|
20 | function addScriptsRule(options) {
|
21 | return async (host) => {
|
22 | const pkgPath = '/package.json';
|
23 | const buffer = host.read(pkgPath);
|
24 | if (buffer === null) {
|
25 | throw new schematics_1.SchematicsException('Could not find package.json');
|
26 | }
|
27 | const serverDist = await (0, utils_1.getOutputPath)(host, options.project, 'server');
|
28 | const pkg = JSON.parse(buffer.toString());
|
29 | pkg.scripts = {
|
30 | ...pkg.scripts,
|
31 | 'dev:ssr': `ng run ${options.project}:${SERVE_SSR_TARGET_NAME}`,
|
32 | 'serve:ssr': `node ${serverDist}/main.js`,
|
33 | 'build:ssr': `ng build && ng run ${options.project}:server`,
|
34 | 'prerender': `ng run ${options.project}:${PRERENDER_TARGET_NAME}`,
|
35 | };
|
36 | host.overwrite(pkgPath, JSON.stringify(pkg, null, 2));
|
37 | };
|
38 | }
|
39 | function updateWorkspaceConfigRule(options) {
|
40 | return () => {
|
41 | return (0, utility_1.updateWorkspace)((workspace) => {
|
42 | const projectName = options.project;
|
43 | const project = workspace.projects.get(projectName);
|
44 | if (!project) {
|
45 | return;
|
46 | }
|
47 | const serverTarget = project.targets.get('server');
|
48 | serverTarget.options.main = (0, core_1.join)((0, core_1.normalize)(project.root), (0, utils_1.stripTsExtension)(options.serverFileName) + '.ts');
|
49 | const serveSSRTarget = project.targets.get(SERVE_SSR_TARGET_NAME);
|
50 | if (serveSSRTarget) {
|
51 | return;
|
52 | }
|
53 | project.targets.add({
|
54 | name: SERVE_SSR_TARGET_NAME,
|
55 | builder: '@nguniversal/builders:ssr-dev-server',
|
56 | defaultConfiguration: 'development',
|
57 | options: {},
|
58 | configurations: {
|
59 | development: {
|
60 | browserTarget: `${projectName}:build:development`,
|
61 | serverTarget: `${projectName}:server:development`,
|
62 | },
|
63 | production: {
|
64 | browserTarget: `${projectName}:build:production`,
|
65 | serverTarget: `${projectName}:server:production`,
|
66 | },
|
67 | },
|
68 | });
|
69 | const prerenderTarget = project.targets.get(PRERENDER_TARGET_NAME);
|
70 | if (prerenderTarget) {
|
71 | return;
|
72 | }
|
73 | project.targets.add({
|
74 | name: PRERENDER_TARGET_NAME,
|
75 | builder: '@nguniversal/builders:prerender',
|
76 | defaultConfiguration: 'production',
|
77 | options: {
|
78 | routes: ['/'],
|
79 | },
|
80 | configurations: {
|
81 | production: {
|
82 | browserTarget: `${projectName}:build:production`,
|
83 | serverTarget: `${projectName}:server:production`,
|
84 | },
|
85 | development: {
|
86 | browserTarget: `${projectName}:build:development`,
|
87 | serverTarget: `${projectName}:server:development`,
|
88 | },
|
89 | },
|
90 | });
|
91 | });
|
92 | };
|
93 | }
|
94 | function updateServerTsConfigRule(options) {
|
95 | return async (host) => {
|
96 | const project = await (0, utils_1.getProject)(host, options.project);
|
97 | const serverTarget = project.targets.get('server');
|
98 | if (!serverTarget || !serverTarget.options) {
|
99 | return;
|
100 | }
|
101 | const tsConfigPath = serverTarget.options.tsConfig;
|
102 | if (!tsConfigPath || typeof tsConfigPath !== 'string') {
|
103 | // No tsconfig path
|
104 | return;
|
105 | }
|
106 | const tsConfig = new json_file_1.JSONFile(host, tsConfigPath);
|
107 | const filesAstNode = tsConfig.get(['files']);
|
108 | const serverFilePath = (0, utils_1.stripTsExtension)(options.serverFileName) + '.ts';
|
109 | if (Array.isArray(filesAstNode) && !filesAstNode.some(({ text }) => text === serverFilePath)) {
|
110 | tsConfig.modify(['files'], [...filesAstNode, serverFilePath]);
|
111 | }
|
112 | };
|
113 | }
|
114 | function routingInitialNavigationRule(options) {
|
115 | return async (host) => {
|
116 | const project = await (0, utils_1.getProject)(host, options.project);
|
117 | const serverTarget = project.targets.get('server');
|
118 | if (!serverTarget || !serverTarget.options) {
|
119 | return;
|
120 | }
|
121 | const tsConfigPath = serverTarget.options.tsConfig;
|
122 | if (!tsConfigPath || typeof tsConfigPath !== 'string' || !host.exists(tsConfigPath)) {
|
123 | // No tsconfig path
|
124 | return;
|
125 | }
|
126 | const parseConfigHost = {
|
127 | useCaseSensitiveFileNames: ts.sys.useCaseSensitiveFileNames,
|
128 | readDirectory: ts.sys.readDirectory,
|
129 | fileExists: function (fileName) {
|
130 | return host.exists(fileName);
|
131 | },
|
132 | readFile: function (fileName) {
|
133 | return host.read(fileName).toString();
|
134 | },
|
135 | };
|
136 | const { config } = ts.readConfigFile(tsConfigPath, parseConfigHost.readFile);
|
137 | const parsed = ts.parseJsonConfigFileContent(config, parseConfigHost, (0, core_1.dirname)((0, core_1.normalize)(tsConfigPath)));
|
138 | const tsHost = ts.createCompilerHost(parsed.options, true);
|
139 | // Strip BOM as otherwise TSC methods (Ex: getWidth) will return an offset,
|
140 | // which breaks the CLI UpdateRecorder.
|
141 | // See: https://github.com/angular/angular/pull/30719
|
142 | tsHost.readFile = function (fileName) {
|
143 | return host
|
144 | .read(fileName)
|
145 | .toString()
|
146 | .replace(/^\uFEFF/, '');
|
147 | };
|
148 | tsHost.directoryExists = function (directoryName) {
|
149 | // When the path is file getDir will throw.
|
150 | try {
|
151 | const dir = host.getDir(directoryName);
|
152 | return !!(dir.subdirs.length || dir.subfiles.length);
|
153 | }
|
154 | catch {
|
155 | return false;
|
156 | }
|
157 | };
|
158 | tsHost.fileExists = function (fileName) {
|
159 | return host.exists(fileName);
|
160 | };
|
161 | tsHost.realpath = function (path) {
|
162 | return path;
|
163 | };
|
164 | tsHost.getCurrentDirectory = function () {
|
165 | return host.root.path;
|
166 | };
|
167 | const program = ts.createProgram(parsed.fileNames, parsed.options, tsHost);
|
168 | const typeChecker = program.getTypeChecker();
|
169 | const sourceFiles = program
|
170 | .getSourceFiles()
|
171 | .filter((f) => !f.isDeclarationFile && !program.isSourceFileFromExternalLibrary(f));
|
172 | const printer = ts.createPrinter();
|
173 | const routerModule = 'RouterModule';
|
174 | const routerSource = '@angular/router';
|
175 | sourceFiles.forEach((sourceFile) => {
|
176 | const routerImport = (0, utils_1.findImport)(sourceFile, routerSource, routerModule);
|
177 | if (!routerImport) {
|
178 | return;
|
179 | }
|
180 | let routerModuleNode;
|
181 | ts.forEachChild(sourceFile, function visitNode(node) {
|
182 | if (ts.isCallExpression(node) &&
|
183 | ts.isPropertyAccessExpression(node.expression) &&
|
184 | ts.isIdentifier(node.expression.expression) &&
|
185 | node.expression.name.text === 'forRoot') {
|
186 | const imp = (0, utils_1.getImportOfIdentifier)(typeChecker, node.expression.expression);
|
187 | if (imp && imp.name === routerModule && imp.importModule === routerSource) {
|
188 | routerModuleNode = node;
|
189 | }
|
190 | }
|
191 | ts.forEachChild(node, visitNode);
|
192 | });
|
193 | if (routerModuleNode) {
|
194 | const print = printer.printNode(ts.EmitHint.Unspecified, (0, utils_1.addInitialNavigation)(routerModuleNode), sourceFile);
|
195 | const recorder = host.beginUpdate(sourceFile.fileName);
|
196 | recorder.remove(routerModuleNode.getStart(), routerModuleNode.getWidth());
|
197 | recorder.insertRight(routerModuleNode.getStart(), print);
|
198 | host.commitUpdate(recorder);
|
199 | }
|
200 | });
|
201 | };
|
202 | }
|
203 | function addDependencies() {
|
204 | return (_host) => {
|
205 | return (0, schematics_1.chain)([
|
206 | (0, utility_1.addDependency)('@nguniversal/builders', '^16.0.2', {
|
207 | type: utility_1.DependencyType.Dev,
|
208 | }),
|
209 | (0, utility_1.addDependency)('@nguniversal/express-engine', '^16.0.2', {
|
210 | type: utility_1.DependencyType.Default,
|
211 | }),
|
212 | (0, utility_1.addDependency)('express', '^4.15.2', {
|
213 | type: utility_1.DependencyType.Default,
|
214 | }),
|
215 | (0, utility_1.addDependency)('@types/express', '^4.17.0', {
|
216 | type: utility_1.DependencyType.Dev,
|
217 | }),
|
218 | ]);
|
219 | };
|
220 | }
|
221 | function addServerFile(options, isStandalone) {
|
222 | return async (host) => {
|
223 | const project = await (0, utils_1.getProject)(host, options.project);
|
224 | const browserDistDirectory = await (0, utils_1.getOutputPath)(host, options.project, 'build');
|
225 | return (0, schematics_1.mergeWith)((0, schematics_1.apply)((0, schematics_1.url)('./files'), [
|
226 | (0, schematics_1.template)({
|
227 | ...core_1.strings,
|
228 | ...options,
|
229 | stripTsExtension: utils_1.stripTsExtension,
|
230 | browserDistDirectory,
|
231 | isStandalone,
|
232 | }),
|
233 | (0, schematics_1.move)(project.root),
|
234 | ]));
|
235 | };
|
236 | }
|
237 | function default_1(options) {
|
238 | return async (host) => {
|
239 | const project = await (0, utils_1.getProject)(host, options.project);
|
240 | const universalOptions = {
|
241 | ...options,
|
242 | skipInstall: true,
|
243 | };
|
244 | const clientBuildTarget = project.targets.get('build');
|
245 | if (!clientBuildTarget) {
|
246 | throw (0, project_targets_1.targetBuildNotFoundError)();
|
247 | }
|
248 | const clientBuildOptions = (clientBuildTarget.options ||
|
249 | {});
|
250 | const isStandalone = (0, ng_ast_utils_1.isStandaloneApp)(host, clientBuildOptions.main);
|
251 | delete universalOptions.serverFileName;
|
252 | delete universalOptions.serverPort;
|
253 | return (0, schematics_1.chain)([
|
254 | project.targets.has('server')
|
255 | ? (0, schematics_1.noop)()
|
256 | : (0, schematics_1.externalSchematic)('@schematics/angular', 'universal', universalOptions),
|
257 | addScriptsRule(options),
|
258 | updateServerTsConfigRule(options),
|
259 | updateWorkspaceConfigRule(options),
|
260 | isStandalone ? (0, schematics_1.noop)() : routingInitialNavigationRule(options),
|
261 | addServerFile(options, isStandalone),
|
262 | addDependencies(),
|
263 | ]);
|
264 | };
|
265 | }
|
266 | exports.default = default_1;
|
267 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../../modules/express-engine/schematics/install/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC 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 */\n\nimport { dirname, join, normalize, strings } from '@angular-devkit/core';\nimport {\n  Rule,\n  SchematicsException,\n  Tree,\n  apply,\n  chain,\n  externalSchematic,\n  mergeWith,\n  move,\n  noop,\n  template,\n  url,\n} from '@angular-devkit/schematics';\nimport { Schema as UniversalOptions } from '@schematics/angular/universal/schema';\nimport { DependencyType, addDependency, updateWorkspace } from '@schematics/angular/utility';\nimport { JSONFile } from '@schematics/angular/utility/json-file';\nimport { isStandaloneApp } from '@schematics/angular/utility/ng-ast-utils';\nimport { targetBuildNotFoundError } from '@schematics/angular/utility/project-targets';\nimport { BrowserBuilderOptions } from '@schematics/angular/utility/workspace-models';\nimport * as ts from 'typescript';\n\nimport {\n  addInitialNavigation,\n  findImport,\n  getImportOfIdentifier,\n  getOutputPath,\n  getProject,\n  stripTsExtension,\n} from '../utils';\n\nimport { Schema as AddUniversalOptions } from './schema';\n\nconst SERVE_SSR_TARGET_NAME = 'serve-ssr';\nconst PRERENDER_TARGET_NAME = 'prerender';\n\nfunction addScriptsRule(options: AddUniversalOptions): Rule {\n  return async (host) => {\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 serverDist = await getOutputPath(host, options.project, 'server');\n    const pkg = JSON.parse(buffer.toString()) as any;\n    pkg.scripts = {\n      ...pkg.scripts,\n      'dev:ssr': `ng run ${options.project}:${SERVE_SSR_TARGET_NAME}`,\n      'serve:ssr': `node ${serverDist}/main.js`,\n      'build:ssr': `ng build && ng run ${options.project}:server`,\n      'prerender': `ng run ${options.project}:${PRERENDER_TARGET_NAME}`,\n    };\n\n    host.overwrite(pkgPath, JSON.stringify(pkg, null, 2));\n  };\n}\n\nfunction updateWorkspaceConfigRule(options: AddUniversalOptions): Rule {\n  return () => {\n    return updateWorkspace((workspace) => {\n      const projectName = options.project;\n      const project = workspace.projects.get(projectName);\n      if (!project) {\n        return;\n      }\n\n      const serverTarget = project.targets.get('server');\n      serverTarget.options.main = join(\n        normalize(project.root),\n        stripTsExtension(options.serverFileName) + '.ts',\n      );\n\n      const serveSSRTarget = project.targets.get(SERVE_SSR_TARGET_NAME);\n      if (serveSSRTarget) {\n        return;\n      }\n\n      project.targets.add({\n        name: SERVE_SSR_TARGET_NAME,\n        builder: '@nguniversal/builders:ssr-dev-server',\n        defaultConfiguration: 'development',\n        options: {},\n        configurations: {\n          development: {\n            browserTarget: `${projectName}:build:development`,\n            serverTarget: `${projectName}:server:development`,\n          },\n          production: {\n            browserTarget: `${projectName}:build:production`,\n            serverTarget: `${projectName}:server:production`,\n          },\n        },\n      });\n\n      const prerenderTarget = project.targets.get(PRERENDER_TARGET_NAME);\n      if (prerenderTarget) {\n        return;\n      }\n\n      project.targets.add({\n        name: PRERENDER_TARGET_NAME,\n        builder: '@nguniversal/builders:prerender',\n        defaultConfiguration: 'production',\n        options: {\n          routes: ['/'],\n        },\n        configurations: {\n          production: {\n            browserTarget: `${projectName}:build:production`,\n            serverTarget: `${projectName}:server:production`,\n          },\n          development: {\n            browserTarget: `${projectName}:build:development`,\n            serverTarget: `${projectName}:server:development`,\n          },\n        },\n      });\n    });\n  };\n}\n\nfunction updateServerTsConfigRule(options: AddUniversalOptions): Rule {\n  return async (host) => {\n    const project = await getProject(host, options.project);\n    const serverTarget = project.targets.get('server');\n    if (!serverTarget || !serverTarget.options) {\n      return;\n    }\n\n    const tsConfigPath = serverTarget.options.tsConfig;\n    if (!tsConfigPath || typeof tsConfigPath !== 'string') {\n      // No tsconfig path\n      return;\n    }\n\n    const tsConfig = new JSONFile(host, tsConfigPath);\n    const filesAstNode = tsConfig.get(['files']);\n    const serverFilePath = stripTsExtension(options.serverFileName) + '.ts';\n    if (Array.isArray(filesAstNode) && !filesAstNode.some(({ text }) => text === serverFilePath)) {\n      tsConfig.modify(['files'], [...filesAstNode, serverFilePath]);\n    }\n  };\n}\n\nfunction routingInitialNavigationRule(options: UniversalOptions): Rule {\n  return async (host) => {\n    const project = await getProject(host, options.project);\n    const serverTarget = project.targets.get('server');\n    if (!serverTarget || !serverTarget.options) {\n      return;\n    }\n\n    const tsConfigPath = serverTarget.options.tsConfig;\n    if (!tsConfigPath || typeof tsConfigPath !== 'string' || !host.exists(tsConfigPath)) {\n      // No tsconfig path\n      return;\n    }\n\n    const parseConfigHost: ts.ParseConfigHost = {\n      useCaseSensitiveFileNames: ts.sys.useCaseSensitiveFileNames,\n      readDirectory: ts.sys.readDirectory,\n      fileExists: function (fileName: string): boolean {\n        return host.exists(fileName);\n      },\n      readFile: function (fileName: string): string {\n        return host.read(fileName).toString();\n      },\n    };\n    const { config } = ts.readConfigFile(tsConfigPath, parseConfigHost.readFile);\n    const parsed = ts.parseJsonConfigFileContent(\n      config,\n      parseConfigHost,\n      dirname(normalize(tsConfigPath)),\n    );\n    const tsHost = ts.createCompilerHost(parsed.options, true);\n    // Strip BOM as otherwise TSC methods (Ex: getWidth) will return an offset,\n    // which breaks the CLI UpdateRecorder.\n    // See: https://github.com/angular/angular/pull/30719\n    tsHost.readFile = function (fileName: string): string {\n      return host\n        .read(fileName)\n        .toString()\n        .replace(/^\\uFEFF/, '');\n    };\n    tsHost.directoryExists = function (directoryName: string): boolean {\n      // When the path is file getDir will throw.\n      try {\n        const dir = host.getDir(directoryName);\n\n        return !!(dir.subdirs.length || dir.subfiles.length);\n      } catch {\n        return false;\n      }\n    };\n    tsHost.fileExists = function (fileName: string): boolean {\n      return host.exists(fileName);\n    };\n    tsHost.realpath = function (path: string): string {\n      return path;\n    };\n    tsHost.getCurrentDirectory = function () {\n      return host.root.path;\n    };\n\n    const program = ts.createProgram(parsed.fileNames, parsed.options, tsHost);\n    const typeChecker = program.getTypeChecker();\n    const sourceFiles = program\n      .getSourceFiles()\n      .filter((f) => !f.isDeclarationFile && !program.isSourceFileFromExternalLibrary(f));\n    const printer = ts.createPrinter();\n    const routerModule = 'RouterModule';\n    const routerSource = '@angular/router';\n\n    sourceFiles.forEach((sourceFile) => {\n      const routerImport = findImport(sourceFile, routerSource, routerModule);\n      if (!routerImport) {\n        return;\n      }\n\n      let routerModuleNode: ts.CallExpression;\n      ts.forEachChild(sourceFile, function visitNode(node: ts.Node) {\n        if (\n          ts.isCallExpression(node) &&\n          ts.isPropertyAccessExpression(node.expression) &&\n          ts.isIdentifier(node.expression.expression) &&\n          node.expression.name.text === 'forRoot'\n        ) {\n          const imp = getImportOfIdentifier(typeChecker, node.expression.expression);\n\n          if (imp && imp.name === routerModule && imp.importModule === routerSource) {\n            routerModuleNode = node;\n          }\n        }\n\n        ts.forEachChild(node, visitNode);\n      });\n\n      if (routerModuleNode) {\n        const print = printer.printNode(\n          ts.EmitHint.Unspecified,\n          addInitialNavigation(routerModuleNode),\n          sourceFile,\n        );\n\n        const recorder = host.beginUpdate(sourceFile.fileName);\n        recorder.remove(routerModuleNode.getStart(), routerModuleNode.getWidth());\n        recorder.insertRight(routerModuleNode.getStart(), print);\n        host.commitUpdate(recorder);\n      }\n    });\n  };\n}\n\nfunction addDependencies(): Rule {\n  return (_host: Tree) => {\n    return chain([\n      addDependency('@nguniversal/builders', '^0.0.0-PLACEHOLDER', {\n        type: DependencyType.Dev,\n      }),\n      addDependency('@nguniversal/express-engine', '^0.0.0-PLACEHOLDER', {\n        type: DependencyType.Default,\n      }),\n      addDependency('express', 'EXPRESS_VERSION', {\n        type: DependencyType.Default,\n      }),\n      addDependency('@types/express', 'EXPRESS_TYPES_VERSION', {\n        type: DependencyType.Dev,\n      }),\n    ]);\n  };\n}\n\nfunction addServerFile(options: UniversalOptions, isStandalone: boolean): Rule {\n  return async (host) => {\n    const project = await getProject(host, options.project);\n    const browserDistDirectory = await getOutputPath(host, options.project, 'build');\n\n    return mergeWith(\n      apply(url('./files'), [\n        template({\n          ...strings,\n          ...options,\n          stripTsExtension,\n          browserDistDirectory,\n          isStandalone,\n        }),\n        move(project.root),\n      ]),\n    );\n  };\n}\n\nexport default function (options: AddUniversalOptions): Rule {\n  return async (host) => {\n    const project = await getProject(host, options.project);\n    const universalOptions = {\n      ...options,\n      skipInstall: true,\n    };\n    const clientBuildTarget = project.targets.get('build');\n    if (!clientBuildTarget) {\n      throw targetBuildNotFoundError();\n    }\n\n    const clientBuildOptions = (clientBuildTarget.options ||\n      {}) as unknown as BrowserBuilderOptions;\n\n    const isStandalone = isStandaloneApp(host, clientBuildOptions.main);\n\n    delete universalOptions.serverFileName;\n    delete universalOptions.serverPort;\n\n    return chain([\n      project.targets.has('server')\n        ? noop()\n        : externalSchematic('@schematics/angular', 'universal', universalOptions),\n      addScriptsRule(options),\n      updateServerTsConfigRule(options),\n      updateWorkspaceConfigRule(options),\n      isStandalone ? noop() : routingInitialNavigationRule(options),\n      addServerFile(options, isStandalone),\n      addDependencies(),\n    ]);\n  };\n}\n"]} |
\ | No newline at end of file |