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