# simple-mock **Repository Path**: lzw/simple-mock ## Basic Information - **Project Name**: simple-mock - **Description**: No description available - **Primary Language**: TypeScript - **License**: MIT - **Default Branch**: master - **Homepage**: None - **GVP Project**: No ## Statistics - **Stars**: 0 - **Forks**: 0 - **Created**: 2021-04-24 - **Last Updated**: 2021-04-24 ## Categories & Tags **Categories**: Uncategorized **Tags**: None ## README # SIMPLE-MOCK [![NPM version][npm-image]][npm-url] [![npm download][download-image]][download-url] [npm-image]: https://img.shields.io/npm/v/@lzwme/simple-mock.svg?style=flat-square [npm-url]: https://npmjs.org/package/@lzwme/simple-mock [download-image]: https://img.shields.io/npm/dm/@lzwme/simple-mock.svg?style=flat-square [download-url]: https://npmjs.org/package/@lzwme/simple-mock 以注入到 `node server` 的 API 代理方式,实现简洁而功能强大的 API MOCK 功能。附带自动保存真实 API 接口返回数据功能。 ## SIMPLE-MOCK 简介 ### 特色 - `不侵入浏览器端代码`:node server 注入模式实现 MOCK,不影响浏览器端编码 - `MOCK 实现简单`:可自动保存后端 API 返回内容,mock 数据编写无需繁琐流程 - `自定义 MCOK 方式简单强大`:commonjs 方式自定义 MOCK 数据文件编写,无学习门槛,能实现各种自定义逻辑 - `自定义 MOCK 文件修改实时生效`:高效复现不同数据场景 - `配置文件实时生效`:自由切换生产与 MOCK 数据,排查比对问题方便高效 - `支持接口过滤规则`:只针对部分接口进行 MOCK 或 不 MOCK,只需修改一下配置文件 - `支持 TypeScript`:配置文件、自定义 mock 规则文件均支持以 TypeScript 方式编写(文件需以 .ts 结尾) - `支持 websocket 服务 MOCK` - more... ### 何时使用 - `与后端同步开发时`:根据约定自定义接口规则,MOCK 数据进行功能开发 - `需重现不同数据场景时`:根据需求描述、测试反馈等,MOCK 相关数据不同的取值实现不同场景重现。重现测试反馈问题 so easy - `后端服务响应慢、不可用时`:默认开启的自动保存功能,已经将平时请求的接口数据进行了基本的保存。只需要修改配置文件开启 Mock 即立即生效,避免阻塞前端研发 - more... ## 安装与使用 ### 安装 ```bash yarn add -d @lzwme/simple-mock ``` ### 使用 在 nodejs 服务中的 API 代理部分,加入 `simpleMock.render`、 `simpleMock.saveApi` 相关 API 接入逻辑。 针对 `websocket` 的 MOCK 方案可参考 `ws-proxy-server` 目录下的示例。 这里以 `http-proxy` 作为代理示例,具体参见 `server/app.js` 中的源码。示例参考: ```js const app = require('express')(); const bodyParser = require('body-parser'); const httpProxy = require('http-proxy'); const apiProxy = httpProxy.createProxyServer(); const queryString = require('querystring'); app.use(bodyParser.json()); app.use(bodyParser.urlencoded({ extended: false })); /** * 接口代理配置与mock */ const simpleMock = require('@lzwme/simple-mock'); app.all('/{api,rest}/**', async function(req, res) { // 开发模式下且可 mock 的情况 if (appConfig.media === 'dev' && await simpleMock.render(req, res)) { return; } console.log('[apiProxy]', req._parsedUrl.pathname); apiProxy.web(req, res, {target: config[proxyTarget]}); }); // 在代理返回时,注入 saveApi 方法 apiProxy.on('proxyRes', function (proxyRes, req, res) { simpleMock.saveApi(req, res, proxyRes.headers['content-encoding']); }); // 以下为针对 post 请求,代理消费了 stream 的情况 // 针对 post 请求,将 (express.js)bodyParser 消费过的 stream 重新写回到 req apiProxy.on('proxyReq', function(proxyReq, req, res, options) { if (!req.body || !Object.keys(req.body).length) { return; } const contentType = proxyReq.getHeader('Content-Type'); let bodyData; if (contentType === 'application/json') { bodyData = JSON.stringify(req.body); } if (contentType === 'application/x-www-form-urlencoded') { bodyData = queryString.stringify(req.body); } if (bodyData) { proxyReq.setHeader('Content-Length', Buffer.byteLength(bodyData)); proxyReq.write(bodyData); } }); ``` 另外,这里还有两篇相关文章可参考: - [在 Vue-CLI 中引入 simple-mock 快速实现简易的 API Mock 接口数据模拟](https://lzw.me/a/vue-cli-simple-mock.html) - [在 Angular-cli 中引入 simple-mock 实现前端开发 API Mock 接口数据模拟](https://lzw.me/a/angular-cli-simple-mock.html) ## MOCK 规则 基本的 MOCK 的规则为: - 以 `config.customSaveFileName` 参数定义 MOCK 文件查找/保存后端返回数据的命名规则。对于 http 服务,默认为将请求 URL.pathname 路径。最后的文件名会将 `/` 替换为 `_` 并以 js 结尾。例如,请求 URL 为 `/a/b/c?t=123`,则对应的默认文件名为 `a_b_c.js` - 关于 MOCK 文件规则: - 导出内容可为普通对象或数组、字符等,将直接作为 API 返回内容 - 导出内容可为函数,函数传入参数 req 和 req 对象。此时: - 可根据 req.query 处理不同的返回内容; - 可直接处理返回信息。如: `res.status(200).send(content)`,`res.status(403).send('禁止访问')`(模拟出错) - Mock 文件可存在于三个目录中,其优先级为:`customdata > mockdata > customdata/autosave` - `mockdata` 目录用于存放常用的公共 API 规则,会提交至 GIT 仓库 - `customdata` 目录用于自定义 API 规则,不会提交至 GIT 仓库。该目录优先级最高 - `customdata/autosave` 目录用于自动保存后端返回数据,也是保底的 mock 规则 - 不 MOCK 的情况(代理至后端 API): - 符合规则的文件不存在 - 文件导出内容为 `undefined`、`null` 或者 `__ignore_mock__` **mock 文件查找规则:** - 先查找 `${config.mockFileDir}/customdata` 目录 - 再查找 `${config.mockFileDir}/mockdata` 目录 - 最后查找 `${config.mockFileDir}/customdata/autosave` 目录 - 都没有,则走转发到真实 API 代理。如果开启了自动保存,则会将返回结果保存到 `autosave` 目录中 ### 自定义 mock 文件内容示例 当开启自动保存 API 内容时,`autosave` 目录内容下会生成 API 请求返回的结果。当需要对这些 API 进行自定义的规则时,可直接编辑它,不过更推荐将其复制到 `mockdata` 目录中,然后重新编辑器内容规则。下面为一个以函数方式定义的规则参考示例(可参考 `mock/mockdata/rest_user_666.ts` 文件内容): ```js module.exports = (req, res) => { console.log(req.body, req.query); // console.log(req.ip, req.ips, req.originalUrl); // 你甚至可以直接发送各种状态来返回结果(这里 server 以 express 为例) // res.status(500).send({ error: 'something blew up' }); const result = { "code": 200, "url": req.url, "data": { "name": "lzw", "webSite": "https://lzw.me" }, "date": new Date().toLocaleString(), "query": req.query, "body": req.body, }; return result; }; ``` ## 配置与 API ### 开启/关闭 mock 的配置方法 - 配置文件方式 项目根目录 `simple-mock-config.js` 为配置文件,应自行创建,并配置 `.gitignore` 中忽略它,以便于随时修改 mock 行为而不影响其他开发者。配置内容示例可参考 `simple-mock-config-sample.js` 文件。 - 环境变量方式 环境变量主要用于开启或关闭相关功能。其功能开启的优先级高于 `simple-mock-config.js` 中的配置。 - 开启MOCK功能 `process.env.MOCKAPI_ENABLE=mock` - 开启自动保存API返回内容 `process.env.MOCKAPI_AUTOSAVE=save` - 强制每次请求都保存API返回内容(未开启MOCK功能时有效,一般不推荐开启) `process.env.MOCKAPI_AUTOSAVE_FORCE=force` ```bash # 开启MOCK功能 set MOCKAPI_ENABLE=mock # process.env.MOCKAPI_ENABLE=mock # 开启自动保存API返回内容 set MOCKAPI_AUTOSAVE=save # process.env.MOCKAPI_AUTOSAVE=save # 强制每次请求都保存API返回内容(未开启MOCK功能时有效,一般不推荐开启) set MOCKAPI_AUTOSAVE_FORCE=force # process.env.MOCKAPI_AUTOSAVE_FORCE=force ``` ### API - `render(req, res, apiPath?): Promise` 判断一个请求是否可 mock,如果满足条件则执行 mock 逻辑。 应在 nodejs 服务中代理转发前执行。返回一个 Promise,结果为 `true` 则表示可 mock,停止继续执行代理转发;否则为不 mock,应继续走代理转发逻辑。 `req` 和 `res` 会传递到 mocK 规则文件中的自定义规则函数参数中,可通过 `req` 和 `res` 参数自行处理 mock 数据逻辑。 - `saveApi(req, res, contentEncoding)` 在代理请求返回时执行,以判定是否需要保存后端返回的 API 数据。 注意: `contentEncoding` 取值为 `encoded` 时, `res` 应为需要直接保存的 JSON 格式的内容。如: ```js simpleMock.saveApi(req, {user: 'zhansan'}, 'encoded'); ``` - `renderWs(reqData, client, filename = '')` 针对 websocket 类型的 mock。client 参数可自行实现一些方法,其将在 mock 文件规则为函数时传入为第二个参数。 - `saveData: (data, filename = '')` 主要是用于 websocket 服务端返回消息的保存。若 `filename` 为空,则具体的文件规需根据 config.customSaveFileName 确定。 ## FAQ / MOCK规则编写示例与技巧 - 如何保存通过代理返回的信息? 关闭 mock 功能,开启自动保存API功能: ```bash process.env.MOCKAPI_ENABLE=N process.env.MOCKAPI_AUTOSAVE=save ``` - 在开启 mock 模式下,如何忽略某个 API 请求的 mock,从真实后端 API 去请求? 在 `mock/customdata`目录中,编辑该 API 对应的 mock 文件,将返回值改为 `__ignore_mock__`。如果需要根据参数来处理,也是可以实现的,示例: ```js // 忽略mock module.exports = '__ignore_mock__'; // or module.exports = req => { const query = Object.assign({}, req.query, req.body); // id 为 1 则不 mock if (+query.id === 1) { return '__ignore_mock__'; } return {...}; } ``` - `小技巧`:对于同一 API,如何快速保存不同参数返回的不同的值? 简单的数据返回,在 `customdata` 目录下自行写 Javascript 逻辑即可。但对于不同参数返回结果复杂、差异巨大这种情况,自行写逻辑就变得繁琐。 一种方式是通过定义 `config.customSaveFileName` 参数,根据不同的参数将他们落地为不同的文件。 另一种方式是可以关闭 MOCK,开启自动保存和强制保存: ```js module.exports = { mockFileDir: 'mock', isEnableMock: false, isAutoSaveApi: true, isForceSaveApi: true, } ``` 然后每触发一次请求成功后,到 `customdata/autosave` 中找到返回内容并复制出来,如此即可快速得到不同的返回值,再到`customdata` 中编写自定义函数处理规则,根据不同参数定义不同的返回逻辑。如果是比较公共性的逻辑,也可以编写到 `mockdata` 目录中。 - mock 模式下,API 对应 mock 文件不存在时,会转发至后端。但此时会报错? 登陆信息为 mock 返回,session 为无效信息,转发至后端登陆认证失败,API 请求自然也不会成功。 此时可关闭 mock 功能,正常登陆一次,再开启mock;也可临时关闭登陆相关 API 的 mock。 - 忽略自定义目录 customdata 的内容,使用公共目录下的 mockdata? 当 customdata 目录下有符合的规则时,会优先使用,否则则使用 mockdata 下的规则定义。因此,删除 customdata 目录下的定义即可。 但是因为开启 `saveApi` 会自动保存到 customdata,所以还有一种办法,就是在该目录下的文件中导出值为 `undefined`: ```js module.exports = void(0); ``` - `saveApi` 保存的某 API 的内容陈旧过时? 手动修改对应 mock 数据文件内容,或者删除该文件,以重新自动保存远端请求的结果。也可开启 `config.isForceSaveApi` 参数自动更新 - more... ## 开发与测试 ```bash git clone https://github.com/lzwme/simple-mock.git cd simple-mock yarn run dev yarn run serve:ws yarn run serve ``` ## 其他相关 - [Simple Mock(PPT)](https://lzw.me/pages/share/ppt/simple-mock.html) - [在 Vue-CLI 中引入 simple-mock 快速实现简易的 API Mock 接口数据模拟](https://lzw.me/a/vue-cli-simple-mock.html) - [在 Angular-cli 中引入 simple-mock 实现前端开发 API Mock 接口数据模拟](https://lzw.me/a/angular-cli-simple-mock.html)