first commit
This commit is contained in:
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
target
|
||||
server
|
||||
1652
bench/Cargo.lock
generated
Normal file
1652
bench/Cargo.lock
generated
Normal file
File diff suppressed because it is too large
Load Diff
8
bench/Cargo.toml
Normal file
8
bench/Cargo.toml
Normal file
@@ -0,0 +1,8 @@
|
||||
[package]
|
||||
name = "bench"
|
||||
version = "0.1.0"
|
||||
edition = "2024"
|
||||
|
||||
[dependencies]
|
||||
tokio = { version = "1", features = ["full"] }
|
||||
reqwest = { version = "0.12", features = ["json", "stream"] }
|
||||
118
bench/src/main.rs
Normal file
118
bench/src/main.rs
Normal file
@@ -0,0 +1,118 @@
|
||||
use std::collections::HashMap;
|
||||
use std::sync::{Arc, Mutex};
|
||||
use std::sync::atomic::{AtomicU64, Ordering};
|
||||
use std::time::{Duration, Instant};
|
||||
use tokio::time::sleep;
|
||||
|
||||
#[tokio::main]
|
||||
async fn main() {
|
||||
let target = "http://127.0.0.1:6008/"; // adjust your port
|
||||
let expected_ip = "1.2.3.4";
|
||||
let start_rps = 10_000;
|
||||
let max_rps = 100_000;
|
||||
let step = 1000;
|
||||
let duration_per_step = Duration::from_secs(2);
|
||||
|
||||
println!("# Benchmarking {target}");
|
||||
println!("rate(req/s)\tavg_latency(ms)\terrors");
|
||||
|
||||
let client = reqwest::Client::builder()
|
||||
.timeout(Duration::from_secs(2))
|
||||
.pool_idle_timeout(None)
|
||||
.build()
|
||||
.unwrap();
|
||||
|
||||
for rate in (start_rps..=max_rps).step_by(step) {
|
||||
let interval = Duration::from_secs_f64(1.0 / rate as f64);
|
||||
let end_time = Instant::now() + duration_per_step;
|
||||
|
||||
let total = Arc::new(AtomicU64::new(0));
|
||||
let errors = Arc::new(AtomicU64::new(0));
|
||||
let lat_sum = Arc::new(AtomicU64::new(0)); // µs
|
||||
let err_kinds = Arc::new(Mutex::new(HashMap::<&'static str, u64>::new()));
|
||||
|
||||
while Instant::now() < end_time {
|
||||
let client = client.clone();
|
||||
let total = total.clone();
|
||||
let errors = errors.clone();
|
||||
let lat_sum = lat_sum.clone();
|
||||
let err_kinds = err_kinds.clone();
|
||||
|
||||
tokio::spawn(async move {
|
||||
let start = Instant::now();
|
||||
let result = client
|
||||
.get(target)
|
||||
.header("X-Forwarded-For", expected_ip)
|
||||
.send()
|
||||
.await;
|
||||
|
||||
total.fetch_add(1, Ordering::Relaxed);
|
||||
|
||||
match result {
|
||||
Ok(resp) => {
|
||||
let status = resp.status();
|
||||
match resp.text().await {
|
||||
Ok(body) => {
|
||||
if !status.is_success() {
|
||||
errors.fetch_add(1, Ordering::Relaxed);
|
||||
let mut map = err_kinds.lock().unwrap();
|
||||
*map.entry("bad_status").or_insert(0) += 1;
|
||||
} else if body.trim() != expected_ip {
|
||||
errors.fetch_add(1, Ordering::Relaxed);
|
||||
let mut map = err_kinds.lock().unwrap();
|
||||
*map.entry("wrong_body").or_insert(0) += 1;
|
||||
} else {
|
||||
let latency = start.elapsed().as_micros() as u64;
|
||||
lat_sum.fetch_add(latency, Ordering::Relaxed);
|
||||
}
|
||||
}
|
||||
Err(_) => {
|
||||
errors.fetch_add(1, Ordering::Relaxed);
|
||||
let mut map = err_kinds.lock().unwrap();
|
||||
*map.entry("read_fail").or_insert(0) += 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
Err(e) => {
|
||||
errors.fetch_add(1, Ordering::Relaxed);
|
||||
let reason: &'static str = if e.is_connect() {
|
||||
"connect_fail"
|
||||
} else if e.is_timeout() {
|
||||
"timeout"
|
||||
} else if e.is_body() {
|
||||
"body_err"
|
||||
} else {
|
||||
"other_err"
|
||||
};
|
||||
let mut map = err_kinds.lock().unwrap();
|
||||
*map.entry(reason).or_insert(0) += 1;
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
sleep(interval).await;
|
||||
}
|
||||
|
||||
sleep(Duration::from_secs(1)).await;
|
||||
|
||||
let total = total.load(Ordering::Relaxed);
|
||||
let errors = errors.load(Ordering::Relaxed);
|
||||
let ok = total.saturating_sub(errors);
|
||||
let lat_sum = lat_sum.load(Ordering::Relaxed);
|
||||
|
||||
let avg_ms = if ok > 0 {
|
||||
(lat_sum as f64 / ok as f64) / 1000.0
|
||||
} else {
|
||||
0.0
|
||||
};
|
||||
|
||||
println!("{rate}\t\t{avg_ms:.3}\t\t{errors}");
|
||||
|
||||
let map = err_kinds.lock().unwrap();
|
||||
if !map.is_empty() {
|
||||
for (k, v) in map.iter() {
|
||||
println!(" {k}: {v}");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
67
server.c
Normal file
67
server.c
Normal file
@@ -0,0 +1,67 @@
|
||||
#include <arpa/inet.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/socket.h>
|
||||
#include <unistd.h>
|
||||
#include <string.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#define PORT 6008
|
||||
#define BACKLOG 16
|
||||
#define RECV_BUF 1024
|
||||
|
||||
int main() {
|
||||
int server_fd = socket(AF_INET, SOCK_STREAM, 0);
|
||||
int opt = 1;
|
||||
setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
|
||||
|
||||
struct sockaddr_in addr = {0};
|
||||
addr.sin_family = AF_INET;
|
||||
addr.sin_addr.s_addr = INADDR_ANY;
|
||||
addr.sin_port = htons(PORT);
|
||||
|
||||
bind(server_fd, (struct sockaddr *)&addr, sizeof(addr));
|
||||
listen(server_fd, BACKLOG);
|
||||
|
||||
printf("listening port %d\nbacklog %d\nrecv buffer %d\n", PORT, BACKLOG, RECV_BUF);
|
||||
|
||||
char buf[RECV_BUF];
|
||||
const char *hdr = "X-Forwarded-For:";
|
||||
const size_t hdrlen = strlen(hdr);
|
||||
|
||||
while(1) {
|
||||
int client = accept(server_fd, NULL, NULL);
|
||||
if (client < 0) continue;
|
||||
ssize_t r = recv(client, buf, sizeof(buf) -1, 0);
|
||||
if (r <= 0) { close(client); continue; }
|
||||
buf[r] = 0;
|
||||
|
||||
//find header
|
||||
char *ip = NULL;
|
||||
for (ssize_t i = 0; i < r - (ssize_t)hdrlen; ++i) {
|
||||
if (strncasecmp(buf + i, hdr, hdrlen) == 0) {
|
||||
ip = buf + i + hdrlen;
|
||||
while (*ip == ' ' || *ip == '\t') ip++;
|
||||
break;
|
||||
}
|
||||
}
|
||||
//find end
|
||||
char *end = ip;
|
||||
if (ip) {
|
||||
while (*end && *end != '\r' && *end != '\n') end++;
|
||||
*end = 0;
|
||||
}
|
||||
|
||||
//response
|
||||
const char *head = "HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\nContent-Length: ";
|
||||
char lenbuf[32];
|
||||
int iplen = ip ? strlen(ip) : 0;
|
||||
int len = snprintf(lenbuf, sizeof(lenbuf), "%d\r\n\r\n", iplen);
|
||||
|
||||
send(client, head, strlen(head), 0);
|
||||
send(client, lenbuf, len, 0);
|
||||
if (iplen > 0) send(client, ip, iplen, 0);
|
||||
|
||||
close(client);
|
||||
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user