164 lines
4.5 KiB
JavaScript
164 lines
4.5 KiB
JavaScript
import { http, toast } from 'uview-plus'
|
||
import Base64 from 'base-64'
|
||
import { env } from '.env';
|
||
|
||
// token 键定义
|
||
export const accessTokenKey = 'access-token';
|
||
export const refreshAccessTokenKey = `x-${accessTokenKey}`;
|
||
|
||
// 清除 token
|
||
export const clearAccessTokens = () => {
|
||
uni.removeStorageSync(accessTokenKey);
|
||
uni.removeStorageSync(refreshAccessTokenKey);
|
||
uni.clearStorageSync();
|
||
};
|
||
|
||
const requestInterceptors = (vm) => {
|
||
/**
|
||
* 请求拦截
|
||
* @param {Object} http
|
||
*/
|
||
http.interceptors.request.use((config) => { // 可使用async await 做异步操作
|
||
let accessToken = uni.getStorageSync(accessTokenKey)
|
||
if (accessToken) {
|
||
config['header'] = {
|
||
Authorization: `Bearer ${accessToken}`
|
||
}
|
||
|
||
// 判断 accessToken 是否过期
|
||
let jwt = decryptJWT(accessToken);
|
||
let exp = getJWTDate(jwt.exp);
|
||
// token 已经过期
|
||
if (new Date() >= exp) {
|
||
// 获取刷新 token
|
||
let refreshAccessToken = uni.getStorageSync(refreshAccessTokenKey);
|
||
// 携带刷新 token
|
||
if (refreshAccessToken) {
|
||
config['header'] = {
|
||
'X-Authorization': `Bearer ${refreshAccessToken}`
|
||
}
|
||
}
|
||
}
|
||
}
|
||
// 初始化请求拦截器时,会执行此方法,此时data为undefined,赋予默认{}
|
||
config.data = config.data || {}
|
||
// 可以在此通过vm引用vuex中的变量,具体值在vm.$store.state中
|
||
return config
|
||
}, (config) => // 可使用async await 做异步操作
|
||
Promise.reject(config))
|
||
}
|
||
|
||
const responseInterceptors = (vm) => {
|
||
/**
|
||
* 响应拦截
|
||
* @param {Object} http
|
||
*/
|
||
http.interceptors.response.use((response) => {
|
||
/* 对响应成功做点什么 可使用async await 做异步操作*/
|
||
let data = response.data
|
||
// 处理 401
|
||
if (data.code === 401) {
|
||
clearAccessTokens();
|
||
// 跳转到登录页面
|
||
uni.redirectTo({
|
||
url: '/pages/login/login'
|
||
});
|
||
}
|
||
|
||
// 处理未进行规范化处理的
|
||
if (data.code >= 400) {
|
||
toast(JSON.stringify(data.statusText))
|
||
}
|
||
|
||
// 处理规范化结果错误
|
||
if (data && data.hasOwnProperty('errors') && data.errors) {
|
||
toast(JSON.stringify(data.errors))
|
||
}
|
||
|
||
// 读取响应报文头 token 信息
|
||
let accessToken = response.header[accessTokenKey];
|
||
let refreshAccessToken = response.header[refreshAccessTokenKey];
|
||
|
||
// 判断是否是无效 token
|
||
if (accessToken === 'invalid_token') {
|
||
clearAccessTokens();
|
||
} else if (refreshAccessToken && accessToken && accessToken !== 'invalid_token') {
|
||
// 判断是否存在刷新 token,如果存在则存储在本地
|
||
uni.setStorageSync(accessTokenKey, accessToken)
|
||
uni.setStorageSync(refreshAccessTokenKey, refreshAccessToken)
|
||
}
|
||
|
||
// 自定义参数
|
||
let custom = response.config?.custom
|
||
if (data.code !== 200) { // 服务端返回的状态码不等于200,则reject()
|
||
// 如果没有显式定义custom的toast参数为false的话,默认对报错进行toast弹出提示
|
||
|
||
if (custom.toast !== false) {
|
||
toast(data.message)
|
||
}
|
||
// 如果需要catch返回,则进行reject
|
||
if (custom?.catch) {
|
||
return Promise.reject(data)
|
||
} else {
|
||
// 否则返回一个pending中的promise
|
||
return new Promise(() => {})
|
||
}
|
||
}
|
||
return data || {}
|
||
}, (response) => {
|
||
/* 对响应错误做点什么 (statusCode !== 200)*/
|
||
clearAccessTokens();
|
||
return Promise.reject(response)
|
||
})
|
||
}
|
||
|
||
/**
|
||
* 解密 JWT token 的信息
|
||
* @param token jwt token 字符串
|
||
* @returns <any>object
|
||
*/
|
||
export function decryptJWT(token) {
|
||
token = token.replace(/_/g, '/').replace(/-/g, '+');
|
||
var json = decodeURIComponent(escape(Base64.decode(token.split('.')[1])));
|
||
return JSON.parse(json);
|
||
}
|
||
|
||
/**
|
||
* 将 JWT 时间戳转换成 Date
|
||
* @description 主要针对 `exp`,`iat`,`nbf`
|
||
* @param timestamp 时间戳
|
||
* @returns Date 对象
|
||
*/
|
||
export function getJWTDate(timestamp) {
|
||
return new Date(timestamp * 1000);
|
||
}
|
||
|
||
// 初始化请求配置
|
||
const initRequest = () => {
|
||
let accessToken = uni.getStorageSync(accessTokenKey);
|
||
if (accessToken) {
|
||
let jwt = decryptJWT(accessToken);
|
||
let exp = getJWTDate(jwt.exp);
|
||
// token 已经过期
|
||
if (new Date() >= exp) {
|
||
// 获取刷新 token
|
||
let refreshAccessToken = uni.getStorageSync(refreshAccessTokenKey);
|
||
// 携带刷新 token
|
||
if (refreshAccessToken) {
|
||
env.header['X-Authorization'] = `Bearer ${refreshAccessToken}`;
|
||
}
|
||
} else {
|
||
env.header['Authorization'] = `Bearer ${accessToken}`
|
||
}
|
||
}
|
||
http.setConfig((config) => {
|
||
config.baseURL = env.baseUrl /* 根域名 */
|
||
return config;
|
||
})
|
||
requestInterceptors();
|
||
responseInterceptors();
|
||
}
|
||
|
||
export {
|
||
initRequest
|
||
} |