前言

今天想写一下eggjs的自定义异常处理中间件,在写的时候遇到了问题,这个错误我捕获不到类型??

处理过程,不喜欢看过程的朋友请直接看解决方法和总结

看一下是什么:

抛出的异常是检验失败异常Validation Failed (code: invalid_param)

我写了个异常处理中间件,用来处理业务中的异常

module.exports = (options, app) => {

  return async function testMiddleware(ctx, next) {

    try{
await next();
}
catch (err) {
// 记录到日志
ctx.logger.error(err);
ctx.throw(err);
}
};
};

具体思路是想要根据异常的类型来实现自定义的处理,如验证失败就不走onerror。如果不是想要处理的错误就ctx.throw(),丢给onerror继续处理。

看了下抛出的错误类型是:UnprocessableEntityError

也没多想,就用instanceof判断一下吧:

module.exports = (options, app) => {

  return async function testMiddleware(ctx, next) {

    try{
await next();
}
catch (err) {
if (err instanceof UnprocessableEntityError) {
// 如果是这个错误那么就自己处理,否则就抛出
// 记录到日志
ctx.logger.error(err);
ctx.body = {
code: err.status,
err
}
} else {
ctx.throw(err);
}
}
};
};

然后就报了‘UnprocessableEntityError’ undefine错误。。

对的,UnprocessableEntityError我没有引入,nodejs自带也没这个错误?那这个错误哪来的?

接着我顺着错误去找抛出的源头,找到了这个:

this.throw是koa 的throw方法:

在node_modules_koa@2.7.0@koa\lib\context.js的throw函数抛出了这个错误,

具体使用的是createError这个函数,再去代码目录下找createError:

webstorm ctrl+鼠标左键定位跳转一下,发现原来使用的是http-errors模块里面的里面的createError方法,下面是具体的函数内容:

// node_modules\_http-errors@1.7.3@http-errors\index.js

/**
* Create a new HTTP Error.
*
* @returns {Error}
* @public
*/ function createError () {
// so much arity going on ~_~
var err
var msg
var status = 500
var props = {}
for (var i = 0; i < arguments.length; i++) {
var arg = arguments[i]
if (arg instanceof Error) { // 如果参数类型就是Error那么err就是arg
err = arg
status = err.status || err.statusCode || status
continue // 跳过以下代码,进入下一轮循环
}
switch (typeof arg) {
case 'string': // 如果arg是string 那么err的msg就是arg
msg = arg
break
case 'number': // 如果是number 那么错误的status就是arg,这里要求statu是第一个参数
status = arg
if (i !== 0) {
deprecate('non-first-argument status code; replace with createError(' + arg + ', ...)')
}
break
case 'object': // 如果是类,那么把类挂载到err上
props = arg
break
}
}
// 循环完毕了,来判断一下状态码
if (typeof status === 'number' && (status < 400 || status >= 600)) {
deprecate('non-error status code; use only 4xx or 5xx status codes')
} if (typeof status !== 'number' ||
(!statuses[status] && (status < 400 || status >= 600))) {
status = 500
} // constructor
var HttpError = createError[status] || createError[codeClass(status)]
// 我们传的是422,所以返回了UnprocessableEntityError这个构造器 if (!err) { // 如果不是直接传入Error
// create error
err = HttpError // 如果HttpError为空,HttpError里面没有这个错误
? new HttpError(msg) // 执行到这一步,返回一个新的错误,
: new Error(msg || statuses[status])
Error.captureStackTrace(err, createError) // 传入堆栈信息
} if (!HttpError || !(err instanceof HttpError) || err.status !== status) { //不执行
// add properties to generic error
err.expose = status < 500
err.status = err.statusCode = status
} for (var key in props) {
if (key !== 'status' && key !== 'statusCode') { //将数据挂载上去
err[key] = props[key]
}
} return err
}

大概搞明白了

就是validate在抛出异常之后,调用了koa的throw又调用了http-errors里面的createError,createError先去找有咩有这个http错误,常见的什么404啊403啊500啊之类的,没找到就直接把用Error构造,找到了就用对应的错误类型构造,这些类型构造又是存在:node_modules/koa/node_modules/http-errors/node_modules/statuses/index.js里面的,比如上面的422



对应的就是UnprocessableEntityError这个异常。

解决方法

到了这里,解决就变的很轻松了,我们要捕获的异常是UnprocessableEntityError,那么可以使用err.name来获取错误的名字,获取到了之后,再根据



validate抛出的固定错误码'Validation Failed'来确定错误。

实际上也可以直接根据这个错误码来判断,这个错误码经历了HttpError的洗礼之后变成了message,可以通过err.message来获取。

那么代码如下:

module.exports = (options, app) => {

  return async function testMiddleware(ctx, next) {

    try{
await next();
}
catch (err) {
if (err.message === 'Validation Failed') {
// 记录到日志
ctx.logger.error(err);
ctx.body = {
code: err.status,
err
}
} else {
ctx.throw(err);
}
}
};
};

再次打开浏览器尝试一下,这个异常被正确的返回给前端了:

总结一下

  1. 错误类型,既错误名可以使用err.name获取
  2. 对应的错误消息(相比HttpError更精细的)使用err.message获取
  3. 错误哪怕被自己处理了也还是写入日志,logger.error()一下比较好
  4. 你不想处理的错误还是交给onerror来办,使用ctx.throw丢给它
  5. 记得处理错误要返回信息,既设置ctx.body,不然会返回404的

