refactor(tvix/eval): remove code and location from struct
Instead, it's passed in the evaluate/compile_only functions, which feels more naturally. It lets us set up the Evaluation struct long before we actually feed it with data to evaluate. Now that Evaluation::new() would be accepting an empty list of arguments, we can simply implement Default, making things a bit more idiomatic. Change-Id: I4369658634909a0c504fdffa18242a130daa0239 Reviewed-on: https://cl.tvl.fyi/c/depot/+/10475 Tested-by: BuildkiteCI Reviewed-by: tazjin <tazjin@tvl.su> Autosubmit: flokli <flokli@flokli.de>
This commit is contained in:
parent
a5c5f1a29e
commit
4fba57c2c9
10 changed files with 72 additions and 69 deletions
|
@ -286,7 +286,7 @@ fn eval(model: &Model) -> Output {
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut eval = tvix_eval::Evaluation::new(&model.code, Some("/nixbolt".into()));
|
let mut eval = tvix_eval::Evaluation::default();
|
||||||
let source = eval.source_map();
|
let source = eval.source_map();
|
||||||
|
|
||||||
let result = {
|
let result = {
|
||||||
|
@ -298,7 +298,7 @@ fn eval(model: &Model) -> Output {
|
||||||
eval.runtime_observer = Some(&mut runtime_observer);
|
eval.runtime_observer = Some(&mut runtime_observer);
|
||||||
}
|
}
|
||||||
|
|
||||||
eval.evaluate()
|
eval.evaluate(&model.code, Some("/nixbolt".into()))
|
||||||
};
|
};
|
||||||
|
|
||||||
if model.display_ast {
|
if model.display_ast {
|
||||||
|
|
|
@ -62,7 +62,7 @@ struct Args {
|
||||||
/// and the result itself. The return value indicates whether
|
/// and the result itself. The return value indicates whether
|
||||||
/// evaluation succeeded.
|
/// evaluation succeeded.
|
||||||
fn interpret(code: &str, path: Option<PathBuf>, args: &Args, explain: bool) -> bool {
|
fn interpret(code: &str, path: Option<PathBuf>, args: &Args, explain: bool) -> bool {
|
||||||
let mut eval = tvix_eval::Evaluation::new_impure(code, path);
|
let mut eval = tvix_eval::Evaluation::new_impure();
|
||||||
|
|
||||||
eval.strict = args.strict;
|
eval.strict = args.strict;
|
||||||
|
|
||||||
|
@ -101,7 +101,7 @@ fn interpret(code: &str, path: Option<PathBuf>, args: &Args, explain: bool) -> b
|
||||||
eval.runtime_observer = Some(&mut runtime_observer);
|
eval.runtime_observer = Some(&mut runtime_observer);
|
||||||
}
|
}
|
||||||
|
|
||||||
eval.evaluate()
|
eval.evaluate(code, path)
|
||||||
};
|
};
|
||||||
|
|
||||||
if args.display_ast {
|
if args.display_ast {
|
||||||
|
@ -135,7 +135,7 @@ fn interpret(code: &str, path: Option<PathBuf>, args: &Args, explain: bool) -> b
|
||||||
/// Interpret the given code snippet, but only run the Tvix compiler
|
/// Interpret the given code snippet, but only run the Tvix compiler
|
||||||
/// on it and return errors and warnings.
|
/// on it and return errors and warnings.
|
||||||
fn lint(code: &str, path: Option<PathBuf>, args: &Args) -> bool {
|
fn lint(code: &str, path: Option<PathBuf>, args: &Args) -> bool {
|
||||||
let mut eval = tvix_eval::Evaluation::new_impure(code, path);
|
let mut eval = tvix_eval::Evaluation::new_impure();
|
||||||
eval.strict = args.strict;
|
eval.strict = args.strict;
|
||||||
|
|
||||||
let source_map = eval.source_map();
|
let source_map = eval.source_map();
|
||||||
|
@ -150,7 +150,7 @@ fn lint(code: &str, path: Option<PathBuf>, args: &Args) -> bool {
|
||||||
eprintln!("warning: --trace-runtime has no effect with --compile-only!");
|
eprintln!("warning: --trace-runtime has no effect with --compile-only!");
|
||||||
}
|
}
|
||||||
|
|
||||||
let result = eval.compile_only();
|
let result = eval.compile_only(code, path);
|
||||||
|
|
||||||
if args.display_ast {
|
if args.display_ast {
|
||||||
if let Some(ref expr) = result.expr {
|
if let Some(ref expr) = result.expr {
|
||||||
|
|
|
@ -68,21 +68,10 @@ pub use crate::io::StdIO;
|
||||||
///
|
///
|
||||||
/// Public fields are intended to be set by the caller. Setting all
|
/// Public fields are intended to be set by the caller. Setting all
|
||||||
/// fields is optional.
|
/// fields is optional.
|
||||||
pub struct Evaluation<'code, 'co, 'ro> {
|
pub struct Evaluation<'co, 'ro> {
|
||||||
/// The Nix source code to be evaluated.
|
|
||||||
code: &'code str,
|
|
||||||
|
|
||||||
/// Optional location of the source code (i.e. path to the file it was read
|
|
||||||
/// from). Used for error reporting, and for resolving relative paths in
|
|
||||||
/// impure functions.
|
|
||||||
location: Option<PathBuf>,
|
|
||||||
|
|
||||||
/// Source code map used for error reporting.
|
/// Source code map used for error reporting.
|
||||||
source_map: SourceCode,
|
source_map: SourceCode,
|
||||||
|
|
||||||
/// Top-level file reference for this code inside the source map.
|
|
||||||
file: Arc<codemap::File>,
|
|
||||||
|
|
||||||
/// Set of all builtins that should be available during the
|
/// Set of all builtins that should be available during the
|
||||||
/// evaluation.
|
/// evaluation.
|
||||||
///
|
///
|
||||||
|
@ -142,27 +131,15 @@ pub struct EvaluationResult {
|
||||||
pub expr: Option<rnix::ast::Expr>,
|
pub expr: Option<rnix::ast::Expr>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'code, 'co, 'ro> Evaluation<'code, 'co, 'ro> {
|
impl<'co, 'ro> Default for Evaluation<'co, 'ro> {
|
||||||
/// Initialise an `Evaluation` for the given Nix source code snippet, and
|
fn default() -> Self {
|
||||||
/// an optional code location.
|
|
||||||
pub fn new(code: &'code str, location: Option<PathBuf>) -> Self {
|
|
||||||
let source_map = SourceCode::default();
|
let source_map = SourceCode::default();
|
||||||
|
|
||||||
let location_str = location
|
|
||||||
.as_ref()
|
|
||||||
.map(|p| p.to_string_lossy().to_string())
|
|
||||||
.unwrap_or_else(|| "[code]".into());
|
|
||||||
|
|
||||||
let file = source_map.add_file(location_str, code.into());
|
|
||||||
|
|
||||||
let mut builtins = builtins::pure_builtins();
|
let mut builtins = builtins::pure_builtins();
|
||||||
builtins.extend(builtins::placeholders()); // these are temporary
|
builtins.extend(builtins::placeholders()); // these are temporary
|
||||||
|
|
||||||
Evaluation {
|
Self {
|
||||||
code,
|
|
||||||
location,
|
|
||||||
source_map,
|
source_map,
|
||||||
file,
|
|
||||||
builtins,
|
builtins,
|
||||||
src_builtins: vec![],
|
src_builtins: vec![],
|
||||||
io_handle: Box::new(DummyIO {}),
|
io_handle: Box::new(DummyIO {}),
|
||||||
|
@ -173,15 +150,21 @@ impl<'code, 'co, 'ro> Evaluation<'code, 'co, 'ro> {
|
||||||
runtime_observer: None,
|
runtime_observer: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'co, 'ro> Evaluation<'co, 'ro> {
|
||||||
#[cfg(feature = "impure")]
|
#[cfg(feature = "impure")]
|
||||||
/// Initialise an `Evaluation` for the given snippet, with all
|
/// Initialise an `Evaluation` for the given snippet, with all
|
||||||
/// impure features turned on by default.
|
/// impure features turned on by default.
|
||||||
pub fn new_impure(code: &'code str, location: Option<PathBuf>) -> Self {
|
pub fn new_impure() -> Self {
|
||||||
let mut eval = Self::new(code, location);
|
let mut eval = Self {
|
||||||
eval.enable_import = true;
|
enable_import: true,
|
||||||
|
io_handle: Box::new(StdIO),
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
|
||||||
eval.builtins.extend(builtins::impure_builtins());
|
eval.builtins.extend(builtins::impure_builtins());
|
||||||
eval.io_handle = Box::new(StdIO);
|
|
||||||
eval
|
eval
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -191,21 +174,30 @@ impl<'code, 'co, 'ro> Evaluation<'code, 'co, 'ro> {
|
||||||
self.source_map.clone()
|
self.source_map.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Only compile the provided source code. This does not *run* the
|
/// Only compile the provided source code, at an optional location of the
|
||||||
/// code, it only provides analysis (errors and warnings) of the
|
/// source code (i.e. path to the file it was read from; used for error
|
||||||
/// compiler.
|
/// reporting, and for resolving relative paths in impure functions)
|
||||||
pub fn compile_only(mut self) -> EvaluationResult {
|
/// This does not *run* the code, it only provides analysis (errors and
|
||||||
|
/// warnings) of the compiler.
|
||||||
|
pub fn compile_only(mut self, code: &str, location: Option<PathBuf>) -> EvaluationResult {
|
||||||
let mut result = EvaluationResult::default();
|
let mut result = EvaluationResult::default();
|
||||||
let source = self.source_map();
|
let source = self.source_map();
|
||||||
|
|
||||||
|
let location_str = location
|
||||||
|
.as_ref()
|
||||||
|
.map(|p| p.to_string_lossy().to_string())
|
||||||
|
.unwrap_or_else(|| "[code]".into());
|
||||||
|
|
||||||
|
let file = source.add_file(location_str, code.into());
|
||||||
|
|
||||||
let mut noop_observer = observer::NoOpObserver::default();
|
let mut noop_observer = observer::NoOpObserver::default();
|
||||||
let compiler_observer = self.compiler_observer.take().unwrap_or(&mut noop_observer);
|
let compiler_observer = self.compiler_observer.take().unwrap_or(&mut noop_observer);
|
||||||
|
|
||||||
parse_compile_internal(
|
parse_compile_internal(
|
||||||
&mut result,
|
&mut result,
|
||||||
self.code,
|
code,
|
||||||
self.file.clone(),
|
file,
|
||||||
self.location,
|
location,
|
||||||
source,
|
source,
|
||||||
self.builtins,
|
self.builtins,
|
||||||
self.src_builtins,
|
self.src_builtins,
|
||||||
|
@ -216,11 +208,20 @@ impl<'code, 'co, 'ro> Evaluation<'code, 'co, 'ro> {
|
||||||
result
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Evaluate the provided source code.
|
/// Evaluate the provided source code, at an optional location of the source
|
||||||
pub fn evaluate(mut self) -> EvaluationResult {
|
/// code (i.e. path to the file it was read from; used for error reporting,
|
||||||
|
/// and for resolving relative paths in impure functions)
|
||||||
|
pub fn evaluate(mut self, code: &str, location: Option<PathBuf>) -> EvaluationResult {
|
||||||
let mut result = EvaluationResult::default();
|
let mut result = EvaluationResult::default();
|
||||||
let source = self.source_map();
|
let source = self.source_map();
|
||||||
|
|
||||||
|
let location_str = location
|
||||||
|
.as_ref()
|
||||||
|
.map(|p| p.to_string_lossy().to_string())
|
||||||
|
.unwrap_or_else(|| "[code]".into());
|
||||||
|
|
||||||
|
let file = source.add_file(location_str, code.into());
|
||||||
|
|
||||||
let mut noop_observer = observer::NoOpObserver::default();
|
let mut noop_observer = observer::NoOpObserver::default();
|
||||||
let compiler_observer = self.compiler_observer.take().unwrap_or(&mut noop_observer);
|
let compiler_observer = self.compiler_observer.take().unwrap_or(&mut noop_observer);
|
||||||
|
|
||||||
|
@ -231,9 +232,9 @@ impl<'code, 'co, 'ro> Evaluation<'code, 'co, 'ro> {
|
||||||
|
|
||||||
let (lambda, globals) = match parse_compile_internal(
|
let (lambda, globals) = match parse_compile_internal(
|
||||||
&mut result,
|
&mut result,
|
||||||
self.code,
|
code,
|
||||||
self.file.clone(),
|
file.clone(),
|
||||||
self.location,
|
location,
|
||||||
source,
|
source,
|
||||||
self.builtins,
|
self.builtins,
|
||||||
self.src_builtins,
|
self.src_builtins,
|
||||||
|
@ -255,7 +256,7 @@ impl<'code, 'co, 'ro> Evaluation<'code, 'co, 'ro> {
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
result.warnings.push(EvalWarning {
|
result.warnings.push(EvalWarning {
|
||||||
kind: WarningKind::InvalidNixPath(err.to_string()),
|
kind: WarningKind::InvalidNixPath(err.to_string()),
|
||||||
span: self.file.span,
|
span: file.span,
|
||||||
});
|
});
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,11 +53,11 @@ fn eval_test(code_path: &str, expect_success: bool) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut eval = crate::Evaluation::new_impure(&code, Some(code_path.into()));
|
let mut eval = crate::Evaluation::new_impure();
|
||||||
eval.strict = true;
|
eval.strict = true;
|
||||||
eval.builtins.extend(mock_builtins::builtins());
|
eval.builtins.extend(mock_builtins::builtins());
|
||||||
|
|
||||||
let result = eval.evaluate();
|
let result = eval.evaluate(&code, Some(code_path.into()));
|
||||||
let failed = match result.value {
|
let failed = match result.value {
|
||||||
Some(Value::Catchable(_)) => true,
|
Some(Value::Catchable(_)) => true,
|
||||||
_ => !result.errors.is_empty(),
|
_ => !result.errors.is_empty(),
|
||||||
|
@ -106,11 +106,13 @@ fn eval_test(code_path: &str, expect_success: bool) {
|
||||||
fn identity(code_path: &str) {
|
fn identity(code_path: &str) {
|
||||||
let code = std::fs::read_to_string(code_path).expect("should be able to read test code");
|
let code = std::fs::read_to_string(code_path).expect("should be able to read test code");
|
||||||
|
|
||||||
let mut eval = crate::Evaluation::new(&code, None);
|
let eval = crate::Evaluation {
|
||||||
eval.strict = true;
|
strict: true,
|
||||||
eval.io_handle = Box::new(crate::StdIO);
|
io_handle: Box::new(crate::StdIO),
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
|
||||||
let result = eval.evaluate();
|
let result = eval.evaluate(&code, None);
|
||||||
assert!(
|
assert!(
|
||||||
result.errors.is_empty(),
|
result.errors.is_empty(),
|
||||||
"evaluation of identity test failed: {:?}",
|
"evaluation of identity test failed: {:?}",
|
||||||
|
|
|
@ -5,10 +5,10 @@ fn test_source_builtin() {
|
||||||
// Test an evaluation with a source-only builtin. The test ensures
|
// Test an evaluation with a source-only builtin. The test ensures
|
||||||
// that the artificially constructed thunking is correct.
|
// that the artificially constructed thunking is correct.
|
||||||
|
|
||||||
let mut eval = Evaluation::new_impure("builtins.testSourceBuiltin", None);
|
let mut eval = Evaluation::new_impure();
|
||||||
eval.src_builtins.push(("testSourceBuiltin", "42"));
|
eval.src_builtins.push(("testSourceBuiltin", "42"));
|
||||||
|
|
||||||
let result = eval.evaluate();
|
let result = eval.evaluate("builtins.testSourceBuiltin", None);
|
||||||
assert!(
|
assert!(
|
||||||
result.errors.is_empty(),
|
result.errors.is_empty(),
|
||||||
"evaluation failed: {:?}",
|
"evaluation failed: {:?}",
|
||||||
|
@ -25,7 +25,7 @@ fn test_source_builtin() {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn skip_broken_bytecode() {
|
fn skip_broken_bytecode() {
|
||||||
let result = Evaluation::new(/* code = */ "x", None).evaluate();
|
let result = Evaluation::default().evaluate(/* code = */ "x", None);
|
||||||
|
|
||||||
assert_eq!(result.errors.len(), 1);
|
assert_eq!(result.errors.len(), 1);
|
||||||
|
|
||||||
|
|
|
@ -51,12 +51,12 @@ fn nix_eval(expr: &str, strictness: Strictness) -> String {
|
||||||
#[track_caller]
|
#[track_caller]
|
||||||
fn compare_eval(expr: &str, strictness: Strictness) {
|
fn compare_eval(expr: &str, strictness: Strictness) {
|
||||||
let nix_result = nix_eval(expr, strictness);
|
let nix_result = nix_eval(expr, strictness);
|
||||||
let mut eval = tvix_eval::Evaluation::new(expr, None);
|
let mut eval = tvix_eval::Evaluation::default();
|
||||||
eval.strict = matches!(strictness, Strictness::Strict);
|
eval.strict = matches!(strictness, Strictness::Strict);
|
||||||
eval.io_handle = Box::new(tvix_eval::StdIO);
|
eval.io_handle = Box::new(tvix_eval::StdIO);
|
||||||
|
|
||||||
let tvix_result = eval
|
let tvix_result = eval
|
||||||
.evaluate()
|
.evaluate(expr, None)
|
||||||
.value
|
.value
|
||||||
.expect("tvix evaluation should succeed")
|
.expect("tvix evaluation should succeed")
|
||||||
.to_string();
|
.to_string();
|
||||||
|
|
|
@ -26,7 +26,7 @@ fn interpret(code: &str) {
|
||||||
// TODO: this is a bit annoying.
|
// TODO: this is a bit annoying.
|
||||||
// It'd be nice if we could set this up once and then run evaluate() with a
|
// It'd be nice if we could set this up once and then run evaluate() with a
|
||||||
// piece of code. b/262
|
// piece of code. b/262
|
||||||
let mut eval = tvix_eval::Evaluation::new_impure(code, None);
|
let mut eval = tvix_eval::Evaluation::new_impure();
|
||||||
|
|
||||||
let known_paths: Rc<RefCell<KnownPaths>> = Default::default();
|
let known_paths: Rc<RefCell<KnownPaths>> = Default::default();
|
||||||
add_derivation_builtins(&mut eval, known_paths.clone());
|
add_derivation_builtins(&mut eval, known_paths.clone());
|
||||||
|
@ -47,7 +47,7 @@ fn interpret(code: &str) {
|
||||||
),
|
),
|
||||||
));
|
));
|
||||||
|
|
||||||
let result = eval.evaluate();
|
let result = eval.evaluate(code, None);
|
||||||
|
|
||||||
assert!(result.errors.is_empty());
|
assert!(result.errors.is_empty());
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,14 +39,14 @@ mod tests {
|
||||||
/// Takes care of setting up the evaluator so it knows about the
|
/// Takes care of setting up the evaluator so it knows about the
|
||||||
// `derivation` builtin.
|
// `derivation` builtin.
|
||||||
fn eval(str: &str) -> EvaluationResult {
|
fn eval(str: &str) -> EvaluationResult {
|
||||||
let mut eval = tvix_eval::Evaluation::new_impure(str, None);
|
let mut eval = tvix_eval::Evaluation::new_impure();
|
||||||
|
|
||||||
let known_paths: Rc<RefCell<KnownPaths>> = Default::default();
|
let known_paths: Rc<RefCell<KnownPaths>> = Default::default();
|
||||||
|
|
||||||
add_derivation_builtins(&mut eval, known_paths.clone());
|
add_derivation_builtins(&mut eval, known_paths.clone());
|
||||||
|
|
||||||
// run the evaluation itself.
|
// run the evaluation itself.
|
||||||
eval.evaluate()
|
eval.evaluate(str, None)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
|
|
@ -375,7 +375,7 @@ mod tests {
|
||||||
/// Takes care of setting up the evaluator so it knows about the
|
/// Takes care of setting up the evaluator so it knows about the
|
||||||
// `derivation` builtin.
|
// `derivation` builtin.
|
||||||
fn eval(str: &str) -> EvaluationResult {
|
fn eval(str: &str) -> EvaluationResult {
|
||||||
let mut eval = tvix_eval::Evaluation::new_impure(str, None);
|
let mut eval = tvix_eval::Evaluation::new_impure();
|
||||||
|
|
||||||
let blob_service = Arc::new(MemoryBlobService::default());
|
let blob_service = Arc::new(MemoryBlobService::default());
|
||||||
let directory_service = Arc::new(MemoryDirectoryService::default());
|
let directory_service = Arc::new(MemoryDirectoryService::default());
|
||||||
|
@ -397,7 +397,7 @@ mod tests {
|
||||||
add_derivation_builtins(&mut eval, known_paths.clone());
|
add_derivation_builtins(&mut eval, known_paths.clone());
|
||||||
|
|
||||||
// run the evaluation itself.
|
// run the evaluation itself.
|
||||||
eval.evaluate()
|
eval.evaluate(str, None)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Helper function that takes a &Path, and invokes a tvix evaluator coercing that path to a string
|
/// Helper function that takes a &Path, and invokes a tvix evaluator coercing that path to a string
|
||||||
|
|
|
@ -46,12 +46,12 @@ where
|
||||||
F: FnOnce(&mut Evaluation),
|
F: FnOnce(&mut Evaluation),
|
||||||
{
|
{
|
||||||
// First step is to evaluate the Nix code ...
|
// First step is to evaluate the Nix code ...
|
||||||
let mut eval = Evaluation::new(src, None);
|
let mut eval = Evaluation::default();
|
||||||
config(&mut eval);
|
config(&mut eval);
|
||||||
|
|
||||||
eval.strict = true;
|
eval.strict = true;
|
||||||
let source = eval.source_map();
|
let source = eval.source_map();
|
||||||
let result = eval.evaluate();
|
let result = eval.evaluate(src, None);
|
||||||
|
|
||||||
if !result.errors.is_empty() {
|
if !result.errors.is_empty() {
|
||||||
return Err(Error::NixErrors {
|
return Err(Error::NixErrors {
|
||||||
|
|
Loading…
Reference in a new issue