import Axios, { AxiosInstance, Method } from 'axios' import NProgress from 'nprogress' import { eventBus } from '@/utils' import { authService } from '@/services' class Http { client: AxiosInstance private static setProgressBar () { NProgress.start() } private static hideProgressBar () { 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 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 => { Http.setProgressBar() config.headers.Authorization = `Bearer ${authService.getApiToken()}` return config }) // Intercept the response and… this.client.interceptors.response.use(response => { Http.hideProgressBar() // …get the tokens from the header or response data if exist, and save them. const token = response.headers.authorization || response.data.token token && authService.setApiToken(token) const audioToken = response.data['audio-token'] audioToken && authService.setAudioToken(audioToken) return response }, error => { Http.hideProgressBar() // 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' && /\/api\/me\/?$/.test(error.config.url))) { // the token must have expired. Log out. eventBus.emit('LOG_OUT') } } return Promise.reject(error) }) } } export const http = new Http()