// eslint-disable-next-line no-restricted-imports import axios from 'axios'; import dayjs from 'dayjs'; import qs from 'qs'; // eslint-disable-next-line no-restricted-imports export * from 'axios'; const isoDateRegex = /^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(?:\.\d{3})?(Z|[+-]\d{2}:\d{2})$/; // For type checking for Date and String types to work export type IsoDateString = string & { readonly __isIsoDateString: unique symbol }; type IsTuple = T extends readonly any[] ? number extends T['length'] ? false // Not a tuple (it's a regular array) : true // It's a tuple : false; // Utility type to decide the resulting type of every input object at compile time. type DeepDateConvert = T extends Date ? T : T extends IsoDateString ? Date : T extends string ? T : IsTuple extends true ? { [K in keyof T]: DeepDateConvert } : T extends (infer U)[] ? DeepDateConvert[] : T extends object ? { [K in keyof T]: DeepDateConvert } : T; /** * Converts string to dates recursively. * * @param data - Data to be transformed to Date if applicable. * @param seen - Set containing data that has been through the function before. * @returns - Data containing transformed Dates. */ function convertStringsToDatesRecursive(data: T, seen: Set): DeepDateConvert { if (typeof data !== 'object' || data === null) { if (typeof data === 'string' && isoDateRegex.test(data)) { return new Date(data) as DeepDateConvert; } return data as DeepDateConvert; } // Check for circular reference if (seen.has(data)) { return data as DeepDateConvert; } seen.add(data); if (Array.isArray(data)) { const resultArray = (data as any[]).map(array => convertStringsToDatesRecursive(array, seen)); return resultArray as DeepDateConvert; } const newData: { [key: string]: any } = {}; for (const key of Object.keys(data as object)) { const value = (data as any)[key]; if (typeof value === 'string' && isoDateRegex.test(value)) { newData[key] = new Date(value); } else if (typeof value === 'object' && value !== null) { newData[key] = convertStringsToDatesRecursive(value, seen); } else { newData[key] = value; } } return newData as DeepDateConvert; } export function convertStringsToDates(data: T): DeepDateConvert { return convertStringsToDatesRecursive(data, new Set()); } // Determine the base array of transformers let baseTransformers = axios.defaults.transformResponse; if (baseTransformers == null) { baseTransformers = []; } else if (!Array.isArray(baseTransformers)) { // If it's a single transformer function, wrap it in an array baseTransformers = [baseTransformers]; } const customAxios = axios.create({ headers: { 'X-Requested-With': 'XMLHttpRequest', 'Content-Type': 'application/json', }, transformResponse: baseTransformers.concat( (data) => { return convertStringsToDates(data); }, ), }); // serialize Date config: https://github.com/axios/axios/issues/1548#issuecomment-548306666 customAxios.interceptors.request.use((config) => { config.paramsSerializer = params => qs.stringify(params, { serializeDate: (date: Date) => dayjs(date).format('YYYY-MM-DDTHH:mm:ssZ') }); return config; }); export default customAxios;