阅读之前你需要知道的知识包括
随着项目不断迭代,原有的项目体积在不断增大。伴随而来的是功能和依赖数量的快速增长。这使得整体项目将越来越难以维护。 并且如果在一个基础旧的功能上进行更新,我们又希望能做到最小代价的开发、测试和构建的话,那么将原有的单体架构拆分成更小的单元,这将是势在必行的。 本篇文章将会以这个项目的迁移为例,讲解整个迁移过程中的思考和实现。
├── src│ ├── assets│ │ ├── fonts│ │ ├── lib # 需要改动开源包│ │ │ ├── redux-undo│ │ │ └── ruler│ │ └── style│ │ └── datePicker│ ├── components│ │ ├── base │ │ ├── common # 系统中的通用组件,包括alert,dialog等│ │ ├── comps # 系统中的展示数据的组件│ │ │ ├── codeFragment│ │ │ ├── commonTitle│ │ │ ├── datasource│ │ │ ├── echarts│ │ │ ├── group│ │ │ └── _template_│ │ ├── form # 系统中的表单组件│ │ └── recursion # 系统中的表单组件生成器│ │ ├── echarts│ │ └── widget│ ├── configurableComponents # 配置化的组件,通用的表单和系统UI主题│ │ ├── form│ │ └── theme│ │ └── overrides│ ├── helpers # 工具函数,数据解析、后端交互│ ├── page # 页面结构│ │ ├── canvas│ │ └── editor│ │ ├── Header│ │ ├── LeftPanel│ │ ├── RightPanel│ │ └── User│ ├── service # 数据模型及处理│ ├── store # 前端状态管理及本地数据持久化│ │ ├── DB│ │ ├── features│ │ │ └── appSlice│ │ └── reducers│ ├── __test__ # 单元测试│ │ ├── components│ │ │ └── recursion│ │ └── utils│ │ └── MockData│ ├── @types # 公共类型及包类型overwrite│ └── utils # 工具函数
从目录中看,需要抽离的功能包含:
从以下角度出发:
这里以系统中Notice组件为例。
Notice组件,顾名思义,这是用来处理系统中所有弹出式通知的组件。只要用户的操作行为,在业务上被定义为需要告知给用户的,都会使用它对消息内容进行展示。它被系统大部分功能依赖,例如现有目录中的:
表单组件生成器
数据模型及处理
前端状态管理
部分工具函数
如果对该组件进行抽离,可以预见的是会产生大量的文件修改和测试部分重写。即使花费如此高昂的代价,也要对这类组件进行抽离,对这种行为,我一般遵循这几个原则:
这里着重说一2和3。
2中提到的 功能可预见较为频繁的修改 我们可以从notice组件的迭代得到答案。
例如,目前系统提供的notice只是提供了单纯的消息展示,并且它的消失时机是几个可选择的常量。如果说后续出现一个消失时机来自不同组件的hook或其他事件的需求。这些碎片化的需求,可能就会使得notice频繁进行发版。
进行拆分之后,我们不必因为某个小功能,对整体系统重新构建。而只需要构建单个功能。
同时结合 3之后,我们对此类功能,抛弃传统的webpack将所有依赖打包成同一个bundle(这里先不讨论 async import),无论是使用script + ESM
还是cjs + new Function
的模式,我们都能在不进行大规模系统构建的前提下,完成对某个功能更新
这是为了让抽离后的项目获得下面的特性:
.├── common # 通用组件│ ├── codeEditor│ ├── dynamicImport│ ├── notice│ ├── recursion│ └── theme├── core # 项目主入口│ ├── config│ ├── public│ ├── scripts│ └── src├── dataComp # 需要热更新和运行时存在多版本的组件│ ├── codeFragment│ ├── commonTitle│ ├── datasource│ ├── echarts│ └── group└── workspace └── devServer # 本地开发命令集
├── package.json├── pnpm-lock.yaml├── README.md├── src│ ├── index.tsx│ └── lib.d.ts└── tsconfig.json
├── package.json├── pnpm-lock.yaml├── src├── tsconfig.json└── webpack.config.js # 可能用到的独特的配置,将会被merge到运行的webpackconfig中
当目录重构后,我们需要对原有的构建流程进行改造。由于项目原本是通过create-react-app
这个命令创建的,但是后续的修改,不可避免的会对webpack的配置产生大量的修改,所以第一步我们需要运行 eject
命令,使得我们可以续改项目的构建配置。 ### 构建模式的选择 —— 单独打包 or 统一打包 运行eject后,我们需要确定对于抽离后的项目,是选择每个项目都配置单独的构建流程,还是使用一个通用的构建方式。
这里出于下面点考虑,我选择了使用通用构建方式:
开发成本低,不必为每个抽离后的项目,开发单独的webpack配置
原本是从单体项目中抽离的,所以构建流程大体一致
更容易控制依赖的版本
同时,我将 /core
目录称之为主项目,它将提供所有项目的构建配置。
common
部分):使用pnpm link
创建抽离项目的软连接并引入dataComp
部分):调用主项目的webpack进行构建后,通过本地的dev服务发送静态资源文件(bundle.js),在主项目中使用 new Function
的方式引入pnpm install
, 以module的方式引入。new Function
的方式引入确定构建模式后,为了提供良好的开发体验。我们仍然期望,抽离后的代码在开发环境出现更新时,项目仍能提供局部热更新的能力。
为了实现这个需求,我们需要:
link
命令,创建一个基于软连接的本地依赖。实现1,需要完成向webpack添加: 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
/**
* @returns {string[]}
*/
const resolveAppsRoot = () => {
const commonPath = path.resolve('../common');
const deps = fs.readdirSync(path.resolve('../common'));
return deps.map((pathVal) => {
return path.join(commonPath, pathVal);
});
};
/**
*
* @param {string[]} mainModulesPath
* @returns {string[]}
*/
const getValidCompilerModulesPath = (mainModulesPath) => {
const appsRoot = resolveAppsRoot();
return mainModulesPath.concat(
appsRoot.reduce((curr, rootPath) => {
return [...curr, path.join(rootPath, 'src'), path.join(rootPath, 'node_modules')];
}, [])
);
};
/**
* @param {string} appSrc
* @returns {string[]}
*/
const resolveAppsSrc = (appSrc) => {
const commonPath = path.resolve('../common');
const deps = fs.readdirSync(path.resolve('../common'));
return [
appSrc,
...deps.map((pathVal) => {
return path.join(commonPath, pathVal, 'src');
}),
];
};
// webpack config modify
{
//...
resolve: {
//...
modules: getValidCompilerModulesPath(
['node_modules', paths.appNodeModules].concat(modules.additionalModulePaths || [])
),
//...
},
//...
module:{
//...
rules: [
//...
{
test: /\.(js|mjs|jsx|ts|tsx)$/,
include: isEnvDevelopment ? resolveAppsSrc(pathsappSrc) : paths.appSrc,
loader: require.resolve('babel-loader'),
//...
}
],
//...
}
//...
}
module.rules
的修改, 扩展了webpack的构建范围,可以使用主项目中的babel(支持typescript),编译抽离后的项目。resolve.modules
的修改,是让主项目webpack可以解析抽离后项目的独立依赖。resolveAppsSrc
函数处理抽离后项目的需要编译的路径getValidCompilerModulesPath
数处理抽离后项目的依赖需要编译的路径step1: 在项目的根目录下创建pnpm-workspace.yaml
和 package.json
。
step2: 然后在根目录运行如下命令: 1
pnpm add -w react react-dom
common/recursion
)目录中运行
1
pnpm add --save-peer react react-dom
1
2
3
4
5
6
7
8
9
10
11
12{
//...
resolve: {
//...
alias: {
//...
react: path.resolve('../node_modules/react'),
'react-dom': path.resolve('../node_modules/react-dom'),
//...
},
}
}
逐条解释它们的作用
useContext
的依赖时,会出现undefined
的问题以formik为例,需要修改主项目的webpack 1
2
3
4
5
6
7
8
9
10
11{
//...
resolve: {
//...
alias: {
//...
formik: path.resolve('../node_modules/formik'),
//...
},
}
}ModuleScopePlugin
,来放开对于依赖范围的检测
我们在开发中,经常会遇到引用一些公共函数的需求,但是,如果引用的层级太深,难免会出现形如 import moduleFunc from '../../../../../utils/getData'
的路径。这样的路径,可读性差,并且如果出现整体目录迁移并且引用该功能的文件非常,会使得这些文件都出现修改。
所以一般我们都会使用形如import moduleFunc from @utils/getData;
的方式进行优化。
但如果在抽离后的项目使用这样的特性,需要对主项目的webpack再做一些更改,这是因为由于构建命令的执行目录在主项目下,它们的相对路径,并没有对应到抽离后的项目目录。需要对webpack做出如下修改: 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100//webpack config
{
//...
resolve: {
//...
alias: {
//...
...(getValidCompilerPaths(modules.webpackAliases) || {}),
//...
},
}
}
/**
* 获取alias真实的编译路径
* @param {Record<string,string>} alias
* @returns {Record<string,string>}
*/
const getValidCompilerPaths = (mainAlias) => {
const appsRoot = resolveAppsRoot();
return appsRoot.reduce((curr, next) => {
const alias = handleTSAlias(next);
return {
...alias,
...curr,
};
}, mainAlias);
};
/**
*
* @typedef {object} Options
* @property {string} options.baseUrl
* @property {{
* [key: string]: string[]
* }} options.paths
* 获取webpack使用的alias的绝对路径
*
* @param {Options} options
* @param {{
* rootPath: string;
* }} extension
*/
function getWebpackAliases(options = {}, extension = {}) {
const { baseUrl, paths: aliasPath } = options;
const { rootPath } = extension;
const appPath = rootPath ? rootPath : paths.appPath;
const appSrc = rootPath ? path.join(rootPath, baseUrl) : paths.appSrc;
if (!baseUrl) {
return {};
}
const baseUrlResolved = path.resolve(appPath, baseUrl);
if (path.relative(appPath, baseUrlResolved) === '') {
return {
src: appSrc,
};
}
if (isEmpty(aliasPath)) {
return {};
}
const aliasPathKeys = Object.keys(aliasPath);
const result = aliasPathKeys.reduce((prev, curr) => {
const aliasPathArr = aliasPath[curr];
return {
...prev,
[curr.replace('/*', '')]: path.resolve(appSrc, aliasPathArr[0].replace('/*', '')),
};
}, {});
console.log(result);
return result;
}
/**
* 获取对应项目的tsconfig
* @param {string} rootPath
* @returns {Record<string, string[]>}
*/
const getTSOption = (rootPath) => {
const appTsConfig = path.join(rootPath, 'tsconfig.json');
const hasTsConfig = fs.existsSync(appTsConfig);
if (!hasTsConfig) {
throw new Error('sub-project tsconfig is not exist');
}
const ts = require(resolve.sync('typescript', {
basedir: paths.appNodeModules,
}));
const config = ts.readConfigFile(appTsConfig, ts.sys.readFile).config;
return config.compilerOptions || {};
};
/**
* 处理TS中声明的alias,使得tsconfig中的alias能与webpack对应上
* @param {string} rootPath
* @returns {string}
*/
const handleTSAlias = (rootPath) => {
return getWebpackAliases(getTSOption(rootPath), { rootPath });
};
在上一个小节中,完成对alias的解析,但是,会出现ts类型错误,如:无法找到模块@utils
。这是因为上面我们只解决了,webpack的编译打包流程,但类型检查仍没有提供抽离后的项目和其目录的对应关系。
这里我们需要将类型检测的范围扩展到整体项目范围,并提供对应的文件路径关系 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46// webpack config
{
//...
plugins: {
//...
new ForkTsCheckerWebpackPlugin({
//...
typescript: {
//...
configOverwrite: {
//...
compilerOptions: {
references: getTypeCheckPaths(),
}
//...
}
//...
}
//...
}),
}
}
const getTypeCheckPaths = () => {
const appsRoot = resolveAppsRoot();
const formatPaths = (aliasPath, appRoot) => {
return Object.keys(aliasPath).reduce((curr, next) => {
const item = aliasPath[next];
return {
...curr,
[next]: path.join(appRoot, item[0]),
};
}, {});
};
const result = appsRoot.reduce((curr, next) => {
//getTSOption的实现参考上面
const { paths: aliasPath } = getTSOption(next);
if (isEmpty(aliasPath)) {
return curr;
} else {
return { ...curr, ...formatPaths(aliasPath, next) };
}
}, {});
return result;
};
上面的一系列改动,解决了抽离后整体项目的构建问题,但目前的构建方式仍然是全量进行构建,预期是通过一个可交互的command,让用户可以选择构建那些抽离后的项目。没被选中的,使用pnpm add <package-names>
,从公共仓库安装到主项目中。
dataComp
中的项目均需要提供在线热更新。
dev环境将提供一个dev server提供编译后的bundle.js
。
prod环境将使用nginx为不同项目的编译产物提供静态服务。
主项目都将使用 fetch + new Fucntion
的方式引入此类组件。
dataComp
中的项目对于不同的用户,可能同时需要存在不同的版本。这需要在它的URL信息中加入版本信息,用来加载不同版本的编译产物。