Request

在 Salvo 中可以通過 Request 獲取用戶請求的數據:

#[handler]
async fn hello(req: &mut Request) -> String {
    req.params().get("id").cloned().unwrap_or_default()
}

獲取查詢參數

可以通過 get_query 獲取查詢參數:

req.query::<String>("id");

獲取 Form 數據

可以通過 get_form 獲取查詢參數, 此函數為異步函數:

req.form::<String>("id").await;

獲取 JSON 反序列化數據

req.parse_json::<User>().await;

提取 Request 數據

Request 提供多個方法將這些數據解析為強類型結構.

  • parse_params: 將請求的 router params 解析為特定的數據類型;
  • parse_queries: 將請求的 URL queries 解析為特定的數據類型;
  • parse_headers: 將請求的 HTTP headers 解析為特定的數據類型;
  • parse_json: 將請求的 HTTP body 部分的數據當作 JSON 格式解析到特定的類型;
  • parse_form: 將請求的 HTTP body 部分的數據當作 Form 表單解析到特定的類型;
  • parse_body: 根據請求的 content-type 的類型, 將 HTTP body 部分的數據解析為特定類型.
  • extract: 可以合並不同的數據源解析出特定的類型.

解析原理

此處通過自定義的 serde::Deserializer 將類似 HashMap<String, String>HashMap<String, Vec<String>> 的數據提取為特定的數據類型.

比如: URL queries 實際上被提取為一個 MultiMap在新窗口打開 類型, MultiMap 可以認為就是一個類似 HashMap<String, Vec<String>> 的數據結構. 如果請求的 URL 是 http://localhost/users?id=123&id=234, 我們提供的目標類型是:

#[derive(Deserialize)]
struct User {
  id: i64
}

則第一個 id=123 會被解析, id=234 則被丟棄:

let user: User = req.parse_queries().unwrap();
assert_eq!(user.id, 123);

如果我們提供的類型是:

#[derive(Deserialize)]
struct Users {
  id: Vec<i64>
}

id=123&id=234 都會被解析:

let users: Users = req.parse_queries().unwrap();
assert_eq!(user.ids, vec![123, 234]);

可以合並多個數據源, 解析出特定類型, 可以先定義一個自定義的類型, 比如:

#[derive(Serialize, Deserialize, Extractible, Debug)]
/// 默認從 body 中獲取數據字段值
#[extract(default_source(from = "body"))]
struct GoodMan<'a> {
    /// 其中, id 號從請求路徑參數中獲取, 並且自動解析數據為 i64 類型.
    #[extract(source(from = "param"))]
    id: i64,
    /// 可以使用引用類型, 避免內存復制.
    username: &'a str,
    first_name: String,
    last_name: String,
}

然後在 Handler 中可以這樣獲取數據:

#[handler]
async fn edit(req: &mut Request) -> String {
    let good_man: GoodMan<'_> = req.extract().await.unwrap();
}

甚至於可以直接把類型作為參數傳入函數, 像這樣:

#[handler]
async fn edit<'a>(good_man: GoodMan<'a>) -> String {
    res.render(Json(good_man));
}

數據類型的定義有相當大的靈活性, 甚至可以根據需要解析為嵌套的結構:

#[derive(Serialize, Deserialize, Extractible, Debug)]
#[extract(default_source(from = "body", format = "json"))]
struct GoodMan<'a> {
    #[extract(source(from = "param"))]
    id: i64,
    #[extract(source(from = "query"))]
    username: &'a str,
    first_name: String,
    last_name: String,
    lovers: Vec<String>,
    /// 這個 nested 字段完全是從 Request 重新解析.
    #[extract(source(from = "request"))]
    nested: Nested<'a>,
}

#[derive(Serialize, Deserialize, Extractible, Debug)]
#[extract(default_source(from = "body", format = "json"))]
struct Nested<'a> {
    #[extract(source(from = "param"))]
    id: i64,
    #[extract(source(from = "query"))]
    username: &'a str,
    first_name: String,
    last_name: String,
    #[extract(rename = "lovers")]
    #[serde(default)]
    pets: Vec<String>,
}

具體實例參見: extract-nested在新窗口打開.