rustfuturerust-actix

Append data to proxied response in Actix-web


I would like to append user's request data to peroxided response in actix-web, but type mismatch resolving error will happend.

I think it's about rust futures but I don't have idea about this problem and how can I fix it.

Sample Code :

use actix_web::*;
use futures::future::ok;
use futures::Future;

fn show_request(
    request: &actix_web::HttpRequest,
) -> Box<Future<Item = HttpResponse, Error = Error>> {
    let mut result: Vec<u8> = Vec::new();
    Box::new(request.body().map_err(|e| e.into()).map(move |f| {
        result.extend(f); 
        client::ClientRequest::get("http://example.com/")
            .finish().unwrap()
            .send()
            .map_err(Error::from)
            .and_then(
                |resp| resp.body() 
                    .from_err()  
                    .and_then(|body| { 
                        result.extend(body); // append request body to response proxied site
                        Ok(HttpResponse::Ok().body(result))
                    }))


    }))
}

pub fn index(scope: actix_web::Scope<()>) -> actix_web::Scope<()> {
    scope.handler("", |req: &actix_web::HttpRequest| {
        show_request(req)
    })
}

fn main() {
    actix_web::server::new(|| {
        vec![
            actix_web::App::new()
                .scope("", index)
                .boxed(),

        ]
    }).bind("127.0.0.1:8000")
        .expect("Can not bind to port 8000")
        .run();
}

Cargo.toml

[package]
name = "temp"
version = "0.1.0"
authors = ["John"]
edition = "2018"

[dependencies]
actix-web = "0.7"
futures = "0.1"

Shown error :

error[E0271]: type mismatch resolving `<[closure@src/main.rs:9:55: 24:6 result:_] as std::ops::FnOnce<(bytes::bytes::Bytes,)>>::Output == actix_web::httpresponse::HttpResponse`
  --> src/main.rs:9:5
   |
9  | /     Box::new(request.body().map_err(|e| e.into()).map(move |f| {
10 | |         result.extend(f);
11 | |         client::ClientRequest::get("http://example.com/")
12 | |             .finish().unwrap()
...  |
23 | |
24 | |     }))
   | |_______^ expected struct `futures::future::and_then::AndThen`, found struct `actix_web::httpresponse::HttpResponse`
   |
   = note: expected type `futures::future::and_then::AndThen<futures::future::map_err::MapErr<actix_web::client::pipeline::SendRequest, fn(actix_web::client::pipeline::SendRequestError) -> actix_web::error::Error {<actix_web::error::Error as std::convert::From<actix_web::client::pipeline::SendRequestError>>::from}>, futures::future::and_then::AndThen<futures::future::from_err::FromErr<actix_web::httpmessage::MessageBody<actix_web::client::response::ClientResponse>, actix_web::error::Error>, std::result::Result<actix_web::httpresponse::HttpResponse, actix_web::error::Error>, [closure@src/main.rs:18:19: 21:10 result:_]>, [closure@src/main.rs:16:8: 21:11 result:_]>`
              found type `actix_web::httpresponse::HttpResponse`
   = note: required because of the requirements on the impl of `futures::future::Future` for `futures::future::map::Map<futures::future::map_err::MapErr<actix_web::httpmessage::MessageBody<actix_web::httprequest::HttpRequest>, [closure@src/main.rs:9:37: 9:49]>, [closure@src/main.rs:9:55: 24:6 result:_]>`
   = note: required for the cast to the object type `dyn futures::future::Future<Error=actix_web::error::Error, Item=actix_web::httpresponse::HttpResponse>

Solution

  • You can solve your problem by using futures combinators.

    for example :

    Box::new(request.body().map_err(|e| e.into()).and_then(|mut x|{
        client::ClientRequest::get("http://example.com/")
            .finish()
            .unwrap()
            .send()
            .map_err(Error::from)
            .and_then(|resp| {
                resp.body()
                    .from_err()
                    .and_then(|body| {
                        x.extend(body);
                        Ok(HttpResponse::Ok().body(x))
                    })
            })
    }))