1
0
mirror of https://github.com/actix/examples synced 2025-01-22 22:05:57 +01:00

162 lines
5.1 KiB
Rust
Raw Normal View History

use std::{io, net::ToSocketAddrs as _};
2019-12-07 23:59:24 +06:00
2023-01-01 20:22:50 +00:00
use actix_web::{
dev::PeerAddr, error, middleware, web, App, Error, HttpRequest, HttpResponse, HttpServer,
2023-01-01 20:22:50 +00:00
};
use awc::Client;
2022-10-26 16:51:27 +01:00
use clap::Parser;
2023-01-01 20:22:50 +00:00
use futures_util::StreamExt as _;
use tokio::sync::mpsc;
2023-01-01 20:22:50 +00:00
use tokio_stream::wrappers::UnboundedReceiverStream;
use url::Url;
2023-01-01 20:22:50 +00:00
const REQWEST_PREFIX: &str = "/using-reqwest";
/// Forwards the incoming HTTP request using `awc`.
2019-12-07 23:59:24 +06:00
async fn forward(
req: HttpRequest,
payload: web::Payload,
2023-01-01 20:22:50 +00:00
peer_addr: Option<PeerAddr>,
url: web::Data<Url>,
2019-03-26 23:33:13 -07:00
client: web::Data<Client>,
2019-12-07 23:59:24 +06:00
) -> Result<HttpResponse, Error> {
2023-01-01 20:22:50 +00:00
let mut new_url = (**url).clone();
new_url.set_path(req.uri().path());
new_url.set_query(req.uri().query());
let forwarded_req = client
.request_from(new_url.as_str(), req.head())
.no_decompress();
2023-01-01 20:22:50 +00:00
// TODO: This forwarded implementation is incomplete as it only handles the unofficial
// X-Forwarded-For header but not the official Forwarded one.
let forwarded_req = match peer_addr {
Some(PeerAddr(addr)) => {
forwarded_req.insert_header(("x-forwarded-for", addr.ip().to_string()))
}
2022-02-02 02:02:47 +00:00
None => forwarded_req,
};
2022-02-02 02:02:47 +00:00
let res = forwarded_req
.send_stream(payload)
.await
.map_err(error::ErrorInternalServerError)?;
2019-12-07 23:59:24 +06:00
let mut client_resp = HttpResponse::build(res.status());
// Remove `Connection` as per
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Connection#Directives
2022-02-18 02:44:02 +00:00
for (header_name, header_value) in res.headers().iter().filter(|(h, _)| *h != "connection") {
client_resp.insert_header((header_name.clone(), header_value.clone()));
2019-12-07 23:59:24 +06:00
}
Ok(client_resp.streaming(res))
}
2023-01-01 20:22:50 +00:00
/// Same as `forward` but uses `reqwest` as the client used to forward the request.
async fn forward_reqwest(
req: HttpRequest,
mut payload: web::Payload,
method: actix_web::http::Method,
2023-01-01 20:22:50 +00:00
peer_addr: Option<PeerAddr>,
url: web::Data<Url>,
client: web::Data<reqwest::Client>,
) -> Result<HttpResponse, Error> {
let path = req
.uri()
.path()
.strip_prefix(REQWEST_PREFIX)
.unwrap_or(req.uri().path());
let mut new_url = (**url).clone();
new_url.set_path(path);
new_url.set_query(req.uri().query());
let (tx, rx) = mpsc::unbounded_channel();
2023-01-01 20:22:50 +00:00
actix_web::rt::spawn(async move {
while let Some(chunk) = payload.next().await {
tx.send(chunk).unwrap();
}
});
let forwarded_req = client
.request(
reqwest::Method::from_bytes(method.as_str().as_bytes()).unwrap(),
new_url,
)
2023-01-01 20:22:50 +00:00
.body(reqwest::Body::wrap_stream(UnboundedReceiverStream::new(rx)));
// TODO: This forwarded implementation is incomplete as it only handles the unofficial
// X-Forwarded-For header but not the official Forwarded one.
let forwarded_req = match peer_addr {
Some(PeerAddr(addr)) => forwarded_req.header("x-forwarded-for", addr.ip().to_string()),
None => forwarded_req,
};
let res = forwarded_req
.send()
.await
.map_err(error::ErrorInternalServerError)?;
let mut client_resp =
HttpResponse::build(actix_web::http::StatusCode::from_u16(res.status().as_u16()).unwrap());
2023-01-01 20:22:50 +00:00
// Remove `Connection` as per
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Connection#Directives
for (header_name, header_value) in res.headers().iter().filter(|(h, _)| *h != "connection") {
client_resp.insert_header((
actix_web::http::header::HeaderName::from_bytes(header_name.as_ref()).unwrap(),
actix_web::http::header::HeaderValue::from_bytes(header_value.as_ref()).unwrap(),
));
2023-01-01 20:22:50 +00:00
}
Ok(client_resp.streaming(res.bytes_stream()))
}
#[derive(clap::Parser, Debug)]
struct CliArguments {
listen_addr: String,
listen_port: u16,
forward_addr: String,
forward_port: u16,
}
2020-09-12 16:49:45 +01:00
#[actix_web::main]
async fn main() -> io::Result<()> {
2022-02-02 02:02:47 +00:00
env_logger::init_from_env(env_logger::Env::new().default_filter_or("info"));
let args = CliArguments::parse();
2022-02-02 02:02:47 +00:00
let forward_socket_addr = (args.forward_addr, args.forward_port)
.to_socket_addrs()?
.next()
.expect("given forwarding address was not valid");
let forward_url = format!("http://{forward_socket_addr}");
2022-02-02 02:02:47 +00:00
let forward_url = Url::parse(&forward_url).unwrap();
log::info!(
2022-02-06 08:19:35 +00:00
"starting HTTP server at http://{}:{}",
2022-02-02 02:02:47 +00:00
&args.listen_addr,
args.listen_port
);
log::info!("forwarding to {forward_url}");
2023-01-01 20:22:50 +00:00
let reqwest_client = reqwest::Client::default();
HttpServer::new(move || {
App::new()
2022-02-02 02:02:47 +00:00
.app_data(web::Data::new(Client::default()))
2023-01-01 20:22:50 +00:00
.app_data(web::Data::new(reqwest_client.clone()))
.app_data(web::Data::new(forward_url.clone()))
2019-03-26 23:33:13 -07:00
.wrap(middleware::Logger::default())
2023-01-01 20:22:50 +00:00
.service(web::scope(REQWEST_PREFIX).default_service(web::to(forward_reqwest)))
2022-02-02 02:02:47 +00:00
.default_service(web::to(forward))
2019-03-09 18:03:09 -08:00
})
.bind((args.listen_addr, args.listen_port))?
2022-02-02 02:02:47 +00:00
.workers(2)
2019-12-25 20:48:33 +04:00
.run()
2019-12-07 23:59:24 +06:00
.await
}