Skip to content

Commit bf4ac0f

Browse files
authored
refactor(plugin): use standard dfs instead of topological for stable sort (#262)
* refactor(plugin): use standard dfs instead of topological for stable sort * refactor(plugin): use recursively dfs to stable sort result & add test case
1 parent 6e591b4 commit bf4ac0f

File tree

6 files changed

+81
-217
lines changed

6 files changed

+81
-217
lines changed

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "@artus/core",
3-
"version": "2.0.5",
3+
"version": "2.1.0",
44
"description": "Core package of Artus",
55
"main": "./lib/index.js",
66
"types": "./lib/index.d.ts",

src/plugin/common.ts

+56-32
Original file line numberDiff line numberDiff line change
@@ -1,62 +1,86 @@
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";
45

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> = {};
912

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+
}
1319

14-
const queue: string[] = [];
20+
if (visited[pluginName]) return;
1521

16-
for (const [name] of pluginInstanceMap) {
17-
if (!indegree.has(name)) {
18-
queue.push(name);
19-
}
20-
}
22+
visited[pluginName] = true;
2123

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));
3041
}
3142
}
43+
sortedPlugins.push(plugin);
3244
}
45+
};
46+
47+
for (const pluginName of pluginInstanceMap.keys()) {
48+
visit(pluginName);
3349
}
34-
return res;
50+
51+
return sortedPlugins;
3552
}
3653

3754
// 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 {
3959
const opts = {
4060
paths: paths.concat(__dirname),
4161
};
4262
return path.dirname(require.resolve(packageName, opts));
4363
}
4464

