mirror of
https://github.com/Cuprate/cuprate.git
synced 2025-01-24 19:45:51 +00:00
Boog900
88f7d1f212
* download monerod in CI * move action file * add macOS arm support * remove reqwest from workspace * undo whitespace changes * fix indentation * update comments * add monerod to .gitignore * Apply suggestions from code review Co-authored-by: hinto-janai <hinto.janai@protonmail.com> * add back spacing --------- Co-authored-by: hinto-janai <hinto.janai@protonmail.com>
188 lines
5.9 KiB
Rust
188 lines
5.9 KiB
Rust
//! Monerod Module
|
|
//!
|
|
//! This module contains a function [`monerod`] to start `monerod` - the core Monero node. Cuprate can then use
|
|
//! this to test compatibility with monerod.
|
|
//!
|
|
use std::{
|
|
env::current_dir,
|
|
ffi::OsStr,
|
|
fs::read_dir,
|
|
io::Read,
|
|
net::{IpAddr, Ipv4Addr, SocketAddr, TcpListener},
|
|
path::PathBuf,
|
|
process::{Child, Command, Stdio},
|
|
str::from_utf8,
|
|
thread::panicking,
|
|
time::Duration,
|
|
};
|
|
|
|
use tokio::{task::yield_now, time::timeout};
|
|
|
|
/// IPv4 local host.
|
|
const LOCALHOST: IpAddr = IpAddr::V4(Ipv4Addr::LOCALHOST);
|
|
|
|
/// The log line `monerod` emits indicated it has successfully started up.
|
|
const MONEROD_STARTUP_TEXT: &str =
|
|
"The daemon will start synchronizing with the network. This may take a long time to complete.";
|
|
|
|
/// The log line `monerod` emits indicated it has stopped.
|
|
const MONEROD_SHUTDOWN_TEXT: &str = "Stopping cryptonote protocol";
|
|
|
|
/// Spawns monerod and returns [`SpawnedMoneroD`].
|
|
///
|
|
/// This function will set `regtest` and the P2P/ RPC ports so these can't be included in the flags.
|
|
pub async fn monerod<T: AsRef<OsStr>>(flags: impl IntoIterator<Item = T>) -> SpawnedMoneroD {
|
|
let path_to_monerod = find_root().join("monerod");
|
|
|
|
let rpc_port = get_available_port(&[]);
|
|
let p2p_port = get_available_port(&[rpc_port]);
|
|
let zmq_port = get_available_port(&[rpc_port, p2p_port]);
|
|
|
|
let data_dir = tempfile::tempdir().unwrap();
|
|
|
|
let mut monerod = Command::new(path_to_monerod)
|
|
.stdout(Stdio::piped())
|
|
.stderr(Stdio::piped())
|
|
.args(flags)
|
|
.arg("--regtest")
|
|
.arg("--log-level=2")
|
|
.arg(format!("--p2p-bind-port={p2p_port}"))
|
|
.arg(format!("--rpc-bind-port={rpc_port}"))
|
|
.arg(format!("--zmq-rpc-bind-port={zmq_port}"))
|
|
.arg(format!("--data-dir={}", data_dir.path().display()))
|
|
.arg("--non-interactive")
|
|
.spawn()
|
|
.expect(
|
|
"Failed to start monerod, you need to have the monerod binary in the root of the repo",
|
|
);
|
|
|
|
let mut logs = String::new();
|
|
|
|
timeout(Duration::from_secs(30), async {
|
|
loop {
|
|
let mut next_str = [0];
|
|
let _ = monerod
|
|
.stdout
|
|
.as_mut()
|
|
.unwrap()
|
|
.read(&mut next_str)
|
|
.unwrap();
|
|
|
|
logs.push_str(from_utf8(&next_str).unwrap());
|
|
|
|
if logs.contains(MONEROD_SHUTDOWN_TEXT) {
|
|
panic!("Failed to start monerod, logs: \n {logs}");
|
|
} else if logs.contains(MONEROD_STARTUP_TEXT) {
|
|
break;
|
|
}
|
|
// this is blocking code but as this is for tests performance isn't a priority. However we should still yield so
|
|
// the timeout works.
|
|
yield_now().await;
|
|
}
|
|
})
|
|
.await
|
|
.unwrap_or_else(|_| panic!("Failed to start monerod in time, logs: {logs}"));
|
|
|
|
SpawnedMoneroD {
|
|
process: monerod,
|
|
rpc_port,
|
|
p2p_port,
|
|
_data_dir: data_dir,
|
|
start_up_logs: logs,
|
|
}
|
|
}
|
|
|
|
/// Finds the root of the repo by finding the `target` directory, this will work up from the current
|
|
/// directory until it finds a `target` directory, then returns the directory that the target is contained
|
|
/// in.
|
|
fn find_root() -> PathBuf {
|
|
let mut current_dir = current_dir().unwrap();
|
|
loop {
|
|
if read_dir(current_dir.join("target")).is_ok() {
|
|
return current_dir;
|
|
} else if !current_dir.pop() {
|
|
panic!("Could not find ./target");
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Fetch an available TCP port on the machine for `monerod` to bind to.
|
|
fn get_available_port(already_taken: &[u16]) -> u16 {
|
|
loop {
|
|
// Using `0` makes the OS return a random available port.
|
|
let port = TcpListener::bind("127.0.0.1:0")
|
|
.unwrap()
|
|
.local_addr()
|
|
.unwrap()
|
|
.port();
|
|
|
|
if !already_taken.contains(&port) {
|
|
return port;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// A struct representing a spawned monerod.
|
|
pub struct SpawnedMoneroD {
|
|
/// A handle to the monerod process, monerod will be stopped when this is dropped.
|
|
process: Child,
|
|
/// The RPC port of the monerod instance.
|
|
rpc_port: u16,
|
|
/// The P2P port of the monerod instance.
|
|
p2p_port: u16,
|
|
/// The data dir for monerod - when this is dropped the dir will be deleted.
|
|
_data_dir: tempfile::TempDir,
|
|
/// The logs upto [`MONEROD_STARTUP_TEXT`].
|
|
start_up_logs: String,
|
|
}
|
|
|
|
impl SpawnedMoneroD {
|
|
/// Returns the p2p port of the spawned monerod
|
|
pub const fn p2p_addr(&self) -> SocketAddr {
|
|
SocketAddr::new(LOCALHOST, self.p2p_port)
|
|
}
|
|
|
|
/// Returns the RPC port of the spawned monerod
|
|
pub const fn rpc_port(&self) -> SocketAddr {
|
|
SocketAddr::new(LOCALHOST, self.rpc_port)
|
|
}
|
|
}
|
|
|
|
impl Drop for SpawnedMoneroD {
|
|
fn drop(&mut self) {
|
|
let mut error = false;
|
|
|
|
if self.process.kill().is_err() {
|
|
error = true;
|
|
println!("Failed to kill monerod, process id: {}", self.process.id());
|
|
}
|
|
|
|
if panicking() {
|
|
// If we are panicking then a test failed so print monerod's logs.
|
|
|
|
let mut out = String::new();
|
|
|
|
if self
|
|
.process
|
|
.stdout
|
|
.as_mut()
|
|
.unwrap()
|
|
.read_to_string(&mut out)
|
|
.is_err()
|
|
{
|
|
println!("Failed to get monerod's logs.");
|
|
}
|
|
|
|
println!("-----START-MONEROD-LOGS-----");
|
|
println!("{}{out}", self.start_up_logs);
|
|
println!("------END-MONEROD-LOGS------");
|
|
}
|
|
|
|
if error && !panicking() {
|
|
// `println` only outputs in a test when panicking so if there is an error while
|
|
// dropping monerod but not an error in the test then we need to panic to make sure
|
|
// the println!s are output.
|
|
panic!("Error while dropping monerod");
|
|
}
|
|
}
|
|
}
|