跳到主要内容

Client

The wasmedge_wasi_socket crate enables Rust developers to create networking applications and compile them into WebAssembly for WasmEdge Runtime. One of the key features of WasmEdge is that it supports non-blocking sockets. That allows even a single-threaded WASM application to handle concurrent network requests. For example, while the program is waiting for data to stream in from one connection, it can start or handle another connection.

While there are many possibilities with sockets, we will demonstrate two familiar use cases, a simple HTTP client and a non-blocking HTTP client application, in this chapter.

备注

Before we start, ensure you have Rust and WasmEdge installed.

A Simple HTTP Client

You can build and run the example in WasmEdge as follows.

git clone https://github.com/second-state/wasmedge_wasi_socket.git
cd wasmedge_wasi_socket/http_client/

# Build the Rust Code
cargo build --target wasm32-wasi --release
# Use the AoT compiler to get better performance
wasmedgec target/wasm32-wasi/release/http_client.wasm http_client.wasm

# Run the example
wasmedge http_client.wasm

The source code for the HTTP client is available as follows.

use wasmedge_http_req::request;

fn main() {
let mut writer = Vec::new(); //container for body of a response
let res = request::get("http://127.0.0.1:1234/get", &mut writer).unwrap();

println!("GET");
println!("Status: {} {}", res.status_code(), res.reason());
println!("Headers {}", res.headers());
println!("{}", String::from_utf8_lossy(&writer));

let mut writer = Vec::new(); //container for body of a response
const BODY: &[u8; 27] = b"field1=value1&field2=value2";
// let res = request::post("https://httpbin.org/post", BODY, &mut writer).unwrap();
// no https , no dns
let res = request::post("http://127.0.0.1:1234/post", BODY, &mut writer).unwrap();

println!("POST");
println!("Status: {} {}", res.status_code(), res.reason());
println!("Headers {}", res.headers());
println!("{}", String::from_utf8_lossy(&writer));
}

A non-blocking HTTP client example

You can build and run the example in WasmEdge as follows.

git clone https://github.com/second-state/wasmedge_wasi_socket
cd wasmedge_wasi_socket/nonblock_http_client/

# Build the Rust Code
cargo build --target wasm32-wasi --release
# Use the AoT compiler for better performance
wasmedgec target/wasm32-wasi/release/nonblock_http_client.wasm nonblock_http_client.wasm

# Run the example
wasmedge nonblock_http_client.wasm
备注

Non-blocking I/O means that the application program can keep multiple connections open simultaneously, and process data in and out of those connections as they come in. The program can either alternatingly poll those open connections or wait for incoming data to trigger async functions. That allows I/O intensive programs to run faster, even in a single-threaded environment.

The source code for a non-blocking HTTP client application is available. The following main() function starts two HTTP connections. The program keeps both connections open and alternatingly checks for incoming data. In other words, the two connections are not blocking each other. Their data are handled concurrently (or alternatingly) as the data comes in.

use httparse::{Response, EMPTY_HEADER};
use std::io::{self, Read, Write};
use std::str::from_utf8;
use wasmedge_wasi_socket::TcpStream;

fn main() {
let req = "GET / HTTP/1.0\n\n";
let mut first_connection = TcpStream::connect("127.0.0.1:80").unwrap();
first_connection.set_nonblocking(true).unwrap();
first_connection.write_all(req.as_bytes()).unwrap();

let mut second_connection = TcpStream::connect("127.0.0.1:80").unwrap();
second_connection.set_nonblocking(true).unwrap();
second_connection.write_all(req.as_bytes()).unwrap();

let mut first_buf = vec![0; 4096];
let mut first_bytes_read = 0;
let mut second_buf = vec![0; 4096];
let mut second_bytes_read = 0;

loop {
let mut first_complete = false;
let mut second_complete = false;
if !first_complete {
match read_data(&mut first_connection, &mut first_buf, first_bytes_read) {
Ok((bytes_read, false)) => {
first_bytes_read = bytes_read;
}
Ok((bytes_read, true)) => {
println!("First connection completed");
if bytes_read != 0 {
parse_data(&first_buf, bytes_read);
}
first_complete = true;
}
Err(e) => {
println!("First connection error: {}", e);
first_complete = true;
}
}
}
if !second_complete {
match read_data(&mut second_connection, &mut second_buf, second_bytes_read) {
Ok((bytes_read, false)) => {
second_bytes_read = bytes_read;
}
Ok((bytes_read, true)) => {
println!("Second connection completed");
if bytes_read != 0 {
parse_data(&second_buf, bytes_read);
}
second_complete = true;
}
Err(e) => {
println!("Second connection error: {}", e);
second_complete = true;
}
}
}
if first_complete && second_complete {
break;
}
}
}