mirror of
https://github.com/tokio-rs/axum.git
synced 2025-01-05 02:22:03 +01:00
Add initial benchmarks (#1243)
* Add initial benchmarks * improve error message if rewrk isn't installed * what env vars does github ci have? * run benchmarks for a bit on ci * debug error * ignore args on ci * actually also locally * apply review suggestions
This commit is contained in:
parent
7faf059234
commit
9e50bf16fb
3 changed files with 217 additions and 0 deletions
|
@ -102,3 +102,7 @@ features = [
|
||||||
"multipart",
|
"multipart",
|
||||||
"ws",
|
"ws",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[bench]]
|
||||||
|
name = "benches"
|
||||||
|
harness = false
|
||||||
|
|
212
axum/benches/benches.rs
Normal file
212
axum/benches/benches.rs
Normal file
|
@ -0,0 +1,212 @@
|
||||||
|
use axum::{
|
||||||
|
routing::{get, post},
|
||||||
|
Json, Router, Server,
|
||||||
|
};
|
||||||
|
use hyper::server::conn::AddrIncoming;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
use std::{
|
||||||
|
io::BufRead,
|
||||||
|
process::{Command, Stdio},
|
||||||
|
};
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
if on_ci() {
|
||||||
|
install_rewrk();
|
||||||
|
} else {
|
||||||
|
ensure_rewrk_is_installed();
|
||||||
|
}
|
||||||
|
|
||||||
|
benchmark("minimal").run(Router::new);
|
||||||
|
|
||||||
|
benchmark("basic").run(|| Router::new().route("/", get(|| async { "Hello, World!" })));
|
||||||
|
|
||||||
|
benchmark("routing").path("/foo/bar/baz").run(|| {
|
||||||
|
let mut app = Router::new();
|
||||||
|
for a in 0..10 {
|
||||||
|
for b in 0..10 {
|
||||||
|
for c in 0..10 {
|
||||||
|
app = app.route(&format!("/foo-{}/bar-{}/baz-{}", a, b, c), get(|| async {}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
app.route("/foo/bar/baz", get(|| async {}))
|
||||||
|
});
|
||||||
|
|
||||||
|
benchmark("receive-json")
|
||||||
|
.method("post")
|
||||||
|
.headers(&[("content-type", "application/json")])
|
||||||
|
.body(r#"{"n": 123, "s": "hi there", "b": false}"#)
|
||||||
|
.run(|| Router::new().route("/", post(|_: Json<Payload>| async {})));
|
||||||
|
|
||||||
|
benchmark("send-json").run(|| {
|
||||||
|
Router::new().route(
|
||||||
|
"/",
|
||||||
|
get(|| async {
|
||||||
|
Json(Payload {
|
||||||
|
n: 123,
|
||||||
|
s: "hi there".to_owned(),
|
||||||
|
b: false,
|
||||||
|
})
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Serialize)]
|
||||||
|
struct Payload {
|
||||||
|
n: u32,
|
||||||
|
s: String,
|
||||||
|
b: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn benchmark(name: &'static str) -> BenchmarkBuilder {
|
||||||
|
BenchmarkBuilder {
|
||||||
|
name,
|
||||||
|
path: None,
|
||||||
|
method: None,
|
||||||
|
headers: None,
|
||||||
|
body: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct BenchmarkBuilder {
|
||||||
|
name: &'static str,
|
||||||
|
path: Option<&'static str>,
|
||||||
|
method: Option<&'static str>,
|
||||||
|
headers: Option<&'static [(&'static str, &'static str)]>,
|
||||||
|
body: Option<&'static str>,
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! config_method {
|
||||||
|
($name:ident, $ty:ty) => {
|
||||||
|
fn $name(mut self, $name: $ty) -> Self {
|
||||||
|
self.$name = Some($name);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BenchmarkBuilder {
|
||||||
|
config_method!(path, &'static str);
|
||||||
|
config_method!(method, &'static str);
|
||||||
|
config_method!(headers, &'static [(&'static str, &'static str)]);
|
||||||
|
config_method!(body, &'static str);
|
||||||
|
|
||||||
|
fn run<F>(self, f: F)
|
||||||
|
where
|
||||||
|
F: FnOnce() -> Router,
|
||||||
|
{
|
||||||
|
// support only running some benchmarks with
|
||||||
|
// ```
|
||||||
|
// cargo bench -- routing send-json
|
||||||
|
// ```
|
||||||
|
let args = std::env::args().collect::<Vec<_>>();
|
||||||
|
if !args.len() == 1 {
|
||||||
|
let names = &args[1..args.len() - 1];
|
||||||
|
if !names.is_empty() && !names.contains(&self.name.to_owned()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let app = f();
|
||||||
|
|
||||||
|
let rt = tokio::runtime::Builder::new_multi_thread()
|
||||||
|
.enable_all()
|
||||||
|
.build()
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let listener = rt
|
||||||
|
.block_on(tokio::net::TcpListener::bind("0.0.0.0:0"))
|
||||||
|
.unwrap();
|
||||||
|
let addr = listener.local_addr().unwrap();
|
||||||
|
|
||||||
|
std::thread::spawn(move || {
|
||||||
|
rt.block_on(async move {
|
||||||
|
let incoming = AddrIncoming::from_listener(listener).unwrap();
|
||||||
|
Server::builder(incoming)
|
||||||
|
.serve(app.into_make_service())
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
let mut cmd = Command::new("rewrk");
|
||||||
|
cmd.stdout(Stdio::piped());
|
||||||
|
|
||||||
|
cmd.arg("--host");
|
||||||
|
cmd.arg(format!("http://{}{}", addr, self.path.unwrap_or("")));
|
||||||
|
|
||||||
|
cmd.args(&["--connections", "10"]);
|
||||||
|
cmd.args(&["--threads", "10"]);
|
||||||
|
|
||||||
|
if on_ci() {
|
||||||
|
// don't slow down CI by running the benchmarks for too long
|
||||||
|
// but do run them for a bit
|
||||||
|
cmd.args(&["--duration", "1s"]);
|
||||||
|
} else {
|
||||||
|
cmd.args(&["--duration", "10s"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(method) = self.method {
|
||||||
|
cmd.args(&["--method", method]);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (key, value) in self.headers.into_iter().flatten() {
|
||||||
|
cmd.arg("--header");
|
||||||
|
cmd.arg(format!("{}: {}", key, value));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(body) = self.body {
|
||||||
|
cmd.args(&["--body", body]);
|
||||||
|
}
|
||||||
|
|
||||||
|
eprintln!("Running {:?} benchmark", self.name);
|
||||||
|
|
||||||
|
// indent output from `rewrk` so its easier to read when running multiple benchmarks
|
||||||
|
let mut child = cmd.spawn().unwrap();
|
||||||
|
let stdout = child.stdout.take().unwrap();
|
||||||
|
let stdout = std::io::BufReader::new(stdout);
|
||||||
|
for line in stdout.lines() {
|
||||||
|
let line = line.unwrap();
|
||||||
|
println!(" {}", line);
|
||||||
|
}
|
||||||
|
|
||||||
|
let status = child.wait().unwrap();
|
||||||
|
|
||||||
|
if !status.success() {
|
||||||
|
eprintln!("`rewrk` command failed");
|
||||||
|
std::process::exit(status.code().unwrap());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn install_rewrk() {
|
||||||
|
println!("installing rewrk");
|
||||||
|
let mut cmd = Command::new("cargo");
|
||||||
|
cmd.args(&[
|
||||||
|
"install",
|
||||||
|
"rewrk",
|
||||||
|
"--git",
|
||||||
|
"https://github.com/ChillFish8/rewrk.git",
|
||||||
|
]);
|
||||||
|
let status = cmd
|
||||||
|
.status()
|
||||||
|
.unwrap_or_else(|_| panic!("failed to install rewrk"));
|
||||||
|
if !status.success() {
|
||||||
|
panic!("failed to install rewrk");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn ensure_rewrk_is_installed() {
|
||||||
|
let mut cmd = Command::new("rewrk");
|
||||||
|
cmd.arg("--help");
|
||||||
|
cmd.stdout(Stdio::null());
|
||||||
|
cmd.stderr(Stdio::null());
|
||||||
|
cmd.status().unwrap_or_else(|_| {
|
||||||
|
panic!("rewrk is not installed. See https://github.com/lnx-search/rewrk")
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_ci() -> bool {
|
||||||
|
std::env::var("GITHUB_ACTIONS").is_ok()
|
||||||
|
}
|
|
@ -109,6 +109,7 @@ impl RequestBuilder {
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
pub(crate) fn multipart(mut self, form: reqwest::multipart::Form) -> Self {
|
pub(crate) fn multipart(mut self, form: reqwest::multipart::Form) -> Self {
|
||||||
self.builder = self.builder.multipart(form);
|
self.builder = self.builder.multipart(form);
|
||||||
self
|
self
|
||||||
|
|
Loading…
Reference in a new issue