address filter
This commit is contained in:
parent
d290a66f32
commit
cb66af7df3
@ -10,19 +10,25 @@ license = "LGPL-2.0-or-later OR MPL-2.0 OR (MIT AND BSD-3-Clause)"
|
|||||||
crate-type = ["cdylib", "staticlib", "rlib"]
|
crate-type = ["cdylib", "staticlib", "rlib"]
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
|
|
||||||
|
# Common features
|
||||||
default = ["enable-crypto-vld0"]
|
default = ["enable-crypto-vld0"]
|
||||||
crypto-test = ["enable-crypto-vld0", "enable-crypto-none"]
|
|
||||||
crypto-test-none = ["enable-crypto-none"]
|
|
||||||
enable-crypto-vld0 = []
|
|
||||||
enable-crypto-none = []
|
|
||||||
verbose-tracing = []
|
|
||||||
rt-async-std = ["async-std", "async-std-resolver", "async_executors/async_std", "rtnetlink/smol_socket", "veilid-tools/rt-async-std"]
|
rt-async-std = ["async-std", "async-std-resolver", "async_executors/async_std", "rtnetlink/smol_socket", "veilid-tools/rt-async-std"]
|
||||||
rt-tokio = ["tokio", "tokio-util", "tokio-stream", "trust-dns-resolver/tokio-runtime", "async_executors/tokio_tp", "async_executors/tokio_io", "async_executors/tokio_timer", "rtnetlink/tokio_socket", "veilid-tools/rt-tokio"]
|
rt-tokio = ["tokio", "tokio-util", "tokio-stream", "trust-dns-resolver/tokio-runtime", "async_executors/tokio_tp", "async_executors/tokio_io", "async_executors/tokio_timer", "rtnetlink/tokio_socket", "veilid-tools/rt-tokio"]
|
||||||
rt-wasm-bindgen = ["veilid-tools/rt-wasm-bindgen", "async_executors/bindgen"]
|
rt-wasm-bindgen = ["veilid-tools/rt-wasm-bindgen", "async_executors/bindgen"]
|
||||||
|
|
||||||
|
# Crypto support features
|
||||||
|
enable-crypto-vld0 = []
|
||||||
|
enable-crypto-none = []
|
||||||
|
|
||||||
|
# Debugging and testing features
|
||||||
|
verbose-tracing = []
|
||||||
|
tracking = []
|
||||||
|
debug-dht = []
|
||||||
|
crypto-test = ["enable-crypto-vld0", "enable-crypto-none"]
|
||||||
|
crypto-test-none = ["enable-crypto-none"]
|
||||||
veilid_core_android_tests = ["dep:paranoid-android"]
|
veilid_core_android_tests = ["dep:paranoid-android"]
|
||||||
veilid_core_ios_tests = ["dep:tracing-oslog"]
|
veilid_core_ios_tests = ["dep:tracing-oslog"]
|
||||||
tracking = []
|
|
||||||
network-result-extra = ["veilid-tools/network-result-extra"]
|
network-result-extra = ["veilid-tools/network-result-extra"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
311
veilid-core/src/network_manager/address_filter.rs
Normal file
311
veilid-core/src/network_manager/address_filter.rs
Normal file
@ -0,0 +1,311 @@
|
|||||||
|
use super::*;
|
||||||
|
use alloc::collections::btree_map::Entry;
|
||||||
|
|
||||||
|
// XXX: Move to config eventually?
|
||||||
|
const PUNISHMENT_DURATION_MIN: usize = 60;
|
||||||
|
|
||||||
|
#[derive(ThisError, Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
|
pub enum AddressFilterError {
|
||||||
|
#[error("Count exceeded")]
|
||||||
|
CountExceeded,
|
||||||
|
#[error("Rate exceeded")]
|
||||||
|
RateExceeded,
|
||||||
|
#[error("Address is punished")]
|
||||||
|
Punished,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(ThisError, Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
|
#[error("Address not in table")]
|
||||||
|
pub struct AddressNotInTableError {}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct AddressFilterInner {
|
||||||
|
conn_count_by_ip4: BTreeMap<Ipv4Addr, usize>,
|
||||||
|
conn_count_by_ip6_prefix: BTreeMap<Ipv6Addr, usize>,
|
||||||
|
conn_timestamps_by_ip4: BTreeMap<Ipv4Addr, Vec<Timestamp>>,
|
||||||
|
conn_timestamps_by_ip6_prefix: BTreeMap<Ipv6Addr, Vec<Timestamp>>,
|
||||||
|
punishments_by_ip4: BTreeMap<Ipv4Addr, Timestamp>,
|
||||||
|
punishments_by_ip6_prefix: BTreeMap<Ipv6Addr, Timestamp>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct AddressFilterUnlockedInner {
|
||||||
|
max_connections_per_ip4: usize,
|
||||||
|
max_connections_per_ip6_prefix: usize,
|
||||||
|
max_connections_per_ip6_prefix_size: usize,
|
||||||
|
max_connection_frequency_per_min: usize,
|
||||||
|
punishment_duration_min: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct AddressFilter {
|
||||||
|
unlocked_inner: Arc<AddressFilterUnlockedInner>,
|
||||||
|
inner: Arc<Mutex<AddressFilterInner>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AddressFilter {
|
||||||
|
pub fn new(config: VeilidConfig) -> Self {
|
||||||
|
let c = config.get();
|
||||||
|
Self {
|
||||||
|
unlocked_inner: Arc::new(AddressFilterUnlockedInner {
|
||||||
|
max_connections_per_ip4: c.network.max_connections_per_ip4 as usize,
|
||||||
|
max_connections_per_ip6_prefix: c.network.max_connections_per_ip6_prefix as usize,
|
||||||
|
max_connections_per_ip6_prefix_size: c.network.max_connections_per_ip6_prefix_size
|
||||||
|
as usize,
|
||||||
|
max_connection_frequency_per_min: c.network.max_connection_frequency_per_min
|
||||||
|
as usize,
|
||||||
|
punishment_duration_min: PUNISHMENT_DURATION_MIN,
|
||||||
|
}),
|
||||||
|
inner: Arc::new(Mutex::new(AddressFilterInner {
|
||||||
|
conn_count_by_ip4: BTreeMap::new(),
|
||||||
|
conn_count_by_ip6_prefix: BTreeMap::new(),
|
||||||
|
conn_timestamps_by_ip4: BTreeMap::new(),
|
||||||
|
conn_timestamps_by_ip6_prefix: BTreeMap::new(),
|
||||||
|
punishments_by_ip4: BTreeMap::new(),
|
||||||
|
punishments_by_ip6_prefix: BTreeMap::new(),
|
||||||
|
})),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn purge_old_timestamps(&self, inner: &mut AddressFilterInner, cur_ts: Timestamp) {
|
||||||
|
// v4
|
||||||
|
{
|
||||||
|
let mut dead_keys = Vec::<Ipv4Addr>::new();
|
||||||
|
for (key, value) in &mut inner.conn_timestamps_by_ip4 {
|
||||||
|
value.retain(|v| {
|
||||||
|
// keep timestamps that are less than a minute away
|
||||||
|
cur_ts.saturating_sub(*v) < TimestampDuration::new(60_000_000u64)
|
||||||
|
});
|
||||||
|
if value.is_empty() {
|
||||||
|
dead_keys.push(*key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for key in dead_keys {
|
||||||
|
inner.conn_timestamps_by_ip4.remove(&key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// v6
|
||||||
|
{
|
||||||
|
let mut dead_keys = Vec::<Ipv6Addr>::new();
|
||||||
|
for (key, value) in &mut inner.conn_timestamps_by_ip6_prefix {
|
||||||
|
value.retain(|v| {
|
||||||
|
// keep timestamps that are less than a minute away
|
||||||
|
cur_ts.saturating_sub(*v) < TimestampDuration::new(60_000_000u64)
|
||||||
|
});
|
||||||
|
if value.is_empty() {
|
||||||
|
dead_keys.push(*key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for key in dead_keys {
|
||||||
|
inner.conn_timestamps_by_ip6_prefix.remove(&key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn purge_old_punishments(&self, inner: &mut AddressFilterInner, cur_ts: Timestamp) {
|
||||||
|
// v4
|
||||||
|
{
|
||||||
|
let mut dead_keys = Vec::<Ipv4Addr>::new();
|
||||||
|
for (key, value) in &mut inner.punishments_by_ip4 {
|
||||||
|
// Drop punishments older than the punishment duration
|
||||||
|
if cur_ts.as_u64().saturating_sub(value.as_u64())
|
||||||
|
> self.unlocked_inner.punishment_duration_min as u64 * 60_000_000u64
|
||||||
|
{
|
||||||
|
dead_keys.push(*key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for key in dead_keys {
|
||||||
|
inner.punishments_by_ip4.remove(&key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// v6
|
||||||
|
{
|
||||||
|
let mut dead_keys = Vec::<Ipv6Addr>::new();
|
||||||
|
for (key, value) in &mut inner.punishments_by_ip6_prefix {
|
||||||
|
// Drop punishments older than the punishment duration
|
||||||
|
if cur_ts.as_u64().saturating_sub(value.as_u64())
|
||||||
|
> self.unlocked_inner.punishment_duration_min as u64 * 60_000_000u64
|
||||||
|
{
|
||||||
|
dead_keys.push(*key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for key in dead_keys {
|
||||||
|
inner.punishments_by_ip6_prefix.remove(&key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_punished_inner(&self, inner: &AddressFilterInner, ipblock: IpAddr) -> bool {
|
||||||
|
match ipblock {
|
||||||
|
IpAddr::V4(v4) => {
|
||||||
|
if inner.punishments_by_ip4.contains_key(&v4) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
IpAddr::V6(v6) => {
|
||||||
|
if inner.punishments_by_ip6_prefix.contains_key(&v6) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
false
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn is_punished(&self, addr: IpAddr) -> bool {
|
||||||
|
let inner = self.inner.lock();
|
||||||
|
let ipblock = ip_to_ipblock(
|
||||||
|
self.unlocked_inner.max_connections_per_ip6_prefix_size,
|
||||||
|
addr,
|
||||||
|
);
|
||||||
|
self.is_punished_inner(&*inner, ipblock)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn punish(&self, addr: IpAddr) {
|
||||||
|
log_net!(debug ">>> PUNISHED: {}", addr);
|
||||||
|
let ts = get_aligned_timestamp();
|
||||||
|
|
||||||
|
let ipblock = ip_to_ipblock(
|
||||||
|
self.unlocked_inner.max_connections_per_ip6_prefix_size,
|
||||||
|
addr,
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut inner = self.inner.lock();
|
||||||
|
match ipblock {
|
||||||
|
IpAddr::V4(v4) => inner
|
||||||
|
.punishments_by_ip4
|
||||||
|
.entry(v4)
|
||||||
|
.and_modify(|v| *v = ts)
|
||||||
|
.or_insert(ts),
|
||||||
|
IpAddr::V6(v6) => inner
|
||||||
|
.punishments_by_ip6_prefix
|
||||||
|
.entry(v6)
|
||||||
|
.and_modify(|v| *v = ts)
|
||||||
|
.or_insert(ts),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn address_filter_task_routine(
|
||||||
|
self,
|
||||||
|
_stop_token: StopToken,
|
||||||
|
_last_ts: Timestamp,
|
||||||
|
cur_ts: Timestamp,
|
||||||
|
) -> EyreResult<()> {
|
||||||
|
//
|
||||||
|
let mut inner = self.inner.lock();
|
||||||
|
self.purge_old_timestamps(&mut *inner, cur_ts);
|
||||||
|
self.purge_old_punishments(&mut *inner, cur_ts);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn add_connection(&self, addr: IpAddr) -> Result<(), AddressFilterError> {
|
||||||
|
let inner = &mut *self.inner.lock();
|
||||||
|
|
||||||
|
let ipblock = ip_to_ipblock(
|
||||||
|
self.unlocked_inner.max_connections_per_ip6_prefix_size,
|
||||||
|
addr,
|
||||||
|
);
|
||||||
|
if self.is_punished_inner(inner, ipblock) {
|
||||||
|
return Err(AddressFilterError::Punished);
|
||||||
|
}
|
||||||
|
|
||||||
|
let ts = get_aligned_timestamp();
|
||||||
|
self.purge_old_timestamps(inner, ts);
|
||||||
|
|
||||||
|
match ipblock {
|
||||||
|
IpAddr::V4(v4) => {
|
||||||
|
// See if we have too many connections from this ip block
|
||||||
|
let cnt = inner.conn_count_by_ip4.entry(v4).or_default();
|
||||||
|
assert!(*cnt <= self.unlocked_inner.max_connections_per_ip4);
|
||||||
|
if *cnt == self.unlocked_inner.max_connections_per_ip4 {
|
||||||
|
warn!("address filter count exceeded: {:?}", v4);
|
||||||
|
return Err(AddressFilterError::CountExceeded);
|
||||||
|
}
|
||||||
|
// See if this ip block has connected too frequently
|
||||||
|
let tstamps = inner.conn_timestamps_by_ip4.entry(v4).or_default();
|
||||||
|
tstamps.retain(|v| {
|
||||||
|
// keep timestamps that are less than a minute away
|
||||||
|
ts.saturating_sub(*v) < TimestampDuration::new(60_000_000u64)
|
||||||
|
});
|
||||||
|
assert!(tstamps.len() <= self.unlocked_inner.max_connection_frequency_per_min);
|
||||||
|
if tstamps.len() == self.unlocked_inner.max_connection_frequency_per_min {
|
||||||
|
warn!("address filter rate exceeded: {:?}", v4);
|
||||||
|
return Err(AddressFilterError::RateExceeded);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If it's okay, add the counts and timestamps
|
||||||
|
*cnt += 1;
|
||||||
|
tstamps.push(ts);
|
||||||
|
}
|
||||||
|
IpAddr::V6(v6) => {
|
||||||
|
// See if we have too many connections from this ip block
|
||||||
|
let cnt = inner.conn_count_by_ip6_prefix.entry(v6).or_default();
|
||||||
|
assert!(*cnt <= self.unlocked_inner.max_connections_per_ip6_prefix);
|
||||||
|
if *cnt == self.unlocked_inner.max_connections_per_ip6_prefix {
|
||||||
|
warn!("address filter count exceeded: {:?}", v6);
|
||||||
|
return Err(AddressFilterError::CountExceeded);
|
||||||
|
}
|
||||||
|
// See if this ip block has connected too frequently
|
||||||
|
let tstamps = inner.conn_timestamps_by_ip6_prefix.entry(v6).or_default();
|
||||||
|
assert!(tstamps.len() <= self.unlocked_inner.max_connection_frequency_per_min);
|
||||||
|
if tstamps.len() == self.unlocked_inner.max_connection_frequency_per_min {
|
||||||
|
warn!("address filter rate exceeded: {:?}", v6);
|
||||||
|
return Err(AddressFilterError::RateExceeded);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If it's okay, add the counts and timestamps
|
||||||
|
*cnt += 1;
|
||||||
|
tstamps.push(ts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn remove_connection(&mut self, addr: IpAddr) -> Result<(), AddressNotInTableError> {
|
||||||
|
let mut inner = self.inner.lock();
|
||||||
|
|
||||||
|
let ipblock = ip_to_ipblock(
|
||||||
|
self.unlocked_inner.max_connections_per_ip6_prefix_size,
|
||||||
|
addr,
|
||||||
|
);
|
||||||
|
|
||||||
|
let ts = get_aligned_timestamp();
|
||||||
|
self.purge_old_timestamps(&mut *inner, ts);
|
||||||
|
|
||||||
|
match ipblock {
|
||||||
|
IpAddr::V4(v4) => {
|
||||||
|
match inner.conn_count_by_ip4.entry(v4) {
|
||||||
|
Entry::Vacant(_) => {
|
||||||
|
return Err(AddressNotInTableError {});
|
||||||
|
}
|
||||||
|
Entry::Occupied(mut o) => {
|
||||||
|
let cnt = o.get_mut();
|
||||||
|
assert!(*cnt > 0);
|
||||||
|
if *cnt == 0 {
|
||||||
|
inner.conn_count_by_ip4.remove(&v4);
|
||||||
|
} else {
|
||||||
|
*cnt -= 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
IpAddr::V6(v6) => {
|
||||||
|
match inner.conn_count_by_ip6_prefix.entry(v6) {
|
||||||
|
Entry::Vacant(_) => {
|
||||||
|
return Err(AddressNotInTableError {});
|
||||||
|
}
|
||||||
|
Entry::Occupied(mut o) => {
|
||||||
|
let cnt = o.get_mut();
|
||||||
|
assert!(*cnt > 0);
|
||||||
|
if *cnt == 0 {
|
||||||
|
inner.conn_count_by_ip6_prefix.remove(&v6);
|
||||||
|
} else {
|
||||||
|
*cnt -= 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
@ -1,176 +0,0 @@
|
|||||||
use super::*;
|
|
||||||
use alloc::collections::btree_map::Entry;
|
|
||||||
|
|
||||||
#[derive(ThisError, Debug, Clone, Copy, PartialEq, Eq)]
|
|
||||||
pub enum AddressFilterError {
|
|
||||||
#[error("Count exceeded")]
|
|
||||||
CountExceeded,
|
|
||||||
#[error("Rate exceeded")]
|
|
||||||
RateExceeded,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(ThisError, Debug, Clone, Copy, PartialEq, Eq)]
|
|
||||||
#[error("Address not in table")]
|
|
||||||
pub struct AddressNotInTableError {}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct ConnectionLimits {
|
|
||||||
max_connections_per_ip4: usize,
|
|
||||||
max_connections_per_ip6_prefix: usize,
|
|
||||||
max_connections_per_ip6_prefix_size: usize,
|
|
||||||
max_connection_frequency_per_min: usize,
|
|
||||||
conn_count_by_ip4: BTreeMap<Ipv4Addr, usize>,
|
|
||||||
conn_count_by_ip6_prefix: BTreeMap<Ipv6Addr, usize>,
|
|
||||||
conn_timestamps_by_ip4: BTreeMap<Ipv4Addr, Vec<Timestamp>>,
|
|
||||||
conn_timestamps_by_ip6_prefix: BTreeMap<Ipv6Addr, Vec<Timestamp>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ConnectionLimits {
|
|
||||||
pub fn new(config: VeilidConfig) -> Self {
|
|
||||||
let c = config.get();
|
|
||||||
Self {
|
|
||||||
max_connections_per_ip4: c.network.max_connections_per_ip4 as usize,
|
|
||||||
max_connections_per_ip6_prefix: c.network.max_connections_per_ip6_prefix as usize,
|
|
||||||
max_connections_per_ip6_prefix_size: c.network.max_connections_per_ip6_prefix_size
|
|
||||||
as usize,
|
|
||||||
max_connection_frequency_per_min: c.network.max_connection_frequency_per_min as usize,
|
|
||||||
conn_count_by_ip4: BTreeMap::new(),
|
|
||||||
conn_count_by_ip6_prefix: BTreeMap::new(),
|
|
||||||
conn_timestamps_by_ip4: BTreeMap::new(),
|
|
||||||
conn_timestamps_by_ip6_prefix: BTreeMap::new(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn purge_old_timestamps(&mut self, cur_ts: Timestamp) {
|
|
||||||
// v4
|
|
||||||
{
|
|
||||||
let mut dead_keys = Vec::<Ipv4Addr>::new();
|
|
||||||
for (key, value) in &mut self.conn_timestamps_by_ip4 {
|
|
||||||
value.retain(|v| {
|
|
||||||
// keep timestamps that are less than a minute away
|
|
||||||
cur_ts.saturating_sub(*v) < TimestampDuration::new(60_000_000u64)
|
|
||||||
});
|
|
||||||
if value.is_empty() {
|
|
||||||
dead_keys.push(*key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for key in dead_keys {
|
|
||||||
self.conn_timestamps_by_ip4.remove(&key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// v6
|
|
||||||
{
|
|
||||||
let mut dead_keys = Vec::<Ipv6Addr>::new();
|
|
||||||
for (key, value) in &mut self.conn_timestamps_by_ip6_prefix {
|
|
||||||
value.retain(|v| {
|
|
||||||
// keep timestamps that are less than a minute away
|
|
||||||
cur_ts.saturating_sub(*v) < TimestampDuration::new(60_000_000u64)
|
|
||||||
});
|
|
||||||
if value.is_empty() {
|
|
||||||
dead_keys.push(*key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for key in dead_keys {
|
|
||||||
self.conn_timestamps_by_ip6_prefix.remove(&key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn add(&mut self, addr: IpAddr) -> Result<(), AddressFilterError> {
|
|
||||||
let ipblock = ip_to_ipblock(self.max_connections_per_ip6_prefix_size, addr);
|
|
||||||
let ts = get_aligned_timestamp();
|
|
||||||
|
|
||||||
self.purge_old_timestamps(ts);
|
|
||||||
|
|
||||||
match ipblock {
|
|
||||||
IpAddr::V4(v4) => {
|
|
||||||
// See if we have too many connections from this ip block
|
|
||||||
let cnt = &mut *self.conn_count_by_ip4.entry(v4).or_default();
|
|
||||||
assert!(*cnt <= self.max_connections_per_ip4);
|
|
||||||
if *cnt == self.max_connections_per_ip4 {
|
|
||||||
warn!("address filter count exceeded: {:?}", v4);
|
|
||||||
return Err(AddressFilterError::CountExceeded);
|
|
||||||
}
|
|
||||||
// See if this ip block has connected too frequently
|
|
||||||
let tstamps = &mut self.conn_timestamps_by_ip4.entry(v4).or_default();
|
|
||||||
tstamps.retain(|v| {
|
|
||||||
// keep timestamps that are less than a minute away
|
|
||||||
ts.saturating_sub(*v) < TimestampDuration::new(60_000_000u64)
|
|
||||||
});
|
|
||||||
assert!(tstamps.len() <= self.max_connection_frequency_per_min);
|
|
||||||
if tstamps.len() == self.max_connection_frequency_per_min {
|
|
||||||
warn!("address filter rate exceeded: {:?}", v4);
|
|
||||||
return Err(AddressFilterError::RateExceeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If it's okay, add the counts and timestamps
|
|
||||||
*cnt += 1;
|
|
||||||
tstamps.push(ts);
|
|
||||||
}
|
|
||||||
IpAddr::V6(v6) => {
|
|
||||||
// See if we have too many connections from this ip block
|
|
||||||
let cnt = &mut *self.conn_count_by_ip6_prefix.entry(v6).or_default();
|
|
||||||
assert!(*cnt <= self.max_connections_per_ip6_prefix);
|
|
||||||
if *cnt == self.max_connections_per_ip6_prefix {
|
|
||||||
warn!("address filter count exceeded: {:?}", v6);
|
|
||||||
return Err(AddressFilterError::CountExceeded);
|
|
||||||
}
|
|
||||||
// See if this ip block has connected too frequently
|
|
||||||
let tstamps = &mut self.conn_timestamps_by_ip6_prefix.entry(v6).or_default();
|
|
||||||
assert!(tstamps.len() <= self.max_connection_frequency_per_min);
|
|
||||||
if tstamps.len() == self.max_connection_frequency_per_min {
|
|
||||||
warn!("address filter rate exceeded: {:?}", v6);
|
|
||||||
return Err(AddressFilterError::RateExceeded);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If it's okay, add the counts and timestamps
|
|
||||||
*cnt += 1;
|
|
||||||
tstamps.push(ts);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn remove(&mut self, addr: IpAddr) -> Result<(), AddressNotInTableError> {
|
|
||||||
let ipblock = ip_to_ipblock(self.max_connections_per_ip6_prefix_size, addr);
|
|
||||||
|
|
||||||
let ts = get_aligned_timestamp();
|
|
||||||
self.purge_old_timestamps(ts);
|
|
||||||
|
|
||||||
match ipblock {
|
|
||||||
IpAddr::V4(v4) => {
|
|
||||||
match self.conn_count_by_ip4.entry(v4) {
|
|
||||||
Entry::Vacant(_) => {
|
|
||||||
return Err(AddressNotInTableError {});
|
|
||||||
}
|
|
||||||
Entry::Occupied(mut o) => {
|
|
||||||
let cnt = o.get_mut();
|
|
||||||
assert!(*cnt > 0);
|
|
||||||
if *cnt == 0 {
|
|
||||||
self.conn_count_by_ip4.remove(&v4);
|
|
||||||
} else {
|
|
||||||
*cnt -= 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
IpAddr::V6(v6) => {
|
|
||||||
match self.conn_count_by_ip6_prefix.entry(v6) {
|
|
||||||
Entry::Vacant(_) => {
|
|
||||||
return Err(AddressNotInTableError {});
|
|
||||||
}
|
|
||||||
Entry::Occupied(mut o) => {
|
|
||||||
let cnt = o.get_mut();
|
|
||||||
assert!(*cnt > 0);
|
|
||||||
if *cnt == 0 {
|
|
||||||
self.conn_count_by_ip6_prefix.remove(&v6);
|
|
||||||
} else {
|
|
||||||
*cnt -= 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
@ -286,6 +286,7 @@ impl ConnectionManager {
|
|||||||
local_addr,
|
local_addr,
|
||||||
&dial_info,
|
&dial_info,
|
||||||
self.arc.connection_initial_timeout_ms,
|
self.arc.connection_initial_timeout_ms,
|
||||||
|
self.network_manager().address_filter(),
|
||||||
)
|
)
|
||||||
.await;
|
.await;
|
||||||
match result_net_res {
|
match result_net_res {
|
||||||
|
@ -29,7 +29,7 @@ pub struct ConnectionTableInner {
|
|||||||
protocol_index_by_id: BTreeMap<NetworkConnectionId, usize>,
|
protocol_index_by_id: BTreeMap<NetworkConnectionId, usize>,
|
||||||
id_by_descriptor: BTreeMap<ConnectionDescriptor, NetworkConnectionId>,
|
id_by_descriptor: BTreeMap<ConnectionDescriptor, NetworkConnectionId>,
|
||||||
ids_by_remote: BTreeMap<PeerAddress, Vec<NetworkConnectionId>>,
|
ids_by_remote: BTreeMap<PeerAddress, Vec<NetworkConnectionId>>,
|
||||||
address_filter: ConnectionLimits,
|
address_filter: AddressFilter,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
@ -58,7 +58,7 @@ impl ConnectionTable {
|
|||||||
protocol_index_by_id: BTreeMap::new(),
|
protocol_index_by_id: BTreeMap::new(),
|
||||||
id_by_descriptor: BTreeMap::new(),
|
id_by_descriptor: BTreeMap::new(),
|
||||||
ids_by_remote: BTreeMap::new(),
|
ids_by_remote: BTreeMap::new(),
|
||||||
address_filter: ConnectionLimits::new(config),
|
address_filter: AddressFilter::new(config),
|
||||||
})),
|
})),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -125,7 +125,7 @@ impl ConnectionTable {
|
|||||||
|
|
||||||
// Filter by ip for connection limits
|
// Filter by ip for connection limits
|
||||||
let ip_addr = descriptor.remote_address().to_ip_addr();
|
let ip_addr = descriptor.remote_address().to_ip_addr();
|
||||||
match inner.address_filter.add(ip_addr) {
|
match inner.address_filter.add_connection(ip_addr) {
|
||||||
Ok(()) => {}
|
Ok(()) => {}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
// Return the connection in the error to be disposed of
|
// Return the connection in the error to be disposed of
|
||||||
@ -258,7 +258,7 @@ impl ConnectionTable {
|
|||||||
let ip_addr = remote.to_socket_addr().ip();
|
let ip_addr = remote.to_socket_addr().ip();
|
||||||
inner
|
inner
|
||||||
.address_filter
|
.address_filter
|
||||||
.remove(ip_addr)
|
.remove_connection(ip_addr)
|
||||||
.expect("Inconsistency in connection table");
|
.expect("Inconsistency in connection table");
|
||||||
conn
|
conn
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,7 @@ mod wasm;
|
|||||||
mod direct_boot;
|
mod direct_boot;
|
||||||
mod send_data;
|
mod send_data;
|
||||||
mod connection_handle;
|
mod connection_handle;
|
||||||
mod connection_limits;
|
mod address_filter;
|
||||||
mod connection_manager;
|
mod connection_manager;
|
||||||
mod connection_table;
|
mod connection_table;
|
||||||
mod network_connection;
|
mod network_connection;
|
||||||
@ -29,7 +29,7 @@ pub use stats::*;
|
|||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////
|
||||||
use connection_handle::*;
|
use connection_handle::*;
|
||||||
use connection_limits::*;
|
use address_filter::*;
|
||||||
use crypto::*;
|
use crypto::*;
|
||||||
use futures_util::stream::FuturesUnordered;
|
use futures_util::stream::FuturesUnordered;
|
||||||
use hashlink::LruCache;
|
use hashlink::LruCache;
|
||||||
@ -54,6 +54,7 @@ pub const PUBLIC_ADDRESS_CHECK_CACHE_SIZE: usize = 8;
|
|||||||
pub const PUBLIC_ADDRESS_CHECK_TASK_INTERVAL_SECS: u32 = 60;
|
pub const PUBLIC_ADDRESS_CHECK_TASK_INTERVAL_SECS: u32 = 60;
|
||||||
pub const PUBLIC_ADDRESS_INCONSISTENCY_TIMEOUT_US: TimestampDuration = TimestampDuration::new(300_000_000u64); // 5 minutes
|
pub const PUBLIC_ADDRESS_INCONSISTENCY_TIMEOUT_US: TimestampDuration = TimestampDuration::new(300_000_000u64); // 5 minutes
|
||||||
pub const PUBLIC_ADDRESS_INCONSISTENCY_PUNISHMENT_TIMEOUT_US: TimestampDuration = TimestampDuration::new(3600_000_000u64); // 60 minutes
|
pub const PUBLIC_ADDRESS_INCONSISTENCY_PUNISHMENT_TIMEOUT_US: TimestampDuration = TimestampDuration::new(3600_000_000u64); // 60 minutes
|
||||||
|
pub const ADDRESS_FILTER_TASK_INTERVAL_SECS: u32 = 60;
|
||||||
pub const BOOT_MAGIC: &[u8; 4] = b"BOOT";
|
pub const BOOT_MAGIC: &[u8; 4] = b"BOOT";
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, Default)]
|
#[derive(Copy, Clone, Debug, Default)]
|
||||||
@ -136,6 +137,7 @@ struct NetworkManagerUnlockedInner {
|
|||||||
#[cfg(feature="unstable-blockstore")]
|
#[cfg(feature="unstable-blockstore")]
|
||||||
block_store: BlockStore,
|
block_store: BlockStore,
|
||||||
crypto: Crypto,
|
crypto: Crypto,
|
||||||
|
address_filter: AddressFilter,
|
||||||
// Accessors
|
// Accessors
|
||||||
routing_table: RwLock<Option<RoutingTable>>,
|
routing_table: RwLock<Option<RoutingTable>>,
|
||||||
components: RwLock<Option<NetworkComponents>>,
|
components: RwLock<Option<NetworkComponents>>,
|
||||||
@ -143,6 +145,7 @@ struct NetworkManagerUnlockedInner {
|
|||||||
// Background processes
|
// Background processes
|
||||||
rolling_transfers_task: TickTask<EyreReport>,
|
rolling_transfers_task: TickTask<EyreReport>,
|
||||||
public_address_check_task: TickTask<EyreReport>,
|
public_address_check_task: TickTask<EyreReport>,
|
||||||
|
address_filter_task: TickTask<EyreReport>,
|
||||||
// Network Key
|
// Network Key
|
||||||
network_key: Option<SharedSecret>,
|
network_key: Option<SharedSecret>,
|
||||||
}
|
}
|
||||||
@ -174,18 +177,20 @@ impl NetworkManager {
|
|||||||
network_key: Option<SharedSecret>,
|
network_key: Option<SharedSecret>,
|
||||||
) -> NetworkManagerUnlockedInner {
|
) -> NetworkManagerUnlockedInner {
|
||||||
NetworkManagerUnlockedInner {
|
NetworkManagerUnlockedInner {
|
||||||
config,
|
config: config.clone(),
|
||||||
storage_manager,
|
storage_manager,
|
||||||
protected_store,
|
protected_store,
|
||||||
table_store,
|
table_store,
|
||||||
#[cfg(feature="unstable-blockstore")]
|
#[cfg(feature="unstable-blockstore")]
|
||||||
block_store,
|
block_store,
|
||||||
crypto,
|
crypto,
|
||||||
|
address_filter: AddressFilter::new(config),
|
||||||
routing_table: RwLock::new(None),
|
routing_table: RwLock::new(None),
|
||||||
components: RwLock::new(None),
|
components: RwLock::new(None),
|
||||||
update_callback: RwLock::new(None),
|
update_callback: RwLock::new(None),
|
||||||
rolling_transfers_task: TickTask::new(ROLLING_TRANSFERS_INTERVAL_SECS),
|
rolling_transfers_task: TickTask::new(ROLLING_TRANSFERS_INTERVAL_SECS),
|
||||||
public_address_check_task: TickTask::new(PUBLIC_ADDRESS_CHECK_TASK_INTERVAL_SECS),
|
public_address_check_task: TickTask::new(PUBLIC_ADDRESS_CHECK_TASK_INTERVAL_SECS),
|
||||||
|
address_filter_task: TickTask::new(ADDRESS_FILTER_TASK_INTERVAL_SECS),
|
||||||
network_key,
|
network_key,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -273,6 +278,9 @@ impl NetworkManager {
|
|||||||
pub fn crypto(&self) -> Crypto {
|
pub fn crypto(&self) -> Crypto {
|
||||||
self.unlocked_inner.crypto.clone()
|
self.unlocked_inner.crypto.clone()
|
||||||
}
|
}
|
||||||
|
pub fn address_filter(&self) -> AddressFilter {
|
||||||
|
self.unlocked_inner.address_filter.clone()
|
||||||
|
}
|
||||||
pub fn routing_table(&self) -> RoutingTable {
|
pub fn routing_table(&self) -> RoutingTable {
|
||||||
self.unlocked_inner
|
self.unlocked_inner
|
||||||
.routing_table
|
.routing_table
|
||||||
@ -894,10 +902,11 @@ impl NetworkManager {
|
|||||||
data.len(),
|
data.len(),
|
||||||
connection_descriptor
|
connection_descriptor
|
||||||
);
|
);
|
||||||
|
let remote_addr = connection_descriptor.remote_address().to_ip_addr();
|
||||||
|
|
||||||
// Network accounting
|
// Network accounting
|
||||||
self.stats_packet_rcvd(
|
self.stats_packet_rcvd(
|
||||||
connection_descriptor.remote_address().to_ip_addr(),
|
remote_addr,
|
||||||
ByteCount::new(data.len() as u64),
|
ByteCount::new(data.len() as u64),
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -911,6 +920,7 @@ impl NetworkManager {
|
|||||||
// Ensure we can read the magic number
|
// Ensure we can read the magic number
|
||||||
if data.len() < 4 {
|
if data.len() < 4 {
|
||||||
log_net!(debug "short packet");
|
log_net!(debug "short packet");
|
||||||
|
self.address_filter().punish(remote_addr);
|
||||||
return Ok(false);
|
return Ok(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -943,6 +953,7 @@ impl NetworkManager {
|
|||||||
Ok(v) => v,
|
Ok(v) => v,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log_net!(debug "envelope failed to decode: {}", e);
|
log_net!(debug "envelope failed to decode: {}", e);
|
||||||
|
self.address_filter().punish(remote_addr);
|
||||||
return Ok(false);
|
return Ok(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -1058,7 +1069,7 @@ impl NetworkManager {
|
|||||||
Ok(v) => v,
|
Ok(v) => v,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log_net!(debug "failed to decrypt envelope body: {}",e);
|
log_net!(debug "failed to decrypt envelope body: {}",e);
|
||||||
// xxx: punish nodes that send messages that fail to decrypt eventually
|
self.address_filter().punish(remote_addr);
|
||||||
return Ok(false);
|
return Ok(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -1078,8 +1089,6 @@ impl NetworkManager {
|
|||||||
};
|
};
|
||||||
source_noderef.add_envelope_version(envelope.get_version());
|
source_noderef.add_envelope_version(envelope.get_version());
|
||||||
|
|
||||||
// xxx: deal with spoofing and flooding here?
|
|
||||||
|
|
||||||
// Pass message to RPC system
|
// Pass message to RPC system
|
||||||
rpc.enqueue_direct_message(
|
rpc.enqueue_direct_message(
|
||||||
envelope,
|
envelope,
|
||||||
|
@ -370,6 +370,14 @@ impl Network {
|
|||||||
c.network.connection_initial_timeout_ms
|
c.network.connection_initial_timeout_ms
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if self
|
||||||
|
.network_manager()
|
||||||
|
.address_filter()
|
||||||
|
.is_punished(dial_info.address().to_ip_addr())
|
||||||
|
{
|
||||||
|
return Ok(NetworkResult::no_connection_other("punished"));
|
||||||
|
}
|
||||||
|
|
||||||
match dial_info.protocol_type() {
|
match dial_info.protocol_type() {
|
||||||
ProtocolType::UDP => {
|
ProtocolType::UDP => {
|
||||||
let peer_socket_addr = dial_info.to_socket_addr();
|
let peer_socket_addr = dial_info.to_socket_addr();
|
||||||
@ -429,6 +437,14 @@ impl Network {
|
|||||||
c.network.connection_initial_timeout_ms
|
c.network.connection_initial_timeout_ms
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if self
|
||||||
|
.network_manager()
|
||||||
|
.address_filter()
|
||||||
|
.is_punished(dial_info.address().to_ip_addr())
|
||||||
|
{
|
||||||
|
return Ok(NetworkResult::no_connection_other("punished"));
|
||||||
|
}
|
||||||
|
|
||||||
match dial_info.protocol_type() {
|
match dial_info.protocol_type() {
|
||||||
ProtocolType::UDP => {
|
ProtocolType::UDP => {
|
||||||
let peer_socket_addr = dial_info.to_socket_addr();
|
let peer_socket_addr = dial_info.to_socket_addr();
|
||||||
|
@ -108,12 +108,29 @@ impl Network {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// XXX
|
// Limit the number of connections from the same IP address
|
||||||
// warn!(
|
// and the number of total connections
|
||||||
// "DEBUGACCEPT: local={} remote={}",
|
// XXX limiting here instead for connection table? may be faster and avoids tls negotiation
|
||||||
// tcp_stream.local_addr().unwrap(),
|
let peer_addr = match tcp_stream.peer_addr() {
|
||||||
// tcp_stream.peer_addr().unwrap(),
|
Ok(addr) => addr,
|
||||||
// );
|
Err(e) => {
|
||||||
|
log_net!(debug "failed to get peer address: {}", e);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let address_filter = self.network_manager().address_filter();
|
||||||
|
// Check to see if it is punished
|
||||||
|
if address_filter.is_punished(peer_addr.ip()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let local_addr = match tcp_stream.local_addr() {
|
||||||
|
Ok(addr) => addr,
|
||||||
|
Err(e) => {
|
||||||
|
log_net!(debug "failed to get local address: {}", e);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
if let Err(e) = tcp_stream.set_linger(Some(core::time::Duration::from_secs(0))) {
|
if let Err(e) = tcp_stream.set_linger(Some(core::time::Duration::from_secs(0))) {
|
||||||
log_net!(debug "Couldn't set TCP linger: {}", e);
|
log_net!(debug "Couldn't set TCP linger: {}", e);
|
||||||
@ -127,24 +144,6 @@ impl Network {
|
|||||||
let listener_state = listener_state.clone();
|
let listener_state = listener_state.clone();
|
||||||
let connection_manager = connection_manager.clone();
|
let connection_manager = connection_manager.clone();
|
||||||
|
|
||||||
// Limit the number of connections from the same IP address
|
|
||||||
// and the number of total connections
|
|
||||||
let peer_addr = match tcp_stream.peer_addr() {
|
|
||||||
Ok(addr) => addr,
|
|
||||||
Err(e) => {
|
|
||||||
log_net!(debug "failed to get peer address: {}", e);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
let local_addr = match tcp_stream.local_addr() {
|
|
||||||
Ok(addr) => addr,
|
|
||||||
Err(e) => {
|
|
||||||
log_net!(debug "failed to get local address: {}", e);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
// XXX limiting here instead for connection table? may be faster and avoids tls negotiation
|
|
||||||
|
|
||||||
log_net!("TCP connection from: {}", peer_addr);
|
log_net!("TCP connection from: {}", peer_addr);
|
||||||
|
|
||||||
// Create a stream we can peek on
|
// Create a stream we can peek on
|
||||||
|
@ -66,8 +66,6 @@ impl Network {
|
|||||||
.await
|
.await
|
||||||
{
|
{
|
||||||
Ok(Ok((size, descriptor))) => {
|
Ok(Ok((size, descriptor))) => {
|
||||||
// XXX: Limit the number of packets from the same IP address?
|
|
||||||
|
|
||||||
// Network accounting
|
// Network accounting
|
||||||
network_manager.stats_packet_rcvd(
|
network_manager.stats_packet_rcvd(
|
||||||
descriptor.remote_address().to_ip_addr(),
|
descriptor.remote_address().to_ip_addr(),
|
||||||
@ -143,7 +141,10 @@ impl Network {
|
|||||||
let socket_arc = Arc::new(udp_socket);
|
let socket_arc = Arc::new(udp_socket);
|
||||||
|
|
||||||
// Create protocol handler
|
// Create protocol handler
|
||||||
let udpv4_handler = RawUdpProtocolHandler::new(socket_arc);
|
let udpv4_handler = RawUdpProtocolHandler::new(
|
||||||
|
socket_arc,
|
||||||
|
Some(self.network_manager().address_filter()),
|
||||||
|
);
|
||||||
|
|
||||||
inner.outbound_udpv4_protocol_handler = Some(udpv4_handler);
|
inner.outbound_udpv4_protocol_handler = Some(udpv4_handler);
|
||||||
}
|
}
|
||||||
@ -164,7 +165,10 @@ impl Network {
|
|||||||
let socket_arc = Arc::new(udp_socket);
|
let socket_arc = Arc::new(udp_socket);
|
||||||
|
|
||||||
// Create protocol handler
|
// Create protocol handler
|
||||||
let udpv6_handler = RawUdpProtocolHandler::new(socket_arc);
|
let udpv6_handler = RawUdpProtocolHandler::new(
|
||||||
|
socket_arc,
|
||||||
|
Some(self.network_manager().address_filter()),
|
||||||
|
);
|
||||||
|
|
||||||
inner.outbound_udpv6_protocol_handler = Some(udpv6_handler);
|
inner.outbound_udpv6_protocol_handler = Some(udpv6_handler);
|
||||||
}
|
}
|
||||||
@ -191,7 +195,8 @@ impl Network {
|
|||||||
let socket_arc = Arc::new(udp_socket);
|
let socket_arc = Arc::new(udp_socket);
|
||||||
|
|
||||||
// Create protocol handler
|
// Create protocol handler
|
||||||
let protocol_handler = RawUdpProtocolHandler::new(socket_arc);
|
let protocol_handler =
|
||||||
|
RawUdpProtocolHandler::new(socket_arc, Some(self.network_manager().address_filter()));
|
||||||
|
|
||||||
// Create message_handler records
|
// Create message_handler records
|
||||||
self.inner
|
self.inner
|
||||||
|
@ -22,7 +22,11 @@ impl ProtocolNetworkConnection {
|
|||||||
local_address: Option<SocketAddr>,
|
local_address: Option<SocketAddr>,
|
||||||
dial_info: &DialInfo,
|
dial_info: &DialInfo,
|
||||||
timeout_ms: u32,
|
timeout_ms: u32,
|
||||||
|
address_filter: AddressFilter,
|
||||||
) -> io::Result<NetworkResult<ProtocolNetworkConnection>> {
|
) -> io::Result<NetworkResult<ProtocolNetworkConnection>> {
|
||||||
|
if address_filter.is_punished(dial_info.address().to_ip_addr()) {
|
||||||
|
return Ok(NetworkResult::no_connection_other("punished"));
|
||||||
|
}
|
||||||
match dial_info.protocol_type() {
|
match dial_info.protocol_type() {
|
||||||
ProtocolType::UDP => {
|
ProtocolType::UDP => {
|
||||||
panic!("Should not connect to UDP dialinfo");
|
panic!("Should not connect to UDP dialinfo");
|
||||||
|
@ -5,13 +5,15 @@ use sockets::*;
|
|||||||
pub struct RawUdpProtocolHandler {
|
pub struct RawUdpProtocolHandler {
|
||||||
socket: Arc<UdpSocket>,
|
socket: Arc<UdpSocket>,
|
||||||
assembly_buffer: AssemblyBuffer,
|
assembly_buffer: AssemblyBuffer,
|
||||||
|
address_filter: Option<AddressFilter>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl RawUdpProtocolHandler {
|
impl RawUdpProtocolHandler {
|
||||||
pub fn new(socket: Arc<UdpSocket>) -> Self {
|
pub fn new(socket: Arc<UdpSocket>, address_filter: Option<AddressFilter>) -> Self {
|
||||||
Self {
|
Self {
|
||||||
socket,
|
socket,
|
||||||
assembly_buffer: AssemblyBuffer::new(),
|
assembly_buffer: AssemblyBuffer::new(),
|
||||||
|
address_filter,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -21,6 +23,13 @@ impl RawUdpProtocolHandler {
|
|||||||
// Get a packet
|
// Get a packet
|
||||||
let (size, remote_addr) = network_result_value_or_log!(self.socket.recv_from(data).await.into_network_result()? => continue);
|
let (size, remote_addr) = network_result_value_or_log!(self.socket.recv_from(data).await.into_network_result()? => continue);
|
||||||
|
|
||||||
|
// Check to see if it is punished
|
||||||
|
if let Some(af) = self.address_filter.as_ref() {
|
||||||
|
if af.is_punished(remote_addr.ip()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Insert into assembly buffer
|
// Insert into assembly buffer
|
||||||
let Some(message) = self.assembly_buffer.insert_frame(&data[0..size], remote_addr) else {
|
let Some(message) = self.assembly_buffer.insert_frame(&data[0..size], remote_addr) else {
|
||||||
continue;
|
continue;
|
||||||
@ -66,6 +75,13 @@ impl RawUdpProtocolHandler {
|
|||||||
bail_io_error_other!("sending too large UDP message");
|
bail_io_error_other!("sending too large UDP message");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check to see if it is punished
|
||||||
|
if let Some(af) = self.address_filter.as_ref() {
|
||||||
|
if af.is_punished(remote_addr.ip()) {
|
||||||
|
return Ok(NetworkResult::no_connection_other("punished"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Fragment and send
|
// Fragment and send
|
||||||
let sender = |framed_chunk: Vec<u8>, remote_addr: SocketAddr| async move {
|
let sender = |framed_chunk: Vec<u8>, remote_addr: SocketAddr| async move {
|
||||||
let len = network_result_try!(self
|
let len = network_result_try!(self
|
||||||
@ -111,6 +127,6 @@ impl RawUdpProtocolHandler {
|
|||||||
// get local wildcard address for bind
|
// get local wildcard address for bind
|
||||||
let local_socket_addr = compatible_unspecified_socket_addr(&socket_addr);
|
let local_socket_addr = compatible_unspecified_socket_addr(&socket_addr);
|
||||||
let socket = UdpSocket::bind(local_socket_addr).await?;
|
let socket = UdpSocket::bind(local_socket_addr).await?;
|
||||||
Ok(RawUdpProtocolHandler::new(Arc::new(socket)))
|
Ok(RawUdpProtocolHandler::new(Arc::new(socket), None))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -609,7 +609,7 @@ impl Network {
|
|||||||
ip_addrs,
|
ip_addrs,
|
||||||
tcp_port,
|
tcp_port,
|
||||||
false,
|
false,
|
||||||
Box::new(move |c, _| Box::new(RawTcpProtocolHandler::new(c))),
|
Box::new(|c, _| Box::new(RawTcpProtocolHandler::new(c))),
|
||||||
)
|
)
|
||||||
.await?;
|
.await?;
|
||||||
trace!("TCP: listener started on {:#?}", socket_addresses);
|
trace!("TCP: listener started on {:#?}", socket_addresses);
|
||||||
|
@ -42,6 +42,20 @@ impl NetworkManager {
|
|||||||
)
|
)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Set address filter task
|
||||||
|
{
|
||||||
|
let this = self.clone();
|
||||||
|
self.unlocked_inner
|
||||||
|
.address_filter_task
|
||||||
|
.set_routine(move |s, l, t| {
|
||||||
|
Box::pin(
|
||||||
|
this.address_filter()
|
||||||
|
.address_filter_task_routine(s, Timestamp::new(l), Timestamp::new(t))
|
||||||
|
.instrument(trace_span!(parent: None, "address filter task routine")),
|
||||||
|
)
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn tick(&self) -> EyreResult<()> {
|
pub async fn tick(&self) -> EyreResult<()> {
|
||||||
|
@ -91,6 +91,14 @@ impl Network {
|
|||||||
c.network.connection_initial_timeout_ms
|
c.network.connection_initial_timeout_ms
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if self
|
||||||
|
.network_manager()
|
||||||
|
.address_filter()
|
||||||
|
.is_punished(dial_info.address().to_ip_addr())
|
||||||
|
{
|
||||||
|
return Ok(NetworkResult::no_connection_other("punished"));
|
||||||
|
}
|
||||||
|
|
||||||
match dial_info.protocol_type() {
|
match dial_info.protocol_type() {
|
||||||
ProtocolType::UDP => {
|
ProtocolType::UDP => {
|
||||||
bail!("no support for UDP protocol")
|
bail!("no support for UDP protocol")
|
||||||
@ -132,6 +140,14 @@ impl Network {
|
|||||||
c.network.connection_initial_timeout_ms
|
c.network.connection_initial_timeout_ms
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if self
|
||||||
|
.network_manager()
|
||||||
|
.address_filter()
|
||||||
|
.is_punished(dial_info.address().to_ip_addr())
|
||||||
|
{
|
||||||
|
return Ok(NetworkResult::no_connection_other("punished"));
|
||||||
|
}
|
||||||
|
|
||||||
match dial_info.protocol_type() {
|
match dial_info.protocol_type() {
|
||||||
ProtocolType::UDP => {
|
ProtocolType::UDP => {
|
||||||
bail!("no support for UDP protocol")
|
bail!("no support for UDP protocol")
|
||||||
|
@ -17,7 +17,11 @@ impl ProtocolNetworkConnection {
|
|||||||
_local_address: Option<SocketAddr>,
|
_local_address: Option<SocketAddr>,
|
||||||
dial_info: &DialInfo,
|
dial_info: &DialInfo,
|
||||||
timeout_ms: u32,
|
timeout_ms: u32,
|
||||||
|
address_filter: AddressFiltter,
|
||||||
) -> io::Result<NetworkResult<ProtocolNetworkConnection>> {
|
) -> io::Result<NetworkResult<ProtocolNetworkConnection>> {
|
||||||
|
if address_filter.is_punished(dial_info.address().to_ip_addr()) {
|
||||||
|
return Ok(NetworkResult::no_connection_other("punished"));
|
||||||
|
}
|
||||||
match dial_info.protocol_type() {
|
match dial_info.protocol_type() {
|
||||||
ProtocolType::UDP => {
|
ProtocolType::UDP => {
|
||||||
panic!("UDP dial info is not supported on WASM targets");
|
panic!("UDP dial info is not supported on WASM targets");
|
||||||
|
@ -74,6 +74,7 @@ impl RPCProcessor {
|
|||||||
vcrypto: vcrypto.clone(),
|
vcrypto: vcrypto.clone(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
#[cfg(feature="debug-dht")]
|
||||||
log_rpc!(debug "{}", debug_string);
|
log_rpc!(debug "{}", debug_string);
|
||||||
|
|
||||||
let waitable_reply = network_result_try!(
|
let waitable_reply = network_result_try!(
|
||||||
@ -210,29 +211,32 @@ impl RPCProcessor {
|
|||||||
.await
|
.await
|
||||||
.map_err(RPCError::internal)?);
|
.map_err(RPCError::internal)?);
|
||||||
|
|
||||||
let debug_string_value = subkey_result.value.as_ref().map(|v| {
|
#[cfg(feature="debug-dht")]
|
||||||
format!(" len={} seq={} writer={}",
|
{
|
||||||
v.value_data().data().len(),
|
let debug_string_value = subkey_result.value.as_ref().map(|v| {
|
||||||
v.value_data().seq(),
|
format!(" len={} seq={} writer={}",
|
||||||
v.value_data().writer(),
|
v.value_data().data().len(),
|
||||||
)
|
v.value_data().seq(),
|
||||||
}).unwrap_or_default();
|
v.value_data().writer(),
|
||||||
|
)
|
||||||
|
}).unwrap_or_default();
|
||||||
|
|
||||||
let debug_string_answer = format!(
|
let debug_string_answer = format!(
|
||||||
"IN ===> GetValueA({} #{}{}{} peers={}) ==> {}",
|
"IN ===> GetValueA({} #{}{}{} peers={}) ==> {}",
|
||||||
key,
|
key,
|
||||||
subkey,
|
subkey,
|
||||||
debug_string_value,
|
debug_string_value,
|
||||||
if subkey_result.descriptor.is_some() {
|
if subkey_result.descriptor.is_some() {
|
||||||
" +desc"
|
" +desc"
|
||||||
} else {
|
} else {
|
||||||
""
|
""
|
||||||
},
|
},
|
||||||
closer_to_key_peers.len(),
|
closer_to_key_peers.len(),
|
||||||
msg.header.direct_sender_node_id()
|
msg.header.direct_sender_node_id()
|
||||||
);
|
);
|
||||||
|
|
||||||
log_rpc!(debug "{}", debug_string_answer);
|
log_rpc!(debug "{}", debug_string_answer);
|
||||||
|
}
|
||||||
|
|
||||||
// Make GetValue answer
|
// Make GetValue answer
|
||||||
let get_value_a = RPCOperationGetValueA::new(
|
let get_value_a = RPCOperationGetValueA::new(
|
||||||
|
@ -88,6 +88,7 @@ impl RPCProcessor {
|
|||||||
vcrypto: vcrypto.clone(),
|
vcrypto: vcrypto.clone(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
#[cfg(feature="debug-dht")]
|
||||||
log_rpc!(debug "{}", debug_string);
|
log_rpc!(debug "{}", debug_string);
|
||||||
|
|
||||||
let waitable_reply = network_result_try!(
|
let waitable_reply = network_result_try!(
|
||||||
@ -239,29 +240,32 @@ impl RPCProcessor {
|
|||||||
(true, new_value)
|
(true, new_value)
|
||||||
};
|
};
|
||||||
|
|
||||||
let debug_string_value = new_value.as_ref().map(|v| {
|
#[cfg(feature="debug-dht")]
|
||||||
format!(" len={} seq={} writer={}",
|
{
|
||||||
v.value_data().data().len(),
|
let debug_string_value = new_value.as_ref().map(|v| {
|
||||||
v.value_data().seq(),
|
format!(" len={} seq={} writer={}",
|
||||||
v.value_data().writer(),
|
v.value_data().data().len(),
|
||||||
)
|
v.value_data().seq(),
|
||||||
}).unwrap_or_default();
|
v.value_data().writer(),
|
||||||
|
)
|
||||||
|
}).unwrap_or_default();
|
||||||
|
|
||||||
let debug_string_answer = format!(
|
let debug_string_answer = format!(
|
||||||
"IN ===> SetValueA({} #{}{}{} peers={}) ==> {}",
|
"IN ===> SetValueA({} #{}{}{} peers={}) ==> {}",
|
||||||
key,
|
key,
|
||||||
subkey,
|
subkey,
|
||||||
if set {
|
if set {
|
||||||
" +set"
|
" +set"
|
||||||
} else {
|
} else {
|
||||||
""
|
""
|
||||||
},
|
},
|
||||||
debug_string_value,
|
debug_string_value,
|
||||||
closer_to_key_peers.len(),
|
closer_to_key_peers.len(),
|
||||||
msg.header.direct_sender_node_id()
|
msg.header.direct_sender_node_id()
|
||||||
);
|
);
|
||||||
|
|
||||||
log_rpc!(debug "{}", debug_string_answer);
|
log_rpc!(debug "{}", debug_string_answer);
|
||||||
|
}
|
||||||
|
|
||||||
// Make SetValue answer
|
// Make SetValue answer
|
||||||
let set_value_a = RPCOperationSetValueA::new(set, new_value, closer_to_key_peers)?;
|
let set_value_a = RPCOperationSetValueA::new(set, new_value, closer_to_key_peers)?;
|
||||||
|
Loading…
Reference in New Issue
Block a user