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

news2024/11/17 23:45:40

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有大坑!
注意仔细参考我的代码!

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.coloradmin.cn/o/1630017.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【神经网络结构可视化】PlotNeuralNet的安装、测试及创建自己的神经网络结构可视化图形

文章目录 前提准备1、下载MikTeX2、下载Git bash3、下载PlotNeuralNet 进行测试1、解压PlotNeuralNet-master.zip2、打开Git bash3、 在my_project中查看生成的pdf文件 创建自己的神经网络结构可视化图形 前提准备 1、下载MikTeX 下载链接&#xff1a; MikTeX ( https://mikt…

RestfulApi RestTemplate代码规范介绍

1.介绍 1.1 RestfulApi Restful API 是一种设计风格&#xff0c;代表了使用 HTTP 协议构建 web 服务的一种架构原则。REST&#xff08;Representational State Transfer&#xff09;的核心思想是&#xff0c;通过 URL 定位资源&#xff0c;使用 HTTP 方法&#xff08;GET, POS…

【多维动态规划】Leetcode 62. 不同路径【中等】

不同路径 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;。 问总共有多少条不同的路径&#xff1…

Vue3+Vite开发的项目进行加密打包

本文主要介绍Vue3+Vite开发的项目如何进行加密打包。 目录 一、vite简介二、混淆工具三、使用方法1. 安装插件:2. 配置插件:3. 运行构建:4. 自定义混淆选项:5. 排除文件:下面是Vue 3+Vite开发的项目进行加密打包的方法。 一、vite简介 Vite 是一个由 Evan You 创造的现代…

【Python从入门到进阶】53、Scrapy日志信息及日志级别

接上篇《52、CrawlSpider链接提取器的使用》 上一篇我们学习了基于规则进行跟踪和自动爬取网页数据的“特殊爬虫”CrawlSpider。本篇我们来学习Scrapy的日志信息及日志级别。 一、引言 1、日志在Scrapy中的重要性 在Scrapy框架中&#xff0c;日志扮演着至关重要的角色。日志…

JAVA面试八股文之JVM

JVM JVM由那些部分组成&#xff0c;运行流程是什么&#xff1f;你能详细说一下 JVM 运行时数据区吗&#xff1f;详细介绍一下程序计数器的作用&#xff1f;你能给我详细的介绍Java堆吗?什么是虚拟机栈&#xff1f;栈内存溢出情况&#xff1f;堆栈的区别是什么吗&#xff1f;解…

2024 年最好的免费数据恢复软件,您可以尝试的几个数据恢复软件

由于系统崩溃而丢失数据可能会给用户带来麻烦。我们将重要的宝贵数据和个人数据保存在我们的 PC、笔记本电脑和其他数字设备上。您可能会因分区丢失、意外删除文件和文件夹、格式化硬盘驱动器而丢失数据。数据丢失是不幸的&#xff0c;如果您不小心从系统中删除了文件或数据&am…

护眼台灯是不是智商税,专业护眼台灯的四大好处需牢记!

在现代社会&#xff0c;电子产品的普及和长时间的屏幕使用已经成为日常生活的一部分&#xff0c;这无疑对我们的视力健康提出了挑战。为了保护眼睛&#xff0c;市场上出现了各种声称能够减轻眼睛疲劳、预防近视等功效的护眼台灯。然而&#xff0c;关于护眼台灯是否真的具有这些…

C#基础:WPF中常见控件的布局基础

一、用ViewBox实现放缩控件不变 二、布局代码 <Window x:Class"WpfApp1.MainWindow"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x"http://schemas.microsoft.com/winfx/2006/xaml"Title"MainWindow"…

传统过程自动化工厂的智能扩展

一 通过NOA概念&#xff0c;公开、安全地迈向未来 随着数字化转型在过程自动化工业中的不断深入&#xff0c;许多公司都面临着同一挑战——如何平衡创新和传统。放眼望去&#xff0c;过程自动化工业和信息技术似乎在以不同的速度发展。虽然过程自动化工厂通过使用传统的自动化…

