AWS Lambda + API Gateway 搭建 Restful API 折腾记录

栏目: IT技术 · 发布时间: 4年前

内容简介:最近自己在家搞小玩具,有一个前后端交互的需求,毫无疑问应该用 Restful API 来进行通信。然而根据之前的经验,在 AWS 上最便宜的机器也要一个月三块五毛钱,对于我这种玩具项目来说太过奢侈,所以就想到了由于暂时不打算透露这个玩具项目具体是做啥的,但又想尽可能详细地记录下搭建的过程,这里我会用一个简单的 ISBN 10 转换到 ISBN 13 的 API 为例。选择 Go 语言作为编写工具,主要是因为我之前开发过一个首先,创建一个新的 Lambda 函数

最近自己在家搞小玩具,有一个前后端交互的需求,毫无疑问应该用 Restful API 来进行通信。然而根据之前的经验,在 AWS 上最便宜的机器也要一个月三块五毛钱,对于我这种玩具项目来说太过奢侈,所以就想到了 AWS Lambda ,兼具价格低廉和无需维护的优点。每个月前一百万次请求免费,之后每一百万次请求收费两毛,再合适(便宜)不过了。

由于暂时不打算透露这个玩具项目具体是做啥的,但又想尽可能详细地记录下搭建的过程,这里我会用一个简单的 ISBN 10 转换到 ISBN 13 的 API 为例。选择 Go 语言作为编写工具,主要是因为我之前开发过一个 ISBN 格式转换的库 ,可以拿来即用。

Lambda 函数的配置

首先,创建一个新的 Lambda 函数

AWS Lambda + API Gateway 搭建 Restful API 折腾记录

名字随便起一个,叫 isbnConverter , Runtime 选择 Go 1.x 。然后需要为 Lambda 函数添加触发器,选择 API Gateway ,然后出现如下所示的界面。

AWS Lambda + API Gateway 搭建 Restful API 折腾记录

似乎 HTTP API 是个新东西,更加轻量,对账单更加友好,反正这次是尝试新鲜事物,所以选择这个。如果打算求稳,建议用右边的 REST API ,但目测两者的使用上不会有太大不同。创建完成后,页面自动跳转回 Lambda ,可以看到在下方出现了一个 API Gateway 的板块,同时也提供给我们一个 API 可以调用,格式类似于

https://*******.execute-api.us-west-2.amazonaws.com/default/isbnConverter

在浏览器中打开,可以得到一个字符串 Hello from Lambda! ,说明我们的 API 成功 Ping 通了!

API Gateway 的配置

我们当然不满足于此,我们希望能把 10 位的 ISBN 号转换成对应的 13 位。先来配置 API Gateway ,然后编写相关的代码。到 API Gateway 的路由配置页面,将 endpoint 接受的方法从 ANY 改成 GET ,因为只是一个简单的转换,不需要支持那么多花里胡哨的方法。

AWS Lambda + API Gateway 搭建 Restful API 折腾记录

然后发送几个 http 请求确保设置生效了

~ » curl https://*******.execute-api.us-west-2.amazonaws.com/default/isbnConverter
Hello from Lambda!%                                                                                                ------------------------------------------------------------
~ » curl -XPOST https://*******.execute-api.us-west-2.amazonaws.com/default/isbnConverter
{"message":"Internal Server Error"}%                                                                               ------------------------------------------------------------
~ » curl -XOPTION https://*******.execute-api.us-west-2.amazonaws.com/default/isbnConverter
{"message":"Internal Server Error"}

当然这个 Internal Server Error 的迷惑性很大,但我还没有研究透如何能自定义 HTTP API 的返回状态,按下不表,总之,这里已经达到想要的效果了。

刚刚我们发请求的时候,不得不写下这一长串的 https://*.execute-api.us-west-2.amazonaws.com ,自己手头既然有域名,为什么不直接绑定上去呢? API Gateway 提供了这样的功能,找到自定义域名,把想要绑定的域名加上去,因为是一个 API endpoint ,所以决定使用 api.old-panda.com

AWS Lambda + API Gateway 搭建 Restful API 折腾记录

ACM 证书可以用 AWS Certificate Manager 签一个,操作非常直观方便。接下来还要给自定义域名添加路径映射,我添加的是

