As of wasmer 4.1 (opens in a new tab), epoll syscall and
TLS clients are now supported in WASIX. This was done by compiling ring.
WASIX with Reqwest
This is a sample project that shows how to use a reqwest client to build an outbound proxy and compile it to WASIX.
Prerequisites
Please check that you have the latest version of wasmer runtime as this tutorial depends on version 4.1.1 or higher.
The project requires the following tools to be installed on your system:
Start a new project
$ cargo new --bin wasix-reqwest-proxy
Created binary (application) `wasix-reqwest-proxy` packageYour wasix-reqwest-proxy directory structure should look like this:
Add dependencies
We will use pinned dependencies from wasix-org to make sure that our project compiles with wasix.
$ cd wasix-reqwest-proxy
$ cargo add tokio --git https://github.com/wasix-org/tokio.git --branch wasix-1.24.2 --features rt-multi-thread,macros,fs,io-util,net,signal
$ cargo add reqwest --git https://github.com/wasix-org/reqwest.git --features json,rustls-tls
$ cargo add hyper --git https://github.com/wasix-org/hyper.git --branch v0.14.27 --features server
$ cargo add anyhowWe also need to add some patch crates to our Cargo.toml so that all other dependencies compile with wasix:
[package]
name = "wasix-reqwest-proxy"
version = "0.1.0"
edition = "2021"
[dependencies]
tokio = { git = "https://github.com/wasix-org/tokio.git", branch = "epoll", default-features = false, features = [
"rt-multi-thread",
"macros",
"fs",
"io-util",
"net",
"signal",
] }
reqwest = { git = "https://github.com/wasix-org/reqwest.git", default-features = false, features = [
"json",
"rustls-tls",
] }
anyhow = { version = "1.0.71" }
hyper = { git = "https://github.com/wasix-org/hyper.git", branch = "v0.14.27", features = [
"server",
] }
[patch.crates-io] # 👈🏼 Added section here
socket2 = { git = "https://github.com/wasix-org/socket2.git", branch = "v0.4.9" } # 👈🏼 Added line here
libc = { git = "https://github.com/wasix-org/libc.git" } # 👈🏼 Added line here
tokio = { git = "https://github.com/wasix-org/tokio.git", branch = "epoll" } # 👈🏼 Added line here
rustls = { git = "https://github.com/wasix-org/rustls.git", branch = "v0.21.5" } # 👈🏼 Added line here
hyper = { git = "https://github.com/wasix-org/hyper.git", branch = "v0.14.27" } # 👈🏼 Added line here
For making certain features such as networking, sockets, threading, etc. work with wasix we need patch dependencies for some crates that use those features.
Writing the Application
Our outbound proxy application will have two parts:
- Listen for incoming requests using the
hyperserver - Forward the request to the destination using the
reqwestclient and return the response to the client
Part 1. - Listening for incoming requests
Let's setup a basic hyper server that listens on port 3000. This code goes inside our main function.
let addr = SocketAddr::from(([127, 0, 0, 1], 3000));
println!("Listening on {}", addr);
// And a MakeService to handle each connection...
let make_service = make_service_fn(|_conn| async { Ok::<_, Infallible>(service_fn(handle)) });
// ^^^^^^
// 🔦 Focus here - This handle function is what connects to the part 2 of our application.
// Then bind and serve...
let server = Server::bind(&addr).serve(make_service);
// And run forever...
Ok(server.await?)Part 2. - Forwarding the request to the destination
Now let's write the handle function that will be called for each incoming request. This function will use the reqwest client to forward the request to the destination and return the response to the client.
async fn handle(req: Request<Body>) -> Result<Response<Body>, Infallible> {
// Create the destination URL
let path = format!(
"https://www.rust-lang.org/{}",
req.uri()
.path_and_query()
.map(|p| p.as_str())
.unwrap_or(req.uri().path())
); // ← 1.
let mut status = StatusCode::OK;
let body = match async { reqwest::get(path).await?.text().await }.await {
Ok(b) => b,
Err(err) => {
status = err.status().unwrap_or(StatusCode::BAD_REQUEST);
format!("{err}")
}
}; // ← 2.
let body = String::from_utf8_lossy(body.as_bytes()).to_string(); // ← 3.
let mut res = Response::new(Body::from(body)); // ← 4.
*res.status_mut() = status; // ← 5.
Ok(res) // ← 6.
}Let's go through the code above:
- Create a
pathvariable that contains the destination URL. We use thereq.uri()to get the request URI and then append it to the destination URL. - Use the
reqwestclient to make a request to the destination URL. We use theawaitkeyword to wait for the response. If the request fails, we set thestatusvariable toBAD_REQUESTand return the error message as the response body. - Convert the response body to a
StringusingString::from_utf8_lossy. This is required because theBodytype requires the body to beSendandSyncandStringimplements both of these traits. - Create a new
Responsewith the body we received from the destination. - Set the status of the response to the
statusvariable we set earlier. - Finally, Return the response.
Your src/main.rs should now look like this:
use hyper::service::{make_service_fn, service_fn};
use hyper::{Body, Request, Response, Server, StatusCode};
use std::convert::Infallible;
use std::net::SocketAddr;
async fn handle(req: Request<Body>) -> Result<Response<Body>, Infallible> {
let path = format!(
"https://www.rust-lang.org/{}",
req.uri()
.path_and_query()
.map(|p| p.as_str())
.unwrap_or(req.uri().path())
);
let mut status = StatusCode::OK;
let body = match async { reqwest::get(path).await?.text().await }.await {
Ok(b) => b,
Err(err) => {
status = err.status().unwrap_or(StatusCode::BAD_REQUEST);
format!("{err}")
}
};
let body = String::from_utf8_lossy(body.as_bytes()).to_string();
let mut res = Response::new(Body::from(body));
*res.status_mut() = status;
Ok(res)
}
#[tokio::main]
async fn main() -> anyhow::Result<()> {
let addr = SocketAddr::from(([127, 0, 0, 1], 3000));
println!("Listening on {}", addr);
// And a MakeService to handle each connection...
let make_service = make_service_fn(|_conn| async { Ok::<_, Infallible>(service_fn(handle)) });
// Then bind and serve...
let server = Server::bind(&addr).serve(make_service);
// And run forever...
Ok(server.await?)
}Running the Application
Let's compile the application to WASIX and run it:
Compiling to WASIX
$ cargo wasix build
Compiling autocfg v1.1.0
Compiling wasi v0.11.0+wasi-snapshot-preview1
Compiling proc-macro2 v1.0.66
Compiling cfg-if v1.0.0
Compiling unicode-ident v1.0.11
Compiling libc v0.2.139 (https://github.com/wasix-org/libc.git#4c0c6c29)
Compiling wasix-reqwest-proxy v0.1.0 (/wasix-reqwest-proxy)
...It could happen that the above command might fail for you, this is because of
dependencies not resolving correctly. You can easily fix this by running
cargo update and then running cargo wasix build again.
Yay, it builds! Now, let's try to run it:
Running the Application with Wasmer
$ wasmer run target/wasm32-wasmer-wasi/debug/wasix-reqwest-proxy.wasm
Currently, we need to run it using wasmer run. See the
issue (opens in a new tab)
Let's try to run it with wasmer:
$ wasmer run target/wasm32-wasmer-wasi/debug/wasix-reqwest-proxy.wasm --net --enable-threadsLet's go through the flags we used above:
--net- This flag enables networking support for wasm files.--enable-threads- This flag enables threading support for wasm files.
Now in a separate terminal, you can use curl to make a request to the server:
$ curl localhost:3000
<!doctype html>
<html lang="en-US">
<head>
<meta charset="utf-8">
<title>
Rust Programming Language
...Congratulations! You have successfully built an outbound proxy server using hyper, reqwest and wasix.
You can also deploy you application to the edge. Checkout this tutorial for deploying your wasix-reqwest-proxy server to wasmer edge.
Exercises
Exercise 1
Try to take the destination URL as a query parameter.
$ curl localhost:3000?url=https://www.rust-lang.orgExercise 2
Try to take the destination URL as a parameter for the .wasm file.
$ wasmer run target/wasm32-wasmer-wasi/debug/wasix-reqwest-proxy.wasm --net --enable-threads -- --url=https://www.rust-lang.orgYou can use the -- to pass arguments to the .wasm file and use the rust's
default std::env::args to parse the arguments. Learn more about
wasmer-cli.
Conclusion
In this tutorial, we learned:
- How to build a simple outbound proxy server using
hyperandreqwest. - How to patch dependencies add WASIX support.
- How to run wasix based
.wasmfiles with Wasmer. - How to enable networking and threading support for wasm files.