refactor: 重新设计build加载方式

This commit is contained in:
wanchun 2022-05-11 17:50:20 +08:00
parent e72b7d2665
commit 0580369042
18 changed files with 114 additions and 117 deletions

2
.gitignore vendored
View File

@ -6,6 +6,7 @@
.cache .cache
.temp .temp
.hound .hound
.fes
dist dist
*.log *.log
node_modules node_modules
@ -14,7 +15,6 @@ npm-debug.log
/packages/fes-doc/docs/.vuepress/dist /packages/fes-doc/docs/.vuepress/dist
/packages/fes-template/package-lock.json /packages/fes-template/package-lock.json
/.changelog /.changelog
/packages/*/lib /packages/*/lib
/packages/*/es /packages/*/es
/packages/*/dist /packages/*/dist

View File

@ -3,6 +3,7 @@ module.exports = {
'create-fes-app', 'create-fes-app',
'fes', 'fes',
'fes-compiler', 'fes-compiler',
'fes-preset-built-in',
'fes-build-vite', 'fes-build-vite',
'fes-build-webpack', 'fes-build-webpack',
'fes-runtime', 'fes-runtime',
@ -20,7 +21,6 @@ module.exports = {
'fes-plugin-sass', 'fes-plugin-sass',
'fes-plugin-vuex', 'fes-plugin-vuex',
'fes-plugin-pinia', 'fes-plugin-pinia',
'fes-preset-built-in',
'fes-plugin-windicss', 'fes-plugin-windicss',
], ],
copy: [], copy: [],

View File

