0

0

Wasm 组件模型和惯用的代码生成

心靈之曲

心靈之曲

发布时间:2024-12-18 12:24:11

|

468人浏览过

|

来源于php中文网

原创

arcjet:使用webassembly组件模型和惯用代码生成构建安全sdk

ArcJet将WebAssembly与我们的安全即代码SDK相结合,允许开发者直接在代码中实现常见的安全功能,例如PII检测和机器人检测。大部分逻辑都嵌入到Wasm中,提供接近原生性能的安全沙箱,符合我们“本地优先安全”的理念。

跨平台运行相同代码的能力非常重要,因为我们构建了从JavaScript到其他技术栈的支持。但这需要一个重要的抽象层来进行跨语言转换(我们的Wasm是从Rust编译的)。

WebAssembly组件模型是实现这一目标的强大工具,但其有效性取决于其周围的实现和工具。对于组件模型,这在主机(执行WebAssembly组件模型的环境)和客户(以任何语言编写并编译到组件模型的WebAssembly模块;在我们的例子中为Rust)的代码生成中最为明显。

组件模型定义了主机和客户之间通信的语言,主要由类型、函数、导入和导出组成。它试图定义一种通用的语言,但某些类型,例如变体、元组和资源,可能在某些编程语言中不存在。

当工具尝试为某种语言生成代码时,开发者通常需要进行创造性的映射,将组件模型类型映射到目标语言。例如,我们使用JCO生成JS绑定,并使用 {tag: string, value: string} 形式的JavaScript对象实现变体。甚至对于 result<_> 类型也有特殊情况,其中错误变体将转换为错误并抛出。

本文探讨了Wasm组件模型如何实现跨语言集成、主机和客户代码生成的复杂性,以及我们为用Go等语言实现惯用代码所做的权衡。

Go的主机代码生成

在ArcJet,我们必须构建一个工具来为主机生成用Go语言编写的代码。虽然我们的SDK尝试在本地分析所有内容,但这并非总是可行,因此我们有使用Go编写的API,它通过附加元数据来增强本地决策。

Go的设计具有非常简洁的语法和类型系统。直到最近,它甚至还没有泛型,并且仍然存在很大的局限性。这使得从组件模型到Go的代码生成变得复杂。

例如,我们可以将 result<_> 生成如下:

type result[v any] struct {
    value v
    err error
}

但这限制了错误位置可以提供的类型。因此,我们需要将其编码为:

type result[v any, e any] struct {
    value v
    err e
}

这可以工作,但与其他惯用的Go代码一起使用会变得很麻烦,后者通常使用 val, err := dosomething() 约定来指示与我们上面定义的 result 类型相同的语义。

此外,构造这个 result 很麻烦:result[int, string]{value: 1, err: ""}。我们可能希望匹配惯用模式,而不是提供 result 类型,以便Go用户能够更自然地使用我们生成的绑定。

惯用映射与直接映射

代码生成可以使语言更自然,也可以更直接地映射到组件模型类型。这两个选项都不适合所有用例,因此由工具开发者决定哪个更有意义。

对于ArcJet工具,我们为 optionresult<_> 类型选择了惯用的Go方法,它们分别映射到 val, ok := dosomething()val, err := dosomething()。对于变体,我们为每个变体创建需要实现的接口,例如:

Ke361开源淘宝客系统
Ke361开源淘宝客系统

Ke361是一个开源的淘宝客系统,基于最新的ThinkPHP3.2版本开发,提供更方便、更安全的WEB应用开发体验,采用了全新的架构设计和命名空间机制, 融合了模块化、驱动化和插件化的设计理念于一体,以帮助想做淘宝客而技术水平不高的朋友。突破了传统淘宝客程序对自动采集商品收费的模式,该程序的自动 采集模块对于所有人开放,代码不加密,方便大家修改。集成淘点金组件,自动转换淘宝链接为淘宝客推广链接。K

下载
type botconfig interface {
    isbotconfig()
}

func (allowedbotconfig) isbotconfig() {}

func (deniedbotconfig) isbotconfig() {}

这在类型安全性和不必要的包装之间取得了良好的平衡。当然,也有一些需要包装的情况,但这些可以作为边缘情况处理。

开发者可能会遇到非惯用模式,导致代码冗长且难以维护。使用既定约定使代码更熟悉,但这确实需要一些额外的努力来实现。

我们决定采用惯用方式来最大限度地减少摩擦,让我们的团队更轻松,这样我们就知道在代码库中移动时会发生什么。

调用约定

工具开发者需要做出的一个重要决定是绑定的调用约定。这包括如何/何时编译导入、是否在设置或实例化期间编译Wasm模块以及清理。

在ArcJet代码库中,我们选择工厂/实例模式来优化性能。编译WebAssembly模块的成本很高,因此我们在 newbotfactory() 构造函数中执行一次。随后的 instantiate() 调用既快速又便宜,从而在生产工作负载中实现高吞吐量。

// ...代码片段...

这种工厂和实例构建模式需要更多代码,但选择它是为了在ArcJet服务的热路径中实现尽可能多的性能。通过预先加载编译成本,我们确保在ArcJet服务的热路径中(延迟最重要)请求处理尽可能高效。这种权衡确实增加了初始化代码的复杂性,但它的回报是每个请求的开销大大降低。

权衡

无论使用原生FFI还是组件模型,任何时候我们需要集成两种或多种语言,都需要做出权衡。

本文讨论了我们在ArcJet中遇到的一些挑战以及我们做出决定的原因。如果我们都基于同一组原语(例如组件模型和WIT)构建,那么我们都可以利用同一组高质量原语,例如 wit-bindgenwit-component,并构建适合每个用例的工具。这就是为什么制定标准对每个人都有帮助。

WebAssembly组件模型为跨语言集成提供了强大的抽象,但将其类型转换为Go等语言会带来微妙的设计挑战。通过选择惯用模式并有选择地优化性能(例如使用工厂/实例模式),我们可以在保持效率的同时提供自然的开发者体验。

随着组件模型工具的发展,我们可以期待更精细的代码生成方法来进一步简化这些集成。

Wasm 组件模型和惯用的代码生成

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

557

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

374

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

754

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

478

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

454

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

1031

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

658

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

553

2023.09.20

Java编译相关教程合集
Java编译相关教程合集

本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

9

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Rust 教程
Rust 教程

共28课时 | 4.6万人学习

Kotlin 教程
Kotlin 教程

共23课时 | 2.7万人学习

Go 教程
Go 教程

共32课时 | 4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号