45-
export async function getInlinePackageEntryPath(packagePath: string): Promise<string> {
65+
export async function getInlinePackageEntryPath(
66+
packagePath: string,
67+
): Promise<string> {
4668
const pkgJson = await compatibleRequire(`${packagePath}/package.json`);
47-
let entryFilePath = '';
69+
let entryFilePath = "";
4870
if (pkgJson.exports) {
4971
if (Array.isArray(pkgJson.exports)) {
5072
throw new Error(`inline package multi exports is not supported`);
51-
} else if (typeof pkgJson.exports === 'string') {
73+
} else if (typeof pkgJson.exports === "string") {
5274
entryFilePath = pkgJson.exports;
53-
} else if (pkgJson.exports?.['.']) {
54-
entryFilePath = pkgJson.exports['.'];
75+
} else if (pkgJson.exports?.["."]) {
76+
entryFilePath = pkgJson.exports["."];
5577
}
5678
}
5779
if (!entryFilePath && pkgJson.main) {
5880
entryFilePath = pkgJson.main;
5981
}
6082
// 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;
6286
}

src/plugin/factory.ts

+5-21
Original file line numberDiff line numberDiff line change
@@ -1,33 +1,17 @@
1-
import { topologicalSort } from './common';
1+
import { sortPlugins } from './common';
22
import { Plugin } from './impl';
33
import { PluginConfigItem, PluginCreateOptions, PluginMap, PluginType } from './types';
44

55
export class PluginFactory {
6-
static async create(name: string, item: PluginConfigItem, opts?: PluginCreateOptions): Promise<PluginType> {
7-
const pluginInstance = new Plugin(name, item, opts);
8-
await pluginInstance.init();
9-
return pluginInstance;
10-
}
11-
126
static async createFromConfig(config: Record<string, PluginConfigItem>, opts?: PluginCreateOptions): Promise<PluginType[]> {
137
const pluginInstanceMap: PluginMap = new Map();
148
for (const [name, item] of Object.entries(config)) {
15-
const pluginInstance = await PluginFactory.create(name, item, opts);
16-
if (pluginInstance.enable) {
9+
if (item.enable) {
10+
const pluginInstance = new Plugin(name, item);
11+
await pluginInstance.init();
1712
pluginInstanceMap.set(name, pluginInstance);
1813
}
1914
}
20-
let pluginDepEdgeList: [string, string][] = [];
21-
// Topological sort plugins
22-
for (const [_name, pluginInstance] of pluginInstanceMap) {
23-
pluginInstance.checkDepExisted(pluginInstanceMap);
24-
pluginDepEdgeList = pluginDepEdgeList.concat(pluginInstance.getDepEdgeList());
25-
}
26-
const pluginSortResult: string[] = topologicalSort(pluginInstanceMap, pluginDepEdgeList);
27-
if (pluginSortResult.length !== pluginInstanceMap.size) {
28-
const diffPlugin = [...pluginInstanceMap.keys()].filter(name => !pluginSortResult.includes(name));
29-
throw new Error(`There is a cycle in the dependencies, wrong plugin is ${diffPlugin.join(',')}.`);
30-
}
31-
return pluginSortResult.map(name => pluginInstanceMap.get(name)!);
15+
return sortPlugins(pluginInstanceMap, opts?.logger);
3216
}
3317
}

src/plugin/impl.ts

+3-34
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,8 @@ import path from 'path';
22
import { loadMetaFile } from '../utils/load_meta_file';
33
import { exists } from '../utils/fs';
44
import { PLUGIN_META_FILENAME } from '../constant';
5-
import { PluginConfigItem, PluginCreateOptions, PluginMap, PluginMetadata, PluginType } from './types';
5+
import { PluginConfigItem, PluginMetadata, PluginType } from './types';
66
import { getPackagePath } from './common';
7-
import { LoggerType } from '../logger';
87

98
export class Plugin implements PluginType {
109
public name: string;
@@ -13,9 +12,7 @@ export class Plugin implements PluginType {
1312
public metadata: Partial<PluginMetadata>;
1413
public metaFilePath = '';
1514

16-
private logger?: LoggerType;
17-
18-
constructor(name: string, configItem: PluginConfigItem, opts?: PluginCreateOptions) {
15+
constructor(name: string, configItem: PluginConfigItem) {
1916
this.name = name;
2017
this.enable = configItem.enable ?? false;
2118
if (this.enable) {
@@ -34,10 +31,9 @@ export class Plugin implements PluginType {
3431
if (configItem.metadata) {
3532
this.metadata = configItem.metadata;
3633
}
37-
this.logger = opts?.logger;
3834
}
3935

40-
async init() {
36+
public async init() {
4137
if (!this.enable) {
4238
return;
4339
}
@@ -50,33 +46,6 @@ export class Plugin implements PluginType {
5046
}
5147
}
5248

53-
public checkDepExisted(pluginMap: PluginMap) {
54-
if (!this.metadata.dependencies) {
55-
return;
56-
}
57-
58-
for (let i = 0; i < this.metadata.dependencies.length; i++) {
59-
const { name: pluginName, optional } = this.metadata.dependencies[i];
60-
const instance = pluginMap.get(pluginName);
61-
if (!instance || !instance.enable) {
62-
if (optional) {
63-
this.logger?.warn(`Plugin ${this.name} need have optional dependency: ${pluginName}.`);
64-
} else {
65-
throw new Error(`Plugin ${this.name} need have dependency: ${pluginName}.`);
66-
}
67-
} else {
68-
// Plugin exist and enabled, need calc edge
69-
this.metadata.dependencies[i]._enabled = true;
70-
}
71-
}
72-
}
73-
74-
public getDepEdgeList(): [string, string][] {
75-
return this.metadata.dependencies
76-
?.filter(({ optional, _enabled }) => !optional || _enabled)
77-
?.map(({ name: depPluginName }) => [this.name, depPluginName]) ?? [];
78-
}
79-
8049
private async checkAndLoadMetadata() {
8150
// check metadata from configItem
8251
if (this.metadata) {

src/plugin/types.ts

-2
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,4 @@ export interface PluginType {
3939
metaFilePath: string;
4040

4141
init(): Promise<void>;
42-
checkDepExisted(map: PluginMap): void;
43-
getDepEdgeList(): [string, string][];
4442
}

0 commit comments

Comments
 (0)