自动化爬虫工具:you-get安装与使用

Windows下的安装命令&#xff1a; pip install you-get linux下的安装命令&#xff1a; pip3 install you-get 下载完成后&#xff0c;我们可以看到如下的警告&#xff0c;意思就是这个工具并未被添加到环境变量中&#xff0c;如果我们想在命令行中直接调用&#xff0c;需要…

Docker Machine 入门实践@FreeBSD

在这篇文档里&#xff0c;我们安装了docker machine&#xff1a;FreeBSD安装Docker记录-CSDN博客 在这篇文档里&#xff0c;我们启动虚拟主机myvm并进入虚拟主机&#xff1a;boot2docker实践FreeBSD-CSDN博客 现在开始更深入的进行 docker的实战演练。参考Docker Machine官方…

Git--多人协作

目录 一、多人协作一二、多人协作二三、 远程分⽀删除后&#xff0c;本地git branch -a依然能看到的解决办法 一、多人协作一 ⽬前&#xff0c;我们所完成的⼯作如下&#xff1a; 1.基本完成Git的所有本地库的相关操作&#xff0c;git基本操作&#xff0c;分⽀理解&#xff0c;…

Java面试八股文-2024

面试指南 TMD&#xff0c;一个后端为什么要了解那么多的知识&#xff0c;真是服了。啥啥都得了解 MySQL MySQL索引可能在以下几种情况下失效&#xff1a; 不遵循最左匹配原则&#xff1a;在联合索引中&#xff0c;如果没有使用索引的最左前缀&#xff0c;即查询条件中没有包含…

RK3568平台开发系列讲解(Linux系统篇)芯片手册的使用:GPIO的寄存器说明

🚀返回专栏总目录 文章目录 一、查找复用寄存器二、查找方向寄存器三、查找数据寄存器沉淀、分享、成长,让自己和他人都能有所收获!😄 📢寄存器GPIO 进行配置, 一般情况下需要对 GPIO 的复用寄存器, 方向寄存器, 数据寄存器进行配置。 GPIO0_B0 配置为例: 一、查…

BTCOIN发布WEB3.0论坛:生态与金融的双重叙事热点驱动自由创新意识

在数字时代&#xff0c;信息的自由流动和透明度是推动经济发展和社会进步的关键。尤其在加密货币和区块链领域&#xff0c;这一点尤为重要。BTCOIN的最新创举——一个基于WEB3.0理念的信息论坛&#xff0c;不仅标志着信息传递自由化的新篇章&#xff0c;也为数字货币市场的信息…

上位机图像处理和嵌入式模块部署(树莓派4b中的软件调试)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 说到软件调试&#xff0c;有一点大家相信都有体会。那就是编写软件的时候&#xff0c;功能本身还是不算太复杂的。只需要前面流程、算法都设计好了…

STM32学习和实践笔记(20):定时器

1.定时器介绍 STM32F1的定时器一共有8个&#xff0c;由2个基本定时器&#xff08;TIM6、TIM7&#xff09;、4个通用定时器&#xff08;TIM2-TIM5&#xff09;和2个高级定时器&#xff08;TIM1、TIM8&#xff09;组成。 基本定时器的功能最为简单&#xff0c;类似于51单片机内定…

BGP选路原则

路由器对BGP路由的处理过程 l 策略机&#xff1a;操纵路径属性&#xff0c;对属性置值。通过影响路由表的生成&#xff0c;从而影响用户数据的转发 输入策略机影响自己及下游路由器 输出策略机不影响自己&#xff0c;只影响下游路由器 l 路由决策&#xff1a;根据BGP路…

AI赋能分层模式,解构未来,智领风潮

​&#x1f308; 个人主页&#xff1a;danci_ &#x1f525; 系列专栏&#xff1a;《设计模式》 &#x1f4aa;&#x1f3fb; 制定明确可量化的目标&#xff0c;坚持默默的做事。 &#x1f680; 转载自热榜文章&#x1f525;&#xff1a;探索设计模式的魅力&#xff1a;AI赋能分…