use std::future::Future;

use anyhow::Result;
use iroh::{
    address_lookup::MemoryLookup, protocol::Router, Endpoint, EndpointId, RelayMode, SecretKey,
};
use iroh_blobs::{store::mem::MemStore, BlobsProtocol};
use iroh_docs::{api::DocsApi, protocol::Docs};
use iroh_gossip::net::Gossip;
use rand::{CryptoRng, Rng, SeedableRng};

/// A test node wrapping all protocols: blobs, gossip, docs.
pub struct TestNode {
    pub router: Router,
    pub store: MemStore,
    pub gossip: Gossip,
    pub docs_api: DocsApi,
}

impl TestNode {
    /// Endpoint ID of this node.
    pub fn id(&self) -> EndpointId {
        self.router.endpoint().id()
    }

    /// The endpoint address (for connecting / registering with MemoryLookup).
    pub fn addr(&self) -> iroh::EndpointAddr {
        self.router.endpoint().addr()
    }

    /// Graceful shutdown.
    pub async fn shutdown(self) -> Result<()> {
        self.router.shutdown().await?;
        Ok(())
    }
}

/// Spawn a single test node with the given MemoryLookup for address discovery.
fn spawn_one(
    rng: &mut (impl CryptoRng + Rng),
    disco: MemoryLookup,
) -> impl Future<Output = Result<TestNode>> + 'static {
    let secret_key = SecretKey::generate(rng);
    async move {
        let ep = Endpoint::empty_builder(RelayMode::Disabled)
            .secret_key(secret_key)
            .address_lookup(disco)
            .bind()
            .await?;

        let store = MemStore::new();
        let blobs_store = (*store).clone(); // iroh_blobs::api::Store
        let gossip = Gossip::builder().spawn(ep.clone());
        let docs = Docs::memory()
            .spawn(ep.clone(), blobs_store.clone(), gossip.clone())
            .await?;

        let router = Router::builder(ep)
            .accept(iroh_blobs::ALPN, BlobsProtocol::new(&blobs_store, None))
            .accept(iroh_docs::ALPN, docs.clone())
            .accept(iroh_gossip::ALPN, gossip.clone())
            .spawn();

        Ok(TestNode {
            router,
            store,
            gossip,
            docs_api: docs.api().clone(),
        })
    }
}

/// Spawn a pair of test nodes that can discover each other.
pub async fn spawn_pair(rng: &mut (impl CryptoRng + Rng)) -> Result<(TestNode, TestNode)> {
    let disco = MemoryLookup::new();
    let fut_a = spawn_one(rng, disco.clone());
    let fut_b = spawn_one(rng, disco.clone());
    let (a, b) = tokio::try_join!(fut_a, fut_b)?;
    disco.add_endpoint_info(a.addr());
    disco.add_endpoint_info(b.addr());
    Ok((a, b))
}

/// Spawn N test nodes that can all discover each other.
pub async fn spawn_nodes(n: usize, rng: &mut (impl CryptoRng + Rng)) -> Result<Vec<TestNode>> {
    let disco = MemoryLookup::new();
    let mut futs = Vec::with_capacity(n);
    for _ in 0..n {
        futs.push(spawn_one(rng, disco.clone()));
    }
    let nodes: Vec<TestNode> = futures_util::future::try_join_all(futs).await?;
    for node in &nodes {
        disco.add_endpoint_info(node.addr());
    }
    Ok(nodes)
}

/// Create a deterministic RNG from a seed string.
pub fn test_rng(seed: &[u8]) -> rand_chacha::ChaCha12Rng {
    let hash = hemera::hash(seed);
    let seed_bytes: [u8; 32] = hash.as_bytes()[..32].try_into().unwrap();
    rand_chacha::ChaCha12Rng::from_seed(seed_bytes)
}

/// Initialize tracing for tests (call once, ignores errors on repeat calls).
pub fn init_tracing() {
    tracing_subscriber::fmt()
        .with_env_filter(
            tracing_subscriber::EnvFilter::try_from_default_env()
                .unwrap_or_else(|_| tracing_subscriber::EnvFilter::new("warn")),
        )
        .try_init()
        .ok();
}

Synonyms

bbg/src/lib.rs
optica/src/lib.rs
zheng/src/lib.rs
nox/rs/lib.rs
honeycrisp/src/lib.rs
trident/src/lib.rs
lens/src/lib.rs
strata/src/lib.rs
rs/macros/src/lib.rs
strata/nebu/rs/lib.rs
honeycrisp/rane/src/lib.rs
honeycrisp/acpu/src/lib.rs
lens/core/src/lib.rs
rs/mir-format/src/lib.rs
rs/core/src/lib.rs
hemera/wgsl/src/lib.rs
strata/kuro/rs/lib.rs
radio/iroh-ffi/src/lib.rs
cyb/src-tauri/src/lib.rs
strata/core/src/lib.rs
radio/iroh-docs/src/lib.rs
strata/compute/src/lib.rs
lens/porphyry/src/lib.rs
radio/cyber-bao/src/lib.rs
radio/iroh-relay/src/lib.rs
lens/assayer/src/lib.rs
lens/brakedown/src/lib.rs
radio/iroh-car/src/lib.rs
honeycrisp/unimem/src/lib.rs
honeycrisp/aruminium/src/lib.rs
lens/binius/src/lib.rs
hemera/rs/src/lib.rs
strata/ext/src/lib.rs
radio/iroh/src/lib.rs
radio/iroh-gossip/src/lib.rs
strata/proof/src/lib.rs
radio/iroh-blobs/src/lib.rs
radio/iroh-base/src/lib.rs
radio/iroh-dns-server/src/lib.rs
radio/iroh-willow/src/lib.rs
lens/ikat/src/lib.rs
rs/tests/macro-integration/src/lib.rs
cw-cyber/contracts/hub-networks/src/lib.rs
cw-cyber/contracts/litium-core/src/lib.rs
strata/trop/wgsl/src/lib.rs
strata/kuro/wgsl/src/lib.rs
cw-cyber/contracts/hub-protocols/src/lib.rs
cw-cyber/contracts/cw-cyber-gift/src/lib.rs
strata/trop/rs/src/lib.rs
cw-cyber/contracts/cybernet/src/lib.rs
cw-cyber/contracts/hub-channels/src/lib.rs
strata/nebu/wgsl/src/lib.rs
cw-cyber/contracts/graph-filter/src/lib.rs
cw-cyber/contracts/litium-stake/src/lib.rs
trident/editor/zed/src/lib.rs
radio/iroh-ffi/iroh-js/src/lib.rs
cw-cyber/contracts/hub-tokens/src/lib.rs
cyb/cyb/cyb-services/src/lib.rs
cw-cyber/packages/hub-base/src/lib.rs
strata/genies/rs/src/lib.rs
cw-cyber/contracts/std-test/src/lib.rs
cw-cyber/packages/cyber-std-test/src/lib.rs
cw-cyber/contracts/litium-refer/src/lib.rs
strata/jali/rs/src/lib.rs
cw-cyber/contracts/hub-libs/src/lib.rs
cw-cyber/contracts/litium-wrap/src/lib.rs
cw-cyber/packages/cyber-std/src/lib.rs
strata/genies/wgsl/src/lib.rs
cw-cyber/contracts/hub-skills/src/lib.rs
strata/jali/wgsl/src/lib.rs
cw-cyber/contracts/cw-cyber-subgraph/src/lib.rs
radio/iroh/bench/src/lib.rs
cw-cyber/contracts/litium-mine/src/lib.rs
cw-cyber/contracts/cw-cyber-passport/src/lib.rs

Neighbours