Ghost/ghost/errors/lib/utils.js
Sam Lord d8ee09d0fa @tryghost/errors Add stack related functions
refs: https://github.com/TryGhost/Toolbox/issues/147

Correctly prepares the stack when wrapping another error, and adds a new method on errors to create an error which can be shown to the user (i.e. remove the stack trace in production)
2021-12-01 17:28:42 +00:00

207 lines
6.1 KiB
JavaScript

const omit = require('lodash/omit');
const merge = require('lodash/merge');
const extend = require('lodash/extend');
const _private = {};
_private.serialize = function serialize(err) {
try {
return {
id: err.id,
status: err.statusCode,
code: err.code || err.errorType,
title: err.name,
detail: err.message,
meta: {
context: err.context,
help: err.help,
errorDetails: err.errorDetails,
level: err.level,
errorType: err.errorType
}
};
} catch (error) {
return {
detail: 'Something went wrong.'
};
}
};
_private.deserialize = function deserialize(obj) {
try {
return {
id: obj.id,
message: obj.detail || obj.error_description || obj.message,
statusCode: obj.status,
code: obj.code || obj.error,
level: obj.meta && obj.meta.level,
help: obj.meta && obj.meta.help,
context: obj.meta && obj.meta.context
};
} catch (err) {
return {
message: 'Something went wrong.'
};
}
};
/**
* @description Serialize error instance into oauth format.
*
* @see https://tools.ietf.org/html/rfc6749#page-45
*
* To not loose any error data when sending errors between internal services, we use the suggested OAuth properties and add ours as well.
*/
_private.OAuthSerialize = function OAuthSerialize(err) {
const matchTable = {};
matchTable[this.NoPermissionError.name] = 'access_denied';
matchTable[this.MaintenanceError.name] = 'temporarily_unavailable';
matchTable[this.BadRequestError.name] = matchTable[this.ValidationError.name] = 'invalid_request';
matchTable.default = 'server_error';
return merge({
error: err.code || matchTable[err.name] || 'server_error',
error_description: err.message
}, omit(_private.serialize(err), ['detail', 'code']));
};
/**
* @description Deserialize oauth error format into GhostError instance.
* @param {Object} errorFormat
* @return {Error}
* @constructor
*/
_private.OAuthDeserialize = function OAuthDeserialize(errorFormat) {
try {
return new this[errorFormat.title || errorFormat.name || this.InternalServerError.name](_private.deserialize(errorFormat));
} catch (err) {
// CASE: you receive an OAuth formatted error, but the error prototype is unknown
return new this.InternalServerError(extend({
errorType: errorFormat.title || errorFormat.name
}, _private.deserialize(errorFormat)));
}
};
/**
* @description Serialize GhostError instance into jsonapi.org format.
* @param {Error} err
* @return {Object}
*/
_private.JSONAPISerialize = function JSONAPISerialize(err) {
const errorFormat = {
errors: [_private.serialize(err)]
};
errorFormat.errors[0].source = {};
if (err.property) {
errorFormat.errors[0].source.pointer = '/data/attributes/' + err.property;
}
return errorFormat;
};
/**
* @description Deserialize JSON api format into GhostError instance.
* @param {Object} errorFormat
* @return {Error}
*/
_private.JSONAPIDeserialize = function JSONAPIDeserialize(errorFormat) {
errorFormat = errorFormat.errors && errorFormat.errors[0] || {};
let internalError;
try {
internalError = new this[errorFormat.title || errorFormat.name || this.InternalServerError.name](_private.deserialize(errorFormat));
} catch (err) {
// CASE: you receive a JSON format error, but the error prototype is unknown
internalError = new this.InternalServerError(extend({
errorType: errorFormat.title || errorFormat.name
}, _private.deserialize(errorFormat)));
}
if (errorFormat.source && errorFormat.source.pointer) {
internalError.property = errorFormat.source.pointer.split('/')[3];
}
return internalError;
};
exports.wrapStack = function wrapStack(err, internalErr) {
const extraLine = err.stack.split(/\n/g)[1];
const [firstLine, ...rest] = internalErr.stack.split(/\n/g);
return [firstLine, extraLine, ...rest].join('\n');
};
/**
* @description Serialize GhostError instance to error JSON format
*
* jsonapi.org error format:
*
* source: {
* parameter: URL query parameter (no support yet)
* pointer: HTTP body attribute
* }
*
* @see http://jsonapi.org/format/#errors
*
* @param {Error} err
* @param {Object} options { format: [String] (jsonapi || oauth) }
*/
exports.serialize = function serialize(err, options) {
options = options || {format: 'jsonapi'};
let errorFormat = {};
try {
if (options.format === 'jsonapi') {
errorFormat = _private.JSONAPISerialize.bind(this)(err);
} else {
errorFormat = _private.OAuthSerialize.bind(this)(err);
}
} catch (error) {
errorFormat.message = 'Something went wrong.';
}
// no need to sanitize the undefined values, on response send JSON.stringify get's called
return errorFormat;
};
/**
* @description Deserialize from error JSON format to GhostError instance
* @param {Object} errorFormat
*/
exports.deserialize = function deserialize(errorFormat) {
let internalError = {};
if (errorFormat.errors) {
internalError = _private.JSONAPIDeserialize.bind(this)(errorFormat);
} else {
internalError = _private.OAuthDeserialize.bind(this)(errorFormat);
}
return internalError;
};
/**
* @description Check whether an error instance is a GhostError.
*/
exports.isGhostError = function isGhostError(err) {
const errorName = this.GhostError.name;
const legacyErrorName = 'IgnitionError';
const recursiveIsGhostError = function recursiveIsGhostError(obj) {
// no super constructor available anymore
if (!obj || !obj.name) {
return false;
}
if (obj.name === errorName || obj.name === legacyErrorName) {
return true;
}
return recursiveIsGhostError(Object.getPrototypeOf(obj));
};
return recursiveIsGhostError(err.constructor);
};