p2panda-discovery 0.5.2

Confidential topic and node discovery protocol
Documentation
// SPDX-License-Identifier: MIT OR Apache-2.0

use std::collections::{BTreeMap, HashSet};
use std::convert::Infallible;
use std::hash::Hash as StdHash;

use rand::Rng;
use rand_chacha::ChaCha20Rng;
use serde::{Deserialize, Serialize};

use crate::DiscoveryResult;
use crate::address_book::NodeInfo;
use crate::address_book::memory::{MemoryStore, current_timestamp};
use crate::traits::LocalTopics;

pub type TestId = usize;

#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, StdHash)]
pub struct TestInfo {
    pub id: TestId,
    pub bootstrap: bool,
    pub transports: Option<TestTransportInfo>,
}

#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, StdHash, Serialize, Deserialize)]
pub struct TestTransportInfo {
    address: String,
    timestamp: u64,
}

impl TestTransportInfo {
    pub fn new(address: &str) -> Self {
        Self {
            address: address.to_owned(),
            timestamp: current_timestamp(),
        }
    }
}

impl TestInfo {
    pub fn new(id: TestId) -> Self {
        Self {
            id,
            bootstrap: false,
            transports: None,
        }
    }

    pub fn new_bootstrap(id: TestId) -> Self {
        Self {
            id,
            bootstrap: true,
            transports: None,
        }
    }

    pub fn with_random_address(mut self, rng: &mut ChaCha20Rng) -> Self {
        self.transports = Some(TestTransportInfo {
            timestamp: current_timestamp(),
            address: {
                // Generate a random, fake IPv4 address
                let segments: [u8; 4] = rng.random();
                segments.map(|s| s.to_string()).join(".")
            },
        });
        self
    }

    /// Returns true if the given transport information is newer than what we have already and it
    /// got updated.
    pub fn update_transports(&mut self, transports: TestTransportInfo) -> bool {
        match self.transports {
            Some(ref current) => {
                if current.timestamp < transports.timestamp {
                    self.transports = Some(transports);
                    return true;
                }
            }
            None => {
                self.transports = Some(transports);
                return true;
            }
        }
        false
    }
}

impl NodeInfo<TestId> for TestInfo {
    type Transports = TestTransportInfo;

    fn id(&self) -> TestId {
        self.id
    }

    fn is_bootstrap(&self) -> bool {
        self.bootstrap
    }

    fn is_stale(&self) -> bool {
        false
    }

    fn transports(&self) -> Option<Self::Transports> {
        self.transports.clone()
    }
}

pub type TestStore<R> = MemoryStore<R, TestId, TestInfo>;

#[derive(Clone, Default, Debug)]
pub struct TestSubscription {
    pub topics: HashSet<[u8; 32]>,
}

impl LocalTopics for TestSubscription {
    type Error = Infallible;

    async fn topics(&self) -> Result<HashSet<[u8; 32]>, Self::Error> {
        Ok(self.topics.clone())
    }
}

impl DiscoveryResult<TestId, TestInfo> {
    pub fn from_neighbors(remote_node_id: TestId, node_ids: &[TestId]) -> Self {
        Self {
            remote_node_id,
            transport_infos: BTreeMap::from_iter(
                node_ids
                    .iter()
                    .map(|id| (*id, TestTransportInfo::new("test"))),
            ),
            topics: HashSet::new(),
        }
    }
}