谢谢观看,有什么问题欢迎留言评论,看到尽量会回复。。。

nodejs异常处理过程/获取nodejs异常类型/写一个eggjs异常处理中间件的更多相关文章

  1. 写一个eggjs权限验证中间件

    关于中间件 https://eggjs.org/zh-cn/basics/middleware.html 官方文档说的很清楚了,不再叙述. 我们要达到怎么样一个效果? 用户没有登录不能访问一些特定的页 ...

  2. 异常处理的方式二:throws+异常类型

    package com.yhqtv.demo01Exception; import java.io.File; import java.io.FileInputStream; import java. ...

  3. 一、NodeJS入门——准备工作(1)——NodeJS的安装

    目录 1.介绍 2.nodejs下载 3.nodejs安装 4.nodejs的简单实用 5.nodejs的经典入门:hello world 6.总结 1    介绍 这是一系列的内容主要是关于我在学习 ...

  4. springmvc在处理请求过程中出现异常信息交由异常处理器进行处理,自定义异常处理器可以实现一个系统的异常处理逻辑。为了区别不同的异常通常根据异常类型自定义异常类,这里我们创建一个自定义系统异常,如果controller、service、dao抛出此类异常说明是系统预期处理的异常信息。

    springmvc在处理请求过程中出现异常信息交由异常处理器进行处理,自定义异常处理器可以实现一个系统的异常处理逻辑. 1.1 异常处理思路 系统中异常包括两类:预期异常和运行时异常RuntimeEx ...

  5. 用 nodejs 写一个命令行工具 :创建 react 组件的命令行工具

    用 nodejs 写一个命令行工具 :创建 react 组件的命令行工具 前言 上周,同事抱怨说 react 怎么不能像 angular 那样,使用命令行工具来生成一个组件.对呀,平时工作时,想要创建 ...

  6. nodejs 通过 get获取数据修改redis数据

    如下代码是没有报错的正确代码 我通过https获取到数据 想用redis set一个键值存储 现在我掉入了回调陷阱res.on 里面接收到的数据是data 里面如果放入 client.on('conn ...

  7. linux系统下nodejs安装过程随记

    首先下载适合的版本.这里我使用的是node v.10.36 先介绍编译安装的详细过程. 下载该版本: wget http://nodejs.org/dist/v0.10.36/node-v0.10.3 ...

  8. python第三十课--异常(异常处理定义格式和常见类型)

    演示: 1).异常处理的定义格式: 2).常见的运行时异常类型: try: print(10/0) num=int('132a') except Exception as e: print('出错了. ...

  9. nodejs:使用puppeteer在服务器中构建一个获取电影电视剧剧集的接口

    首先我们看下数据来源: 来源于这个网站:https://z1.m1907.cn/ 可以说这个网站上能找到很多你想看的很多电影或电视剧,最重要的是很多电影电视剧在别的网站是收费的,但是在这里看是免费的, ...

随机推荐

  1. English--动词语态

    English|动词语态 动词的语态在理解长难句中很重要.了解被动语态与主动语态,掌握语态的变化方式. 前言 目前所有的文章思想格式都是:知识+情感. 知识:对于所有的知识点的描述.力求不含任何的自我 ...

  2. js事件【续】(事件类型)

    一.UI事件[使用时需要添加on eg: onload 页面加载完成事件]load    [一张页面或一幅图像完成加载]页面加载后触发的事件,即进入页面后 unload    [用户退出页面]页面卸载 ...

  3. 解决javaScript在不同时区new Date()显示值不同问题

    在日期格式化时遇到的问题,日期格式化方法在最下面 如果在中国时区    formatDate('2019-07-09')  结果是 ‘2019-07-09’ 如果 在夏威夷时区 utc-10:00 或 ...

  4. 48.vue-awesome-swipe使用

    1.安装:npm install vue-awesome-swiper --save 2.局部引入: import "swiper/dist/css/swiper.css"; im ...

  5. 前端项目中公共方法汇总utils.js

    目录 判断手机类型IOS Android 格式化金钱 金钱字符串变回数字 用aa替换中文 并返回 去除文件后缀,得到文件名称(不带后缀) 获取浏览器类型(名称) post方式下载文件流 动态设置img ...

  6. div css字间距

    div:nth-child(1){ color: #FFFFFF; font-size: 13px; font-weight: bold; margin-bottom: 25px; letter-sp ...

  7. APS应用案例|纽威阀门实现高效排产

    企业背景: 苏州纽威阀门股份有限公司(下文简称:纽威阀门)成立于1997年,总部设在江苏苏州.自成立以来一直致力于工业阀门的研发与制造,以为客户提供全套工业阀门解决方案为目标.纽威阀门通过企业的努力发 ...

  8. Odoo中的ORM API(模型数据增删改查)

    转载请注明原文地址:https://www.cnblogs.com/ygj0930/p/10826214.html 一:增 1:create():返回新创建的记录对象 self.create({'na ...

  9. linux设备驱动程序--bus

    linux 中bus驱动解析 总线(bus)是linux发展过程中抽象出来的一种设备模型,为了统一管理所有的设备,内核中每个设备都会被挂载在总线上,这个bus可以是对应硬件的bus(i2c bus.s ...

  10. git远程上的分支到本地

    先想一个自己要在本地新建的分支名称,qianjinyan git checkout -b qianjinyan origin/SELLER-2248-1018 git branch 查看分支 git ...