Rust 实战练习 - 12. Axum Web 简单demo

embedded/2024/9/24 6:21:21/

Rust Web 历程

Rust 的异步框架tokio非他莫属,而web框架一直是悬而未决,说到底还是因为没有官方成熟的方案指引,大家各玩各的,互不兼容,白白浪费精力。

这个事情一直等到半官方组织tokio推出axum有了改善。但是市场上仍然乱七八糟,具体细节可以参考:https://zhuanlan.zhihu.com/p/398232138

现在相对靠谱的发展方向参考如下图:
在这里插入图片描述

但是 tower和tower-http这2个项目比较奇葩,sample和docment严重缺少,所以建议做如下研究:

  • tokio
  • axum
  • tonic
  • sqlx

axum 简单demo

这个框架也不是特别成熟,在multipart有大坑!
注意仔细参考我的代码!

rust">use axum::{extract::{DefaultBodyLimit, Form, Multipart, Path, Query}, http::{Method, StatusCode}, response::{Html, IntoResponse}, routing::{get, post}, Json, Router};
use serde::{Deserialize, Serialize};#[tokio::main]
async fn main() {let routes = Router::new().route("/", get(page_index)).route("/hello", get(|| async {println!("{:<12} - hello", "HANDLER");Html("hello world!")})).route("/user", post(page_user)).route("/user2", get(page_user2)).route("/user3/:username", get(page_user3)).route("/form", get(form_get).post(form_post))// 复杂的提取,需要参考 https://docs.rs/axum/latest/axum/extract/index.html.route("/form2", get(form_get_file).post(form_post_file).layer(DefaultBodyLimit::disable()));let listener = tokio::net::TcpListener::bind("0.0.0.0:8080").await.unwrap();println!("Server listen on: {:?}", listener.local_addr());axum::serve(listener, routes).await.unwrap();}#[derive(Deserialize, Debug)]
struct ReqUser {username: String,
}#[derive(Serialize, Debug,Deserialize)]
struct RespUser {id: u64,username: String,
}async fn page_index() -> &'static str{"welcome to axum based on tokio!"
}// e.g.: Post /user  Body: {"username": "xxxx"}
async fn page_user(Json(req): Json<ReqUser>,
) -> (StatusCode, Json<RespUser>) {println!("{:<12} - page_user - {req:?}", "HANDLER");let user = RespUser {id: 1337,username: req.username,};(StatusCode::OK, Json(user))
}// e.g.: GET /user2?username=abc
async fn page_user2(Query(req) : Query<ReqUser>
) -> impl IntoResponse {println!("{:<12} - page_user2 - {req:?}", "HANDLER");let user = RespUser {id: 1338,username: req.username,};Html(format!("{user:?}"))
}// e.g.: GET /user3/username
async fn page_user3(Path(req_name) : Path<String>
) -> impl IntoResponse {println!("{:<12} - page_user3 - {req_name:?}", "HANDLER");let user = RespUser {id: 1339,username: req_name,};Html(format!("{user:?}"))
}async fn form_get() -> Html<&'static str> {Html(r#"<!doctype html><html><head>form test</head><body><h2> normal form </h2><form action="/form" method="post"><label for="username">Enter your name:<input type="text" name="username"></label><br><label>Enter your id:<input type="text" name="id"></label><br><input type="submit" value="Ok"></form></body></html>"#,)
}
async fn form_get_file() -> Html<&'static str> {Html(r#"<!doctype html><html><head>form test</head><body><h2> normal form </h2><form action="/form2" method="post" enctype="multipart/form-data"><label>Enter your id:<input type="text" name="id"></label><br><label>Upload file:<input type="file" name="myfile" multiple></label><br><input type="submit" value="Upload files"></form></body></html>"#,)
}// 支持多个提取器
async fn form_post(_method: Method,Form(user): Form<RespUser>
) -> String {//dbg!(&user);format!("{user:?}")
}// body 部分只支持一种,不冲突的支持多种
// Form与Multipart冲突,保留multipart
async fn form_post_file(_method: Method,mut multipart: Multipart,
) -> String {while let Some(field) = multipart.next_field().await.unwrap() {let name = field.name().unwrap().to_string();if name == "myfile" {let file_name = field.file_name().unwrap().to_string();let content_type = field.content_type().unwrap().to_string();let data = field.bytes().await.unwrap();println!("form upload [{name}] = {file_name}, data len: {}, type: {content_type}",data.len());}else{let val = field.text().await.unwrap();println!("form field [{name}] = {val}");}}format!("{_method:?}")
}

