import Axios, { AxiosInstance, Method, AxiosRequestConfig, AxiosError, AxiosResponse } from 'axios' import NProgress from 'nprogress' import { eventBus } from '@/utils' import { authService } from '@/services' class Http { client: AxiosInstance private silent = false private showLoadingIndicator () { NProgress.start() } private hideLoadingIndicator () { NProgress.done(true) } public request (method: Method, url: string, data: Record = {}, onUploadProgress?: any) { return this.client.request({ url, data, method, onUploadProgress }) as Promise<{ data: T }> } public async get (url: string) { return (await this.request('get', url)).data } public async post (url: string, data: Record = {}, onUploadProgress?: any) { return (await this.request('post', url, data, onUploadProgress)).data } public async put (url: string, data: Record) { return (await this.request('put', url, data)).data } public async patch (url: string, data: Record) { return (await this.request('patch', url, data)).data } public async delete (url: string, data: Record = {}) { return (await this.request('delete', url, data)).data } constructor () { this.client = Axios.create({ baseURL: `${window.BASE_URL}api`, headers: { 'X-Api-Version': 'v6' } }) // Intercept the request to make sure the token is injected into the header. this.client.interceptors.request.use((config: AxiosRequestConfig) => { this.silent || this.showLoadingIndicator() config.headers.Authorization = `Bearer ${authService.getApiToken()}` return config }) // Intercept the response and… this.client.interceptors.response.use((response: AxiosResponse) => { this.silent || this.hideLoadingIndicator() this.silent = false // …get the tokens from the header if exist, and save them // This occurs during user updating password. const token = response.headers.authorization token && authService.setApiToken(token) return response }, (error: AxiosError) => { this.silent || this.hideLoadingIndicator() this.silent = false // Also, if we receive a Bad Request / Unauthorized error if (error.response?.status === 400 || error.response?.status === 401) { // and we're not trying to log in if (!(error.config.method === 'post' && error.config.url === 'me')) { // the token must have expired. Log out. eventBus.emit('LOG_OUT') } } return Promise.reject(error) }) } public get silently () { this.silent = true return this } } export const http = new Http()