Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix/error-status-codes #596

Open
wants to merge 2 commits into
base: stable
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,9 @@
# 4.0.3

_Fixes:_

- Corrected status codes of the most custom errors.

# 4.0.2

_Hotfix_:
Expand Down
3 changes: 2 additions & 1 deletion bin/cli.js
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,8 @@ const start = async () => {
}
} else {
throw new ExportError(
'[cli] No valid options provided. Please check your input and try again.'
'[cli] No valid options provided. Please check your input and try again.',
400
);
}
} catch (error) {
Expand Down
7 changes: 4 additions & 3 deletions lib/browser.js
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ let browser;
*/
export function get() {
if (!browser) {
throw new ExportError('[browser] No valid browser has been created.');
throw new ExportError('[browser] No valid browser has been created.', 500);
}
return browser;
}
Expand Down Expand Up @@ -119,12 +119,13 @@ export async function create(puppeteerArgs) {
}
} catch (error) {
throw new ExportError(
'[browser] Maximum retries to open a browser instance reached.'
'[browser] Maximum retries to open a browser instance reached.',
500
).setError(error);
}

if (!browser) {
throw new ExportError('[browser] Cannot find a browser to open.');
throw new ExportError('[browser] Cannot find a browser to open.', 500);
}
}

Expand Down
20 changes: 12 additions & 8 deletions lib/cache.js
Original file line number Diff line number Diff line change
Expand Up @@ -88,9 +88,10 @@ export const saveConfigToManifest = async (config, fetchedModules) => {
'utf8'
);
} catch (error) {
throw new ExportError('[cache] Error writing the cache manifest.').setError(
error
);
throw new ExportError(
'[cache] Error writing the cache manifest.',
500
).setError(error);
}
};

Expand Down Expand Up @@ -139,7 +140,8 @@ export const fetchAndProcessScript = async (

if (shouldThrowError) {
throw new ExportError(
`Could not fetch the ${script}.js. The script might not exist in the requested version (status code: ${response.statusCode}).`
`Could not fetch the ${script}.js. The script might not exist in the requested version (status code: ${response.statusCode}).`,
404
).setError(response);
} else {
log(
Expand Down Expand Up @@ -185,9 +187,10 @@ export const fetchScripts = async (
port: proxyPort
});
} catch (error) {
throw new ExportError('[cache] Could not create a Proxy Agent.').setError(
error
);
throw new ExportError(
'[cache] Could not create a Proxy Agent.',
500
).setError(error);
}
}

Expand Down Expand Up @@ -269,7 +272,8 @@ export const updateCache = async (
return fetchedModules;
} catch (error) {
throw new ExportError(
'[cache] Unable to update the local Highcharts cache.'
'[cache] Unable to update the local Highcharts cache.',
500
).setError(error);
}
};
Expand Down
32 changes: 24 additions & 8 deletions lib/chart.js
Original file line number Diff line number Diff line change
Expand Up @@ -70,7 +70,7 @@ export const startExport = async (settings, endCallback) => {
return result;
} catch (error) {
return endCallback(
new ExportError('[chart] Error loading SVG input.').setError(error)
new ExportError('[chart] Error loading SVG input.', 400).setError(error)
);
}
}
Expand All @@ -84,7 +84,9 @@ export const startExport = async (settings, endCallback) => {
return exportAsString(options.export.instr.trim(), options, endCallback);
} catch (error) {
return endCallback(
new ExportError('[chart] Error loading input file.').setError(error)
new ExportError('[chart] Error loading input file.', 400).setError(
error
)
);
}
}
Expand Down Expand Up @@ -112,15 +114,16 @@ export const startExport = async (settings, endCallback) => {
);
} catch (error) {
return endCallback(
new ExportError('[chart] Error loading raw input.').setError(error)
new ExportError('[chart] Error loading raw input.', 400).setError(error)
);
}
}