http://www.ppmy.cn/embedded/21059.html

相关文章

【高阶数据结构】B树 {B树的概念;B树的实现:节点设计,查找,插入,遍历,删除;B树的性能分析;B+树和B*树;B树的应用}

一、常见的搜索结构 以上结构适合用于数据量相对不是很大&#xff0c;能够一次性存放在内存中&#xff0c;进行数据查找的场景。如果数据量很大&#xff0c;比如有100G数据&#xff0c;无法一次放进内存中&#xff0c;那就只能放在磁盘上了&#xff0c;如果放在磁盘上&#xff…

Shift+Delete删除的文件还可以恢复吗?3个小妙招立马解决!

在计算机使用过程中我们经常会遇到误删文件的情况&#xff0c;有时时由于操作失误或者急于清理磁盘空间&#xff0c;我们会使用ShiftDelete组合键直接删除文件而不经过回收站。这样的操作往往会让用户感到焦虑&#xff0c;因为文件似乎被“永久删除”了。那么ShiftDelete删除的…

Mac如何更换终端默认的python版本

一、Mac中如何更换终端默认的python版本 1、打开终端输入下列代码&#xff0c;打开配置文件 open ~/.bash_profile 2.在打开的配置文件中输入&#xff08;以下代码3.12替换为所下载的版本即可&#xff09; #下面是修改python版本的命令就是环境变量 export PYTHON_HOME/Lib…

小龙虾优化算法(Crayfish Optimization Algorithm,COA)

小龙虾优化算法&#xff08;Crayfish Optimization Algorithm&#xff0c;COA&#xff09; 前言一、小龙虾优化算法的实现1.初始化阶段2.定义温度和小龙虾的觅食量3.避暑阶段&#xff08;探索阶段&#xff09;4.竞争阶段&#xff08;开发阶段&#xff09;5.觅食阶段&#xff08…

Linux网络配置以及DHCP

一、网络配置 将Linux主机接入到网络&#xff0c;需要配置网络相关设置 一般包括如下内容&#xff1a; 主机名 IP/netmask 路由&#xff1a;默认网关 DNS服务器 主DNS服务器 次DNS服务器 第三个DNS服务器 1.1显示网络接口的信息 ifconfig命令用于显示和配置网络接口的信…

web前端框架设计第六课-样式绑定

web前端框架设计第六课-样式绑定 一.预习笔记 1.class属性绑定 给P标签绑定一个类样式&#xff0c;类名为active。当active取值为true时&#xff0c;表示绑定样式成功&#xff0c;取值为false时&#xff0c;取消绑定 以对象形式给P标签绑定多个类样式 以数组形式给P标签绑定多…

【综述】DSP处理器芯片

文章目录 TI DSP C2000系列 TMS320F28003X 典型应用 开发工具链 参考资料 TI DSP TI C2000系列 控制领域 TI C5000系列 通信领域 TI C6000系列 图像领域 C2000系列 第三代集成了C28浮点DSP内核&#xff0c;采用了65nm工艺&#xff08;上一代180nm&#xff09; 第四代正在…

【Camera KMD ISP SubSystem笔记】Request 流转与Bubble机制

ISP中断类型 SOF: 一帧图像数据开始传输 EOF: 一帧图像数据传输完成 REG_UPDATE: ISP寄存器更新完成(每个reg group都有独立的这个中断) EPOCH: ISP某一行结尾(默认20)就会产生此中断 BUFFER DONE: 一帧图像数据ISP完全写到DDR了 管理Isp request的几个List st…