2022-12-20 07:47:02 -08:00
|
|
|
import { AxiosError } from 'axios';
|
|
|
|
import React from 'react';
|
2022-12-21 11:14:31 -08:00
|
|
|
import toast from 'react-hot-toast';
|
|
|
|
import { defineMessages, MessageDescriptor } from 'react-intl';
|
2022-12-20 07:47:02 -08:00
|
|
|
|
2022-12-21 11:14:31 -08:00
|
|
|
import { Toast } from './components/ui';
|
2022-12-20 07:47:02 -08:00
|
|
|
import { httpErrorMessages } from './utils/errors';
|
|
|
|
|
2022-12-21 11:14:31 -08:00
|
|
|
export type ToastText = string | MessageDescriptor
|
|
|
|
export type ToastType = 'success' | 'error' | 'info'
|
2022-12-20 07:47:02 -08:00
|
|
|
|
2024-03-20 15:58:53 -07:00
|
|
|
export interface IToastOptions {
|
2023-10-02 11:54:02 -07:00
|
|
|
action?(): void;
|
|
|
|
actionLink?: string;
|
|
|
|
actionLabel?: ToastText;
|
|
|
|
duration?: number;
|
|
|
|
summary?: string;
|
2022-12-20 07:47:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
const DEFAULT_DURATION = 4000;
|
|
|
|
|
|
|
|
const createToast = (type: ToastType, message: ToastText, opts?: IToastOptions) => {
|
|
|
|
const duration = opts?.duration || DEFAULT_DURATION;
|
|
|
|
|
2022-12-21 11:14:31 -08:00
|
|
|
toast.custom((t) => <Toast t={t} message={message} type={type} {...opts} />, {
|
2022-12-20 07:47:02 -08:00
|
|
|
duration,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
function info(message: ToastText, opts?: IToastOptions) {
|
|
|
|
createToast('info', message, opts);
|
|
|
|
}
|
|
|
|
|
|
|
|
function success(message: ToastText, opts?: IToastOptions) {
|
|
|
|
createToast('success', message, opts);
|
|
|
|
}
|
|
|
|
|
|
|
|
function error(message: ToastText, opts?: IToastOptions) {
|
|
|
|
createToast('error', message, opts);
|
|
|
|
}
|
|
|
|
|
|
|
|
const messages = defineMessages({
|
2023-10-11 12:57:53 -07:00
|
|
|
unexpectedMessage: { id: 'alert.unexpected.message', defaultMessage: 'Something went wrong.' },
|
2022-12-20 07:47:02 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
function showAlertForError(networkError: AxiosError<any>) {
|
|
|
|
if (networkError?.response) {
|
|
|
|
const { data, status, statusText } = networkError.response;
|
|
|
|
|
|
|
|
if (status === 502) {
|
|
|
|
return error('The server is down');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status === 404 || status === 410) {
|
|
|
|
// Skip these errors as they are reflected in the UI
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
let message: string | undefined = statusText;
|
|
|
|
|
|
|
|
if (data?.error) {
|
|
|
|
message = data.error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!message) {
|
|
|
|
message = httpErrorMessages.find((httpError) => httpError.code === status)?.description;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message) {
|
|
|
|
return error(message);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
console.error(networkError);
|
|
|
|
return error(messages.unexpectedMessage);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default {
|
|
|
|
info,
|
|
|
|
success,
|
|
|
|
error,
|
|
|
|
showAlertForError,
|
|
|
|
};
|