Skip to content

应用开发

应用开发规范

快速开始

Serverless Devs的应用开发案例已经被集成到Serverless Devs命令行工具中,通过对Serverless Devs的命令行工具,可以进行空白应用项目的初始化。

s init

Terminal window
$ s init
🚀 More applications: https://registry.serverless-devs.com
? Hello Serverless for Cloud Vendors (Use arrow keys or type to search)
Alibaba Cloud Serverless
AWS Cloud Serverless
Tencent Cloud Serverless
Baidu Cloud Serverless
Dev Template for Serverless Devs

此时,选择最后的Dev Template for Serverless Devs,并按回车:

Terminal window
$ s init
🚀 More applications: https://registry.serverless-devs.com
? Hello Serverless for Cloud Vendors Dev Template for Serverless Devs
? Please select an Serverless-Devs Application (Use arrow keys or type to search)
Application Scaffolding
Component Scaffolding
Plugin Scaffolding

此时,选择Application Scaffolding,并按回车,即可完成一个完整的Serverless Devs的Application项目的初始化,可以通过命令查看文件树:

Terminal window
$ find . -print | sed -e 's;[^/]*/;|____;g;s;____|; |;g'
.
|____readme.md
|____version.md
|____publish.yaml
|____src
| |____s.yaml
| |____index.js
|____hook
| |____index.js

目录结构

推荐的应用目录结构为:

|- src # 目录名字不可以变更
| └── 应用目录
| └── s.yml: 应用描述文件
|- hook # 目录名字不可以变更
| └── 应用目录
| └── index.js: 应用钩子
|- publish.yaml: 项目的资源描述
|- readme.md: 项目简介
|- version.md: 版本更新内容

其中:

目录必须含义
src必须存在应用所在目录
s.yml必须存在应用的资源描述Yaml,需要符合该应用对应的publish,yaml规范
publish.yaml必须存在Serverless Devs Package的开发识别文档
readme.md必须存在对该应用的描述,或帮助文档信息
version.md推荐存在版本的描述,例如当前版本的更新内容等
hook推荐存在init时的钩子函数目录

应用模型元数据

应用模型元数据将会在publish.yaml中进行描述,并在Serverless Registry和Serverless Devs开发者工具侧进行识别和初始化。

publish.yaml文件的基本格式如下所示:

Edition: 3.0.0
Type: Project
Name: 名称
Provider:
- XXXXX # 取值内容参考:https://api.devsapp.cn/v3/common/args.html
Version: 版本,例如0.0.1
Description: 简短的描述/介绍
HomePage: 项目首页地址
Tags: #标签详情
- 部署函数
- 部署组件
Category: XXXXX # 取值内容参考:https://api.devsapp.cn/v3/common/args.html
Service: # 使用的服务
XXXXX: # 取值内容参考:https://api.devsapp.cn/v3/common/args.html
Authorities: #权限描述
- XXXXX # 所需要的权限,例如AliyunFCFullAccess
Organization: 组织名称
Effective: 可视 / Public, Private,Organization
Parameters: # 标准的JSON Scheme
type: object
additionalProperties: false # 不允许增加其他属性
required: # 必填项
- mysqlName
- regionName
properties:
mysqlName: # 正则校验
type: string, # 类型
description: Mysql连接串 # 描述
title: Mysql连接串
pattern: '^mysql:.*$' # 正则表达式

参数详解

目录必须结构含义
EditionString当前Yaml的版本,推荐3.0.0
TypeString类型,包括Component和Project,Plugin三个取值,此处取值Project
NameString应用名称
ProviderList应用所支持的云厂商信息
VersionString应用版本号,例如0.0.1
DescriptionString应用描述(一句话的简短描述)
HomePageString应用的主页,可以填写应用的仓库地址
TagsList应用的标签
CategoryString应用的分类
ServiceStruct应用所需要的服务和相关的权限等描述,例如该应用需要函数计算,Serverless工作流等产品/服务作为支持
OrganizationString应用的组织名称
EffectiveString应用的可视权限
ParametersStruct应用中Yaml内需要填写的字段,严格遵守Json Schema规范标准, 具体可以参考 Parameters 规范

Provider

取值范围:阿里云, 百度智能云, 华为云, 腾讯云, AWS, Azure, Google Cloud, 其它

Provider:
- 阿里云
- 百度智能云

Version

通过不同格式的版本号,来进行开发,灰度,正式的版本划分。

  • x.x.x指正式版本,获取时会取最高版本号的对应内容。
  • dev.x.xdev指开发版本,可以通过拉版本列表获取,也可以通过指定对应版本获取,但是不会通过获取最新版本获取,例如最新版本是1.0.0,之后发布dev.1.1版本,此时获取最新版本依旧是1.0.0版本。
  • beta.x.x#x指灰度版本,可以通过拉版本列表获取,也可以通过指定对应版本获取或拉最新版本获取,其中#后面是一个小于 10 大于 0 的正整数,用于表示当前版本的灰度百分比,例如beta.1.0#4,表示的是,用户在获取最新版本时,有40%概率获得到当前的beta版本。

格式参考:

Version: 1.0.0
Version: dev.0.1
Version: beta.1.0#4

Category

