summaryrefslogtreecommitdiffstats
path: root/lib/src/gossip/handler.rs
blob: e524da8b23bc2cbd61bf6a3f4c165b6b636b5a44 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
//! Low-level module for gossip'ing code
//!
//! This module implements the low-level gossiping functionality that other modules use to
//! implement actual behaviours on
//!

use std::sync::Arc;

use anyhow::Result;
use futures::Stream;
use futures::StreamExt;
use tokio::sync::RwLock;

use crate::profile::Profile;
use crate::gossip::GossipMessage;

#[derive(Debug)]
pub struct GossipHandler<Strategy = LogStrategy>
    where Strategy: GossipHandlingStrategy + Sync + Send
{
    profile: Arc<RwLock<Profile>>,
    strategy: std::marker::PhantomData<Strategy>,
}

impl<Strat> GossipHandler<Strat>
    where Strat: GossipHandlingStrategy + Sync + Send
{
    pub fn new(profile: Arc<RwLock<Profile>>) -> Self {
        Self {
            profile,
            strategy: std::marker::PhantomData,
        }
    }

    pub fn run<S>(self, input: S) -> impl Stream<Item = (GossipMessage, Result<()>)>
        where S: Stream<Item = (ipfs::PeerId, GossipMessage)>
    {
        input.then(move |(source, msg)| {
            let pr = self.profile.clone();
            async move {
                log::trace!("Received gossip message from {}: {:?}", source, msg);
                let res = Strat::handle_gossip_message(pr.clone(), &source, &msg).await;
                (msg, res)
            }
        })
    }
}

#[async_trait::async_trait]
pub trait GossipHandlingStrategy: Sync + Send {
    async fn handle_gossip_message(profile: Arc<RwLock<Profile>>, source: &ipfs::PeerId, msg: &GossipMessage) -> Result<()>;
}

pub struct LogStrategy;

#[async_trait::async_trait]
impl GossipHandlingStrategy for LogStrategy {
    async fn handle_gossip_message(_profile: Arc<RwLock<Profile>>, source: &ipfs::PeerId, msg: &GossipMessage) -> Result<()> {
        use std::convert::TryFrom;
        use std::ops::Deref;

        match msg {
            GossipMessage::CurrentProfileState { peer_id, cid } => {
                let peer_id = ipfs::PeerId::from_bytes(peer_id);
                let cid = cid::Cid::try_from(cid.deref());

                log::trace!("{:?} told me that {:?} is at {:?}", source, peer_id, cid);
            }
        }

        Ok(())
    }
}