mirror of
https://github.com/YGGverse/aquatic.git
synced 2026-04-01 10:15:31 +00:00
182 lines
No EOL
3.8 KiB
Rust
182 lines
No EOL
3.8 KiB
Rust
use std::net::{SocketAddr, IpAddr, Ipv4Addr, Ipv6Addr};
|
|
use std::sync::{Arc, atomic::AtomicUsize};
|
|
use std::hash::Hash;
|
|
|
|
use hashbrown::HashMap;
|
|
use indexmap::IndexMap;
|
|
use parking_lot::Mutex;
|
|
|
|
pub use aquatic_common::ValidUntil;
|
|
pub use aquatic_udp_protocol::*;
|
|
|
|
|
|
pub const MAX_PACKET_SIZE: usize = 4096;
|
|
|
|
|
|
pub trait Ip: Hash + PartialEq + Eq + Clone + Copy {
|
|
fn ip_addr(self) -> IpAddr;
|
|
}
|
|
|
|
|
|
impl Ip for Ipv4Addr {
|
|
fn ip_addr(self) -> IpAddr {
|
|
IpAddr::V4(self)
|
|
}
|
|
}
|
|
|
|
|
|
impl Ip for Ipv6Addr {
|
|
fn ip_addr(self) -> IpAddr {
|
|
IpAddr::V6(self)
|
|
}
|
|
}
|
|
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
|
pub struct ConnectionKey {
|
|
pub connection_id: ConnectionId,
|
|
pub socket_addr: SocketAddr
|
|
}
|
|
|
|
|
|
pub type ConnectionMap = HashMap<ConnectionKey, ValidUntil>;
|
|
|
|
|
|
#[derive(PartialEq, Eq, Hash, Clone, Copy, Debug)]
|
|
pub enum PeerStatus {
|
|
Seeding,
|
|
Leeching,
|
|
Stopped
|
|
}
|
|
|
|
|
|
impl PeerStatus {
|
|
/// Determine peer status from announce event and number of bytes left.
|
|
///
|
|
/// Likely, the last branch will be taken most of the time.
|
|
#[inline]
|
|
pub fn from_event_and_bytes_left(
|
|
event: AnnounceEvent,
|
|
bytes_left: NumberOfBytes
|
|
) -> Self {
|
|
if event == AnnounceEvent::Stopped {
|
|
Self::Stopped
|
|
} else if bytes_left.0 == 0 {
|
|
Self::Seeding
|
|
} else {
|
|
Self::Leeching
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub struct Peer<I: Ip> {
|
|
pub ip_address: I,
|
|
pub port: Port,
|
|
pub status: PeerStatus,
|
|
pub valid_until: ValidUntil
|
|
}
|
|
|
|
|
|
impl <I: Ip>Peer<I> {
|
|
#[inline(always)]
|
|
pub fn to_response_peer(&self) -> ResponsePeer {
|
|
ResponsePeer {
|
|
ip_address: self.ip_address.ip_addr(),
|
|
port: self.port
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#[derive(PartialEq, Eq, Hash, Clone)]
|
|
pub struct PeerMapKey<I: Ip> {
|
|
pub ip: I,
|
|
pub peer_id: PeerId
|
|
}
|
|
|
|
|
|
pub type PeerMap<I> = IndexMap<PeerMapKey<I>, Peer<I>>;
|
|
|
|
|
|
pub struct TorrentData<I: Ip> {
|
|
pub peers: PeerMap<I>,
|
|
pub num_seeders: usize,
|
|
pub num_leechers: usize,
|
|
}
|
|
|
|
|
|
impl <I: Ip>Default for TorrentData<I> {
|
|
fn default() -> Self {
|
|
Self {
|
|
peers: IndexMap::new(),
|
|
num_seeders: 0,
|
|
num_leechers: 0,
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
pub type TorrentMap<I> = HashMap<InfoHash, TorrentData<I>>;
|
|
|
|
|
|
#[derive(Default)]
|
|
pub struct TorrentMaps {
|
|
pub ipv4: TorrentMap<Ipv4Addr>,
|
|
pub ipv6: TorrentMap<Ipv6Addr>,
|
|
}
|
|
|
|
|
|
#[derive(Default)]
|
|
pub struct Statistics {
|
|
pub requests_received: AtomicUsize,
|
|
pub responses_sent: AtomicUsize,
|
|
pub readable_events: AtomicUsize,
|
|
pub bytes_received: AtomicUsize,
|
|
pub bytes_sent: AtomicUsize,
|
|
}
|
|
|
|
|
|
#[derive(Clone)]
|
|
pub struct State {
|
|
pub connections: Arc<Mutex<ConnectionMap>>,
|
|
pub torrents: Arc<Mutex<TorrentMaps>>,
|
|
pub statistics: Arc<Statistics>,
|
|
}
|
|
|
|
|
|
impl Default for State {
|
|
fn default() -> Self {
|
|
Self {
|
|
connections: Arc::new(Mutex::new(HashMap::new())),
|
|
torrents: Arc::new(Mutex::new(TorrentMaps::default())),
|
|
statistics: Arc::new(Statistics::default()),
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
#[test]
|
|
fn test_peer_status_from_event_and_bytes_left(){
|
|
use crate::common::*;
|
|
|
|
use PeerStatus::*;
|
|
|
|
let f = PeerStatus::from_event_and_bytes_left;
|
|
|
|
assert_eq!(Stopped, f(AnnounceEvent::Stopped, NumberOfBytes(0)));
|
|
assert_eq!(Stopped, f(AnnounceEvent::Stopped, NumberOfBytes(1)));
|
|
|
|
assert_eq!(Seeding, f(AnnounceEvent::Started, NumberOfBytes(0)));
|
|
assert_eq!(Leeching, f(AnnounceEvent::Started, NumberOfBytes(1)));
|
|
|
|
assert_eq!(Seeding, f(AnnounceEvent::Completed, NumberOfBytes(0)));
|
|
assert_eq!(Leeching, f(AnnounceEvent::Completed, NumberOfBytes(1)));
|
|
|
|
assert_eq!(Seeding, f(AnnounceEvent::None, NumberOfBytes(0)));
|
|
assert_eq!(Leeching, f(AnnounceEvent::None, NumberOfBytes(1)));
|
|
}
|
|
} |