2024-03-05 09:18:22 +00:00
|
|
|
use std::{io::Write, path::Path};
|
|
|
|
|
|
2024-08-19 12:35:32 +01:00
|
|
|
use anyhow::Context;
|
|
|
|
|
use axum::{response::IntoResponse, routing::get, Router};
|
2024-03-05 09:18:22 +00:00
|
|
|
use librqbit_core::Id20;
|
2024-08-14 15:11:41 +01:00
|
|
|
use rand::{thread_rng, Rng, RngCore, SeedableRng};
|
2024-03-05 09:18:22 +00:00
|
|
|
use tempfile::TempDir;
|
2024-08-20 21:53:43 +01:00
|
|
|
use tracing::{debug, info};
|
2024-03-05 09:18:22 +00:00
|
|
|
|
|
|
|
|
pub fn create_new_file_with_random_content(path: &Path, mut size: usize) {
|
|
|
|
|
let mut file = std::fs::OpenOptions::new()
|
|
|
|
|
.create_new(true)
|
|
|
|
|
.write(true)
|
|
|
|
|
.open(path)
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
2024-08-20 21:53:43 +01:00
|
|
|
debug!(?path, "creating temp file");
|
2024-03-05 09:18:22 +00:00
|
|
|
|
|
|
|
|
const BUF_SIZE: usize = 8192 * 16;
|
2024-03-05 01:53:41 -08:00
|
|
|
let mut rng = rand::rngs::SmallRng::from_entropy();
|
2024-03-05 09:18:22 +00:00
|
|
|
let mut write_buf = [0; BUF_SIZE];
|
|
|
|
|
while size > 0 {
|
|
|
|
|
rng.fill_bytes(&mut write_buf[..]);
|
|
|
|
|
let written = file.write(&write_buf[..size.min(BUF_SIZE)]).unwrap();
|
|
|
|
|
size -= written;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn create_default_random_dir_with_torrents(
|
|
|
|
|
num_files: usize,
|
|
|
|
|
file_size: usize,
|
|
|
|
|
tempdir_prefix: Option<&str>,
|
|
|
|
|
) -> TempDir {
|
|
|
|
|
let dir = TempDir::with_prefix(tempdir_prefix.unwrap_or("rqbit_test")).unwrap();
|
2024-08-20 21:53:43 +01:00
|
|
|
info!(path=?dir.path(), "created tempdir");
|
2024-03-05 09:18:22 +00:00
|
|
|
for f in 0..num_files {
|
|
|
|
|
create_new_file_with_random_content(&dir.path().join(&format!("{f}.data")), file_size);
|
|
|
|
|
}
|
|
|
|
|
dir
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
|
pub struct TestPeerMetadata {
|
|
|
|
|
pub server_id: u8,
|
|
|
|
|
pub max_random_sleep_ms: u8,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl TestPeerMetadata {
|
2024-08-14 15:11:41 +01:00
|
|
|
pub fn good() -> Self {
|
|
|
|
|
Self {
|
|
|
|
|
server_id: 0,
|
|
|
|
|
max_random_sleep_ms: 0,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-05 09:18:22 +00:00
|
|
|
pub fn as_peer_id(&self) -> Id20 {
|
|
|
|
|
let mut peer_id = Id20::default();
|
2024-08-14 15:11:41 +01:00
|
|
|
thread_rng().fill(&mut peer_id.0);
|
2024-03-05 09:18:22 +00:00
|
|
|
peer_id.0[0] = self.server_id;
|
|
|
|
|
peer_id.0[1] = self.max_random_sleep_ms;
|
|
|
|
|
peer_id
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn from_peer_id(peer_id: Id20) -> Self {
|
|
|
|
|
Self {
|
|
|
|
|
server_id: peer_id.0[0],
|
|
|
|
|
max_random_sleep_ms: peer_id.0[1],
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn disconnect_probability(&self) -> f64 {
|
2024-08-14 11:53:12 +01:00
|
|
|
if self.server_id % 2 == 1 {
|
2024-03-05 09:18:22 +00:00
|
|
|
return 0.05f64;
|
|
|
|
|
}
|
|
|
|
|
0f64
|
|
|
|
|
}
|
2024-03-29 20:32:08 +00:00
|
|
|
|
|
|
|
|
pub fn bad_data_probability(&self) -> f64 {
|
2024-08-14 11:53:12 +01:00
|
|
|
if self.server_id % 2 == 1 {
|
2024-03-29 20:32:08 +00:00
|
|
|
return 0.05f64;
|
|
|
|
|
}
|
|
|
|
|
0f64
|
|
|
|
|
}
|
2024-03-05 09:18:22 +00:00
|
|
|
}
|
2024-08-19 12:35:32 +01:00
|
|
|
|
|
|
|
|
async fn debug_server() -> anyhow::Result<()> {
|
|
|
|
|
async fn backtraces() -> impl IntoResponse {
|
2024-08-19 13:40:01 +01:00
|
|
|
#[cfg(feature = "async-bt")]
|
|
|
|
|
{
|
|
|
|
|
async_backtrace::taskdump_tree(true)
|
|
|
|
|
}
|
|
|
|
|
#[cfg(not(feature = "async-bt"))]
|
|
|
|
|
{
|
|
|
|
|
use crate::ApiError;
|
|
|
|
|
ApiError::from(anyhow::anyhow!(
|
|
|
|
|
"backtraces not enabled, enable async-bt feature"
|
|
|
|
|
))
|
|
|
|
|
}
|
2024-08-19 12:35:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let app = Router::new().route("/backtrace", get(backtraces));
|
|
|
|
|
let app = app.into_make_service();
|
|
|
|
|
|
|
|
|
|
let addr = "127.0.0.1:3032";
|
|
|
|
|
|
|
|
|
|
info!(%addr, "starting HTTP server");
|
|
|
|
|
|
|
|
|
|
use tokio::net::TcpListener;
|
|
|
|
|
|
|
|
|
|
let listener = TcpListener::bind(addr)
|
|
|
|
|
.await
|
|
|
|
|
.with_context(|| format!("error binding to {addr}"))?;
|
|
|
|
|
axum::serve(listener, app).await?;
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn spawn_debug_server() {
|
|
|
|
|
tokio::spawn(debug_server());
|
|
|
|
|
}
|