2025-08-14 21:04:04 +08:00
|
|
|
|
import type { CustomRequestOptions } from '@/http/types'
|
|
|
|
|
|
import { useUserStore } from '@/store'
|
|
|
|
|
|
import { getEnvBaseUrl } from '@/utils'
|
|
|
|
|
|
import { platform } from '@/utils/platform'
|
2025-08-15 13:46:23 +08:00
|
|
|
|
import { checkAndRefreshToken } from '@/utils/tokenRefresh'
|
2025-08-14 21:04:04 +08:00
|
|
|
|
import { stringifyQuery } from './tools/queryString'
|
|
|
|
|
|
|
|
|
|
|
|
// 请求基准地址
|
|
|
|
|
|
const baseUrl = getEnvBaseUrl()
|
|
|
|
|
|
|
|
|
|
|
|
// 拦截器配置
|
|
|
|
|
|
const httpInterceptor = {
|
|
|
|
|
|
// 拦截前触发
|
2025-08-15 13:46:23 +08:00
|
|
|
|
async invoke(options: CustomRequestOptions) {
|
2025-08-14 21:04:04 +08:00
|
|
|
|
// 接口请求支持通过 query 参数配置 queryString
|
|
|
|
|
|
if (options.query) {
|
|
|
|
|
|
const queryStr = stringifyQuery(options.query)
|
|
|
|
|
|
if (options.url.includes('?')) {
|
|
|
|
|
|
options.url += `&${queryStr}`
|
|
|
|
|
|
}
|
|
|
|
|
|
else {
|
|
|
|
|
|
options.url += `?${queryStr}`
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
// 非 http 开头需拼接地址
|
|
|
|
|
|
if (!options.url.startsWith('http')) {
|
|
|
|
|
|
// #ifdef H5
|
|
|
|
|
|
// console.log(__VITE_APP_PROXY__)
|
|
|
|
|
|
if (JSON.parse(__VITE_APP_PROXY__)) {
|
|
|
|
|
|
// 自动拼接代理前缀
|
|
|
|
|
|
options.url = import.meta.env.VITE_APP_PROXY_PREFIX + options.url
|
|
|
|
|
|
}
|
|
|
|
|
|
else {
|
|
|
|
|
|
options.url = baseUrl + options.url
|
|
|
|
|
|
}
|
|
|
|
|
|
// #endif
|
|
|
|
|
|
// 非H5正常拼接
|
|
|
|
|
|
// #ifndef H5
|
|
|
|
|
|
options.url = baseUrl + options.url
|
|
|
|
|
|
// #endif
|
|
|
|
|
|
// TIPS: 如果需要对接多个后端服务,也可以在这里处理,拼接成所需要的地址
|
|
|
|
|
|
}
|
|
|
|
|
|
// 1. 请求超时
|
|
|
|
|
|
options.timeout = 10000 // 10s
|
|
|
|
|
|
// 2. (可选)添加小程序端请求头标识
|
|
|
|
|
|
options.header = {
|
|
|
|
|
|
...options.header,
|
|
|
|
|
|
}
|
2025-08-15 13:46:23 +08:00
|
|
|
|
// 3. 检查并刷新token(如果即将过期)
|
2025-08-14 21:04:04 +08:00
|
|
|
|
const userStore = useUserStore()
|
2025-08-15 13:46:23 +08:00
|
|
|
|
const { accessToken } = userStore
|
|
|
|
|
|
|
|
|
|
|
|
// 如果有访问令牌且不是刷新token的请求,检查是否需要刷新
|
|
|
|
|
|
if (accessToken && !options.url.includes('/auth/refresh')) {
|
|
|
|
|
|
try {
|
|
|
|
|
|
await checkAndRefreshToken()
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (error) {
|
|
|
|
|
|
console.error('Token刷新失败:', error)
|
|
|
|
|
|
// 刷新失败时继续原请求,让业务层处理401错误
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 4. 添加 token 请求头标识
|
|
|
|
|
|
// 重新获取token(可能已经刷新过了)
|
|
|
|
|
|
const currentToken = useUserStore().accessToken
|
|
|
|
|
|
if (currentToken) {
|
|
|
|
|
|
options.header.Authorization = `Bearer ${currentToken}`
|
2025-08-14 21:04:04 +08:00
|
|
|
|
}
|
2025-08-15 21:30:51 +08:00
|
|
|
|
|
|
|
|
|
|
return options
|
2025-08-14 21:04:04 +08:00
|
|
|
|
},
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export const requestInterceptor = {
|
|
|
|
|
|
install() {
|
|
|
|
|
|
// 拦截 request 请求
|
|
|
|
|
|
uni.addInterceptor('request', httpInterceptor)
|
|
|
|
|
|
// 拦截 uploadFile 文件上传
|
|
|
|
|
|
uni.addInterceptor('uploadFile', httpInterceptor)
|
|
|
|
|
|
},
|
|
|
|
|
|
}
|