axios取消器重写

This commit is contained in:
ray_wuhao 2023-02-27 17:19:19 +08:00
parent e105726dde
commit ce83a9c30b
4 changed files with 127 additions and 19 deletions

83
src/axios/canceler.ts Normal file
View File

@ -0,0 +1,83 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-02-27
*
* @workspace ray-template
*
* @remark
*/
/**
*
*
*
*
*/
import type { AxiosRequestConfig } from 'axios'
export default class RequestCanceler {
pendingRequest: Map<string, AbortController>
constructor() {
this.pendingRequest = new Map<string, AbortController>()
}
/**
*
* @param config config
* @returns key
*
* @remark config request key
*/
generateRequestKey(config: AxiosRequestConfig): string {
const { method, url } = config
return [
url || '',
method || '',
JSON.stringify(config.params),
JSON.stringify(config.data),
].join('&')
}
/**
*
* @param config axios request config
*
* @remark signal ,
*/
addPendingRequest(config: AxiosRequestConfig) {
const requestKey: string = this.generateRequestKey(config)
if (!this.pendingRequest.has(requestKey)) {
const controller = new AbortController()
config.signal = controller.signal
this.pendingRequest.set(requestKey, controller)
} else {
// 如果已经有该 key 则重新挂载 signal
config.signal = (
this.pendingRequest.get(requestKey) as AbortController
).signal
}
}
/**
*
* @param config axios request config
*
* @remark , map generateRequestKey value
*/
removePendingRequest(config: AxiosRequestConfig) {
const requestKey = this.generateRequestKey(config)
if (this.pendingRequest.has(requestKey)) {
;(this.pendingRequest.get(requestKey) as AbortController).abort()
this.pendingRequest.delete(requestKey)
}
}
}

View File

@ -1,28 +1,30 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-02-27
*
* @workspace ray-template
*
* @remark
*/
/**
*
* :
* - ,
* - 西, axios instance config
* - , 使 instance
*/
import request from './instance'
import type { AxiosRequestConfig } from 'axios'
let controller: AbortController
const useRequest = (config: AxiosRequestConfig) => {
controller && controller.abort() // 调用控制器, 取消请求
controller = new AbortController() // 实例化控制器对象(可以中止一个或多个 `Web` 请求)
const cfg = Object.assign({}, config, {
signal: controller.signal, // 取消请求信号
})
const cfg = Object.assign({}, config, {})
return request(cfg)
}
export default useRequest
/**
*
* 使 `axios`
*
*
*
* 使, 使 `import { xxx } from '@use-api/xxx' 导入
*/

View File

@ -1,9 +1,23 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2022-11-18
*
* @workspace ray-template
*
* @remark
*/
import axios from 'axios'
import { useDetermineEnv } from '@use-utils/hook'
import RequestCanceler from './canceler'
import type { RawAxiosRequestHeaders, AxiosRequestConfig } from 'axios'
import type { RequestHeaderOptions } from './type'
const canceler = new RequestCanceler()
/**
*
* @param instance axios instance
@ -26,6 +40,9 @@ const server = axios.create({
baseURL: '', // `import.meta.env`,
withCredentials: false, // 是否允许跨域携带 `cookie`
timeout: 5 * 1000,
headers: {
'Content-Type': 'application/json',
},
})
server.interceptors.request.use(
@ -47,6 +64,9 @@ server.interceptors.request.use(
},
]) // 自定义请求头
canceler.removePendingRequest(request) // 检查是否存在重复请求, 若存在则取消已发的请求
canceler.addPendingRequest(request) // 把当前的请求信息添加到 pendingRequest 表中
return request
},
(error) => {
@ -56,11 +76,15 @@ server.interceptors.request.use(
server.interceptors.response.use(
(response) => {
canceler.removePendingRequest(response.config)
const { data } = response
return Promise.resolve(data)
},
(error) => {
canceler.removePendingRequest(error.config || {})
return Promise.reject(error)
},
)

View File

@ -63,7 +63,6 @@ const Axios = defineComponent({
<NLayoutHeader bordered>
<NCard title="请求函数">
axios , , .
,
</NCard>
</NLayoutHeader>
<NLayoutHeader bordered>