2024-03-05 01:29:57 +00:00
|
|
|
//! This file contains a test for a handshake with monerod but uses fragmented messages.
|
|
|
|
use std::{
|
|
|
|
net::SocketAddr,
|
|
|
|
pin::Pin,
|
|
|
|
sync::Arc,
|
|
|
|
task::{Context, Poll},
|
|
|
|
time::Duration,
|
|
|
|
};
|
|
|
|
|
|
|
|
use futures::{Stream, StreamExt};
|
|
|
|
use tokio::{
|
|
|
|
net::{
|
|
|
|
tcp::{OwnedReadHalf, OwnedWriteHalf},
|
|
|
|
TcpListener, TcpStream,
|
|
|
|
},
|
2024-05-02 22:58:22 +00:00
|
|
|
sync::Semaphore,
|
2024-03-05 01:29:57 +00:00
|
|
|
time::timeout,
|
|
|
|
};
|
|
|
|
use tokio_util::{
|
|
|
|
bytes::BytesMut,
|
|
|
|
codec::{Encoder, FramedRead, FramedWrite},
|
|
|
|
};
|
|
|
|
use tower::{Service, ServiceExt};
|
|
|
|
|
|
|
|
use cuprate_helper::network::Network;
|
|
|
|
use monero_p2p::{
|
|
|
|
client::{ConnectRequest, Connector, DoHandshakeRequest, HandShaker, InternalPeerID},
|
|
|
|
network_zones::ClearNetServerCfg,
|
|
|
|
ConnectionDirection, NetworkZone,
|
|
|
|
};
|
|
|
|
use monero_wire::{
|
|
|
|
common::PeerSupportFlags,
|
|
|
|
levin::{message::make_fragmented_messages, LevinMessage, Protocol},
|
|
|
|
BasicNodeData, Message, MoneroWireCodec,
|
|
|
|
};
|
|
|
|
|
|
|
|
use cuprate_test_utils::monerod::monerod;
|
|
|
|
|
|
|
|
mod utils;
|
|
|
|
use utils::*;
|
|
|
|
|
|
|
|
/// A network zone equal to clear net where every message sent is turned into a fragmented message.
|
|
|
|
/// Does not support sending fragmented or dummy messages manually.
|
|
|
|
#[derive(Clone, Copy)]
|
|
|
|
pub enum FragNet {}
|
|
|
|
|
|
|
|
#[async_trait::async_trait]
|
|
|
|
impl NetworkZone for FragNet {
|
|
|
|
const NAME: &'static str = "FragNet";
|
2024-05-02 22:58:22 +00:00
|
|
|
const SEEDS: &'static [Self::Addr] = &[];
|
2024-03-05 01:29:57 +00:00
|
|
|
const ALLOW_SYNC: bool = true;
|
|
|
|
const DANDELION_PP: bool = true;
|
|
|
|
const CHECK_NODE_ID: bool = true;
|
|
|
|
|
|
|
|
type Addr = SocketAddr;
|
|
|
|
type Stream = FramedRead<OwnedReadHalf, MoneroWireCodec>;
|
|
|
|
type Sink = FramedWrite<OwnedWriteHalf, FragmentCodec>;
|
|
|
|
type Listener = InBoundStream;
|
|
|
|
|
|
|
|
type ServerCfg = ClearNetServerCfg;
|
|
|
|
|
|
|
|
async fn connect_to_peer(
|
|
|
|
addr: Self::Addr,
|
|
|
|
) -> Result<(Self::Stream, Self::Sink), std::io::Error> {
|
|
|
|
let (read, write) = TcpStream::connect(addr).await?.into_split();
|
|
|
|
Ok((
|
|
|
|
FramedRead::new(read, MoneroWireCodec::default()),
|
|
|
|
FramedWrite::new(write, FragmentCodec::default()),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn incoming_connection_listener(
|
|
|
|
config: Self::ServerCfg,
|
2024-06-04 17:19:25 +00:00
|
|
|
port: u16,
|
2024-03-05 01:29:57 +00:00
|
|
|
) -> Result<Self::Listener, std::io::Error> {
|
2024-06-04 17:19:25 +00:00
|
|
|
let listener = TcpListener::bind(SocketAddr::new(config.ip, port)).await?;
|
2024-03-05 01:29:57 +00:00
|
|
|
Ok(InBoundStream { listener })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct InBoundStream {
|
|
|
|
listener: TcpListener,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Stream for InBoundStream {
|
|
|
|
type Item = Result<
|
|
|
|
(
|
|
|
|
Option<SocketAddr>,
|
|
|
|
FramedRead<OwnedReadHalf, MoneroWireCodec>,
|
|
|
|
FramedWrite<OwnedWriteHalf, FragmentCodec>,
|
|
|
|
),
|
|
|
|
std::io::Error,
|
|
|
|
>;
|
|
|
|
|
|
|
|
fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
|
|
|
self.listener
|
|
|
|
.poll_accept(cx)
|
|
|
|
.map_ok(|(stream, addr)| {
|
|
|
|
let (read, write) = stream.into_split();
|
|
|
|
(
|
|
|
|
Some(addr),
|
|
|
|
FramedRead::new(read, MoneroWireCodec::default()),
|
|
|
|
FramedWrite::new(write, FragmentCodec::default()),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.map(Some)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct FragmentCodec(MoneroWireCodec);
|
|
|
|
|
|
|
|
impl Encoder<LevinMessage<Message>> for FragmentCodec {
|
|
|
|
type Error = <MoneroWireCodec as Encoder<LevinMessage<Message>>>::Error;
|
|
|
|
|
|
|
|
fn encode(
|
|
|
|
&mut self,
|
|
|
|
item: LevinMessage<Message>,
|
|
|
|
dst: &mut BytesMut,
|
|
|
|
) -> Result<(), Self::Error> {
|
|
|
|
match item {
|
|
|
|
LevinMessage::Body(body) => {
|
|
|
|
// 66 is the minimum fragment size.
|
|
|
|
let fragments = make_fragmented_messages(&Protocol::default(), 66, body).unwrap();
|
|
|
|
|
|
|
|
for frag in fragments {
|
|
|
|
self.0.encode(frag.into(), dst)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => unreachable!("Handshakes should only send bucket bodys"),
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test]
|
|
|
|
async fn fragmented_handshake_cuprate_to_monerod() {
|
|
|
|
let semaphore = Arc::new(Semaphore::new(10));
|
|
|
|
let permit = semaphore.acquire_owned().await.unwrap();
|
|
|
|
|
|
|
|
let monerod = monerod(["--fixed-difficulty=1", "--out-peers=0"]).await;
|
|
|
|
|
|
|
|
let our_basic_node_data = BasicNodeData {
|
|
|
|
my_port: 0,
|
2024-05-02 22:58:22 +00:00
|
|
|
network_id: Network::Mainnet.network_id(),
|
2024-03-05 01:29:57 +00:00
|
|
|
peer_id: 87980,
|
|
|
|
support_flags: PeerSupportFlags::from(1_u32),
|
|
|
|
rpc_port: 0,
|
|
|
|
rpc_credits_per_hash: 0,
|
|
|
|
};
|
|
|
|
|
2024-05-02 22:58:22 +00:00
|
|
|
let handshaker = HandShaker::<FragNet, _, _, _, _, _>::new(
|
2024-03-05 01:29:57 +00:00
|
|
|
DummyAddressBook,
|
2024-05-02 22:58:22 +00:00
|
|
|
DummyPeerSyncSvc,
|
2024-03-05 01:29:57 +00:00
|
|
|
DummyCoreSyncSvc,
|
|
|
|
DummyPeerRequestHandlerSvc,
|
2024-05-02 22:58:22 +00:00
|
|
|
|_| futures::stream::pending(),
|
2024-03-05 01:29:57 +00:00
|
|
|
our_basic_node_data,
|
|
|
|
);
|
|
|
|
|
|
|
|
let mut connector = Connector::new(handshaker);
|
|
|
|
|
|
|
|
connector
|
|
|
|
.ready()
|
|
|
|
.await
|
|
|
|
.unwrap()
|
|
|
|
.call(ConnectRequest {
|
|
|
|
addr: monerod.p2p_addr(),
|
|
|
|
permit,
|
|
|
|
})
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test]
|
|
|
|
async fn fragmented_handshake_monerod_to_cuprate() {
|
|
|
|
let semaphore = Arc::new(Semaphore::new(10));
|
|
|
|
let permit = semaphore.acquire_owned().await.unwrap();
|
|
|
|
|
|
|
|
let our_basic_node_data = BasicNodeData {
|
|
|
|
my_port: 18081,
|
2024-05-02 22:58:22 +00:00
|
|
|
network_id: Network::Mainnet.network_id(),
|
2024-03-05 01:29:57 +00:00
|
|
|
peer_id: 87980,
|
|
|
|
support_flags: PeerSupportFlags::from(1_u32),
|
|
|
|
rpc_port: 0,
|
|
|
|
rpc_credits_per_hash: 0,
|
|
|
|
};
|
|
|
|
|
2024-05-02 22:58:22 +00:00
|
|
|
let mut handshaker = HandShaker::<FragNet, _, _, _, _, _>::new(
|
2024-03-05 01:29:57 +00:00
|
|
|
DummyAddressBook,
|
2024-05-02 22:58:22 +00:00
|
|
|
DummyPeerSyncSvc,
|
2024-03-05 01:29:57 +00:00
|
|
|
DummyCoreSyncSvc,
|
|
|
|
DummyPeerRequestHandlerSvc,
|
2024-05-02 22:58:22 +00:00
|
|
|
|_| futures::stream::pending(),
|
2024-03-05 01:29:57 +00:00
|
|
|
our_basic_node_data,
|
|
|
|
);
|
|
|
|
|
2024-06-04 17:19:25 +00:00
|
|
|
let ip = "127.0.0.1".parse().unwrap();
|
2024-03-05 01:29:57 +00:00
|
|
|
|
2024-06-04 17:19:25 +00:00
|
|
|
let mut listener = FragNet::incoming_connection_listener(ClearNetServerCfg { ip }, 18081)
|
2024-03-05 01:29:57 +00:00
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let _monerod = monerod(["--add-exclusive-node=127.0.0.1:18081"]).await;
|
|
|
|
|
|
|
|
// Put a timeout on this just in case monerod doesn't make the connection to us.
|
|
|
|
let next_connection_fut = timeout(Duration::from_secs(30), listener.next());
|
|
|
|
|
|
|
|
if let Some(Ok((addr, stream, sink))) = next_connection_fut.await.unwrap() {
|
|
|
|
let _ = handshaker
|
|
|
|
.ready()
|
|
|
|
.await
|
|
|
|
.unwrap()
|
|
|
|
.call(DoHandshakeRequest {
|
|
|
|
addr: InternalPeerID::KnownAddr(addr.unwrap()), // This is clear net all addresses are known.
|
|
|
|
peer_stream: stream,
|
|
|
|
peer_sink: sink,
|
|
|
|
direction: ConnectionDirection::InBound,
|
|
|
|
permit,
|
|
|
|
})
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
} else {
|
|
|
|
panic!("Failed to receive connection from monerod.");
|
|
|
|
};
|
|
|
|
}
|