Skip to content

Commit 3a4bf7c

Browse files
authored
refactor: 支持 Plugin.package 字段在扫描阶段的预处理 (#110)
1 parent e1bd4d6 commit 3a4bf7c

File tree

10 files changed

+161
-62
lines changed

10 files changed

+161
-62
lines changed

src/loader/impl/config.ts

+37-8
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,12 @@ import { DefineLoader } from '../decorator';
66
import { ManifestItem, Loader, LoaderFindOptions } from '../types';
77
import compatibleRequire from '../../utils/compatible_require';
88
import { isMatch } from '../../utils';
9+
import { Application } from '../../types';
10+
11+
export interface ConfigFileMeta {
12+
env: string;
13+
namespace?: string;
14+
}
915

1016
@DefineLoader('config')
1117
class ConfigLoader implements Loader {
@@ -15,6 +21,14 @@ class ConfigLoader implements Loader {
1521
this.container = container;
1622
}
1723

24+
protected get app(): Application {
25+
return this.container.get(ArtusInjectEnum.Application);
26+
}
27+
28+
protected get configurationHandler(): ConfigurationHandler {
29+
return this.container.get(ConfigurationHandler);
30+
}
31+
1832
static async is(opts: LoaderFindOptions): Promise<boolean> {
1933
if (this.isConfigDir(opts)) {
2034
return isMatch(opts.filename, CONFIG_PATTERN);
@@ -28,24 +42,39 @@ class ConfigLoader implements Loader {
2842
}
2943

3044
async load(item: ManifestItem) {
31-
const originConfigObj = await compatibleRequire(item.path);
45+
const { namespace, env } = await this.getConfigFileMeta(item);
46+
let configObj = await this.loadConfigFile(item);
47+
if (namespace) {
48+
configObj = {
49+
[namespace]: configObj
50+
};
51+
}
52+
this.configurationHandler.setConfig(env, configObj);
53+
}
54+
55+
protected async getConfigFileMeta(item: ManifestItem): Promise<ConfigFileMeta> {
3256
let [namespace, env, extname] = item.filename.split('.');
3357
if (!extname) {
3458
// No env flag, set to Default
3559
env = ARTUS_DEFAULT_CONFIG_ENV.DEFAULT;
3660
}
61+
const meta: ConfigFileMeta = {
62+
env
63+
};
64+
if (namespace !== 'config') {
65+
meta.namespace = namespace;
66+
}
67+
return meta
68+
}
69+
70+
protected async loadConfigFile(item: ManifestItem): Promise<Record<string, any>> {
71+
const originConfigObj = await compatibleRequire(item.path);
3772
let configObj = originConfigObj;
3873
if(typeof originConfigObj === 'function') {
3974
const app = this.container.get(ArtusInjectEnum.Application);
4075
configObj = originConfigObj(app);
4176
}
42-
if (namespace !== 'config') {
43-
configObj = {
44-
[namespace]: configObj
45-
};
46-
}
47-
const configHandler = this.container.get(ConfigurationHandler);
48-
configHandler.setConfig(env, configObj);
77+
return configObj;
4978
}
5079
}
5180

src/loader/impl/framework_config.ts

+5-17
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,7 @@
1-
import ConfigurationHandler from '../../configuration';
1+
import { FrameworkObject } from '../../configuration';
22
import { DefineLoader } from '../decorator';
33
import { ManifestItem, Loader, LoaderFindOptions } from '../types';
4-
import compatibleRequire from '../../utils/compatible_require';
5-
import { ArtusInjectEnum, ARTUS_DEFAULT_CONFIG_ENV, FRAMEWORK_PATTERN } from '../../constant';
4+
import { FRAMEWORK_PATTERN } from '../../constant';
65
import ConfigLoader from './config';
76
import { isMatch } from '../../utils';
87

@@ -20,20 +19,9 @@ class FrameworkConfigLoader extends ConfigLoader implements Loader {
2019
}
2120

2221
async load(item: ManifestItem) {
23-
const originConfigObj = await compatibleRequire(item.path);
24-
let [, env, extname] = item.filename.split('.');
25-
if (!extname) {
26-
// No env flag, set to Default
27-
env = ARTUS_DEFAULT_CONFIG_ENV.DEFAULT;
28-
}
29-
let configObj = originConfigObj;
30-
if (typeof originConfigObj === 'function') {
31-
const app = this.container.get(ArtusInjectEnum.Application);
32-
configObj = originConfigObj(app);
33-
}
34-
35-
const configHandler = this.container.get(ConfigurationHandler);
36-
configHandler.addFramework(item.source || 'app', configObj, {
22+
const { env } = await this.getConfigFileMeta(item);
23+
const configObj = await this.loadConfigFile(item) as FrameworkObject;
24+
this.configurationHandler.addFramework(item.source || 'app', configObj, {
3725
env,
3826
unitName: item.unitName || '',
3927
});

src/loader/impl/plugin_config.ts

+21-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
import { PLUGIN_CONFIG_PATTERN } from '../../constant';
2+
import { ArtusPlugin } from '../../plugin';
3+
import { PluginConfigItem } from '../../plugin/types';
24
import { isMatch } from '../../utils';
35
import { DefineLoader } from '../decorator';
46
import { ManifestItem, Loader, LoaderFindOptions } from '../types';
@@ -15,7 +17,25 @@ class PluginConfigLoader extends ConfigLoader implements Loader {
1517
}
1618

1719
async load(item: ManifestItem) {
18-
await super.load(item);
20+
const { env } = await this.getConfigFileMeta(item);
21+
let configObj = await this.loadConfigFile(item);
22+
for (const pluginName of Object.keys(configObj)) {
23+
const pluginConfigItem: PluginConfigItem = configObj[pluginName];
24+
if (pluginConfigItem.package) {
25+
// convert package to path when load plugin config
26+
if (pluginConfigItem.path) {
27+
throw new Error(`Plugin ${pluginName} config can't have both package and path at ${item.path}`);
28+
}
29+
if (pluginConfigItem.enable) {
30+
pluginConfigItem.path = ArtusPlugin.getPath(pluginConfigItem.package);
31+
}
32+
delete pluginConfigItem.package;
33+
configObj[pluginName] = pluginConfigItem;
34+
}
35+
}
36+
this.configurationHandler.setConfig(env, {
37+
plugin: configObj
38+
});
1939
}
2040
}
2141

src/plugin/base.ts

+18-9
Original file line numberDiff line numberDiff line change
@@ -3,23 +3,32 @@ import path from 'path';
33
type PluginMap = Map<string, BasePlugin>;
44

55
export class BasePlugin implements Plugin {
6+
static getPath(packageName: string): string {
7+
return path.resolve(require.resolve(`${packageName}/package.json`), '..');
8+
}
9+
610
public name: string;
711
public enable: boolean;
8-
public importPath: string;
12+
public importPath: string = '';
913
public metadata: Partial<PluginMetadata> = {};
1014
public metaFilePath: string = '';
1115

1216
constructor(name: string, configItem: PluginConfigItem) {
1317
this.name = name;
14-
let importPath = configItem.path ?? '';
15-
if (configItem.package) {
16-
importPath = path.resolve(require.resolve(`${configItem.package}/package.json`), '..');
17-
}
18-
if (!importPath) {
19-
throw new Error(`Plugin ${name} need have path or package field`);
20-
}
21-
this.importPath = importPath;
2218
this.enable = configItem.enable ?? false;
19+
if (this.enable) {
20+
let importPath = configItem.path ?? '';
21+
if (configItem.package) {
22+
if (importPath) {
23+
throw new Error(`plugin ${name} config error, package and path can't be set at the same time.`);
24+
}
25+
importPath = BasePlugin.getPath(configItem.package);
26+
}
27+
if (!importPath) {
28+
throw new Error(`Plugin ${name} need have path or package field`);
29+
}
30+
this.importPath = importPath;
31+
}
2332
}
2433

2534
async init() { }

src/plugin/impl.ts

+3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,9 @@ import { exisis } from '../utils/fs';
55

66
export class ArtusPlugin extends BasePlugin {
77
async init() {
8+
if (!this.enable) {
9+
return;
10+
}
811
await this.checkAndLoadMetadata();
912
if (!this.metadata) {
1013
throw new Error(`${this.name} is not have metadata.`);

src/scanner/scan.ts

+52-23
Original file line numberDiff line numberDiff line change
@@ -10,20 +10,19 @@ import {
1010
DEFAULT_LOADER_LIST_WITH_ORDER,
1111
LOADER_NAME_META,
1212
} from '../constant';
13-
import { Manifest, ManifestItem } from '../loader';
13+
import { LoaderFactory, Manifest, ManifestItem } from '../loader';
1414
import { ScannerOptions, WalkOptions } from './types';
1515
import ConfigurationHandler, { ConfigObject } from '../configuration';
16-
import { ConfigLoader } from '../loader/impl';
1716
import { FrameworkConfig, FrameworkHandler } from '../framework';
1817
import { BasePlugin, PluginFactory } from '../plugin';
1918
import { ScanUtils } from './utils';
2019

2120
export class Scanner {
2221
private moduleExtensions = ['.js', '.json', '.node'];
2322
private options: ScannerOptions;
24-
private itemMap: Map<string, ManifestItem[]>;
25-
private configList: ConfigObject[];
26-
private configHandle: ConfigurationHandler;
23+
private itemMap: Map<string, ManifestItem[]> = new Map();
24+
private tmpConfigStore: Map<string, ConfigObject[]> = new Map();
25+
private configHandle: ConfigurationHandler = new ConfigurationHandler();
2726

2827
constructor(options: Partial<ScannerOptions> = {}) {
2928
this.options = {
@@ -36,7 +35,9 @@ export class Scanner {
3635
excluded: DEFAULT_EXCLUDES.concat(options.excluded ?? []),
3736
extensions: [...new Set(this.moduleExtensions.concat(options.extensions ?? [], ['.yaml']))],
3837
};
38+
}
3939

40+
private async initItemMap(): Promise<void> {
4041
this.itemMap = new Map(
4142
this.options.loaderListGenerator(DEFAULT_LOADER_LIST_WITH_ORDER).map(loaderNameOrClazz => {
4243
if (typeof loaderNameOrClazz === 'string') {
@@ -50,8 +51,6 @@ export class Scanner {
5051
return [loaderName, []];
5152
})
5253
);
53-
this.configList = [];
54-
this.configHandle = new ConfigurationHandler();
5554
}
5655

5756
private async scanEnvList(root: string): Promise<string[]> {
@@ -87,6 +86,9 @@ export class Scanner {
8786
}
8887

8988
private async scanManifestByEnv(root: string, env: string): Promise<Manifest> {
89+
// 0. init clean itemMap
90+
await this.initItemMap();
91+
9092
const config = await this.getAllConfig(root, env);
9193

9294
// 1. scan all file in framework
@@ -95,8 +97,12 @@ export class Scanner {
9597
await this.walk(frameworkDir, this.formatWalkOptions('framework', frameworkDir));
9698
}
9799

100+
98101
// 2. scan all file in plugin
99-
this.configList.forEach(config => this.configHandle.setConfig(env, config));
102+
if (this.tmpConfigStore.has(env)) {
103+
const configList = this.tmpConfigStore.get(env) ?? [];
104+
configList.forEach(config => this.configHandle.setConfig(env, config));
105+
}
100106
const { plugin } = this.configHandle.getMergedConfig(env);
101107
const pluginSortedList = await PluginFactory.createFromConfig(plugin || {});
102108
for (const plugin of pluginSortedList.reverse()) {
@@ -134,28 +140,48 @@ export class Scanner {
134140
});
135141
}
136142

137-
private async getAllConfig(root: string, env: string) {
138-
const configDir = this.getConfigDir(root, this.options.configDir);
143+
private async getAllConfig(baseDir: string, env: string) {
144+
const configDir = this.getConfigDir(baseDir, this.options.configDir);
139145
if (!configDir) {
140146
return {};
141147
}
142-
const configFileList = await fs.readdir(path.resolve(root, configDir));
148+
const root = path.resolve(baseDir, configDir);
149+
const configFileList = await fs.readdir(root);
143150
const container = new Container(ArtusInjectEnum.DefaultContainerName);
144151
container.set({ type: ConfigurationHandler });
145-
const configHandler = new ConfigLoader(container);
146-
for (const pluginConfigFile of configFileList) {
147-
const extname = path.extname(pluginConfigFile);
148-
if (ScanUtils.isExclude(pluginConfigFile, extname, this.options.excluded, this.options.extensions)) {
149-
continue;
152+
const loaderFactory = LoaderFactory.create(container);
153+
const configItemList: (ManifestItem|null)[] = await Promise.all(configFileList.map(async filename => {
154+
const extname = path.extname(filename);
155+
if (ScanUtils.isExclude(filename, extname, this.options.excluded, this.options.extensions)) {
156+
return null;
150157
}
151-
await configHandler.load({
152-
path: path.join(root, configDir, pluginConfigFile),
153-
extname: extname,
154-
filename: pluginConfigFile,
158+
let loader = await loaderFactory.findLoaderName({
159+
filename,
160+
baseDir,
161+
root,
162+
configDir
155163
});
164+
if (loader === 'framework-config') {
165+
// SEEME: framework-config is a special loader, cannot be used when scan, need refactor later
166+
loader = 'config';
167+
}
168+
return {
169+
path: path.resolve(root, filename),
170+
extname,
171+
filename,
172+
loader,
173+
source: 'config',
174+
};
175+
}));
176+
await loaderFactory.loadItemList(configItemList.filter(v => v) as ManifestItem[]);
177+
const configurationHandler = container.get(ConfigurationHandler);
178+
const config = configurationHandler.getMergedConfig(env);
179+
let configList = [config];
180+
if (this.tmpConfigStore.has(env)) {
181+
// equal unshift config to configList
182+
configList = configList.concat(this.tmpConfigStore.get(env) ?? []);
156183
}
157-
const config = container.get(ConfigurationHandler).getMergedConfig(env);
158-
this.configList.unshift(config);
184+
this.tmpConfigStore.set(env, configList);
159185
return config;
160186
}
161187

@@ -213,7 +239,10 @@ export class Scanner {
213239
items = items.concat(unitItems);
214240
}
215241
relative && items.forEach(item => (item.path = path.relative(appRoot, item.path)));
216-
return items;
242+
return items.filter(item => (
243+
// remove PluginConfig to avoid re-merge on application running
244+
item.loader !== 'plugin-config'
245+
));
217246
}
218247

219248
private async writeFile(filename: string = 'manifest.json', data: string) {

test/fixtures/app_koa_with_ts/src/config/plugin.default.ts

+4
Original file line numberDiff line numberDiff line change
@@ -9,4 +9,8 @@ export default {
99
enable: false,
1010
path: path.resolve(__dirname, '../mysql_plugin')
1111
},
12+
testDuplicate: {
13+
enable: false,
14+
package: 'unimportant-package'
15+
},
1216
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
import path from 'path';
2+
3+
export default {
4+
testDuplicate: {
5+
enable: true,
6+
path: path.resolve(__dirname, '../test_duplicate_plugin')
7+
},
8+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
name: testDuplicate

test/scanner.test.ts

+12-4
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,13 @@ describe('test/scanner.test.ts', () => {
88
const scanner = new Scanner({ needWriteFile: false, extensions: ['.ts', '.js', '.json'] });
99
const scanResults = await scanner.scan(path.resolve(__dirname, './fixtures/app_koa_with_ts'));
1010
const { default: manifest } = scanResults;
11-
expect(Object.entries(scanResults).length).toBe(1);
11+
expect(Object.entries(scanResults).length).toBe(2);
1212
expect(manifest).toBeDefined();
1313
expect(manifest.items).toBeDefined();
1414
// console.log('manifest', manifest);
15-
expect(manifest.items.length).toBe(12);
15+
expect(manifest.items.length).toBe(11);
1616

17-
expect(manifest.items.filter(item => item.loader === 'plugin-config').length).toBe(1);
17+
expect(manifest.items.filter(item => item.loader === 'plugin-config').length).toBe(0);
1818
expect(manifest.items.filter(item => item.loader === 'plugin-meta').length).toBe(1);
1919
expect(manifest.items.filter(item => item.loader === 'exception').length).toBe(1);
2020
expect(manifest.items.filter(item => item.loader === 'lifecycle-hook-unit').length).toBe(2);
@@ -23,7 +23,15 @@ describe('test/scanner.test.ts', () => {
2323

2424
expect(manifest.items.filter(item => item.unitName === 'redis').length).toBe(2);
2525
expect(manifest.items.filter(item => item.unitName === 'mysql').length).toBe(0);
26-
expect(manifest.items.filter(item => item.source === 'app').length).toBe(10);
26+
expect(manifest.items.filter(item => item.source === 'app').length).toBe(9);
27+
28+
const { dev: devManifest } = scanResults;
29+
// console.log('devManifest', devManifest);
30+
expect(devManifest).toBeDefined();
31+
expect(devManifest.items).toBeDefined();
32+
expect(devManifest.items.length).toBe(12);
33+
expect(devManifest.items.filter(item => item.loader === 'plugin-meta').length).toBe(2);
34+
expect(devManifest.items.find(item => item.unitName === 'testDuplicate')).toBeDefined();
2735
});
2836

2937
it('should scan module with custom loader', async () => {

0 commit comments

Comments
 (0)