|
1 |
| -import path from 'path'; |
2 |
| -import compatibleRequire from '../utils/compatible_require'; |
3 |
| -import { PluginType } from './types'; |
| 1 | +import path from "path"; |
| 2 | +import compatibleRequire from "../utils/compatible_require"; |
| 3 | +import { PluginType } from "./types"; |
| 4 | +import { LoggerType } from "../logger"; |
4 | 5 |
|
5 |
| -// A utils function that toplogical sort plugins |
6 |
| -export function topologicalSort(pluginInstanceMap: Map<string, PluginType>, pluginDepEdgeList: [string, string][]): string[] { |
7 |
| - const res: string[] = []; |
8 |
| - const indegree: Map<string, number> = new Map(); |
| 6 | +export function sortPlugins( |
| 7 | + pluginInstanceMap: Map<string, PluginType>, |
| 8 | + logger: LoggerType, |
| 9 | +): PluginType[] { |
| 10 | + const sortedPlugins: PluginType[] = []; |
| 11 | + const visited: Record<string, boolean> = {}; |
9 | 12 |
|
10 |
| - pluginDepEdgeList.forEach(([to]) => { |
11 |
| - indegree.set(to, (indegree.get(to) ?? 0) + 1); |
12 |
| - }); |
| 13 | + const visit = (pluginName: string, depChain: string[] = []) => { |
| 14 | + if (depChain.includes(pluginName)) { |
| 15 | + throw new Error( |
| 16 | + `Circular dependency found in plugins: ${depChain.join(", ")}`, |
| 17 | + ); |
| 18 | + } |
13 | 19 |
|
14 |
| - const queue: string[] = []; |
| 20 | + if (visited[pluginName]) return; |
15 | 21 |
|
16 |
| - for (const [name] of pluginInstanceMap) { |
17 |
| - if (!indegree.has(name)) { |
18 |
| - queue.push(name); |
19 |
| - } |
20 |
| - } |
| 22 | + visited[pluginName] = true; |
21 | 23 |
|
22 |
| - while(queue.length) { |
23 |
| - const cur = queue.shift()!; |
24 |
| - res.push(cur); |
25 |
| - for (const [to, from] of pluginDepEdgeList) { |
26 |
| - if (from === cur) { |
27 |
| - indegree.set(to, (indegree.get(to) ?? 0) - 1); |
28 |
| - if (indegree.get(to) === 0) { |
29 |
| - queue.push(to); |
| 24 | + const plugin = pluginInstanceMap.get(pluginName); |
| 25 | + if (plugin) { |
| 26 | + for (const dep of plugin.metadata.dependencies ?? []) { |
| 27 | + const depPlugin = pluginInstanceMap.get(dep.name); |
| 28 | + if (!depPlugin || !depPlugin.enable) { |
| 29 | + if (dep.optional) { |
| 30 | + logger?.warn( |
| 31 | + `Plugin ${plugin.name} need have optional dependency: ${dep.name}.`, |
| 32 | + ); |
| 33 | + } else { |
| 34 | + throw new Error( |
| 35 | + `Plugin ${plugin.name} need have dependency: ${dep.name}.`, |
| 36 | + ); |
| 37 | + } |
| 38 | + } else { |
| 39 | + // Plugin exist and enabled, need visit |
| 40 | + visit(dep.name, depChain.concat(pluginName)); |
30 | 41 | }
|
31 | 42 | }
|
| 43 | + sortedPlugins.push(plugin); |
32 | 44 | }
|
| 45 | + }; |
| 46 | + |
| 47 | + for (const pluginName of pluginInstanceMap.keys()) { |
| 48 | + visit(pluginName); |
33 | 49 | }
|
34 |
| - return res; |
| 50 | + |
| 51 | + return sortedPlugins; |
35 | 52 | }
|
36 | 53 |
|
37 | 54 | // A util function of get package path for plugin
|
38 |
| -export function getPackagePath(packageName: string, paths: string[] = []): string { |
| 55 | +export function getPackagePath( |
| 56 | + packageName: string, |
| 57 | + paths: string[] = [], |
| 58 | +): string { |
39 | 59 | const opts = {
|
40 | 60 | paths: paths.concat(__dirname),
|
41 | 61 | };
|
42 | 62 | return path.dirname(require.resolve(packageName, opts));
|
43 | 63 | }
|
44 | 64 |
|
45 |
| -export async function getInlinePackageEntryPath(packagePath: string): Promise<string> { |
| 65 | +export async function getInlinePackageEntryPath( |
| 66 | + packagePath: string, |
| 67 | +): Promise<string> { |
46 | 68 | const pkgJson = await compatibleRequire(`${packagePath}/package.json`);
|
47 |
| - let entryFilePath = ''; |
| 69 | + let entryFilePath = ""; |
48 | 70 | if (pkgJson.exports) {
|
49 | 71 | if (Array.isArray(pkgJson.exports)) {
|
50 | 72 | throw new Error(`inline package multi exports is not supported`);
|
51 |
| - } else if (typeof pkgJson.exports === 'string') { |
| 73 | + } else if (typeof pkgJson.exports === "string") { |
52 | 74 | entryFilePath = pkgJson.exports;
|
53 |
| - } else if (pkgJson.exports?.['.']) { |
54 |
| - entryFilePath = pkgJson.exports['.']; |
| 75 | + } else if (pkgJson.exports?.["."]) { |
| 76 | + entryFilePath = pkgJson.exports["."]; |
55 | 77 | }
|
56 | 78 | }
|
57 | 79 | if (!entryFilePath && pkgJson.main) {
|
58 | 80 | entryFilePath = pkgJson.main;
|
59 | 81 | }
|
60 | 82 | // will use package root path if no entry file found
|
61 |
| - return entryFilePath ? path.resolve(packagePath, entryFilePath, '..') : packagePath; |
| 83 | + return entryFilePath |
| 84 | + ? path.resolve(packagePath, entryFilePath, "..") |
| 85 | + : packagePath; |
62 | 86 | }
|
0 commit comments