如何开发自己的第一个 Serverless Component

serverless-components

by yugasun from https://yugasun.com/post/serverless-fullstack-vue-practice.html
本文可全文转载,但需要保留原作者和出处。

前言

上一篇 基于 Serverless Component 的全栈解决方案 介绍 Serverless Component 是什么和如何使用 Serverless Component 开发一个全栈应用。但是目前社区还不够完善,当我们需要一个还没有的组件时,怎么办呢?

与其向官方提交 issue,说明需求,不如自己动手撸一个,岂不快哉~

Serverless Component 运行机制

在开始开发之前,我们先来了解下 Serverless Component 的运行机制:

每个 Serverless Component 实际上就是一个 npm 包,你可以通过 npm install 命令直接安装。当我们在一个依赖 Serverless Component 的应用中,执行命令 serverless --debug 部署时,它首先会读取 serverless.yml 文件中的 component 参数指定组件模块,它会像安装 npm 包一样,自动安装到本地,然后自动注入该组件模块,同时执行组件中的 default 函数(之后会讲到),从而完成部署流程。

开发步骤

一个完整组件的开发流程应该包括以下流程:

  1. 明确功能需求
  2. 定义组件配置:输入和输出参数
  3. 组件开发:default 函数、remove 函数(可选)
  4. 测试组件
  5. 发布 npm 包

接下来将按照以上步骤,一步一步实现腾讯云 CDN 组件。

1. 明确功能需求

腾讯云 CDN 控制台 已经提供了手动配置加速域名的功能,但是作为一名懒惰的程序员, “手动” 一直都是我尝试规避的问题。于是去看了看腾讯云文档,看看官方有没有提供相应便捷的方式。果不其然腾讯云 API 已经提供了相关接口,那么我们为什么不借助 API 实现一个能够帮助我们自动配置的 CDN 组件呢?

需求很明确:开发一个能够自动配置 CDN 加速域名的组件,帮助我们节省手动配置时间。

2. 定义组件配置

要实现 CDN 域名的添加,需要借助2个腾讯云API 接口:新增加速域名HTTPS 配置。通过阅读这两份接口文档,总结出一份配置说明文件 config.md ,内容如下:

MyCDN:
  component: '@serverless/tencent-cdn'
  inputs:
    host: abc.com
    hostType: cos
    origin: www.test.com
    backupOrigin: www.test.com
    serviceType: web
    fullUrl: on
    fwdHost: ww.test.com
    cache:
      - type: 0
        rule: all
        time: 1000
      - type: 0
        rule: all
        time: 1000
    cacheMode: simple
    refer:
      - type: 1
        list:
          - 'qq.baidu.com'
          - '*.baidu.com'
    accessIp:
      type: 1
      list:
        - '1.2.3.4'
        - '2.3.4.5'
    https:
      certId: 123
      cert: 123
      privateKey: 123
      http2: off
      httpsType: 2
      forceSwitch: -2

其中 inputs 就是组件的输入参数,其实这些参数都是从接口文档中拷贝出来而已,实际开发时,需根据自己组件功能,定制化配置就好。

无服务框架的配置都是 yaml 文件,所以在定义组件配置时,需要将 API 的参数做好 yaml 规范映射。比如 yaml 文件中,符号 - 是用来定义数组的。如果对 yaml 语法还不太熟,可以参考这份 YAML 语言教程

组件输入定义好了,还需要定义输出内容,只需要大致的组织 API 请求返回结构就行,尽量简洁明了:

{
    host: 'abc.com',
    hostId: '123'
    origin: 'www.test.com',
    cname: 'www.test.com.cdn.dnsv1.com',
    https: true
}

3. 组件开发

对于一个标准的 Serverless Component ,结构如下:

// serverless.js
const { Component } = require('@serverless/core')
class MyComponent extends Component {
  /*
   * Default (必须)
   * - default 是用来执行、准备和更新你的组建的函数
   * - 执行命令 `$ serverless` 会运行此函数
   * - You can run this function by running the "$ serverless" command
   */
  async default(inputs = {}) {
    return {}
  }

  /*
   * Remove (可选)
   * - 如果你的组件需要删除基础设施,推荐你添加他
   * - 执行命令 `$ serverless remove` 会运行此函数
   */
  async remove(inputs = {}) {
    return {}
  }

  /*
   * Anything (可选)
   * - 如果你想发布带有额外功能的组件,你可以将逻辑写在一个函数里,函数名可以自定义
   * - 执行命令 `$ serverless anything` 会运行此函数
   */
  async anything(inputs = {}) {
    return {}
  }
}
module.exports = MyComponent

了解了组件的结构,接下来,就开始开发吧~

3.1 初始化项目

创建项目目录 tencent-cdn,执行 npm init 初始化项目,根据命令指引,填写相关信息就行:

$ npm init
This utility will walk you through creating a package.json file.
It only covers the most common items, and tries to guess sensible defaults.

See `npm help json` for definitive documentation on these fields
and exactly what they do.

Use `npm install <pkg>` afterwards to install a package and
save it as a dependency in the package.json file.