取值范围:基础云服务, Web框架, 全栈应用, 人工智能, 音视频处理, 图文处理, 监控告警, 大数据, IoT, 新手入门, 其它, 开源项目其他`

Category: 基础云服务

Service

取值范围:函数计算, 容器服务, 镜像服务, 消息队列, 工作流, CDN, 对象存储, 表格存储, MNS, 日志服务, API网关, 数据库, 解析服务, 云应用, 其它

Service: # 使用的服务
函数计算:
Authorities: #权限描述
- AliyunFCFullAccess # 所需要的权限,例如AliyunFCFullAccess

Effective

取值范围:Public,Private,Organization

Effective: Public

根据Organization字段的有无(是否加入组织),此字段的作用范围也不一样。可概括如下:

Effective/有无组织
Public所有人可见所有人可见
Private只有自己可见只有自己可见
Organization组织内可见invalid

Parameters

在应用模型中,尽管已经有一个完整的s.yaml用来描述应用的信息,但是实际上还会存在诸如下面的情况:

  • 某些s.yaml中的参数需要使用者来填写,例如某些应用需要连接数据库,此时需要用户在初始化应用的时候进行参数的填写;
  • 某些s.yaml中的参数尽管存在默认值,但是仍任需要用户关注,或者需要用户在某些情况下自定义;

所以,Serverless Package模型,针对Application类型,提供了Parameters参数。通过该参数,可以描述s.yaml中的相关参数,例如:

Parameters: # 标准的JSON Scheme
type: object
additionalProperties: false # 不允许增加其他属性
required: # 必填项
- mysqlName
- regionName
properties:
mysqlName: # 正则校验
type: string, # 类型
description: Mysql连接串 # 描述
title: Mysql连接串
pattern: '^mysql:.*$' # 正则表达式
regionName: # 枚举类型
type: string,
description: 地域Region
default: cn-hangzhou # 默认值
title: 地域
enum: # 枚举类型
- cn-beijing
- cn-hangzhou
- cn-shanghai

此时,在s.yaml中可以引用该字段,例如:

edition: 3.0.0 # 命令行YAML规范版本,遵循语义化版本(Semantic Versioning)规范
name: component-test # 项目名称
vars: # [全局变量,提供给各个服务使用]
domain: xxxx.yyy.com
resources:
component-test:
component: demo
props:
mysqlName: {{ mysqlName }}
regionName: {{ regionName }}

这样,当用户使用s init创建你的应用时,将会被要求输入mysqlNameregionName两个参数,并替换掉双括号内容。

开发与调试

为了兼容spec 0.0.1 中,关于s.yaml的特殊格式定义,在当前版本中:

  1. 如果s.yaml中存在类似'{{ bucket | alibaba oss bucket }}'的内容 ,则直接提醒用户需要输入bucket这样的一个参数,作为Yaml中所必须的参数,并以|之后的内容”alibaba oss bucket”作为解释这个参数的含义;

  2. 如果s.yaml中存在类似"{{ access }}"内容,则判断publish.yaml中是否存在Parameters参数以及相关的Key:

    • 如果存在,则默认进行对应;
    • 如果不存在,直接提醒用户需要输入access这样的一个参数,作为Yaml中所必须的参数;

关于Parameters参数的格式,严格遵循JSON Scheme的规范标准,更多使用示例可查看Parameters参数文档。

私有应用调试

外部应用模版下载

在初始化应用时,指定 uri 参数能够从指定路径下载应用模版

Terminal window
s init --uri https://images.devsapp.cn/goat/start-fc3-nodejs.zip

注意文件内结构和publish.yaml内容必须依照此文档,不然会初始化失败。

发布流程

开发者可以在 src 下完成应用的开发,并对项目进行publish.yaml文件的编写。完成之后,即可通过以下几个步骤发布项目:

  • 更改 publish.yaml 里的 Version 字段。确保版本号比现有最高版本号大 1,例如:1.0.0 -> 1.0.1。

    您可以使用固定的 dev 版本用于持续发布测试版本

  • 首次发布需要通过 registry 命令先登录 Serverless Devs Registry。

    Terminal window
    s registry login

    随后浏览器会跳出登陆窗口,根据提示进行操作即可。

  • 后续直接执行 s registry publish 即可进行发布

  • 测试应用

    如果您使用 dev 版本进行了应用的发布, 假设您的应用名字为 start-application-v3, 那么您可以使用:

查看已发布的应用

详细可见 registry 命令文档

可以通过s registry list指令查看当前登陆到 Serverless Registry 账号所发布的组件。例如:

Terminal window
$ s registry list
-
type: Project
name: start-qwen-api-messages
description: 使用函数计算 FC 快速体验通义千问 API,通过 messages 以文本指令对话
category: 人工智能
tags:
- Web框架
- Flask
- 人工智能
- 通义千问
...

list指令会输出所有组件。在组件过多的情况下,可以通过category, tagpage参数进行筛选,还可以通过search参数搜索特定的组件。

使用钩子函数

hook目录下,可以通过编写index.js内的代码来在s init初始化你的应用时,在初始化前后执行特定操作。index.js初始值如下:

async function preInit(inputObj) {
}
async function postInit(inputObj) {
}
module.exports = {
postInit,
preInit
}

把代码放到preInitpostInit函数中,即可在init操作执行前后执行特定操作,例如:

async function preInit(inputObj) {
console.log(`
Serverless Devs Application Case
Cloud services required:
- FC : https://fc.console.aliyun.com/
Tips:
- FC Component: https://docs.serverless-devs.com/user-guide/aliyun/#fc3`)
}
async function postInit(inputObj) {
console.log(`
* Before using, please check whether the actions command in Yaml file is available
* Carefully reading the notes in s.yaml is helpful for the use of the tool
`)
}
module.exports = {
postInit,
preInit
}

此时,当你执行 s init 命令初始化时,会在初始化之前执行 preInit 函数,初始化之后执行 postInit 函数。

其中传入的inputObj是一个对象,包含以下字段:

{
"name": "(string) 应用名",
"version": "(string) 应用版本",
"appPath": "(string) 保存地址",
"tempAppPath": "(string) 项目临时存储地址",
"logger": "(object) 日志对象",
"fs": "(object) fs-extra库",
"lodash": "(object) lodash库",
}