数据 mock&联调

开发环境

如果前端应用和后端接口服务器没有运行在同一个主机上,你需要在开发环境下将接口请求代理到接口服务器。

如果是同一个主机,可以直接请求具体的接口地址。

配置

开发环境时候,接口地址在项目根目录下

.env.development 文件配置

  1. # vite 本地跨域代理
  2. VITE_PROXY=[["/api","http://localhost:8000"]]
  3. # 接口地址
  4. VITE_GLOB_API_URL=/api

::: tip

  • .env 文件中的字段如果是字符串,则无需加引号,默认全部为字符串
  • VITE_PROXY 不能换行

:::

跨域处理

如果你在 src/api/ 下面的接口为下方代码,且 .env.development 文件配置如下注释,则在控制台看到的地址为 http://localhost:8001/api/login

由于 /api 匹配到了设置的 VITE_PROXY,所以上方实际是请求 http://localhost:8000/login,这样同时也解决了跨域问题。(8001为项目端口号,http://localhost:8000为PROXY代理的目标地址)

  1. // .env.development
  2. // VITE_PROXY=[["/api","http://localhost:8000"]]
  3. // VITE_GLOB_API_URL=/api
  4. enum Api {
  5. Login = '/login',
  6. }
  7. /**
  8. * @description: 用户登陆
  9. */
  10. export function loginApi(params: LoginParams) {
  11. return http.request<LoginResultModel>({
  12. url: Api.Login,
  13. method: 'POST',
  14. params,
  15. });
  16. }

没有跨域时的配置

如果没有跨域问题,可以直接忽略 VITE_PROXY 配置,直接将接口地址设置在 VITE_GLOB_API_URL

  1. # 例如接口地址为 http://localhost:8000 则
  2. VITE_GLOB_API_URL=http://localhost:8000

如果有跨域问题,将 VITE_GLOB_API_URL 设置为跟 VITE_PROXY 内其中一个数组的第一个项一致的值即可。

下方的接口地址设置为 /api,当请求发出的时候会经过 Vite 的 proxy 代理,匹配到了我们设置的 VITE_PROXY 规则,将 /api 转化为 http://localhost:8000 进行请求

  1. # 例如接口地址为 http://localhost:8000 则
  2. VITE_PROXY=[["/basic-api","http://localhost:8000"]]
  3. # 接口地址
  4. VITE_GLOB_API_URL=/api

跨域原理解析

vite.config.ts 配置文件中,提供了 server 的 proxy 功能,用于代理 API 请求。

  1. server: {
  2. proxy: {
  3. "/api":{
  4. target: 'http://localhost:8000',
  5. changeOrigin: true,
  6. ws: true,
  7. rewrite: (path) => path.replace(new RegExp(`^/api`), ''),
  8. }
  9. },
  10. },

::: tip 注意

从浏览器控制台的 Network 看,请求是 http://localhost:8000/api/xxx,这是因为 proxy 配置不会改变本地请求的 url。

:::

生产环境

生产环境接口地址在项目根目录下 .env.production 文件配置。

生产环境接口地址值需要修改 VITE_GLOB_API_URL,如果出现跨域问题,可以使用 nginx 或者后台开启 cors 进行处理

::: tip 打包后如何进行地址修改?

VITEGLOB\* 开头的变量会在打包的时候注入 _app.config.js 文件内。

dist/_app.config.js 修改相应的接口地址后刷新页面即可,不需要在根据不同环境打包多次,一次打包可以用于多个不同接口环境的部署。

:::

接口请求

在 naive-ui-admin 中:

  1. 页面交互操作;
  2. 调用统一管理的 api 请求函数;
  3. 使用封装的 axios.ts 发送请求;
  4. 获取服务端返回数据
  5. 更新 data;

接口统一存放于 src/api/ 下面管理

以登陆接口为例:

src/api/ 内新建模块文件,其中参数与返回值最好定义一下类型,方便校验。虽然麻烦,但是后续维护字段很方便。

::: tip

类型定义文件可以抽取出去统一管理,具体参考项目,至于参数类型校验,可以单独抽离 model 引入

