mirror of
https://github.com/darkrenaissance/darkfi.git
synced 2026-04-28 03:00:18 -04:00
328 lines
9.5 KiB
Rust
328 lines
9.5 KiB
Rust
/* This file is part of DarkFi (https://dark.fi)
|
|
*
|
|
* Copyright (C) 2020-2023 Dyne.org foundation
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
use std::{collections::VecDeque, fmt::Debug};
|
|
|
|
use async_std::sync::{Arc, Mutex};
|
|
use async_trait::async_trait;
|
|
use darkfi_serial::{Decodable, Encodable, SerialDecodable, SerialEncodable};
|
|
use log::{debug, info};
|
|
use rand::{rngs::OsRng, RngCore};
|
|
|
|
use super::EventMsg;
|
|
use crate::{
|
|
event_graph::model::{Event, EventId, ModelPtr},
|
|
net,
|
|
util::async_util::sleep,
|
|
Result,
|
|
};
|
|
|
|
const SIZE_OF_SEEN_BUFFER: usize = 65536;
|
|
// const MAX_CONFIRM: u8 = 3;
|
|
|
|
#[derive(Clone)]
|
|
struct RingBuffer<T> {
|
|
pub items: VecDeque<T>,
|
|
}
|
|
|
|
impl<T: Eq + PartialEq + Clone> RingBuffer<T> {
|
|
pub fn new(capacity: usize) -> Self {
|
|
let items = VecDeque::with_capacity(capacity);
|
|
Self { items }
|
|
}
|
|
|
|
pub fn push(&mut self, val: T) {
|
|
if self.items.len() == self.items.capacity() {
|
|
self.items.pop_front();
|
|
}
|
|
self.items.push_back(val);
|
|
}
|
|
|
|
pub fn contains(&self, val: &T) -> bool {
|
|
self.items.contains(val)
|
|
}
|
|
}
|
|
|
|
type InvId = u64;
|
|
|
|
#[derive(SerialEncodable, SerialDecodable, Clone, Debug, PartialEq, Eq, Hash)]
|
|
struct InvItem {
|
|
id: InvId,
|
|
hash: EventId,
|
|
}
|
|
|
|
#[derive(SerialDecodable, SerialEncodable, Clone, Debug)]
|
|
struct Inv {
|
|
invs: Vec<InvItem>,
|
|
}
|
|
|
|
#[derive(SerialDecodable, SerialEncodable, Clone, Debug)]
|
|
struct SyncEvent {
|
|
leaves: Vec<EventId>,
|
|
}
|
|
|
|
#[derive(SerialDecodable, SerialEncodable, Clone, Debug)]
|
|
struct GetData {
|
|
events: Vec<EventId>,
|
|
}
|
|
|
|
pub type SeenPtr<T> = Arc<Seen<T>>;
|
|
|
|
pub struct Seen<T> {
|
|
seen: Mutex<RingBuffer<T>>,
|
|
}
|
|
|
|
impl<T: Eq + PartialEq + Clone> Seen<T> {
|
|
pub fn new() -> SeenPtr<T> {
|
|
Arc::new(Self { seen: Mutex::new(RingBuffer::new(SIZE_OF_SEEN_BUFFER)) })
|
|
}
|
|
|
|
pub async fn push(&self, item: &T) -> bool {
|
|
let seen = &mut self.seen.lock().await;
|
|
if !seen.contains(item) {
|
|
seen.push(item.clone());
|
|
return true
|
|
}
|
|
false
|
|
}
|
|
}
|
|
|
|
pub struct ProtocolEvent<T>
|
|
where
|
|
T: Send + Sync + Encodable + Decodable + Debug + 'static,
|
|
{
|
|
jobsman: net::ProtocolJobsManagerPtr,
|
|
event_sub: net::MessageSubscription<Event<T>>,
|
|
inv_sub: net::MessageSubscription<Inv>,
|
|
getdata_sub: net::MessageSubscription<GetData>,
|
|
syncevent_sub: net::MessageSubscription<SyncEvent>,
|
|
p2p: net::P2pPtr,
|
|
channel: net::ChannelPtr,
|
|
model: ModelPtr<T>,
|
|
seen_event: SeenPtr<EventId>,
|
|
seen_inv: SeenPtr<InvId>,
|
|
}
|
|
|
|
impl<T> ProtocolEvent<T>
|
|
where
|
|
T: Send + Sync + Encodable + Decodable + Clone + EventMsg + Debug + 'static,
|
|
{
|
|
pub async fn init(
|
|
channel: net::ChannelPtr,
|
|
p2p: net::P2pPtr,
|
|
model: ModelPtr<T>,
|
|
seen_event: SeenPtr<EventId>,
|
|
seen_inv: SeenPtr<InvId>,
|
|
) -> net::ProtocolBasePtr {
|
|
let message_subsytem = channel.get_message_subsystem();
|
|
message_subsytem.add_dispatch::<Event<T>>().await;
|
|
message_subsytem.add_dispatch::<Inv>().await;
|
|
message_subsytem.add_dispatch::<GetData>().await;
|
|
message_subsytem.add_dispatch::<SyncEvent>().await;
|
|
|
|
let event_sub =
|
|
channel.clone().subscribe_msg::<Event<T>>().await.expect("Missing Event dispatcher!");
|
|
|
|
let inv_sub = channel.subscribe_msg::<Inv>().await.expect("Missing Inv dispatcher!");
|
|
|
|
let getdata_sub =
|
|
channel.clone().subscribe_msg::<GetData>().await.expect("Missing GetData dispatcher!");
|
|
|
|
let syncevent_sub = channel
|
|
.clone()
|
|
.subscribe_msg::<SyncEvent>()
|
|
.await
|
|
.expect("Missing SyncEvent dispatcher!");
|
|
|
|
Arc::new(Self {
|
|
jobsman: net::ProtocolJobsManager::new("ProtocolEvent", channel.clone()),
|
|
event_sub,
|
|
inv_sub,
|
|
getdata_sub,
|
|
syncevent_sub,
|
|
p2p,
|
|
channel,
|
|
model,
|
|
seen_event,
|
|
seen_inv,
|
|
})
|
|
}
|
|
|
|
async fn handle_receive_event(self: Arc<Self>) -> Result<()> {
|
|
debug!(target: "ircd", "ProtocolEvent::handle_receive_event() [START]");
|
|
let exclude_list = vec![self.channel.address()];
|
|
loop {
|
|
let event = self.event_sub.receive().await?;
|
|
let event = (*event).to_owned();
|
|
|
|
if !self.seen_event.push(&event.hash()).await {
|
|
continue
|
|
}
|
|
|
|
info!("[P2P] Received: {:?}", event.action);
|
|
|
|
self.new_event(&event).await?;
|
|
self.send_inv(&event).await?;
|
|
|
|
// Broadcast the msg
|
|
self.p2p.broadcast_with_exclude(event, &exclude_list).await?;
|
|
}
|
|
}
|
|
|
|
async fn handle_receive_inv(self: Arc<Self>) -> Result<()> {
|
|
debug!(target: "ircd", "ProtocolEvent::handle_receive_inv() [START]");
|
|
let exclude_list = vec![self.channel.address()];
|
|
loop {
|
|
let inv = self.inv_sub.receive().await?;
|
|
let inv = (*inv).to_owned();
|
|
let inv_item = inv.invs[0].clone();
|
|
|
|
// for inv in inv.invs.iter() {
|
|
if !self.seen_inv.push(&inv_item.id).await {
|
|
continue
|
|
}
|
|
|
|
if self.model.lock().await.get_event(&inv_item.hash).is_none() {
|
|
self.send_getdata(vec![inv_item.hash]).await?;
|
|
}
|
|
|
|
// }
|
|
|
|
// Broadcast the inv msg
|
|
self.p2p.broadcast_with_exclude(inv, &exclude_list).await?;
|
|
}
|
|
}
|
|
async fn handle_receive_getdata(self: Arc<Self>) -> Result<()> {
|
|
debug!(target: "ircd", "ProtocolEvent::handle_receive_getdata() [START]");
|
|
loop {
|
|
let getdata = self.getdata_sub.receive().await?;
|
|
let events = (*getdata).to_owned().events;
|
|
|
|
for event_id in events {
|
|
let model_event = self.model.lock().await.get_event(&event_id);
|
|
if let Some(event) = model_event {
|
|
self.channel.send(event).await?;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
async fn handle_receive_syncevent(self: Arc<Self>) -> Result<()> {
|
|
debug!(target: "ircd", "ProtocolEvent::handle_receive_syncevent() [START]");
|
|
loop {
|
|
let syncevent = self.syncevent_sub.receive().await?;
|
|
|
|
let model = self.model.lock().await;
|
|
let leaves = model.find_leaves();
|
|
|
|
if leaves == syncevent.leaves {
|
|
continue
|
|
}
|
|
|
|
for leaf in syncevent.leaves.iter() {
|
|
if leaves.contains(leaf) {
|
|
continue
|
|
}
|
|
|
|
let children = model.get_offspring(leaf);
|
|
|
|
for child in children {
|
|
self.channel.send(child).await?;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// every 6 seconds send a SyncEvent msg
|
|
async fn send_sync_hash_loop(self: Arc<Self>) -> Result<()> {
|
|
loop {
|
|
sleep(6).await;
|
|
let leaves = self.model.lock().await.find_leaves();
|
|
self.channel.send(SyncEvent { leaves }).await?;
|
|
}
|
|
}
|
|
|
|
async fn new_event(&self, event: &Event<T>) -> Result<()> {
|
|
let mut model = self.model.lock().await;
|
|
model.add(event.clone()).await;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn send_inv(&self, event: &Event<T>) -> Result<()> {
|
|
let id = OsRng.next_u64();
|
|
self.p2p.broadcast(Inv { invs: vec![InvItem { id, hash: event.hash() }] }).await?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn send_getdata(&self, events: Vec<EventId>) -> Result<()> {
|
|
self.channel.send(GetData { events }).await?;
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[async_trait]
|
|
impl<T> net::ProtocolBase for ProtocolEvent<T>
|
|
where
|
|
T: Send + Sync + Encodable + Decodable + Clone + EventMsg + Debug,
|
|
{
|
|
async fn start(self: Arc<Self>, executor: Arc<smol::Executor<'_>>) -> Result<()> {
|
|
debug!(target: "ircd", "ProtocolEvent::start() [START]");
|
|
self.jobsman.clone().start(executor.clone());
|
|
self.jobsman.clone().spawn(self.clone().handle_receive_event(), executor.clone()).await;
|
|
self.jobsman.clone().spawn(self.clone().handle_receive_inv(), executor.clone()).await;
|
|
self.jobsman.clone().spawn(self.clone().handle_receive_getdata(), executor.clone()).await;
|
|
self.jobsman.clone().spawn(self.clone().handle_receive_syncevent(), executor.clone()).await;
|
|
self.jobsman.clone().spawn(self.clone().send_sync_hash_loop(), executor.clone()).await;
|
|
debug!(target: "ircd", "ProtocolEvent::start() [END]");
|
|
Ok(())
|
|
}
|
|
|
|
fn name(&self) -> &'static str {
|
|
"ProtocolEvent"
|
|
}
|
|
}
|
|
|
|
impl<T> net::Message for Event<T>
|
|
where
|
|
T: Send + Sync + Decodable + Encodable + 'static,
|
|
{
|
|
fn name() -> &'static str {
|
|
"event"
|
|
}
|
|
}
|
|
|
|
impl net::Message for Inv {
|
|
fn name() -> &'static str {
|
|
"inv"
|
|
}
|
|
}
|
|
|
|
impl net::Message for SyncEvent {
|
|
fn name() -> &'static str {
|
|
"syncevent"
|
|
}
|
|
}
|
|
|
|
impl net::Message for GetData {
|
|
fn name() -> &'static str {
|
|
"getdata"
|
|
}
|
|
}
|