mirror of https://github.com/msgbyte/tailchat
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
88 lines
2.1 KiB
TypeScript
88 lines
2.1 KiB
TypeScript
import axios, { AxiosRequestConfig } from 'axios';
|
|
import _get from 'lodash/get';
|
|
import _isFunction from 'lodash/isFunction';
|
|
import { t } from '../i18n';
|
|
import { getErrorHook, tokenGetter } from '../manager/request';
|
|
import { getServiceUrl, onServiceUrlChange } from '../manager/service';
|
|
|
|
export type CommonRequestResult<T> =
|
|
| ({
|
|
result: false;
|
|
msg: string;
|
|
} & Partial<T>) // 并上一个T是为了方便取值, 但需要判定
|
|
| ({
|
|
result: true;
|
|
} & T);
|
|
|
|
class RequestError extends Error {}
|
|
|
|
export type RequestConfig = AxiosRequestConfig;
|
|
|
|
/**
|
|
* 创建请求实例
|
|
*/
|
|
function createRequest() {
|
|
const ins = axios.create({
|
|
baseURL: getServiceUrl(),
|
|
});
|
|
onServiceUrlChange((getUrl) => {
|
|
// 重置请求地址
|
|
ins.defaults.baseURL = getUrl();
|
|
});
|
|
|
|
ins.interceptors.request.use(async (val) => {
|
|
if (
|
|
['post', 'get'].includes(String(val.method).toLowerCase()) &&
|
|
!val.headers['X-Token']
|
|
) {
|
|
// 任何请求都尝试增加token
|
|
val.headers['X-Token'] = await tokenGetter();
|
|
}
|
|
|
|
return val;
|
|
});
|
|
|
|
ins.interceptors.response.use(
|
|
(val) => {
|
|
/**
|
|
* 预处理返回的数据
|
|
*/
|
|
val.data = _get(val.data, 'data', val.data);
|
|
|
|
return val;
|
|
},
|
|
(err) => {
|
|
// 尝试获取错误信息
|
|
const responseData = _get(err, 'response.data') ?? {};
|
|
let errorMsg: string = responseData.message;
|
|
const code: number = responseData.code;
|
|
|
|
if (responseData.type === 'VALIDATION_ERROR') {
|
|
// 校验失败
|
|
errorMsg = t('请求参数校验失败');
|
|
|
|
if (Array.isArray(responseData.data)) {
|
|
try {
|
|
errorMsg += `: ${responseData.data
|
|
.map((item: any) => item.field)
|
|
.join(', ')}`;
|
|
} catch (e) {}
|
|
}
|
|
}
|
|
|
|
if (_isFunction(getErrorHook)) {
|
|
const isContinue = getErrorHook(err);
|
|
if (isContinue === false) {
|
|
return { data: { result: false, msg: errorMsg, code } };
|
|
}
|
|
}
|
|
|
|
throw new RequestError(errorMsg ?? err.message);
|
|
}
|
|
);
|
|
|
|
return ins;
|
|
}
|
|
|
|
export const request = createRequest();
|