:::

  1. import http from '@/utils/http/axios';
  2. import { LoginParams, LoginResultModel } from './model/userModel';
  3. export function getTableList(params) {
  4. return http.request({
  5. url: '/table/list',
  6. method: 'get',
  7. params,
  8. });
  9. }

axios 配置

axios 请求封装存放于 src/utils/http/axios 文件夹内部

index.ts 文件内容需要根据项目自行修改外,其余文件无需修改

  1. ├── Axios.ts // axios实例
  2. ├── axiosCancel.ts // axiosCancel实例,取消重复请求
  3. ├── axiosTransform.ts // 数据转换类
  4. ├── checkStatus.ts // 返回状态值校验
  5. ├── index.ts // 接口返回统一处理

index.ts 配置说明

  1. const axios = new VAxios({
  2. timeout: 10 * 1000,
  3. // 接口前缀
  4. prefixUrl: urlPrefix,
  5. headers: { 'Content-Type': ContentTypeEnum.JSON },
  6. // 数据处理方式
  7. transform,
  8. // 配置项,下面的选项都可以在独立的接口请求中覆盖
  9. requestOptions: {
  10. // 默认将prefix 添加到url
  11. joinPrefix: true,
  12. // 需要对返回数据进行处理
  13. isTransformRequestResult: true,
  14. // post请求的时候添加参数到url
  15. joinParamsToUrl: false,
  16. // 格式化提交参数时间
  17. formatDate: true,
  18. // 消息提示类型
  19. errorMessageMode: 'none',
  20. // 接口地址
  21. apiUrl: globSetting.apiUrl as string,
  22. },
  23. withCredentials: false,
  24. });

transform 数据处理说明

类型定义,见 axiosTransform.ts 文件

  1. export abstract class AxiosTransform {
  2. /**
  3. * @description: 请求之前处理配置
  4. */
  5. beforeRequestHook?: (config: AxiosRequestConfig, options: RequestOptions) => AxiosRequestConfig;
  6. /**
  7. * @description: 请求成功处理
  8. */
  9. transformRequestData?: (res: AxiosResponse<Result>, options: RequestOptions) => any;
  10. /**
  11. * @description: 请求失败处理
  12. */
  13. requestCatch?: (e: Error) => Promise<any>;
  14. /**
  15. * @description: 请求之前的拦截器
  16. */
  17. requestInterceptors?: (config: AxiosRequestConfig) => AxiosRequestConfig;
  18. /**
  19. * @description: 请求之后的拦截器
  20. */
  21. responseInterceptors?: (res: AxiosResponse<any>) => AxiosResponse<any>;
  22. /**
  23. * @description: 请求之前的拦截器错误处理
  24. */
  25. requestInterceptorsCatch?: (error: Error) => void;
  26. /**
  27. * @description: 请求之后的拦截器错误处理
  28. */
  29. responseInterceptorsCatch?: (error: Error) => void;
  30. }

