mirror of
https://github.com/YGGverse/aquatic.git
synced 2026-04-02 10:45:30 +00:00
Merge pull request #19 from greatest-ape/access-list-signals
aquatic_udp: mio: update access list on SIGUSR1; update TODO and README
This commit is contained in:
commit
a4d131359c
10 changed files with 117 additions and 62 deletions
21
Cargo.lock
generated
21
Cargo.lock
generated
|
|
@ -79,6 +79,7 @@ dependencies = [
|
||||||
"hashbrown 0.11.2",
|
"hashbrown 0.11.2",
|
||||||
"hex",
|
"hex",
|
||||||
"indexmap",
|
"indexmap",
|
||||||
|
"log",
|
||||||
"privdrop",
|
"privdrop",
|
||||||
"rand",
|
"rand",
|
||||||
"serde",
|
"serde",
|
||||||
|
|
@ -181,6 +182,7 @@ dependencies = [
|
||||||
"quickcheck_macros",
|
"quickcheck_macros",
|
||||||
"rand",
|
"rand",
|
||||||
"serde",
|
"serde",
|
||||||
|
"signal-hook",
|
||||||
"socket2 0.4.2",
|
"socket2 0.4.2",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
@ -1754,6 +1756,25 @@ dependencies = [
|
||||||
"opaque-debug",
|
"opaque-debug",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "signal-hook"
|
||||||
|
version = "0.3.10"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "9c98891d737e271a2954825ef19e46bd16bdb98e2746f2eec4f7a4ef7946efd1"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
"signal-hook-registry",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "signal-hook-registry"
|
||||||
|
version = "1.4.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "e51e73328dc4ac0c7ccbda3a494dfa03df1de2f46018127f60c693f2648455b0"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "simd-json"
|
name = "simd-json"
|
||||||
version = "0.4.8"
|
version = "0.4.8"
|
||||||
|
|
|
||||||
|
|
@ -171,8 +171,8 @@ paths in the configuration file, e.g.:
|
||||||
```toml
|
```toml
|
||||||
[network]
|
[network]
|
||||||
address = '0.0.0.0:3000'
|
address = '0.0.0.0:3000'
|
||||||
tls_certificate_path = './cert.crt'
|
tls_certificate_path = './cert.pem'
|
||||||
tls_private_key_path = './key.pk8'
|
tls_private_key_path = './key.pem'
|
||||||
```
|
```
|
||||||
|
|
||||||
### aquatic_ws: WebTorrent tracker
|
### aquatic_ws: WebTorrent tracker
|
||||||
|
|
|
||||||
5
TODO.md
5
TODO.md
|
|
@ -3,6 +3,8 @@
|
||||||
* readme
|
* readme
|
||||||
* document privilige dropping, cpu pinning
|
* document privilige dropping, cpu pinning
|
||||||
|
|
||||||
|
* socket_recv_size and ipv6_only in glommio implementations
|
||||||
|
|
||||||
* config: fail on unrecognized keys
|
* config: fail on unrecognized keys
|
||||||
|
|
||||||
* access lists:
|
* access lists:
|
||||||
|
|
@ -13,14 +15,12 @@
|
||||||
* aquatic_udp
|
* aquatic_udp
|
||||||
* CI for both implementations
|
* CI for both implementations
|
||||||
* glommio
|
* glommio
|
||||||
* ipv6 only flag
|
|
||||||
* consider sending local responses immediately
|
* consider sending local responses immediately
|
||||||
* consider adding ConnectedScrapeRequest::Scrape(PendingScrapeRequest)
|
* consider adding ConnectedScrapeRequest::Scrape(PendingScrapeRequest)
|
||||||
containing TransactionId and BTreeMap<usize, InfoHash>, and same for
|
containing TransactionId and BTreeMap<usize, InfoHash>, and same for
|
||||||
response
|
response
|
||||||
|
|
||||||
* aquatic_http:
|
* aquatic_http:
|
||||||
* ipv6 only flag
|
|
||||||
* optimize?
|
* optimize?
|
||||||
* get_peer_addr only once (takes 1.2% of runtime)
|
* get_peer_addr only once (takes 1.2% of runtime)
|
||||||
* queue response: allocating takes 2.8% of runtime
|
* queue response: allocating takes 2.8% of runtime
|
||||||
|
|
@ -37,7 +37,6 @@
|
||||||
Relevant for mio implementation too.
|
Relevant for mio implementation too.
|
||||||
|
|
||||||
* aquatic_ws
|
* aquatic_ws
|
||||||
* ipv6 only flag
|
|
||||||
* load test cpu pinning
|
* load test cpu pinning
|
||||||
* test with multiple socket and request workers
|
* test with multiple socket and request workers
|
||||||
* should it send back error on message parse error, or does that
|
* should it send back error on message parse error, or does that
|
||||||
|
|
|
||||||
|
|
@ -16,6 +16,7 @@ arc-swap = "1"
|
||||||
hashbrown = "0.11.2"
|
hashbrown = "0.11.2"
|
||||||
hex = "0.4"
|
hex = "0.4"
|
||||||
indexmap = "1"
|
indexmap = "1"
|
||||||
|
log = "0.4"
|
||||||
privdrop = "0.5"
|
privdrop = "0.5"
|
||||||
rand = { version = "0.8", features = ["small_rng"] }
|
rand = { version = "0.8", features = ["small_rng"] }
|
||||||
serde = { version = "1", features = ["derive"] }
|
serde = { version = "1", features = ["derive"] }
|
||||||
|
|
|
||||||
|
|
@ -3,6 +3,7 @@ use std::io::{BufRead, BufReader};
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
use anyhow::Context;
|
||||||
use arc_swap::ArcSwap;
|
use arc_swap::ArcSwap;
|
||||||
use hashbrown::HashSet;
|
use hashbrown::HashSet;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
@ -59,7 +60,11 @@ impl AccessList {
|
||||||
let mut new_list = Self::default();
|
let mut new_list = Self::default();
|
||||||
|
|
||||||
for line in reader.lines() {
|
for line in reader.lines() {
|
||||||
new_list.insert_from_line(&line?)?;
|
let line = line?;
|
||||||
|
|
||||||
|
new_list
|
||||||
|
.insert_from_line(&line)
|
||||||
|
.with_context(|| format!("Invalid line in access list: {}", line))?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(new_list)
|
Ok(new_list)
|
||||||
|
|
@ -75,15 +80,15 @@ impl AccessList {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait AccessListQuery {
|
pub trait AccessListQuery {
|
||||||
fn update_from_path(&self, path: &PathBuf) -> anyhow::Result<()>;
|
fn update(&self, config: &AccessListConfig) -> anyhow::Result<()>;
|
||||||
fn allows(&self, list_mode: AccessListMode, info_hash_bytes: &[u8; 20]) -> bool;
|
fn allows(&self, list_mode: AccessListMode, info_hash_bytes: &[u8; 20]) -> bool;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type AccessListArcSwap = ArcSwap<AccessList>;
|
pub type AccessListArcSwap = ArcSwap<AccessList>;
|
||||||
|
|
||||||
impl AccessListQuery for AccessListArcSwap {
|
impl AccessListQuery for AccessListArcSwap {
|
||||||
fn update_from_path(&self, path: &PathBuf) -> anyhow::Result<()> {
|
fn update(&self, config: &AccessListConfig) -> anyhow::Result<()> {
|
||||||
self.store(Arc::new(AccessList::create_from_path(path)?));
|
self.store(Arc::new(AccessList::create_from_path(&config.path)?));
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -34,6 +34,7 @@ mimalloc = { version = "0.1", default-features = false }
|
||||||
parking_lot = "0.11"
|
parking_lot = "0.11"
|
||||||
rand = { version = "0.8", features = ["small_rng"] }
|
rand = { version = "0.8", features = ["small_rng"] }
|
||||||
serde = { version = "1", features = ["derive"] }
|
serde = { version = "1", features = ["derive"] }
|
||||||
|
signal-hook = { version = "0.3" }
|
||||||
|
|
||||||
# mio
|
# mio
|
||||||
crossbeam-channel = { version = "0.5", optional = true }
|
crossbeam-channel = { version = "0.5", optional = true }
|
||||||
|
|
|
||||||
|
|
@ -9,15 +9,17 @@ use anyhow::Context;
|
||||||
use aquatic_common::privileges::drop_privileges_after_socket_binding;
|
use aquatic_common::privileges::drop_privileges_after_socket_binding;
|
||||||
use crossbeam_channel::unbounded;
|
use crossbeam_channel::unbounded;
|
||||||
|
|
||||||
|
use aquatic_common::access_list::AccessListQuery;
|
||||||
|
use signal_hook::consts::SIGUSR1;
|
||||||
|
use signal_hook::iterator::Signals;
|
||||||
|
|
||||||
|
use crate::config::Config;
|
||||||
|
|
||||||
pub mod common;
|
pub mod common;
|
||||||
pub mod handlers;
|
pub mod handlers;
|
||||||
pub mod network;
|
pub mod network;
|
||||||
pub mod tasks;
|
pub mod tasks;
|
||||||
|
|
||||||
use aquatic_common::access_list::{AccessListArcSwap, AccessListMode, AccessListQuery};
|
|
||||||
|
|
||||||
use crate::config::Config;
|
|
||||||
|
|
||||||
use common::State;
|
use common::State;
|
||||||
|
|
||||||
pub fn run(config: Config) -> ::anyhow::Result<()> {
|
pub fn run(config: Config) -> ::anyhow::Result<()> {
|
||||||
|
|
@ -29,36 +31,38 @@ pub fn run(config: Config) -> ::anyhow::Result<()> {
|
||||||
|
|
||||||
let state = State::default();
|
let state = State::default();
|
||||||
|
|
||||||
update_access_list(&config, &state.access_list);
|
update_access_list(&config, &state)?;
|
||||||
|
|
||||||
|
let mut signals = Signals::new(::std::iter::once(SIGUSR1))?;
|
||||||
|
|
||||||
|
{
|
||||||
|
let config = config.clone();
|
||||||
|
let state = state.clone();
|
||||||
|
|
||||||
|
::std::thread::spawn(move || run_inner(config, state));
|
||||||
|
}
|
||||||
|
|
||||||
|
for signal in &mut signals {
|
||||||
|
match signal {
|
||||||
|
SIGUSR1 => {
|
||||||
|
let _ = update_access_list(&config, &state);
|
||||||
|
}
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn run_inner(config: Config, state: State) -> ::anyhow::Result<()> {
|
||||||
|
if config.cpu_pinning.active {
|
||||||
|
core_affinity::set_for_current(core_affinity::CoreId {
|
||||||
|
id: config.cpu_pinning.offset,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
let num_bound_sockets = Arc::new(AtomicUsize::new(0));
|
let num_bound_sockets = Arc::new(AtomicUsize::new(0));
|
||||||
|
|
||||||
start_workers(config.clone(), state.clone(), num_bound_sockets.clone())?;
|
|
||||||
|
|
||||||
drop_privileges_after_socket_binding(
|
|
||||||
&config.privileges,
|
|
||||||
num_bound_sockets,
|
|
||||||
config.socket_workers,
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
loop {
|
|
||||||
::std::thread::sleep(Duration::from_secs(config.cleaning.interval));
|
|
||||||
|
|
||||||
update_access_list(&config, &state.access_list);
|
|
||||||
|
|
||||||
state
|
|
||||||
.torrents
|
|
||||||
.lock()
|
|
||||||
.clean(&config, state.access_list.load_full().deref());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn start_workers(
|
|
||||||
config: Config,
|
|
||||||
state: State,
|
|
||||||
num_bound_sockets: Arc<AtomicUsize>,
|
|
||||||
) -> ::anyhow::Result<()> {
|
|
||||||
let (request_sender, request_receiver) = unbounded();
|
let (request_sender, request_receiver) = unbounded();
|
||||||
let (response_sender, response_receiver) = unbounded();
|
let (response_sender, response_receiver) = unbounded();
|
||||||
|
|
||||||
|
|
@ -132,16 +136,36 @@ pub fn start_workers(
|
||||||
.with_context(|| "spawn statistics worker")?;
|
.with_context(|| "spawn statistics worker")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
drop_privileges_after_socket_binding(
|
||||||
|
&config.privileges,
|
||||||
|
num_bound_sockets,
|
||||||
|
config.socket_workers,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
::std::thread::sleep(Duration::from_secs(config.cleaning.interval));
|
||||||
|
|
||||||
|
state
|
||||||
|
.torrents
|
||||||
|
.lock()
|
||||||
|
.clean(&config, state.access_list.load_full().deref());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn update_access_list(config: &Config, access_list: &Arc<AccessListArcSwap>) {
|
fn update_access_list(config: &Config, state: &State) -> anyhow::Result<()> {
|
||||||
match config.access_list.mode {
|
if config.access_list.mode.is_on() {
|
||||||
AccessListMode::White | AccessListMode::Black => {
|
match state.access_list.update(&config.access_list) {
|
||||||
if let Err(err) = access_list.update_from_path(&config.access_list.path) {
|
Ok(()) => {
|
||||||
::log::error!("Update access list from path: {:?}", err);
|
::log::info!("Access list updated")
|
||||||
|
}
|
||||||
|
Err(err) => {
|
||||||
|
::log::error!("Updating access list failed: {:#}", err);
|
||||||
|
|
||||||
|
return Err(err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
AccessListMode::Off => {}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -17,7 +17,7 @@ use futures_lite::StreamExt;
|
||||||
use futures_rustls::server::TlsStream;
|
use futures_rustls::server::TlsStream;
|
||||||
use futures_rustls::TlsAcceptor;
|
use futures_rustls::TlsAcceptor;
|
||||||
use glommio::channels::channel_mesh::{MeshBuilder, Partial, Role, Senders};
|
use glommio::channels::channel_mesh::{MeshBuilder, Partial, Role, Senders};
|
||||||
use glommio::channels::local_channel::{LocalReceiver, LocalSender, new_unbounded};
|
use glommio::channels::local_channel::{new_unbounded, LocalReceiver, LocalSender};
|
||||||
use glommio::channels::shared_channel::ConnectedReceiver;
|
use glommio::channels::shared_channel::ConnectedReceiver;
|
||||||
use glommio::net::{TcpListener, TcpStream};
|
use glommio::net::{TcpListener, TcpStream};
|
||||||
use glommio::timer::TimerActionRepeat;
|
use glommio::timer::TimerActionRepeat;
|
||||||
|
|
@ -95,8 +95,7 @@ pub async fn run_socket_worker(
|
||||||
while let Some(stream) = incoming.next().await {
|
while let Some(stream) = incoming.next().await {
|
||||||
match stream {
|
match stream {
|
||||||
Ok(stream) => {
|
Ok(stream) => {
|
||||||
let (out_message_sender, out_message_receiver) =
|
let (out_message_sender, out_message_receiver) = new_unbounded();
|
||||||
new_unbounded();
|
|
||||||
let out_message_sender = Rc::new(out_message_sender);
|
let out_message_sender = Rc::new(out_message_sender);
|
||||||
|
|
||||||
let key = RefCell::borrow_mut(&connection_slab).insert(ConnectionReference {
|
let key = RefCell::borrow_mut(&connection_slab).insert(ConnectionReference {
|
||||||
|
|
@ -160,7 +159,7 @@ async fn receive_out_messages(
|
||||||
.get(channel_out_message.0.connection_id.0)
|
.get(channel_out_message.0.connection_id.0)
|
||||||
{
|
{
|
||||||
match reference.out_message_sender.try_send(channel_out_message) {
|
match reference.out_message_sender.try_send(channel_out_message) {
|
||||||
Ok(()) | Err(GlommioError::Closed(_)) => {},
|
Ok(()) | Err(GlommioError::Closed(_)) => {}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
::log::error!(
|
::log::error!(
|
||||||
"Couldn't send out_message from shared channel to local receiver: {:?}",
|
"Couldn't send out_message from shared channel to local receiver: {:?}",
|
||||||
|
|
|
||||||
|
|
@ -97,7 +97,6 @@ fn create_tls_config() -> anyhow::Result<Arc<rustls::ClientConfig>> {
|
||||||
Ok(Arc::new(config))
|
Ok(Arc::new(config))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
fn monitor_statistics(state: LoadTestState, config: &Config) {
|
fn monitor_statistics(state: LoadTestState, config: &Config) {
|
||||||
let start_time = Instant::now();
|
let start_time = Instant::now();
|
||||||
let mut report_avg_response_vec: Vec<f64> = Vec::new();
|
let mut report_avg_response_vec: Vec<f64> = Vec::new();
|
||||||
|
|
|
||||||
|
|
@ -7,12 +7,12 @@ use std::{
|
||||||
};
|
};
|
||||||
|
|
||||||
use aquatic_ws_protocol::{InMessage, JsonValue, OfferId, OutMessage, PeerId};
|
use aquatic_ws_protocol::{InMessage, JsonValue, OfferId, OutMessage, PeerId};
|
||||||
use async_tungstenite::{WebSocketStream, client_async};
|
use async_tungstenite::{client_async, WebSocketStream};
|
||||||
use futures::{StreamExt, SinkExt};
|
use futures::{SinkExt, StreamExt};
|
||||||
use futures_rustls::{TlsConnector, client::TlsStream};
|
use futures_rustls::{client::TlsStream, TlsConnector};
|
||||||
use glommio::net::TcpStream;
|
use glommio::net::TcpStream;
|
||||||
use glommio::{prelude::*, timer::TimerActionRepeat};
|
use glommio::{prelude::*, timer::TimerActionRepeat};
|
||||||
use rand::{Rng, SeedableRng, prelude::SmallRng};
|
use rand::{prelude::SmallRng, Rng, SeedableRng};
|
||||||
|
|
||||||
use crate::{common::LoadTestState, config::Config, utils::create_random_request};
|
use crate::{common::LoadTestState, config::Config, utils::create_random_request};
|
||||||
|
|
||||||
|
|
@ -80,7 +80,9 @@ impl Connection {
|
||||||
let stream = TcpStream::connect(config.server_address)
|
let stream = TcpStream::connect(config.server_address)
|
||||||
.await
|
.await
|
||||||
.map_err(|err| anyhow::anyhow!("connect: {:?}", err))?;
|
.map_err(|err| anyhow::anyhow!("connect: {:?}", err))?;
|
||||||
let stream = TlsConnector::from(tls_config).connect("example.com".try_into().unwrap(), stream).await?;
|
let stream = TlsConnector::from(tls_config)
|
||||||
|
.connect("example.com".try_into().unwrap(), stream)
|
||||||
|
.await?;
|
||||||
let request = format!(
|
let request = format!(
|
||||||
"ws://{}:{}",
|
"ws://{}:{}",
|
||||||
config.server_address.ip(),
|
config.server_address.ip(),
|
||||||
|
|
@ -114,8 +116,12 @@ impl Connection {
|
||||||
async fn run_connection_loop(&mut self) -> anyhow::Result<()> {
|
async fn run_connection_loop(&mut self) -> anyhow::Result<()> {
|
||||||
loop {
|
loop {
|
||||||
if self.can_send {
|
if self.can_send {
|
||||||
let request =
|
let request = create_random_request(
|
||||||
create_random_request(&self.config, &self.load_test_state, &mut self.rng, self.peer_id);
|
&self.config,
|
||||||
|
&self.load_test_state,
|
||||||
|
&mut self.rng,
|
||||||
|
self.peer_id,
|
||||||
|
);
|
||||||
|
|
||||||
// If self.send_answer is set and request is announce request, make
|
// If self.send_answer is set and request is announce request, make
|
||||||
// the request an offer answer
|
// the request an offer answer
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue