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 = [];
};
相关推荐
初遇你时动了情3 小时前
css中backdrop-filter 详细使用 ios毛玻璃效果、filter和backdrop-filter使用说明
前端·css
景彡先生4 小时前
Python Selenium详解:从入门到实战,Web自动化的“瑞士军刀”
前端·python·selenium
Liudef066 小时前
DeepseekV3.2 实现构建简易版Wiki系统:从零开始的HTML实现
前端·javascript·人工智能·html
景早7 小时前
vue 记事本案例详解
前端·javascript·vue.js
wangjialelele8 小时前
Qt中的常用组件:QWidget篇
开发语言·前端·c++·qt
乔冠宇9 小时前
vue需要学习的点
前端·vue.js·学习
用户47949283569159 小时前
同样是 #,锚点和路由有什么区别
前端·javascript
Hero_11279 小时前
在pycharm中install不上需要的包
服务器·前端·pycharm
爱上妖精的尾巴9 小时前
5-26 WPS JS宏数组元素添加删除应用
开发语言·前端·javascript·wps·js宏
是谁眉眼9 小时前
wpsapi
前端·javascript·html