使用 actix_web 对部分结构进行 Serde 反序列化

Sam*_*ena 1 json rust serde rust-actix actix-web

我有一个 API 端点,actix_web用于反序列化传入的 JSON 有效负载(actix_web最终serde用于 JSON 反序列化)。

例如,我有一些看起来像这样的东西:

pub struct IncomingPayload {
    pub field1: i32,
    pub field2: String
}

pub async fn update_platforms(
    pool: web::Data<Pool>,
    req: web::Json<Vec<IncomingPayload>>,
) -> Result<HttpResponse, error::Error> { 
    println!(req.field1); // will be an i32
    println!(req.field2); // will be a String
}
Run Code Online (Sandbox Code Playgroud)

目前,只有serde能够反序列化结构的所有字段,此端点才会成功返回。即一个请求必须包含field1field2键。

例如,这将是成功的:

curl --header "Content-Type: application/json" \
  --request POST \
  --data '{"field1": 1,"field2":"something"}' \
  http://localhost:8080
Run Code Online (Sandbox Code Playgroud)

但这不会(因为field2有效载荷中缺少):

curl --header "Content-Type: application/json" \
  --request POST \
  --data '{"field1": 1}' \
  http://localhost:8080
Run Code Online (Sandbox Code Playgroud)

所以我的问题是,这能做到吗?是否可以将包含键值字段子集的 JSON 有效负载发送到actix_web期望所有字段都存在的端点?

或者更确切地说,是否有一种通用的方法可以使用actix_webor反序列化这样的部分结构serde

Frx*_*rem 6

您可以将其Option<T>用作字段的类型以使其可选。如果在反序列化过程中缺少某个字段,则该字段设置为None,否则设置为Some(value)

#[derive(Deserialize)]
struct IncomingPayload {
    pub field1: Option<i32>,
    pub field2: Option<String>,
}
Run Code Online (Sandbox Code Playgroud)

对于实现 的类型,如果字段缺失Default,您还可以使用#[serde(default)]将字段设置为默认值。

#[derive(Deserialize)]
struct IncomingPayload {
    #[serde(default)] // default = 0
    pub field1: i32,
    #[serde(default)] // default = empty string
    pub field2: String,
}
Run Code Online (Sandbox Code Playgroud)