Как получить правильную обратную трассировку для пользовательского класса ошибок в NodeJS?
Мое понимание "правильного" способа создания собственного класса Error в JavaScript выглядит примерно так:
function MyError(message) {
this.name = "MyError";
this.message = message || "Default Message";
}
MyError.prototype = new Error();
MyError.prototype.constructor = MyError;
(Фрагмент кода, созданный по https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Error.)
С NodeJS, если я попытаюсь проверить на ошибку этого типа как:
var err = new MyError("whoops");
assert.ifError(err);
... обратная трассировка покажет контекст объекта Error, который я создал во время компиляции в качестве прототипа для MyError, а не объект MyError, который я создал с помощью "new MyError()".
Есть ли какой-то способ, которым я могу получить правильные данные обратной трассировки для фактической ошибки, а не прототип?
2 ответа
Нам нужно вызвать супер функцию - captureStackTrace
var util = require('util');
function MyError(message) {
Error.call(this); //super constructor
Error.captureStackTrace(this, this.constructor); //super helper method to include stack trace in error object
this.name = this.constructor.name; //set our function’s name as error name.
this.message = message; //set the error message
}
// inherit from Error
util.inherits(MyError, Error);
ОБНОВИТЬ:
Вы можете использовать этот модуль узла для легкого расширения типов ошибок https://github.com/jayyvis/extend-error
@ Джей Кумар, здесь есть один хороший ответ. Однако, может быть, здесь есть другое подобное решение
module.exports = function CustomError(message, extra) {
Error.captureStackTrace(this, this.constructor);
this.name = this.constructor.name;
this.message = message;
this.extra = extra;
};
require('util').inherits(module.exports, Error);
Error.call(this) - создает другой объект ошибки (тратит кучу времени) и не затрагивает это вообще
Так как ECMAScript6
может быть поддержан в последней Node.js
версия. Ответ под ES6
можно сослаться на эту ссылку.
class MyError extends Error {
constructor(message) {
super(message);
this.message = message;
this.name = 'MyError';
}
}
Вот тестовые коды под Node v4.2.1
class MyError extends Error{
constructor(msg, extra) {
super(msg);
this.message = msg;
this.name = 'MyError';
this.extra = extra;
}
};
var myerr = new MyError("test", 13);
console.log(myerr.stack);
console.log(myerr);
Выход:
MyError: test
at MyError (/home/bsadmin/test/test.js:5:8)
at Object.<anonymous> (/home/bsadmin/test/test.js:12:13)
at Module._compile (module.js:435:26)
at Object.Module._extensions..js (module.js:442:10)
at Module.load (module.js:356:32)
at Function.Module._load (module.js:311:12)
at Function.Module.runMain (module.js:467:10)
at startup (node.js:134:18)
at node.js:961:3
{ [MyError: test] name: 'MyError', extra: 13 }