2022-01-19 07:39:39 +00:00
|
|
|
import { ipcRenderer } from 'electron';
|
2022-01-18 13:03:16 +00:00
|
|
|
import frappe from 'frappejs';
|
|
|
|
import { MandatoryError, ValidationError } from 'frappejs/common/errors';
|
2022-01-19 07:39:39 +00:00
|
|
|
import { IPC_ACTIONS } from './messages';
|
2022-01-19 10:59:13 +00:00
|
|
|
import { showMessageDialog, showToast } from './utils';
|
2022-01-18 13:03:16 +00:00
|
|
|
|
|
|
|
function shouldNotStore(error) {
|
|
|
|
return [MandatoryError, ValidationError].some(
|
|
|
|
(errorClass) => error instanceof errorClass
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-01-19 10:59:13 +00:00
|
|
|
function reportError(errorLogObj) {
|
|
|
|
// push errorlog to frappebooks.com
|
|
|
|
console.log(errorLogObj);
|
|
|
|
}
|
|
|
|
|
2022-01-19 11:13:50 +00:00
|
|
|
function getToastProps(errorLogObj) {
|
|
|
|
const props = {
|
|
|
|
message: _(`Error: `) + errorLogObj.name,
|
|
|
|
type: 'error',
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!frappe.SystemSettings.autoReportErrors) {
|
|
|
|
Object.assign(props, {
|
|
|
|
actionText: frappe._('Report Error'),
|
|
|
|
action: () => {
|
|
|
|
reportError(errorLogObj);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return props;
|
|
|
|
}
|
|
|
|
|
2022-01-18 13:03:16 +00:00
|
|
|
export function handleError(shouldLog, error, more = {}) {
|
|
|
|
if (shouldLog) {
|
|
|
|
console.error(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (shouldNotStore(error)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const { name, stack, message } = error;
|
|
|
|
const errorLogObj = { name, stack, message, more };
|
2022-01-19 11:13:50 +00:00
|
|
|
|
2022-01-18 13:03:16 +00:00
|
|
|
frappe.errorLog.push(errorLogObj);
|
|
|
|
|
2022-01-19 11:13:50 +00:00
|
|
|
showToast(getToastProps(errorLogObj));
|
|
|
|
if (frappe.SystemSettings.autoReportErrors) {
|
|
|
|
reportError(errorLogObj);
|
|
|
|
}
|
2022-01-18 13:03:16 +00:00
|
|
|
}
|
2022-01-19 06:14:12 +00:00
|
|
|
|
2022-01-19 07:39:39 +00:00
|
|
|
export function getErrorMessage(e, doc) {
|
|
|
|
let errorMessage = e.message || _('An error occurred.');
|
|
|
|
const { doctype, name } = doc;
|
|
|
|
const canElaborate = doctype && name;
|
|
|
|
if (e.type === frappe.errors.LinkValidationError && canElaborate) {
|
|
|
|
errorMessage = _('{0} {1} is linked with existing records.', [
|
|
|
|
doctype,
|
|
|
|
name,
|
|
|
|
]);
|
|
|
|
} else if (e.type === frappe.errors.DuplicateEntryError && canElaborate) {
|
|
|
|
errorMessage = _('{0} {1} already exists.', [doctype, name]);
|
|
|
|
}
|
|
|
|
return errorMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function handleErrorWithDialog(error, doc = {}) {
|
|
|
|
let errorMessage = getErrorMessage(error, doc);
|
|
|
|
handleError(false, error, { errorMessage, doc });
|
|
|
|
|
|
|
|
showMessageDialog({ message: errorMessage });
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function showErrorDialog({ title, content }) {
|
|
|
|
// To be used for show stopper errors
|
|
|
|
title ??= frappe._('Error');
|
|
|
|
content ??= frappe._(
|
|
|
|
'Something has gone terribly wrong. Please check the console and raise an issue.'
|
|
|
|
);
|
|
|
|
|
|
|
|
await ipcRenderer.invoke(IPC_ACTIONS.SHOW_ERROR, { title, content });
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wrapper Functions
|
|
|
|
|
2022-01-19 06:14:12 +00:00
|
|
|
export function getErrorHandled(func) {
|
|
|
|
return async function errorHandled(...args) {
|
|
|
|
try {
|
|
|
|
return await func(...args);
|
|
|
|
} catch (error) {
|
|
|
|
handleError(false, error, {
|
|
|
|
functionName: func.name,
|
|
|
|
functionArgs: args,
|
|
|
|
});
|
|
|
|
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getErrorHandledSync(func) {
|
|
|
|
return function errorHandledSync(...args) {
|
|
|
|
try {
|
|
|
|
return func(...args);
|
|
|
|
} catch (error) {
|
|
|
|
handleError(false, error, {
|
|
|
|
functionName: func.name,
|
|
|
|
functionArgs: args,
|
|
|
|
});
|
|
|
|
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|