uni-open-bridge.md 30.6 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

W
wanganxp 已提交
52
## 凭据托管在不同平台的处理
53

d-u-a's avatar
d-u-a 已提交
54 55 56 57 58 59 60
|凭据																		|微信小程序	|微信公众号	|微信PC网页	|微信App		|
|:-:																		|:-:				|:-:				|:-:				|:-:				|
|[access_token](#access_token)					|定时刷新		|定时刷新		|						|开发者操作	|
|[user_access_token](#user_access_token)|						|开发者操作	|						|						|
|[session_key](#session_key)						|开发者操作	|						|						|						|
|[encrypt_key](#encrypt_key)						|开发者操作	|						|						|						|
|[ticket](#ticket)											|						|定时刷新		|						|						|
61 62 63


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

W
wanganxp 已提交
65
**微信凭据分应用级、用户级、一次性等凭据,如果你之前未接触过微信这些凭据,请务必阅读[微信凭据详细介绍](#wxtoken)**
66

W
wanganxp 已提交
67
## 使用
d-u-a's avatar
d-u-a 已提交
68

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

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

首先向微信的[公众平台](https://mp.weixin.qq.com/)申请 `appid``secret` 固定凭据。
74 75

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

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

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

W
wanganxp 已提交
81
**uni-id-config**
82

83
```json
d-u-a's avatar
d-u-a 已提交
84
// uniCloud/cloudfunctions/common/uni-config-center/uni-id/config.json
85 86 87 88 89 90 91 92 93 94 95 96 97
{
  "dcloudAppid": "__UNI__xxxxxx", // 在项目的 manifest.json 
  "mp-weixin": {
    "tokenExpiresIn": 259200,
    "oauth": {
      "weixin": {
        "appid": "", // 微信公众平台申请的小程序 appid
        "appsecret": "" // 微信公众平台申请的小程序 secret
      }
    }
  },
  "web": {
    "oauth": {
98
      "weixin-h5": {
d-u-a's avatar
d-u-a 已提交
99 100
        "appid": "", // 微信公众平台申请的网页授权 appid
        "appsecret": "" // 微信公众平台申请的网页授权 secret
101 102 103 104 105
      }
    }
  }
}
```
d-u-a's avatar
d-u-a 已提交
106

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

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

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

113 114 115
**示例代码**

```json
d-u-a's avatar
d-u-a 已提交
116
// uniCloud/cloudfunctions/common/uni-config-center/uni-open-bridge/config.json
117 118 119 120
{
  "schedule": {
    "__UNI__xxxxxx": { // dcloudAppid, 需要和 `uni-config-center` uni-id中的配置一致
      "enable": true, // 任务全局开关,优先级最高
121
      "weixin-mp": { // 平台,目前仅支持 微信小程序、微信 H5,详情参见 https://uniapp.dcloud.net.cn/uniCloud/uni-open-bridge#platform
122 123 124
        "enable": true, // 当前平台任务开关
        "tasks": ["accessToken"] // 要执行的任务,微信小程序支持 accessToken
      },
125
      "weixin-h5": {
126 127 128 129 130 131 132 133 134 135
        "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 已提交
136

W
wanganxp 已提交
137 138 139 140 141
### 4. 将插件上传到服务空间

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

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

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

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

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

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

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

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

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

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

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

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

W
wanganxp 已提交
163
```js
164
let uobc = require('uni-open-bridge-common')
W
wanganxp 已提交
165

166
// 应用级凭据
W
wanganxp 已提交
167 168
const key = {
  dcloudAppid: '__UNI__xxx', // DCloud Appid
169
  platform: 'weixin-mp' // 平台,解释见下
170
}
W
wanganxp 已提交
171 172 173
uobc.getAccessToken(key)
uobc.getTicket(key)

174

175
// 用户级凭据,需要同时传入 openid 才能获得
176 177
const userKey = {
  dcloudAppid: '__UNI__xxx', // DCloud Appid
178 179
  platform: 'weixin-mp', // 平台,解释见下
  openid: '' // 用户唯一标识,解释见下
180 181 182 183 184
}
uobc.getUserAccessToken(userKey)
uobc.getSessionKey(userKey)
uobc.getEncryptKey(userKey)

W
wanganxp 已提交
185 186 187 188
```

#### Platform@platform

W
wanganxp 已提交
189
存储数据key对应平台的值。注意不同于前端条件编译使用的uniPlatform。
W
wanganxp 已提交
190 191 192

|值					|描述				|
|:-:				|:-:				|
193 194 195 196 197 198
|weixin-mp	|微信小程序	|
|weixin-h5	|微信公众号	|
|weixin-web	|微信pc网页	|
|weixin-app	|微信 App		|
|qq-mp			|QQ 小程序	|
|qq-app			|QQ App			|
W
wanganxp 已提交
199

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

202 203 204 205
#### getAccessToken(key: Object, fallback: Function)

读取 access_token

206 207
#### setAccessToken(key: Object, value: Object, expiresIn: Number)

W
wanganxp 已提交
208
写入 access_token。开发者一般只需使用get类方法,用不到set、remove类方法。下同
209 210 211

#### removeAccessToken(key: Object)

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


W
wanganxp 已提交
215
**key 属性**
216 217 218 219 220 221

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

W
wanganxp 已提交
222
**value 属性**
223

d-u-a's avatar
d-u-a 已提交
224 225 226
|参数					|类型		|描述									|
|:-:					|:-:		|:-:									|
|access_token	|String	|[详情](#access_token)|
227

W
wanganxp 已提交
228
**expiresIn**
229 230 231 232

有效时间(秒)


W
wanganxp 已提交
233
**示例代码**
234 235 236 237 238 239

```js
'use strict';

const {
  getAccessToken,
240 241
  setAccessToken,
  removeAccessToken
242 243 244 245
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
246
    dcloudAppid: '__UNI__xxx',
247
    platform: 'weixin-mp'
248 249 250 251 252 253 254 255 256 257 258 259
  }
  const value = {
    access_token: ''
  }
  const expiresIn = 7200

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

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

260 261 262 263 264 265 266
  // 删除
  await removeAccessToken(key)

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

267 268 269 270
  return null
};
```

271 272
#### getUserAccessToken(key: Object, fallback: Function)

W
wanganxp 已提交
273
读取 `user_access_token`
274

275 276
#### setUserAccessToken(key: Object, value: Object, expiresIn: Number)

W
wanganxp 已提交
277
写入 `user_access_token`
278 279 280

#### removeUserAccessToken(key: Object)

W
wanganxp 已提交
281
删除 `user_access_token`
282 283 284 285 286 287 288


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


**key 属性**

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

**value 属性**

d-u-a's avatar
d-u-a 已提交
297 298 299
|参数					|类型		|描述																									|
|:-:					|:-:		|:-:																									|
|access_token	|String	|微信公众平台用户会话密钥,[详情](#user_access_token)	|
300 301 302 303 304 305 306 307 308 309 310 311

**expiresIn**

有效时间(秒)

**示例代码**

```js
'use strict';

const {
  getUserAccessToken,
d-u-a's avatar
d-u-a 已提交
312 313
  setUserAccessToken,
  removeUserAccessToken
314 315 316 317
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
318
    dcloudAppid: '__UNI__xxx',
319
    platform: 'weixin-h5',
320 321 322 323 324 325 326 327 328 329 330 331 332
    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 已提交
333 334 335 336 337 338 339 340
  // 删除
  await removeUserAccessToken(key)


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

341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356
  return null
};
```


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

读取 session_key

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

写入 session_key

#### removeSessionKey(key: Object)

删除 session_key
357 358


W
wanganxp 已提交
359
**key 属性**
360

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

W
wanganxp 已提交
367
**value 属性**
368

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

W
wanganxp 已提交
373
**expiresIn**
374 375 376 377

有效时间(秒)


W
wanganxp 已提交
378
**示例代码**
379 380 381 382 383

```js
'use strict';

const {
384 385 386
  getSessionKey,
  setSessionKey,
  removeSessionKey
387 388 389 390
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
391
    dcloudAppid: '__UNI__xxx',
392
    platform: 'weixin-mp',
393 394 395 396 397 398 399 400
    openid: ''
  }
  const value = {
    'session_key': ''
  }
  const expiresIn = 7200

  // 写入 (redis / 数据库)
401
  await setSessionKey(key, value, expiresIn)
402 403

  // 读取 (redis / 数据库)
404 405 406 407 408 409 410 411 412
  let result1 = await getSessionKey(key)

  // 删除
  await removeSessionKey(key)


  // 删除后读取, 返回 null
  let result2 = await getSessionKey(key)
  console.log(result2) // null
413 414 415 416 417 418 419 420 421 422

  return null
};
```


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

读取 encrypt_key

423 424 425 426 427 428 429 430 431
#### setEncryptKey(key: Object, value: Object, expiresIn: Number)

写入 encrypt_key

#### removeEncryptKey(key: Object)

删除 encrypt_key


W
wanganxp 已提交
432
**key 属性**
433

434 435 436 437 438 439
|参数				|类型		|必填	|描述																															|
|:-:				|:-:		|:-:	|:-:																															|
|dcloudAppid|String	|是		|DCloud应用appid。[详情](https://ask.dcloud.net.cn/article/35907)	|
|platform		|String	|是		|[详情](#platform)																								|
|openid			|String	|是		|[详情](#openid)																									|
|version		|Number	|是		|版本																															|
440 441


W
wanganxp 已提交
442
**value 属性**
443

d-u-a's avatar
d-u-a 已提交
444 445 446 447
|参数				|类型		|描述														|
|:-:				|:-:		|:-:														|
|encrypt_key|String	|加密 key,[详情](#encrypt_key)	|
|iv					|String	|加密 iv												|
448

W
wanganxp 已提交
449
**expiresIn**
450 451 452 453

有效时间(秒)


W
wanganxp 已提交
454
**示例代码**
455 456 457 458 459 460

```js
'use strict';

const {
  getEncryptKey,
461 462
  setEncryptKey,
  removeEncryptKey
463 464 465 466
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
467
    dcloudAppid: '__UNI__xxx',
468
    platform: 'weixin-mp',
469 470 471 472 473 474 475 476 477 478 479 480 481 482 483
    openid: '',
    version: 1
  }
  const value = {
    encrypt_key: '',
    iv: ''
  }
  const expiresIn = 7200

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

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

484 485 486 487 488 489 490
  // 删除
  await removeEncryptKey(key)

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

491 492 493 494 495 496 497 498 499
  return null
};
```


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

读取 ticket

d-u-a's avatar
d-u-a 已提交
500
#### setTicket(key: Object, value: Object, expiresIn: Number)
501 502 503

写入 ticket

d-u-a's avatar
d-u-a 已提交
504
#### removeTicket(key: Object)
505 506 507 508

删除 ticket


W
wanganxp 已提交
509
**key 属性**
510 511 512 513 514 515

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

W
wanganxp 已提交
516
**value 属性**
517 518 519

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

W
wanganxp 已提交
522
**expiresIn**
523 524 525 526

有效时间(秒)


W
wanganxp 已提交
527
**示例代码**
528 529 530 531 532 533

```js
'use strict';

const {
  getTicket,
534 535
  setTicket,
  removeTicket
536 537 538 539
} = require('uni-open-bridge-common')

exports.main = async (event, context) => {
  const key = {
d-u-a's avatar
d-u-a 已提交
540
    dcloudAppid: '__UNI__xxx',
541
    platform: 'weixin-h5'
542 543 544 545 546 547 548 549 550 551 552 553
  }
  const value = {
    ticket: ''
  }
  const expiresIn = 7200

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

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

554 555 556 557 558 559 560 561
  // 删除
  await removeTicket(key)


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

562 563 564 565 566 567
  return null
};
```



W
wanganxp 已提交
568
#### fallback
569 570 571 572 573 574 575 576 577 578

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

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

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

W
wanganxp 已提交
581
#### 注意事项
582 583

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


d-u-a's avatar
d-u-a 已提交
587
### 云对象URL化方式@cloudurl
W
wanganxp 已提交
588 589 590 591 592 593 594

云对象 `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 已提交
595
配置URL化后,其他系统可以通过下面的http接口,读写删各种开放平台凭据。
W
wanganxp 已提交
596 597 598 599 600 601 602 603 604 605 606 607 608 609

#### getAccessToken

Url

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

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
610
  "platform": "weixin-mp"
W
wanganxp 已提交
611 612 613
}
```

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

616 617
#### setAccessToken

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

620 621 622 623 624 625
Url

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

626 627
参数

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

630 631 632
```json
{
  "dcloudAppid": "__UNI__xxx",
633
  "platform": "weixin-mp",
634 635 636 637 638 639 640
  "value": {
    "access_token": ""
  },
  "expiresIn": 7200
}
```

641

642 643 644 645 646 647 648 649 650 651 652 653 654
#### removeAccessToken

Url

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

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
655
  "platform": "weixin-mp"
656 657 658
}
```

W
wanganxp 已提交
659

660 661 662 663 664 665 666 667 668 669 670 671 672
#### getUserAccessToken

Url

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

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
673
  "platform": "weixin-h5",
674 675 676 677
  "openid": ""
}
```

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

680
#### setUserAccessToken
W
wanganxp 已提交
681 682 683 684

Url

```
685 686 687 688 689
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/setUserAccessToken
```

参数

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

692 693 694
```json
{
  "dcloudAppid": "__UNI__xxx",
695
  "platform": "weixin-h5",
696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716
  "openid": "",
  "value": {
    "access_token": ""
  },
  "expiresIn": 7200
}
```

#### removeUserAccessToken

Url

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

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
717
  "platform": "weixin-h5",
718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734
  "openid": ""
}
```

#### getSessionKey

Url

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

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
735
  "platform": "weixin-mp",
736 737 738 739 740 741 742 743 744 745 746 747 748 749
  "openid": ""
}
```

#### setSessionKey

Url

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

参数

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

752 753 754
```json
{
  "dcloudAppid": "__UNI__xxx",
755
  "platform": "weixin-mp",
756 757 758 759 760 761 762 763 764 765 766 767 768 769
  "openid": "",
  "value": {
    "session_key": ""
  },
  "expiresIn": 7200
}
```

#### removeSessionKey

Url

```
https://xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx.bspapp.com/uni-open-bridge/removeSessionKey
W
wanganxp 已提交
770 771 772 773 774 775 776
```

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
777
  "platform": "weixin-mp",
W
wanganxp 已提交
778 779 780 781
  "openid": ""
}
```

782 783 784 785 786 787 788 789 790 791 792 793 794
#### getEncryptKey

Url

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

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
795
  "platform": "weixin-mp",
796 797 798 799 800
  "openid": "",
  "version": 1 // 此版本号应根据客户端传递的版本号
}
```

801 802 803 804 805 806 807 808 809 810
#### setEncryptKey

Url

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

参数

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

813 814 815
```json
{
  "dcloudAppid": "__UNI__xxx",
816
  "platform": "weixin-mp",
817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838
  "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",
839
  "platform": "weixin-mp",
840 841 842 843 844
  "openid": "",
  "version": 1
}
```

W
wanganxp 已提交
845 846 847 848 849 850 851 852 853 854 855 856 857 858

#### getTicket

Url

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

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
859
  "platform": "weixin-h5"
860 861 862 863 864 865 866 867 868 869 870 871 872
}
```

