生成式AI领域的最新成果都在这里!抢 QCon 展区门票 了解详情
写点什么

适用于 AWS Lambda 的 Dart 运行时简介

  • 2020-02-27
  • 本文字数:5331 字

    阅读完需:约 17 分钟

适用于 AWS Lambda 的 Dart 运行时简介

Dart 是一种发展迅速的开源编程语言,并且能为一些快速发展的开源项目提供支持,如 Flutter。借助自定义 AWS Lambda 运行时,您可以在 AWS Lambda 中运行 Dart。以 Dart 编写函数有助于让您使用自己的技能开发移动应用程序,从而建立无服务器后端。您也可以在应用程序和后端之间共享代码,并且使用静态类型语言的强大功能。本篇博文将阐释我们如何通过自定义 Lambda 运行时来提供语言支持,以及如何以 Dart 创建 Lambda 函数。文章还将分享一些诀窍,在您的构建过程中助您一臂之力。

什么是 Dart?

Dart 是一种热门的开源编程语言,它以对象为导向,是通用型编程语言。Dart 可用于编写命令行脚本、移动应用程序、Web 应用程序或服务器端应用程序。这是一种静态类型语言,具有可靠的类型系统,并且配备功能强大的包管理器和开发人员工具。


这种语言是 2018-2019 年发展最迅速的语言之一,这并不让人感到意外,因为最热门的 Flutter 移动开发工具包正是使用 Dart 编写。此开发工具包是 GitHub 上最为流行的开源项目之一。深入研究之前,您可以先访问 Dart 网站了解这种语言及其功能,并使用 DartPad 对它进行实验。

Dart 和 AWS Lambda 自定义运行时

利用自定义 AWS Lambda 运行时,您可以在任何编程语言中执行运行时。自定义 Lambda 运行时会在运行的时候使用 AWS Lambda 运行时界面。它定义了基于 HTTP 的 Lambda 编程模型规范,自定义运行时使用此规范来处理调用请求。借助 AWS Lambda 自定义运行时,您还可以利用最近发布的 Dart 编程语言运行时,在 Lambda 中执行 Dart。借助我们的运行时,您也可以用 Dart 来进行无服务器计算。

入门

在下面的步骤中,我们将使用 Dart 自定义运行时来设置 Lambda 函数,然后对其进行配置以通过事件源接收请求,对此,我们以 Application Load Balancer 为例。此项目将向请求者返回简单的消息。


对于目标为设备(移动、桌面或服务器)的程序,Dart 支持 Dart VM 使用适时 (JIT) 编译和预先 (AOT) 编译器来生成各自的机器代码。对于目标为 Web 的程序,Dart 提供 Dart to JavaScript 编译器。AWS Lambda 是服务器设备目标,并且运行时项目使用 AOT 编译器来为目标生成机器代码。


Dart 2.6 推出了 dart2native 命令行工具,可更轻松生成本机机器代码。此工具还可将 Dart 程序编译为所需的本机 x64 机器代码,此外还有小型 Dart 运行时,可处理类型检查和垃圾收集。运行时项目使用此工具来将 Lambda 函数和运行时编译为 AWS Lambda 的本机机器代码。创建的二进制可在自定义运行时使用的标准 Lambda 执行环境中执行。


AWS Lambda 可运行创建的函数以响应多个本机事件,例如通过 Application Load Balancer (ALB) 的 HTTP 请求。ALB 可触发函数来处理请求。



调用函数的事件包含请求元数据和主体。在本文中,我将说明如何构建可处理事件的函数。函数将返回 HTTP 响应作为处理结果。

先决条件

您需要源代码编辑器来为 Lambda 函数创建源文件,以及为包管理器创建包规范。如果您喜欢开源,并且希望快速开始创建函数,那么 Visual Studio Code 和 Dart Code 插件是理想之选。插件需要您获取 Dart 开发工具包来使用其大部分功能。

创建、构建和部署 Dart 函数

要开始,先为您的 Dart 包创建文件夹,然后为其添加包规范。您需要通过规范来使用 pub 包管理器,其可通过规范中的信息来管理 Dart 程序的依赖项和环境。


以下命令可为包创建文件夹:


Bash


bash $ > mkdir my_lambda && cd $_
复制代码


在文件夹中,通过以下内容创建 pubspec.yaml 文件:


