@nf-beta/angular 0.0.1 → 0.0.2
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/LICENSE +8 -0
- package/README.md +458 -0
- package/builders.json +10 -0
- package/collection.json +27 -0
- package/generators.json +12 -0
- package/migration-collection.json +13 -0
- package/package.json +18 -8
- package/src/builders/build/builder.d.ts +6 -0
- package/src/builders/build/builder.d.ts.map +1 -0
- package/src/builders/build/builder.js +348 -0
- package/src/builders/build/federation-build-notifier.d.ts +70 -0
- package/src/builders/build/federation-build-notifier.d.ts.map +1 -0
- package/src/builders/build/federation-build-notifier.js +186 -0
- package/src/builders/build/schema.d.ts +21 -0
- package/src/builders/build/schema.json +84 -0
- package/src/config.d.ts +3 -0
- package/src/config.d.ts.map +1 -0
- package/src/config.js +2 -0
- package/src/generators/native-federation/files/src/index.ts__template__ +1 -0
- package/src/generators/native-federation/generator.d.ts +4 -0
- package/src/generators/native-federation/generator.d.ts.map +1 -0
- package/src/generators/native-federation/generator.js +43 -0
- package/src/generators/native-federation/schema.d.ts +5 -0
- package/src/generators/native-federation/schema.json +29 -0
- package/src/index.d.ts +2 -0
- package/src/index.d.ts.map +1 -0
- package/src/index.js +1 -0
- package/src/patch-angular-build.d.ts +2 -0
- package/src/patch-angular-build.d.ts.map +1 -0
- package/src/patch-angular-build.js +5 -0
- package/src/plugin/dev-externals-mixin.d.ts +3 -0
- package/src/plugin/dev-externals-mixin.d.ts.map +1 -0
- package/src/plugin/dev-externals-mixin.js +29 -0
- package/src/plugin/externals-skip-list.d.ts +3 -0
- package/src/plugin/externals-skip-list.d.ts.map +1 -0
- package/src/plugin/externals-skip-list.js +4 -0
- package/src/plugin/index.d.ts +4 -0
- package/src/plugin/index.d.ts.map +1 -0
- package/src/plugin/index.js +75 -0
- package/src/schematics/appbuilder/schema.d.ts +3 -0
- package/src/schematics/appbuilder/schema.json +17 -0
- package/src/schematics/appbuilder/schematic.d.ts +5 -0
- package/src/schematics/appbuilder/schematic.d.ts.map +1 -0
- package/src/schematics/appbuilder/schematic.js +83 -0
- package/src/schematics/init/files/federation.config.js__tmpl__ +33 -0
- package/src/schematics/init/schema.d.ts +6 -0
- package/src/schematics/init/schema.json +34 -0
- package/src/schematics/init/schematic.d.ts +7 -0
- package/src/schematics/init/schematic.d.ts.map +1 -0
- package/src/schematics/init/schematic.js +422 -0
- package/src/schematics/remove/schema.d.ts +3 -0
- package/src/schematics/remove/schema.json +17 -0
- package/src/schematics/remove/schematic.d.ts +5 -0
- package/src/schematics/remove/schematic.d.ts.map +1 -0
- package/src/schematics/remove/schematic.js +109 -0
- package/src/schematics/update18/schema.json +7 -0
- package/src/schematics/update18/schematic.d.ts +3 -0
- package/src/schematics/update18/schematic.d.ts.map +1 -0
- package/src/schematics/update18/schematic.js +7 -0
- package/src/tools/fstart-as-data-url.d.ts +2 -0
- package/src/tools/fstart-as-data-url.d.ts.map +1 -0
- package/src/tools/fstart-as-data-url.js +1 -0
- package/src/utils/angular-esbuild-adapter.d.ts +10 -0
- package/src/utils/angular-esbuild-adapter.d.ts.map +1 -0
- package/src/utils/angular-esbuild-adapter.js +289 -0
- package/src/utils/angular-locales.d.ts +19 -0
- package/src/utils/angular-locales.d.ts.map +1 -0
- package/src/utils/angular-locales.js +18 -0
- package/src/utils/create-awaitable-compiler-plugin.d.ts +6 -0
- package/src/utils/create-awaitable-compiler-plugin.d.ts.map +1 -0
- package/src/utils/create-awaitable-compiler-plugin.js +29 -0
- package/src/utils/create-compiler-options.d.ts +5 -0
- package/src/utils/create-compiler-options.d.ts.map +1 -0
- package/src/utils/create-compiler-options.js +42 -0
- package/src/utils/event-source.d.ts +10 -0
- package/src/utils/event-source.d.ts.map +1 -0
- package/src/utils/event-source.js +10 -0
- package/src/utils/i18n.d.ts +23 -0
- package/src/utils/i18n.d.ts.map +1 -0
- package/src/utils/i18n.js +61 -0
- package/src/utils/mem-resuts.d.ts +29 -0
- package/src/utils/mem-resuts.d.ts.map +1 -0
- package/src/utils/mem-resuts.js +50 -0
- package/src/utils/patch-angular-build.d.ts +4 -0
- package/src/utils/patch-angular-build.d.ts.map +1 -0
- package/src/utils/patch-angular-build.js +29 -0
- package/src/utils/rebuild-events.d.ts +8 -0
- package/src/utils/rebuild-events.d.ts.map +1 -0
- package/src/utils/rebuild-events.js +4 -0
- package/src/utils/shared-mappings-plugin.d.ts +4 -0
- package/src/utils/shared-mappings-plugin.d.ts.map +1 -0
- package/src/utils/shared-mappings-plugin.js +28 -0
- package/src/utils/updateIndexHtml.d.ts +5 -0
- package/src/utils/updateIndexHtml.d.ts.map +1 -0
- package/src/utils/updateIndexHtml.js +34 -0
|
@@ -0,0 +1,422 @@
|
|
|
1
|
+
import { apply, chain, mergeWith, move, noop, template, url, } from '@angular-devkit/schematics';
|
|
2
|
+
import { strings } from '@angular-devkit/core';
|
|
3
|
+
import { NodePackageInstallTask } from '@angular-devkit/schematics/tasks';
|
|
4
|
+
import { patchAngularBuildPackageJson, privateEntrySrc } from '../../utils/patch-angular-build.js';
|
|
5
|
+
import { addPackageJsonDependency, getPackageJsonDependency, NodeDependencyType, } from '@schematics/angular/utility/dependencies';
|
|
6
|
+
import * as path from 'path';
|
|
7
|
+
const SSR_VERSION = '^3.3.4';
|
|
8
|
+
export function updatePackageJson(tree) {
|
|
9
|
+
const packageJson = tree.readJson('package.json') ?? {};
|
|
10
|
+
const scriptCall = 'node node_modules/@angular-architects/native-federation/src/patch-angular-build.js';
|
|
11
|
+
if (!packageJson?.['scripts']) {
|
|
12
|
+
packageJson['scripts'] = {};
|
|
13
|
+
}
|
|
14
|
+
let postInstall = (packageJson['scripts']?.['postinstall'] || '');
|
|
15
|
+
if (!postInstall) {
|
|
16
|
+
return;
|
|
17
|
+
}
|
|
18
|
+
if (postInstall.includes(scriptCall)) {
|
|
19
|
+
postInstall = postInstall.replace(scriptCall, '');
|
|
20
|
+
}
|
|
21
|
+
if (postInstall.endsWith(' && ')) {
|
|
22
|
+
postInstall = postInstall.substring(0, postInstall.length - 4);
|
|
23
|
+
}
|
|
24
|
+
packageJson['scripts']['postinstall'] = postInstall;
|
|
25
|
+
tree.overwrite('package.json', JSON.stringify(packageJson, null, 2));
|
|
26
|
+
}
|
|
27
|
+
export default function config(options) {
|
|
28
|
+
return async function (tree, context) {
|
|
29
|
+
const workspaceFileName = getWorkspaceFileName(tree);
|
|
30
|
+
const workspace = JSON.parse(tree.read(workspaceFileName)?.toString('utf8') ?? '{}');
|
|
31
|
+
const normalized = normalizeOptions(options, workspace, tree);
|
|
32
|
+
const { polyfills, projectName, projectRoot, projectSourceRoot, manifestPath, manifestRelPath, main, } = normalized;
|
|
33
|
+
updatePolyfills(tree, polyfills);
|
|
34
|
+
const remoteMap = await generateRemoteMap(workspace, projectName);
|
|
35
|
+
if (options.type === 'dynamic-host' && !tree.exists(manifestPath)) {
|
|
36
|
+
tree.create(manifestPath, JSON.stringify(remoteMap, null, '\t'));
|
|
37
|
+
}
|
|
38
|
+
const federationConfigPath = path.join(projectRoot, 'federation.config.js');
|
|
39
|
+
const exists = tree.exists(federationConfigPath);
|
|
40
|
+
const cand1 = path.join(projectSourceRoot, 'app', 'app.component.ts').replace(/\\/g, '/');
|
|
41
|
+
const cand2 = path.join(projectSourceRoot, 'app', 'app.ts').replace(/\\/g, '/');
|
|
42
|
+
const appComponent = tree.exists(cand1) ? cand1 : tree.exists(cand2) ? cand2 : 'update-this.ts';
|
|
43
|
+
const generateRule = !exists
|
|
44
|
+
? await generateFederationConfig(remoteMap, projectRoot, projectSourceRoot, appComponent, options)
|
|
45
|
+
: noop;
|
|
46
|
+
const ssr = isSsrProject(normalized);
|
|
47
|
+
const server = ssr ? getSsrFilePath(normalized) : '';
|
|
48
|
+
if (ssr) {
|
|
49
|
+
console.log('SSR detected ...');
|
|
50
|
+
console.log('Activating CORS ...');
|
|
51
|
+
addPackageJsonDependency(tree, {
|
|
52
|
+
name: 'cors',
|
|
53
|
+
type: NodeDependencyType.Default,
|
|
54
|
+
version: '^2.8.5',
|
|
55
|
+
overwrite: false,
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
updateWorkspaceConfig(tree, normalized, workspace, workspaceFileName, ssr);
|
|
59
|
+
// updatePackageJson(tree);
|
|
60
|
+
// patchAngularBuild(tree);
|
|
61
|
+
addPackageJsonDependency(tree, {
|
|
62
|
+
name: '@angular/animations',
|
|
63
|
+
type: NodeDependencyType.Default,
|
|
64
|
+
version: getPackageJsonDependency(tree, '@angular/core')?.version || 'latest',
|
|
65
|
+
overwrite: false,
|
|
66
|
+
});
|
|
67
|
+
addPackageJsonDependency(tree, {
|
|
68
|
+
name: '@angular-devkit/build-angular',
|
|
69
|
+
type: NodeDependencyType.Dev,
|
|
70
|
+
version: getPackageJsonDependency(tree, '@angular/build')?.version || 'latest',
|
|
71
|
+
overwrite: false,
|
|
72
|
+
});
|
|
73
|
+
addPackageJsonDependency(tree, {
|
|
74
|
+
name: 'es-module-shims',
|
|
75
|
+
type: NodeDependencyType.Default,
|
|
76
|
+
version: '^1.5.12',
|
|
77
|
+
overwrite: false,
|
|
78
|
+
});
|
|
79
|
+
addPackageJsonDependency(tree, {
|
|
80
|
+
name: '@nf-beta/core-node',
|
|
81
|
+
type: NodeDependencyType.Default,
|
|
82
|
+
version: SSR_VERSION,
|
|
83
|
+
overwrite: true,
|
|
84
|
+
});
|
|
85
|
+
context.addTask(new NodePackageInstallTask());
|
|
86
|
+
return chain([
|
|
87
|
+
generateRule,
|
|
88
|
+
makeMainAsync(main, options, remoteMap, manifestRelPath),
|
|
89
|
+
ssr ? makeServerAsync(server, options, remoteMap) : noop(),
|
|
90
|
+
]);
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
function isSsrProject(normalized) {
|
|
94
|
+
return !!normalized.projectConfig?.architect?.build.options?.ssr;
|
|
95
|
+
}
|
|
96
|
+
function getSsrFilePath(normalized) {
|
|
97
|
+
return normalized.projectConfig.architect.build.options.ssr.entry;
|
|
98
|
+
}
|
|
99
|
+
export function patchAngularBuild(tree) {
|
|
100
|
+
const packagePath = 'node_modules/@angular/build/package.json';
|
|
101
|
+
const privatePath = 'node_modules/@angular/build/private.js';
|
|
102
|
+
if (!tree.exists(packagePath)) {
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
const packageJson = JSON.parse(tree.read(packagePath)?.toString('utf8') ?? '{}');
|
|
106
|
+
patchAngularBuildPackageJson(packageJson);
|
|
107
|
+
tree.overwrite(packagePath, JSON.stringify(packageJson, null, 2));
|
|
108
|
+
if (!tree.exists(privatePath)) {
|
|
109
|
+
tree.create(privatePath, privateEntrySrc);
|
|
110
|
+
}
|
|
111
|
+
else {
|
|
112
|
+
tree.overwrite(privatePath, privateEntrySrc);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
function updateWorkspaceConfig(tree, options, workspace, workspaceFileName, ssr) {
|
|
116
|
+
const { projectConfig, projectName, port } = options;
|
|
117
|
+
if (!projectConfig?.architect?.build || !projectConfig?.architect?.serve) {
|
|
118
|
+
throw new Error(`The project doesn't have a build or serve target in angular.json!`);
|
|
119
|
+
}
|
|
120
|
+
const originalBuild = projectConfig.architect.build;
|
|
121
|
+
if (originalBuild.builder !== '@angular-devkit/build-angular:application' ||
|
|
122
|
+
originalBuild.builder !== '@angular/build:application') {
|
|
123
|
+
console.log('Switching project to the application builder using esbuild ...');
|
|
124
|
+
originalBuild.builder = '@angular/build:application';
|
|
125
|
+
delete originalBuild.configurations?.development?.buildOptimizer;
|
|
126
|
+
delete originalBuild.configurations?.development?.vendorChunk;
|
|
127
|
+
}
|
|
128
|
+
if (originalBuild.options.main) {
|
|
129
|
+
const main = originalBuild.options.main;
|
|
130
|
+
delete originalBuild.options.main;
|
|
131
|
+
originalBuild.options.browser = main;
|
|
132
|
+
}
|
|
133
|
+
delete originalBuild.options.commonChunk;
|
|
134
|
+
projectConfig.architect.esbuild = originalBuild;
|
|
135
|
+
projectConfig.architect.build = {
|
|
136
|
+
builder: '@angular-architects/native-federation:build',
|
|
137
|
+
options: {},
|
|
138
|
+
configurations: {
|
|
139
|
+
production: {
|
|
140
|
+
target: `${projectName}:esbuild:production`,
|
|
141
|
+
},
|
|
142
|
+
development: {
|
|
143
|
+
target: `${projectName}:esbuild:development`,
|
|
144
|
+
dev: true,
|
|
145
|
+
},
|
|
146
|
+
},
|
|
147
|
+
defaultConfiguration: 'production',
|
|
148
|
+
};
|
|
149
|
+
if (ssr) {
|
|
150
|
+
projectConfig.architect.build.options.ssr = true;
|
|
151
|
+
// projectConfig.architect.esbuild.options.prerender = false;
|
|
152
|
+
}
|
|
153
|
+
const serve = projectConfig.architect.serve;
|
|
154
|
+
serve.options ??= {};
|
|
155
|
+
serve.options.port = port;
|
|
156
|
+
delete serve.options.commonChunk;
|
|
157
|
+
const serveProd = projectConfig.architect.serve.configurations?.production;
|
|
158
|
+
if (serveProd) {
|
|
159
|
+
serveProd.buildTarget = `${projectName}:esbuild:production`;
|
|
160
|
+
delete serveProd.browserTarget;
|
|
161
|
+
}
|
|
162
|
+
const serveDev = projectConfig.architect.serve.configurations?.development;
|
|
163
|
+
if (serveDev) {
|
|
164
|
+
serveDev.buildTarget = `${projectName}:esbuild:development`;
|
|
165
|
+
delete serveDev.browserTarget;
|
|
166
|
+
}
|
|
167
|
+
projectConfig.architect['serve-original'] = projectConfig.architect.serve;
|
|
168
|
+
projectConfig.architect.serve = {
|
|
169
|
+
builder: '@angular-architects/native-federation:build',
|
|
170
|
+
options: {
|
|
171
|
+
target: `${projectName}:serve-original:development`,
|
|
172
|
+
rebuildDelay: 500,
|
|
173
|
+
dev: true,
|
|
174
|
+
cacheExternalArtifacts: false,
|
|
175
|
+
port: 0,
|
|
176
|
+
},
|
|
177
|
+
};
|
|
178
|
+
const serveSsr = projectConfig.architect['serve-ssr'];
|
|
179
|
+
if (serveSsr && !serveSsr.options) {
|
|
180
|
+
serveSsr.options = {};
|
|
181
|
+
}
|
|
182
|
+
if (serveSsr) {
|
|
183
|
+
serveSsr.options.port = port;
|
|
184
|
+
}
|
|
185
|
+
// projectConfig.architect.serve.builder = serveBuilder;
|
|
186
|
+
// TODO: Register further builders when ready
|
|
187
|
+
tree.overwrite(workspaceFileName, JSON.stringify(workspace, null, '\t'));
|
|
188
|
+
}
|
|
189
|
+
function normalizeOptions(options, workspace, tree) {
|
|
190
|
+
if (!options.project) {
|
|
191
|
+
options.project = workspace.defaultProject;
|
|
192
|
+
}
|
|
193
|
+
const projects = Object.keys(workspace.projects);
|
|
194
|
+
if (!options.project && projects.length === 0) {
|
|
195
|
+
throw new Error(`No default project found. Please specifiy a project name!`);
|
|
196
|
+
}
|
|
197
|
+
if (!options.project) {
|
|
198
|
+
console.log('Using first configured project as default project: ' + projects[0]);
|
|
199
|
+
options.project = projects[0];
|
|
200
|
+
}
|
|
201
|
+
const projectName = options.project;
|
|
202
|
+
const projectConfig = workspace.projects[projectName];
|
|
203
|
+
if (!projectConfig) {
|
|
204
|
+
throw new Error(`Project ${projectName} not found in angular.json.`);
|
|
205
|
+
}
|
|
206
|
+
const projectRoot = projectConfig.root?.replace(/\\/g, '/');
|
|
207
|
+
const projectSourceRoot = projectConfig.sourceRoot?.replace(/\\/g, '/');
|
|
208
|
+
const publicPath = path.join(projectRoot, 'public').replace(/\\/g, '/');
|
|
209
|
+
let manifestPath = path.join(publicPath, 'federation.manifest.json').replace(/\\/g, '/');
|
|
210
|
+
let manifestRelPath = 'federation.manifest.json';
|
|
211
|
+
const hasPublicFolder = tree
|
|
212
|
+
.getDir(projectRoot)
|
|
213
|
+
.subdirs.map(p => String(p))
|
|
214
|
+
.includes('public');
|
|
215
|
+
if (!hasPublicFolder) {
|
|
216
|
+
manifestPath = path
|
|
217
|
+
.join(projectRoot, 'src/assets/federation.manifest.json')
|
|
218
|
+
.replace(/\\/g, '/');
|
|
219
|
+
manifestRelPath = 'assets/federation.manifest.json';
|
|
220
|
+
}
|
|
221
|
+
const main = projectConfig.architect.build.options.main || projectConfig.architect.build.options.browser;
|
|
222
|
+
if (!projectConfig.architect.build.options.polyfills) {
|
|
223
|
+
projectConfig.architect.build.options.polyfills = [];
|
|
224
|
+
}
|
|
225
|
+
if (typeof projectConfig.architect.build.options.polyfills === 'string') {
|
|
226
|
+
projectConfig.architect.build.options.polyfills = [
|
|
227
|
+
projectConfig.architect.build.options.polyfills,
|
|
228
|
+
];
|
|
229
|
+
}
|
|
230
|
+
const polyfills = projectConfig.architect.build.options.polyfills;
|
|
231
|
+
return {
|
|
232
|
+
polyfills,
|
|
233
|
+
projectName,
|
|
234
|
+
projectRoot,
|
|
235
|
+
projectSourceRoot,
|
|
236
|
+
manifestPath,
|
|
237
|
+
manifestRelPath,
|
|
238
|
+
projectConfig,
|
|
239
|
+
main,
|
|
240
|
+
port: +(options.port || 4200),
|
|
241
|
+
};
|
|
242
|
+
}
|
|
243
|
+
function updatePolyfills(tree, polyfills) {
|
|
244
|
+
if (typeof polyfills === 'string') {
|
|
245
|
+
updatePolyfillsFile(tree, polyfills);
|
|
246
|
+
}
|
|
247
|
+
else {
|
|
248
|
+
updatePolyfillsArray(tree, polyfills);
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
function updatePolyfillsFile(tree, polyfills) {
|
|
252
|
+
let polyfillsContent = tree.readText(polyfills);
|
|
253
|
+
if (!polyfillsContent.includes('es-module-shims')) {
|
|
254
|
+
polyfillsContent += `\nimport 'es-module-shims';\n`;
|
|
255
|
+
tree.overwrite(polyfills, polyfillsContent);
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
function updatePolyfillsArray(_tree, polyfills) {
|
|
259
|
+
const polyfillsConfig = polyfills;
|
|
260
|
+
if (!polyfillsConfig.includes('es-module-shims')) {
|
|
261
|
+
polyfillsConfig.push('es-module-shims');
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
function generateRemoteMap(workspace, projectName) {
|
|
265
|
+
const result = {};
|
|
266
|
+
for (const p in workspace.projects) {
|
|
267
|
+
const project = workspace.projects[p];
|
|
268
|
+
const projectType = project.projectType ?? 'application';
|
|
269
|
+
if (p !== projectName &&
|
|
270
|
+
projectType === 'application' &&
|
|
271
|
+
project?.architect?.serve &&
|
|
272
|
+
project?.architect?.build) {
|
|
273
|
+
const pPort = project.architect['serve-original']?.options?.port ??
|
|
274
|
+
project.architect.serve?.options?.port ??
|
|
275
|
+
4200;
|
|
276
|
+
result[strings.camelize(p)] = `http://localhost:${pPort}/remoteEntry.json`;
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
if (Object.keys(result).length === 0) {
|
|
280
|
+
result['mfe1'] = `http://localhost:3000/remoteEntry.json`;
|
|
281
|
+
}
|
|
282
|
+
return result;
|
|
283
|
+
}
|
|
284
|
+
function makeMainAsync(main, options, remoteMap, manifestRelPath) {
|
|
285
|
+
return async function (tree) {
|
|
286
|
+
const mainPath = path.dirname(main);
|
|
287
|
+
const bootstrapName = path.join(mainPath, 'bootstrap.ts');
|
|
288
|
+
if (tree.exists(bootstrapName)) {
|
|
289
|
+
console.info(`${bootstrapName} already exists.`);
|
|
290
|
+
return;
|
|
291
|
+
}
|
|
292
|
+
const mainContent = tree.read(main);
|
|
293
|
+
if (mainContent)
|
|
294
|
+
tree.create(bootstrapName, mainContent);
|
|
295
|
+
let newMainContent = '';
|
|
296
|
+
if (options.type === 'dynamic-host') {
|
|
297
|
+
newMainContent = `import { initFederation } from '@angular-architects/native-federation';
|
|
298
|
+
|
|
299
|
+
initFederation('${manifestRelPath}')
|
|
300
|
+
.catch(err => console.error(err))
|
|
301
|
+
.then(_ => import('./bootstrap'))
|
|
302
|
+
.catch(err => console.error(err));
|
|
303
|
+
`;
|
|
304
|
+
}
|
|
305
|
+
else if (options.type === 'host') {
|
|
306
|
+
const manifest = JSON.stringify(remoteMap, null, 2).replace(/"/g, "'");
|
|
307
|
+
newMainContent = `import { initFederation } from '@angular-architects/native-federation';
|
|
308
|
+
|
|
309
|
+
initFederation(${manifest})
|
|
310
|
+
.catch(err => console.error(err))
|
|
311
|
+
.then(_ => import('./bootstrap'))
|
|
312
|
+
.catch(err => console.error(err));
|
|
313
|
+
`;
|
|
314
|
+
}
|
|
315
|
+
else {
|
|
316
|
+
newMainContent = `import { initFederation } from '@angular-architects/native-federation';
|
|
317
|
+
|
|
318
|
+
initFederation()
|
|
319
|
+
.catch(err => console.error(err))
|
|
320
|
+
.then(_ => import('./bootstrap'))
|
|
321
|
+
.catch(err => console.error(err));
|
|
322
|
+
`;
|
|
323
|
+
}
|
|
324
|
+
tree.overwrite(main, newMainContent);
|
|
325
|
+
};
|
|
326
|
+
}
|
|
327
|
+
function makeServerAsync(server, options, remoteMap) {
|
|
328
|
+
return async function (tree) {
|
|
329
|
+
const mainPath = path.dirname(server);
|
|
330
|
+
const bootstrapName = path.join(mainPath, 'bootstrap-server.ts');
|
|
331
|
+
if (tree.exists(bootstrapName)) {
|
|
332
|
+
console.info(`${bootstrapName} already exists.`);
|
|
333
|
+
return;
|
|
334
|
+
}
|
|
335
|
+
const cors = `import { createRequire } from "module";
|
|
336
|
+
const require = createRequire(import.meta.url);
|
|
337
|
+
const cors = require("cors");
|
|
338
|
+
`;
|
|
339
|
+
const mainContent = tree.read(server)?.toString('utf8');
|
|
340
|
+
const updatedContent = (cors + mainContent)
|
|
341
|
+
.replace(`const port = process.env['PORT'] || 4000`, `const port = process.env['PORT'] || ${options.port || 4000}`)
|
|
342
|
+
.replace(`const app = express();`, `const app = express();\n\tapp.use(cors());\n app.set('view engine', 'html');`)
|
|
343
|
+
.replace(`if (isMainModule(import.meta.url)) {`, ``)
|
|
344
|
+
.replace(/\}(?![\s\S]*\})/, '');
|
|
345
|
+
tree.create(bootstrapName, updatedContent);
|
|
346
|
+
let newMainContent = '';
|
|
347
|
+
if (options.type === 'dynamic-host') {
|
|
348
|
+
newMainContent = `import { initNodeFederation } from '@nf-beta/core-node';
|
|
349
|
+
|
|
350
|
+
console.log('Starting SSR for Shell');
|
|
351
|
+
|
|
352
|
+
(async () => {
|
|
353
|
+
|
|
354
|
+
await initNodeFederation({
|
|
355
|
+
remotesOrManifestUrl: '../browser/federation.manifest.json',
|
|
356
|
+
relBundlePath: '../browser/',
|
|
357
|
+
});
|
|
358
|
+
|
|
359
|
+
await import('./bootstrap-server');
|
|
360
|
+
|
|
361
|
+
})();
|
|
362
|
+
`;
|
|
363
|
+
}
|
|
364
|
+
else if (options.type === 'host') {
|
|
365
|
+
const manifest = JSON.stringify(remoteMap, null, 2).replace(/"/g, "'");
|
|
366
|
+
newMainContent = `import { initNodeFederation } from '@nf-beta/core-node';
|
|
367
|
+
|
|
368
|
+
console.log('Starting SSR for Shell');
|
|
369
|
+
|
|
370
|
+
(async () => {
|
|
371
|
+
|
|
372
|
+
await initNodeFederation({
|
|
373
|
+
remotesOrManifestUrl: ${manifest},
|
|
374
|
+
relBundlePath: '../browser/',
|
|
375
|
+
});
|
|
376
|
+
|
|
377
|
+
await import('./bootstrap-server');
|
|
378
|
+
|
|
379
|
+
})();
|
|
380
|
+
`;
|
|
381
|
+
}
|
|
382
|
+
else {
|
|
383
|
+
newMainContent = `import { initNodeFederation } from '@nf-beta/core-node';
|
|
384
|
+
|
|
385
|
+
(async () => {
|
|
386
|
+
|
|
387
|
+
await initNodeFederation({
|
|
388
|
+
relBundlePath: '../browser/'
|
|
389
|
+
});
|
|
390
|
+
|
|
391
|
+
await import('./bootstrap-server');
|
|
392
|
+
|
|
393
|
+
})();
|
|
394
|
+
`;
|
|
395
|
+
}
|
|
396
|
+
tree.overwrite(server, newMainContent);
|
|
397
|
+
};
|
|
398
|
+
}
|
|
399
|
+
export function getWorkspaceFileName(tree) {
|
|
400
|
+
if (tree.exists('angular.json')) {
|
|
401
|
+
return 'angular.json';
|
|
402
|
+
}
|
|
403
|
+
if (tree.exists('workspace.json')) {
|
|
404
|
+
return 'workspace.json';
|
|
405
|
+
}
|
|
406
|
+
throw new Error("angular.json or workspace.json expected! Did you call this in your project's root?");
|
|
407
|
+
}
|
|
408
|
+
async function generateFederationConfig(remoteMap, projectRoot, projectSourceRoot, appComponentPath, options) {
|
|
409
|
+
const tmpl = url('./files');
|
|
410
|
+
const applied = apply(tmpl, [
|
|
411
|
+
template({
|
|
412
|
+
projectRoot,
|
|
413
|
+
projectSourceRoot,
|
|
414
|
+
appComponentPath,
|
|
415
|
+
remoteMap,
|
|
416
|
+
...options,
|
|
417
|
+
tmpl: '',
|
|
418
|
+
}),
|
|
419
|
+
move(projectRoot),
|
|
420
|
+
]);
|
|
421
|
+
return mergeWith(applied);
|
|
422
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
{
|
|
2
|
+
"$schema": "http://json-schema.org/schema",
|
|
3
|
+
"$id": "nf",
|
|
4
|
+
"title": "",
|
|
5
|
+
"type": "object",
|
|
6
|
+
"properties": {
|
|
7
|
+
"project": {
|
|
8
|
+
"type": "string",
|
|
9
|
+
"description": "The project to remove native federation from",
|
|
10
|
+
"$default": {
|
|
11
|
+
"$source": "argv",
|
|
12
|
+
"index": 0
|
|
13
|
+
},
|
|
14
|
+
"x-prompt": "Project name (press enter for default project)"
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import type { Rule, Tree } from '@angular-devkit/schematics';
|
|
2
|
+
import type { NfSchematicSchema } from './schema.js';
|
|
3
|
+
export default function remove(options: NfSchematicSchema): Rule;
|
|
4
|
+
export declare function getWorkspaceFileName(tree: Tree): string;
|
|
5
|
+
//# sourceMappingURL=schematic.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"schematic.d.ts","sourceRoot":"","sources":["../../../../src/schematics/remove/schematic.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,IAAI,EAAE,MAAM,4BAA4B,CAAC;AAE7D,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAcrD,MAAM,CAAC,OAAO,UAAU,MAAM,CAAC,OAAO,EAAE,iBAAiB,GAAG,IAAI,CAgB/D;AAmHD,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,IAAI,GAAG,MAAM,CAUvD"}
|
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
import * as path from 'path';
|
|
2
|
+
export default function remove(options) {
|
|
3
|
+
return async function (tree) {
|
|
4
|
+
const workspaceFileName = getWorkspaceFileName(tree);
|
|
5
|
+
const workspace = JSON.parse(tree.read(workspaceFileName)?.toString('utf8') ?? '{}');
|
|
6
|
+
const normalized = normalizeOptions(options, workspace);
|
|
7
|
+
const { polyfills, projectRoot } = normalized;
|
|
8
|
+
const bootstrapPath = path.join(projectRoot, 'src/bootstrap.ts');
|
|
9
|
+
const mainPath = path.join(projectRoot, 'src/main.ts');
|
|
10
|
+
makeMainSync(tree, bootstrapPath, mainPath);
|
|
11
|
+
updatePolyfills(tree, polyfills);
|
|
12
|
+
updateWorkspaceConfig(tree, normalized, workspace, workspaceFileName);
|
|
13
|
+
};
|
|
14
|
+
}
|
|
15
|
+
function makeMainSync(tree, bootstrapPath, mainPath) {
|
|
16
|
+
if (tree.exists(bootstrapPath) && tree.exists(mainPath)) {
|
|
17
|
+
tree.delete(mainPath);
|
|
18
|
+
tree.rename(bootstrapPath, mainPath);
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
function updateWorkspaceConfig(tree, options, workspace, workspaceFileName) {
|
|
22
|
+
const { projectConfig } = options;
|
|
23
|
+
if (!projectConfig?.architect?.build || !projectConfig?.architect?.serve) {
|
|
24
|
+
throw new Error(`The project doesn't have a build or serve target in angular.json!`);
|
|
25
|
+
}
|
|
26
|
+
if (projectConfig.architect.esbuild) {
|
|
27
|
+
projectConfig.architect.build = projectConfig.architect.esbuild;
|
|
28
|
+
delete projectConfig.architect.esbuild;
|
|
29
|
+
}
|
|
30
|
+
if (projectConfig.architect['serve-original']) {
|
|
31
|
+
projectConfig.architect.serve = projectConfig.architect['serve-original'];
|
|
32
|
+
delete projectConfig.architect['serve-original'];
|
|
33
|
+
}
|
|
34
|
+
if (projectConfig.architect.serve) {
|
|
35
|
+
const conf = projectConfig.architect.serve.configurations;
|
|
36
|
+
conf.production.buildTarget = conf.production.buildTarget.replace(':esbuild:', ':build:');
|
|
37
|
+
conf.development.buildTarget = conf.development.buildTarget.replace(':esbuild:', ':build:');
|
|
38
|
+
}
|
|
39
|
+
tree.overwrite(workspaceFileName, JSON.stringify(workspace, null, '\t'));
|
|
40
|
+
}
|
|
41
|
+
function normalizeOptions(options, workspace) {
|
|
42
|
+
if (!options.project) {
|
|
43
|
+
options.project = workspace.defaultProject;
|
|
44
|
+
}
|
|
45
|
+
const projects = Object.keys(workspace.projects);
|
|
46
|
+
if (!options.project && projects.length === 0) {
|
|
47
|
+
throw new Error(`No default project found. Please specifiy a project name!`);
|
|
48
|
+
}
|
|
49
|
+
if (!options.project) {
|
|
50
|
+
console.log('Using first configured project as default project: ' + projects[0]);
|
|
51
|
+
options.project = projects[0];
|
|
52
|
+
}
|
|
53
|
+
const projectName = options.project;
|
|
54
|
+
const projectConfig = workspace.projects[projectName];
|
|
55
|
+
if (!projectConfig) {
|
|
56
|
+
throw new Error(`Project ${projectName} not found in angular.json.`);
|
|
57
|
+
}
|
|
58
|
+
const projectRoot = projectConfig.root?.replace(/\\/g, '/');
|
|
59
|
+
const projectSourceRoot = projectConfig.sourceRoot?.replace(/\\/g, '/');
|
|
60
|
+
const manifestPath = path
|
|
61
|
+
.join(projectRoot, 'src/assets/federation.manifest.json')
|
|
62
|
+
.replace(/\\/g, '/');
|
|
63
|
+
const main = projectConfig.architect.build.options.main;
|
|
64
|
+
if (!projectConfig.architect.build.options.polyfills) {
|
|
65
|
+
projectConfig.architect.build.options.polyfills = [];
|
|
66
|
+
}
|
|
67
|
+
const polyfills = projectConfig.architect.build.options.polyfills;
|
|
68
|
+
return {
|
|
69
|
+
polyfills,
|
|
70
|
+
projectName,
|
|
71
|
+
projectRoot,
|
|
72
|
+
projectSourceRoot,
|
|
73
|
+
manifestPath,
|
|
74
|
+
projectConfig,
|
|
75
|
+
main,
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
function updatePolyfills(tree, polyfills) {
|
|
79
|
+
if (typeof polyfills === 'string') {
|
|
80
|
+
updatePolyfillsFile(tree, polyfills);
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
updatePolyfillsArray(tree, polyfills);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
function updatePolyfillsFile(tree, polyfills) {
|
|
87
|
+
let polyfillsContent = tree.readText(polyfills);
|
|
88
|
+
if (polyfillsContent.includes('es-module-shims')) {
|
|
89
|
+
polyfillsContent = polyfillsContent.replace(`import 'es-module-shims';`, '');
|
|
90
|
+
tree.overwrite(polyfills, polyfillsContent);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
function updatePolyfillsArray(_tree, polyfills) {
|
|
94
|
+
const polyfillsConfig = polyfills;
|
|
95
|
+
const index = polyfillsConfig.findIndex(p => p === 'es-module-shims');
|
|
96
|
+
if (index === -1) {
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
polyfillsConfig.splice(index, 1);
|
|
100
|
+
}
|
|
101
|
+
export function getWorkspaceFileName(tree) {
|
|
102
|
+
if (tree.exists('angular.json')) {
|
|
103
|
+
return 'angular.json';
|
|
104
|
+
}
|
|
105
|
+
if (tree.exists('workspace.json')) {
|
|
106
|
+
return 'workspace.json';
|
|
107
|
+
}
|
|
108
|
+
throw new Error("angular.json or workspace.json expected! Did you call this in your project's root?");
|
|
109
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"schematic.d.ts","sourceRoot":"","sources":["../../../../src/schematics/update18/schematic.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAQ,MAAM,4BAA4B,CAAC;AAI7D,MAAM,CAAC,OAAO,UAAU,QAAQ,IAAI,IAAI,CAKvC"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export declare const fstart = "// libs/native-federation-node/src/lib/node/init-node-federation.ts
import { register } from "node:module";
import { pathToFileURL } from "node:url";
import * as fs2 from "node:fs/promises";
import * as path2 from "node:path";

// libs/native-federation-runtime/src/lib/model/global-cache.ts
var nfNamespace = "__NATIVE_FEDERATION__";
var global2 = globalThis;
global2[nfNamespace] ??= {
  externals: /* @__PURE__ */ new Map(),
  remoteNamesToRemote: /* @__PURE__ */ new Map(),
  baseUrlToRemoteNames: /* @__PURE__ */ new Map()
};
var globalCache = global2[nfNamespace];

// libs/native-federation-runtime/src/lib/model/externals.ts
var externals = globalCache.externals;
function getExternalKey(shared) {
  return `${shared.packageName}@${shared.version}`;
}
function getExternalUrl(shared) {
  const packageKey = getExternalKey(shared);
  return externals.get(packageKey);
}
function setExternalUrl(shared, url2) {
  const packageKey = getExternalKey(shared);
  externals.set(packageKey, url2);
}

// libs/native-federation-runtime/src/lib/model/import-map.ts
function mergeImportMaps(map1, map2) {
  return {
    imports: { ...map1.imports, ...map2.imports },
    scopes: { ...map1.scopes, ...map2.scopes }
  };
}

// libs/native-federation-runtime/src/lib/model/remotes.ts
var remoteNamesToRemote = globalCache.remoteNamesToRemote;
var baseUrlToRemoteNames = globalCache.baseUrlToRemoteNames;
function addRemote(remoteName, remote) {
  remoteNamesToRemote.set(remoteName, remote);
  baseUrlToRemoteNames.set(remote.baseUrl, remoteName);
}

// libs/native-federation-runtime/src/lib/utils/path-utils.ts
function getDirectory(url2) {
  const parts = url2.split("/");
  parts.pop();
  return parts.join("/");
}
function joinPaths(path1, path22) {
  while (path1.endsWith("/")) {
    path1 = path1.substring(0, path1.length - 1);
  }
  if (path22.startsWith("./")) {
    path22 = path22.substring(2, path22.length);
  }
  return `${path1}/${path22}`;
}

// libs/native-federation-runtime/src/lib/watch-federation-build.ts
function watchFederationBuildCompletion(endpoint) {
  const eventSource = new EventSource(endpoint);
  eventSource.onmessage = function(event) {
    const data = JSON.parse(event.data);
    if (data.type === "federation-rebuild-complete" /* COMPLETED */) {
      console.log("[Federation] Rebuild completed, reloading...");
      window.location.reload();
    }
  };
  eventSource.onerror = function(event) {
    console.warn("[Federation] SSE connection error:", event);
  };
}

// libs/native-federation-runtime/src/lib/init-federation.ts
async function processRemoteInfos(remotes, options = { throwIfRemoteNotFound: false }) {
  const processRemoteInfoPromises = Object.keys(remotes).map(
    async (remoteName) => {
      try {
        let url2 = remotes[remoteName];
        if (options.cacheTag) {
          const addAppend = remotes[remoteName].includes("?") ? "&" : "?";
          url2 += `${addAppend}t=${options.cacheTag}`;
        }
        return await processRemoteInfo(url2, remoteName);
      } catch (e) {
        const error = `Error loading remote entry for ${remoteName} from file ${remotes[remoteName]}`;
        if (options.throwIfRemoteNotFound) {
          throw new Error(error);
        }
        console.error(error);
        return null;
      }
    }
  );
  const remoteImportMaps = await Promise.all(processRemoteInfoPromises);
  const importMap = remoteImportMaps.reduce(
    (acc, remoteImportMap) => remoteImportMap ? mergeImportMaps(acc, remoteImportMap) : acc,
    { imports: {}, scopes: {} }
  );
  return importMap;
}
async function processRemoteInfo(federationInfoUrl, remoteName) {
  const baseUrl = getDirectory(federationInfoUrl);
  const remoteInfo = await loadFederationInfo(federationInfoUrl);
  if (!remoteName) {
    remoteName = remoteInfo.name;
  }
  if (remoteInfo.buildNotificationsEndpoint) {
    watchFederationBuildCompletion(
      baseUrl + remoteInfo.buildNotificationsEndpoint
    );
  }
  const importMap = createRemoteImportMap(remoteInfo, remoteName, baseUrl);
  addRemote(remoteName, { ...remoteInfo, baseUrl });
  return importMap;
}
function createRemoteImportMap(remoteInfo, remoteName, baseUrl) {
  const imports = processExposed(remoteInfo, remoteName, baseUrl);
  const scopes = processRemoteImports(remoteInfo, baseUrl);
  return { imports, scopes };
}
async function loadFederationInfo(url2) {
  const info = await fetch(url2).then((r) => r.json());
  return info;
}
function processRemoteImports(remoteInfo, baseUrl) {
  const scopes = {};
  const scopedImports = {};
  for (const shared of remoteInfo.shared) {
    const outFileName = getExternalUrl(shared) ?? joinPaths(baseUrl, shared.outFileName);
    setExternalUrl(shared, outFileName);
    scopedImports[shared.packageName] = outFileName;
  }
  scopes[baseUrl + "/"] = scopedImports;
  return scopes;
}
function processExposed(remoteInfo, remoteName, baseUrl) {
  const imports = {};
  for (const exposed of remoteInfo.exposes) {
    const key = joinPaths(remoteName, exposed.key);
    const value = joinPaths(baseUrl, exposed.outFileName);
    imports[key] = value;
  }
  return imports;
}
async function processHostInfo(hostInfo, relBundlesPath = "./") {
  const imports = hostInfo.shared.reduce(
    (acc, cur) => ({
      ...acc,
      [cur.packageName]: relBundlesPath + cur.outFileName
    }),
    {}
  );
  for (const shared of hostInfo.shared) {
    setExternalUrl(shared, relBundlesPath + shared.outFileName);
  }
  return { imports, scopes: {} };
}

// libs/native-federation-node/src/lib/utils/import-map-loader.js
import path from "path";
import url from "url";
import { promises as fs } from "fs";
var IMPORT_MAP_FILE_NAME = "node.importmap";
var baseURL = url.pathToFileURL(process.cwd()) + path.sep;
function resolveAndComposeImportMap(parsed) {
  if (!isPlainObject(parsed)) {
    throw Error(`Invalid import map - top level must be an object`);
  }
  let sortedAndNormalizedImports = {};
  if (Object.prototype.hasOwnProperty.call(parsed, "imports")) {
    if (!isPlainObject(parsed.imports)) {
      throw Error(`Invalid import map - "imports" property must be an object`);
    }
    sortedAndNormalizedImports = sortAndNormalizeSpecifierMap(
      parsed.imports,
      baseURL
    );
  }
  let sortedAndNormalizedScopes = {};
  if (Object.prototype.hasOwnProperty.call(parsed, "scopes")) {
    if (!isPlainObject(parsed.scopes)) {
      throw Error(`Invalid import map - "scopes" property must be an object`);
    }
    sortedAndNormalizedScopes = sortAndNormalizeScopes(parsed.scopes, baseURL);
  }
  const invalidKeys = Object.keys(parsed).filter(
    (key) => key !== "imports" && key !== "scopes"
  );
  if (invalidKeys.length > 0) {
    console.warn(
      `Invalid top-level key${invalidKeys.length > 0 ? "s" : ""} in import map - ${invalidKeys.join(", ")}`
    );
  }
  return {
    imports: sortedAndNormalizedImports,
    scopes: sortedAndNormalizedScopes
  };
}
function sortAndNormalizeSpecifierMap(map, baseURL2) {
  const normalized = {};
  for (let specifierKey in map) {
    const value = map[specifierKey];
    const normalizedSpecifierKey = normalizeSpecifierKey(specifierKey, baseURL2);
    if (normalizedSpecifierKey === null) {
      continue;
    }
    let addressURL = parseURLLikeSpecifier(value, baseURL2);
    if (addressURL === null) {
      console.warn(
        `Invalid URL address for import map specifier '${specifierKey}'`
      );
      normalized[normalizedSpecifierKey] = null;
      continue;
    }
    if (specifierKey.endsWith("/") && !addressURL.endsWith("/")) {
      console.warn(
        `Invalid URL address for import map specifier '${specifierKey}' - since the specifier ends in slash, so must the address`
      );
      normalized[normalizedSpecifierKey] = null;
      continue;
    }
    normalized[normalizedSpecifierKey] = addressURL;
  }
  return normalized;
}
function normalizeSpecifierKey(key) {
  if (key === "") {
    console.warn(`Specifier keys in import maps may not be the empty string`);
    return null;
  }
  return parseURLLikeSpecifier(key, baseURL) || key;
}
function parseURLLikeSpecifier(specifier, baseURL2) {
  const useBaseUrlAsParent = specifier.startsWith("/") || specifier.startsWith("./") || specifier.startsWith("../");
  try {
    return new URL(specifier, useBaseUrlAsParent ? baseURL2 : void 0).href;
  } catch {
    return null;
  }
}
function sortAndNormalizeScopes(map, baseURL2) {
  let normalized = {};
  for (let scopePrefix in map) {
    const potentialSpecifierMap = map[scopePrefix];
    if (!isPlainObject(potentialSpecifierMap)) {
      throw TypeError(
        `The value of scope ${scopePrefix} must be a JSON object`
      );
    }
    let scopePrefixURL;
    try {
      scopePrefixURL = new URL(scopePrefix, baseURL2).href;
    } catch {
      console.warn(
        `Scope prefix URL '${scopePrefix}' was not parseable in import map`
      );
      continue;
    }
    normalized[scopePrefixURL] = sortAndNormalizeSpecifierMap(
      potentialSpecifierMap,
      baseURL2
    );
  }
  return normalized;
}
function isPlainObject(obj) {
  return obj === Object(obj) && !Array.isArray(obj);
}
var importMapPromise = getImportMapPromise();
async function getImportMapPromise() {
  const relativePath = process.env.IMPORT_MAP_PATH || IMPORT_MAP_FILE_NAME;
  const importMapPath = path.resolve(process.cwd(), relativePath);
  let str;
  try {
    str = await fs.readFile(importMapPath);
  } catch (err) {
    return emptyMap();
  }
  let json;
  try {
    json = await JSON.parse(str);
  } catch (err) {
    throw Error(
      `Import map at ${importMapPath} contains invalid json: ${err.message}`
    );
  }
  return resolveAndComposeImportMap(json);
}
global.nodeLoader = global.nodeLoader || {};
global.nodeLoader.setImportMapPromise = function setImportMapPromise(promise) {
  importMapPromise = promise.then((map) => {
    return resolveAndComposeImportMap(map);
  });
};
function emptyMap() {
  return { imports: {}, scopes: {} };
}

// libs/native-federation-node/src/lib/utils/loader-as-data-url.js
var resolver = "import path from 'path';
import url from 'url';
import { promises as fs } from 'fs';

export const IMPORT_MAP_FILE_NAME = 'node.importmap';

const baseURL = url.pathToFileURL(process.cwd()) + path.sep;

// https://wicg.github.io/import-maps/#new-resolve-algorithm
export function resolveSpecifier(importMap, specifier, parentURL) {
  let currentBaseURL;
  if (parentURL) {
    const lastSlashIndex = parentURL.lastIndexOf(path.sep);
    currentBaseURL = parentURL.slice(0, lastSlashIndex + 1);
  } else {
    currentBaseURL = baseURL;
  }
  const normalizedSpecifier =
    parseURLLikeSpecifier(specifier, currentBaseURL) || specifier;
  for (let scopePrefix in importMap.scopes) {
    if (
      scopePrefix === currentBaseURL ||
      (scopePrefix.endsWith('/') && currentBaseURL.startsWith(scopePrefix))
    ) {
      const scopeImportsMatch = resolveImportsMatch(
        normalizedSpecifier,
        importMap.scopes[scopePrefix],
      );
      if (scopeImportsMatch) {
        return scopeImportsMatch;
      }
    } else {
      const topLevelImportsMatch = resolveImportsMatch(
        normalizedSpecifier,
        importMap.imports,
      );
      if (topLevelImportsMatch) {
        return topLevelImportsMatch;
      }
    }
  }

  return resolveImportsMatch(normalizedSpecifier, importMap.imports);
}

// https://wicg.github.io/import-maps/#resolve-an-imports-match
function resolveImportsMatch(normalizedSpecifier, specifierMap) {
  for (let specifierKey in specifierMap) {
    const resolutionResult = specifierMap[specifierKey];

    if (specifierKey === normalizedSpecifier) {
      if (resolutionResult === null) {
        throw TypeError(
          `The import map resolution of ${specifierKey} failed due to a null entry`,
        );
      }
      return resolutionResult;
    } else if (
      specifierKey.endsWith('/') &&
      normalizedSpecifier.startsWith(specifierKey)
    ) {
      if (resolutionResult === null) {
        throw TypeError(
          `The import map resolution of ${specifierKey} failed due to a null entry`,
        );
      }
      const afterPrefix = normalizedSpecifier.slice(specifierKey.length);
      try {
        return new URL(afterPrefix, resolutionResult).href;
      } catch {
        throw TypeError(
          `The import map resolution of ${specifierKey} failed due to URL parse failure`,
        );
      }
    }
  }

  return null;
}

// https://wicg.github.io/import-maps/#parsing
export function resolveAndComposeImportMap(parsed) {
  // Step 2
  if (!isPlainObject(parsed)) {
    throw Error(`Invalid import map - top level must be an object`);
  }

  // Step 3
  let sortedAndNormalizedImports = {};

  // Step 4
  if (Object.prototype.hasOwnProperty.call(parsed, 'imports')) {
    // Step 4.1
    if (!isPlainObject(parsed.imports)) {
      throw Error(`Invalid import map - "imports" property must be an object`);
    }

    // Step 4.2
    sortedAndNormalizedImports = sortAndNormalizeSpecifierMap(
      parsed.imports,
      baseURL,
    );
  }

  // Step 5
  let sortedAndNormalizedScopes = {};

  // Step 6
  if (Object.prototype.hasOwnProperty.call(parsed, 'scopes')) {
    // Step 6.1
    if (!isPlainObject(parsed.scopes)) {
      throw Error(`Invalid import map - "scopes" property must be an object`);
    }

    // Step 6.2
    sortedAndNormalizedScopes = sortAndNormalizeScopes(parsed.scopes, baseURL);
  }

  // Step 7
  const invalidKeys = Object.keys(parsed).filter(
    (key) => key !== 'imports' && key !== 'scopes',
  );
  if (invalidKeys.length > 0) {
    console.warn(
      `Invalid top-level key${
        invalidKeys.length > 0 ? 's' : ''
      } in import map - ${invalidKeys.join(', ')}`,
    );
  }

  // Step 8
  return {
    imports: sortedAndNormalizedImports,
    scopes: sortedAndNormalizedScopes,
  };
}

// https://wicg.github.io/import-maps/#sort-and-normalize-a-specifier-map
function sortAndNormalizeSpecifierMap(map, baseURL) {
  const normalized = {};

  for (let specifierKey in map) {
    const value = map[specifierKey];

    const normalizedSpecifierKey = normalizeSpecifierKey(specifierKey, baseURL);
    if (normalizedSpecifierKey === null) {
      continue;
    }

    let addressURL = parseURLLikeSpecifier(value, baseURL);
    if (addressURL === null) {
      console.warn(
        `Invalid URL address for import map specifier '${specifierKey}'`,
      );
      normalized[normalizedSpecifierKey] = null;
      continue;
    }

    if (specifierKey.endsWith('/') && !addressURL.endsWith('/')) {
      console.warn(
        `Invalid URL address for import map specifier '${specifierKey}' - since the specifier ends in slash, so must the address`,
      );
      normalized[normalizedSpecifierKey] = null;
      continue;
    }

    normalized[normalizedSpecifierKey] = addressURL;
  }

  return normalized;
}

// https://wicg.github.io/import-maps/#normalize-a-specifier-key
function normalizeSpecifierKey(key) {
  if (key === '') {
    console.warn(`Specifier keys in import maps may not be the empty string`);
    return null;
  }

  return parseURLLikeSpecifier(key, baseURL) || key;
}

// https://wicg.github.io/import-maps/#parse-a-url-like-import-specifier
function parseURLLikeSpecifier(specifier, baseURL) {
  const useBaseUrlAsParent =
    specifier.startsWith('/') ||
    specifier.startsWith('./') ||
    specifier.startsWith('../');

  try {
    return new URL(specifier, useBaseUrlAsParent ? baseURL : undefined).href;
  } catch {
    return null;
  }
}

// https://wicg.github.io/import-maps/#sort-and-normalize-scopes
function sortAndNormalizeScopes(map, baseURL) {
  let normalized = {};

  for (let scopePrefix in map) {
    const potentialSpecifierMap = map[scopePrefix];
    if (!isPlainObject(potentialSpecifierMap)) {
      throw TypeError(
        `The value of scope ${scopePrefix} must be a JSON object`,
      );
    }

    let scopePrefixURL;
    try {
      scopePrefixURL = new URL(scopePrefix, baseURL).href;
    } catch {
      console.warn(
        `Scope prefix URL '${scopePrefix}' was not parseable in import map`,
      );
      continue;
    }

    normalized[scopePrefixURL] = sortAndNormalizeSpecifierMap(
      potentialSpecifierMap,
      baseURL,
    );
  }

  return normalized;
}

function isPlainObject(obj) {
  return obj === Object(obj) && !Array.isArray(obj);
}

// ---

let importMapPromise = getImportMapPromise();

export async function resolve(specifier, context, defaultResolve) {
  const { parentURL = null } = context;
  const importMap = await importMapPromise;
  const importMapUrl = resolveSpecifier(importMap, specifier, parentURL);

  return defaultResolve(importMapUrl ?? specifier, context, defaultResolve);
}

export async function load(url, context, defaultLoad) {
  if (url.startsWith('http://') || url.startsWith('https://')) {
    const res = await fetch(url);
    if (!res.ok) {
      throw new Error(`Failed to fetch module from ${url}`);
    }
    const source = await res.text();
    return {
      shortCircuit: true,
      format: 'module',
      source,
    };
  }

  if (!url.startsWith('node:')) {
    context.format = 'module';
  }

  return defaultLoad(url, context, defaultLoad);
}

async function getImportMapPromise() {
  const relativePath = process.env.IMPORT_MAP_PATH || IMPORT_MAP_FILE_NAME;
  const importMapPath = path.resolve(process.cwd(), relativePath);

  let str;
  try {
    str = await fs.readFile(importMapPath);
  } catch (err) {
    return emptyMap();
  }

  let json;
  try {
    json = await JSON.parse(str);
  } catch (err) {
    throw Error(
      `Import map at ${importMapPath} contains invalid json: ${err.message}`,
    );
  }

  return resolveAndComposeImportMap(json);
}

global.nodeLoader = global.nodeLoader || {};

global.nodeLoader.setImportMapPromise = function setImportMapPromise(promise) {
  importMapPromise = promise.then((map) => {
    return resolveAndComposeImportMap(map);
  });
};

function emptyMap() {
  return { imports: {}, scopes: {} };
}
";

// libs/native-federation-node/src/lib/node/init-node-federation.ts
var defaultOptions = {
  remotesOrManifestUrl: {},
  relBundlePath: "../browser",
  throwIfRemoteNotFound: false
};
async function initNodeFederation(options) {
  const mergedOptions = { ...defaultOptions, ...options };
  const importMap = await createNodeImportMap(mergedOptions);
  await writeImportMap(importMap);
  await writeResolver();
  register(pathToFileURL("./federation-resolver.mjs").href);
}
async function createNodeImportMap(options) {
  const { remotesOrManifestUrl, relBundlePath } = options;
  const remotes = typeof remotesOrManifestUrl === "object" ? remotesOrManifestUrl : await loadFsManifest(remotesOrManifestUrl);
  const hostInfo = await loadFsFederationInfo(relBundlePath);
  const hostImportMap = await processHostInfo(hostInfo, "./" + relBundlePath);
  const remotesImportMap = await processRemoteInfos(remotes, {
    throwIfRemoteNotFound: options.throwIfRemoteNotFound,
    cacheTag: options.cacheTag
  });
  const importMap = mergeImportMaps(hostImportMap, remotesImportMap);
  return importMap;
}
async function loadFsManifest(manifestUrl) {
  const content = await fs2.readFile(manifestUrl, "utf-8");
  const manifest = JSON.parse(content);
  return manifest;
}
async function loadFsFederationInfo(relBundlePath) {
  const manifestPath = path2.join(relBundlePath, "remoteEntry.json");
  const content = await fs2.readFile(manifestPath, "utf-8");
  const manifest = JSON.parse(content);
  return manifest;
}
async function writeImportMap(map) {
  await fs2.writeFile(
    IMPORT_MAP_FILE_NAME,
    JSON.stringify(map, null, 2),
    "utf-8"
  );
}
async function writeResolver() {
  const buffer = Buffer.from(resolver, "base64");
  await fs2.writeFile("federation-resolver.mjs", buffer, "utf-8");
}

// libs/native-federation-node/src/lib/utils/fstart-args-parser.ts
import * as fs3 from "node:fs";
var defaultArgs = {
  entry: "./server.mjs",
  remotesOrManifestUrl: "../browser/federation.manifest.json",
  relBundlePath: "../browser/"
};
function parseFStartArgs() {
  const args2 = {
    entry: "",
    remotesOrManifestUrl: "",
    relBundlePath: ""
  };
  let key = "";
  for (let i = 2; i < process.argv.length; i++) {
    const cand = process.argv[i];
    if (cand.startsWith("--")) {
      const candKey = cand.substring(2);
      if (defaultArgs[candKey]) {
        key = candKey;
      } else {
        console.error(`switch ${cand} not supported!`);
        exitWithUsage(defaultArgs);
      }
    } else if (key) {
      args2[key] = cand;
      key = "";
    } else {
      console.error(`unreladed value ${cand}!`);
      exitWithUsage(defaultArgs);
    }
  }
  applyDefaultArgs(args2);
  return args2;
}
function applyDefaultArgs(args2) {
  if (args2.relBundlePath && !args2.remotesOrManifestUrl) {
    const cand = defaultArgs.relBundlePath + "federation.manifest.json";
    if (fs3.existsSync(cand)) {
      args2.remotesOrManifestUrl = cand;
    }
  }
  args2.entry = args2.entry || defaultArgs.entry;
  args2.relBundlePath = args2.relBundlePath || defaultArgs.relBundlePath;
  args2.remotesOrManifestUrl = args2.remotesOrManifestUrl || defaultArgs.remotesOrManifestUrl;
  if (!fs3.existsSync(args2.remotesOrManifestUrl)) {
    args2.remotesOrManifestUrl = void 0;
  }
}
function exitWithUsage(defaultArgs2) {
  let args2 = "";
  for (const key in defaultArgs2) {
    args2 += `[--${key} ${defaultArgs2[key]}] `;
  }
  console.log("usage: nfstart " + args2);
  process.exit(1);
}

// libs/native-federation-node/src/lib/utils/fstart.ts
var args = parseFStartArgs();
(async () => {
  await initNodeFederation({
    ...args.remotesOrManifestUrl ? { remotesOrManifestUrl: args.remotesOrManifestUrl } : {},
    relBundlePath: args.relBundlePath
  });
  await import(args.entry);
})();
";
|
|
2
|
+
//# sourceMappingURL=fstart-as-data-url.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fstart-as-data-url.d.ts","sourceRoot":"","sources":["../../../src/tools/fstart-as-data-url.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,MAAM,i8/BAC66/B,CAAC"}
|