This commit is contained in:
吴红兵
2025-12-02 10:37:49 +08:00
commit 1f645dad3e
1183 changed files with 147673 additions and 0 deletions

138
src/utils/request.ts Normal file
View File

@@ -0,0 +1,138 @@
import axios, { AxiosInstance, AxiosResponse, InternalAxiosRequestConfig } from 'axios';
import { Session } from '/@/utils/storage';
import { useMessageBox } from '/@/hooks/message';
import qs from 'qs';
import other from './other';
import {paramsFilter} from "/@/flow";
import { wrapEncryption, encryptRequestParams, decrypt } from './apiCrypto';
// 常用header
export enum CommonHeaderEnum {
'TENANT_ID' = 'TENANT-ID',
'ENC_FLAG' = 'Enc-Flag',
'AUTHORIZATION' = 'Authorization',
'VERSION' = 'VERSION',
}
/**
* 创建并配置一个 Axios 实例对象
*/
const service: AxiosInstance = axios.create({
baseURL: import.meta.env.VITE_API_URL,
timeout: 50000, // 全局超时时间
paramsSerializer: {
serialize: (params: any) => {
return qs.stringify(paramsFilter(params), { arrayFormat: 'repeat' });
},
},
});
/**
* Axios请求拦截器对请求进行处理
* 1. 序列化get请求参数
* 2. 统一增加Authorization和TENANT-ID请求头
* 3. 自动适配单体、微服务架构不同的URL
* @param config AxiosRequestConfig对象包含请求配置信息
*/
service.interceptors.request.use(
(config: InternalAxiosRequestConfig) => {
config.headers['VERSION'] = 'A' // 目标版本
// 统一增加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;
}
// 增加 gray_version 请求头
const version = import.meta.env.VITE_GRAY_VERSION;
if (version) {
config.headers![CommonHeaderEnum.VERSION] = version;
}
// 请求报文加密 ,如果请求头中不包含 ENC_FLAG false 则加密
if (config.data && !config.headers![CommonHeaderEnum.ENC_FLAG]) {
config.data = wrapEncryption(config.data);
}
// 如果是 GET ,加密 config.param 的每一个参数并URLencode
if (config.method === 'get' && config.params) {
config.params = encryptRequestParams(config.params);
}
// 自动适配单体和微服务架构不同的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) {
response.data = decrypt(response.data.encryption);
return response.data;
}
return response.data;
};
/**
* 添加 Axios 的响应拦截器,用于全局响应结果处理
*/
service.interceptors.response.use(handleResponse, (error) => {
const status = Number(error.response.status) || 200;
if (status === 423) {
return Promise.reject({ msg: '"演示环境,仅供预览"' });
}
if (status === 424) {
useMessageBox()
.confirm('令牌状态已过期,请点击重新登录')
.then(() => {
Session.clear(); // 清除浏览器全部临时缓存
window.location.href = '/'; // 去登录页
return;
});
}
if (status === 426) {
useMessageBox()
.confirm('租户状态已过期,请联系管理员')
.then(() => {
Session.clear(); // 清除浏览器全部临时缓存
return;
});
}
// 针对密文返回解密
if (error.response?.data?.encryption) {
error.response.data = decrypt(error.response?.data.encryption);
}
return Promise.reject(error.response.data);
});
// 导出 axios 实例
export default service;