Bash


name: my_lambdaenvironment:  sdk: ">=2.6.0 <3.0.0"dependencies:  aws_lambda_dart_runtime:    git: https://github.com/awslabs/aws-lambda-dart-runtime.gitdev_dependencies:  build_runner:
复制代码


这是包规范。如此可将 Dart 运行时指定为程序的依赖项。将从 Dart Lambda 运行时存储库的主分支获取包,直至 AWS 在 pub.dev(一个公开可用的包存储库)上成为发布者。


接下来,创建 main.dart 文件并在您偏好的源代码编辑器中打开。将以下 Dart 程序复制到文件中:


Bash


import 'package:aws_lambda_dart_runtime/aws_lambda_dart_runtime.dart';
void main() async { /// This demo's handling an ALB request. final Handler<AwsALBEvent> helloALB = (context, event) async { final response = '''<html><header><title>My Lambda Function</title></header><body>Success! I created my first Dart Lambda function.</body></html>'''; /// Returns the response to the ALB. return InvocationResult( context.requestId, AwsALBResponse.fromString(response)); }; /// The Runtime is a singleton. /// You can define the handlers as you wish. Runtime() ..registerHandler<AwsALBEvent>("hello.ALB", helloALB) ..invoke();}
复制代码


程序会在运行时中将 helloALB 函数注册为 hello.ALB 句柄。句柄是代码中特定函数的标识符,AWS Lambda 运行函数时可调用此标识符。调用请求包含运行时应执行的句柄,并且运行时会将此句柄映射到届时运行的真实函数。程序中的 hello.ALB 句柄的类型将归为 AwsALBEvent 事件。因此,AWS Lambda 调用 hello.ALB 句柄时,运行时预计会收到 Application Load Balancer 事件。AwsALBEvent 就是 Dart 运行时中一个便捷的封套,可从 Application Load Balancer 事件中将 JSON 提取到 Dart 数据结构。请参阅 AWS Lambda 的 Dart 运行时文档,了解所有可用事件。


下一步是编译函数,其中包括警告。dart2native 工具不提供跨平台编译器支持(请参阅 #28617)。如果您未使用 Linux x86-64 平台,这一点十分重要。未在此平台上编译的二进制无法在 AWS Lambda 上执行。您可学习如何使用 Docker 在其他平台上编译函数。

使用 Docker 编译

首先,如果尚未安装 Docker,则您必须在操作系统中安装 Docker。如果您不了解 Docker,请参阅“什么是 Docker?”。如需编译函数,您需要使用官方的 google/dart Docker 容器。


接下来,继续逐步操作,生成二进制。如果您使用持续集成,则以下步骤均可自动执行。如需查看 build.sh 脚本示例,请查看 Dart Lambda 运行时项目示例文件夹中的脚本。


首先,在容器中挂载包,然后为其创建交互式 Shell:


Bash


bash $ > docker run -v $PWD:/app -w /app -it google/dart /bin/bash
复制代码


该命令会在 /app 输入工作目录。通过 pub get 获取所需依赖项。这样便准备就绪,可以将 Dart 程序编译为机器代码,并且 Dart 运行时为:


Bash


bash $ > dart2native main.dart -o bootstrap
复制代码


运行 ls -lh 应该具有以下输出。bootstrap 文件为编译后的 main.dart 程序;有意将此文件命名为 bootstrap。Lambda 运行时在启动该函数时会执行名为 bootstrap 的任何文件(请参阅“自定义 AWS Lambda 运行时”文档):


Bash


-rwxr-xr-x 1 root root 8.0M Feb 3 10:03 bootstrap-rw-r--r-- 1 root root 665 Feb 3 09:55 main.dart-rw-r--r-- 1 root root 9.2K Feb 3 10:02 pubspec.lock-rw-r--r-- 1 root root 193 Feb 3 09:25 pubspec.yaml
复制代码


最后一步,使用 exit 在容器中退出交互式 Shell。

在 Linux 上编译

如果您是在 Linux x86-64 上运行,则不必使用 Docker 容器;不过,您必须在开发环境中安装 Dart


Bash


bash $ > pub get && dart2native  main.dart -o bootstrap
复制代码


这些命令获取依赖项,并将 main.dart 中的程序编译为引导二进制文件,此二进制文件可在自定义 AWS Lambda 中执行。您将会看到一条确认已成功生成该二进制文件的消息。由于自定义运行时在启动函数时将执行任何名为 bootstrap 的文件,因此该二进制文件命名为 bootstrap

构建函数

bootstrap 二进制文件为最后一步准备就绪时,即可对其进行部署:通过 zip -j lambda.zip bootstrap 创建该二进制文件的 .zip 文件。lambda.zip 文件包用于在 AWS Lambda 中部署函数。

在 AWS Lambda 中部署函数

您所创建的函数的目的是通过 Application Load Balancer HTTP(S) 请求对该函数进行调用(请参阅文章“Lambda 函数作为 Application Load Balancers 的目标”)。该函数预期运行时将返回一个 ALB 调用事件,此调用事件由函数封装在 Dart 对象中。


下一步,导航至 AWS Lambda 控制台,然后单击创建函数



从零开始创建函数并将其命名为 myLambda。下一步,从运行时下拉菜单中提供您自己的 bootstrap。为 Lambda 创建一个新执行角色并将其命名为myLambdaExecutionRole,不过,您也可以使用现有角色。最后,单击创建函数以创建函数。会出现一条告知您已成功创建函数的消息,您也会被重定向至该函数的详细信息页面。


下一步,部署 Lambda 函数。在函数代码部分下,选择上传 .zip 文件。将处理器重命名为 hello.ALB,此名称为您在我们 main.dart 中注册的处理器的名称。然后,在函数包下单击上传,从 my_lambda 文件夹选择 lambda.zip。按保存以上传函数包。



您已成功部署 Lambda 函数。

测试函数

下一步是测试函数是否有效以及处理器是否由运行时调用。因此,您必须创建一个测试事件并调用函数。


要测试事件并调用函数,单击选择测试事件,然后选择配置测试事件选项。



将测试事件命名为 myLambdaALBTestEvent,并将文档中的 JSON 用于事件主体:


Json


{    "requestContext": {        "elb": {            "targetGroupArn": "arn:aws:elasticloadbalancing:us-east-2:123456789012:targetgroup/lambda-279XGJDqGZ5rsrHC2Fjr/49e9d65c45c6791a"        }    },    "httpMethod": "GET",    "path": "/lambda",    "queryStringParameters": {        "query": "1234ABCD"    },    "headers": {        "accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8",        "accept-encoding": "gzip",        "accept-language": "en-US,en;q=0.9",        "connection": "keep-alive",        "host": "lambda-alb-123578498.us-east-2.elb.amazonaws.com",        "upgrade-insecure-requests": "1",        "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36",        "x-amzn-trace-id": "Root=1-5c536348-3d683b8b04734faae651f476",        "x-forwarded-for": "72.12.164.125",        "x-forwarded-port": "80",        "x-forwarded-proto": "http",        "x-imforwards": "20"    },    "body": "",    "isBase64Encoded": false}
复制代码



单击创建来创建测试事件。myLambdaALBTestEvent 应已创建并按预选设置出现在下拉菜单中。



准备就绪,可以测试事件了。单击测试按钮以触发测试事件。展开执行过程的详情,将显示对请求的成功响应,其中包括示例 HTTP 响应主体。



既然您已经成功测试了函数,说明您已经准备好了将函数与实际 ALB 进行关联。您可以参阅“Lambda 函数作为 Application Load Balancer 的目标”一文了解更多信息。

自定义事件

Dart 运行时支持自定义事件。此类事件并非 AWS 平台及其服务的原生事件,而是由客户自己所创建的。要创建自定义事件,您需要创建事件的类别,并使用运行时的实例注册事件。运行时以单例模式执行,以便在您的代码结构灵活使用并可避免创建多个实例。


为新的自定义事件创建 MyCustomEvent 类别。类别仅需实施 MyCustomEvent.fromJson(Map<String, dynamic> json) 工厂方法。在稍后查找事件处理器的 MyCustomEvent 以及通过事件的 JSON 表达方式创建事件时,运行时会调用此方法。


Bash


class MyCustomEvent {  factory MyCustomEvent.fromJson(Map<String, dynamic> json) =>      MyCustomEvent(json);  const MyCustomEvent();}
Register the MyCustomEvent with the runtime as follows:Runtime() ..registerEvent((Map<String, dynamic> json) => MyCustomEvent.from(json);The full example of a MyCustomEvent looks like this:import 'package:aws_lambda_dart_runtime/aws_lambda_dart_runtime.dart';class MyCustomEvent { factory MyCustomEvent.fromJson(Map<String, dynamic> json) => MyCustomEvent(json); const MyCustomEvent();}void main() async { final Handler successHandler = (context, event) async { return InvocationResult(context.requestId, "SUCCESS"); }; Runtime() ..registerEvent((Map<String, dynamic> json) => MyCustomEvent.from(json)) ..registerHandler("doesnt.matter", successHandler) ..invoke();}.
复制代码


处理器函数在 Context 对象中传递。该对象包含关于来自 Lambda 运行时界面的调用的信息,例如 requestIdregionfunctionName。参阅 AWS Lambda 的 Dart 运行时文档,了解有关自定义事件和完整 API 的更多信息。

结论

Dart Lambda 运行时目前仍处于早期阶段。维护人员希望获得您的反馈意见以帮助其发展。维护人员尤其希望了解您实验所用的工作负载,或尚不完善和需要提供支持的工作负载。您可以访问 aws-lambda-dart-runtime Github 存储库,分享您的经验或提交问题。


本文转载自 AWS 技术博客。


原文链接:https://amazonaws-china.com/cn/blogs/china/introducing-a-dart-runtime-for-aws-lambda/


2020-02-27 17:05533

评论

发布
暂无评论
发现更多内容

POI内存溢出故障排查

Season

JVM POI jvm调优

Redis系列(六):你说要看Redis线程模型?安排

z小赵

redis 高并发

OAM 深入解读:如何基于 OAM Runtime 编写一个扩展 Trait?

钱王骞

云原生 k8s OAM

架构师训练营第八周学习总结

张明森

Python Kafka 报错:ImportError: cannot import name 'KafkaConsumer'

BigYoung

Python kafka importerror 报错

【架构训练 Week07 作业】

Rex

除了技术,加密货币开发者更应关注可使用性

CECBC

加密货币 用户为本 可使用性 容错机制

JVM系列之:对象的锁状态和同步

程序那些事

JVM GC 同步

我的 20 条工作原则

霍太稳@极客邦科技

成长 知识管理 职场成长

Docker网络学习第四篇-Namespace通信实战

Lazy

Docker Linux 网络

22种超全用户触点采集,易观方舟SDK又更新了

易观大数据

性能优化

独孤魂

Demo 示例:如何原生的在 K8s 上运行 Flink?

Apache Flink

flink

【区块链+通证经济】从量变到质变区块链发展的下一阶段是什么?

CECBC

数字货币 防篡改 通证

眼见为实,华为鲲鹏架构服务器生态大揭秘

华为云开发者联盟

华为 鲲鹏920 服务器 云服务 华为云

PromiseKit 源码阅读

fuyoufang

架构师训练营第八周笔记

Melo

腾讯面试题: 百度搜索为什么那么快?

小松漫步

面试

信创舆情一线--两部门发文加强对数字货币等新型权益的保护

统小信uos

LeetCode题解:1. 两数之和,JavaScript,双循环暴力解法,详细注释

Lee Chen

大前端 LeetCode

如何识别刷屏文章中的伪科学

Lee Chen

大前端 随笔杂谈

关于中台,可能都是正确的废话

FinClip

中台 业务中台

读《我们为什么要去火星》随笔

Jackchang234987

产品 人生 读书 随笔杂谈

压测工具试验

独孤魂

四十个鹏城春夏,一场数字繁花

脑极体

最高法主张加强数字货币产权保护有法可依

CECBC

数字货币 法偿货币 中国人民银行 虚拟财产

高能预警!Apache Flink Meetup · 上海站返场啦

Apache Flink

flink

性能优化概述

superman

SpreadJS 纯前端表格控件应用案例:雷鸟365在线文档系统

葡萄城技术团队

大前端 SpreadJS 在线文档

B站新一代golang规则引擎的设计与实现

calo

B站 高并发 AST 规则引擎 Go 语言

脑洞:基于Enterprise Continuum证明DDD用于构建汽车的可行性

冯文辉

企业架构 领域驱动设计 DDD 架构演进

适用于 AWS Lambda 的 Dart 运行时简介_语言 & 开发_AWS_InfoQ精选文章