uni-open-bridge.md 32.1 KB
Newer Older
d-u-a's avatar
d-u-a 已提交
1 2
# uni-open-bridge

3
`uni-open-bridge` 是统一接管微信等三方平台认证凭据(包括但不限于`access_token``session_key``encrypt_key``ticket`)的开源库。
d-u-a's avatar
d-u-a 已提交
4

W
wanganxp 已提交
5
## 背景
d-u-a's avatar
d-u-a 已提交
6

W
wanganxp 已提交
7
调用微信等三方开放平台时,涉及众多凭据。有的是固定凭据,没有有效期。有的是临时凭据,会在一定时间或一定操作后失效。
d-u-a's avatar
d-u-a 已提交
8

9
尤其是临时凭据,比如微信的`access_token``session_key``encrypt_key``ticket`, 开发者需要动态从微信服务器获取,统一保存。
d-u-a's avatar
d-u-a 已提交
10

W
wanganxp 已提交
11
但实际上这里面的坑很多:
d-u-a's avatar
d-u-a 已提交
12

W
wanganxp 已提交
13 14 15
1. 微信官方建议公众号开发者使用中控服务器统一获取和刷新 `access_token`,其他业务逻辑服务器所使用的 `access_token` 均来自于该中控服务器,不应该各自去刷新,否则容易造成冲突,导致 `access_token` 覆盖而影响业务;
2. 有的凭据有效期较短,比如`ticket` 的有效期为7200秒,需要定时请求,避免过期。并且由于获取 `ticket` 的 api 调用次数非常有限,频繁刷新 `ticket` 会导致 api 调用受限,影响自身业务,开发者必须在自己的服务全局缓存 `ticket `
3. 在客户端任意地方调用 `wx.login()` 后,会让上一个 `session_key` 立即过期
d-u-a's avatar
d-u-a 已提交
16

W
wanganxp 已提交
17
当多个业务都需要这些临时凭据时,无法让每个业务各自请求微信服务器,会非常混乱和容易冲突。
d-u-a's avatar
d-u-a 已提交
18

W
wanganxp 已提交
19
所以需要在一个中央系统,在定时任务里统一请求微信服务器,保存到数据库。
d-u-a's avatar
d-u-a 已提交
20

W
wanganxp 已提交
21
然后各个业务需要这些凭据时,从这个中央系统的接口中获取,而不是自己向微信服务器请求。
d-u-a's avatar
d-u-a 已提交
22

W
wanganxp 已提交
23
这个中央系统就是`uni-open-bridge`
d-u-a's avatar
d-u-a 已提交
24

W
wanganxp 已提交
25
## 流程介绍
d-u-a's avatar
d-u-a 已提交
26

