Javascript 如何全面接管xhr请求
背景及思考
为什么需要接管xhr请求?这就需要我们了解它的一些应用场景。我们如何统一项目中xhr请求的行为,监控请求的整个生命周期、如何自定义拦截请求并返回mock数据、如何制定完全可控的控制台(如vconsole那样) 、如何监控所有api请求的健康状态 等等!
有一种最常见的情况。比如项目中发起请求的方式不一,有的在js sdk或私有npm库
中发起、有的在引入了第三方js cdn
中发起、有的由项目中统一的ajax、axios
发起。如果我们需要对项目中所有请求增加某些统一的行为该如何处理了?
原生XMLHttpRequest 回顾
使用xhr发起请求
注:以下只针对xhr的处理,不考虑使用ActiveXObject
来处理兼容性,不考虑使用fetch请求。
// 创建 XMLHttpRequest 对象
var xhr = new XMLHttpRequest ();
// 建立连接
xhr.open(method, url, async, username, password);
// 在open后,send前 可对报文进行处理,如设置请求头
xhr.setRequestHeader('customId', 666)
// 对于异步请求,绑定响应状态事件监听函数
xhr.onreadystatechange = function () {
//监听readyState状态、http状态码
if (xhr.readyState == 4 && xhr.status == 200) {
console.log(xhr.responseText); // 接收数据
}
}
// 使用 send() 方法发送请求
xhr.send(body);
//对于同步请求,可直接接收数据
console.log(xhr.responseText);
//中止请求
xhr.onreadystatechange = function () {}; //清理事件响应函数(IE、火狐兼容性处理)
xhr.abort();
ES5实现局部拦截
假设使用ajax、axios、原生xhr在请求时增加了自定义的请求头custom-trace-id:'aa,bb'
。 我们如何通过拦截获取到其值,并增加两个新的请求头'custom-a': 'aa'
和'custom-b': 'bb'
(分割custom-trace-id的值获取到'aa'和'bb')
拦截项目中所有xhr, 并给有'custom-trace'的头增加新的自定义请求头(仅拦截open和setRequestHeader)
(function(w){
w.rewriteXhr = {
// 随机生成uuid
_setUUID: function () {
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
var r = Math.random() * 16 | 0,
v = c == 'x' ? r : (r & 0x3 | 0x8);
return v.toString(16);
});
},
// 储存xhr原型
xhrProto:w.XMLHttpRequest.prototype,
// 存储需要拦截的局部属性或方法
tempXhrProto: function(){
this._open = this.xhrProto.open
this._setRequestHeader = this.xhrProto.setRequestHeader
},
// 拦截处理
proxy: function(){
var _this = this
this.xhrProto.open = function () {
this._open_args = [...arguments];
return _this._open.apply(this, arguments);
}
// 拦截setRequestHeader方法
this.xhrProto.setRequestHeader = function () {
var headerKey = arguments[0]
// 需要给所有请求增加的头
var keys = ['custom-a', 'custom-b', 'custom-uuid']
// 可使用url做过滤处理
// var url = this.open_args && this.open_args[1]
if(/^custom-trace-id$/.test(headerKey)){
var value = arguments[1] && arguments[1].split(',')
value[2] = _this._setUUID()
keys.forEach((key, index)=>{
// 也可以直接使用_this._setRequestHeader.apply(this, arguments)
this.setRequestHeader(key, value[index])
})
return
}
return _this._setRequestHeader.apply(this, arguments)
}
},
init: function(){
this.tempXhrProto()
this.proxy()
}
}
w.rewriteXhr.init()
})(window)
以上,我们重新定义了open
和setRequestHeader
的原型方法(拦截open的目的在于只能在该方法的参数中获取到url等信息),同时也储存了原始的open
和setRequestHeader
。在每次有请求调用到setRequestHeader时,实际调用的是我们自己重写的setRequestHeader
方法,在该方法里面再去调用原始的setRequestHeader
,从而实现拦截设置请求头的目的。
了解了局部的xhr拦截,我们可以以此来思索如何封装实现全局的请求拦截?
ES5实现全局拦截
在项目中使用xhrHook
xhrHook({
open: function (args, xhr) {
console.log("open called!", args, xhr)
},
setRequestHeader: function (args, xhr) {
console.log("setRequestHeader called!", args, xhr)
},
onload: function (xhr) {
// 对响应结果做处理
this.responseText = xhr.responseText.replace('abc', '')
}
})
xhrHook 的实现
在全局拦截中,我们需要考虑到实例的属性、方法及事件的处理。
function xhrHook(config){
// 存储真实的xhr构造器, 在取消hook时,可恢复
window.realXhr = window.realXhr || XMLHttpRequest
// 重写XMLHttpRequest构造函数
XMLHttpRequest = function(){
var xhr = new window.realXhr()
// 遍历实例及其原型上的属性(实例和原型链上有相同属性时,取实例属性)
for (var attr in xhr) {
if (Object.prototype.toString.call(a) === '[object Function]') {
this[attr] = hookFunction(attr); // 接管xhr function
} else {
Object.defineProperty(this, attr, { // 接管xhr attr、event
get: getterFactory(attr),
set: setterFactory(attr),
enumerable: true
})
}
}
// 真实的xhr实例存储到自定义的xhr属性中
this.xhr = xhr
}
}
xhr中的方法拦截
// xhr中的方法拦截,eg: open、send etc.
function hookFunction(fun) {
return function () {
var args = Array.prototype.slice.call(arguments)
// 将open参数存入xhr, 在其它事件回调中可以获取到。
if(fun === 'open'){
this.xhr.open_args = args
}
if (config[fun]) {
// 配置的函数执行结果返回为true时终止调用
var result = config[fun].call(this, args, this.xhr)
if (result) return result;
}
return this.xhr[fun].apply(this.xhr, args);
}
}
xhr中的属性和事件的拦截
// 属性及回调方法拦截
function getterFactory() {
var value = this.xhr[attr]
var getter = (proxy[attr] || {})["getter"]
return getterHook && getterHook(value, this) || value
}
// 在赋值时触发该工厂函数(如onload等事件)
function setterFactory(attr) {
return function (value) {
var xhr = this.xhr;
var _this = this;
var hook = config[attr]; // 方法或对象
if (/^on/.test(attr)) {
// 在真实的xhr上给事件绑定函数
xhr[attr] = function (e) {
e = configEvent(e, _this)
var result = hook && hook.call(_this, xhr, e)
result || value.call(_this, e);
}
} else {
var attrSetterHook = (hook || {})["setter"]
value = attrSetterHook && attrSetterHook(value, _this) || value
try {
xhr[attr] = value;
} catch (e) {
console.warn('xhr的'+attr+'属性不可写')
}
}
}
}
解除xhr拦截,归还xhr管理权
// 归还xhr管理权
function unXhrHook() {
if (window[realXhr]) XMLHttpRequest = window[realXhr];
window[realXhr] = undefined;
}
ES6实现全局拦截
夜已深,等待整理中......
总结
xhr的全局拦截总体来说比较简单,除了对事件的托管流程有点复杂。不管是局部还是全局处理,共同的特点是都要存储原生的xhr, 但在执行原生的属性、方法、事件时,会先执行自己的处理函数,在函数中执行一些操作,最后再去执行原生的方法。
对于事件的拦截,比如我们在定义xhr.onload = function(){}
时,实际触发的是自己定义的onload
的setter
方法,在该方法中会去给真实的xhr绑定回调函数onload,并在回调函数中去执行config.onload
中的逻辑、如果config.onload()
没有返回或返回false
, 会继续执行之前在外面绑定的xhr.onload函数。
如有不足之处、疑问或建议,欢迎大家留言指出。
作者:tager
链接:https://juejin.cn/post/7019704757556084750
说明:稀土掘金同步更新
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。