@ -1,6 +1,7 @@
export default function () { export default function () {
return { return {
plugins: [ plugins: [
require.resolve('./registerBuilder'),
require.resolve('./registerMethods'), require.resolve('./registerMethods'),
require.resolve('./registerType'), require.resolve('./registerType'),

View File

@ -0,0 +1,5 @@
export default function (api) {
api.registerBuilder({
name: 'vite',
});
}

View File

@ -1,6 +1,8 @@
export default function () { export default function () {
return { return {
plugins: [ plugins: [
require.resolve('./plugins/registerBuilder'),
// register methods // register methods
require.resolve('./plugins/registerMethods'), require.resolve('./plugins/registerMethods'),
require.resolve('./plugins/registerType'), require.resolve('./plugins/registerType'),

View File

@ -0,0 +1,5 @@
export default function (api) {
api.registerBuilder({
name: 'webpack',
});
}

View File

@ -9,7 +9,7 @@ import { AsyncSeriesWaterfallHook } from 'tapable';
import { existsSync } from 'fs'; import { existsSync } from 'fs';
import { lodash, chalk } from '@fesjs/utils'; import { lodash, chalk } from '@fesjs/utils';
import { Command, Option } from 'commander'; import { Command, Option } from 'commander';
import { resolvePresets, filterBuilder, pathToObj, resolvePlugins } from './utils/pluginUtils'; import { resolvePresets, pathToObj, resolvePlugins } from './utils/pluginUtils';
import loadDotEnv from './utils/loadDotEnv'; import loadDotEnv from './utils/loadDotEnv';
import isPromise from './utils/isPromise'; import isPromise from './utils/isPromise';
import BabelRegister from './babelRegister'; import BabelRegister from './babelRegister';
@ -38,6 +38,9 @@ export default class Service extends EventEmitter {
// including plugins // including plugins
plugins = {}; plugins = {};
// 构建
builder = {};
// plugin methods // plugin methods
pluginMethods = {}; pluginMethods = {};
@ -89,11 +92,6 @@ export default class Service extends EventEmitter {
this.env = opts.env || process.env.NODE_ENV; this.env = opts.env || process.env.NODE_ENV;
this.fesPkg = opts.fesPkg || {}; this.fesPkg = opts.fesPkg || {};
const builderPkgPath = filterBuilder(this.pkg);
this.builder = {
isVite: (builderPkgPath[0] || '').includes('build-vite'),
};
assert(existsSync(this.cwd), `cwd ${this.cwd} does not exist.`); assert(existsSync(this.cwd), `cwd ${this.cwd} does not exist.`);
// register babel before config parsing // register babel before config parsing

View File

@ -21,21 +21,14 @@ export default class PluginAPI {
} }
// TODO: reversed keys // TODO: reversed keys
describe({ describe({ id, key, config, enableBy } = {}) {
id,
key,
config,
enableBy
} = {}) {
const { plugins } = this.service; const { plugins } = this.service;
// this.id and this.key is generated automatically // this.id and this.key is generated automatically
// so we need to diff first // so we need to diff first
if (id && this.id !== id) { if (id && this.id !== id) {
if (plugins[id]) { if (plugins[id]) {
const name = plugins[id].isPreset ? 'preset' : 'plugin'; const name = plugins[id].isPreset ? 'preset' : 'plugin';
throw new Error( throw new Error(`api.describe() failed, ${name} ${id} is already registered by ${plugins[id].path}.`);
`api.describe() failed, ${name} ${id} is already registered by ${plugins[id].path}.`
);
} }
plugins[id] = plugins[this.id]; plugins[id] = plugins[this.id];
plugins[id].id = id; plugins[id].id = id;
@ -55,54 +48,35 @@ export default class PluginAPI {
} }
register(hook) { register(hook) {
assert( assert(hook.key && typeof hook.key === 'string', `api.register() failed, hook.key must supplied and should be string, but got ${hook.key}.`);
hook.key && typeof hook.key === 'string', assert(hook.fn && typeof hook.fn === 'function', `api.register() failed, hook.fn must supplied and should be function, but got ${hook.fn}.`);
`api.register() failed, hook.key must supplied and should be string, but got ${hook.key}.` this.service.hooksByPluginId[this.id] = (this.service.hooksByPluginId[this.id] || []).concat(hook);
);
assert(
hook.fn && typeof hook.fn === 'function',
`api.register() failed, hook.fn must supplied and should be function, but got ${hook.fn}.`
);
this.service.hooksByPluginId[this.id] = (
this.service.hooksByPluginId[this.id] || []
).concat(hook);
} }
registerCommand(commandOption) { registerCommand(commandOption) {
const { command, fn } = commandOption; const { command, fn } = commandOption;
assert( assert(!this.service.commands[command], `api.registerCommand() failed, the command ${command} is exists.`);
!this.service.commands[command], assert(typeof command === 'string', 'api.registerCommand() failed, the command must be string.');
`api.registerCommand() failed, the command ${command} is exists.` assert(typeof fn === 'function', 'api.registerCommand() failed, the fn must be function.');
);
assert(
typeof command === 'string',
'api.registerCommand() failed, the command must be String.'
);
assert(
typeof fn === 'function',
'api.registerCommand() failed, the fn must be function.'
);
this.service.commands[command] = commandOption; this.service.commands[command] = commandOption;
} }
// 在 preset 初始化阶段放后面,在插件注册阶段放前面 // 在 preset 初始化阶段放后面,在插件注册阶段放前面
registerPlugins(plugins) { registerPlugins(plugins) {
assert( assert(
this.service.stage === ServiceStage.initPresets this.service.stage === ServiceStage.initPresets || this.service.stage === ServiceStage.initPlugins,
|| this.service.stage === ServiceStage.initPlugins, 'api.registerPlugins() failed, it should only be used in registering stage.',
'api.registerPlugins() failed, it should only be used in registering stage.'
); );
assert( assert(Array.isArray(plugins), 'api.registerPlugins() failed, plugins must be Array.');
Array.isArray(plugins), const extraPlugins = plugins.map((plugin) =>
'api.registerPlugins() failed, plugins must be Array.' isValidPlugin(plugin)
); ? plugin
const extraPlugins = plugins.map(plugin => (isValidPlugin(plugin)
? (plugin)
: pathToObj({ : pathToObj({
type: PluginType.plugin, type: PluginType.plugin,
path: plugin, path: plugin,
cwd: this.service.cwd cwd: this.service.cwd,
}))); }),
);
if (this.service.stage === ServiceStage.initPresets) { if (this.service.stage === ServiceStage.initPresets) {
this.service._extraPlugins.push(...extraPlugins); this.service._extraPlugins.push(...extraPlugins);
} else { } else {
@ -111,52 +85,51 @@ export default class PluginAPI {
} }
registerPresets(presets) { registerPresets(presets) {
assert( assert(this.service.stage === ServiceStage.initPresets, 'api.registerPresets() failed, it should only used in presets.');
this.service.stage === ServiceStage.initPresets, assert(Array.isArray(presets), 'api.registerPresets() failed, presets must be Array.');
'api.registerPresets() failed, it should only used in presets.' const extraPresets = presets.map((preset) =>
); isValidPlugin(preset)
assert( ? preset
Array.isArray(presets),
'api.registerPresets() failed, presets must be Array.'
);
const extraPresets = presets.map(preset => (isValidPlugin(preset)
? (preset)
: pathToObj({ : pathToObj({
type: PluginType.preset, type: PluginType.preset,
path: preset, path: preset,
cwd: this.service.cwd cwd: this.service.cwd,
}))); }),
);
// 插到最前面,下个 while 循环优先执行 // 插到最前面,下个 while 循环优先执行
this.service._extraPresets.splice(0, 0, ...extraPresets); this.service._extraPresets.splice(0, 0, ...extraPresets);
} }
registerMethod({ registerMethod({ name, fn, exitsError = true }) {
name,
fn,
exitsError = true
}) {
if (this.service.pluginMethods[name]) { if (this.service.pluginMethods[name]) {
if (exitsError) { if (exitsError) {
throw new Error( throw new Error(`api.registerMethod() failed, method ${name} is already exist.`);
`api.registerMethod() failed, method ${name} is already exist.`
);
} else { } else {
return; return;
} }
} }
this.service.pluginMethods[name] = fn this.service.pluginMethods[name] =
fn ||
// 这里不能用 arrow functionthis 需指向执行此方法的 PluginAPI // 这里不能用 arrow functionthis 需指向执行此方法的 PluginAPI
// 否则 pluginId 会不会,导致不能正确 skip plugin // 否则 pluginId 会不会,导致不能正确 skip plugin
|| function (hookFn) { function (hookFn) {
const hook = { const hook = {
key: name, key: name,
...(utils.lodash.isPlainObject(hookFn) ? hookFn : { fn: hookFn }) ...(utils.lodash.isPlainObject(hookFn) ? hookFn : { fn: hookFn }),
}; };
// @ts-ignore // @ts-ignore
this.register(hook); this.register(hook);
}; };
} }
registerBuilder(builder) {
assert(typeof builder === 'object', 'api.registerBuilder() failed, the builder must be object.');
const { name } = builder;
assert(typeof name === 'string', 'api.registerBuilder() failed, the builder.name must be string.');
assert(typeof this.service.builder.name !== 'string', `检测到您使用了 builder: ${name},已经加载 builder: ${this.service.builder.name} 请保留一个`);
this.service.builder = builder;
}
skipPlugins(pluginIds) { skipPlugins(pluginIds) {
pluginIds.forEach((pluginId) => { pluginIds.forEach((pluginId) => {
this.service.skipPluginIds.add(pluginId); this.service.skipPluginIds.add(pluginId);

View File

@ -1,14 +1,11 @@
import { dirname, join, basename, relative, extname } from 'path'; import { dirname, join, basename, relative, extname } from 'path';
import { compatESModuleRequire, resolve, winPath, pkgUp, lodash } from '@fesjs/utils'; import { compatESModuleRequire, resolve, winPath, pkgUp, lodash } from '@fesjs/utils';
import Logger from '../../logger';
import { PluginType } from '../enums'; import { PluginType } from '../enums';
const logger = new Logger('fes:compiler');
const RE = { const RE = {
[PluginType.plugin]: /^(@fesjs\/|@webank\/fes-|fes-)plugin-/, [PluginType.plugin]: /^(@fesjs\/|@webank\/fes-|fes-)plugin-/,
[PluginType.preset]: /^(@fesjs\/|@webank\/fes-|fes-)preset-/, [PluginType.preset]: /^(@fesjs\/|@webank\/fes-|fes-)(preset|build)-/,
}; };
export function isPluginOrPreset(type, name) { export function isPluginOrPreset(type, name) {
@ -26,18 +23,6 @@ function filterPluginAndPreset(type, pkg) {
.filter(isPluginOrPreset.bind(null, type)); .filter(isPluginOrPreset.bind(null, type));
} }
export function filterBuilder(pkg) {
const builders = Object.keys(pkg.devDependencies || {})
.concat(Object.keys(pkg.dependencies || {}))
.filter((name) => /^@fesjs\/build-/.test(name));
if (builders.length > 1) {
logger.warn(`检测到您使用了多个个 builder: ${builders},当前生效的是 ${builders[0]}, 请保留一个`);
}
return builders.slice(0, 1);
}
export function getPluginsOrPresets(type, opts) { export function getPluginsOrPresets(type, opts) {
const upperCaseType = type.toUpperCase(); const upperCaseType = type.toUpperCase();
return [ return [
@ -46,8 +31,6 @@ export function getPluginsOrPresets(type, opts) {
// env // env
...(process.env[`FES_${upperCaseType}S`] || '').split(',').filter(Boolean), ...(process.env[`FES_${upperCaseType}S`] || '').split(',').filter(Boolean),
...filterPluginAndPreset(type, opts.pkg), ...filterPluginAndPreset(type, opts.pkg),
// 构建只允许是 presets
...(type === PluginType.preset ? filterBuilder(opts.pkg) : []),
// user config // user config
...(opts[type === PluginType.preset ? 'userConfigPresets' : 'userConfigPlugins'] || []), ...(opts[type === PluginType.preset ? 'userConfigPresets' : 'userConfigPlugins'] || []),
].map((path) => ].map((path) =>
@ -122,13 +105,29 @@ export function pathToObj({ path, type, cwd }) {
export function resolvePresets(opts) { export function resolvePresets(opts) {
const type = PluginType.preset; const type = PluginType.preset;
const presets = [...getPluginsOrPresets(type, opts)]; const presets = [...getPluginsOrPresets(type, opts)];
return presets.map((path) => return presets
.map((path) =>
pathToObj({ pathToObj({
type, type,
path, path,
cwd: opts.cwd, cwd: opts.cwd,
}), }),
); )
.sort((a, b) => {
if (a.id === '@fesjs/preset-built-in') {
return -1;
}
if (b.id === '@fesjs/preset-built-in') {
return 1;
}
if (/^(@fesjs\/|@webank\/fes-|fes-)build-/.test(a.id)) {
return -1;
}
if (/^(@fesjs\/|@webank\/fes-|fes-)build-/.test(b.id)) {
return 1;
}
return 0;
});
} }
export function resolvePlugins(opts) { export function resolvePlugins(opts) {

View File

@ -74,7 +74,7 @@ export default (api) => {
api.addRuntimePlugin(() => `@@/${absRuntimeFilePath}`); api.addRuntimePlugin(() => `@@/${absRuntimeFilePath}`);
if (api.builder.isVite) { if (api.builder.name === 'vite') {
api.modifyBundleConfig((config) => { api.modifyBundleConfig((config) => {
const monacoEditorPlugin = require('vite-plugin-monaco-editor').default; const monacoEditorPlugin = require('vite-plugin-monaco-editor').default;
config?.plugins?.push(monacoEditorPlugin(api.config?.monacoEditor || {})); config?.plugins?.push(monacoEditorPlugin(api.config?.monacoEditor || {}));

View File

@ -19,7 +19,7 @@ export default function (api) {
enableBy: () => isSlaveEnable(api), enableBy: () => isSlaveEnable(api),
}); });
if (api.builder.isVite) { if (api.builder.name === 'vite') {
// 处理 // 处理
} else { } else {
api.modifyDefaultConfig((memo) => { api.modifyDefaultConfig((memo) => {

View File

@ -17,7 +17,7 @@ export default (api) => {
}, },
}); });
if (api.builder.isVite) { if (api.builder.name === 'vite') {
// vite 不需要处理 // vite 不需要处理
} else { } else {
api.chainWebpack((memo, { createCSSRule }) => { api.chainWebpack((memo, { createCSSRule }) => {

View File

@ -61,7 +61,7 @@ export default (api) => {
api.addEntryImportsAhead(() => [{ source: 'windi-base.css' }, { source: 'windi-components.css' }, { source: 'windi-utilities.css' }]); api.addEntryImportsAhead(() => [{ source: 'windi-base.css' }, { source: 'windi-components.css' }, { source: 'windi-utilities.css' }]);
if (api.builder.isVite) { if (api.builder.name === 'vite') {
buildWindicssWithVite(api); buildWindicssWithVite(api);
} else { } else {
buildWindicssWithWebpack(api); buildWindicssWithWebpack(api);

View File

@ -24,6 +24,7 @@ export default function () {
require.resolve('./plugins/features/mock'), require.resolve('./plugins/features/mock'),
require.resolve('./plugins/features/outputPath'), require.resolve('./plugins/features/outputPath'),
require.resolve('./plugins/features/plugins'), require.resolve('./plugins/features/plugins'),
require.resolve('./plugins/features/presets'),
require.resolve('./plugins/features/proxy'), require.resolve('./plugins/features/proxy'),
require.resolve('./plugins/features/publicPath'), require.resolve('./plugins/features/publicPath'),
require.resolve('./plugins/features/singular'), require.resolve('./plugins/features/singular'),

View File

@ -0,0 +1,10 @@
export default (api) => {
api.describe({
key: 'presets',
config: {
schema(joi) {
return joi.array().items(joi.string());
},
},
});
};

View File

@ -77,6 +77,7 @@ export interface InnerBuildConfig {
}; };
mountElementId?: string; mountElementId?: string;
plugins?: string[]; plugins?: string[];
presets?: string[];
proxy?: { proxy?: {
[apiPrefix: string]: { [apiPrefix: string]: {
target: string; target: string;

View File

@ -91,5 +91,8 @@ export default {
dynamicImport: true, dynamicImport: true,
monacoEditor: { monacoEditor: {
languages: ['javascript', 'typescript', 'html', 'json'] languages: ['javascript', 'typescript', 'html', 'json']
} },
presets: [
require.resolve('../fes-build-webpack/lib'),
]
}; };

View File

@ -58,7 +58,6 @@
"@fesjs/plugin-windicss": "^2.0.0", "@fesjs/plugin-windicss": "^2.0.0",
"@fesjs/plugin-pinia": "^2.0.0", "@fesjs/plugin-pinia": "^2.0.0",
"@fesjs/fes-design": "^0.3.3", "@fesjs/fes-design": "^0.3.3",
"@fesjs/build-webpack": "^1.0.0",
"vue": "^3.0.5", "vue": "^3.0.5",
"vuex": "^4.0.0", "vuex": "^4.0.0",
"pinia": "^2.0.11" "pinia": "^2.0.11"