#### setTicket

Url

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

参数

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

875 876 877
```json
{
  "dcloudAppid": "__UNI__xxx",
878
  "platform": "weixin-h5",
879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897
  "value": {
    "ticket": ""
  }
}
```

#### removeTicket

Url

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

参数

```json
{
  "dcloudAppid": "__UNI__xxx",
898
  "platform": "weixin-h5"
W
wanganxp 已提交
899 900 901 902
}
```


W
wanganxp 已提交
903
## 微信凭据详细介绍@wxtoken
W
wanganxp 已提交
904

905
### access_token(应用级)@access_token
W
wanganxp 已提交
906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924

- 微信小程序 `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 白名单列表。

925
### user_access_token(用户级)@user_access_token
926

W
wanganxp 已提交
927
因微信的众多凭据命名都叫`access_token`,无法有效区分。对于用户级的`access_token`,在 uni-open-bridge 中改名 `user_access_token`
d-u-a's avatar
d-u-a 已提交
928

929 930 931
|平台							|值						|描述																																																													|
|:-:							|:-:					|:-:																																																													|
|微信内置浏览器H5	|access_token	|微信内置浏览器H5用户会话密钥。[详情](https://developers.weixin.qq.com/doc/offiaccount/OA_Web_Apps/Wechat_webpage_authorization.html)	|
d-u-a's avatar
d-u-a 已提交
932

933 934 935 936 937 938 939
对应微信公众平台网页用户授权 `access_token`

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

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

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

942 943 944 945 946 947 948

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

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

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

949
`code` 仅可在服务器使用一次,客户端调用频率限制每个用户每分钟100次
950 951 952 953 954 955 956

### openid(用户级)@openid

微信小程序用户唯一标识

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

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

d-u-a's avatar
d-u-a 已提交
959
### session_key(用户级)@session_key
W
wanganxp 已提交
960 961 962 963 964 965 966 967 968 969 970

平台对应的值

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

会话密钥 `session_key` 有效性

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

971
`uni.login` 调用时,用户的 `session_key` 可能会被更新而致使旧 `session_key` 失效(刷新机制存在最短周期,如果同一个用户短时间内多次调用 `uni.login`,并非每次调用都导致 `session_key` 刷新)。
W
wanganxp 已提交
972 973 974 975 976 977 978 979 980

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

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

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

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

d-u-a's avatar
d-u-a 已提交
981
### encrypt_key(用户级)@encrypt_key
W
wanganxp 已提交
982 983 984 985 986

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

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

d-u-a's avatar
d-u-a 已提交
987
### ticket(用户级)@ticket
W
wanganxp 已提交
988 989 990

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

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


## 不使用 `uni-open-bridge` 托管的情况@nouseuniopenbridge

W
wanganxp 已提交
996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010
如开发者的老业务里已经获取了微信的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 已提交
1011

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

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

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

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