项目默认 transform 处理逻辑,可以根据各自项目进行处理。一般需要更改的部分为下方代码,见代码注释说明

  1. /**
  2. * @description: 数据处理,方便区分多种处理方式
  3. */
  4. const transform: AxiosTransform = {
  5. /**
  6. * @description: 处理请求数据
  7. */
  8. transformRequestData: (res: AxiosResponse<Result>, options: RequestOptions) => {
  9. const {
  10. isTransformRequestResult,
  11. isShowMessage = true,
  12. isShowErrorMessage,
  13. isShowSuccessMessage,
  14. successMessageText,
  15. errorMessageText,
  16. } = options;
  17. const reject = Promise.reject;
  18. const { data } = res;
  19. // 这里 code,result,message为 后台统一的字段,需要在 types.ts内修改为项目自己的接口返回格式
  20. const { code, result, message } = data;
  21. // 请求成功
  22. const hasSuccess = data && Reflect.has(data, 'code') && code === ResultEnum.SUCCESS;
  23. // 是否显示提示信息
  24. if (isShowMessage) {
  25. if (hasSuccess && (successMessageText || isShowSuccessMessage)) {
  26. // 是否显示自定义信息提示
  27. Message.success(successMessageText || message || '操作成功!');
  28. } else if (!hasSuccess && (errorMessageText || isShowErrorMessage)) {
  29. // 是否显示自定义信息提示
  30. Message.error(message || errorMessageText || '操作失败!');
  31. } else if (!hasSuccess && options.errorMessageMode === 'modal') {
  32. // errorMessageMode=‘custom-modal’的时候会显示modal错误弹窗,而不是消息提示,用于一些比较重要的错误
  33. Modal.confirm({ title: '错误提示', content: message });
  34. }
  35. }
  36. // 不进行任何处理,直接返回
  37. // 用于页面代码可能需要直接获取code,data,message这些信息时开启
  38. if (!isTransformRequestResult) {
  39. return res.data;
  40. }
  41. if (!data) {
  42. // return '[HTTP] Request has no return value';
  43. return reject(data);
  44. }
  45. // 接口请求成功,直接返回结果
  46. if (code === ResultEnum.SUCCESS) {
  47. return result;
  48. }
  49. // 接口请求错误,统一提示错误信息
  50. if (code === ResultEnum.ERROR) {
  51. if (message) {
  52. Message.error(data.message);
  53. Promise.reject(new Error(message));
  54. } else {
  55. const msg = '操作失败,系统异常!';
  56. Message.error(msg);
  57. Promise.reject(new Error(msg));
  58. }
  59. return reject();
  60. }
  61. // 登录超时
  62. if (code === ResultEnum.TIMEOUT) {
  63. if (router.currentRoute.value.name == 'login') return;
  64. // 到登录页
  65. const timeoutMsg = '登录超时,请重新登录!';
  66. Modal.destroyAll();
  67. Modal.warning({
  68. title: '提示',
  69. content: '登录身份已失效,请重新登录!',
  70. onOk: () => {
  71. router.replace({
  72. name: 'login',
  73. query: {
  74. redirect: router.currentRoute.value.fullPath,
  75. },
  76. });
  77. storage.clear();
  78. },
  79. });
  80. return reject(new Error(timeoutMsg));
  81. }
  82. // 这里逻辑可以根据项目进行修改
  83. if (!hasSuccess) {
  84. return reject(new Error(message));
  85. }
  86. return data;
  87. },
  88. // 请求之前处理config
  89. beforeRequestHook: (config, options) => {
  90. const { apiUrl, joinPrefix, joinParamsToUrl, formatDate, joinTime = true } = options;
  91. if (joinPrefix) {
  92. config.url = `${urlPrefix}${config.url}`;
  93. }
  94. if (apiUrl && isString(apiUrl)) {
  95. config.url = `${apiUrl}${config.url}`;
  96. }
  97. const params = config.params || {};
  98. if (config.method?.toUpperCase() === RequestEnum.GET) {
  99. if (!isString(params)) {
  100. // 给 get 请求加上时间戳参数,避免从缓存中拿数据。
  101. config.params = Object.assign(params || {}, joinTimestamp(joinTime, false));
  102. } else {
  103. // 兼容restful风格
  104. config.url = config.url + params + `${joinTimestamp(joinTime, true)}`;
  105. config.params = undefined;
  106. }
  107. } else {
  108. if (!isString(params)) {
  109. formatDate && formatRequestDate(params);
  110. config.data = params;
  111. config.params = undefined;
  112. if (joinParamsToUrl) {
  113. config.url = setObjToUrlParams(config.url as string, config.data);
  114. }
  115. } else {
  116. // 兼容restful风格
  117. config.url = config.url + params;
  118. config.params = undefined;
  119. }
  120. }
  121. return config;
  122. },
  123. /**
  124. * @description: 请求拦截器处理
  125. */
  126. requestInterceptors: (config) => {
  127. // 请求之前处理config
  128. const userStore = useUserStoreWidthOut();
  129. const token = userStore.getToken;
  130. if (token) {
  131. // jwt token
  132. config.headers.token = token;
  133. }
  134. return config;
  135. },
  136. /**
  137. * @description: 响应错误处理
  138. */
  139. responseInterceptorsCatch: (error: any) => {
  140. const { response, code, message } = error || {};
  141. const msg: string =
  142. response && response.data && response.data.error ? response.data.error.message : '';
  143. const err: string = error.toString();
  144. try {
  145. if (code === 'ECONNABORTED' && message.indexOf('timeout') !== -1) {
  146. Message.error('接口请求超时,请刷新页面重试!');
  147. return;
  148. }
  149. if (err && err.includes('Network Error')) {
  150. Modal.confirm({
  151. title: '网络异常',
  152. content: '请检查您的网络连接是否正常!',
  153. });
  154. return;
  155. }
  156. } catch (error) {
  157. throw new Error(error);
  158. }
  159. // 请求是否被取消
  160. const isCancel = axios.isCancel(error);
  161. if (!isCancel) {
  162. checkStatus(error.response && error.response.status, msg);
  163. } else {
  164. console.warn(error, '请求被取消!');
  165. }
  166. return error;
  167. },
  168. };

