vue无感刷新Token并重新请求

vue 拦截器拦截401重新请求Token 无感刷新Token 之后重新请求报401的接口

instance.interceptors.response.use(
  async (response) => {
    let { data } = response;

    if (data.code === 401 || data.code === 403) {
      return await handleExpiredToken(response.config);
    }
    if (data.code !== 200) {
      return Promise.reject(data);
    } else {
      return Promise.resolve(data);
    }
  },
  async (error) => {
    if (!error || !error.response) {
      handleNetworkError();
      return Promise.reject('Network anomaly');
    }

    switch (error.response.status) {
      case 401:
      case 403:
        return await handleExpiredToken(error.config);
      case 500:
        handleServerError(error.response.data.code);
        break;
      case 502:
        Message.error('Server error');
        break;
      default:
        break;
    }

    return Promise.reject(error.response.data.message);
  }
);

let isRefreshing = false;
let pendingRequests = [];

/// 处理 Token 过期
const handleExpiredToken = async (originalRequest) => {
  if (isRefreshing) {
    // 如果正在刷新 Token,将请求推入队列等待
    return new Promise((resolve, reject) => {
      pendingRequests.push({ originalRequest, resolve, reject });
    });
  }
  isRefreshing = true; // 标记为正在刷新
  // 请求新的 token
  try {
    const params = {
      refreshToken: localStorage.getItem('refreshToken'),
      grantType: 'refreshToken'
    };
    // 刷新Token接口
    const response = await login(params);
    const { accessToken, refreshToken } = response.data;

    localStorage.setItem('token', accessToken);
    localStorage.setItem('refreshToken', refreshToken);

    // 处理通过原请求重试其他待处理请求
    processPendingRequests(accessToken);

    // 更新原请求的 Authorization 头
    originalRequest.headers['Authorization'] = `Bearer ${accessToken}`;

    // 重新发送原请求
    return await instance(originalRequest);
  } catch (error) {
    console.log('error--------', error);
    localStorage.clear();
    router.push('/login');
    // 清空待处理请求队列
    processPendingRequests(null);
  } finally {
    isRefreshing = false; // 清理标记
  }
}

// 处理待处理请求
const processPendingRequests = (accessToken) => {
  pendingRequests.forEach(({ originalRequest, resolve, reject }) => {
    if (accessToken) {
      // 更新请求的头部
      originalRequest.headers['Authorization'] = `Bearer ${accessToken}`;
      resolve(instance(originalRequest));
    } else {
      reject('Token refresh failed');
    }
  });
  // 清空队列
  pendingRequests = [];
};
Vue中实现无感刷新token的方法可以通过拦截器来实现。首先,你可以在axios的响应拦截器中判断返回的状态码是否为401,如果是的话,说明token已过期。然后,你可以设置一个变量isRefreshing来控制是否正在刷新token的状态。当第一个请求触发token过期时,isRefreshing会被设置为true,然后发送刷新token请求,获取最新的token进行覆盖。在刷新token的过程中,其他请求会被拦截等待token刷新完成后再继续发送。当token刷新完成后,isRefreshing会被设置为false,其他请求继续发送。以下是一个示例代码: ```javascript import axios from 'axios' let isRefreshing = false axios.interceptors.response.use( response => { if (response.status === 401) { if (!isRefreshing) { isRefreshing = true return refreshToken().then(res => { const { token } = res.data // 更新token response.headers.Authorization = `${token}` // 继续发送之前被拦截的请求 return axios(response.config) }).catch(err => { // 刷新token失败,跳转到登录页 router.push('/login') return Promise.reject(err) }).finally(() => { isRefreshing = false }) } } return response && response.data }, error => { // 处理错误 return Promise.reject(error) } ) ``` 在上述代码中,refreshToken()是一个发送刷新token请求的函数,你可以根据你的实际情况进行实现。当token过期时,其他请求会被拦截等待token刷新完成后再继续发送。这样就实现了Vue中的无感刷新token的功能。\[1\]\[2\]\[3\] #### 引用[.reference_title] - *1* [VUE中拦截请求无感知刷新token](https://blog.csdn.net/Wangyuan_wo/article/details/121209540)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [Vue 无感刷新token](https://blog.csdn.net/weixin_45308405/article/details/127643153)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [VUE前端实现token无感刷新,即refresh_token](https://blog.csdn.net/yu1431/article/details/130835868)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值