diff options
Diffstat (limited to 'tvix/cli/src/nix_compat.rs')
-rw-r--r-- | tvix/cli/src/nix_compat.rs | 122 |
1 files changed, 122 insertions, 0 deletions
diff --git a/tvix/cli/src/nix_compat.rs b/tvix/cli/src/nix_compat.rs new file mode 100644 index 000000000000..bda238983839 --- /dev/null +++ b/tvix/cli/src/nix_compat.rs @@ -0,0 +1,122 @@ +//! This module implements (temporary) compatibility shims between +//! Tvix and C++ Nix. +//! +//! These are not intended to be long-lived, but should bootstrap Tvix +//! by piggybacking off functionality that already exists in Nix and +//! is still being implemented in Tvix. + +use std::collections::HashMap; +use std::path::Path; +use std::process::Command; +use std::sync::RwLock; +use std::{io, path::PathBuf}; + +use smol_str::SmolStr; +use tvix_eval::{EvalIO, FileType, StdIO}; + +/// Compatibility implementation of [`EvalIO`] that uses C++ Nix to +/// write files to the Nix store. +pub struct NixCompatIO { + /// Most IO requests are tunneled through to [`tvix_eval::StdIO`] + /// instead. + underlying: StdIO, + + /// Cache paths for identical files being imported to the store. + // TODO(tazjin): This could be done better by having a thunk cache + // for these calls on the eval side, but that is a little more + // complex. + import_cache: RwLock<HashMap<PathBuf, PathBuf>>, +} + +impl EvalIO for NixCompatIO { + fn store_dir(&self) -> Option<String> { + Some("/nix/store".into()) + } + + // Pass path imports through to `nix-store --add` + fn import_path(&self, path: &Path) -> Result<PathBuf, io::Error> { + let path = path.to_owned(); + if let Some(path) = self + .import_cache + .read() + .map_err(|e| io::Error::new(io::ErrorKind::Other, e.to_string()))? + .get(&path) + { + return Ok(path.to_path_buf()); + } + + let store_path = self.add_to_store(&path)?; + + self.import_cache + .write() + .map_err(|e| io::Error::new(io::ErrorKind::Other, e.to_string()))? + .insert(path, store_path.clone()); + + Ok(store_path) + } + + // Pass the rest of the functions through to `Self::underlying` + fn path_exists(&self, path: &Path) -> Result<bool, io::Error> { + if path.starts_with("/__corepkgs__") { + return Ok(true); + } + + self.underlying.path_exists(path) + } + + fn read_to_string(&self, path: &Path) -> Result<String, io::Error> { + // Bundled version of corepkgs/fetchurl.nix. This workaround + // is similar to what cppnix does for passing the path + // through. + // + // TODO: this comparison is bad and allocates, we should use + // the sane path library. + if path.starts_with("/__corepkgs__/fetchurl.nix") { + return Ok(include_str!("fetchurl.nix").to_string()); + } + + self.underlying.read_to_string(path) + } + + fn read_dir(&self, path: &Path) -> Result<Vec<(SmolStr, FileType)>, io::Error> { + self.underlying.read_dir(path) + } +} + +impl NixCompatIO { + pub fn new() -> Self { + NixCompatIO { + underlying: StdIO, + import_cache: RwLock::new(HashMap::new()), + } + } + + /// Add a path to the Nix store using the `nix-store --add` + /// functionality from C++ Nix. + fn add_to_store(&self, path: &Path) -> Result<PathBuf, io::Error> { + if !path.try_exists()? { + return Err(io::Error::from(io::ErrorKind::NotFound)); + } + + let mut cmd = Command::new("nix-store"); + cmd.arg("--add"); + cmd.arg(path); + + let out = cmd.output()?; + + if !out.status.success() { + return Err(io::Error::new( + io::ErrorKind::Other, + String::from_utf8_lossy(&out.stderr).trim().to_owned(), + )); + } + + let out_path_str = String::from_utf8(out.stdout) + .map_err(|err| io::Error::new(io::ErrorKind::InvalidData, err))?; + let out_path_trimmed = out_path_str.trim(); + + let mut out_path = PathBuf::new(); + out_path.push(out_path_trimmed); + Ok(out_path) + } +} |