Docker + Wasm 入门讲解
liuian 2024-12-13 14:56 25 浏览
Wasm是WebAssembly 的缩写, 发音[waz-um,汉语拼音类似wozem],WASM 是 Linux 容器技术的继任者,是目前在 Docker 中使用的 Linux 和 Windows 容器的快速、轻型替代方案。
今天来介绍如何在Docker中运行Wasm 代码。
当然,Docker的Wasm 功能目前处于测试阶段。 Docker官方建议不要在生产环境中使用此功能,此功能可能会改变,也许从未来版本中删除。
Docker+Wasm 整体架构
同上一篇文章讲的Kubernetes中运行Wasm的例子一样, Docker 中启用Wasm,也是通过Containerd + shim的方案进行的:
Docker最近正在做的一件事情是, 将镜像管理迁移到containerd,containerd提供了使用OCI兼容工件和containerd shim的能力。同时,Docker与 WasmEdge 合作创建了一个容器shim。 该填充程序从 OCI 工件中提取 Wasm 模块并使用 WasmEdge 运行时运行它。Docker同时添加了对声明 Wasm 运行时的支持,这将允许使用这个新的 shim。
如果仅仅这么说, 可能很多对于Docker 了解不是很深的人,会感到困惑。先来回顾一下Docker 的基本架构。
回顾Docker 架构
Docker 使用客户端-服务器架构。 Docker 客户端与 Docker 守护进程通信,后者负责构建、运行和分发 Docker 容器的繁重工作。
Docker 客户端和守护进程可以在同一系统上运行,也可以将 Docker 客户端连接到远程 Docker 守护进程。 Docker 客户端和守护进程使用 REST API 通过 UNIX 套接字或网络接口进行通信。 另一个 Docker 客户端是 Docker Compose,它允许您使用由一组容器组成的应用程序。
下面来详细看一下Docker 本身的组件:
- dockerd - Docker 守护进程本身。 Docker 体系中最高级别的组件,也是列出的唯一“Docker”产品。 提供 Docker 所有优秀的 UX 功能。
- (docker-)containerd - 也是一个守护进程,监听 Unix 套接字,公开 gRPC 端点。 处理所有低级容器管理任务、存储、镜像分发、网络连接等......
- (docker-)containerd-ctr - 一个轻量级 CLI,可直接与 containerd 通信。 可以将其视为“docker”与“dockerd”的关系。
- (docker-)runc - 用于实际运行容器的轻量级二进制文件。 处理与 Linux 功能(如 cgroup、命名空间等)的低级接口...
- (docker-)containerd-shim - runC 实际运行容器后,它退出(允许我们没有任何长时间运行的进程负责我们的容器)。 shim 是位于 containerd 和 runc 之间的组件,以促进这一点。
关于Dockerd与Containerd, 这里还需要更加详细的讲解一下。
Docker + Containerd关系
Containerd 是 Docker 生态系统中最近的项目之一,其目的是打破 Docker 架构的更多模块化,并相对于其他行业参与者(云提供商和其他编排服务)更具中立性。
据Docker创始人之一 Solomon Hykes 介绍,自 2016 年 4 月纳入 Docker 1.11 以来,containerd 已部署在数百万台机器上。 宣布的扩展 Containerd 路线图得到了阿里云、AWS、谷歌、IBM、微软和容器生态系统其他活跃成员等云提供商和参与者的意见。
更多 Docker 引擎功能将被添加到 Containerd 中,以便 Containerd 1.0 将提供在 Linux 和 Windows 主机上管理容器所需的所有核心原语:
- 容器执行和监督
- 图片分发
- 网络接口管理
- 本地存储
- 原生管道级别 API
- 完整的 OCI 支持,包括扩展的 OCI 图像规范
要构建、发布和运行容器化应用程序,既可以继续使用 Docker,但如果希望寻找专用组件,您可以考虑使用 containerd。
Docker Engine 1.11 是第一个基于 runC(基于开放容器倡议技术的运行时)和 containerd 构建的版本, 结构如下:
注意, 上面的Docker Engine, 在Docker 二进制代码中,就是Docker;执行顺序如下:
更详细的介绍,参见下图:
各个组件的详细功能介绍:
当我们详细理解Docker Contaier的运行机制之后, 再来看看Docker + Wasm的运行架构图, 就可以更加理解Docker 是如何运行Wasm的了。
说了这么多理论, 运行一个例子看看吧。
二、运行Docker + Wasm的例子
要在 docker 中使用 WASM,需要 Docker Desktop预览版。 如果现有版本的 docker Desktop,请卸载该版本并使用以下与桌面操作系统相关的任何链接安装 docker 桌面技术预览版, 然后更改设置:
使用Rust 编写代码, 首先新建一个项目:
cargo new hello-docker
结果:
更改main.rs:
fn main() {
println!("你好, Docker + Wasm!");
}
此时 Rust 应用程序已编写完成, 下面开始将Rust 代码编译成Wasm代码。
将应用程序编译为 Wasm 二进制文件
切换到 hello-docker 目录并运行以下 Cargo 命令将 Rust 应用程序编译为 wasm32-wasi 二进制文件。 这将创建一个 Wasm 字节码二进制文件,该二进制文件将在任何具有 WebAssembly 运行时的系统上运行。
rustup target add wasm32-wasi
cargo build --target wasm32-wasi --release
该命令将编译后的 hello-docker.wasm Wasm 二进制文件输出到 hello-docker/target/wasm32-wasi/release 文件夹中。
我们将在接下来的步骤中在 Docker 的帮助下执行它。 但是,在此之前,我们将其构建为 OCI 映像,以便可以将其存储在 Docker Hub 中并由 Docker 执行。
将 Wasm 应用程序构建到 OCI 镜像中
Docker 可以将 Wasm 模块打包到 OCI 镜像中。
现在开始编写Dockerfile,从临时基础映像开始,复制 Wasm 模块,并将程序设置为作为 Wasm 二进制文件执行。
FROM scratch
COPY ./target/wasm32-wasi/release/hello-docker.wasm /hello-docker.wasm
ENTRYPOINT [ "hello-docker.wasm" ]
创建 Dockerfile 后,运行以下命令来构建映像。 该命令需要在命令行中进入 Dockerfile 位于同一目录中:
docker buildx build --platform wasi/wasm --provenance=false -t docker-wasm:0.1 .
命令中,--platform wasi/wasm 标志将映像的目标操作系统设置为 wasi,将目标架构设置为 wasm。 -t docker-wasm:0.1 标记镜像“docker-wasm:0.1”,末尾的句点告诉 Docker 使用当前目录中的 Dockerfile。
执行结果:
运行以下命令来验证新映像是否存在:
docker images
至此,Wasm 应用程序已打包在 OCI 镜像中。
将 Wasm 应用程序推送到 Docker Hub
这是一个可选步骤。 如果不关心使用 Docker Hub,请跳过它。 但是,如果您执行此步骤,则需要一个 Docker ID。
运行以下命令以使用您自己的 Docker ID 标记映像,以便您可以将其推送到您自己的 Docker Hub 存储库。 我的 Docker ID 是 hintcnuie,所以我将运行下面的命令。
docker tag docker-wasm:0.1 hintcnuie/docker-wasm:0.1
docker push hintcnuie/docker-wasm:0.1
执行结果:
在Docker Hub 中的镜像:
使用 Docker 运行 Wasm 应用程序
需要 Docker Desktop 4.15 或更高版本才能完成此步骤。
运行以下命令告诉 Docker 运行 OCI 镜像中打包的 Wasm 应用程序。 这是一个包含多行的单个命令:
docker run --rm --name=dockerwasm \
--runtime=io.containerd.wasmedge.v1 \
--platform=wasi/wasm \
hintcnuie/docker-wasm:0.1
--runtime=io.containerd.wasmedge.v1 标志是 Docker 告诉 containerd 使用 runwasi containerd shim 并调用 WasmEdge 来运行镜像中打包的 Wasm 模块的方式。
结果:
可以看到, 在命令行中, “你好, Docker+Wasm!”被成功打印出来, 这也证明我们的Wasm 代码成功在Docker 上执行了。
这个例子, 演示了使用 Docker 构建、共享和运行 Wasm 应用程序!
添加对更多Wasm运行时的支持
Docker 不断在 Docker Desktop 中添加功能并改进对 Wasm 的支持。 最新的技术预览版 (4.24) 支持以下 Wasm 运行时:
- wasmtime
- WasmEdge
- Spin
- slight
- wasmer
- lunatic
- wws
结论
尽管现在来说, Docker 与Wasm的应用还是处于早期, 但是, 在云原生方面, Wasm的方向是明确的, WebAssembly 应用程序必将改变云计算的游戏规则。
这也只是 Docker 和 Wasm 的冰山一角!
相关推荐
- Java 8日期时间API新特性揭秘与深度解析
-
Java8日期时间API新特性揭秘与深度解析在这个万物互联的时代,精准的时间管理显得尤为重要。Java8在日期和时间处理方面为我们带来了革命性的改变。本文将带你全面了解Java8日期时间API的...
- SimpleDateFormat线程不安全的5种解决方案
-
1.什么是线程不安全?线程不安全也叫非线程安全,是指多线程执行中,程序的执行结果和预期的结果不符的情况就叫着线程不安全。线程不安全的代码SimpleDateFormat就是一个典型的线程不安全事例,...
- 软件测试 | Java数据持久化技术(java对象持久化到数据库)
-
TKMyBatis简介TKMybatis是基于Mybatis框架开发的一个工具,内部实现了对单表的基本数据操作,只需要简单继承TKMybatis提供的接口,就能够实现无需编写任何sql...
- 读Java实战(第二版)笔记06_新的日期和时间API
-
1.Java8之前的库对日期和时间的支持非常不理想2.TemporalField接口2.1.定义了如何访问temporal对象某个字段的值的接口2.2.ChronoField枚举2.2.1.实现Te...
- 侠说java8-LocalDateTime等时间使用手册(全),先mark后看
-
前言java8的时间日期api给我们提供了极大的便利。如何更好的熟悉使用时间api也是学习java8的一个很重要的知识点,下面我们一起来学习学习。本篇文章代码比较多,可以作为工具,需要使用时,再来查阅...
- Java培训 | JAVA日期类(java 日期处理类)
-
我们在程序开发过程中经常遇到对日期进行处理的需求,此时能熟练掌握JavaAPI中提供的日期相关类就显得非常重要,可以顺利帮助我们解决遇到的问题。然而在Java8以前,日期和时间处理一直被程序员抱怨太...
- 吊打面试官(十一)--Java语言中日期处理相关问题总结
-
导读在Java中,日期处理是必然遇到的场景,也是问题较多的场景。下面我们将收集的日期问题场景做一个汇总。祝大家面试必过,吊打面试官。让我们0帧起手,开始说具体问题。1.日期格式化大小写问题问题描述:...
- 服务器时间不准,相差 8 个小时?教你轻松解决!
-
在服务器管理中,时间的准确性至关重要。然而,有时我们可能会遇到服务器时间不准,与实际时间相差8个小时的问题。别担心,本文将为你详细介绍解决此问题的方法。218.0.48.3:8800赶紧去搜218...
- Java 8时间类,越用越香(java时间类的定义)
-
为什么会在Jdk8中加入很多时间类非线程安全java.util.Date是非线程安全的,所有的日期类都是可变的,这是Java日期类最大的问题之一。Datedate=newDate();fo...
- 【开源推荐】给大家推荐个基于ChatGPT的PHP开发库 openai-php-api
-
有了这个库大家就可以愉快的使用PHP对接chatGPT的官方接口了,至于对接了官方接口想要做什么就看你自己的啦环境要求PHP7.4或以上composer1.6.5以上支持框架Laravel、Sym...
- 3个最流行的大模型应用开发框架【LLM】
-
随着人工智能的能力,特别是大型语言模型(LLM)的不断发展和演变,开发人员正在寻求将AI功能整合到他们的应用程序中。虽然文本完成和摘要等简单任务可以通过直接调用OpenAI或Coher...
- 都说PHP性能差,但PHP性能真的差吗?
-
今天本能是想测试一个PDO持久化,会不会带来会话混乱的问题先贴一下PHP代码,代码丑了点,但是坚持能run就行,反正就是做个测试。<?php$dsn='mysql:host=l...
- saas介绍和原理 · laravel-独立站-商城SaaS – 湾区梁工
-
SaaS就是多租户,一个应用可以分给很多用户使用,而应用只需要维护一个。那么应用就需要做好各种资源的隔离(数据库,文件,缓存,队列,后台,命令行等等)。有两种类型的多租户SaaS形式:1,单数据库Sa...
- 3分钟短文 | Laravel SQL筛选两个日期之间的记录,怎么写?
-
引言今天说一个细分的需求,在模型中,或者使用laravel提供的EloquentORM功能,构造查询语句时,返回位于两个指定的日期之间的条目。应该怎么写?本文通过几个例子,为大家梳理一下。学习时...
- pip 2(根细胞吸收水借助pip2)
-
pip-h#-helppipdownload#下载.whl文件,然后是可以安装的UV:Python包管理神器-比pip快100倍安装方法Windows:1powershe...
- 一周热门
-
-
Python实现人事自动打卡,再也不会被批评
-
Psutil + Flask + Pyecharts + Bootstrap 开发动态可视化系统监控
-
【验证码逆向专栏】vaptcha 手势验证码逆向分析
-
一个解决支持HTML/CSS/JS网页转PDF(高质量)的终极解决方案
-
再见Swagger UI 国人开源了一款超好用的 API 文档生成框架,真香
-
网页转成pdf文件的经验分享 网页转成pdf文件的经验分享怎么弄
-
C++ std::vector 简介
-
系统C盘清理:微信PC端文件清理,扩大C盘可用空间步骤
-
10款高性能NAS丨双十一必看,轻松搞定虚拟机、Docker、软路由
-
python使用fitz模块提取pdf中的图片
-
- 最近发表
-
- Java 8日期时间API新特性揭秘与深度解析
- SimpleDateFormat线程不安全的5种解决方案
- 软件测试 | Java数据持久化技术(java对象持久化到数据库)
- 读Java实战(第二版)笔记06_新的日期和时间API
- 侠说java8-LocalDateTime等时间使用手册(全),先mark后看
- Java培训 | JAVA日期类(java 日期处理类)
- 吊打面试官(十一)--Java语言中日期处理相关问题总结
- 服务器时间不准,相差 8 个小时?教你轻松解决!
- Java 8时间类,越用越香(java时间类的定义)
- 【开源推荐】给大家推荐个基于ChatGPT的PHP开发库 openai-php-api
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- table.render (33)
- uniapp textarea (33)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)