W
wanganxp 已提交
27
`uni-open-bridge` 包括:
W
wanganxp 已提交
28
1. 一个云对象 `uni-open-bridge`,插件下载地址:[https://ext.dcloud.net.cn/plugin?id=9002](https://ext.dcloud.net.cn/plugin?id=9002)。(其依赖了下面的公共模块,但不是一个插件)
29
2. 一个公共模块 `uni-open-bridge-common` ,插件下载地址:[https://ext.dcloud.net.cn/plugin?id=9177](https://ext.dcloud.net.cn/plugin?id=9177)
W
wanganxp 已提交
30
3. 配套的数据库,保存这些凭据,表名为 `opendb-open-data`。在redis中的key格式为 `uni-id:[dcloudAppid]:[platform]:[openid]:[access-token|user-access-token|session-key|encrypt-key-version|ticket]`
d-u-a's avatar
d-u-a 已提交
31

W
wanganxp 已提交
32
`uni-open-bridge`系统中,有一个同名云对象`uni-open-bridge`,它默认就是定时运行的,在package.json中配置了每小时定时运行一次(部署线上系统生效)。
d-u-a's avatar
d-u-a 已提交
33

34
该云对象根据在 `uni-config-center`[配置](#uni-id-config)固定凭据,从而有权定时向微信服务器发请求,将获取到的 `access_token``ticket` 保存到数据库 `opendb-open-data` 表中。
d-u-a's avatar
d-u-a 已提交
35

36
当所在服务空间开通redis时,还会缓存在redis的key。这会让系统性能更好。
d-u-a's avatar
d-u-a 已提交
37

W
wanganxp 已提交
38
[uni-open-bridge](#uni-open-bridge) 云对象提供了定时任务,外部系统访问能力,读写数据时依赖 [uni-open-bridge-common](#uni-open-bridge-common)。安装 `uni-open-bridge` 时会自动安装依赖插件 [uni-open-bridge-common](#uni-open-bridge-common)
d-u-a's avatar
d-u-a 已提交
39

W
wanganxp 已提交
40
[uni-open-bridge-common](#uni-open-bridge-common) 提供了多层读写Redis或数据库的能力,是为业务云函数/云对象使用这些凭据而设计的。
d-u-a's avatar
d-u-a 已提交
41

W
wanganxp 已提交
42
上述获取到微信的各种临时凭据后,当各个业务代码需要这些凭据时,通过如下方式获取。
d-u-a's avatar
d-u-a 已提交
43

W
wanganxp 已提交
44 45
- 云函数/云对象获取这些临时凭据,可引用公共模块 `uni-open-bridge-common` ,通过该模块的API获取,比如getAccessToken。[见下](#uni-open-bridge-common)
- 非uniCloud系统,比如传统云,获取这些凭据,需要将云对象`uni-open-bridge`进行URL化,通过Http方式请求凭据。[见下](#http)
d-u-a's avatar
d-u-a 已提交
46

d-u-a's avatar
d-u-a 已提交
47

W
wanganxp 已提交
48
流程图如下:
d-u-a's avatar
d-u-a 已提交
49

50
![](https://vkceyugu.cdn.bspapp.com/VKCEYUGU-a90b5f95-90ba-4d30-a6a7-cd4d057327db/b80cec3b-e106-489d-9075-90b5ecb02963.png)
d-u-a's avatar
d-u-a 已提交
51

52 53 54 55 56 57 58 59 60 61 62 63 64
## 凭据使用场景

### 微信小程序

1. 客户端登陆需要保存 [session_key](#session_key)
2. 解密用户敏感数据需要 [access_token](#access_token)[session_key](#session_key),例如获取用户授权的手机号、用户敏感资料
3. 解密安全网络通道使用的加密数据需要 [access_token](#access_token)[session_key](#session_key)[encrypt_key](#encrypt_key)

### 微信公众号

1. 微信应用内公众号页面用户登陆需要用到 [user_access_token](#user_access_token)[ticket](#ticket)

## 凭据托管状态
65

d-u-a's avatar
d-u-a 已提交
66 67
|凭据																		|微信小程序	|微信公众号	|微信PC网页	|微信App		|
|:-:																		|:-:				|:-:				|:-:				|:-:				|
68 69 70 71 72
|[access_token](#access_token)					|定时刷新		|定时刷新		|开发者操作	|开发者操作	|
|[user_access_token](#user_access_token)|						|开发者操作	|-					|-					|
|[session_key](#session_key)						|开发者操作	|-					|-					|-					|
|[encrypt_key](#encrypt_key)						|开发者操作	|-					|-					|-					|
|[ticket](#ticket)											|-					|定时刷新		|-					|-					|
73

74 75 76 77 78 79
`定时刷新` 由云对象 `uni-open-bridge` 的定时任务触发,从微信服务器获取数据,通过调用 `uni-open-bridge-common` 写入到Redis或数据库
`开发者操作` 由开发者引入公共模块 `uni-open-bridge-common`,调用相关[方法](#uni-open-bridge-common)

`session_key` 如果使用了uni-id,uni-id用户登陆时会读写
`encrypt_key` 依赖 `access_token``session_key`,如果依赖的值已存在,可直接读取 `encrypt_key`,如果不存在自动向微信服务器获取、开发者应该仅读取该值,如果有外部系统依赖时可写入
`ticket` 依赖 `access_token`,直接获取 `ticket` 会检查 `access_token`,如果不存在默认先请求微信服务器获取并保存,继续请求 `ticket`
80 81

还有一些不常用的凭据暂不列出,例如:微信App access_token
82

d-u-a's avatar
d-u-a 已提交
83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
### Platform@platform

存储数据key对应平台的值。注意不同于前端条件编译使用的uniPlatform。

|值					|描述				|
|:-:				|:-:				|
|weixin-mp	|微信小程序	|
|weixin-h5	|微信公众号	|
|weixin-web	|微信pc网页	|
|weixin-app	|微信 App		|
|qq-mp			|QQ 小程序	|
|qq-app			|QQ App			|

提示:自动刷新固定应用级凭据目前仅支持 `weixin-mp``weixin-h5`。 后续补充其他平台

98
## 微信凭据详细介绍@wxtoken
99

100
**微信凭据分应用级、用户级、一次性等凭据,如果你之前未接触过微信这些凭据,请务必阅读下面的详细介绍
101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189

### access_token(应用级)@access_token

- 微信小程序 `access_token` 是微信小程序全局唯一后台接口调用凭据,调用绝大多数后台接口时都需使用。[详情](https://developers.weixin.qq.com/miniprogram/dev/framework/server-ability/backend-api.html#access_token)

- 微信H5 `access_token` 是公众号的全局唯一接口调用凭据,公众号调用各接口时都需使用 `access_token`。开发者需要进行妥善保存。`access_token` 的存储至少要保留512个字符空间。`access_token` 的有效期目前为2个小时,需定时刷新,重复获取将导致上次获取的 `access_token` 失效。

公众平台的 API 调用所需的 `access_token` 的使用及生成方式说明:

1、建议公众号开发者使用中控服务器统一获取和刷新 `access_token`,其他业务逻辑服务器所使用的 `access_token` 均来自于该中控服务器,不应该各自去刷新,否则容易造成冲突,导致 `access_token` 覆盖而影响业务;

2、目前`access_token` 的有效期通过返回的expires_in来传达,目前是7200秒之内的值。中控服务器需要根据这个有效时间提前去刷新新 `access_token`。在刷新过程中,中控服务器可对外继续输出的老 `access_token`,此时公众平台后台会保证在5分钟内,新老 `access_token` 都可用,这保证了第三方业务的平滑过渡;

3、`access_token` 的有效时间可能会在未来有调整,所以中控服务器不仅需要内部定时主动刷新,还需要提供被动刷新 `access_token` 的接口,这样便于业务服务器在 API 调用获知 `access_token` 已超时的情况下,可以触发 `access_token` 的刷新流程。

4、对于可能存在风险的调用,在开发者进行获取 `access_token` 调用时进入风险调用确认流程,需要用户管理员确认后才可以成功获取。具体流程为:

开发者通过某 IP 发起调用->平台返回错误码[89503]并同时下发模板消息给公众号管理员->公众号管理员确认该 IP 可以调用->开发者使用该 IP 再次发起调用->调用成功。

如公众号管理员第一次拒绝该 IP 调用,用户在1个小时内将无法使用该 IP 再次发起调用,如公众号管理员多次拒绝该 IP 调用,该 IP 将可能长期无法发起调用。平台建议开发者在发起调用前主动与管理员沟通确认调用需求,或请求管理员开启 IP 白名单功能并将该 IP 加入 IP 白名单列表。

### user_access_token(用户级)@user_access_token

因微信的众多凭据命名都叫`access_token`,无法有效区分。对于用户级的`access_token`,在 uni-open-bridge 中改名 `user_access_token`

|平台							|值						|描述																																																													|
|:-:							|:-:					|:-:																																																													|
|微信内置浏览器H5	|access_token	|微信内置浏览器H5用户会话密钥。[详情](https://developers.weixin.qq.com/doc/offiaccount/OA_Web_Apps/Wechat_webpage_authorization.html)	|

对应微信公众平台网页用户授权 `access_token`

微信公众平台网页授权有两个相同名字 `access_token`,分别用于

1、公众号的全局唯一接口调用凭据,公众号调用各接口时都需使用 `access_token`
2、网页授权接口调用凭证,用户授权的作用域 `access_token`

在微信内置浏览器H5无法区分两个相同名称值不同的 `access_token`,所以在 uni-open-bridge 中对用户级凭据进行改名,以更直观的名称 `user_access_token` 对应用户授权 `access_token`


### code(临时凭据)@code

微信小程序用户登录凭证校验

在客户端通过调用 `uni.login()` 获得临时登录凭证 `code` 后传到开发者服务器在请求微信服务器获得 `session_key``openid``unionid`

`code` 仅可在服务器使用一次,客户端调用频率限制每个用户每分钟100次

### openid(用户级)@openid

微信小程序用户唯一标识

需要在开发者服务器请求微信服务器获得,依赖参数 code,[详情](#code)

可通过 `uni-id-co` 获取,[详情](https://uniapp.dcloud.net.cn/uniCloud/uni-id-summary.html#save-user-token)

### session_key(用户级)@session_key

平台对应的值

|平台				|值					|描述																																																								|
|:-:				|:-:				|:-:																																																								|
|微信小程序	|session_key|微信小程序会话密钥。[详情](https://developers.weixin.qq.com/miniprogram/dev/OpenApiDoc/user-login/code2Session.html)	|

会话密钥 `session_key` 有效性

开发者如果遇到因为 `session_key` 不正确而校验签名失败或解密失败,请关注下面几个与 `session_key` 有关的注意事项。

`uni.login` 调用时,用户的 `session_key` 可能会被更新而致使旧 `session_key` 失效(刷新机制存在最短周期,如果同一个用户短时间内多次调用 `uni.login`,并非每次调用都导致 `session_key` 刷新)。

开发者应该在明确需要重新登录时才调用 `uni.login`,及时通过 `code2Session` 接口更新服务器存储的 `session_key`

微信不会把 `session_key` 的有效期告知开发者。我们会根据用户使用小程序的行为对 `session_key` 进行续期。用户越频繁使用小程序,`session_key` 有效期越长。

开发者在 `session_key` 失效时,可以通过重新执行登录流程获取有效的 `session_key`。使用接口 `uni.checkSession` 可以校验 `session_key` 是否有效,从而避免小程序反复执行登录流程。

当开发者在实现自定义登录态时,可以考虑以 `session_key` 有效期作为自身登录态有效期,也可以实现自定义的时效性策略。

### encrypt_key(用户级)@encrypt_key

为了避免小程序与开发者后台通信时数据被截取和篡改,微信侧维护了一个用户维度的可靠key,用于小程序和后台通信时进行加密和签名。[详情](https://developers.weixin.qq.com/miniprogram/dev/framework/open-ability/user-encryptkey.html)

开发者可以分别通过小程序前端和微信后台提供的接口,获取用户的加密 key。

### ticket(用户级)@ticket

`ticket` 是公众号用于调用微信 JS 接口的临时票据。正常情况下,`ticket` 的有效期为7200秒,通过 `access_token` 来获取。

由于获取 `ticket` 的 api 调用次数非常有限,频繁刷新 `ticket` 会导致 api 调用受限,影响自身业务,开发者必须在自己的服务全局缓存 `ticket `[详情](https://developers.weixin.qq.com/doc/offiaccount/OA_Web_Apps/JS-SDK.html#62)

190

W
wanganxp 已提交
191
## 使用
d-u-a's avatar
d-u-a 已提交
192

W
wanganxp 已提交
193 194 195 196
### 1. **下载插件[uni-open-bridge](https://ext.dcloud.net.cn/plugin?id=9002)到项目中。

### 2. 在`uni-config-center`的 `uni-id` 下配置固定凭据,详情见下面的示例代码

197 198
微信小程序或微信公众号,首先向微信的[公众平台](https://mp.weixin.qq.com/)申请 `appid``secret` 固定凭据。
微信App或PC网页,首先向微信的[开放平台](https://open.weixin.qq.com/)申请 `appid``secret` 固定凭据。
199 200

然后在项目的 uniCloud/cloudfunctions/common/uni-config-center/uni-id/config.json 文件中配置
d-u-a's avatar
d-u-a 已提交
201

d-u-a's avatar
d-u-a 已提交
202 203
如果不需要定时刷新 `access_token``ticket`、也不需要通过外部系统访问凭据时可单独引入 [uni-open-bridge-common](#uni-open-bridge-common),然后在云函数或云对象中直接调用相关方法

W
wanganxp 已提交
204
**示例代码**
d-u-a's avatar
d-u-a 已提交
205

W
wanganxp 已提交
206
**uni-id-config**
207

208
```json
d-u-a's avatar
d-u-a 已提交
209
// uniCloud/cloudfunctions/common/uni-config-center/uni-id/config.json
210 211 212 213 214 215 216 217 218 219 220 221 222
{
  "dcloudAppid": "__UNI__xxxxxx", // 在项目的 manifest.json 
  "mp-weixin": {
    "tokenExpiresIn": 259200,
    "oauth": {
      "weixin": {
        "appid": "", // 微信公众平台申请的小程序 appid
        "appsecret": "" // 微信公众平台申请的小程序 secret
      }
    }
  },
  "web": {
    "oauth": {
223
      "weixin-h5": {
d-u-a's avatar
d-u-a 已提交
224 225
        "appid": "", // 微信公众平台申请的网页授权 appid
        "appsecret": "" // 微信公众平台申请的网页授权 secret
226 227 228 229 230
      }
    }
  }
}
```
d-u-a's avatar
d-u-a 已提交
231

232 233
注意:拷贝此文件内容时需要移除 `注释`

W
wanganxp 已提交
234
### 3. 在`uni-config-center`目录下新建子目录`uni-open-bridge`, 新增 `config.json`,配置 dcloudAppid ,详情见下面的示例代码
235

W
wanganxp 已提交
236
#### uni-open-bridge-config@uniopenbridgeconfig
d-u-a's avatar
d-u-a 已提交
237

238 239 240
**示例代码**

```json
d-u-a's avatar
d-u-a 已提交
241
// uniCloud/cloudfunctions/common/uni-config-center/uni-open-bridge/config.json
242 243 244 245
{
  "schedule": {
    "__UNI__xxxxxx": { // dcloudAppid, 需要和 `uni-config-center` uni-id中的配置一致
      "enable": true, // 任务全局开关,优先级最高
246
      "weixin-mp": { // 平台,目前仅支持 微信小程序、微信 H5,详情参见 https://uniapp.dcloud.net.cn/uniCloud/uni-open-bridge#platform
247 248 249
        "enable": true, // 当前平台任务开关
        "tasks": ["accessToken"] // 要执行的任务,微信小程序支持 accessToken
      },
250
      "weixin-h5": {
251 252 253 254 255 256 257 258 259 260
        "enable": false,
        "tasks": ["ticket"] // 支持微信 H5 ticket,因 ticker 依赖微信 H5 accessToken,内部自动先获取 accessToken。此处的 accessToken 和微信小程序的 accessToken 不是一个值
      }
    }
  },
  "ipWhiteList": ["0.0.0.0"] // 用于 http 调用的服务器IP白名单
}
```

注意:拷贝此文件内容时需要移除 `注释`
d-u-a's avatar
d-u-a 已提交
261

W
wanganxp 已提交
262 263 264 265 266
### 4. 将插件上传到服务空间

云对象上传到服务空间后,会每隔一个小时自动运行一次,从微信服务器获取相关凭据并保存到数据库。

在数据库`opendb-open-data`中会看到数据。如开通redis则在redis的`uni-id`分组中查看(推荐开通redis以获取更好的性能)。
d-u-a's avatar
d-u-a 已提交
267

268
如果异常,请在 [uniCloud Web控制台](https://unicloud.dcloud.net.cn/),找到云函数/云对象 `uni-open-bridge` 检查运行日志。很可能是第一步或第二步的配置出错了。
269

W
wanganxp 已提交
270
## 业务系统获取相关凭据的方法
271

272
`uni-open-bridge`云对象获取到相关凭据后,当业务系统需要使用这些凭据时,通过以下方式获取。
273

W
wanganxp 已提交
274
### 云函数公共模块方式@uni-open-bridge-common
275

W
wanganxp 已提交
276
当你的业务在uniCloud上时,在你的业务云函数/云对象中引用公共模块`uni-open-bridge-common`,然后调用下面的API。
277 278 279

> `云函数公共模块`是不同云函数共享代码的一种方式。如果你不了解什么是`云函数公共模块`,请另读文档[公共模块](https://uniapp.dcloud.io/uniCloud/cf-common)

W
wanganxp 已提交
280
`uni-open-bridge-common` 提供了 [access_token](#access_token)[user_access_token](#user_access_token)[session_key](#session_key)[encrypt_key](#encrypt_key)[ticket](#ticket) 的读取、写入、删除操作。
281 282 283

`uni-open-bridge-common` 支持多层 读取 / 写入 机制,`redis -> database -> fallback`,优先级如下:

284
如果用户没有开通 `redis` 或者操作失败,透传到 `database``database` 失败后,如果用户配置了 `fallback`,继续调用 `fallback` 方法,否则抛出 `Error``database` 对应的表为: `opendb-open-data`
285

W
wanganxp 已提交
286
在常见的情况下,在你的云函数/云对象中调用`uni-open-bridge-common`的几个get方法即可。
287

W
wanganxp 已提交
288
```js
289
let uobc = require('uni-open-bridge-common')
W
wanganxp 已提交
290

291
// 应用级凭据
W
wanganxp 已提交
292 293
const key = {
  dcloudAppid: '__UNI__xxx', // DCloud Appid
294
  platform: 'weixin-mp' // 平台,解释见下
295
}
W
wanganxp 已提交
296 297 298
uobc.getAccessToken(key)
uobc.getTicket(key)

299

300
// 用户级凭据,需要同时传入 openid 才能获得
301 302
const userKey = {
  dcloudAppid: '__UNI__xxx', // DCloud Appid
303 304
  platform: 'weixin-mp', // 平台,解释见下
  openid: '' // 用户唯一标识,解释见下
305 306 307 308 309
}
uobc.getUserAccessToken(userKey)
uobc.getSessionKey(userKey)
uobc.getEncryptKey(userKey)

W
wanganxp 已提交
310 311
```

312 313 314 315
#### getAccessToken(key: Object, fallback: Function)

读取 access_token

316 317
#### setAccessToken(key: Object, value: Object, expiresIn: Number)

W
wanganxp 已提交
318
写入 access_token。开发者一般只需使用get类方法,用不到set、remove类方法。下同
319 320 321

#### removeAccessToken(key: Object)

W
wanganxp 已提交
322
删除 access_token。开发者一般只需使用get类方法,用不到set、remove类方法。下同
323 324


W
wanganxp 已提交
325
**key 属性**
326 327 328 329 330 331

|参数				|类型			|必填	|描述																															|
|:-:				|:-:			|:-:	|:-:																															|
|dcloudAppid|String		|是		|DCloud应用appid。[详情](https://ask.dcloud.net.cn/article/35907)	|
|platform		|String		|是		|[详情](#platform)																								|

W
wanganxp 已提交
332
**value 属性**
333

d-u-a's avatar
d-u-a 已提交
334 335 336
|参数					|类型		|描述									|
|:-:					|:-:		|:-:									|
|access_token	|String	|[详情](#access_token)|
337

W
wanganxp 已提交
338
**expiresIn**
339 340 341 342

有效时间(秒)


W
wanganxp 已提交
343
**示例代码**
344 345 346 347 348 349

```js
'use strict';

const {
  getAccessToken,
350 351
  setAccessToken,
  removeAccessToken
352 353 354 355
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
356
    dcloudAppid: '__UNI__xxx',
357
    platform: 'weixin-mp'
358 359 360 361 362 363 364 365 366 367 368 369
  }
  const value = {
    access_token: ''
  }
  const expiresIn = 7200

  // 写入 (redis / 数据库)
  await setAccessToken(key, value, expiresIn)

  // 读取 (redis / 数据库)
  let result1 = await getAccessToken(key)

370 371 372 373 374 375 376
  // 删除
  await removeAccessToken(key)

  // 删除后读取, 返回 null
  let result2 = await getAccessToken(key)
  console.log(result2) // null

377 378 379 380
  return null
};
```

381 382
#### getUserAccessToken(key: Object, fallback: Function)

W
wanganxp 已提交
383
读取 `user_access_token`
384

385 386
#### setUserAccessToken(key: Object, value: Object, expiresIn: Number)

W
wanganxp 已提交
387
写入 `user_access_token`
388 389 390

#### removeUserAccessToken(key: Object)

W
wanganxp 已提交
391
删除 `user_access_token`
392 393 394 395 396 397 398


对应微信公众平台网页用户授权 `access_token`,详情见下文说明


**key 属性**

399 400 401 402 403
|参数				|类型		|必填	|描述																															|
|:-:				|:-:		|:-:	|:-:																															|
|dcloudAppid|String	|是		|DCloud应用appid。[详情](https://ask.dcloud.net.cn/article/35907)	|
|platform		|String	|是		|[详情](#platform)																								|
|openid			|String	|是		|[详情](#openid)																									|
404 405 406

**value 属性**

d-u-a's avatar
d-u-a 已提交
407 408 409
|参数					|类型		|描述																									|
|:-:					|:-:		|:-:																									|
|access_token	|String	|微信公众平台用户会话密钥,[详情](#user_access_token)	|
410 411 412 413 414 415 416 417 418 419 420 421

**expiresIn**

有效时间(秒)

**示例代码**

```js
'use strict';

const {
  getUserAccessToken,
d-u-a's avatar
d-u-a 已提交
422 423
  setUserAccessToken,
  removeUserAccessToken
424 425 426 427
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
428
    dcloudAppid: '__UNI__xxx',
429
    platform: 'weixin-h5',
430 431 432 433 434 435 436 437 438 439 440 441 442
    openid: ''
  }
  const value = {
    'access_token': ''
  }
  const expiresIn = 7200

  // 写入 (redis / 数据库)
  await setUserAccessToken(key, value, expiresIn)

  // 读取 (redis / 数据库)
  let result1 = await getUserAccessToken(key)

d-u-a's avatar
d-u-a 已提交
443 444 445 446 447 448 449 450
  // 删除
  await removeUserAccessToken(key)


  // 删除后读取, 返回 null
  let result2 = await getUserAccessToken(key)
  console.log(result2) // null

451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466
  return null
};
```


#### getSessionKey(key: Object, fallback: Function)

读取 session_key

#### setSessionKey(key: Object, value: Object, expiresIn: Number)

写入 session_key

#### removeSessionKey(key: Object)

删除 session_key
467 468


W
wanganxp 已提交
469
**key 属性**
470

471 472 473 474 475
|参数				|类型		|必填	|描述																															|
|:-:				|:-:		|:-:	|:-:																															|
|dcloudAppid|String	|是		|DCloud应用appid。[详情](https://ask.dcloud.net.cn/article/35907)	|
|platform		|String	|是		|[详情](#platform)																								|
|openid			|String	|是		|[详情](#openid)																									|
476

W
wanganxp 已提交
477
**value 属性**
478

d-u-a's avatar
d-u-a 已提交
479 480 481
|参数				|类型		|描述																			|
|:-:				|:-:		|:-:																			|
|session_key|String	|微信小程序会话密钥,[详情](#session_key)	|
482

W
wanganxp 已提交
483
**expiresIn**
484 485 486 487

有效时间(秒)


W
wanganxp 已提交
488
**示例代码**
489 490 491 492 493

```js
'use strict';

const {
494 495 496
  getSessionKey,
  setSessionKey,
  removeSessionKey
497 498 499 500
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
501
    dcloudAppid: '__UNI__xxx',
502
    platform: 'weixin-mp',
503 504 505 506 507 508 509 510
    openid: ''
  }
  const value = {
    'session_key': ''
  }
  const expiresIn = 7200

  // 写入 (redis / 数据库)
511
  await setSessionKey(key, value, expiresIn)
512 513

  // 读取 (redis / 数据库)
514 515 516 517 518 519 520 521 522
  let result1 = await getSessionKey(key)

  // 删除
  await removeSessionKey(key)


  // 删除后读取, 返回 null
  let result2 = await getSessionKey(key)
  console.log(result2) // null
523 524 525 526 527 528 529 530 531 532

  return null
};
```


#### getEncryptKey(key: Object, fallback: Function)

读取 encrypt_key

533 534 535 536 537 538 539 540 541
#### setEncryptKey(key: Object, value: Object, expiresIn: Number)

写入 encrypt_key

#### removeEncryptKey(key: Object)

删除 encrypt_key


W
wanganxp 已提交
542
**key 属性**
543

544 545 546 547 548 549
|参数				|类型		|必填	|描述																															|
|:-:				|:-:		|:-:	|:-:																															|
|dcloudAppid|String	|是		|DCloud应用appid。[详情](https://ask.dcloud.net.cn/article/35907)	|
|platform		|String	|是		|[详情](#platform)																								|
|openid			|String	|是		|[详情](#openid)																									|
|version		|Number	|是		|版本																															|
550 551


W
wanganxp 已提交
552
**value 属性**
553

d-u-a's avatar
d-u-a 已提交
554 555 556 557
|参数				|类型		|描述														|
|:-:				|:-:		|:-:														|
|encrypt_key|String	|加密 key,[详情](#encrypt_key)	|
|iv					|String	|加密 iv												|
558

W
wanganxp 已提交
559
**expiresIn**
560 561 562 563

有效时间(秒)


W
wanganxp 已提交
564
**示例代码**
565 566 567 568 569 570

```js
'use strict';

const {
  getEncryptKey,
571 572
  setEncryptKey,
  removeEncryptKey
573 574 575 576
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
577
    dcloudAppid: '__UNI__xxx',
578
    platform: 'weixin-mp',
579 580 581 582 583 584 585 586 587 588 589 590 591 592 593
    openid: '',
    version: 1
  }
  const value = {
    encrypt_key: '',
    iv: ''
  }
  const expiresIn = 7200

  // 写入 (redis / 数据库)
  await setEncryptKey(key, value, expiresIn)

  // 读取 (redis / 数据库)
  let result1 = await getEncryptKey(key)

594 595 596 597 598 599 600
  // 删除
  await removeEncryptKey(key)

  // 删除后读取, 返回 null
  let result2 = await getEncryptKey(key)
  console.log(result2) // null

601 602 603 604 605 606 607 608 609
  return null
};
```


#### getTicket(key: Object, fallback: Function)

读取 ticket

d-u-a's avatar
d-u-a 已提交
610
#### setTicket(key: Object, value: Object, expiresIn: Number)
611 612 613

写入 ticket

d-u-a's avatar
d-u-a 已提交
614
#### removeTicket(key: Object)
615 616 617 618

删除 ticket


W
wanganxp 已提交
619
**key 属性**
620 621 622 623 624 625

|参数				|类型			|必填	|描述																															|
|:-:				|:-:			|:-:	|:-:																															|
|dcloudAppid|String		|是		|DCloud应用appid。[详情](https://ask.dcloud.net.cn/article/35907)	|
|platform		|String		|是		|[详情](#platform)																								|

W
wanganxp 已提交
626
**value 属性**
627 628 629

|参数				|类型		|描述			|
|:-:				|:-:		|:-:			|
d-u-a's avatar
d-u-a 已提交
630
|ticket			|String	|[详情](#ticket)					|
631

W
wanganxp 已提交
632
**expiresIn**
633 634 635 636

有效时间(秒)


W
wanganxp 已提交
637
**示例代码**
638 639 640 641 642 643

```js
'use strict';

const {
  getTicket,
644 645
  setTicket,
  removeTicket
646 647 648 649
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
650
    dcloudAppid: '__UNI__xxx',
651
    platform: 'weixin-h5'
652 653 654 655 656 657 658 659 660 661 662 663
  }
  const value = {
    ticket: ''
  }
  const expiresIn = 7200

  // 写入 (redis / 数据库)
  await setTicket(key, value, expiresIn)

  // 读取 (redis / 数据库)
  let result1 = await getTicket(key)

664 665 666 667 668 669 670 671
  // 删除
  await removeTicket(key)


  // 删除后读取, 返回 null
  let result2 = await getTicket(key)
  console.log(result2) // null

672 673 674 675 676 677
  return null
};
```



W
wanganxp 已提交
678
#### fallback
679 680 681 682 683 684 685 686 687 688

可选 `async function fallback()`,当 `reids -> database` 都找不到对应 `key` 时,调用此方法,需要返回数据格式如下

```json
{
  value: null,
  duration: 1
}
```

689
为了简化调用 `getAccessToken()``getTicket()` 已内置 `fallback` 到微信的服务器,需要在 `config-center` 中配置 `appid` `appsecret`[详情](#uni-id-config)
690

W
wanganxp 已提交
691
#### 注意事项
692 693

- 所有方法类型为 `async`,需要使用 `await`
694
- 所有方法校验 `key` 属性是否有效,无效则 `throw new Error()`,对 `value` 仅校验是否为 `Object`
W
wanganxp 已提交
695 696


d-u-a's avatar
d-u-a 已提交
697
### 云对象URL化方式@cloudurl
W
wanganxp 已提交
698 699 700 701 702 703 704

云对象 `uni-open-bridge` URL化后,让非uniCloud系统可通过 http 方式访问凭据。

[URL化](http.md),是一种让云函数或云对象暴露为Http接口的方式,[详见](http.md)。可以在 [uniCloud Web控制台](https://unicloud.dcloud.net.cn/) 操作。

请求类型 `POST`, 可以配置IP白名单字段 `ipWhiteList`,参见 `config.json`

W
wanganxp 已提交
705
配置URL化后,其他系统可以通过下面的http接口,读写删各种开放平台凭据。
W
wanganxp 已提交
706 707 708 709 710 711 712 713 714 715 716 717 718 719

#### getAccessToken

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/getAccessToken
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
720
  "platform": "weixin-mp"
W
wanganxp 已提交
721 722 723
}
```

W
wanganxp 已提交
724 725
其中参数platform值域[详见](#platform)。下同,不再复述。

726 727
#### setAccessToken

W
wanganxp 已提交
728 729
如果各种开放平台凭据由`uni-open-bridge`托管,那么只需要调用各种get方法,是用不到set等方法的。但在某些情况下,相关凭据没有由`uni-open-bridge`从微信服务器获取,就需要这些set方法了。[详见](#nouseuniopenbridge)

730 731 732 733 734 735
Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/setAccessToken
```

736 737
参数

W
wanganxp 已提交
738
由外部系统从微信获取到相关凭据,然后写入。[详见](#nouseuniopenbridge)
739

740 741 742
```json
{
  "dcloudAppid": "__UNI__xxx",
743
  "platform": "weixin-mp",
744 745 746 747 748 749 750
  "value": {
    "access_token": ""
  },
  "expiresIn": 7200
}
```

751

752 753 754 755 756 757 758 759 760 761 762 763 764
#### removeAccessToken

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/removeAccessToken
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
765
  "platform": "weixin-mp"
766 767 768
}
```

W
wanganxp 已提交
769

770 771 772 773 774 775 776 777 778 779 780 781 782
#### getUserAccessToken

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/getUserAccessToken
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
783
  "platform": "weixin-h5",
784 785 786 787
  "openid": ""
}
```

788 789
其中参数openid值域[详见](#openid)。下同,不再复述。

790
#### setUserAccessToken
W
wanganxp 已提交
791 792 793 794

Url

```
795 796 797 798 799
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/setUserAccessToken
```

参数

W
wanganxp 已提交
800
由外部系统从微信获取到相关凭据,然后写入。[详见](#nouseuniopenbridge)
801

802 803 804
```json
{
  "dcloudAppid": "__UNI__xxx",
805
  "platform": "weixin-h5",
806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826
  "openid": "",
  "value": {
    "access_token": ""
  },
  "expiresIn": 7200
}
```

#### removeUserAccessToken

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/removeUserAccessToken
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
827
  "platform": "weixin-h5",
828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844
  "openid": ""
}
```

#### getSessionKey

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/getSessionKey
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
845
  "platform": "weixin-mp",
846 847 848 849 850 851 852 853 854 855 856 857 858 859
  "openid": ""
}
```

#### setSessionKey

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/setSessionKey
```

参数

W
wanganxp 已提交
860
由外部系统从微信获取到相关凭据,然后写入。[详见](#nouseuniopenbridge)
861

862 863 864
```json
{
  "dcloudAppid": "__UNI__xxx",
865
  "platform": "weixin-mp",
866 867 868 869 870 871 872 873 874 875 876 877 878 879
  "openid": "",
  "value": {
    "session_key": ""
  },
  "expiresIn": 7200
}
```

#### removeSessionKey

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/removeSessionKey
W
wanganxp 已提交
880 881 882 883 884 885 886
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
887
  "platform": "weixin-mp",
W
wanganxp 已提交
888 889 890 891
  "openid": ""
}
```

892 893 894 895 896 897 898 899 900 901 902 903 904
#### getEncryptKey

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/getEncryptKey
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
905
  "platform": "weixin-mp",
906 907 908 909 910
  "openid": "",
  "version": 1 // 此版本号应根据客户端传递的版本号
}
```

911 912 913 914 915 916 917 918 919 920
#### setEncryptKey

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/setEncryptKey
```

参数

W
wanganxp 已提交
921
由外部系统从微信获取到相关凭据,然后写入。[详见](#nouseuniopenbridge)
922

923 924 925
```json
{
  "dcloudAppid": "__UNI__xxx",
926
  "platform": "weixin-mp",
927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948
  "openid": "",
  "version": 1,
  "value": {
    "encrypt_key": "",
    "iv": ""
  }
}
```

#### removeEncryptKey

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/removeEncryptKey
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
949
  "platform": "weixin-mp",
950 951 952 953 954
  "openid": "",
  "version": 1
}
```

W
wanganxp 已提交
955 956 957 958 959 960 961 962 963 964 965 966 967 968

#### getTicket

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/getTicket
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
969
  "platform": "weixin-h5"
970 971 972 973 974 975 976 977 978 979 980 981 982
}
```

#### setTicket

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/setTicket
```

参数

W
wanganxp 已提交
983
由外部系统从微信获取到相关凭据,然后写入。[详见](#nouseuniopenbridge)
984

985 986 987
```json
{
  "dcloudAppid": "__UNI__xxx",
988
  "platform": "weixin-h5",
989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007
  "value": {
    "ticket": ""
  }
}
```

#### removeTicket

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/removeTicket
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
1008
  "platform": "weixin-h5"
W
wanganxp 已提交
1009 1010 1011 1012
}
```


d-u-a's avatar
d-u-a 已提交
1013 1014
## 不使用 `uni-open-bridge` 托管的情况@nouseuniopenbridge

W
wanganxp 已提交
1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029
如开发者的老业务里已经获取了微信的access_token等凭据,难以迁移到由`uni-open-bridge`来托管微信相关凭据。

那么`uni-open-bridge`也暴露了允许三方系统给`uni-open-bridge`写入微信相关凭据的接口。

因为其他插件会依赖`uni-open-bridge`,比如:
1. `uni-ad`微信小程序激励视频广告服务器回调
2. uni云端一体安全网络

如果`uni-open-bridge`里没有相关凭据,上述插件或功能就无法使用。

因此,开发者即不想改成由`uni-open-bridge`托管微信凭据,又需要使用上述依赖`uni-open-bridge`的功能或插件,就只能将老系统获取到的相关凭据写入到`uni-open-bridge`中。

此时,开发者需通过以下方式处理:

1. 取消`uni-open-bridge`云对象的定时任务,不再定时向微信服务器请求凭据
d-u-a's avatar
d-u-a 已提交
1030

W
wanganxp 已提交
1031
`uni-open-bridge`云对象的package.json中找到定时器节点`triggers`,删除该节点。本地修改package.json后需重新上传到服务空间方生效。
d-u-a's avatar
d-u-a 已提交
1032

W
wanganxp 已提交
1033
参考[定时任务配置](cf-functions.md#packagejson))。
d-u-a's avatar
d-u-a 已提交
1034

W
wanganxp 已提交
1035
2. 老系统从微信服务器获取到相关凭据后调用`uni-open-bridge`的set方法写入凭据
d-u-a's avatar
d-u-a 已提交
1036

W
wanganxp 已提交
1037
先将云对象`uni-open-bridge`进行URL化,暴露出http接口。然后老系统调用setAccessToken、setUserAccessToken、setSessionKey、setEncryptKey、setTicket等接口。[参考](#cloudurl)