Radish alpha
h
rad:z3gqcJUoA1n9HaHKufZs5FCSGazv5
Radicle Heartwood Protocol & Stack
Radicle
Git
heartwood crates radicle src storage git transport remote mock.rs
//! Mock git transport used for mocking the remote transport in tests.
use std::collections::HashMap;
use std::path::{Path, PathBuf};
use std::str::FromStr;
use std::sync::LazyLock;
use std::sync::{Mutex, Once};
use std::thread::ThreadId;
use std::{process, thread};

use super::Url;
use crate::git;
use crate::storage::RemoteId;
use crate::storage::git::transport::ChildStream;

/// Nodes registered with the mock transport.
static NODES: LazyLock<Mutex<HashMap<(ThreadId, RemoteId), PathBuf>>> =
    LazyLock::new(|| Mutex::new(HashMap::new()));

/// The mock transport.
#[derive(Default)]
struct MockTransport;

impl git::raw::transport::SmartSubtransport for MockTransport {
    fn action(
        &self,
        url: &str,
        service: git::raw::transport::Service,
    ) -> Result<Box<dyn git::raw::transport::SmartSubtransportStream>, git::raw::Error> {
        let url =
            Url::from_str(url).map_err(|e| git::raw::Error::from_str(e.to_string().as_str()))?;
        let id = thread::current().id();
        let nodes = NODES.lock().expect("lock cannot be poisoned");
        let storage = if let Some(storage) = nodes.get(&(id, url.node)) {
            match service {
                git::raw::transport::Service::ReceivePack
                | git::raw::transport::Service::ReceivePackLs => {
                    return Err(git::raw::Error::from_str(
                        "git-receive-pack is not supported with the mock transport",
                    ));
                }
                _ => {}
            }
            storage
        } else {
            return Err(git::raw::Error::from_str(&format!(
                "node {} was not registered with the mock transport",
                url.node
            )));
        };
        assert!(
            storage.exists(),
            "The storage path {} must exist",
            storage.display()
        );

        let git_dir = storage.join(url.repo.canonical());
        assert!(
            git_dir.exists(),
            "The repository {} must exist",
            git_dir.display()
        );

        let mut cmd = process::Command::new("git");
        let mut child = cmd
            .arg("upload-pack")
            .arg("--strict")
            .arg(&git_dir)
            .stdin(process::Stdio::piped())
            .stdout(process::Stdio::piped())
            .stderr(process::Stdio::inherit())
            .spawn()
            .expect("the `git` command is available");

        let stdin = child.stdin.take().expect("stdin is safe to take");
        let stdout = child.stdout.take().expect("stdout is safe to take");

        thread::spawn(move || child.wait());

        Ok(Box::new(ChildStream { stdout, stdin }))
    }

    fn close(&self) -> Result<(), git::raw::Error> {
        Ok(())
    }
}

/// Register a new node with the given storage path.
pub fn register(node: &RemoteId, path: &Path) {
    static REGISTER: Once = Once::new();

    REGISTER.call_once(|| unsafe {
        git::raw::transport::register(Url::SCHEME, move |remote| {
            git::raw::transport::Transport::smart(remote, false, MockTransport)
        })
        .expect("transport registration is successful");
    });
    let id = thread::current().id();

    NODES
        .lock()
        .expect("the lock isn't poisoned")
        .insert((id, *node), path.to_owned());
}