Files
weRequest/README.md

436 lines
19 KiB
Markdown
Raw Normal View History

2018-12-21 23:28:33 +08:00
<p align="center"><img src="./image/logo.png" alt="weRequest" height="160"/></p>
<h2 align="center">v1.2.8</h2>
2018-12-21 23:28:33 +08:00
<p align="center"><b>解决繁琐的小程序会话管理,一款自带登录态管理的网络请求组件。</b></p>
2017-11-23 19:45:53 +08:00
## 目标
让业务逻辑更专注不用再关注底层登录态问题。小程序对比以往的H5登录态管理逻辑要复杂很多。通过`weRequest`这个组件,希望能帮助开发者把更多精力放在业务逻辑上,而登录态管理问题只需通过一次简单配置,以后就不用再花精力管理了。
2019-03-26 20:14:46 +08:00
## 如何安装
2019-01-24 10:26:51 +08:00
2019-03-26 20:14:46 +08:00
### a) 通过npm安装
2019-01-24 10:26:51 +08:00
```
npm install --save we-request
```
2017-07-31 17:23:46 +08:00
2019-03-26 20:14:46 +08:00
### b) 或直接下载`build/weRequest.min.js`放到小程序包内
2019-01-24 10:26:51 +08:00
## 怎么使用
2019-03-26 20:14:46 +08:00
### 1) 引入`weRequest`组件
#### 1.aES6模式
2017-07-31 17:27:28 +08:00
```javascript
2019-01-24 10:26:51 +08:00
import weRequest from 'we-request';
2019-03-26 20:14:46 +08:00
2019-01-24 10:26:51 +08:00
// 若下载文件到本地,则直接引入对应文件,具体路径自己根据情况修改
// import weRequest from '../lib/weRequest.min'
2019-03-26 20:14:46 +08:00
```
#### 1.b) commonJs 模式
2019-01-24 10:26:51 +08:00
2019-03-26 20:14:46 +08:00
```javascript
2019-01-24 10:26:51 +08:00
const weRequest= require('we-request');
2019-03-26 20:14:46 +08:00
2019-01-24 10:26:51 +08:00
// 若下载文件到本地,则直接引入对应文件,具体路径自己根据情况修改
// const weRequest = require('../lib/weRequest.min');
2019-03-26 20:14:46 +08:00
```
### 2) 初始化组件配置
2017-07-31 17:23:46 +08:00
2019-03-26 20:14:46 +08:00
```javascript
2017-07-31 17:23:46 +08:00
// 初始化配置
weRequest.init({
// 关于配置内容,将在后文详述
// 此处暂时省略...
})
2019-03-26 20:14:46 +08:00
```
### 3) **就像使用`wx.request`那样去使用它**
2017-07-31 17:23:46 +08:00
2019-03-26 20:14:46 +08:00
```javascript
2017-07-31 17:23:46 +08:00
// 发起请求
weRequest.request({
url: 'order/detail',
data: {
id: '107B7615E04AE64CFC10'
},
success: function (data) {
2018-11-23 16:20:25 +08:00
// 省略...
2017-07-31 17:23:46 +08:00
}
})
2019-01-24 10:26:51 +08:00
// 同时也支持Promise形式使用
weRequest.request({
url: 'order/detail',
data: {
id: '107B7615E04AE64CFC10'
}
}).then((data)=>{
// 省略...
})
2017-07-31 17:23:46 +08:00
```
2019-03-26 20:14:46 +08:00
## 使用贴士
正常情况下组件只需要`init`一次即可,因此初始化可以封装在一个文件中:
```javascript
// 举例: 以下代码封装在 util/weRequest.js
import weRequest from 'we-request';
weRequest.init({
// ...
});
export default weRequest;
```
后续业务逻辑直接引用封装好的文件即可使用:
```javascript
import weRequest from 'util/weRequest';
weRequest.request({
// ...
})
```
2017-07-31 17:23:46 +08:00
2018-09-24 22:07:25 +08:00
## 为什么需要它
![登录时序图](http://mp.weixin.qq.com/debug/wxadoc/dev/image/login.png)
上图是小程序官方文档中的**登录时序图**。此图涵盖了前后端,详细讲解了包括登录态的生成,维护,传输等各方面的问题。
具体到业务开发过程中的前端来说,我认为上图还不够完整,于是我画了下面这张以**前端逻辑**为出发点的、包含循环的**流程图**。
我认为前端每一次**发起网络请求**,跟后台进行数据交互,都适用于下图的**流程**
2019-04-01 18:05:43 +08:00
![](https://raw.githubusercontent.com/IvinWu/weRequest/1.2.0/image/flow_login.png)
2018-09-24 22:07:25 +08:00
- **hasChecked** 用一状态标识本生命周期内是否执行过`wx.checkSession`,判断该标识,若否,开始执行`wx.checkSession`,若是,进入下一步
- **wx.checkSession()** 调用接口判断登录态是否过期,若是,重新登录;若否,进入下一步
> wx.checkSession()是小程序提供的检测登录态是否过期的接口,生命周期内只需调用一次即可。用户越久未使用小程序,用户登录态越有可能失效。反之如果用户一直在使用小程序,则用户登录态一直保持有效。具体时效逻辑由微信维护,对开发者透明
- **wx.getStorage(session)** 尝试获取本地的`session`。如果之前曾经登录过,则能获取到;否则,本地无`session`
- **wx.login()** 小程序提供的接口,用于获取`code`code有效期为5分钟
- **wx.request(code)** 将`code`通过后台提供的接口,换取`session`
- **wx.setStorage(session)** 将后台接口返回的`session`存入到localStorage以备后续使用
- **wx.request(session)** 真正发起业务请求,请求中带上`session`
- **parse(data)** 对后台返回的数据进行预解析,若发现登录态失效,则重新执行登录;若成功,则真正获取到业务数据
只要遵循上图的流程,我们就无需在业务逻辑中关注登录态的问题了,相当于把登录态的管理问题**耦合**到了发起网络请求当中,本组件则完成了上述流程的封装,让开发者不用再关心以上逻辑,把精力放回在业务的开发上。
2017-11-23 19:45:53 +08:00
## 演示DEMO
### 自动带上登录态参数
2017-07-31 17:27:28 +08:00
2018-11-23 16:20:25 +08:00
通过`weRequest`发出的请求,将会自动带上登录态参数。
2017-07-31 17:23:46 +08:00
对应的流程为下图中**红色**的指向:
2019-04-01 18:05:43 +08:00
![自动带上登录态参数](https://raw.githubusercontent.com/IvinWu/weRequest/1.2.0/image/flow1.png)
2017-07-31 17:27:28 +08:00
2017-11-23 19:45:53 +08:00
### 没有登录态时,自动登录
2017-07-31 17:27:28 +08:00
2017-07-31 17:23:46 +08:00
当本地没有登录态时,按照流程图,`weRequest`将会自动执行`wx.login()`后的一系列流程,得到`code`并调用后台接口换取`session`储存在localStorage之后重新发起业务请求。
对应的流程为下图中**红色**的指向:
2019-04-01 18:05:43 +08:00
![没有登录态时,自动登录](https://raw.githubusercontent.com/IvinWu/weRequest/1.2.0/image/flow2.png)
2017-07-31 17:27:28 +08:00
2017-11-23 19:45:53 +08:00
### 登录态过期时,自动重新登录
2017-07-31 17:27:28 +08:00
2017-07-31 17:23:46 +08:00
对后台数据进行预解析之后,发现登录态过期,于是重新执行登录流程,获取新的`session`之后,重新发起请求。
对应的流程为下图中**红色**的指向:
2019-04-01 18:05:43 +08:00
![登录态过期时,自动重新登录](https://raw.githubusercontent.com/IvinWu/weRequest/1.2.0/image/flow3.png)
2017-11-23 19:45:53 +08:00
## 文档
### .init(OBJECT)
对组件进行初始化配置,使用组件发起请求前必须进行至少一次的配置
#### OBJECT参数说明
|参数名|类型|必填|默认值|说明|
| :-------- | :-------| :------ | :------ |:------ |
|sessionName|String|否|session|储存在localStorage的session名称且CGI请求的data中会自动带上以此为名称的session值可不配置默认为session|
|urlPerfix|String or Function|否||请求URL的固定前缀如果配置了后续请求的URL都会自动加上这个前缀如果是函数则为函数的返回值|
2017-11-23 19:45:53 +08:00
|loginTrigger|Function|是||触发重新登录的条件参数为CGI返回的数据返回需要重新登录的条件|
|codeToSession|Object|是||用code换取session的CGI配置|
|reLoginLimit|Int|否|3|登录重试次数,当连续请求登录接口返回失败次数超过这个次数,将不再重试登录|
|successTrigger|Function|是||触发请求成功的条件参数为CGI返回的数据返回接口逻辑成功的条件|
|successData|Function|否||成功之后返回数据参数为CGI返回的数据返回逻辑需要使用的数据|
|errorTitle|String/Function|否|操作失败|接口逻辑失败时,错误弹窗的标题|
|errorContent|String/Function|否||接口逻辑失败时,错误弹窗的内容|
|errorCallback|Function|否||当出现接口逻辑错误时,会执行统一的回调函数,这里可以做统一的错误上报等处理|
|doNotCheckSession|Boolean|否|false|是否需要调用checkSession验证小程序的登录态过期若业务不需要使用到session_key则可配置为true|
|reportCGI|Function|否||接口返回成功之后,会执行统一的回调函数,这里可以做统一的耗时上报等处理|
|mockJson|Object|否||可为接口提供mock数据|
|globalData|Object/Function|否||所有请求都会自动带上这里的参数|
2018-09-24 21:58:20 +08:00
|sessionExpireTime|Int|否|null|为用户登陆态设置本地缓存时间单位为ms一旦过期直接废弃缓存中的登陆态|
|sessionExpireKey|String|否|sessionExpireKey|如果为用户登陆态设置了本地缓存时间则过期时间将以此值为key存储在Storage中|
|doNotUseQueryString|Boolean|否|false|默认情况下POST请求登陆态除了带在请求body中也会带在queryString上如果配置了这个为true则登陆态不带在queryString中|
2017-11-23 19:45:53 +08:00
##### codeToSession参数说明
|参数名|类型|必填|默认值|说明|
| :-------- | :-------| :------ | :------ |:------ |
|url|String|是||CGI的url|
|method|String|否|GET|调用改CGI的方法|
|codeName|String|否|code|CGI中传参时存放code的名称|
|data|Object|否||登录接口需要的其他参数|
|success|Function|是||接口返回成功的函数需要返回session的值|
##### reportCGI返回参数说明
|参数名|类型|说明|
| :-------- | :-------| :------ |
|name|String|调用的接口名字可在request接口的report字段配置|
|startTime|Int|发起请求时的时间戳|
|endTime|Int|请求返回时的时间戳|
|request|Function|请求方法,可用于上报|
#### 示例代码
2017-07-31 17:27:28 +08:00
```javascript
2017-07-31 17:23:46 +08:00
weRequest.init({
2018-09-24 22:07:25 +08:00
// [可选] 存在localStorage的session名称且CGI请求的data中会自动带上以此为名称的session值可不配置默认为session
2017-07-31 17:23:46 +08:00
sessionName: "session",
2018-09-24 22:07:25 +08:00
// [可选] 请求URL的固定前缀可不配置默认为空
2017-07-31 17:23:46 +08:00
urlPerfix: "https://www.example.com/",
2018-09-24 22:07:25 +08:00
// [必填] 触发重新登录的条件res为CGI返回的数据
2017-07-31 17:23:46 +08:00
loginTrigger: function (res) {
// 此处例子当返回数据中的字段errcode等于-1会自动触发重新登录
return res.errcode == -1;
},
2018-09-24 22:07:25 +08:00
// [必填] 用code换取session的CGI配置
2017-07-31 17:23:46 +08:00
codeToSession: {
2018-09-24 22:07:25 +08:00
// [必填] CGI的URL
2017-07-31 17:23:46 +08:00
url: 'user/login',
2018-09-24 22:07:25 +08:00
// [可选] 调用改CGI的方法可不配置默认为GET
2017-07-31 17:23:46 +08:00
method: 'GET',
2018-09-24 22:07:25 +08:00
// [可选] CGI中传参时存放code的名称此处例子名称就是code可不配置默认值为code
2017-07-31 17:23:46 +08:00
codeName: 'code',
2018-09-24 22:07:25 +08:00
// [可选] 登录接口需要的其他参数;可不配置,默认为{}
2017-11-23 19:45:53 +08:00
data: {},
2018-09-24 22:07:25 +08:00
// [必填] CGI中返回的session值
2017-07-31 17:23:46 +08:00
success: function (res) {
// 此处例子CGI返回数据中的字段session即为session值
return res.session;
}
},
2018-09-24 22:07:25 +08:00
// [可选] 登录重试次数当连续请求登录接口返回失败次数超过这个次数将不再重试登录可不配置默认为重试3次
2017-07-31 17:23:46 +08:00
reLoginLimit: 2,
2018-09-24 22:07:25 +08:00
// [必填] 触发请求成功的条件
2017-07-31 17:23:46 +08:00
successTrigger: function (res) {
// 此处例子当返回数据中的字段errcode等于0时代表请求成功其他情况都认为业务逻辑失败
return res.errcode == 0;
},
2018-09-24 22:07:25 +08:00
// [可选] 成功之后返回数据;可不配置
2017-07-31 17:23:46 +08:00
successData: function (res) {
// 此处例子返回数据中的字段data为业务接受到的数据
return res.data;
},
2018-09-24 22:07:25 +08:00
// [可选] 当CGI返回错误时弹框提示的标题文字
2017-07-31 17:23:46 +08:00
errorTitle: function(res) {
// 此处例子当返回数据中的字段errcode等于0x10040730时错误弹框的标题是“温馨提示”其他情况下则是“操作失败”
return res.errcode == 0x10040730 ? '温馨提示' : '操作失败'
},
2018-09-24 22:07:25 +08:00
// [可选] 当CGI返回错误时弹框提示的内容文字
2017-07-31 17:23:46 +08:00
errorContent: function(res) {
// 此处例子返回数据中的字段msg为错误弹框的提示内容文字
2018-09-24 22:07:25 +08:00
return res.msg ? res.msg : '服务可能存在异常,请稍后重试'
2017-11-23 19:45:53 +08:00
},
2018-09-24 22:07:25 +08:00
// [可选] 当出现CGI错误时统一的回调函数这里可以做统一的错误上报等处理
2017-11-23 19:45:53 +08:00
errorCallback: function(obj, res) {
// do some report
},
2018-09-24 22:07:25 +08:00
// [可选] 是否需要调用checkSession验证小程序的登录态过期可不配置默认为false
2017-11-23 19:45:53 +08:00
doNotCheckSession: true,
2018-11-23 16:20:25 +08:00
// [可选] 上报耗时的函数name为上报名称startTime为接口调用开始时的时间戳endTime为接口返回时的时间戳
2017-11-23 19:45:53 +08:00
reportCGI: function(name, startTime, endTime, request) {
//wx.reportAnalytics(name, {
// time: endTime - startTime
//});
//request({
// url: 'reportCGI',
// data: {
// name: name,
// cost: endTime - startTime
// },
// fail: function() {
//
// }
//})
console.log(name + ":" + (endTime - startTime));
},
// [可选] 提供接口的mock若不需使用请设置为false。url为调用weRequest.request()时的url。mock数据的格式与正式接口提供的数据格式一致。
mockJson: {
url1: require("../../mock1.json"),
url2: require("../../mock2.json"),
url3: require("../../mock3.json")
}
2018-09-24 22:07:25 +08:00
// [可选] 所有请求都会自动带上globalData里的参数
2018-11-22 20:54:22 +08:00
globalData: function() {
2017-11-23 19:45:53 +08:00
return {
version: getApp().version
}
2018-09-24 22:07:25 +08:00
},
// [可选] session本地缓存时间(单位为ms),可不配置,默认不设置本地缓存时间
sessionExpireTime: 24 * 60 * 60 * 1000,
// [可选] session本地缓存时间存在Storage中的名字可不配置默认为 sessionExpireKey
sessionExpireKey: "sessionExpireKey"
2017-11-23 19:45:53 +08:00
})
```
### .request(OBJECT)
2019-01-24 10:26:51 +08:00
[return Promise]
2017-11-23 19:45:53 +08:00
带上登录态发起一个请求,参数大部分与`wx.request`一致
#### OBJECT参数说明
|参数名|类型|必填|默认值|说明|是否与wx.request不一致|
| :-------- | :-------| :------ | :------ |:------ |:------ |
|url|String|是||开发者服务器接口地址若在init()时有配置urlPerfix则这里会自动拼接前缀|是|
|data|Object/String|否||请求的参数||
|header|Object|否||设置请求的 headerheader 中不能设置 Referer。||
|method|String|否|GET|需大写有效值OPTIONS, GET, HEAD, POST, PUT, DELETE, TRACE, CONNECT||
|dataType|String|否|json|如果设为json会尝试对返回的数据做一次 JSON.parse||
|beforeSend|Function|否||发起请求前执行的函数|是|
|success|Function|否||收到开发者服务成功返回,且执行`successTrigger`成功后的回调函数,参数为`successData`返回的参数|是|
|fail|Function|否||接口调用失败,或执行`successTrigger`失败后的回调函数,若这里有配置,则不再默认弹窗报错|是|
|complete|Function|否||接口调用结束的回调函数(调用成功、失败都会执行)||
2018-12-21 23:28:33 +08:00
|showLoading|Boolean/String|否|false|请求过程页面是否展示全屏的loading当值为字符串时将展示相关文案的loading|是|
2017-11-23 19:45:53 +08:00
|report|String|否||接口请求成功后将自动执行init()中配置的reportCGI函数其中的name字段值为这里配置的值|是|
2018-12-28 17:35:09 +08:00
|cache|Boolean|否||接口是否启用缓存机制若为true将以url为key将结果存储在storage中下次带cache的请求优先返回缓存内容success回调中第二个参数对象的isCache值将标识内容是否为缓存|是|
|noCacheFlash|Boolean|否||当启用缓存时,决定除了返回缓存内容外,是否还返回接口实时内容,以防止页面多次渲染的抖动|是|
2017-11-23 19:45:53 +08:00
#### 示例代码
```javascript
weRequest.request({
url: 'order/detail',
showLoading: true,
report: 'detail',
data: {
id: '123'
},
2018-12-28 17:35:09 +08:00
success: function (data, ext) {
console.log(data, ext.isCache);
2017-11-23 19:45:53 +08:00
},
2018-11-22 20:54:22 +08:00
fail: function(res) {
2017-07-31 17:23:46 +08:00
}
})
```
2017-07-31 17:37:38 +08:00
2017-11-23 19:45:53 +08:00
### .uploadFile(Object)
2019-01-24 10:26:51 +08:00
[return Promise]
2017-11-23 19:45:53 +08:00
带上登录态,将本地资源上传到开发者服务器,客户端发起一个 HTTPS POST 请求,其中 content-type 为 multipart/form-data参数大部分与`wx.uploadFile`一致
#### OBJECT参数说明
|参数名|类型|必填|默认值|说明|是否与wx.uploadFile不一致|
| :-------- | :-------| :------ | :------ |:------ |:------ |
|url|String|是||开发者服务器接口地址若在init()时有配置urlPerfix则这里会自动拼接前缀|是|
|filePath|String|是||要上传文件资源的路径||
|name|String|是||文件对应的 key , 开发者在服务器端通过这个 key 可以获取到文件二进制内容||
|header|Object|否||设置请求的 headerheader 中不能设置 Referer。||
|formData|Object|否||HTTP 请求中其他额外的 form data||
|beforeSend|Function|否||发起请求前执行的函数|是|
|success|Function|否||收到开发者服务成功返回,且执行`successTrigger`成功后的回调函数,参数为`successData`返回的参数|是|
|fail|Function|否||接口调用失败,或执行`successTrigger`失败后的回调函数,若这里有配置,则不再默认弹窗报错|是|
|complete|Function|否||接口调用结束的回调函数(调用成功、失败都会执行)||
2018-12-21 23:28:33 +08:00
|showLoading|Boolean/String|否|false|请求过程页面是否展示全屏的loading当值为字符串时将展示相关文案的loading|是|
2017-11-23 19:45:53 +08:00
|report|String|否||接口请求成功后将自动执行init()中配置的reportCGI函数其中的name字段值为这里配置的值|是|
2018-11-22 20:54:22 +08:00
#### 示例代码
```javascript
wx.chooseImage({
count: 1,
success: function (res) {
weRequest.uploadFile({
url: 'upload/img',
filePath: res.tempFilePaths[0],
name: 'pic',
showLoading: "提交中",
success(data){
console.log(data.imgPath);
}
})
}
})
```
### .getSession()
[return String]
获取本地缓存中用户票据的值
### .getConfig()
[return Object]
2018-09-24 21:58:20 +08:00
获取weRequest的配置。返回的Object内容如下
|参数名|类型|说明|
| :-------- | :-------| :------ |
|urlPerfix|String or Function|在组件初始化时传入的请求URL的固定前缀|
2018-09-24 21:58:20 +08:00
|sessionExpireTime|Int|在组件初始化时传入的用户登陆态设置本地缓存时间|
|sessionExpireKey|String|在组件初始化时传入的用户登陆态本地缓存时间Storage的key|
|sessionExpire|Int|用户登陆态本地缓存过期的时间戳|
2017-11-23 19:45:53 +08:00
### .login()
2018-11-23 16:20:25 +08:00
```[不建议使用]``` 在不发起业务请求的情况下,单独执行登录逻辑
2017-11-23 19:45:53 +08:00
### .setSession(String)
2018-11-23 16:20:25 +08:00
```[不建议使用]``` 设置用户票据的值
2017-11-23 19:45:53 +08:00
### .version
获取 weRequest 版本号
2017-11-23 19:45:53 +08:00
## FAQ
### 我希望在请求时候页面能出现最简单的loading状态该怎么办
2017-07-31 17:37:38 +08:00
只需要在请求的时候,加上参数`showLoading: true`即可,如:
```javascript
weRequest.request({
url: 'order/detail',
showLoading: true,
data: {
id: '123'
},
success: function (data) {
console.log(data);
}
})
```
当然如果你希望使用个性化的loading样式你可以直接使用beforeSend参数来进行自定义展示个性化的loading并且在complete的时候将它隐藏。
2017-11-23 19:45:53 +08:00
### 某些请求在返回错误时,我不希望触发通用的错误提示框,而想用特别的逻辑去处理,该怎么办?
2017-07-31 17:37:38 +08:00
只需要在请求的时候,加上参数`fail: function(){ ... }`即可,如:
```javascript
weRequest.request({
url: 'order/detail',
slience: true,
data: {
id: '123'
},
success: function (data) {
console.log(data);
},
fail: function(res) {
console.log(res);
}
})
```
此时如果接口返回错误码将触发这里定义的fail函数且默认错误弹框将不会出现。
2018-12-21 23:28:33 +08:00
## 贡献者
<table><tbody><tr>
<td><a target="_blank" href="https://github.com/IvinWu"><img width="60px" src="https://avatars0.githubusercontent.com/u/7484381?s=60&v=4"></a></td>
<td><a target="_blank" href="https://github.com/godbasin"><img width="60px" src="https://avatars3.githubusercontent.com/u/11885123?s=60&v=4"></a></td>
</tr></tbody></table>