一个 Node 相关的项目中,总是少不了跑脚本。跑一个脚本拉取配置、处理一些数据以及定时任务更是家常便饭。
一个 Node 相关的项目中,总是少不了跑脚本。跑一个脚本拉取配置、处理一些数据以及定时任务更是家常便饭。
在一些重要流程中能够看到脚本的身影:
- CI,用以测试、质量保障及部署等
- Docker,用以构建镜像
- Cron,用以定时任务
如果在这些重要流程中脚本出错无法及时发现问题,将有可能引发更加隐蔽的问题。
最近观察项目镜像构建,会偶尔发现一两个镜像虽然构建成功,但容器却跑不起来的情况。「究其原因,是因为 Exit Code 的问题」。
Exit Code
什么是 exit code?
exit code 代表一个进程的返回码,通过系统调用 exit_group 来触发。在 POSIX 中,0 代表正常的返回码,1-255 代表异常返回码,一般主动抛出的错误码都是 1。在 Node 应用中使用 process.exitCode = 1 来代表因不期望的异常而中断。
这里有一张关于异常码的附表 Appendix E. Exit Codes With Special Meanings[1]。
异常码在操作系统中随处可见,以下是一个关于 cat 命令的异常以及它的 exit code,并使用 strace 追踪系统调用。
- $cata
- cat:a:Nosuchfileordirectory
- #使用strace查看cat的系统调用
- #-e只显示write与exit_group的系统调用
- $strace-ewrite,exit_groupcata
- write(2,"cat:",5cat:)=5
- write(2,"a",1a)=1
- write(2,":Nosuchfileordirectory",27:Nosuchfileordirectory)=27
- write(2,"\n",1
- )=1
- exit_group(1)=?
- +++exitedwith1+++
从系统调用的最后一行可以看出,该进行的 exit code 是 1,并把错误信息输出到 stderr (标准错误的 fd 为 2) 中
如何查看 exit code
从 strace 中可以来判断进程的 exit code,但是不够方便过于冗余,特别身处 shell 编程环境中。
「有一种简单的方法,通过 echo $? 来确认返回码」
- $cata
- cat:a:Nosuchfileordirectory
- $echo$?
- 1
throw new Error与Promise.reject区别
以下是两段代码,第一段抛出一个异常,第二段 Promise.reject,两段代码都会如下打印出一段异常信息,那么两者有什么区别?
- functionerror(){
- thrownewError('hello,error')
- }
- error()
- //Output:
- ///Users/shanyue/Documents/note/demo.js:2
- //thrownewError('hello,world')
- //^
- //
- //Error:hello,world
- //aterror(/Users/shanyue/Documents/note/demo.js:2:9)
- //atObject.<anonymous>(/Users/shanyue/Documents/note/demo.js:5:1)
- //atModule._compile(internal/modules/cjs/loader.js:701:30)
- functionerror(){
- thrownewError('hello,error')
- }
- error()
- //Output:
- ///Users/shanyue/Documents/note/demo.js:2
- //thrownewError('hello,world')
- //^
- //
- //Error:hello,world
- //aterror(/Users/shanyue/Documents/note/demo.js:2:9)
- //atObject.<anonymous>(/Users/shanyue/Documents/note/demo.js:5:1)
- //atModule._compile(internal/modules/cjs/loader.js:701:30)
在对上述两个测试用例使用 echo $? 查看 exit code,我们会发现 throw new Error() 的 exit code 为 1,而 Promise.reject() 的为 0。
「从操作系统的角度来讲,exit code 为 0 代表进程成功运行并退出,此时即使有 Promise.reject,操作系统也会视为它执行成功。」
这在 Dockerfile 与 CI 中将留有安全隐患。
Dockerfile 在 node 中的注意点
当使用 Dockerfile 构建镜像时,如果 RUN 的进程返回非 0 的返回码,构建就会失败。
「而在 Node 中的错误处理中,我们倾向于所有的异常都交由 async/await 来处理,而当发生异常时,由于此时 exit code 为 0 并不会导致镜像构建失败。」
这是一个浅显易懂的含 Promise.reject() 问题的镜像。
- FROMnode:12-alpine
- RUNnode-e"Promise.reject('hello,world')"
构建镜像过程如下:「即使在构建过程打印出了 unhandledPromiseRejection 信息,但是镜像仍然构建成功。」
- $dockerbuild-tdemo.
- SendingbuildcontexttoDockerdaemon33.28kB
- Step1/2:FROMnode:12-alpine
- --->18f4bc975732
- Step2/2:RUNnode-e"Promise.reject('hello,world')"
- --->Runningin79a6d53c5aa6
- (node:1)UnhandledPromiseRejectionWarning:hello,world
- (node:1)UnhandledPromiseRejectionWarning:Unhandledpromiserejection.Thiserrororiginatedeitherbythrowinginsideofanasyncfunctionwithoutacatchblock,orbyrejectingapromisewhichwasnothandledwith.catch().Toterminatethenodeprocessonunhandledpromiserejection,usetheCLIflag`--unhandled-rejections=strict`(seehttps://nodejs.org/api/cli.html#cli_unhandled_rejections_mode).(rejectionid:1)
- (node:1)[DEP0018]DeprecationWarning:Unhandledpromiserejectionsaredeprecated.Inthefuture,promiserejectionsthatarenothandledwillterminatetheNode.jsprocesswithanon-zeroexitcode.
- Removingintermediatecontainer79a6d53c5aa6
- --->09f07eb993fe
- Successfullybuilt09f07eb993fe
- Successfullytaggeddemo:latest
Promise.reject 脚本解决方案
能在编译时能发现的问题,绝不要放在运行时。所以,构建镜像或 CI 中需要执行 node 脚本时,对异常处理需要手动指定 process.exitCode = 1 来提前暴露问题
- runScript().catch(()=>{
- process.exitCode=1
- })
在构建镜像时,也有关于异常解决方案的建议:
(node:1) UnhandledPromiseRejectionWarning: Unhandled promise rejection. This error originated either by throwing inside of an async function without a catch block, or by rejecting a promise which was not handled with .catch(). To terminate the node process on unhandled promise rejection, use the CLI flag --unhandled-rejections=strict (see https://nodejs.org/api/cli.html#cli_unhandled_rejections_mode). (rejection id: 1) |
根据提示,--unhandled-rejections=strict 将会把 Promise.reject 的退出码设置为 1,并在将来的 node 版本中修正 Promise 异常退出码。
- $node--unhandled-rejections=stricterror.js
--unhandled-rejections=strict 的配置对 node 有版本要求:
Added in: v12.0.0, v10.17.0
By default all unhandled rejections trigger a warning plus a deprecation warning for the very first unhandled rejection in case no unhandledRejection hook is used. |
总结
- 当进程结束的 exit code 为非 0 时,系统会认为该进程执行失败
- 通过 echo $? 可查看终端上一进程的 exit code
- Node 中 Promise.reject 时 exit code 为 0
- Node 中可以通过 process.exitCode = 1 显式设置 exit code
- 在 Node12+ 中可以通过 node --unhandled-rejections=strict error.js执行脚本,视 Promise.reject 的 exit code 为 1
©本文为清一色官方代发,观点仅代表作者本人,与清一色无关。清一色对文中陈述、观点判断保持中立,不对所包含内容的准确性、可靠性或完整性提供任何明示或暗示的保证。本文不作为投资理财建议,请读者仅作参考,并请自行承担全部责任。文中部分文字/图片/视频/音频等来源于网络,如侵犯到著作权人的权利,请与我们联系(微信/QQ:1074760229)。转载请注明出处:清一色财经