2
0
mirror of https://github.com/frappe/books.git synced 2025-01-24 23:58:27 +00:00
books/src/errorHandling.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

193 lines
4.8 KiB
TypeScript
Raw Normal View History

import { ipcRenderer } from 'electron';
2022-04-20 12:08:47 +05:30
import { t } from 'fyo';
import { ConfigKeys } from 'fyo/core/types';
import { Doc } from 'fyo/model/doc';
import { BaseError } from 'fyo/utils/errors';
import { ErrorLog } from 'fyo/utils/types';
import { truncate } from 'lodash';
2022-03-31 13:03:58 +05:30
import { IPC_ACTIONS, IPC_MESSAGES } from 'utils/messages';
2022-04-20 12:08:47 +05:30
import { fyo } from './initFyo';
import router from './router';
import { getErrorMessage, stringifyCircular } from './utils';
import { MessageDialogOptions, ToastOptions } from './utils/types';
2022-04-20 12:08:47 +05:30
import { showMessageDialog, showToast } from './utils/ui';
2022-01-18 18:33:16 +05:30
function shouldNotStore(error: Error) {
const shouldLog = (error as BaseError).shouldStore ?? true;
return !shouldLog;
2022-01-18 18:33:16 +05:30
}
2022-05-27 13:20:14 +05:30
async function reportError(errorLogObj: ErrorLog) {
if (!errorLogObj.stack) {
return;
}
const body = {
error_name: errorLogObj.name,
message: errorLogObj.message,
stack: errorLogObj.stack,
more: stringifyCircular(errorLogObj.more ?? {}),
};
2022-03-17 11:33:02 +05:30
2022-04-20 12:08:47 +05:30
if (fyo.store.isDevelopment) {
2022-03-17 11:33:02 +05:30
console.log(body);
}
await ipcRenderer.invoke(IPC_ACTIONS.SEND_ERROR, JSON.stringify(body));
2022-01-19 16:29:13 +05:30
}
2022-05-27 13:20:14 +05:30
function getToastProps(errorLogObj: ErrorLog) {
2022-04-20 12:08:47 +05:30
const props: ToastOptions = {
2022-05-20 16:42:32 +05:30
message: errorLogObj.name ?? t`Error`,
type: 'error',
actionText: t`Report Error`,
action: () => reportIssue(errorLogObj),
};
return props;
}
2022-03-16 17:12:06 +05:30
export function getErrorLogObject(
error: Error,
more: Record<string, unknown>
): ErrorLog {
const { name, stack, message } = error;
const errorLogObj = { name, stack, message, more };
2022-04-20 12:08:47 +05:30
fyo.errorLog.push(errorLogObj);
return errorLogObj;
}
export async function handleError(
logToConsole: boolean,
error: Error,
2022-05-27 13:20:14 +05:30
more?: Record<string, unknown>
) {
if (logToConsole) {
2022-01-18 18:33:16 +05:30
console.error(error);
}
if (shouldNotStore(error)) {
return;
}
2022-03-16 17:12:06 +05:30
const errorLogObj = getErrorLogObject(error, more ?? {});
2022-01-18 18:33:16 +05:30
2022-05-27 13:20:14 +05:30
await reportError(errorLogObj);
const toastProps = getToastProps(errorLogObj);
await showToast(toastProps);
2022-01-18 18:33:16 +05:30
}
export async function handleErrorWithDialog(
error: Error,
doc?: Doc,
reportError?: false,
dontThrow?: false
) {
const errorMessage = getErrorMessage(error, doc);
await handleError(false, error, { errorMessage, doc });
2022-05-20 16:42:32 +05:30
const name = error.name ?? t`Error`;
const options: MessageDialogOptions = { message: name, detail: errorMessage };
if (reportError) {
options.detail = truncate(options.detail, { length: 128 });
options.buttons = [
{
label: t`Report`,
action() {
reportIssue(getErrorLogObject(error, { errorMessage }));
},
},
{ label: t`OK`, action() {} },
];
}
await showMessageDialog(options);
if (dontThrow) {
if (fyo.store.isDevelopment) {
console.error(error);
}
return;
}
throw error;
}
export async function showErrorDialog(title?: string, content?: string) {
// To be used for show stopper errors
2022-01-24 12:14:11 +05:30
title ??= t`Error`;
content ??= t`Something has gone terribly wrong. Please check the console and raise an issue.`;
await ipcRenderer.invoke(IPC_ACTIONS.SHOW_ERROR, { title, content });
}
// Wrapper Functions
export function getErrorHandled(func: Function) {
return async function errorHandled(...args: unknown[]) {
try {
return await func(...args);
} catch (error) {
await handleError(false, error as Error, {
functionName: func.name,
functionArgs: args,
});
throw error;
}
};
}
export function getErrorHandledSync(func: Function) {
return function errorHandledSync(...args: unknown[]) {
try {
return func(...args);
} catch (error) {
handleError(false, error as Error, {
functionName: func.name,
functionArgs: args,
}).then(() => {
throw error;
});
}
};
}
function getIssueUrlQuery(errorLogObj?: ErrorLog): string {
const baseUrl = 'https://github.com/frappe/books/issues/new?labels=bug';
const body = ['<h2>Description</h2>', 'Add some description...', ''];
if (errorLogObj) {
body.push(
'<h2>Error Info</h2>',
'',
`**Error**: _${errorLogObj.name}: ${errorLogObj.message}_`,
''
);
}
if (errorLogObj?.stack) {
body.push('**Stack**:', '```', errorLogObj.stack, '```', '');
}
body.push(`**Version**: \`${fyo.store.appVersion}\``);
2022-07-07 17:59:06 +05:30
body.push(`**Platform**: \`${fyo.store.platform}\``);
body.push(`**Path**: \`${router.currentRoute.value.fullPath}\``);
body.push(`**Language**: \`${fyo.config.get(ConfigKeys.Language)}\``);
if (fyo.singles.SystemSettings?.countryCode) {
body.push(`**Country**: \`${fyo.singles.SystemSettings.countryCode}\``);
}
const url = [baseUrl, `body=${body.join('\n')}`].join('&');
return encodeURI(url);
}
export function reportIssue(errorLogObj?: ErrorLog) {
const urlQuery = getIssueUrlQuery(errorLogObj);
ipcRenderer.send(IPC_MESSAGES.OPEN_EXTERNAL, urlQuery);
}