# 1. 前言

在官方的axios中,默认配置对象里面还提供了transformRequesttransformResponse 这两个属性,它们的值可以是一个函数或者是一个由多个函数组成的数组。官方对这个属性介绍如下:

  • transformRequest: 允许你在将请求数据发送到服务器之前对其进行修改,这只适用于请求方法 putpostpatch,如果值是数组,则数组中的最后一个函数必须返回一个字符串或 FormDataURLSearchParamsBlob 等类型作为 xhr.send 方法的参数,而且在 transform 过程中可以修改 headers 对象。

    // `transformRequest` allows changes to the request data before it is sent to the server
    // This is only applicable for request methods 'PUT', 'POST', 'PATCH' and 'DELETE'
    // The last function in the array must return a string or an instance of Buffer, ArrayBuffer,
    // FormData or Stream
    // You may modify the headers object.
    transformRequest: [
      function(data, headers) {
        // Do whatever you want to transform the data
    
        return data;
      },
    ];
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
  • transformResponse: 允许你在把响应数据传递给 then 或者 catch 之前对它们进行修改。

    // `transformResponse` allows changes to the response data to be made before
    // it is passed to then/catch
    transformResponse: [
      function(data) {
        // Do whatever you want to transform the data
    
        return data;
      },
    ];
    
    1
    2
    3
    4
    5
    6
    7
    8
    9

当值为数组的时候,数组的每一个函数都是一个转换函数,数组中的函数就像管道一样依次执行,前者的输出作为后者的输入。

OK,了解了这两个属性的作用后,我们就来实现它。其实实现起来也没啥难度,因为我们之前在封装dispatchRequest函数时,在该函数内部的processConfig函数中我们已经对请求的dataheaders和响应的data做了转换,大不了现在把它们都抽离到默认配置的transformRequesttransformResponse 这两个属性里就完事了。

# 2. 修改默认配置对象

我们需要修改之前创建的默认配置对象defaults,为其添加transformRequesttransformResponse 这两个属性,并且把之前封装dispatchRequest函数时对请求的dataheaders和响应的data做的转换分别抽离到这两个属性内。

修改defaults之前我们先来修改AxiosRequestConfig接口类型定义,因为defaults的类型是AxiosRequestConfig,我们要在AxiosRequestConfig接口定义中添加transformRequesttransformResponse 这两个属性,修改如下:

// src/types/index.ts

export interface AxiosRequestConfig {
  url?: string;
  method?: Method;
  headers?: any;
  data?: any;
  params?: any;
  responseType?: XMLHttpRequestResponseType;
  timeout?: number;
  transformRequest?: AxiosTransformer | AxiosTransformer[];
  transformResponse?: AxiosTransformer | AxiosTransformer[];
  [propName: string]: any;
}

export interface AxiosTransformer {
  (data: any, headers?: any): any;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

由于transformRequesttransformResponse 它们的值都要么是一个转换函数,要么是一个由转换函数组成的数组,所以我们为转换函数单独定义了接口类型AxiosTransformer

接口定义修改好之后,我们就可以在src/defaults.ts中修改之前创建好的默认配置对象了,如下:

import { processHeaders } from "./helpers/header";
import { transformRequest, transformResponse } from "./helpers/data";

const defaults: AxiosRequestConfig = {
  timeout: 0,
  headers: {
    common: {
      Accept: "application/json, text/plain, */*",
    },
  },
  transformRequest: [
    function(data: any, headers: any): any {
      processHeaders(headers, data);
      return transformRequest(data);
    },
  ],
  transformResponse: [
    function(data: any) {
      return transformResponse(data);
    },
  ],
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22

我们把之前封装dispatchRequest函数和processConfig函数时对请求的dataheaders和响应的data做的转换分别抽离到这两个属性内,然后在dispatchRequest函数和processConfig函数内调用默认配置里的transformRequesttransformResponse 就好了,如下:

function processConfig(config: AxiosRequestConfig): void {
  config.url = transformUrl(config);
  config.data = transform(config.data, config.headers, config.transformRequest);
  config.headers = flattenHeaders(config.headers, config.method!);
}

function transformResponseData(res: AxiosResponse): AxiosResponse {
  res.data = transform(res.data, res.headers, res.config.transformResponse);
  return res;
}
1
2
3
4
5
6
7
8
9
10

由于transformRequesttransformResponse 这两个属性值有可能是多个转换函数构成的数组,而且当执行这些转换函数的时候,前一个转换函数的返回输出值是后一个转换函数的输入值,针对这一问题,我们创建了transform函数,在该函数内部遍历执行所有的转换函数,并且把前一个转换函数的返回值作为参数传给后一个转换函数。在外层我们只需调用transform函数即可。

# 3. 实现 transform 函数

正如上面所说,该函数的主要作用是执行所有的转换函数,并且把前一个转换函数的返回值作为参数传给后一个转换函数。我们在src/core目录下创建transform.ts,其实现如下:

import { AxiosTransformer } from "../types";

export default function transform(
  data: any,
  headers: any,
  fns?: AxiosTransformer | AxiosTransformer[]
) {
  if (!fns) {
    return data;
  }
  if (!Array.isArray(fns)) {
    fns = [fns];
  }
  fns.forEach((fn) => {
    data = fn(data, headers);
  });
  return data;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

代码说明:

  • 该函数接收三个参数,待转换的数据data、待转换的headers以及所有的转换函数。
  • 首先判断转换函数是否为空,若为空,表示不进行任何转换,则直接把data返回;
  • 然后再判断转换函数是否为数组,若不为数组,则将其强制变成一个长度为 1 的数组,这是为了下面可以统一遍历;
  • 遍历所有的转换函数并执行,执行的时候每个转换函数返回的 data 会作为下一个转换函数的参数 data 传入。

OK,transformRequesttransformResponse 这两个属性的添加以及必要的逻辑就已经实现完毕了,接下来我们就编写demo测试下效果如何。

# 4. demo 编写

examples 目录下创建 transformData目录,在 transformData目录下创建 index.html:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>transformData demo</title>
  </head>
  <body>
    <script src="/__build__/transformData.js"></script>
  </body>
</html>
1
2
3
4
5
6
7
8
9
10

接着再创建 app.ts 作为入口文件:

import axios from "../../src/axios";
import { AxiosTransformer } from "../../src/types";

axios({
  url: "/api/transformData",
  method: "post",
  data: {
    a: 1,
  },
  transformRequest: [
    function(data) {
      data.a = data.a + 1;
      return data;
    },
    ...(axios.defaults.transformRequest as AxiosTransformer[]),
  ],
  transformResponse: [
    ...(axios.defaults.transformResponse as AxiosTransformer[]),
    function(data) {
      data.b = "对响应进行了转换";
      return data;
    },
  ],
}).then((res) => {
  console.log(res.data);
});
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

接着在 server/server.js 添加新的接口路由:

// 添加transformRequest 和 transformResponse
router.post("/api/transformData", function(req, res) {
  res.json(req.body);
});
1
2
3
4

最后在根目录下的index.html中加上启动该demo的入口:

<li><a href="examples/transformData">transformData</a></li>
1

OK,我们在命令行中执行:

# 同时开启客户端和服务端
npm run server | npm start
1
2

接着我们打开 chrome 浏览器,访问 http://localhost:8000/ (opens new window) 即可访问我们的 demo 了,我们点击 transformData,通过F12network 部分我们可以看到请求已正常发出,并且对请求的data.a加了 1,响应的data也添加了b属性,如下:

OK,请求和响应的配置化就完成了。