更改参数格式

项目接口默认为 Json 参数格式,即 headers: { 'Content-Type': ContentTypeEnum.JSON },

如果需要更改为 form-data 格式,更改 headers 的 'Content-TypeContentTypeEnum.FORM_URLENCODED 即可

多个接口地址

当项目中需要用到多个接口地址时, 可以在 src/utils/http/axios/index.ts 导出多个 axios 实例

  1. // 目前只导出一个默认实例,接口地址对应的是环境变量中的 VITE_GLOB_API_URL 接口地址
  2. export const Axios = new VAxios({});
  3. // 需要有其他接口地址的可以在后面添加
  4. // other api url
  5. export const AxiosTow = new VAxios({
  6. requestOptions: {
  7. apiUrl: 'xxx',
  8. },
  9. });

删除请求 URL 携带的时间戳参数

如果不需要 url 上面默认携带的时间戳参数 ?_t=xxx

  1. const axios = new VAxios({
  2. requestOptions: {
  3. // 是否加入时间戳
  4. joinTime: false,
  5. },
  6. });

Mock 服务

Mock 数据是前端开发过程中必不可少的一环,是分离前后端开发的关键链路。通过预先跟服务器端约定好的接口,模拟请求数据甚至逻辑,能够让前端开发独立自主,不会被服务端的开发进程所阻塞。

本项目使用 vite-plugin-mock 来进行 mock 数据处理。项目内 mock 服务分本地和线上

本地 Mock

本地 mock 采用 Node.js 中间件进行参数拦截(不采用 mock.js 的原因是本地开发看不到请求参数和响应结果)。

如何新增 mock 接口

如果你想添加 mock 数据,只要在根目录下找到 mock 文件,添加对应的接口,对其进行拦截和模拟数据。

在 mock 文件夹内新建文件

::: tip

文件新增后会自动更新,不需要手动重启,可以在代码控制台查看日志信息 mock 文件夹内会自动注册,排除以_开头的文件夹及文件

:::

例:

  1. import { Random } from 'mockjs';
  2. import { resultSuccess, doCustomTimes } from '../_util';
  3. const tableList = (pageSize) => {
  4. const result: any[] = [];
  5. doCustomTimes(pageSize, () => {
  6. result.push({
  7. id: '@integer(10,100)',
  8. beginTime: '@datetime',
  9. endTime: '@datetime',
  10. address: '@city()',
  11. name: '@cname()',
  12. avatar: Random.image('400x400', Random.color(), Random.color(), Random.first()),
  13. date: `@date('yyyy-MM-dd')`,
  14. time: `@time('HH:mm')`,
  15. 'no|100000-10000000': 100000,
  16. 'status|1': [true, false],
  17. });
  18. });
  19. return result;
  20. };
  21. export default [
  22. //表格数据列表
  23. {
  24. url: '/api/table/list',
  25. timeout: 1000,
  26. method: 'get',
  27. response: ({ query }) => {
  28. const { page = 1, pageSize = 10 } = query;
  29. const list = tableList(Number(pageSize));
  30. return resultSuccess({
  31. page: Number(page),
  32. pageSize: Number(pageSize),
  33. pageCount: 60,
  34. list,
  35. });
  36. },
  37. },
  38. ];

::: tip

