11个版本 (6个重大更改)

0.7.0 2024年1月17日
0.6.1 2022年7月31日
0.6.0 2022年3月22日
0.5.0 2021年1月22日
0.1.0 2018年5月31日

#42 in HTTP客户端

Download history 649/week @ 2024-04-20 765/week @ 2024-04-27 744/week @ 2024-05-04 776/week @ 2024-05-11 886/week @ 2024-05-18 903/week @ 2024-05-25 719/week @ 2024-06-01 1213/week @ 2024-06-08 950/week @ 2024-06-15 1368/week @ 2024-06-22 1424/week @ 2024-06-29 1215/week @ 2024-07-06 1059/week @ 2024-07-13 1023/week @ 2024-07-20 1563/week @ 2024-07-27 1205/week @ 2024-08-03

5,106 每月下载量
4 个库中使用了(3个直接使用)

MIT/Apache

67KB
1K SLoC

Rust Multipart Async

此库允许创建用于与std futures一起使用的客户端/服务器多部分流。

快速使用

使用客户端,您需要创建一个 MultipartRequest 并添加您的字段和文件。

Hyper客户端示例

这是一个如何使用客户端与hyper一起使用的示例(使用cargo run --example hyper运行)

use hyper::{header::CONTENT_TYPE, Body, Client, Request};
use hyper::{service::make_service_fn, service::service_fn, Response, Server};
use mpart_async::client::MultipartRequest;

type Error = Box<dyn std::error::Error + Send + Sync + 'static>;

#[tokio::main]
async fn main() -> Result<(), Error> {
    //Setup a mock server to accept connections.
    setup_server();

    let client = Client::new();

    let mut mpart = MultipartRequest::default();

    mpart.add_field("foo", "bar");
    mpart.add_file("test", "Cargo.toml");

    let request = Request::post("https://127.0.0.1:3000")
        .header(
            CONTENT_TYPE,
            format!("multipart/form-data; boundary={}", mpart.get_boundary()),
        )
        .body(Body::wrap_stream(mpart))?;

    client.request(request).await?;

    Ok(())
}

fn setup_server() {
    let addr = ([127, 0, 0, 1], 3000).into();
    let make_svc = make_service_fn(|_conn| async { Ok::<_, Error>(service_fn(mock)) });
    let server = Server::bind(&addr).serve(make_svc);

    tokio::spawn(server);
}

async fn mock(_: Request<Body>) -> Result<Response<Body>, Error> {
    Ok(Response::new(Body::from("")))
}

Warp服务器示例

这是一个如何使用它与warp服务器一起使用的示例(使用cargo run --example warp运行)

use warp::Filter;

use bytes::Buf;
use futures::stream::TryStreamExt;
use futures::Stream;
use mime::Mime;
use mpart_async::server::MultipartStream;
use std::convert::Infallible;

#[tokio::main]
async fn main() {
    // Match any request and return hello world!
    let routes = warp::any()
        .and(warp::header::<Mime>("content-type"))
        .and(warp::body::stream())
        .and_then(mpart);

    warp::serve(routes).run(([127, 0, 0, 1], 3030)).await;
}

async fn mpart(
    mime: Mime,
    body: impl Stream<Item = Result<impl Buf, warp::Error>> + Unpin,
) -> Result<impl warp::Reply, Infallible> {
    let boundary = mime.get_param("boundary").map(|v| v.to_string()).unwrap();

    let mut stream = MultipartStream::new(
        boundary,
        body.map_ok(|mut buf| buf.copy_to_bytes(buf.remaining())),
    );

    while let Ok(Some(mut field)) = stream.try_next().await {
        println!("Field received:{}", field.name().unwrap());
        if let Ok(filename) = field.filename() {
            println!("Field filename:{}", filename);
        }

        while let Ok(Some(bytes)) = field.try_next().await {
            println!("Bytes received:{}", bytes.len());
        }
    }

    Ok(format!("Thanks!\n"))
}

与服务交互

curl -F test=field -F [email protected] https://127.0.0.1:3030/

依赖项

~1.9–4MB
~75K SLoC