Press ^C at any time to quit.
package name: (cdn-module) tencent-cdn
version: (1.0.0)
description: Tencent Cloud CDN Component
entry point: (index.js) serverless.js
test command:
git repository:
keywords: cdn,serverless,serverless-component,serverlesscomponent,tencent
author: yugasun
license: (ISC) MIT
About to write to /Users/yugasun/Desktop/Develop/serverless/cdn-module/package.json:

{
  "name": "tencent-cdn",
  "version": "1.0.0",
  "description": "Tencent Cloud CDN Component",
  "main": "serverless.js",
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "keywords": [
    "cdn",
    "serverless",
    "serverless-component",
    "serverlesscomponent",
    "tencent"
  ],
  "author": "yugasun",
  "license": "MIT"
}


Is this OK? (yes)

然后新建 serverless.js 文件,复制上面的模板代码到 serverless.js 文件中。

3.2 编写 default 函数

default 函数代码,这里就不贴出来了,有点多 o(╯□╰)o。

主要思路就是,根据 inputs 输入参数,规范成接口请求参数,然后请求接口,执行配置就好。

对于腾讯云 API,所有的接口请求都需要鉴权,所以这里需要先实例化一个 Capi,如下:

import { Capi } from '@tencent-sdk/capi'
const capi = new Capi({
    SecretId: this.context.credentials.tencent.SecretId,
  SecretKey: this.context.credentials.tencent.SecretKey,
  ServiceType: 'cdn',
})

注意:关于请求云 API 库 @tencent-sdk/capi 说明文档已经很全面了,当然你也可以在这里看到 源码.

它需要传入 SecretIdSecretKeyServiceType 三个参数,SecretIdSecretKey 可以通过 this.context.credentials.tencent 来获取,执行 serverless 命令在执行时,它会根据用户项目根目录配置的 .env 文件,自动注入到 this.context.credentials.tencent 上。ServiceType 是当前服务类型,这是腾讯云 API 定义的,针对不同业务配置相应参数就行。

注意:不同的云服务商挂到 this.context.credentials 上的属性也是不一样,比如这里腾讯云是 tencent,AWS 是 aws,目前支持的所有云服务商的属性配置源码,在这里可以找到,@serverless/cli

然后请求 新增加速域名 接口:

// cdnInputs 就是我们组装好的请求参数
await AddCdnHost(capi, cdnInputs)

这里有个重点:请求 新增加速域名 接口成功返回后,CDN 并不会立即部署成功,这个是需要时间的,所以我们执行后,需要轮训当前新增域名的状态,当为部署成功时,我们才能进行之后的逻辑。

3.3 组件状态保存

Serverless Component 在执行 default 函数时,它会产生一些状态,比如新增 CDN 域名成功后,会产生一个 hostId,我们可以保存在 this.state 对象中,通过执行 this.save() 函数,它会将 this.state 保存到项目根目录的 .serverless 文件夹中一个名为 Template.MyCDN.jsonMyCDN 是我定义的当前 Serverless 应用的名称)文件中,方便之后在做组件建删除时使用。

3.4 编写 remove 函数

Serverless Component 删除的逻辑,就是再 serverless remove 命令时,它会读取 default 函数执行保存到 .serverless 中的状态文件,并注入到 this.state 上 , 然后我们可以根据 state 中的值进行移除,比如我这里会用到 host, 因为 删除加速域名接口 需要传递 host 参数。

3.5 完善说明文档

开源项目的 README 一定要写的清晰明了,方便开发者顺利的使用和开发。

4. 测试组件

到这里我们组件的基本开发完成了,在发布之前,还得进行本地测试,Serverless Framework 提供了一个很好地本地调试方法,就是应用的 serverless.ymlcomponent 可以指定本地项目路径,比如在 tencent-cdn 目录下,创建 test 文件夹,然后新增 serverless.yml 配置如下:

MyCDN:
  component: ../
  inputs:
    host: abc.com
    ...

这里的 ../ 就是相对路径,因为 tencent-cdn 组件的 serverless.js 文件在 tencent-cdn 根目录下,之后我们就可以进入 test 目录,执行部署和移除操作,来测试我们的组件了。

注意:虽然一个 Serverless Component 是一个 npm 模块,我们可以通过 package.json 中的 main 属性指定项目中任意的文件入口,但是如果没有 serverless.js 文件,serverless 命令是没法通过 component 指定的本地路径调试的。

5. 发布 npm 包

发布 npm 包,首先需要你拥有一个 npm 账号,请先前往 npm官网 注册,然后本地执行 npm login 登录你的账号。

经过测试没问题,就可以执行 npm publish 就可以发布到 npm 仓库了。

源码

最终实现源码:@serverless/tencent-cdn

组件引用(补充)

每个组件实例,都会有个 load 方法,我们可以通过此方法来加载其他组件,如下:

const cdnComp = await this.load('@serverless/tencent-cdn', 'cdnComp');

借助此功能,我们可以实现很多高阶组件,比如 @serverless/tencent-website 就是一个很好地案例。

至于如何组合你的组件,实现自己的需求,就靠你自己去天马行空了,是不是已经跃跃欲试了?快来社区贡献你的一份力量吧~

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,287评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,346评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,277评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,132评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,147评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,106评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,019评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,862评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,301评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,521评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,682评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,405评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,996评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,651评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,803评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,674评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,563评论 2 352