//! 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 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<(Vec<u8>, 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)
}
}