mock 的值可以直接使用 mockjs 的语法。

:::

接口格式

  1. {
  2. url: string; // mock 接口地址
  3. method?: MethodType; // 请求方式
  4. timeout?: number; // 延时时间
  5. statusCode: number; // 响应状态码
  6. response: ((opt: { // 响应结果
  7. body: any;
  8. query: any;
  9. }) => any) | object;
  10. }

参数获取

GET 接口:({ query }) => { }

POST 接口:({ body }) => { }

util 说明

可在 代码 中查看

::: tip

util 只作为服务处理结果数据使用。可以不用,如需使用可自行封装,需要将对应的字段改为接口的返回结构

:::

匹配

src/api 下面,如果接口匹配到 mock,则会优先使用 mock 进行响应

  1. import http from '@/utils/http/axios';
  2. //获取table
  3. export function getTableList(params) {
  4. return http.request({
  5. url: '/table/list',
  6. method: 'get',
  7. params,
  8. });
  9. }
  10. // 会匹配到上方的
  11. export default [
  12. //表格数据列表
  13. {
  14. url: '/api/table/list',
  15. timeout: 1000,
  16. method: 'get',
  17. response: ({ query }) => {
  18. const { page = 1, pageSize = 10 } = query;
  19. const list = tableList(Number(pageSize));
  20. return resultSuccess({
  21. page: Number(page),
  22. pageSize: Number(pageSize),
  23. pageCount: 60,
  24. list,
  25. });
  26. },
  27. },
  28. ];

接口有了,如何去掉 mock

当后台接口已经开发完成,只需要将相应的 mock 函数去掉即可。

以上方接口为例,假如后台接口 login 已经开发完成,则只需要删除/注释掉下方代码即可

  1. export default [
  2. //表格数据列表
  3. {
  4. url: '/api/table/list',
  5. timeout: 1000,
  6. method: 'get',
  7. response: ({ query }) => {
  8. const { page = 1, pageSize = 10 } = query;
  9. const list = tableList(Number(pageSize));
  10. return resultSuccess({
  11. page: Number(page),
  12. pageSize: Number(pageSize),
  13. pageCount: 60,
  14. list,
  15. });
  16. },
  17. },
  18. ];

线上 mock

由于该项目是一个展示类项目,线上也是用 mock 数据,所以在打包后同时也集成了 mock。通常项目线上一般为正式接口。

项目线上 mock 采用的是 mockjs 进行 mock 数据模拟。

线上如何开启 mock

::: warning 注意

线上开启 mock 只适用于一些简单的示例网站及预览网站。一定不要在正式的生产环境开启!!!

:::

  1. 修改 .env.production 文件内的 VITE_USE_MOCK 的值为 true
  1. VITE_USE_MOCK = true;
  1. mock/_createProductionServer.ts 文件中引入需要的 mock 文件
  1. import { createProdMockServer } from 'vite-plugin-mock/es/createProdMockServer';
  2. const modules = import.meta.globEager('./**/*.ts');
  3. const mockModules: any[] = [];
  4. Object.keys(modules).forEach((key) => {
  5. if (key.includes('/_')) {
  6. return;
  7. }
  8. mockModules.push(...modules[key].default);
  9. });
  10. export function setupProdMockServer() {
  11. createProdMockServer(mockModules);
  12. }
  1. build/vite/plugin/mock.ts 里面引入
  1. import { viteMockServe } from 'vite-plugin-mock';
  2. export function configMockPlugin(isBuild: boolean) {
  3. return viteMockServe({
  4. injectCode: `
  5. import { setupProdMockServer } from '../mock/_createProductionServer';
  6. setupProdMockServer();
  7. `,
  8. });
  9. }

::: tip 为什么通过插件注入代码而不是直接在 main.ts 内插入

在插件内通过 injectCode 插入代码,方便控制 mockjs 是否被打包到最终代码内。如果在 main.ts 内判断,如果关闭了 mock 功能,mockjs 也会打包到构建文件内,这样会增加打包体积。

:::

到这里线上 mock 就配置完成了。线上与本地差异不大,比较大的区别是线上在控制台内看不到接口请求日志。