2016-12-06 23:06:45 +02:00
|
|
|
import Raven from 'raven-js';
|
|
|
|
|
2017-03-29 08:46:20 +03:00
|
|
|
import abbreviate from './abbreviate';
|
|
|
|
|
2019-12-09 09:47:24 +02:00
|
|
|
const isTest = process.env.NODE_ENV === 'test';
|
|
|
|
const isProduction = process.env.NODE_ENV === 'production';
|
2016-12-06 23:06:45 +02:00
|
|
|
|
2018-05-02 20:48:50 +03:00
|
|
|
class Logger {
|
2020-01-14 15:21:38 +03:00
|
|
|
init({ sentryDSN }: { sentryDSN: string }) {
|
|
|
|
if (sentryDSN) {
|
|
|
|
Raven.config(sentryDSN, {
|
2019-11-27 11:03:32 +02:00
|
|
|
logger: 'accounts-js-app',
|
|
|
|
level: 'info',
|
2019-12-09 09:47:24 +02:00
|
|
|
environment: process.env.APP_ENV,
|
|
|
|
release: process.env.__VERSION__,
|
2019-11-27 11:03:32 +02:00
|
|
|
shouldSendCallback: () => !isTest,
|
|
|
|
dataCallback: data => {
|
|
|
|
if (!data.level) {
|
|
|
|
// log unhandled errors as info
|
|
|
|
data.level = 'info';
|
|
|
|
}
|
|
|
|
|
|
|
|
return data;
|
|
|
|
},
|
|
|
|
whitelistUrls: isProduction ? [/ely\.by/] : [],
|
|
|
|
}).install();
|
|
|
|
|
|
|
|
window.addEventListener('unhandledrejection', event => {
|
|
|
|
const error = event.reason || {};
|
|
|
|
|
|
|
|
let message = error.message || error;
|
|
|
|
|
|
|
|
if (typeof message === 'string') {
|
|
|
|
message = `: ${message}`;
|
|
|
|
} else {
|
|
|
|
message = '';
|
2016-12-06 23:06:45 +02:00
|
|
|
}
|
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
this.info(`Unhandled rejection${message}`, {
|
|
|
|
error,
|
|
|
|
event,
|
2016-12-06 23:06:45 +02:00
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-12-06 23:06:45 +02:00
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
}
|
2016-12-06 23:06:45 +02:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
setUser(user: {
|
|
|
|
username: string | null;
|
|
|
|
email: string | null;
|
|
|
|
id: number | null;
|
|
|
|
}) {
|
2019-11-27 11:03:32 +02:00
|
|
|
Raven.setUserContext({
|
|
|
|
username: user.username,
|
|
|
|
email: user.email,
|
|
|
|
id: user.id,
|
|
|
|
});
|
|
|
|
}
|
2018-05-05 12:01:25 +03:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
unexpected(message: string | Error, previous: any) {
|
2019-11-27 11:03:32 +02:00
|
|
|
// TODO: check whether previous was already handled. Cover with tests
|
|
|
|
this.error(message, {
|
|
|
|
error: previous,
|
|
|
|
});
|
|
|
|
}
|
2017-03-02 07:58:33 +02:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
error(message: string | Error, context?: { [key: string]: any }) {
|
2019-11-27 11:03:32 +02:00
|
|
|
log('error', message, context);
|
|
|
|
}
|
2017-03-29 08:46:20 +03:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
info(message: string | Error, context?: { [key: string]: any }) {
|
2019-11-27 11:03:32 +02:00
|
|
|
log('info', message, context);
|
|
|
|
}
|
2018-05-02 21:22:13 +03:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
warn(message: string | Error, context?: { [key: string]: any }) {
|
2019-11-27 11:03:32 +02:00
|
|
|
log('warning', message, context);
|
|
|
|
}
|
|
|
|
|
|
|
|
getLastEventId(): string | void {
|
|
|
|
return Raven.lastEventId();
|
|
|
|
}
|
2018-05-02 20:48:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
function log(
|
2019-11-27 11:03:32 +02:00
|
|
|
level: 'error' | 'warning' | 'info' | 'debug',
|
|
|
|
message: string | Error,
|
2019-12-07 13:28:52 +02:00
|
|
|
context?: { [key: string]: any },
|
2018-05-02 20:48:50 +03:00
|
|
|
) {
|
2019-11-27 11:03:32 +02:00
|
|
|
const method: 'error' | 'warn' | 'info' | 'debug' =
|
|
|
|
level === 'warning' ? 'warn' : level;
|
|
|
|
|
|
|
|
if (isTest) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof context !== 'object') {
|
|
|
|
// it would better to always have an object here
|
|
|
|
context = {
|
|
|
|
message: context,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
prepareContext(context).then(context => {
|
|
|
|
console[method](message, context); // eslint-disable-line
|
|
|
|
|
|
|
|
Raven.captureException(message, {
|
|
|
|
level,
|
|
|
|
extra: context,
|
|
|
|
...(typeof message === 'string' ? { fingerprint: [message] } : {}),
|
2018-05-02 20:48:50 +03:00
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2018-05-02 20:48:50 +03:00
|
|
|
}
|
2016-12-06 23:06:45 +02:00
|
|
|
|
2017-04-15 09:59:11 +03:00
|
|
|
/**
|
|
|
|
* prepare data for JSON.stringify
|
|
|
|
*
|
|
|
|
* @param {object} context
|
|
|
|
*
|
2019-11-27 11:03:32 +02:00
|
|
|
* @returns {Promise}
|
2017-04-15 09:59:11 +03:00
|
|
|
*/
|
2019-12-07 13:28:52 +02:00
|
|
|
function prepareContext(context: { [key: string]: any }) {
|
2019-11-27 11:03:32 +02:00
|
|
|
if (context instanceof Response) {
|
|
|
|
// TODO: rewrite abbreviate to use promises and recursively find Response
|
|
|
|
return context
|
|
|
|
.json()
|
|
|
|
.catch(() => context.text())
|
|
|
|
.then(body =>
|
|
|
|
abbreviate({
|
|
|
|
type: context.type,
|
|
|
|
url: context.url,
|
|
|
|
status: context.status,
|
|
|
|
statusText: context.statusText,
|
|
|
|
body,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
} else if (context.originalResponse instanceof Response) {
|
|
|
|
return prepareContext(context.originalResponse).then(originalResponse =>
|
|
|
|
abbreviate({
|
|
|
|
...context,
|
|
|
|
originalResponse,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.resolve(abbreviate(context));
|
2017-04-15 09:59:11 +03:00
|
|
|
}
|
|
|
|
|
2018-05-02 20:48:50 +03:00
|
|
|
export default new Logger();
|