前面我们学过如何调试 Nest 项目,那如何调试 Nest 源码呢?
有的同学说,调试 Nest 项目的时候,调用栈里不就有源码部分么?
其实不是的,这部分是编译后的代码:
我们新建个 Nest 项目:
nest new debug-nest-source
点击 debug 面板的 create a launch.json file按钮:
输入 npm,选择 launch via npm,创建一个调试 npm scripts 的配置:
改为这样:
{
"name": "Launch via NPM",
"request": "launch",
"runtimeArgs": [
"run-script",
"start:dev"
],
"runtimeExecutable": "npm",
"console": "integratedTerminal",
"skipFiles": [
"<node_internals>/**"
],
"type": "node"
}
这个就是跑 npm run start:dev 的调试配置。
指定 console 为 integratedTerminal,这样日志会输出在 terminal。
不然,日志会输出在 debug console。颜色等都不一样:
在 AppController 的 getHello 打个断点:
点击 debug 启动:
然后浏览器重新访问 http://localhost:3000
这时候代码就会在断点处断住:
这样就可以断点调试 Nest 项目了。
但如果想调试源码,还需要再做一步:
因为现在调用栈里的 Nest 源码部分都是编译后的:
我们想调试 Nest 的 ts 源码,这就需要用到 sourcemap 了。
用 npm install 下载的包没有 sourcemap 的代码,想要 sourcemap,需要自己 build 源码。
把 Nest 项目下载下来,并安装依赖(加个 --depth=1 是下载单 commit,--single-branch 是下载单个分支,这样速度会快很多):
git clone --depth=1 --single-branch https://github.com/nestjs/nest
build 下:
npm install
npm run build
会在 node_modules/@nestjs 下生成编译后的代码。
看下 npm scripts:
可以看到它做的事情就是 tsc 编译代码,然后把编译后的文件移动到 node_modules/@nestjs 目录下。
move 的具体实现可以看 tools/gulp/tasks/move.ts 的代码:
所以,执行 npm run build,你就会在 node_modules/@nestjs 下看到这样的代码:
只包含了 js 和 ts,没有 sourcemap:
生成 sourcemap 需要改下 tsc 编译配置,也就是 packages/tsconfig.build.json 文件:
设置 sourceMap 为 true 也就是生成 sourcemap,但默认的 sourcemap 里不包含内联的源码,也就是 sourcesContent 部分,需要设置 inlineSources 来包含。
再次执行 npm run build,就会生成带有 sourcemap 的代码:
并且 sourcemap 是内联了源码的:
然后我们跑一下 Nest 的项目,直接跑 samples 目录下的项目即可,这是 Nest 内置的一些案例项目:
进入 01-cats-app 目录,安装依赖
npm install
然后把根目录 node_modules 下生成的代码覆盖下 cats 项目的 node_modules 下的代码:
创建一个调试配置:
改成这样:
{
"name": "调试 nest 源码",
"request": "launch",
"runtimeArgs": [
"run-script",
"start:dev"
],
"runtimeExecutable": "npm",
"console": "integratedTerminal",
"cwd": "${workspaceFolder}/sample/01-cats-app/",
"resolveSourceMapLocations": [
"${workspaceFolder}/**",
// "!**/node_modules/**"
],
"skipFiles": [
"<node_internals>/**"
],
"type": "node"
}
指定 cwd 为那个项目的目录,也就是在那个目录下执行 npm run start:dev。
resolveSourceMapLocations 是从哪里找 sourcemap,默认排除掉了 node_modules,我们把它去掉。
在 sample/01-cats-app 的 src/cats/cats.controller.ts 打个断点:
然后点击 debug 调试:
如果提示端口被占用,你需要先 kill 掉之前的进程再跑:
浏览器访问 http://localhost:3000/cats
断住之后你看下调用栈:
这时候 sourcemap 就生效了,可以看到调用栈中的就是 Nest 的 ts 源码。
这样就可以调试 Nest 源码了。
比如我们看下 Nest 的 AOP 部分的源码:
点击这个调用栈:
可以看到它先创建了所有的 pipes、interceptors、guards 的实例,然后封装了调用 pipe 和 guard 的函数:
下面调用 hander 的时候,先调用 guard、再调用 interceptor,然后调用 handler,并且 handler 里会先用 pipe 处理参数:
这就是 AOP 机制的源码。
而如果你想在你的项目里调试 Nest 源码,只要把 node_modules/@nestjs 覆盖你项目下那个就好了。
总结
这节我们学习了如何调试 Nest 源码。
vscode 里创建 npm scripts 的调试配置,就可以调试 npm run start:dev 的服务。
但如果想调试源码,需要把 Nest 源码下载下来,build 出一份带有 sourcemap 版本的代码。
同时还要设置 resolveSourcemapLocations 去掉排除 node_modules 的配置。
然后再调试,就可以直接调试 Nest 的 ts 源码了。
我们调试了下 AOP 部分的源码,以后你对哪部分的实现原理感兴趣,也可以自己调试源码了。