// No input specified, pass an error message to the callback
return endCallback(
new ExportError(
`[chart] No valid input specified. Check if at least one of the following parameters is correctly set: 'infile', 'instr', 'options', or 'svg'.`
`[chart] No valid input specified. Check if at least one of the following parameters is correctly set: 'infile', 'instr', 'options', or 'svg'.`,
400
)
);
};
Expand Down Expand Up @@ -183,7 +186,8 @@ export const batchExport = async (options) => {
await killPool();
} catch (error) {
throw new ExportError(
'[chart] Error encountered during batch export.'
'[chart] Error encountered during batch export.',
400
).setError(error);
}
};
Expand Down Expand Up @@ -344,7 +348,8 @@ const doExport = async (options, chartJson, endCallback, svg) => {
// these settings.
return endCallback(
new ExportError(
`[chart] The 'callback', 'resources' and 'customCode' options have been disabled for this server.`
`[chart] The 'callback', 'resources' and 'customCode' options have been disabled for this server.`,
403
)
);
}
Expand Down Expand Up @@ -489,7 +494,8 @@ const doStraightInject = (options, endCallback) => {
} catch (error) {
return endCallback(
new ExportError(
`[chart] Malformed input detected for ${options.export?.requestId || '?'}. Please make sure that your JSON/JavaScript options are sent using the "options" attribute, and that if you're using SVG, it is unescaped.`
`[chart] Malformed input detected for ${options.export?.requestId || '?'}. Please make sure that your JSON/JavaScript options are sent using the "options" attribute, and that if you're using SVG, it is unescaped.`,
400
).setError(error)
);
}
Expand Down Expand Up @@ -522,6 +528,15 @@ const exportAsString = (stringToExport, options, endCallback) => {
// Try to parse to JSON and call the doExport function
const chartJSON = JSON.parse(stringToExport.replaceAll(/\t|\n|\r/g, ' '));

if (!chartJSON || typeof chartJSON !== 'object') {
return endCallback(
new ExportError(
'[chart] Invalid configuration provided - the options must be an object, not a string',
400
)
);
}

// If a correct JSON, do the export
return doExport(options, chartJSON, endCallback);
} catch (error) {
Expand All @@ -532,7 +547,8 @@ const exportAsString = (stringToExport, options, endCallback) => {
// Do not allow straight injection without the allowCodeExecution flag
return endCallback(
new ExportError(
'[chart] Only JSON configurations and SVG are allowed for this server. If this is your server, JavaScript custom code can be enabled by starting the server with the --allowCodeExecution flag.'
'[chart] Only JSON configurations and SVG are allowed for this server. If this is your server, JavaScript custom code can be enabled by starting the server with the --allowCodeExecution flag.',
403
).setError(error)
);
}
Expand Down
15 changes: 14 additions & 1 deletion lib/errors/ExportError.js
Original file line number Diff line number Diff line change
@@ -1,22 +1,35 @@
class ExportError extends Error {
constructor(message) {
/**
* @param {string} message - The error message.
* @param {number} statusCode - The status code (400, 500, etc.).
*/
constructor(message, statusCode) {
super();

this.message = message;
this.stackMessage = message;

if (statusCode) {
this.statusCode = statusCode;
}
}

setError(error) {
this.error = error;

if (error.name) {
this.name = error.name;
}

if (error.statusCode) {
this.statusCode = error.statusCode;
}

if (error.stack) {
this.stackMessage = error.message;
this.stack = error.stack;
}

return this;
}
}
Expand Down
14 changes: 10 additions & 4 deletions lib/errors/HttpError.js
Original file line number Diff line number Diff line change
@@ -1,13 +1,19 @@
import ExportError from './ExportError.js';

class HttpError extends ExportError {
constructor(message, status) {
/**
* @param {string} message - The error message.
* @param {number} statusCode - The status code (400, 500, etc.).
*/
constructor(message, statusCode) {
super(message);
this.status = this.statusCode = status;

this.statusCode = statusCode;
}

setStatus(status) {
this.status = status;
setStatus(statusCode) {
this.statusCode = statusCode;

return this;
}
}
Expand Down
5 changes: 3 additions & 2 deletions lib/export.js
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@ const createImage = (page, type, encoding, clip, rasterizationTimeout) =>
}),
new Promise((_resolve, reject) =>
setTimeout(
() => reject(new ExportError('Rasterization timeout')),
() => reject(new ExportError('Rasterization timeout', 408)),
rasterizationTimeout || 1500
)
)
Expand Down Expand Up @@ -292,7 +292,8 @@ export default async (page, chart, options) => {
);
} else {
throw new ExportError(
`[export] Unsupported output format ${exportOptions.type}.`
`[export] Unsupported output format ${exportOptions.type}.`,
400
);
}

Expand Down
6 changes: 1 addition & 5 deletions lib/highcharts.js
Original file line number Diff line number Diff line change
Expand Up @@ -124,11 +124,7 @@ export async function triggerExport(chartOptions, options, displayErrors) {
let constr = options.export.constr || 'chart';
constr = typeof Highcharts[constr] !== 'undefined' ? constr : 'chart';

Highcharts[constr](
'container',
finalOptions,
finalCallback
);
Highcharts[constr]('container', finalOptions, finalCallback);

// Get the current global options
const defaultOptions = getOptions();
Expand Down
19 changes: 13 additions & 6 deletions lib/pool.js
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ const factory = {
page = await newPage();

if (!page || page.isClosed()) {
throw new ExportError('The page is invalid or closed.');
throw new ExportError('The page is invalid or closed.', 400);
}

log(
Expand All @@ -73,7 +73,8 @@ const factory = {
);
} catch (error) {
throw new ExportError(
'Error encountered when creating a new page.'
'Error encountered when creating a new page.',
400
).setError(error);
}

Expand Down Expand Up @@ -225,7 +226,8 @@ export const initPool = async (config) => {
);
} catch (error) {
throw new ExportError(
'[pool] Could not create the pool of workers.'
'[pool] Could not create the pool of workers.',
500
).setError(error);
}
};
Expand Down Expand Up @@ -283,7 +285,10 @@ export const postWork = async (chart, options) => {
}

if (!pool) {
throw new ExportError('Work received, but pool has not been started.');
throw new ExportError(
'Work received, but pool has not been started.',
500
);
}

// Acquire the worker along with the id of resource and work count
Expand All @@ -307,14 +312,16 @@ export const postWork = async (chart, options) => {
(options.payload?.requestId
? `For request with ID ${options.payload?.requestId} - `
: '') +
`Error encountered when acquiring an available entry: ${acquireCounter()}ms.`
`Error encountered when acquiring an available entry: ${acquireCounter()}ms.`,
400
).setError(error);
}
log(4, '[pool] Acquired a worker handle.');

