3
3
4
4
在 < https://github.com/npmstudy/your-node-v20-monorepo-project > 里,我们用到的技术栈如下。
5
5
6
- ``` markdown
7
- - [Tsup](https://tsup.egoist.dev/) as a TypeScript universal package.
8
- - [Tsx](https://github.com/esbuild-kit/tsx) as a Node.js enhanced with esbuild to run TypeScript & ESM
9
- - [Tsd](https://github.com/SamVerschueren/tsd) as type test runner
10
- - [Tsdoc](https://tsdoc.org/) as document
11
- - [PNPM](https://pnpm.io/workspaces) as workspace manager and package manager.
12
- - [Vitest](https://vitest.dev/) as a test runner.
13
- - [Size Limit](https://github.com/ai/size-limit) as a size limit plugin.
14
- - [Prettier](https://prettier.io/) as a code formatter.
15
- - [ESLint](https://eslint.org/) as a code linter.
16
- - [NX](https://nx.dev) as cacheable operations.
17
- - [Changesets](https://github.com/changesets/changesets/) as a way to manage changes and releases.
18
- - [c8](https://www.npmjs.com/package/c8) as coverage
19
- - [supertest](https://www.npmjs.com/package/supertest) as server test
20
- - [cypress](https://www.cypress.io/) as e2e test
21
- ```
6
+ > - [ Tsup] ( https://tsup.egoist.dev/ ) as a TypeScript universal package.
7
+ > - [ Tsx] ( https://github.com/esbuild-kit/tsx ) as a Node.js enhanced with esbuild to run TypeScript & ESM
8
+ > - [ Tsd] ( https://github.com/SamVerschueren/tsd ) as type test runner
9
+ > - [ Tsdoc] ( https://tsdoc.org/ ) as document
10
+ > - [ PNPM] ( https://pnpm.io/workspaces ) as workspace manager and package manager.
11
+ > - [ Vitest] ( https://vitest.dev/ ) as a test runner.
12
+ > - [ Size Limit] ( https://github.com/ai/size-limit ) as a size limit plugin.
13
+ > - [ Prettier] ( https://prettier.io/ ) as a code formatter.
14
+ > - [ ESLint] ( https://eslint.org/ ) as a code linter.
15
+ > - [ NX] ( https://nx.dev ) as cacheable operations.
16
+ > - [ Changesets] ( https://github.com/changesets/changesets/ ) as a way to manage changes and releases.
17
+ > - [ c8] ( https://www.npmjs.com/package/c8 ) as coverage
18
+ > - [ supertest] ( https://www.npmjs.com/package/supertest ) as server test
19
+ > - [ cypress] ( https://www.cypress.io/ ) as e2e test
22
20
23
21
运行TS的4个(已讲过)
24
22
46
44
4 . Prettier
47
45
5 . Changesets
48
46
49
- 其中ESLint和Prettier前端同学都熟悉,这里就细讲了 。
47
+ 其中ESLint和Prettier前端同学都熟悉,这里就不细讲了 。
50
48
51
49
- ESLint是一个用于JavaScript和JSX代码的静态代码分析工具。它可以帮助开发者在编写代码的过程中发现并修复潜在的问题,以确保代码的质量和一致性。
52
50
- Prettier是一个代码格式化工具,用于自动格式化代码以保持一致的代码风格。它支持多种编程语言,包括JavaScript、CSS、HTML、JSON等。Prettier可以帮助开发者快速、准确地格式化代码,提高代码的可读性和可维护性。它可以与其他代码检查工具(如ESLint)配合使用,以确保代码的质量和一致性。
@@ -81,15 +79,15 @@ pnpm在2018年发布,作为npm的更快速和更高效的替代品。
81
79
82
80
pnpm使用的是npm version 2.x类似的树形结构,同时使用.pnpm 以平铺的形式储存着所有的包。这里的.pnpm为虚拟存储目录,该目录通过` <package-name>@<version> ` 来实现相同模块不同版本之间隔离和复用,由于它只会根据项目中的依赖生成,并不存在提升,所以它不会存在之前提到的** Phantom dependencies** 问题!
83
81
84
- 然后使用Store + Links和文件资源进行关联。简单说pnpm把会包下载到一个公共目录 ,如果某个依赖在 store 目录中存在了话,那么就会直接从 store 目录里面去 hard-link,避免了二次安装带来的时间消耗,如果依赖在 store 目录里面不存在的话,就会去下载一次。
82
+ 然后使用Store + Links和文件资源进行关联。简单说pnpm会把包下载到一个公共目录 ,如果某个依赖在 store 目录中存在了话,那么就会直接从 store 目录里面去 hard-link,避免了二次安装带来的时间消耗,如果依赖在 store 目录里面不存在的话,就会去下载一次。
85
83
86
84
通过Store + hard link的方式,不仅解决了项目中的NPM doppelgangers问题,项目之间也不存在该问题,从而完美解决了npm3+和yarn中的包重复问题!
87
85
88
86
![ Untitled] ( img/Untitled%204.png )
89
87
90
88
pnpm除了安装速度快,节省磁盘空间,避免幽灵依赖等优化,也内置了对monorepo的支持。使用起来比较简单,在项目根目录中新建pnpm-workspace.yaml文件,并声明对应的工作区就好。
91
89
92
- ``` tsx
90
+ ``` bash
93
91
$ cat pnpm-workspace.yaml
94
92
packages:
95
93
- ' example'
@@ -99,11 +97,11 @@ packages:
99
97
100
98
剩下的就是运行时处理。之前nx run-many解决了monorepo子模块运行的问题。很多时候我们需要在根目录执行某个项目 的构建脚本,需要通过—filter或-F过滤之后,再执行,示例如下。
101
99
102
- ``` tsx
100
+ ``` bash
103
101
$ pnpm -F example dev
104
102
```
105
103
106
- 以上都是pnpm的有点 ,其实pnpm的演进还是比较快的,比如pnpm v7和pnpm v8版本差异还是比较大的,且不完全兼容。但整个社区是比较活跃的,
104
+ 以上都是pnpm的优点 ,其实pnpm的演进还是比较快的,比如pnpm v7和pnpm v8版本差异还是比较大的,且不完全兼容。但整个社区是比较活跃的,
107
105
108
106
<aside >
109
107
💡 都说Node.js和前端复杂,其实是背后工程复杂度导致的,如果没有这么多依赖,这么多版本,也不会出现幻影依赖这样的问题。其实,对用的人来说无感的,我就用pnpm v8,像其他社区一样,也没问题的。真正的问题是Node.js和前端变化太快,你在较短时间内,总可能会遇到pnpm v7的代码,甚至互相切换。
@@ -129,7 +127,7 @@ $ pnpm -F example dev
129
127
130
128
先写一个简单的Koa的Hello world。见app.js
131
129
132
- ``` tsx
130
+ ``` js
133
131
import Koa from " koa" ;
134
132
const app = new Koa ();
135
133
@@ -143,15 +141,15 @@ export default app;
143
141
144
142
再写一个测试脚本,run.js
145
143
146
- ``` tsx
144
+ ``` js
147
145
import app from " ./index.js" ;
148
146
149
147
app .listen (3000 );
150
148
```
151
149
152
150
此时,执行node run.js就可以启动服务了。然后我们看一下测试如何编写。
153
151
154
- ``` tsx
152
+ ``` js
155
153
import { expect , test } from " vitest" ;
156
154
import supertest from " supertest" ;
157
155
import app from " ./index.js" ;
@@ -167,7 +165,7 @@ test("koa app", async () => {
167
165
168
166
这个代码和我们在Node.js v20的test runner里几乎是一模一样的。通过下面命令即可运行测试。
169
167
170
- ``` tsx
168
+ ``` bash
171
169
$ npx vitest run
172
170
173
171
RUN v0.34.6 /Users/alfred/workspace/npmstudy/vitest-with-supertest
@@ -190,13 +188,13 @@ $ npx vitest run
190
188
191
189
安装
192
190
193
- ``` tsx
191
+ ``` bash
194
192
$ npm install cypress --save-dev --registry=https://registry.npmmirror.com
195
193
```
196
194
197
195
通过npx cypress open打开。
198
196
199
- ``` tsx
197
+ ``` bash
200
198
$ npx cypress open
201
199
It looks like this is your first time using Cypress: 13.3.0
202
200
@@ -217,7 +215,7 @@ DevTools listening on ws://127.0.0.1:54097/devtools/browser/dc560ea2-0aad-47a4-9
217
215
218
216
修改代码,手动执行node run.js启动服务。
219
217
220
- ``` tsx
218
+ ``` js
221
219
describe (" template spec" , () => {
222
220
it (" passes" , () => {
223
221
cy .visit (" http://127.0.0.1:3000" );
@@ -230,7 +228,7 @@ describe("template spec", () => {
230
228
231
229
为了简化操作,我们把启动服务放到cypress.config.js里。
232
230
233
- ``` tsx
231
+ ``` js
234
232
import { defineConfig } from " cypress" ;
235
233
import app from " ./index.js" ;
236
234
export default defineConfig ({
@@ -248,7 +246,7 @@ export default defineConfig({
248
246
249
247
此时,执行结果如下。
250
248
251
- ``` tsx
249
+ ``` bash
252
250
$ npx cypress run
253
251
254
252
DevTools listening on ws://127.0.0.1:61755/devtools/browser/0a2f2716-10cb-49b9-a2c9-5163469574e9
@@ -364,7 +362,7 @@ nx 会计算当前执行的 target 的 Hash 作为 cache key,在 target 执行
364
362
365
363
比如依赖如下。
366
364
367
- ``` tsx
365
+ ``` bash
368
366
$ pnpm project-graph
369
367
370
368
> your-node-v20-monorepo-project@ project-graph /Users/alfred/workspace/npmstudy/your-node-v20-monorepo-project
@@ -379,13 +377,17 @@ nx除了本地换成,其实还支持云服务(remote cache)。即nx.app,
379
377
380
378
![ Untitled] ( img/Untitled%2015.png )
381
379
382
- 在< https://github.com/npmstudy/your-node-v20-monorepo -project > 项目里,我们其实只用了非常简单的npm nx run-many
380
+ 在< https://github.com/npmstudy/your-node-v20-monoreopo -project > 项目里,我们其实只用了非常简单的npm nx run-many
383
381
384
- ``` tsx
385
- " build" : " nx run-many -t build" ,
382
+ ``` json
383
+ {
384
+ "scripts" : {
385
+ "build" : " nx run-many -t build" ,
386
386
"build:fast" : " nx run-many -t build:fast" ,
387
387
"dev" : " nx run-many -t dev" ,
388
388
"test" : " nx run-many -t test" ,
389
+ }
390
+ }
389
391
```
390
392
391
393
虽然多了一个nx模块,但执行的时候能够利用本地缓存,加快打包、测试速度,已经是非常好的事儿。
0 commit comments