// Copyright 2018 Parity Technologies (UK) Ltd. // This file is part of Parity. // Parity is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // Parity 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 General Public License for more details. // You should have received a copy of the GNU General Public License // along with Parity. If not, see . extern crate parking_lot; extern crate parity_bytes; extern crate substrate_network_libp2p; use std::sync::atomic::{AtomicBool, Ordering as AtomicOrdering}; use std::sync::Arc; use std::thread; use std::time::*; use parking_lot::Mutex; use parity_bytes::Bytes; use substrate_network_libp2p::*; use TimerToken; pub struct TestProtocol { drop_session: bool, pub packet: Mutex, pub got_timeout: AtomicBool, pub got_disconnect: AtomicBool, } impl TestProtocol { pub fn new(drop_session: bool) -> Self { TestProtocol { packet: Mutex::new(Vec::new()), got_timeout: AtomicBool::new(false), got_disconnect: AtomicBool::new(false), drop_session: drop_session, } } pub fn got_packet(&self) -> bool { self.packet.lock()[..] == b"hello"[..] } pub fn got_timeout(&self) -> bool { self.got_timeout.load(AtomicOrdering::Relaxed) } pub fn got_disconnect(&self) -> bool { self.got_disconnect.load(AtomicOrdering::Relaxed) } } impl NetworkProtocolHandler for TestProtocol { fn initialize(&self, io: &NetworkContext) { io.register_timer(0, Duration::from_millis(10)).unwrap(); } fn read(&self, _io: &NetworkContext, _peer: &NodeIndex, packet_id: u8, data: &[u8]) { assert_eq!(packet_id, 33); self.packet.lock().extend(data); } fn connected(&self, io: &NetworkContext, peer: &NodeIndex) { if self.drop_session { io.report_peer(*peer, Severity::Bad("We are evil and just want to drop")) } else { io.respond(33, "hello".to_owned().into_bytes()); } } fn disconnected(&self, _io: &NetworkContext, _peer: &NodeIndex) { self.got_disconnect.store(true, AtomicOrdering::Relaxed); } /// Timer function called after a timeout created with `NetworkContext::timeout`. fn timeout(&self, _io: &NetworkContext, timer: TimerToken) { assert_eq!(timer, 0); self.got_timeout.store(true, AtomicOrdering::Relaxed); } } #[test] fn net_service() { let _service = NetworkService::new( NetworkConfiguration::new_local(), vec![(Arc::new(TestProtocol::new(false)), *b"myp", &[(1u8, 1)])] ).expect("Error creating network service"); } #[test] #[ignore] // TODO: how is this test even supposed to work? fn net_disconnect() { let mut config1 = NetworkConfiguration::new_local(); config1.boot_nodes = vec![ ]; let handler1 = Arc::new(TestProtocol::new(false)); let service1 = NetworkService::new(config1, vec![(handler1.clone(), *b"tst", &[(42u8, 1), (43u8, 1)])]).unwrap(); let mut config2 = NetworkConfiguration::new_local(); config2.boot_nodes = vec![ service1.external_url().unwrap() ]; let handler2 = Arc::new(TestProtocol::new(true)); let _service2 = NetworkService::new(config2, vec![(handler2.clone(), *b"tst", &[(42u8, 1), (43u8, 1)])]).unwrap(); while !(handler1.got_disconnect() && handler2.got_disconnect()) { thread::sleep(Duration::from_millis(50)); } assert!(handler1.got_disconnect()); assert!(handler2.got_disconnect()); } #[test] fn net_timeout() { let config = NetworkConfiguration::new_local(); let handler = Arc::new(TestProtocol::new(false)); let _service = NetworkService::new(config, vec![(handler.clone(), *b"tst", &[(42u8, 1), (43u8, 1)])]).unwrap(); while !handler.got_timeout() { thread::sleep(Duration::from_millis(50)); } }