Skip to content

Commit

Permalink
feat(sdk)!: ban addresses failed in sdk (dashpay#2351)
Browse files Browse the repository at this point in the history
Co-authored-by: lklimek <[email protected]>
  • Loading branch information
shumkov and lklimek authored Dec 4, 2024
1 parent d8eb951 commit 30a5aad
Show file tree
Hide file tree
Showing 17 changed files with 245 additions and 194 deletions.
1 change: 1 addition & 0 deletions packages/rs-dapi-client/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -37,5 +37,6 @@ lru = { version = "0.12.3" }
serde = { version = "1.0.197", optional = true, features = ["derive"] }
serde_json = { version = "1.0.120", optional = true }
chrono = { version = "0.4.38", features = ["serde"] }

[dev-dependencies]
tokio = { version = "1.40", features = ["macros"] }
168 changes: 97 additions & 71 deletions packages/rs-dapi-client/src/address_list.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,22 +4,20 @@ use chrono::Utc;
use dapi_grpc::tonic::codegen::http;
use dapi_grpc::tonic::transport::Uri;
use rand::{rngs::SmallRng, seq::IteratorRandom, SeedableRng};
use std::collections::HashSet;
use std::collections::hash_map::Entry;
use std::collections::HashMap;
use std::hash::{Hash, Hasher};
use std::mem;
use std::str::FromStr;
use std::sync::{Arc, RwLock};
use std::time::Duration;

const DEFAULT_BASE_BAN_PERIOD: Duration = Duration::from_secs(60);

/// DAPI address.
#[derive(Debug, Clone, Eq)]
#[cfg_attr(feature = "mocks", derive(serde::Serialize, serde::Deserialize))]
pub struct Address {
ban_count: usize,
banned_until: Option<chrono::DateTime<Utc>>,
#[cfg_attr(feature = "mocks", serde(with = "http_serde::uri"))]
uri: Uri,
}
pub struct Address(#[cfg_attr(feature = "mocks", serde(with = "http_serde::uri"))] Uri);

impl FromStr for Address {
type Err = AddressListError;
Expand All @@ -33,35 +31,46 @@ impl FromStr for Address {

impl PartialEq<Self> for Address {
fn eq(&self, other: &Self) -> bool {
self.uri == other.uri
self.0 == other.0
}
}

impl PartialEq<Uri> for Address {
fn eq(&self, other: &Uri) -> bool {
self.uri == *other
self.0 == *other
}
}

impl Hash for Address {
fn hash<H: Hasher>(&self, state: &mut H) {
self.uri.hash(state);
self.0.hash(state);
}
}

impl From<Uri> for Address {
fn from(uri: Uri) -> Self {
Address {
ban_count: 0,
banned_until: None,
uri,
}
Address(uri)
}
}

impl Address {
/// Get [Uri] of a node.
pub fn uri(&self) -> &Uri {
&self.0
}
}

/// Address status
/// Contains information about the number of bans and the time until the next ban is lifted.
#[derive(Debug, Default, Clone)]
pub struct AddressStatus {
ban_count: usize,
banned_until: Option<chrono::DateTime<Utc>>,
}

impl AddressStatus {
/// Ban the [Address] so it won't be available through [AddressList::get_live_address] for some time.
fn ban(&mut self, base_ban_period: &Duration) {
pub fn ban(&mut self, base_ban_period: &Duration) {
let coefficient = (self.ban_count as f64).exp();
let ban_period = Duration::from_secs_f64(base_ban_period.as_secs_f64() * coefficient);

Expand All @@ -75,24 +84,16 @@ impl Address {
}

/// Clears ban record.
fn unban(&mut self) {
pub fn unban(&mut self) {
self.ban_count = 0;
self.banned_until = None;
}

/// Get [Uri] of a node.
pub fn uri(&self) -> &Uri {
&self.uri
}
}

/// [AddressList] errors
#[derive(Debug, thiserror::Error)]
#[cfg_attr(feature = "mocks", derive(serde::Serialize, serde::Deserialize))]
pub enum AddressListError {
/// Specified address is not present in the list
#[error("address {0} not found in the list")]
AddressNotFound(#[cfg_attr(feature = "mocks", serde(with = "http_serde::uri"))] Uri),
/// A valid uri is required to create an Address
#[error("unable parse address: {0}")]
#[cfg_attr(feature = "mocks", serde(skip))]
Expand All @@ -103,7 +104,7 @@ pub enum AddressListError {
/// for [DapiRequest](crate::DapiRequest) execution.
#[derive(Debug, Clone)]
pub struct AddressList {
addresses: HashSet<Address>,
addresses: Arc<RwLock<HashMap<Address, AddressStatus>>>,
base_ban_period: Duration,
}

Expand All @@ -115,7 +116,7 @@ impl Default for AddressList {

impl std::fmt::Display for Address {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.uri.fmt(f)
self.0.fmt(f)
}
}

Expand All @@ -128,43 +129,70 @@ impl AddressList {
/// Creates an empty [AddressList] with adjustable base ban time.
pub fn with_settings(base_ban_period: Duration) -> Self {
AddressList {
addresses: HashSet::new(),
addresses: Arc::new(RwLock::new(HashMap::new())),
base_ban_period,
}
}

/// Bans address
pub(crate) fn ban_address(&mut self, address: &Address) -> Result<(), AddressListError> {
if !self.addresses.remove(address) {
return Err(AddressListError::AddressNotFound(address.uri.clone()));
};
/// Returns false if the address is not in the list.
pub fn ban(&self, address: &Address) -> bool {
let mut guard = self.addresses.write().unwrap();

let mut banned_address = address.clone();
banned_address.ban(&self.base_ban_period);
let Some(status) = guard.get_mut(address) else {
return false;
};

self.addresses.insert(banned_address);
status.ban(&self.base_ban_period);

Ok(())
true
}

/// Clears address' ban record
pub(crate) fn unban_address(&mut self, address: &Address) -> Result<(), AddressListError> {
if !self.addresses.remove(address) {
return Err(AddressListError::AddressNotFound(address.uri.clone()));
/// Returns false if the address is not in the list.
pub fn unban(&self, address: &Address) -> bool {
let mut guard = self.addresses.write().unwrap();

let Some(status) = guard.get_mut(address) else {
return false;
};

let mut unbanned_address = address.clone();
unbanned_address.unban();
status.unban();

true
}

self.addresses.insert(unbanned_address);
/// Check if the address is banned.
pub fn is_banned(&self, address: &Address) -> bool {
let guard = self.addresses.read().unwrap();

Ok(())
guard
.get(address)
.map(|status| status.is_banned())
.unwrap_or(false)
}

/// Adds a node [Address] to [AddressList]
/// Returns false if the address is already in the list.
pub fn add(&mut self, address: Address) -> bool {
self.addresses.insert(address)
let mut guard = self.addresses.write().unwrap();

match guard.entry(address) {
Entry::Occupied(_) => false,
Entry::Vacant(e) => {
e.insert(AddressStatus::default());

true
}
}
}

/// Remove address from the list
/// Returns [AddressStatus] if the address was in the list.
pub fn remove(&mut self, address: &Address) -> Option<AddressStatus> {
let mut guard = self.addresses.write().unwrap();

guard.remove(address)
}

// TODO: this is the most simple way to add an address
Expand All @@ -173,46 +201,53 @@ impl AddressList {
/// Add a node [Address] to [AddressList] by [Uri].
/// Returns false if the address is already in the list.
pub fn add_uri(&mut self, uri: Uri) -> bool {
self.addresses.insert(uri.into())
self.add(Address::from(uri))
}

/// Randomly select a not banned address.
pub fn get_live_address(&self) -> Option<&Address> {
let mut rng = SmallRng::from_entropy();
pub fn get_live_address(&self) -> Option<Address> {
let guard = self.addresses.read().unwrap();

self.unbanned().into_iter().choose(&mut rng)
}
let mut rng = SmallRng::from_entropy();

/// Get all addresses that are not banned.
fn unbanned(&self) -> Vec<&Address> {
let now = chrono::Utc::now();

self.addresses
guard
.iter()
.filter(|addr| {
addr.banned_until
.filter(|(_, status)| {
status
.banned_until
.map(|banned_until| banned_until < now)
.unwrap_or(true)
})
.collect()
}

/// Get number of available, not banned addresses.
pub fn available(&self) -> usize {
self.unbanned().len()
.choose(&mut rng)
.map(|(addr, _)| addr.clone())
}

/// Get number of all addresses, both banned and not banned.
pub fn len(&self) -> usize {
self.addresses.len()
self.addresses.read().unwrap().len()
}

/// Check if the list is empty.
/// Returns true if there are no addresses in the list.
/// Returns false if there is at least one address in the list.
/// Banned addresses are also counted.
pub fn is_empty(&self) -> bool {
self.addresses.is_empty()
self.addresses.read().unwrap().is_empty()
}
}

impl IntoIterator for AddressList {
type Item = (Address, AddressStatus);
type IntoIter = std::collections::hash_map::IntoIter<Address, AddressStatus>;

fn into_iter(self) -> Self::IntoIter {
let mut guard = self.addresses.write().unwrap();

let addresses_map = mem::take(&mut *guard);

addresses_map.into_iter()
}
}

Expand All @@ -238,12 +273,3 @@ impl FromIterator<Uri> for AddressList {
address_list
}
}

impl IntoIterator for AddressList {
type Item = Address;
type IntoIter = std::collections::hash_set::IntoIter<Address>;

fn into_iter(self) -> Self::IntoIter {
self.addresses.into_iter()
}
}
Loading

0 comments on commit 30a5aad

Please sign in to comment.