2022-12-12 15:19:27 +01:00
|
|
|
//! Interface for injecting I/O-related functionality into tvix-eval.
|
|
|
|
//!
|
|
|
|
//! The Nix language contains several builtins (e.g. `builtins.readDir`), as
|
|
|
|
//! well as language feature (e.g. string-"coercion" of paths) that interact
|
|
|
|
//! with the filesystem.
|
|
|
|
//!
|
|
|
|
//! The language evaluator implemented by this crate does not depend on any
|
|
|
|
//! particular filesystem interaction model. Instead, this module provides a
|
|
|
|
//! trait that can be implemented by tvix-eval callers to provide the
|
|
|
|
//! functionality they desire.
|
|
|
|
//!
|
|
|
|
//! In theory this can be used to implement "mocked" filesystem interactions, or
|
|
|
|
//! interaction with remote filesystems, etc.
|
|
|
|
//!
|
|
|
|
//! In the context of Nix builds, callers also use this interface to determine
|
|
|
|
//! how store paths are opened and so on.
|
|
|
|
|
2022-12-12 18:02:39 +01:00
|
|
|
use smol_str::SmolStr;
|
2022-12-12 15:19:27 +01:00
|
|
|
use std::path::PathBuf;
|
2022-12-12 18:02:39 +01:00
|
|
|
use std::rc::Rc;
|
2022-12-12 15:19:27 +01:00
|
|
|
|
|
|
|
use crate::errors::ErrorKind;
|
|
|
|
|
2022-12-12 18:02:39 +01:00
|
|
|
/// Types of files as represented by `builtins.readDir` in Nix.
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum FileType {
|
|
|
|
Directory,
|
|
|
|
Regular,
|
|
|
|
Symlink,
|
|
|
|
Unknown,
|
|
|
|
}
|
|
|
|
|
2022-12-12 15:19:27 +01:00
|
|
|
/// Defines how filesystem interaction occurs inside of tvix-eval.
|
|
|
|
pub trait EvalIO {
|
2022-12-12 17:36:45 +01:00
|
|
|
/// Verify whether the file at the specified path exists.
|
|
|
|
fn path_exists(&self, path: PathBuf) -> Result<bool, ErrorKind>;
|
|
|
|
|
2022-12-12 15:19:27 +01:00
|
|
|
/// Read the file at the specified path to a string.
|
|
|
|
fn read_to_string(&self, path: PathBuf) -> Result<String, ErrorKind>;
|
2022-12-12 18:02:39 +01:00
|
|
|
|
|
|
|
/// Read the directory at the specified path and return the names
|
|
|
|
/// of its entries associated with their [`FileType`].
|
|
|
|
fn read_dir(&self, path: PathBuf) -> Result<Vec<(SmolStr, FileType)>, ErrorKind>;
|
2022-12-12 15:19:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Implementation of [`EvalIO`] that simply uses the equivalent
|
|
|
|
/// standard library functions, i.e. does local file-IO.
|
2022-12-12 17:10:55 +01:00
|
|
|
#[cfg(feature = "impure")]
|
2022-12-12 15:38:28 +01:00
|
|
|
pub struct StdIO;
|
2022-12-12 15:19:27 +01:00
|
|
|
|
2022-12-12 17:10:55 +01:00
|
|
|
#[cfg(feature = "impure")]
|
2022-12-12 15:19:27 +01:00
|
|
|
impl EvalIO for StdIO {
|
2022-12-12 17:36:45 +01:00
|
|
|
fn path_exists(&self, path: PathBuf) -> Result<bool, ErrorKind> {
|
|
|
|
path.try_exists().map_err(|e| ErrorKind::IO {
|
|
|
|
path: Some(path),
|
2022-12-12 18:02:39 +01:00
|
|
|
error: Rc::new(e),
|
2022-12-12 17:36:45 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-12-12 15:19:27 +01:00
|
|
|
fn read_to_string(&self, path: PathBuf) -> Result<String, ErrorKind> {
|
|
|
|
std::fs::read_to_string(&path).map_err(|e| ErrorKind::IO {
|
|
|
|
path: Some(path),
|
2022-12-12 18:02:39 +01:00
|
|
|
error: Rc::new(e),
|
2022-12-12 15:19:27 +01:00
|
|
|
})
|
|
|
|
}
|
2022-12-12 18:02:39 +01:00
|
|
|
|
|
|
|
fn read_dir(&self, path: PathBuf) -> Result<Vec<(SmolStr, FileType)>, ErrorKind> {
|
|
|
|
let mut result = vec![];
|
|
|
|
|
|
|
|
let mk_err = |err| ErrorKind::IO {
|
|
|
|
path: Some(path.clone()),
|
|
|
|
error: Rc::new(err),
|
|
|
|
};
|
|
|
|
|
|
|
|
for entry in path.read_dir().map_err(mk_err)? {
|
|
|
|
let entry = entry.map_err(mk_err)?;
|
|
|
|
let file_type = entry
|
|
|
|
.metadata()
|
|
|
|
.map_err(|err| ErrorKind::IO {
|
|
|
|
path: Some(entry.path()),
|
|
|
|
error: Rc::new(err),
|
|
|
|
})?
|
|
|
|
.file_type();
|
|
|
|
|
|
|
|
let val = if file_type.is_dir() {
|
|
|
|
FileType::Directory
|
|
|
|
} else if file_type.is_file() {
|
|
|
|
FileType::Regular
|
|
|
|
} else if file_type.is_symlink() {
|
|
|
|
FileType::Symlink
|
|
|
|
} else {
|
|
|
|
FileType::Unknown
|
|
|
|
};
|
|
|
|
|
|
|
|
result.push((SmolStr::new(entry.file_name().to_string_lossy()), val));
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(result)
|
|
|
|
}
|
2022-12-12 15:19:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Dummy implementation of [`EvalIO`], can be used in contexts where
|
|
|
|
/// IO is not available but code should "pretend" that it is.
|
2022-12-12 15:38:28 +01:00
|
|
|
pub struct DummyIO;
|
2022-12-12 15:19:27 +01:00
|
|
|
|
|
|
|
impl EvalIO for DummyIO {
|
2022-12-12 17:36:45 +01:00
|
|
|
fn path_exists(&self, _: PathBuf) -> Result<bool, ErrorKind> {
|
|
|
|
Err(ErrorKind::NotImplemented(
|
|
|
|
"I/O methods are not implemented in DummyIO",
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
2022-12-12 15:19:27 +01:00
|
|
|
fn read_to_string(&self, _: PathBuf) -> Result<String, ErrorKind> {
|
|
|
|
Err(ErrorKind::NotImplemented(
|
|
|
|
"I/O methods are not implemented in DummyIO",
|
|
|
|
))
|
|
|
|
}
|
2022-12-12 18:02:39 +01:00
|
|
|
|
|
|
|
fn read_dir(&self, _: PathBuf) -> Result<Vec<(SmolStr, FileType)>, ErrorKind> {
|
|
|
|
Err(ErrorKind::NotImplemented(
|
|
|
|
"I/O methods are not implemented in DummyIO",
|
|
|
|
))
|
|
|
|
}
|
2022-12-12 15:19:27 +01:00
|
|
|
}
|