vue无感刷新Token并重新请求

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

javascript 复制代码
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 = [];
};
相关推荐
练习前端两年半15 分钟前
【Vue3 高级技巧】函数重载+Watch:打造类型安全的通用事件监听 Hook
前端·javascript·vue.js
一只小鸟儿26 分钟前
门户短信发送验证码及验证功能
前端·javascript·jquery
elangyipi12329 分钟前
pnpm :下一代包管理工具的原理与实践
前端·npm
代码的奴隶(艾伦·耶格尔)38 分钟前
Sentinel限流熔断
java·前端·sentinel
talenteddriver40 分钟前
mysql: MySQL中between子句和limit子句的区别
前端·javascript·数据库
A24207349301 小时前
深入浅出理解AJAX:核心原理与POST/GET区别详解
前端·ajax·okhttp
LYFlied1 小时前
【每日算法】LeetCode 300. 最长递增子序列
前端·数据结构·算法·leetcode·职场和发展
张较瘦_1 小时前
前端 | 代码可读性 + SEO 双提升!HTML 语义化标签实战教程
前端·html
似水流年QC1 小时前
前端国际化实战指南:i18n 工程化最佳实践总结
前端
GISer_Jing1 小时前
企业级前端脚手架:原理与实战指南
前端·前端框架