cf-functions.md 83.2 KB
Newer Older
D
DCloud_LXH 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
## 简介@intro

云函数是运行在云端的 `JavaScript` 代码,是基于 `Node.js` 的扩展。

在常规的 `Node API` 基础上,uniCloud的云函数环境内置了`uniCloud`对象,这个对象内置了网络、数据库等各种API。开发者未学习过 `Node.js` 也没有关系,只需要看uniCloud的文档,掌握这个`uniCloud`对象的API即可。

每个云函数是一个js包,在云函数被调用时,由 serverless 调度系统分配硬件资源启动一个 node 环境来运行这个云函数。

在HBuilderX中可以新建云函数(HBuilderX 3.4 同时可以新建云对象)。
![](https://qiniu-web-assets.dcloud.net.cn/unidoc/zh/createFun-a.jpg)

每个云函数是一个目录,其中普通云函数有`index.js`入口文件,云对象的入口文件则是`index.obj.js`

一个最简单的云函数只需要这个入口js文件,在里面编写代码即可。当然也可以在这个js中require该云函数目录下的其他js、json文件。

16
云函数的配置文件和 npm规范 相同,在云函数目录下可新建一个 package.json 来存放配置。uniCloud云函数扩展了 package.json,增加了一些特有的配置项。[详见](./cf-functions?id=packagejson)
D
DCloud_LXH 已提交
17 18 19 20 21 22 23

云函数启动后实例会保留一段时间(如15分钟),超过保留期后若该云函数一直没有被再调用,那这个实例会被释放。所以云函数有冷启动的概念。不过由于js实例的启动要比php和java更快,所以js更适合serverless方式。

**注意事项**
- 云函数内使用commonjs规范,不可使用import、export,参考:[commonjs模块](http://nodejs.cn/api/modules.html#modules_modules_commonjs_modules)
- 不同项目使用同一个服务空间时,不可使用同名云函数。同名云函数会相互覆盖。
- 在HBuilderX创建云函数时,如果新云函数与服务器上已存在同名云函数,会用新函数覆盖。所以应先选择从服务空间下载云函数。
24 25
- 单个云函数大小限制为10M(包含`node_modules`),过大的云函数影响运行性能,也会增加计费的gbs。同时支付宝云与腾讯云支持在云端安装`node_modules`,此时不占用云函数体积。
- uniCloud的阿里云版与支付宝云版,暂不可使用相对路径读取文件(比如`fs.readFileSync('./info.txt')`),可以使用绝对路径`fs.readFileSync(path.resolve(__dirname,'./info.txt'))`
D
DCloud_LXH 已提交
26 27 28 29 30 31

## 云函数的分类

云函数有若干子概念,包括 普通云函数、云对象、公共模块、clientDB的action云函数、uniCloud扩展库。

- 云函数:通过传统json接口方式和客户端通信,客户端使用`uniCloud.callfunction("")`调用云函数
32
- 云对象:是通过前端导入对象来操作的,客户端使用`uniCloud.importObject("")`导入云对象。详见[云对象](./cloud-obj)
D
DCloud_LXH 已提交
33
- 公共模块:用于不同的云函数/云对象,抽取和共享相同代码,详见[公共模块文档](#common)
34 35
- action云函数(不推荐使用):为了弥补clientDB客户端直接操作数据库的局限而设计的,详见[clientDB action文档](./clientdb?id=action)。从HBuilderX 3.6.11开始,推荐使用[数据库触发器](jql-schema-ext.md)替代action云函数。
- uniCloud扩展库:为了裁剪和控制云函数体积而设计的,一些不太常用的功能比如Redis,独立为可选扩展库,避免增大每个云函数的体积,详见[uniCloud扩展库](./cf-functions?id=扩展库)
D
DCloud_LXH 已提交
36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64

HBuilderX中uniCloud项目的云函数均在项目的`uniCloud/cloudfunctions`目录下,目录结构如下:

<pre v-pre="" data-lang="">
	<code class="lang-" style="padding:0">
|——— cloudfunctions               云函数目录
|   │───common                    云函数公用模块目录 <a target="_blank" href="https://uniapp.dcloud.net.cn/uniCloud/cf-common">详情</a>
|   |   └──hello-common           云函数公用模块
|   |      │──index.js            公用模块代码
|   |      └──package.json        公用模块package.json
|   │───uni-clientDB-actions      (推荐用数据库触发器替代action云函数)
|   │      └──new_action.js       clientDB action代码 <a target="_blank" href="https://uniapp.dcloud.net.cn/uniCloud/clientdb?id=action">详情</a>
|   │───function-name             云函数目录
|   │     │──index.js             云函数代码
|   │     └──package.json         包含云函数的配置信息,如url化、定时设置、可用内存等内容 <a target="_blank" href="https://uniapp.dcloud.net.cn/uniCloud/cf-functions?id=packagejson">详情</a>
|   └───object-name               云对象目录
|         │──index.obj.js         云对象代码
|         └──package.json         包含云对象的配置信息,可用内存等内容 <a target="_blank" href="https://uniapp.dcloud.net.cn/uniCloud/cf-functions?id=packagejson">详情</a>
	</code>
</pre>

## 客户端和云函数的通信@clientcallfunction

uni-app客户端和传统服务器通信时,使用`uni.request`的ajax请求方式。uniCloud下不再使用它,有更好的云端一体的通信方式。

uniCloud体系里,客户端和服务端的云函数通信,有4种方式:

|			|传统的restful方式|callfunction方式|云对象方式|clientDB方式|
|:-:|:-:|:-:|:-:|:-:|
65
|简述		|通过配置[云函数URL化](./http),把云函数转为传统的http链接	|云函数默认并不自带http链接|把callfunction的函数式调用,升级为模块化的对象调用|客户端直接操作云数据库|
D
DCloud_LXH 已提交
66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
|前端调用方式|传统ajax|uni-app客户端通过`uniCloud.callFunction(functionname)`来调用云函数|uni-app客户端通过`uniCloud.importObject(objectname)`导入一个云对象,直接使用这个对象的方法	|uni-app客户端通过`<uniCloud-db>`组件或`uniCloud.database()` API来访问uniCloud数据库。也支持搭配action云函数追加服务器逻辑	|
|适用场景	|http链接需要自己注册域名。如果前端是uni-app,则不推荐使用URL化。如果是非uni-app的系统需要访问云函数,只能使用URL化	|相比云函数URL,callfunction更加安全、更serverless,不暴露域名和ip,不怕攻击,也无需注册域名|uni-app 3.4起支持。相比callfunction方式。代码更加精简、逻辑更加清晰、开发更加高效	|如果uni-app前端发起的服务器请求目的主要是查询或操作数据库,则推荐使用clientDB方式|

云函数是uniCloud的基础,本质上 clientDB 和 云对象 都是建立在云函数上针对特定场景的优化。
- clientDB针对的场景是数据库操作,它优化了可以不写或少写服务器代码
- 云对象针对的场景是非数据库操作或不宜前端暴露的数据库操作时,和uni-app客户端的通信方式。它优化了代码结构,更精简、简单


### clientDB方式

**直观体验代码示例**

clientDB分API方式和组件方式,此处使用API方式来演示
```js
// 客户端js直接操作云数据库,查询list表的数据。无需服务器代码
const db = uniCloud.database() // 获取云数据库的引用
db.collection('list').get()
  .then((res)=>{
    // res 为数据库查询结果
  }).catch((err)=>{
D
DCloud_LXH 已提交
86
    console.log(err);
D
DCloud_LXH 已提交
87 88 89 90 91 92 93 94 95 96 97 98 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
  })
```

由于篇幅较长,学习clientDB需另见文档[clientDB](clientdb.md)

- clientDB适用的情况:

如果客户端使用uni-app开发,且向uniCloud服务空间的请求主要是为了操作云数据库(无论增删改查),那么推荐使用clientDB方式,由uni-app客户端直接操作云数据库。

如果操作数据库的同时,还需要同时执行一些云端逻辑:HBuilderX 3.6.11以前使用action云函数;从HBuilderX 3.6.11开始,推荐使用[数据库触发器](jql-schema-ext.md)替代action云函数。

- clientDB不适用的情况:

1. 请求不操作云数据库,比如向外部web系统发请求、操作redis、删除云文件等;
2. 操作的云数据库请求不希望暴露在前端;
3. 数据库表和字段数量多而接口数量少。给每个数据配置权限的工作量超过了控制少数接口权限的工作量;
4. 权限体系非常复杂,除了用户和管理员外还有较多其他权限条件或动态权限。此时在schema.json/schema.ext.js中编写代码的复杂度超过了写接口。

### 云对象方式

云对象和clientDB最大的区别,是云对象把数据库操作(以及其他逻辑)封装在云对象的方法里面。

它无法像clientDB那样无需开发服务器代码,它仍需在客户端和云端分别写代码。但它的应用场景不受限制。上文中不适用clientDB的情况,都可以使用云对象解决。

**直观体验代码示例**

云端云对象代码,云对象名称:testco,有一个sum方法

```js
module.exports = {
	sum(a, b) {
		// 此处省略a和b的有效性校验
		return a + b
	}
}
```

然后在客户端的js中,import这个testco对象,调用它的sum方法

```js
const testco = uniCloud.importObject('testco') //第一步导入云对象
async function sum () { //注意方法或生命周期需使用async异步方式
	try {
		const res = await testco.sum(1,2) //导入云对象后就可以直接调用该对象的方法了,注意使用异步await
		console.log(res) // 结果是3
	} catch (e) {
		console.log(e)
	}
}
```

由于篇幅较长,学习云对象需另见文档[云对象](cloud-obj.md)

clientDB和云对象可以混合使用:
D
DCloud_LXH 已提交
141
1. 比如官方提供了[uni-id-pages](uni-id/app.md),是基于云对象的登录注册系统,开发者可以导出这个插件处理账户体系,然后剩余的业务如果不算复杂,就可以使用clientDB搞定。
D
DCloud_LXH 已提交
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
2. 一个业务的用户端和admin端也可以是不同的技术栈。比如业务端有复杂的动态权限,而管理端只有一个admin管理员使用,那么admin端使用[schema2code](schema2code.md)会非常高效,而这些技术都基于clientDB。

### 普通云函数callFunction方式

- 普通云函数适用的情况:

在HBuilderX 3.5.2之前,需要URL化和定时运行时,只能使用普通云函数;在HBuilderX 3.5.2+,云对象也支持了URL化和定时运行。

官方不推荐开发者使用云函数,有相关需求推荐使用云对象替代云函数。

目前官方还未提供基于云对象的router模式的框架,有相关需求可以使用三方框架。

**直观体验代码示例**

```js
// 客户端发起调用云函数hellocf,并传入data数据
uniCloud.callFunction({
	name: 'hellocf',
	data: {a:1,b:2}
}).then((res) => {
	console.log(res.result) // 结果是 {sum: 3}
}).catch((err) => {
	console.error(err)
})
```

```js
// 云函数hellocf的代码,接收到客户端传递的data,并对其中a和b相加返回给客户端
'use strict';
exports.main = async (event, context) => {
	//event为客户端上传的参数
	console.log('event : ', event)
	//此处省略event.a和event.b的有效性校验
	//返回数据给客户端
	return {sum : event.a + event.b}
};

```

181
由于篇幅较长,需另见文档[云函数callfunction方式](./cf-callfunction)
D
DCloud_LXH 已提交
182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216

### 云函数URL化方式

可以让云函数/云对象生成一个HTTP URL。这样非uni-app应用,可以通过ajax请求和云函数/云对象通信。

在 uniCloud Web控制台进行URL化配置。

由于篇幅较长,需另见文档[云函数URL化](http.md)


### 云函数请求中的中间状态通知通道

云函数在执行期间可以将中间状态发送给客户端,详情参考:[云函数请求中的中间状态通知通道](sse-channel.md)


### uniCloud响应体规范@resformat

`uniCloud响应体规范`(uniCloud response format),是DCloud制定的、服务器给客户端返回json数据的一种建议格式。

云对象、clientDB、uni-id公共模块均支持此规范。

**由来**

uniCloud服务器给客户端返回的数据格式一般是json,但json的格式具体是什么没有约定。比如返回错误码,是叫`code`还是叫`errCode`?错误内容是`message`还是`errMsg`?内容的国际化如何处理?

如果没有一套统一的格式,在客户端将无法编写有效的网络拦截器,无法统一处理错误。

另外,如果不同的插件,云端返回的数据格式千差万别,那使用者整合这些插件也会非常麻烦。国际化更无法落地。

为此DCloud推出了`uniCloud响应体规范`

为了与uni-app前端的API错误回调风格统一,uniCloud响应体规范定义的云端返回信息(尤其是报错时)应包含`errCode``errMsg`

#### HBuilderX 3.6.10及之后版本的错误规范

雪洛's avatar
雪洛 已提交
217
错误规范继承自[uni错误规范](https://uniapp.dcloud.net.cn/tutorial/err-spec)
D
DCloud_LXH 已提交
218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261

#### HBuilderX 3.6.10之前版本的错误规范

```json
// 失败返回值
{
  "errCode": 'uni-id-account-banned',
  "errMsg": '账号被禁用'
}
```

- errCode

errCode在成功时应返回数字`0`,失败时应返回一个以插件id开头的“字符串”,每个单词以连字符(`-`)分割。做出这样的规定是为了防止不同插件之间出现重复错误码

`'uni-id-account-banned'`错误码为例,`uni-id`为插件id,`account-banned`为错误缩写。

如果业务开发的代码并不发布插件市场,那么为了避免下载了一个市场的插件产生冲突,推荐使用不包含“-”的字符串来做errCode(插件市场的所有插件ID必须包含“-”)。

后续uniCloud会提供自动根据errCode对errMsg进行国际化处理的功能,开发者仅需保证云函数返回值满足`uniCloud响应体规范`即可。

- errMsg

errMsg用于存放具体错误信息,包括展示给开发者、终端用户的错误信息

#### 请求成功的响应

除此之外响应体规范还包含`newToken`字段,用于token的自动续期(云对象接收含有newToken的响应后会自动更新storage内存储的`uni_id_token``uni_id_token_expired`,此行为新增于`HBuilderX 3.4.13`)。开发者仅在自行调用uni-id-common的checkToken等会产生新token的接口时才需要返回新token,uni-app客户端和uni-id-co之间会自动管理token及续期。

`uniCloud响应体`示例如下:

```json
// 成功返回值
{
  "errCode": 0,
  "errMsg": '登录成功',
  "uid": 'xxx', // 其他信息
  "newToken": { // 用于下发新token给客户端
	  "token": 'xxx',
	  "tokenExpired": 'xxx'
  }
}
```

D
DCloud_LXH 已提交
262
HBuilderX内使用代码块`returnu`可以快速输入以下代码(`HBuilderX 3.4.0`及以上版本):
D
DCloud_LXH 已提交
263 264 265 266 267 268 269 270 271 272 273 274 275 276 277

```js
return {
	errSubject: '', // HBuilderX 3.6.10新增
	errCode: 0,
	errMsg: ''
}
```

## uniCloud API列表

云函数支持 js 和 nodejs 的标准API,如`console.log()``setTimeout()`,另见[nodejs官网](https://nodejs.org/en/docs/)。nodejs版本,详见[云函数运行环境](?id=runtime)

除了标准API外,云函数环境中内置了`uniCloud`对象,扩展了一批新API,实际开发中更常用的是uniCloud的扩展API。见下:

WangMoYang's avatar
WangMoYang 已提交
278 279 280 281
|API						| 描述																																			                                                                                             |
|--							|-----------------------------------------------------------------------------------------------------------------------------------|
|uniCloud.database()		| 云数据库对象 [详情](cf-database.md)																									                                                                              |
|uniCloud.databaseJQL()		| 云函数中使用JQL语法操作数据库 [详见](jql-cloud.md),需添加扩展库																			                                                                     |
D
DCloud_LXH 已提交
282
|uniCloud.redis()			| 使用redis [详见](redis.md),需添加扩展库
WangMoYang's avatar
WangMoYang 已提交
283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298
|uniCloud.uploadFile()		| 云函数上传文件到云存储 [详情](storage?id=clouduploadfile)																				                                                                  |
|uniCloud.downloadFile()	| 云函数下载云存储的文件到云函数运行环境 [详情](storage?id=clouddownloadfile)															                                                             |
|uniCloud.deleteFile()		| 云函数删除云存储的文件 [详情](storage?id=clouddeletefile)																				                                                                  |
|uniCloud.getTempFileURL()	| 获取云存储文件的临时路径 [详情](storage?id=cloudgettempfileurl)																		                                                               |
|uniCloud.customAuth()		| 使用云厂商自定义登录,仅腾讯云支持[详情](storage/authentication.md?id=cloud-custom-auth)														                                               |
|uniCloud.callFunction()	| 云函数/云对象中调用另一个云函数 [见下](#callbyfunction)	                                                                                           |
|uniCloud.importObject()	| 云函数/云对象中调用另一个云对象 [详情](cloud-obj.md?id=call-by-cloud)	                                                                             |
|uniCloud.httpclient		| 云函数中通过http访问其他系统 [见下](#httpclient)																		                                                                              |
|uniCloud.httpProxyForEip	| 使用云厂商代理访问http服务(阿里云的解决微信需要固定IP的方案),仅阿里云云端环境支持 [详见](#aliyun-eip),新增于`HBuilderX 3.5.5`                                              |
|uniCloud.sendSms()			| 发送短信,需添加扩展库 [详见](send-sms.md)																											                                                                          |
|uniCloud.getPhoneNumber()	| 获取一键登录手机号,需添加扩展库 [详见](univerify.md?id=cloud)																						                                                                |
|uniCloud.init()			| 获取指定服务空间的uniCloud实例 [详见](concepts/space.md?id=multi-space)														                                                          |
|uniCloud.logger			| 云函数中打印日志到[uniCloud web控制台](https://unicloud.dcloud.net.cn/)的日志系统(非HBuilderX控制台)[详情](rundebug.md?id=uniCloudlogger)															 |
|uniCloud.getRequestList	| 获取当前云函数实例内正在处理的请求Id列表 [详见](#get-request-list),新增于`HBuilderX 3.5.5`                                                                |
|uniCloud.getClientInfos	| 获取当前云函数实例内正在处理的请求对应的客户端信息列表 [详见](#get-client-infos),新增于`HBuilderX 3.5.5`                                                          |
|uniCloud.getCloudInfos		| 获取当前云函数实例内正在处理的请求对应的云端信息列表 [详见](#get-cloud-infos),新增于`HBuilderX 3.5.5`                                                            |
D
DCloud_LXH 已提交
299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316

## 错误对象@uni-cloud-error

云函数调用uniCloud接口时(包括请求云函数、云对象、云存储等)可能存在抛出错误的场景,此时会抛出uniCloud标准的错误对象(以下记为uniCloudError),uniCloudError包含以下属性

|属性		|类型	|必备	|说明												|
|--			|--		|--		|--													|
|errCode	|string	|是		|错误码												|
|errMsg		|string	|是		|错误信息											|
|requestId	|string	|否		|请求Id,用于排查错误								|
|detail		|object	|否		|仅云对象主动返回错误对应的响应体规范时会有此属性	|

另外uniCloudError对象上还有code属性和message属性,两者均不推荐使用。

## 访问数据库

云函数中支持访问本服务空间下的、或经授权的其他服务空间下的,数据库。

317 318
- 使用 JQL 语法操作数据库,另见[文档](jql-cloud.md)
- 使用 MongoDB 语法操作数据库,另见[文档](cf-database.md)
D
DCloud_LXH 已提交
319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654

## 访问其他HTTP服务@httpclient

### uniCloud.httpclient.request@unicloud-httpclient-request

云函数中如需要请求其他http服务,则使用`uniCloud.httpclient`。无需额外依赖,就可以请求任何 HTTP 和 HTTPS 协议的 Web 服务。`uniCloud.httpclient`返回的是一个[urllib实例](https://github.com/node-modules/urllib)

**uniCloud.httpclient.request(URL,requestOptions)**

**requestOptions参数说明**

|参数名							|类型																																																				|是否必填	|默认值	|说明																																																																																			|
|----								|----																																																				|----			|----		|----																																																																																			|
|method							|String																																																			| -				|GET		|HTTP 请求方法, 默认为:GET. 可选值: GET, POST, DELETE, PUT																																																							|
|data								|Object																																																			| -				|-			|发送的数据																																																																																|
|dataAsQueryString	|Boolean																																																		| -				|true		|是否强制转换data为queryString																																																																						|
|content						|String &#124; Buffer																																												| -				|-			|手动设置请求的payload,设置后会忽略data																																																																	|
|stream							|ReadStream																																																	|-				|-			|发送请求正文的可读数据流																																																																									|
|writeStream				|WriteStream																																																|-				|-			|接受响应数据的可写数据流																																																																									|
|consumeWriteStream	|Boolean																																																		|-				|true		|是否等待 writeStream 完全写完才算响应全部接收完毕																																																												|
|files							|Array&lt;ReadStream&#124;Buffer&#124;String&gt; &#124; Object &#124; ReadStream &#124; Buffer &#124; String| -				|-			|上传的文件,设置后将会使用 multipart/form-data 格式。如果未设置method,将会自动将method设置为POST																																				|
|contentType				|String																																																			| -				|-			|上传数据的格式,设为`json`会自动在`header`内设置`Content-Type: application/json`																																													|
|nestedQuerystring	|Boolean																																																		| -				|-			|转换data为queryString时默认不支持嵌套Object,此选项设置为true则支持转换嵌套Object																																												|
|dataType						|String																																																			| -				|-			|返回的数据格式,可选值为 'json'(返回数据转为JSON),'text'(返回数据转为字符串), ''(返回数据不做处理,默认值)																												|
|fixJSONCtlChars		|Boolean																																																		|-				|false	|在JSON.parse之前处理响应结果中的控制字符(Control Character)																																																						|
|headers						|Object																																																			| -				|-			|请求头																																																																																		|
|timeout						|Number &#124; Array																																												| -				|5000			|超时时间设置。设置为数组时第一项为请求超时,第二项为返回超时。设置为数字时相当于同时设置请求超时和返回超时,即`timeout:3000`效果等于`timeouut:[3000,3000]`								|
|auth								|String																																																			|-				|-			|简单登录授权(Basic Authentication)参数,必须按照 `user:password` 格式设置																																															|
|digestAuth					|String																																																			|-				|-			|摘要登录授权(Digest Authentication)参数,必须按照 `user:password` 格式设置																																															|
|agent							|[http.Agent](https://nodejs.org/api/http.html#http_class_http_agent)																				|-				|-			|http代理,如不使用可设为false																																																																						|
|httpsAgent					|[https.Agent](https://nodejs.org/api/https.html#https_class_https_agent)																		|-				|-			|https代理,如不使用可设为false																																																																						|
|ca									|String&#124;Buffer&#124;Array																																							|-				|-			|证书内容																																																																																	|
|rejectUnauthorized	|Boolean																																																		|-				|true		|是否在证书不受信任时返回错误																																																																							|
|pfx								|String&#124;Buffer																																													|-				|-			|包含了私钥, 证书和CA certs, 一般是 PFX 或者 PKCS12 格式																																																									|
|key								|String&#124;Buffer																																													|-				|-			|PEF格式的服务器的私钥																																																																										|
|cert								|String&#124;Buffer																																													|-				|-			|PEM格式的服务器证书密钥																																																																									|
|passphrase					|String																																																			|-				|-			|私钥或pfx密码的字符串																																																																										|
|ciphers						|String																																																			|-				|-			|使用或排除的cipher																																																																												|
|secureProtocol			|String																																																			|-				|-			|SSL 使用的方法,例如,`SSLv3_method` 强制 SSL 版本为3。																																																									|
|followRedirect			|Boolean																																																		|-				|false	|收到3xx响应时是否自动重定向																																																																							|
|maxRedirects				|Number																																																			|-				|10			|最高重定向次数																																																																														|
|formatRedirectUrl	|Function																																																		|-				|-			|手动格式化url																																																																														|
|beforeRequest			|Function																																																		|-				|-			|请求发送前的钩子																																																																													|
|streaming					|Boolean																																																		|-				|false	|是否直接返回响应流,开启 streaming 之后,HttpClient 会在拿到响应对象 res 之后马上返回, 此时 result.headers 和 result.status 已经可以读取到,只是没有读取 data 数据而已。|
|gzip								|Boolean																																																		|-				|false	|是否支持 gzip 响应格式。开启 gzip 之后,HttpClient 将自动设置 Accept-Encoding: gzip 请求头, 并且会自动解压带 Content-Encoding: gzip 响应头的数据。											|
|timing							|Boolean																																																		|-				|false	|是否开启请求各阶段的时间测量																																																																							|
|enableProxy				|Boolean																																																		|-				|false	|是否启用代理																																																																															|
|proxy							|String																																																			|-				|null		| 代理地址																																																																																|
|lookup							|Function																																																		|-				|-			|自定义DNS查询函数																																																																												|
|checkAddress				|Function																																																		|-				|-			|校验请求地址																																																																															|
|trace							|Boolean																																																		|-				|false	|是否启用捕获堆栈																																																																													|

**注意**

默认情况下request接口不会处理返回的数据,即不传`dataType`参数时会返回buffer类型的数据,如需自动解析json格式的返回结果,需要将`dataType`设置为`"json"`

**示例代码**

```js
const res = await uniCloud.httpclient.request(apiUrl, {
    method: 'POST',
    data: {
      test: 'testValue'
    },
    contentType: 'json', // 指定以application/json发送data内的数据
    dataType: 'json' // 指定返回值为json格式,自动进行parse
  })
console.log(res)
```

返回数据结构如下

```js
{
	"data": {"name": "DCloud"}, // 响应内容
	"status": 200, // 状态码
	"headers": { // 响应头,仅作示例,不同服务器返回的有差异
		"date": "Tue, 29 Dec 2020 08:10:30 GMT",
		"content-type": "application/json",
		"content-length": "276",
		"connection": "keep-alive",
		"server": "gunicorn/19.9.0",
		"access-control-allow-origin": "*",
		"access-control-allow-credentials": "true"
	}
}

```


### uniCloud.request@unicloud-request

> 新增于HBuilderX 3.8.10

为简化http请求的调用uniCloud新增了`uni.request`调用方法类似的http请求接口`uniCloud.request`

**参数说明**

|属性					|类型											|必填	|默认值	|说明																																					|
|:-						|:-												|:-		|:-			|:-																																						|
|url					|String										|是		|				|服务器接口地址																																|
|data					|Object/String/ArrayBuffer|否		|				|请求的参数																																		|
|header				|Object										|否		|				|请求头																																				|
|method				|String										|否		|GET		|请求方法,GET、POST、DELETE、PUT																							|
|timeout			|Number										|否		|60000	|超时时间,单位 ms																														|
|dataType			|String										|否		|json		|如果设为 json,会对返回的数据进行一次 JSON.parse,responseType非text时不生效	|
|responseType	|String										|否		|text		|设置响应的数据类型。合法值:text、buffer																			|
|sslVerify		|Boolean									|否		|true		|验证 ssl 证书																																|

**返回值说明**

|属性				|类型											|必备	|说明									|
|:-					|:-												|:-		|:-										|
|statusCode	|number										|是		|开发者服务器接口地址	|
|data				|Object/String/ArrayBuffer|是		|响应结果							|
|header			|Object										|是		|响应头								|

**代码示例**

```js
const res = await uniCloud.request({
  url: 'https://example.com'
})
console.log(res.statusCode)
console.log(res.data)
```

### 发送formdata类型数据

实际业务中常有使用云函数发送formdata类型数据的需求,比如微信小程序提供的一些服务端接口(图片内容安全检测、识别图片二维码等),可以参考以下示例进行发送

```js
'use strict';
const fs = require('fs')
const path = require('path')
const FormData = require('form-data'); // 此form-data需要使用npm安装,地址:https://www.npmjs.com/package/form-data
exports.main = async (event, context) => {
  const form = new FormData()
  form.append('media', fs.readFileSync(path.resolve(__dirname, './test.jpg')), { // 为方便演示此处直接使用云函数目录下的test.jpg文件
    filename: 'test.jpg',
    contentType: 'image/jpeg'
  });
  form.append('otherParam', 'otherParam content');
  const res = await uniCloud.httpclient.request('https://httpbin.org/post', {
    method: 'POST',
    content: form.getBuffer(), // 请求内容
    headers: form.getHeaders(), // 请求头
    dataType: 'json' // 此处指定为json表示将此请求的返回值解析为json
  })
  return res
};

```

## 访问其他websocket服务@websocket-client

云函数无法作为websocket服务器使用,如有相关需求可以尝试使用uni-push 2.0实现,参考:[uni-push 2.0](uni-push/introduction.md)

本章节内容介绍云函数如何作为websocket客户端使用。为简化调用方式uniCloud新增了`uni.connectSocket`方法类似的接口`uniCloud.connectSocket`

**参数说明**

|参数名		|类型								|必填	|说明						|
|:-				|:-									|:-		|:-							|
|url			|String							|是		|服务器接口地址	|
|header		|Object							|否		|请求头					|
|protocols|Array&lt;String&gt;|否		|子协议数组			|

**返回值说明**

调用此接口返回SocketTask对象,见下一章节介绍

### SocketTask@socket-task


#### SocketTask.onMessage(CALLBACK)

监听 WebSocket 接受到服务器的消息事件

**回调函数**

WebSocket 接受到服务器的消息事件的回调函数

**回调函数中的参数**

|属性	|类型								|说明							|
|:-		|:-									|:-								|
|data	|String/Buffer	|服务器返回的消息	|

#### SocketTask.send(OBJECT)

通过 WebSocket 连接发送数据

**参数**

|属性	|类型								|是否必填	|说明						|
|:-		|:-									|:-				|:-							|
|data	|String/Buffer	|是				|需要发送的内容	|

#### SocketTask.close(OBJECT)

关闭 WebSocket 连接

**参数**

|属性		|类型		|默认值										|是否必填	|说明																										|
|:-			|:-			|:-												|:-				|:-																											|
|code		|Number	|1000(表示正常关闭连接)	|否				|一个数字值表示关闭连接的状态号,表示连接被关闭的原因。	|
|reason	|String	|													|否				|一个可读的字符串,表示连接被关闭的原因。								|

#### SocketTask.onOpen(CALLBACK)

监听 WebSocket 连接打开事件

**回调函数**

WebSocket 连接打开事件的回调函数

**回调函数中的参数**

|属性	|类型								|说明							|
|:-		|:-									|:-								|
|data	|String/ArrayBuffer	|服务器返回的消息	|

#### SocketTask.onClose(CALLBACK)

监听 WebSocket 连接关闭事件

**回调函数**

WebSocket 连接关闭事件的回调函数

**回调函数中的参数**

|属性		|类型		|说明																										|
|:-			|:-			|:-																											|
|code		|number	|一个数字值表示关闭连接的状态号,表示连接被关闭的原因。	|
|reason	|string	|一个可读的字符串,表示连接被关闭的原因。								|

#### SocketTask.onError(CALLBACK)

监听 WebSocket 错误事件

**回调函数**

WebSocket 错误事件的回调函数

**回调函数中的参数**

|属性		|类型		|说明			|
|:-			|:-			|:-				|
|errMsg	|String	|错误信息	|


### 示例@socket-example

以下云函数示例代码,从websocket服务器获取消息拼接后返回给客户端,如果遇到错误会抛出错误

```js
'use strict';
exports.main = async (event, context) => {
  const socketTask = uniCloud.connectSocket({
    url: 'wss://xxx.com'
  })
  socketTask.onOpen(async () => {
    await socketTask.send({
      data: 'send some data to server'
    })
    console.log('send data complete')
  })
  const SOCKET_TIMEOUT = 10000

  let error
  let result = ''
  socketTask.onMessage(({
    data
  } = {}) => {
    console.log('message data: ', data);
    result += data
    if (data === '[DONE]') {
      socketTask.close()
    }
  })
  socketTask.onError(function(err) {
    console.log('error', err)
    error = err
  })
  let isClosed = false
  const timeout = setTimeout(() => {
    if (isClosed) {
      return
    }
    error = new Error('socket timeout')
    socketTask.close()
  }, SOCKET_TIMEOUT)
  return new Promise((resolve, reject) => {
    socketTask.onClose(function(...args) {
      isClosed = true
      clearTimeout(timeout)
      console.log('close', ...args)
      if (error) {
        reject(error)
      } else {
        resolve({
          result
        })
      }
    })
  })
};
```


## 请求和环境API

由于存在[单实例多并发](#concurrency)的情况,实例级的uniCloud对象,和每个请求request是一对多的关系。

这也造成了与请求相关的上下文,比如客户端信息,需要通过请求来获取。

为了更好的管理请求和请求相关的上下文,uniCloud提供了下面一批API。

### 获取请求id列表@get-request-list

**示例**

```js
uniCloud.getRequestList()
// 返回值:['3228166e-3c17-4d58-9707-xxxxxxxx']
```

如没有配置[单实例多并发](#concurrency),数组里只会返回一项内容。配置后可能会多项,正在并发的所有请求的requestId都会返回。

当返回多项时,在uniCloud对象上无法明确当前请求是数组中的哪一个。所以提供了其他方法来获取当前请求:
- 云对象通过`this.getUniCloudRequestId()`[详情](cloud-obj.md#get-request-id)
- 云函数通过函数自带参数context。[详情](cf-callfunction.md#context)

A
Anne_LXM 已提交
655
### 获取客户端信息列表@get-client-infos
D
DCloud_LXH 已提交
656

657 658 659 660 661 662 663 664
> HBuilderX 4.21版本客户端上传clientInfo信息时不会将所有信息都传到云端。具体字段白名单如下:`'appId', 'appLanguage', 'appName', 'appVersion', 'appVersionCode', 'appWgtVersion',
    'browserName', 'browserVersion',
    'deviceBrand', 'deviceId', 'deviceModel', 'deviceType',
    'osName', 'osVersion',
    'romName', 'romVersion', 'ua',
    'hostName', 'hostVersion',
    'uniPlatform', 'uniRuntimeVersion', 'uniRuntimeVersionCode', 'uniCompilerVersion', 'uniCompilerVersionCode'`。如需让客户端上传更多clientInfo字段到云端,可以使用客户端api:[uniCloud.setCustomClientInfo](client-sdk.md#set-custom-client-info)

D
DCloud_LXH 已提交
665 666 667
同理,考虑到单实例多并发,`uniCloud.getClientInfos()`获取客户端信息也是一个数组。

```js
D
DCloud_LXH 已提交
668
const clientInfos = uniCloud.getClientInfos()
D
DCloud_LXH 已提交
669 670 671 672 673 674 675 676 677 678 679 680 681 682 683
```

返回值
```js
clientInfos = [{
  appId: '__UNI_xxxxx',
  requestId: '3228166e-3c17-4d58-9707-xxxxxxxx'
  // ...
}]
```

如未开启单实例多并发,那么数组只有1项。单实例多并发场景下返回正在并发的所有请求的客户端信息列表。

**返回值**

雪洛's avatar
雪洛 已提交
684
getClientInfos返回的信息,是在客户端的[uni.getSystemInfo](https://uniapp.dcloud.net.cn/api/system/info#getsysteminfo)的基础之上,增加了一些额外的信息。
D
DCloud_LXH 已提交
685 686 687 688 689 690 691 692 693

除了`getSystemInfo`返回字段外,还包含以下信息

|属性名		|类型		|说明																																																																					|
|--				|--			|--																																																																						|
|requestId|string	|请求Id,可以使用此字段筛选出当前请求的客户端信息																																															|
|clientIP	|string	|客户端ip																																																																			|
|userAgent|string	|客户端ua,注意非本地运行环境下客户端getSystemInfoSync也会获取ua参数并上传给云对象,但是云对象会从http请求头里面获取ua而不是clientInfo里面的ua|
|source		|string	|调用来源,返回值见下。																																																												|
雪洛's avatar
雪洛 已提交
694
|scene		|string	|场景值。客户端[uni.getLaunchOptionsSync](https://uniapp.dcloud.net.cn/api/getLaunchOptionsSync#getlaunchoptionssync)返回的scene参数,													|
D
DCloud_LXH 已提交
695 696 697 698 699 700 701 702 703 704 705 706 707 708 709

云函数调用来源source,它的值域为:

| 取值			     | 说明													    |
|-----------|--------------------|
| client		  | uni-app客户端导入云对象调用	 |
| function	 | 由其他云函数或云对象调用			    |
| http			   | 云对象URL化后通过http访问调用 |
| timing		  | 定时任务调用云对象						    |
| server		  | 云函数右键"上传并运行"						   |

**注意事项**

- 客户端上报的信息在理论上存在被篡改可能,实际业务中应验证前端传来的数据的合法性
- 除了clientIP外,其他客户端信息只有使用uni-app客户端以callFunction或者importObject方式访问云函数或云对象时才有
D
DCloud_LXH 已提交
710
- 云对象与云函数内获取客户端platform稍有不同,云函数未拉齐vue2、vue3版本app平台的platform值,vue2为`app-plus`,vue3为`app`。云对象无论客户端是vue2还是vue3,在app平台获取的platform均为`app`。这一点在使用uni-id时需要特别注意,详情见:[uni-id文档 preferedAppPlatform](uni-id/old.md?id=prefered-app-platform)
D
DCloud_LXH 已提交
711 712 713 714 715 716 717 718 719 720 721 722

除了`uniCloud.getClientInfos()`API,在云函数context和云对象this中,也可以直接获取当前客户端信息。
- 云对象通过`this.getClientInfo()`。[详情](cloud-obj.md#get-client-info)
- 云函数通过函数自带参数context。[详情](cf-callfunction.md#context)

### 获取云端信息@get-cloud-infos

同上,为了兼容并发场景,获取云端信息`uniCloud.getCloudInfos()`返回的也是数组。

**示例**

```js
D
DCloud_LXH 已提交
723
const cloudInfos = uniCloud.getCloudInfos()
D
DCloud_LXH 已提交
724 725 726 727 728 729 730 731 732 733 734 735 736
cloudInfos = [{
  provider: 'aliyun',
  spaceId: 'xxxxx',
  functionName: 'xxx',
  functionType: 'xxxx',
  requestId: '3228166e-3c17-4d58-9707-xxxxxxxx'
}]
```

**返回值**

| 参数名			        | 类型	     | 必备	 | 说明													                                       |
|---------------|---------|-----|-------------------------------------------------------|
737
| provider		    | string	 | 是		 | 服务空间供应商,支付宝云:`alipay`,阿里云为:`aliyun`,腾讯云为:`tencent` |
D
DCloud_LXH 已提交
738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767
| spaceId		     | string	 | 是		 | 服务空间Id												                                    |
| functionName	 | string	 | 是		 | 云函数名称												                                     |
| functionType	 | string	 | 是		 | 云对象为`cloudobject`、云函数为`cloudfunction`			              |
| requestId		   | string	 | 是		 | 请求Id,可以使用此字段筛选出当前请求的云端信息			                           |

除了`uniCloud.getCloudInfos()`API,在云函数context和云对象this中,也可以直接获取当前请求的云端信息。
- 云对象通过`this.getCloudInfo()`。[详情](cloud-obj.md#get-cloud-info)
- 云函数通过函数自带参数context。[详情](cf-callfunction.md#context)


上述3个API,都因为单实例多并发而被设计成数组方式。实际上,大多数开发者并不使用单实例多并发。

在不考虑单实例多并发时,也可以直接使用uniCloud的getRequestList、getClientInfos、getCloudInfos方法中第一个数组项。

或者在云对象的this和云函数的context里获取相关上下文信息也可以。

## 扩展库@extension

uniCloud的api中,有些api对应的实现,其代码体积较大,且这些功能并不是每一个云函数都会使用。为了方便开发者控制云函数的体积,设计了`uniCloud扩展库`的概念。

开发者可以对云函数目录点右键,管理公共模块和扩展库依赖,在其中选择要加载的扩展库。这个可视化界面对应的数据在云函数目录下的 package.json 内的`extensions`字段下。

注意:未引用扩展库的,使用uniCloud相应api时会报错。

![](https://qiniu-web-assets.dcloud.net.cn/unidoc/zh/uniPush-glkzk.jpg)

![](https://qiniu-web-assets.dcloud.net.cn/unidoc/zh/uniPush-kzk.jpg)

**目前支持的扩展库如下**

768 769 770 771 772
- JQL扩展库[uni-cloud-jql]:用于在云函数内使用JQL语法操作数据库,详见:[JQL扩展库](jql-cloud.md)
- redis扩展库[uni-cloud-redis]:云函数内使用redis,详见:[redis扩展库](redis.md)
- 发送短信扩展[uni-cloud-sms]:云函数中发送短信,详见:[sms扩展](send-sms?id=extension)
- 一键登录与实人认证扩展[uni-cloud-verify]:手机App调用运营商一键登录服务时,云函数中获取到真实手机号, 详见:[一键登录扩展库](univerify?id=extension)。核验终端操作者的真实身份,详见:[uni实人认证](frv/intro.md)
- 统一推送服务扩展库[uni-cloud-push]:云函数内使用uni-push,详见:[统一推送服务扩展库](uni-cloud-push/api.md)
D
DCloud_LXH 已提交
773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789

以下是一个开启了redis扩展库的云函数package.json示例,注意此文件不支持注释,下方示例中的注释仅为演示

```js
{
  "name": "add-article",
  "version": "1.0.0",
  "description": "新增文章",
  "main": "index.js",
	"extensions": {
		"uni-cloud-redis": {} // 配置为空对象即可,后续如有扩展参数会在此处配置
	}
}
```

## 公共模块@common

790
云函数支持公共模块。多个云函数/云对象的共享部分,可以抽离为公共模块,然后被多个云函数引用。由于篇幅较长,[详见](cf-common)
D
DCloud_LXH 已提交
791 792 793 794 795 796 797

## 使用npm

云函数的运行环境是 `Node.js`,因此我们可以使用 `npm` 安装第三方依赖。

注意:阿里云目前仅支持全量上传云函数(整个`node_modules`文件夹全部上传,会在上传前自动在本地安装依赖,不会直接使用云函数目录下的node_modules),因此提醒开发者精简依赖,否则可能会每次上传时间很慢,影响开发体验。并且太大的npm库影响云函数的运行性能。

798
腾讯云、支付宝云会在上传云函数后自动安装需要的npm依赖。
D
DCloud_LXH 已提交
799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875

Tips:
- 目前每个云函数上传包大小限制为10M。如果npm包很大,阿里云的整体上传机制会无法满足需求。此时只能选择腾讯云,交给腾讯云自动安装依赖。


## 云函数/云对象中调用云函数@callbyfunction

### 调用三方云函数或云对象

用法同客户端调用云函数,仍然是`uniCloud.callfunction`,但不支持callback形式。

**请求参数**

|字段			|类型			|必填	|说明					|
|---			|---			|---	|---					|
|name			|String		|是		|云函数名称。	|
|data			|Object		|否		|云函数参数。	|

**响应参数**

|字段			|类型		|必备	|说明												|
|---			|---		|---	|---												|
|errCode	|String	|否		|状态码,操作成功则不返回。	|
|errMsg		|String	|否		|错误描述。									|
|result		|Object	|否		|云函数执行结果。						|
|requestId|String	|否		|请求序列号,用于错误排查。	|

**示例代码**

```javascript
let callFunctionResult = await uniCloud.callFunction({
    name: "test",
    data: { a: 1 }
})
```

**注意**

由于调用方不是uni-app客户端,云函数的context、云对象的this.getClientInfo等API无法获取客户端信息,包括 uni-id-token。

可以在云函数互调时手动传递 token ,或者校验调用来源(source)为云函数(function)时不验证用户 token。

云函数/云对象互相调用时调用方会通过公网访问被调用方,访问速度不如直接将逻辑放在调用方执行。使用前请确保你确实需要此功能。

`HBuilderX 3.4.0`版本之前**云函数右键本地运行时**,里面的代码再次callFunction会调用云端的云函数而不是本地云函数,此bug后续版本已修复。

### 云函数递归调用自身@recurrence

除了调用三方云函数,事实上云函数还可以递归调用自己。

当一个云函数实例的资源不能满足需求,或超时时间不够用时。比如要给10万个用户发送短信,而短信发送接口一次调用最多支持50个手机号码,这样最少需要调用2000次接口才能完成;而一个云函数实例完成不了2000次接口的调用。这种场景就可以使用云函数递归调用,分解任务实现。

示例代码如下:

```js
// 当前云函数名称 send-sms-cf
'use strict';
const db = uniCloud.database();
const dbCmd = db.command
const userTable = db.collection('uni-id-users')
exports.main = async (event, context) => {
	//执行业务逻辑
	let res = await sendSms(event.before_id)
	if (res.errCode) {
		return res
	}else{
		// 如果没有报错,就让当前云函数 调用当前云函数(云对象同理)。注意:这里是异步的
		uniCloud.callFunction({
			name: 'send-sms-cf',
			data: {
				before_id: res.before_id
			}
		}).catch(e=>{
			console.log(e.message);
		}).then(e=>{
			console.log(e.result);
		})
D
DCloud_LXH 已提交
876

D
DCloud_LXH 已提交
877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894
		// 等待500毫秒,给一个请求发出去的时间
		return await new Promise((resolve, reject) => {
			setTimeout(() => {
				resolve(res)
			}, 500)
		})
	}

	async function sendSms(before_id) {
		console.log('before_id',before_id);
		let where = {
			phone: dbCmd.exists(true),
			//..这里可以写你自己的其他条件,如超过多久没登录的用户 last_login_date < Date.now() - 3600*24*...
		}
		if(before_id){
			//高性能分页查询,以上一次查询的最后一条数据的id被起始id
			where._id = dbCmd.gt(before_id)
		}
D
DCloud_LXH 已提交
895

D
DCloud_LXH 已提交
896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932
		let res = await userTable.where(where)
			.limit(50)
			.orderBy("_id", "asc")
			.get()

		if (!res.data.length) {
			return {
				errCode: 'sendSms-invalid',
				errMsg: '结束,没有符合条件的接收者'
			}
		}
		let phoneList = res.data.map(item => item.phone)
		let sendSmsRes = await uniCloud.sendSms({
			phoneList,
			appid: '__UNI__xxxxxxx',
			templateId: '100**', // 请替换为自己申请的模板id
			data: {
				text1: 'xxx',
				text2: 'xxx'
			}
		})
		if (sendSmsRes.errCode) {
			return sendSmsRes
		}
		return {
			errCode: 0,
			before_id: res.data[res.data.length - 1]._id
		}
	}
};
```

注意:如果不小心把递归云函数写成死循环,就把云函数的内容全部删除,重新上传覆盖即可


### 云函数内访问其他服务空间@call-by-function-cross-space

933
> 仅支付宝云与腾讯云支持
D
DCloud_LXH 已提交
934

935
在支付宝云与腾讯云服务空间的云函数内支持获取**同账号**下其他服务空间的uniCloud实例,参考:[一个应用访问多个服务空间](concepts/space.md?id=multi-space),并使用此实例调用对应服务空间的云函数。
D
DCloud_LXH 已提交
936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966

```javascript
//开发者创建了多个服务空间,则需手动初始化。注意这是前端代码,不是云函数代码
const myCloud = uniCloud.init({
  provider: 'tencent',
  spaceId: 'xxxx-yyy'
});
//通过uniCloud实例调用云开发的API
myCloud.callFunction()
myCloud.uploadFile()
```

**注意**

- 连接本地云函数调试时,如果存在跨服务空间调用,则callFunction会使用云端云函数

## serverless环境说明@runtime

serverless是动态分配计算资源的,由此会引发的出一批特有概念:冷启动、实例、并发请求、无状态、伪全局变量。

### 云函数冷启动、热启动@launchtype

基于云函数按需执行的特点, 函数在不被触发的时候, 计算资源是不被激活的。

当一个云函数初次被触发时,其完整过程如下:

1. severless实例化计算实例
2. 加载函数代码
3. 启动 node
4. 执行云函数代码

D
DCloud_LXH 已提交
967
函数被调用时,执行这些完整步骤的过程称作`冷启动`, 冷启动的耗时一般在一秒左右。
D
DCloud_LXH 已提交
968 969 970 971 972 973 974 975 976

一个云函数实例冷启动后,serverless调度中心会保留这个实例一定时间。在实例保留期间,客户端再次请求云函数,不会触发冷启动,速度会更快。实例的详细定义[见下](#instance)

云函数实例和执行进程都被复用的情况下称之为`热启动`, 热启动性能要好非常多,因为它只有一个步骤:
1. 执行云函数代码

如果一个云函数实例长时间没有被再次调用,则该计算实例会被**回收**;后续再次调用该云函数时,就会再次触发云函数的**冷启动**。

不同云厂商的函数实例回收时间不同:
977
- 支付宝云:60秒
D
DCloud_LXH 已提交
978 979 980 981 982 983 984 985 986 987 988
- 阿里云:15分钟内没有第二次访问的云函数,就会被回收
- 腾讯云:30分钟

直观的体验表现为:隔了很久不用的云函数,第一次用就会比较慢,然后立即访问第二次,则很快,毫秒级响应。

注:冷启动虽慢但也不会超过1.5秒,如超过1.5秒应该是云函数写的有问题或网络有问题。

三家云厂商仍然在优化冷启动问题。目前给开发者的建议是:
1. 使用clientDB可以减少遇到冷启动问题的概率
2. 非高频访问的云函数,合并到高频云函数中。也有的开发者使用单路由方式编写云函数,即在一个云函数中通过路由处理实现了整个应用的所有后台逻辑。参考[插件](https://ext.dcloud.net.cn/search?q=%E8%B7%AF%E7%94%B1&cat1=7&orderBy=UpdatedDate)。
3. 非高频访问的云函数,可以通过定时任务持续运行它(注意阿里云公测版的定时任务最短周期大于资源回收周期)
989
4. 支付宝云与阿里云支持配置云函数的单实例多并发,请参考:[单实例多并发](cf-functions.md?id=concurrency)
D
DCloud_LXH 已提交
990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071

### 实例与请求@instance

`实例`,指云函数的一个执行环境,可以简单的理解为一个node进程。

每次客户端发起`请求`(request)时,serverless的调度中心会查看已启动、且空闲的实例,分配一个实例来接收这个请求。如果没有空闲实例,则新起一个实例。

新起一个实例需要一定时间,也即上文说的冷启动问题。详见[冷启动](#launchtype)

一个实例启动后,一般在1秒内就会完成请求,但serverless调度中心会保留这个实例一定时间(时间见上一节)。在实例保留期间,客户端再次请求云函数,不会触发冷启动。

也就是说,**在实例保留期间,1个实例会接受多个客户端请求。**

所以要注意`实例`和`请求`不是一对一关系。

`请求`(request),指一次连接云函数的网络请求。不同请求有不同的上下文信息(比如客户端UA)。

所以想要获取客户端信息,一定注意不是在实例的全局对象上获取,而是需要在请求的上下文中获取。[详见]()

在uniCloud阿里云版与支付宝小程序版中还提供了1个实例的多并发请求配置,即同一时间多个请求可以并发执行。
也就是同一时间的请求发到云函数时,没有配置单实例多并发会新开一个云函数实例,配置了单实例多并发则不会新开实例,在一个实例中增加并发。
详见[单实例多并发](#concurrency)。

一个云函数,可以同时存在多个实例。比如cf1云函数,如未配置单实例多并发,10个请求同时达到云函数,就会开10个实例。

不管开了多少实例,云函数的计费是按请求计费的。实例响应请求后到保留期结束之间,如果不发生新请求是不会计费的。

### 云函数的无状态和全局变量@state-less

因为实例可能第一次启动,也可能已经启动,所以云函数中的js全局变量其实是伪全局变量。也就是**云函数是无状态的**。

在云对象中,写在`module.exports = {}`之前,云函数写在`exports.main = async (event, context) => {}`之前的变量定义,是伪全局变量。

它们在实例有效期内的多次请求中会复用。

以如下代码为例,`count`作为全局变量,当多次调用该云函数时,可能会出现变量累加的情况。

- 云对象示例
```js
let count = 0;
module.exports = {
	methoda() {
		return count++
	}
}
```

- 云函数示例

```javascript
let count = 0;
exports.main = async (event, context) => {
	return count++
}
```

上面2个示例中,实例未复用时,也就是冷启动时,count的值每次都是0;若实例被复用,则可能返回1、2、3等各种意外情况。

当然,可以用这个方法来获知一个实例被重用了多少次请求。

**require由于存在缓存,也存在同样的问题。尽量不要直接修改require返回的内容。**

**uniCloud全局对象也是跨请求的,与请求相关的内容不应该挂载到uniCloud全局对象上。**

**正确的全局变量,应该使用如下方案:**
- uni-config-center:静态全局变量可以使用uni提供的配置中心。[详见](uni-config-center.md)
- redis:动态全局变量使用redis。[详见](redis-introduction.md)

### 请求的上下文

由于上节提到的,uniCloud全局对象是实例级的、跨请求的,1个实例内uniCloud只会在冷启动时进行一次初始化。

所以与请求相关的上下文,比如客户端信息,需要通过请求来获取。

为了让开发者清晰的了解实例和请求的关系,uniCloud提供了如下方案。

1. 通过uniCloud.getRequestList(),可以获得当前实例的请求id列表
每个请求,都有一个requestId,在运行回调里、云端日志里都有体现。
```js
uniCloud.getRequestList()
// 返回值:['3228166e-3c17-4d58-9707-xxxxxxxx']
```
D
DCloud_LXH 已提交
1072

D
DCloud_LXH 已提交
1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090
	- 如果未配置阿里云的单实例多并发,getRequestList()返回的数组里面只有一项,即只能拿到当前的请求id。
	- 如果配置了阿里云的单实例多并发,当并发发生时,这个列表就会返回多项,当前并发的每个requestId都在里面。

2. uniCloud.getClientInfos(),可以返回当前所有请求的客户端信息。
该方法返回一个数组,当前每个并发执行中的请求的客户端信息都在里面。
```json
[
  {
    "appId": "__UNI_xxxxx",
    "requestId": "3228166e-3c17-4d58-9707-xxxxxxxx"
    // ...
  }
]
```

	- 如果未配置阿里云的单实例多并发,getRequestList()返回的数组里面只有一项,即只能拿到当前的请求id。
	- 如果配置了阿里云的单实例多并发,当并发发生时,这个列表就会返回多项,当前并发的每个requestId都在里面。

D
DCloud_LXH 已提交
1091
3.
D
DCloud_LXH 已提交
1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102

如果是uniCloud私有云,
如果想获取与请求相关的信息,比如这次请求的客户端UA,或云函数环境信息,无法直接在uniCloud全局对象中获取。


云函数入口入参包含一个event和一个context,这两个参数和请求相关,每次有新请求到云函数实例时就会有一个新的event对象和一个新的context对象

云对象的this和event、context类似,每个请求都对应一个单独的this。

### 单实例多并发@concurrency

1103
> 仅支付宝云与阿里云支持
D
DCloud_LXH 已提交
1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121

默认情况下云函数仅支持单实例单并发,即同一时间一个实例仅可为一个请求服务(不同请求同一时间访问会被分派到不同实例进行处理)。不过在uniCloud web控制台中,阿里云可以通过修改云函数单实例并发度,可以修改云函数同一时间最多能处理多少请求。

假设同时有3个请求需要处理:

当实例并发度设置为1时,需要创建3个实例来处理这3个请求,每个实例分别处理1个请求。而每开启一个实例都会引发云函数冷启动;

当云函数的实例并发度设置为10时(即1个实例可以同时处理10个请求),只需要创建1个实例就能处理这3个请求。这样后面2个并发请求不会有因云函数实例创建带来的冷启动问题。

相关文档:[云函数实例及部分变量说明](#instance) 、[云函数的无状态](#state-less)

**开启方式**

云函数详情页面配置单实例并发度即可,支持1-100之间的数值

**效果**

- 有效减少并发请求时云函数冷启动次数
D
DCloud_LXH 已提交
1122

D
DCloud_LXH 已提交
1123 1124
**使用注意**

1125
- 虽然支付宝云与阿里云云函数支持配置多并发,但在高并发下异步请求排队效果未必好于新开一个实例。尤其是并发操作数据库性能不佳。**一般情况下不要设置过大的并发度,可以自己针对业务代码测试比较下是否启用并发或并发数配成多少**
D
DCloud_LXH 已提交
1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139
- 云函数内存使用量会随着并发量增大而增加,过大的内存可能导致OOM
- 注意云函数是有超时时间的。设置过大的单实例多并发可能会导致实例底层网络请求排队从而导致请求超时,
- 如果并发的不同请求对全局变量同时进行读写会污染全局变量,可能会导致意想不到的后果,详见[全局变量](#state-less)

**适用场景**

|场景									|适用性	|理由																	|
|:-:									|:-:	|:-:																	|
|函数中有较多时间在等待下游服务的响应	|适用	|等待响应一般不消耗资源,在一个实例内并发处理可以节省费用。				|
|函数中有共享状态且不能并发访问			|不适用	|例如全局变量,多请求并发执行修改共享状态可能会导致错误。				|
|单个请求的执行要消耗大量CPU及内存资源	|不适用	|多请求并发执行会造成资源争抢,可能会导致内存不足(OOM)或者延时增加。	|

**关于旧版本uni-id公共模块的特殊说明**

D
DCloud_LXH 已提交
1140
旧版本uni-id公共模块指uni-id-common推出之前的版本。[详见](uni-id/old.md)
D
DCloud_LXH 已提交
1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244

```js
// 开启单实例多并发前的uni-id用法
const uniID = require('uni-id')
exports.main = async function(event, context) {
  const res = uniID.login({
    // ...一些参数
  })
  return res
}

// 由于uni-id默认会从一个内置全局变量上获取客户端平台信息,不同请求会修改此全局变量可能造成混乱,开启单实例多并发后需要将uni-id修改为如下写法
let uniID = require('uni-id')
exports.main = async function(event, context) {
  let uniIDIns = uniID.createInstance({ // 创建uni-id实例,其上方法同uniID
    context: context // 传入context防止不同请求互相影响
    // config: {} // 完整uni-id配置信息,使用config.json进行配置时无需传此参数
  })
  const res = uniIDIns.login({
    // ...一些参数
  })
  return res
}
```

不同于旧版uni-id公共模块,新版uni-id-common不可直接require后使用,必须使用createInstance方法

**进阶**

开启单实例多并发后的全局变量复用并非一定是坏的结果,如果你很了解此行为,也可以对此进行有效的利用

例:[ip-filter](https://ext.dcloud.net.cn/plugin?id=4619)中就利用云函数全局缓存一些ip访问信息来限制单ip访问频率,可以下载示例项目体验一下

受单实例多并发的影响,云函数全局存放与本次请求有关的信息会造成混乱。因此uniCloud提供了根据当前requestId获取客户端信息和云端信息。参考以下文档

- [云函数获取当前requestId](cf-callfunction.md#context)
- [云对象获取当前requestId](cloud-obj.md#get-request-id)
- [获取当前云函数实例正在处理的请求对应的requestId列表](#get-request-list)
- [获取当前云函数实例正在处理的请求对应的客户端信息列表](#get-client-infos)
- [获取当前云函数实例正在处理的请求对应的云端信息列表](#get-cloud-infos)


### 临时存储空间

云函数是运行在云端的代码,运行环境由云服务器弹性调配,这是和传统`Node.js`应用很大的区别。

换言之,云函数每次执行的宿主环境(可简单理解为虚拟机或服务器硬件)可能相同,也可能不同,因此传统`Node.js`开发中将部分信息存储本地硬盘或内存的方案就不再适合。

所以,不建议使用node的fs文件系统相关的API。建议通过云数据库、云存储、redis的方案替代。

### 云函数中的异步行为

书写云函数时应注意`async`、`await`的使用,`nodejs`有内置模块`util`可以将符合`error-first`形式`callback`的函数转换为`promise`形式,[详情参考](https://nodejs.org/api/util.html#util_util_promisify_original),比如以下示例:

```js
const {
	promisify
} = require('util')

let testCallback = {
	value: 'testCallbackValue',
	echo: function(num, callback) {
		setTimeout(() => {
      // 第一个参数为error,第二个为返回值
			callback(null, `${this.value}:${num}`)
		}, 2000)
	}
}

exports.main = async function() {
  // num=2,不传入callback参数,callback会自动作为回调函数处理
	let val = await promisify(testCallback.echo).call(testCallback, 2)
	console.log(val)
	return val
}

```

如果想在云函数内使用回调形式可以让云函数返回一个promise,如以下示例:

```js
exports.main = async function() {
	return new Promise((resolve, reject) => {
		setTimeout(() => {
			resolve('some return value')
		}, 1000)
	})
}
```


### return的策略@return

- 阿里云 return 之后云函数立即终止,逻辑不会继续执行,包括 settimeout 或其他异步操作都会立即终止。
- 腾讯云 node8 return 之后也不会继续执行,但 node12 可以配置是否继续执行
- HBuilderX 本地运行
	* 不通过客户端发起,直接本地运行云函数/云对象,return 之后还可以执行300ms
	* 通过客户端连接本地云函数/云对象,return 之后可以继续执行

**腾讯云因为按 GBS 对云函数计费,在 node12 时,尤其要注意,如果未有效终止云函数,会一直计费**

### node版本
云函数运行在 node 环境中。可以使用 node api `process.version` 获取 node 版本。

1245
- uniCloud 支付宝云默认是 node18, 也可以在 package.json 中选择 node16
WangMoYang's avatar
WangMoYang 已提交
1246
- uniCloud 阿里云默认是 node16,也可以在 package.json 中选择 node12
D
DCloud_LXH 已提交
1247 1248 1249 1250 1251 1252 1253 1254 1255
- uniCloud 腾讯云默认是 node8.9.4,也可以在 package.json 中选择 node12
- HBuilderX 本地运行环境使用的是 HBuilderX 自带的 node 版本,目前为 node16。在 package.json 选择 node版本 只云端生效,且只在第一次上传云函数时生效。

**注意**
- 本地开发一旦使用了 node12 的专用 api,上传云函数时必须在package.json里手动配置选择 node12 的运行环境。
	之所以没有在云端默认统一使用 node12,是因为腾讯云 node12 的 return 策略有一些特殊情况,[见下](?id=return)。
- 运行环境在云端云函数创建时设定,不可通过更新云函数来修改。
	也就是第一次上传云函数的时候,package.json里配了什么,就是什么。如果需要修改node环境,需先删除云端云函数,重新上传。

1256
node版本可以在云函数的package.json文件的`cloudfunction-config->runtime`字段进行配置,详情参考:[云函数package.json](cf-functions.md?id=packagejson)
D
DCloud_LXH 已提交
1257 1258 1259 1260


### 时区

crlfe's avatar
crlfe 已提交
1261 1262 1263
支付宝云云端云函数使用的时区是`UTC+8`。

阿里云和腾讯云云端云函数使用的时区是`UTC+0`,而不是 `UTC+8`,在云函数中使用时间时需特别注意。
D
DCloud_LXH 已提交
1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274

云函数在HBuilderX本地运行时,时区则是电脑的时区,很可能是 `UTC+8`。建议统一使用时间戳,可以规避时区问题。

## 云函数配置

云函数除了代码,还有配置。在[uniCloud web控制台](https://unicloud.dcloud.net.cn/)可以配置;在HBuilderX项目中,云函数根目录的`package.json`也是存放配置的地方。

### 运行内存@memory

云函数运行内存为单个云函数实例使用的内存。

1275
支付宝云云函数默认运行内存512MB,阿里云正式版默认512MB,腾讯云云函数默认运行内存大小为256MB
D
DCloud_LXH 已提交
1276 1277 1278 1279 1280 1281 1282 1283 1284

计算云函数GBs资源消耗时腾讯云会以此内存*运行时间(100ms为阶梯向上取整)得到消耗量。阿里云会以实际运行时间计算GBs,不会按100ms阶梯向上取整

::: warning 注意
阿里云一般场景(比如包含数据库访问时)不建议将内存配置为128MB。如果开发起见遇到数据库响应缓慢,在排除索引等因素的影响后可以检查下是不是云函数内存配置为了128MB,如果是建议调整为256MB
:::

### 超时时间@timeout

WangMoYang's avatar
WangMoYang 已提交
1285
支付宝云定时任务触发最大支持3小时超时时间(需开启异步超时),非定时触发时超时时间为180秒,客户端请求云函数如果超出180秒云函数断开连接后会停止运行。
D
DCloud_LXH 已提交
1286

VK1688's avatar
VK1688 已提交
1287
阿里云定时任务触发最大支持7200秒超时时间,非定时触发时超时时间为120秒,客户端请求云函数如果超出120秒云函数断开连接后会停止运行。
D
DCloud_LXH 已提交
1288 1289 1290

腾讯云定时任务触发最大支持900秒超时时间。非定时触发时超时时间为30秒,客户端请求云函数时如果超出30秒云函数断开链接后会继续运行,最大能运行到配置的超时时间。

1291
如果超时时间仍然不够用,可以参考云函数递归调用,连续执行多个云函数处理一个任务[详情查看](cf-functions.md?id=recurrence)
D
DCloud_LXH 已提交
1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313

### 固定出口IP@eip

#### 腾讯云@tencent-eip

serverless默认是没有固定的服务器IP的,因为有很多服务器资源在后台供随时调用,每次调用到哪个服务器、哪个ip都不固定。

但一些三方系统,要求配置固定ip白名单,比如微信公众号的js sdk,此时只能提供固定ip地址。

腾讯云的收费版,提供了云函数的固定出口ip,在uniCloud [Web控制台](https://unicloud.dcloud.net.cn),创建付费的腾讯云服务空间,选择一个云函数,在云函数的详情界面可以开启固定出口ip。开启后界面上会显示可用的固定ip。拿着这个ip去需要固定ip的界面(如微信公众号管理界面)配置即可。

**注意**

- 如果你是免费版升配到付费版,开启`固定IP`功能后,会导致付费版到期无法自动降级到免费版,请注意按时续费

腾讯云原本的设计是同一个服务空间内所有开启固定出口IP的云函数使用的是同一个IP。但是对于开通vpc的云函数无法和未开通vpc的函数共用同一个出口ip。具体使用中有以下表现

- 开通redis扩展的云函数和未开通redis扩展的云函数会分配不同的ip
- 如果一个云函数已经开通固定出口ip,再关联redis扩展库时固定ip会发生变化

建议已开通redis的服务空间先将云函数关联redis扩展再开通固定出口IP,**2022年7月20日起新上传的云函数会默认开启vpc功能,如需旧云函数和新云函数保持一致可以把旧云函数关联redis扩展后上传一次,注意这样操作会改变旧云函数的固定出口IP**

雪洛's avatar
雪洛 已提交
1314 1315 1316 1317 1318 1319 1320 1321 1322
#### 支付宝云@alipay-eip

**出口IP列表**

```
47.97.38.108
112.124.10.115
```

D
DCloud_LXH 已提交
1323 1324 1325 1326
#### 阿里云@aliyun-eip

> 新增于 HBuilderX 3.5.5

雪洛's avatar
雪洛 已提交
1327
**阿里云必须使用uniCloud.httpProxyForEip发送请求才会固定出口ip**,其原理是通过代理请求获得固定出口IP的能力。IP为轮转不固定,因此三方服务要求使用白名单时开发者需要将代理服务器可能的IP均加入到白名单中,见下方代理服务器列表。此外对于代理的域名有限制,当前仅持`weixin.qq.com`泛域名。若开发者有其他域名代理需求,发送邮件到service@dcloud.io申请,邮件模板参考:[申请解除限制邮件模板](price.md#apply-email-template)。
D
DCloud_LXH 已提交
1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346

**代理服务器IP列表**

```
47.92.132.2
47.92.152.34
47.92.87.58
47.92.207.183
8.142.185.204
```

如需在获取微信公众号access_token场景使用,请将上述ip配置到`微信公众平台 -> 基本配置 -> IP白名单`内,相关链接:[微信公众平台](https://mp.weixin.qq.com/)


##### 发送Get请求@http-proxy-get

**用法**

```js
crlfe's avatar
crlfe 已提交
1347
uniCloud.httpProxyForEip.get(url: String, params?: Object, headers?: Object)
D
DCloud_LXH 已提交
1348 1349 1350 1351 1352 1353 1354 1355
```

**示例**

```js
await uniCloud.httpProxyForEip.get(
  'https://api.weixin.qq.com/cgi-bin/token',
  {
D
DCloud_LXH 已提交
1356
    grant_type: 'client_credential',
D
DCloud_LXH 已提交
1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375
    appid: 'xxxx',
    secret: 'xxxx'
  }
)
```

##### 发送POST请求携带表单数据@http-proxy-post-form

注意,此接口以`application/x-www-form-urlencoded`格式发送数据而不是`multipart/form-data`

**用法**

```js
uniCloud.httpProxyForEip.postForm(url: String, data?: Object, headers?: Object)
```

**示例**

```js
D
DCloud_LXH 已提交
1376
uniCloud.httpProxyForEip.postForm(
D
DCloud_LXH 已提交
1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397
  'https://www.example.com/search',
  {
    q: 'nodejs',
    cat: '1001'
  }
)
```

##### 发送POST请求携带JSON数据@http-proxy-post-json

以`application/json`格式post数据

**用法**

```js
uniCloud.httpProxyForEip.postJson(url: String, json?: Object, headers?: Object)
```

**示例**

```js
D
DCloud_LXH 已提交
1398
uniCloud.httpProxyForEip.postJson(
D
DCloud_LXH 已提交
1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417
  'https://www.example.com/search',
  {
    q: 'nodejs',
    cat: '1001'
  }
)
```

##### POST通用数据@http-proxy-post

**用法**

```js
uniCloud.httpProxyForEip.post(url: String, text?: String, headers?: Object)
```

**示例**

```js
D
DCloud_LXH 已提交
1418
uniCloud.httpProxyForEip.post(
D
DCloud_LXH 已提交
1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466
  'https://www.example.com/search',
  'abcdefg',
  {
    "Content-Type": "text/plain"
  }
)
```

**注意**

- 不支持发送multipart格式的内容
- 代理请求超时时间为5秒
- 上述接口支持本地运行


## 云函数package.json@packagejson

HBuilderX 3.0版本之前,package.json只是一个标准的package.json,安装依赖或公共模块才需要。HBuilderX 3.0及以上版本,package.json也可以用来配置云函数。

uniCloud web控制台提供了很多云函数的设置,比如内存大小、url化、定时触发等,从HBuilderX 3.0起,在云函数的package.json里也可以编写这些设置。

开发者在本地编写云函数的设置,上传云函数,这些设置会自动在云端生效。(本地不生效)

在云端设置了非默认参数后,HBuilderX下载云函数到本地时,也会自动把设置项放入package.json中下载下来。

package.json是一个标准json文件,不可带注释。下面是一个package.json示例。

```json
{
  "name": "add-article",
  "version": "1.0.0",
  "description": "新增文章",
  "main": "index.js",
  "dependencies": {
    // 云函数的依赖,包括公共模块及自行安装的npm依赖
  },
	"extensions": {
		// 云函数使用的扩展库
	},
  "cloudfunction-config": {
		"memorySize": 256,
		"timeout": 5,
		"triggers": [{
				"name": "myTrigger",
				"type": "timer",
				"config": "0 0 2 1 * * *"
		}],
		"path": "",
D
DCloud_LXH 已提交
1467
		"runtime": "Nodejs8"
D
DCloud_LXH 已提交
1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478
	}
}
```

### cloudfunction-config@cloudfunction-config

其中cloudfunction-config字段是云函数配置,支持的配置如下

```js
{
  "concurrency": 10, // 单个云函数实例最大并发量,不配置的情况下默认是1
VK1688's avatar
VK1688 已提交
1479 1480
  "memorySize": 512, // 函数的最大可用内存,单位MB,可选值: 128|256|512|1024|2048,默认值256,阿里云正式版默认512
  "timeout": 60, // 函数的超时时间,单位秒,默认值5。阿里云最长为120秒,阿里云在定时触发时最长可以是7200秒
D
DCloud_LXH 已提交
1481 1482 1483 1484 1485 1486 1487 1488 1489 1490 1491
  // triggers 字段是触发器数组,目前仅支持一个触发器,即数组只能填写一个,不可添加多个
  "triggers": [{ // 阿里云腾讯云均为此形式,请阅读下方说明
      // name: 触发器的名字,规则见https://uniapp.dcloud.net.cn/uniCloud/trigger,name不对阿里云生效
      "name": "myTrigger",
      // type: 触发器类型,目前仅支持 timer (即 定时触发器),type不对阿里云生效
      "type": "timer",
      // config: 触发器配置,在定时触发器下,config 格式为 cron 表达式,规则见https://uniapp.dcloud.net.cn/uniCloud/trigger。使用阿里云时会自动忽略最后一位,即代表年份的一位在阿里云不生效
      "config": "0 0 2 1 * * *"
  }],
  // 云函数Url化path部分
  "path": "",
WangMoYang's avatar
WangMoYang 已提交
1492
  "runtime": "", // nodejs版本,可选Nodejs8、Nodejs12、Nodejs14、Nodejs16、Nodejs18、Nodejs20
D
DCloud_LXH 已提交
1493 1494 1495
  "keepRunningAfterReturn": true // 是否在云函数return之后继续执行,仅腾讯云nodejs12生效,详情见下方说明
}
```
WangMoYang's avatar
WangMoYang 已提交
1496
**runtime 支持情况**
WangMoYang's avatar
WangMoYang 已提交
1497

WangMoYang's avatar
WangMoYang 已提交
1498
阿里云:Nodejs12、Nodejs14、Nodejs16、Nodejs18、Nodejs20,默认Nodejs16
WangMoYang's avatar
WangMoYang 已提交
1499

WangMoYang's avatar
WangMoYang 已提交
1500
腾讯云:Nodejs8(即将下线,不推荐使用)、Nodejs12、Nodejs16、Nodejs18,默认Nodejs16
WangMoYang's avatar
WangMoYang 已提交
1501

WangMoYang's avatar
WangMoYang 已提交
1502
支付宝云:Nodejs16、Nodejs18,默认Nodejs18
WangMoYang's avatar
WangMoYang 已提交
1503

WangMoYang's avatar
WangMoYang 已提交
1504
阿里云Nodejs8已终止支持:终止支持阶段一:禁止新建(2024年06月01日);终止支持阶段二:禁止新建和更新(2024年09月01日)。**终止支持不影响函数继续运行。建议您及时升级**。
WangMoYang's avatar
WangMoYang 已提交
1505

1506
**使用腾讯云Nodejs12及以上版本时,务必仔细阅读此文档:[keepRunningAfterReturn](#keep-running)**
D
DCloud_LXH 已提交
1507 1508 1509

#### 定时任务triggers@triggers

1510
支付宝云与阿里云定时触发的cron表达式不支持代表年的第七位,但是在package.json内配置时仍需将第七位设置为*。
D
DCloud_LXH 已提交
1511

1512
**在web控制台配置trigger请参考:[定时触发](trigger.md)**
D
DCloud_LXH 已提交
1513

1514
package.json内统一了支付宝云、腾讯云、阿里云三家厂商的配置,三个平台都需要配置为如下形式
D
DCloud_LXH 已提交
1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527 1528 1529 1530 1531 1532 1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544 1545 1546 1547 1548 1549 1550 1551 1552

```js
{
	"name": "myTrigger",
	"type": "timer",
	"config": "0 0 2 1 * * *"
}
```

#### keepRunningAfterReturn@keep-running

> 新增于HBuilderX 3.5.1

阿里云、腾讯云nodejs8在云函数return之后其余逻辑会被冻结不再执行。腾讯云nodejs12表现恰好相反,云函数return之后还会等待其余逻辑执行后才会将此云函数实例空闲出来。

以下面的代码为例

```js
exports.main = async function(event, context) {
	setTimeout(()=>{
	  console.log('delay 5 seconds')
	}, 5000)
	return {}
}
```

如果此云函数运行在阿里云或腾讯云nodejs8,setTimeout里面的console.log不会在本次云函数调用执行,但是可能在云函数实例再次被复用时继续执行。

如果此云函数运行在腾讯云nodejs12,setTimeout里面的console.log会在本次云函数调用内,同样的本次云函数**计费时间(与云函数GBs指标相关)**也会按照最终执行完成的时间计算(5000ms+return耗时)。但是前端无需等待5秒即可收到响应。注意:如果有未断开的长连接(例如:redis连接)会导致云函数一直运行到配置的超时时间

当在云函数package.json内的cloudfunction-config内配置了`keepRunningAfterReturn: false`时,可以改变腾讯云nodejs12的表现,云函数return之后将不再继续执行,未断开的长连接也不会增加云函数实际运行时间,云函数return后长连接也不会被中断,简单来说其表现和腾讯云nodejs8一致。

**在云函数中发送网络请求**

将上述示例中的setTimeout换成网络请求、调用其他云函数或数据库请求同理,如果在阿里云或腾讯云nodejs8直接return会导致网络请求可能无法发送(即使成功发送也是在下一次云函数实例被复用的时候),这是与传统开发不太一样的地方。

```js
exports.main = async function(event, context) {
D
DCloud_LXH 已提交
1553
	uniCloud.callFunction({
D
DCloud_LXH 已提交
1554 1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578 1579 1580 1581 1582 1583
    name: 'test',
    data: {}
  })
	return {} // callFunction后不等待直接return时无法调用到test云函数
}
```

**腾讯云nodejs12使用redis**

由于redis需要和服务器建立连接,此连接会阻止云函数结束执行。如果没有云函数return之后还需要继续执行的需求,可以简单的在`cloudfunction-config`内配置`keepRunningAfterReturn: false`。这样redis的连接并不会中断,下次请求来时依然可以使用之前建立的连接。

如果需要return之后继续执行,那么需要在使用完毕后断开redis连接,调用`redis.quit()`方法即可断开连接。需要注意的是断开连接后之前建立的连接将不再可用,下个请求到来时需要使用`uniCloud.redis()`方法重新建立连接。

**如未按照上述说明进行操作,redis连接将会一直占用云函数实例,导致云厂商持续计算云函数执行时间,可能会导致消耗大量云资源而产生额外费用**

**务必确定自己已理解此文档内容,因未按照文档说明使用导致的额外计费DCloud不承担任何责任**

### 注意事项

- 插件作者在发布插件时,如果云函数有特殊设置,应该放入package.json中,然后发布到插件市场。这样就不用再通过说明文档一步一步引导用户去配置云函数定时触发器、内存、url化路径等
- 在web控制台修改云函数配置后,通过HBuilderX的下载云函数菜单会在package.json内添加修改后的云函数配置
- 上传云函数时,如果项目下的package.json内包含云函数配置会同时进行云函数的配置更新
- package.json只有云端部署才生效,本地运行不生效。
- cloudfunction-config不可删除云端配置。例:云端已配置triggers(定时触发器),删除cloudfunction-config内的trigger不会删掉云端的定时触发器
- runtime参数(nodejs版本)仅可在创建云函数时生效,不可修改


### 云函数的数量、体积、冷启动的平衡

鉴于:
1584
- 每个服务空间的云函数数量是有限的,支付宝云是499个,阿里云是48个,腾讯云是149个,[详见](price.md)
D
DCloud_LXH 已提交
1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614 1615 1616
- 每个云函数的体积限制是10M(含node_modules)
- 云函数有冷启动问题

基于以上情况,对开发模式有如下建议:

1. 一般不建议使用体积较大、依赖较深的node_modules。多使用DCloud官方或插件市场提供的库。
2. 优先使用clientDB,不占用云函数数量,也不用编写服务器代码。
3. 对云对象或云函数做适当的合并和拆解。
	- 低频且对用户体验影响较大的操作不建议独立使用云函数,合并到高频云函数中。
	- 控制好单个云函数体积,有的开发者喜欢使用[单路由云函数](https://ext.dcloud.net.cn/search?q=%E8%B7%AF%E7%94%B1&orderBy=WeekDownload&cat1=7),整个服务空间就一个云函数。这也得根据实际情况,如果云函数体积超过6M也还是建议分拆。
	- 用户体系方面,官方已经提供uni-id-co云对象,再搭配clientDB,常规业务就够了。有特殊需求可以再适度补若干云对象。不太会发生云函数数量不足的情况。
4. 必要时可以使用多个服务空间,跨服务空间使用

## cloudfunctions_init(已废弃)

`HBuilderX 2.9`版本,`uniCloud`提供了`cloudfunctions_init.json`来方便开发者快速进行云函数的初始化操作。

**注意:HBuilderX 3.0.0版本起不再使用cloudfunctions_init.json来初始化云函数。改为使用在云函数目录下通过package.json进行配置,具体见上个章节**

详细调整如下:

不再使用cloudfunctions_init.json,内容被分散到每个云函数的package.json的`cloudfunction-config`字段下

package.json是一个标准json文件,不可带注释。下面是一个package.json示例

```json
{
  "name": "add-article",
  "version": "1.0.0",
  "description": "新增文章",
  "main": "index.js",
  "dependencies": {
D
DCloud_LXH 已提交
1617

D
DCloud_LXH 已提交
1618 1619
  },
  "cloudfunction-config": {
VK1688's avatar
VK1688 已提交
1620 1621
      "memorySize": 512,
      "timeout": 60,
D
DCloud_LXH 已提交
1622 1623 1624 1625 1626 1627 1628 1629 1630 1631 1632 1633 1634 1635
      "triggers": [{
          "name": "myTrigger",
          "type": "timer",
          "config": "0 0 2 1 * * *"
      }],
      "path": ""
    }
}
```

cloudfunction-config说明如下

```js
{
VK1688's avatar
VK1688 已提交
1636 1637
  "memorySize": 512, // 函数的最大可用内存,单位MB,可选值: 128|256|512|1024|2048,默认值256,阿里云正式版默认512
  "timeout": 60, // 函数的超时时间,单位秒,默认值5。阿里云最长为120秒,阿里云在定时触发时最长可以是7200秒
D
DCloud_LXH 已提交
1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665 1666 1667 1668 1669 1670 1671 1672 1673 1674 1675 1676 1677 1678 1679
  // triggers 字段是触发器数组,目前仅支持一个触发器,即数组只能填写一个,不可添加多个
  "triggers": [{
      // name: 触发器的名字,规则见https://uniapp.dcloud.net.cn/uniCloud/trigger,name不对阿里云生效
      "name": "myTrigger",
      // type: 触发器类型,目前仅支持 timer (即 定时触发器),type不对阿里云生效
      "type": "timer",
      // config: 触发器配置,在定时触发器下,config 格式为 cron 表达式,规则见https://uniapp.dcloud.net.cn/uniCloud/trigger。使用阿里云时会自动忽略最后一位,即代表年份的一位在阿里云不生效
      "config": "0 0 2 1 * * *"
  }],
  // 云函数Url化path部分
  "path": ""
}
```

**HBuilderX 3.0.0之前版本,请继续阅读下面文档**

**使用方式**
- 在`cloudfucntions`目录右键即可创建`cloudfunctions_init.json`,
- 编写好json内容,在`cloudfunctions_init.json`上右键初始化云函数配置。

**cloudfunctions_init.json形式如下**

```json
{
    "fun-name": { // 云函数名称
        "memorySize": 256, // 函数的最大可用内存,单位MB,可选值: 128|256|512|1024|2048,默认值256,阿里云正式版默认512
        "timeout": 5, // 函数的超时时间,单位秒,默认值5。
        // triggers 字段是触发器数组,目前仅支持一个触发器,即数组只能填写一个,不可添加多个
        "triggers": [{
            // name: 触发器的名字,规则见https://uniapp.dcloud.net.cn/uniCloud/trigger,name不对阿里云生效
            "name": "myTrigger",
            // type: 触发器类型,目前仅支持 timer (即 定时触发器),type不对阿里云生效
            "type": "timer",
            // config: 触发器配置,在定时触发器下,config 格式为 cron 表达式,规则见https://uniapp.dcloud.net.cn/uniCloud/trigger。使用阿里云时会自动忽略最后一位,即代表年份的一位在阿里云不生效
            "config": "0 0 2 1 * * *"
        }],
        // 云函数Url化path部分
        "path": ""
    }
}

```