import axios, { AxiosInstance, AxiosRequestConfig, AxiosResponse } from 'axios';
import { Session, Local } from '/@/utils/storage';
import { useMessage, useMessageBox } from '/@/hooks/message';
import qs from 'qs';
import other from './other';
import errorCode from './errorCode';
import { useThemeConfig } from '/@/stores/themeConfig';
const storesThemeConfig = useThemeConfig();
const { themeConfig } = storeToRefs(storesThemeConfig);
/**
* 创建并配置一个 Axios 实例对象
*/
const service: AxiosInstance = axios.create({
baseURL: import.meta.env.VITE_API_URL,
timeout: 50000, // 全局超时时间
});
/**
* Axios请求拦截器,对请求进行处理
* 1. 序列化get请求参数
* 2. 统一增加Authorization和TENANT-ID请求头
* 3. 自动适配单体、微服务架构不同的URL
* @param config AxiosRequestConfig对象,包含请求配置信息
*/
service.interceptors.request.use(
(config: AxiosRequestConfig) => {
// 对get请求参数进行序列化
if (config.method === 'get') {
// @ts-ignore 使用qs库来序列化查询参数
config.paramsSerializer = (params: any) => {
return qs.stringify(params, { arrayFormat: 'repeat' });
};
}
// 统一增加Authorization请求头, skipToken 跳过增加token
const token = Session.getToken();
if (token && !config.headers?.skipToken) {
config.headers![CommonHeaderEnum.AUTHORIZATION] = `Bearer ${token}`;
}
// 统一增加TENANT-ID请求头
const tenantId = Session.getTenant();
if (tenantId) {
config.headers![CommonHeaderEnum.TENANT_ID] = tenantId;
}
// 请求报文加密
if (config.headers![CommonHeaderEnum.ENC_FLAG]) {
const enc = other.encryption(JSON.stringify(config.data), import.meta.env.VITE_PWD_ENC_KEY);
config.data = {
encryption: enc,
};
}
// 自动适配单体和微服务架构不同的URL
config.url = other.adaptationUrl(config.url);
// 处理完毕,返回config对象
return config;
},
(error) => {
// 对请求错误进行处理
return Promise.reject(error);
},
);
/**
* 响应拦截器处理函数
* @param response 响应结果
* @returns 如果响应成功,则返回响应的data属性;否则,抛出错误或者执行其他操作
*/
const handleResponse = (response: AxiosResponse<any>) => {
if (response.data.code === 1) {
throw response.data;
}
// 针对密文返回解密
if (response.data.encryption) {
const originData = JSON.parse(other.decryption(response.data.encryption, import.meta.env.VITE_PWD_ENC_KEY));
response.data = originData;
return response.data;
}
// 206针对当点登录
if (response.status === 206) {
return response;
} else {
return response.data;
}
};
/**
* 添加 Axios 的响应拦截器,用于全局响应结果处理
*/
service.interceptors.response.use(handleResponse, (error) => {
const status = Number(error.response.status) || 200;
if (status === 424) {
useMessageBox()
.confirm('令牌状态已过期,请点击重新登录')
.then(() => {
themeConfig.value.isLockScreen = false;
themeConfig.value.lockScreenTime = 30;
// 将最新的主题配置保存到本地存储中
themeConfig.value.isDrawer = false;
Local.set('themeConfig', themeConfig.value);
Session.clear(); // 清除浏览器全部临时缓存
window.location.href = '/'; // 去登录页
});
return;
}
if (status !== 200) {
const errMsg = errorCode[status] || errorCode['default'];
useMessage().error(errMsg);
}
return Promise.reject(error.response.data);
});
// 常用header
export enum CommonHeaderEnum {
'TENANT_ID' = 'TENANT-ID',
'ENC_FLAG' = 'Enc-Flag',
'AUTHORIZATION' = 'Authorization',
}
// 导出 axios 实例
export default service;
axios request 封装
©著作权归作者所有,转载或内容合作请联系作者
- 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
- 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
- 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
推荐阅读更多精彩内容
- 前端小菜鸡一枚,总结项目中发现的小技巧,有什么不对和错误希望各位指出和补充,谢谢大家观看!············...
- 用户发送了请求,通过组件页面发送到了api.js页面, api.js页面根据用户的请求, 发送不同的请求发送给re...
- 搭建VUE项目时,需要考虑封装一个全局的接口请求文件,可以对登录进行拦截以及请求的拦截,还有对通用的状态错误码和异...