if (!workerHandle.page) {
throw new ExportError(
'Resolved worker page is invalid: the pool setup is wonky.'
'Resolved worker page is invalid: the pool setup is wonky.',
400
);
}

Expand Down
20 changes: 11 additions & 9 deletions lib/server/error.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,11 @@ import { logWithStack } from '../logger.js';
* Middleware for logging errors with stack trace and handling error response.
*
* @param {Error} error - The error object.
* @param {Express.Request} req - The Express request object.
* @param {Express.Response} res - The Express response object.
* @param {Express.Request} request - The Express request object.
* @param {Express.Response} response - The Express response object.
* @param {Function} next - The next middleware function.
*/
const logErrorMiddleware = (error, req, res, next) => {
const logErrorMiddleware = (error, request, response, next) => {
// Display the error with stack in a correct format
logWithStack(1, error);

Expand All @@ -26,17 +26,19 @@ const logErrorMiddleware = (error, req, res, next) => {
* Middleware for returning error response.
*
* @param {Error} error - The error object.
* @param {Express.Request} req - The Express request object.
* @param {Express.Response} res - The Express response object.
* @param {Express.Request} request - The Express request object.
* @param {Express.Response} response - The Express response object.
* @param {Function} next - The next middleware function.
*/
const returnErrorMiddleware = (error, req, res, next) => {
const returnErrorMiddleware = (error, request, response, next) => {
// Gather all requied information for the response
const { statusCode: stCode, status, message, stack } = error;
const statusCode = stCode || status || 400;
const { message, stack } = error;

// Use the error's status code or the default 400
const statusCode = error.statusCode || 400;

// Set and return response
res.status(statusCode).json({ statusCode, message, stack });
response.status(statusCode).json({ statusCode, message, stack });
};

export default (app) => {
Expand Down
2 changes: 1 addition & 1 deletion lib/server/routes/change_hc_version.js
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,7 @@ export default (app) =>
} catch (error) {
throw new HttpError(
`Version change: ${error.message}`,
error.statusCode
400
).setError(error);
}

Expand Down
Loading
Loading