AWS Lambda + API Gateway 搭建 Restful API 折腾记录

这样当我们访问 https://api.old-panda.com/book 的时候,会被自动替换为 https://*.execute-api.us-west-2.amazonaws.com/default 。要达到最后的效果,还差一步,就是 DNS 配置,否则当看到 api.old-panda.com 时,域名服务器无从解析到 AWS 为我们自动生成的地址。到 AWS Route 53 添加一条新的 CNAME 记录,子域名自然是 api.old-panda.com ,映射到的目标地址是 API Gateway 自定义域名界面上的 Target Domain Name ,这里很容易跟 API 页面的 Invoke URL 混淆,这俩 URL 的格式非常相似,再次强调, DNS 的 CNAME 配置要映射到自定义域名界面上的 Target Domain Name不是 Invoke URL

验证配置效果

~ » curl https://api.old-panda.com/book/isbnConverter
Hello from Lambda!

编码

因为上面已经决定了, API 只支持 GET 请求,所以传入 ISBN10 的方式也就只能通过 URL 参数(我知道还有其他的方式诸如把数据塞在请求头里,但一般没人那么干),比如说

https://api.old-panda.com/book/isbnConverter?isbn10=7532736555

因此代码首先要拿到 URL 中的参数 isbn10 。首先在代码中定义 URL 参数的格式, Lambda 函数收到的事件格式是一个 json ,其中有一个域 queryStringParameters 包含所有的 URL 参数,所以很直接地,两个结构体如下所示

type requestParams struct {
	QueryStringParameters isbnParam `json:"queryStringParameters"`
}

type isbnParam struct {
	ISBN10 string `json:"isbn10,omitempty"`
}

对照 官方文档 , Handler 函数的签名选择 func (context.Context, TIn) (TOut, error) ,因为我们希望得到一个 json 格式的输入( TIn )同时返回转换后的结果( TOut )。结合 go-isbn 库 ,很容易写出转换函数。整理后的 main.go 文件如下所示

func HandleLambdaEvent(ctx context.Context, eventJSON json.RawMessage) (string, error) {
	var params requestParams
	if err := json.Unmarshal(eventJSON, &params); err != nil {
		return "", fmt.Errorf("Failed to parse url parameters: %v\nError: %v", string(eventJSON), err)
	}

	isbn10 := params.QueryStringParameters.ISBN10
	if isbn10 == "" {
		log.Warn("ISBN10 is not given")
		return "", nil
	}

	isbn13, err := isbn.ConvertToIsbn13(isbn10)
	if err != nil {
		log.Warn("Cannot convert given isbn10: %v to isbn13", isbn10)
		return "", nil
	}
	return fmt.Sprintf("%v", isbn13), nil
}

完整的代码以及 go.mod 文件我放在了 GitHub

部署

同样通过参考 官方文档 ,很容易将本地的 Go 程序部署上线。

BUILD_BIN="isbn10-converter"
GOOS=linux go build -o $BUILD_BIN
zip function.zip $BUILD_BIN
aws lambda update-function-code --function-name isbnConverter --zip-file fileb://function.zip

验证成果

~ » curl https://api.old-panda.com/book/isbnConverter\?isbn10\=7532736555
"9787532736553"

这样一个简单的 Restful API endpoint 就搭建好了。


以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

鸟哥的Linux私房菜 基础学习篇(第二版)

鸟哥的Linux私房菜 基础学习篇(第二版)

鸟哥 / 人民邮电出版社 / 2007-9 / 65.00元

《鸟哥的Linux私房菜基础学习篇(第二版)》全面而详细地介绍了Linux操作系统。全书分为5个部分:第一部分着重说明Linux的起源及功能,如何规划和安装Linux主机;第二部分介绍Linux的文件系统、文件、目录与磁盘的管理;第三部分介绍文字模式接口shell和管理系统的好帮手shell脚本,另外还介绍了文字编辑器vi和vim的使用方法;第四部分介绍了对于系统安全非常重要的Linux账号的管理......一起来看看 《鸟哥的Linux私房菜 基础学习篇(第二版)》 这本书的介绍吧!

随机密码生成器
随机密码生成器

多种字符组合密码

SHA 加密
SHA 加密

SHA 加密工具

XML、JSON 在线转换
XML、JSON 在线转换

在线XML、JSON转换工具