一种基于宏和serde_json实现的rust web中统一返回类

本人rust萌新,写web碰到了这个,基于ChatGPT和文心一言学了宏,强行把这玩意实现出来了,做个学习记录,如果有更好的方法,勿喷。

先看效果,注意不支持嵌套,且kv映射要用=>(因为它这个只支持用箭头),即在这个宏语法内只支持单层kv,但是你可以传value为HashMap。

对此我曾尝试用#[proc_macro]TokenStream拿到变量名和值的方式实现正常使用花括号和冒号和深层字典,但是太菜了搞不会。

如果能基于我的想法实现出深层kv,欢迎评论区分享。

rust 复制代码
//第一个变量是msg,后面的全部存入data{key1:...,key2:...}
success!(
   "success",
    "key1"=> vec![1,2,3,4,5],
    "key2" =>"qwer",
    "key3"=>String::from("value")
);
rust 复制代码
//第一个变量没了就是默认空msg,后面的全部存入data{key1:...,key2:...}
success!(
 	"key"=>"qwer",
    "key2"=>1324,
);


fail!同理

rust中实现统一返回类有点麻烦,比如我想返回固定格式

json 复制代码
{
	"success":true,
	"msg":"登录成功",
	"data":{
		"id":15,
		"name":"qwer"
	}
}

我目前用的是actix-web,虽然支持直接传入结构体对象作为json返回值,但我有时候想对于data进行灵活的处理,不想建那么多结构体

这里我基于serde_jsonjson!进行进一步封装

这里一个宏里面写两个形式,因为要匹配传入msg和不传入msg的两种情形。最后一个$(,)?是允许最后一个多余的逗号,换行时好看一点

rust 复制代码
// 随便找个文件放
// 用了#[macro_export]的宏会直接放在crate下

//下面三个库需要在使用下面宏的地方进行use,此处use没用
// use actix_web::web;
// use serde_json::json;
// use std::collections::HashMap;

#[macro_export]
macro_rules! success {
    // 没有msg的情况
    ($($key:expr => $value:expr),* $(,)?) => {
        {
            let mut resp = HashMap::new();
            resp.insert("success", json!(true));
            resp.insert("msg", json!("")); // 提供一个默认的消息
            let mut data:HashMap<String,Value> = HashMap::new();
            $(
                data.insert($key, json!($value));
            )*
            resp.insert("data",json!(data));
            web::Json(json!(resp))
        }
    };
    // 有msg的情况
    ($msg:expr,  $($key:expr => $value:expr),* $(,)?) => {
       {
            let mut resp = HashMap::new();
            resp.insert("success", json!(true));
            resp.insert("msg", json!($msg)); // 提供一个默认的消息
            let mut data:HashMap<String,Value> = HashMap::new();
            $(
                data.insert($key, json!($value));
            )*
            resp.insert("data",json!(data));
            web::Json(json!(resp))
        }
    };
}

#[macro_export]
macro_rules! fail {
    // 没有$msg的情况
    ($($key:expr => $value:expr),* $(,)?) => {
        {
            let mut resp = HashMap::new();
            resp.insert("success", json!(false));
            resp.insert("msg", json!(""));
            let mut data:HashMap<String,Value> = HashMap::new();
            $(
                data.insert($key, json!($value));
            )*
            resp.insert("data",data);
            web::Json(json!(resp))
        }
    };
    // 有$msg的情况
    ($msg:expr,  $($key:expr => $value:expr),* $(,)?) => {
       {
            let mut resp = HashMap::new();
            resp.insert("success", json!(false));
            resp.insert("msg", json!($msg));
            let mut data:HashMap<String,Value> = HashMap::new();
            $(
                data.insert($key, json!($value));
            )*
            resp.insert("data",json!(data));
            web::Json(json!(resp))
        }
    };
}

注意:过程宏需要定义在一个单独的crate中,主要是因为过程宏是一段在编译crate前,对其代码进行加工的代码,而这段是需要在编译后执行的。若是将定义过程宏和使用过程宏放到同一个crate中,就会陷入编译"死锁":

另一个文件对其进行调用

rust 复制代码
// 注意是在crate下面的
use crate::success;
use actix_web::{
    get,web::{self, Json},
};
use serde_json::{json, Value};
use std::collections::HashMap;

#[get("/login")]
async fn login() -> Json<Value> {
    let resp = success!(
        "success",
        "key1"=> vec![1,2,3,4,5],
        "key2" =>"qwer",
        "key3"=>String::from("value")
    );
    resp
}

我理解的这个宏就是字符串替换,因为他是编译阶段处理的,所以宏所在rs文件的use是无效的,即使用了宏的文件需要对于宏使用的第三方库进行use,这个有点恶心,问题不大。

相关推荐
@一枝梅9 分钟前
vue3 vite.config.js 引入bem.scss文件报错
javascript·rust·vue·scss
DongLi013 小时前
Rust基础学习 - 第1天:语法与变量
rust
维维酱8 小时前
Rust - 读写锁 (RwLock)
rust
Source.Liu8 小时前
【PhysUnits】15.17 比例因子模块 (ratio.rs)
rust
UestcXiye9 小时前
Rust 学习笔记:关于智能指针的练习题
rust
维维酱1 天前
Rust - 互斥锁
rust
维维酱1 天前
Rust - 共享状态的并发
rust
ArcX1 天前
从 JS 到 Rust 的旅程
前端·javascript·rust
Humbunklung1 天前
Rust Floem UI 框架使用简介
开发语言·ui·rust
寻月隐君1 天前
深入解析 Rust 的面向对象编程:特性、实现与设计模式
后端·rust·github