uniapp 实现 请求拦截 和 对请求的封装

229 阅读1分钟
const baseUrl = 'https://pcapi-xiaotuxian-front-devtest.itheima.net'

// 拦截器配置
/* 拦截器配置核心:addInterceptor
通过 addINterceptor 配置拦截,配置相对应的对象 httpIntertor
httpIntertor 中 通过 invoke 进行拦截
invoke 中配置
头字段是否为 http
超时响应
配置 header
是否携带 token
 */
const httpIntertor = {
  invoke(options: UniApp.RequestOptions) {
    if (!options.url.startsWith('http')) {
      options.url = baseUrl + options.url
    }
    options.timeout = 10000
    options.header = {
      ...options.header,
      'source-client': 'miniapp',
    }
    // 判断token
    const memberStore = useMemberStore()
    const token = memberStore.profile?.token
    if (token) {
      options.header.Authorization = token
    }
  },
}

type Data<T> = {
  code: string
  msg: string
  result: T
}

// 返回请求结果
/*
对返回的请求进行处理,
  返回成功的结果 success
    resolve:返回成功的结果。
    若返回结果不为理想的结果,因为 success 会直接走 resolve,
    所以需要通过对返回的状态码进行判断返回的结果。
      状态码在 200至300
      状态码为401
      其他错误
  返回失败的结果 fail
    reject:返回失败的结果
 */
export const http = <T>(options: UniApp.RequestOptions) => {
  return new Promise<Data<T>>((resolve, reject) => {
    uni.request({
      ...options,
      success(res) {
        // 响应成功
        if (res.statusCode >= 200 && res.statusCode < 300) {
          resolve(res.data as Data<T>)
        } else if (res.statusCode === 401) {
          const memberStore = useMemberStore()
          memberStore.clearProfile()
          uni.navigateTo({ url: '/pages/login/login' })
          reject(res)
        } else {
          uni.showToast({
            icon: 'none',
            title: (res.data as Data<T>).msg || '请求错误',
          })
          reject(res)
        }
      },

      fail(err) {
        uni.showToast({
          icon: 'none',
          title: '未连接到网络',
        })
        reject(err)
      },
    })
  })
}

uni.addInterceptor('request', httpIntertor)