2023-01-14 13:45:22 +01:00
|
|
|
use std::cell::RefCell;
|
|
|
|
use std::rc::Rc;
|
2023-06-09 17:22:25 +02:00
|
|
|
use std::sync::Arc;
|
2022-10-08 20:22:26 +02:00
|
|
|
use std::{fs, path::PathBuf};
|
2023-11-03 12:34:37 +01:00
|
|
|
use tvix_glue::known_paths::KnownPaths;
|
2023-11-03 15:40:32 +01:00
|
|
|
use tvix_glue::{builtins::add_derivation_builtins, configure_nix_path};
|
2022-08-11 23:27:02 +02:00
|
|
|
|
2022-09-18 21:59:59 +02:00
|
|
|
use clap::Parser;
|
2022-08-11 23:27:02 +02:00
|
|
|
use rustyline::{error::ReadlineError, Editor};
|
2023-09-21 21:32:44 +02:00
|
|
|
use tvix_castore::blobservice::MemoryBlobService;
|
|
|
|
use tvix_castore::directoryservice::MemoryDirectoryService;
|
2022-12-09 11:49:36 +01:00
|
|
|
use tvix_eval::observer::{DisassemblingObserver, TracingObserver};
|
2023-06-22 17:57:50 +02:00
|
|
|
use tvix_eval::Value;
|
2023-11-03 12:34:37 +01:00
|
|
|
use tvix_glue::tvix_store_io::TvixStoreIO;
|
2023-05-14 20:21:27 +02:00
|
|
|
use tvix_store::pathinfoservice::MemoryPathInfoService;
|
2022-08-04 15:43:51 +02:00
|
|
|
|
2022-09-18 21:59:59 +02:00
|
|
|
#[derive(Parser)]
|
|
|
|
struct Args {
|
|
|
|
/// Path to a script to evaluate
|
|
|
|
script: Option<PathBuf>,
|
|
|
|
|
2022-10-11 01:41:42 +02:00
|
|
|
#[clap(long, short = 'E')]
|
|
|
|
expr: Option<String>,
|
2022-12-09 11:29:08 +01:00
|
|
|
|
2022-12-09 11:49:36 +01:00
|
|
|
/// Dump the raw AST to stdout before interpreting
|
|
|
|
#[clap(long, env = "TVIX_DISPLAY_AST")]
|
|
|
|
display_ast: bool,
|
|
|
|
|
|
|
|
/// Dump the bytecode to stdout before evaluating
|
|
|
|
#[clap(long, env = "TVIX_DUMP_BYTECODE")]
|
|
|
|
dump_bytecode: bool,
|
|
|
|
|
|
|
|
/// Trace the runtime of the VM
|
|
|
|
#[clap(long, env = "TVIX_TRACE_RUNTIME")]
|
|
|
|
trace_runtime: bool,
|
|
|
|
|
2023-01-05 10:56:19 +01:00
|
|
|
/// Only compile, but do not execute code. This will make Tvix act
|
|
|
|
/// sort of like a linter.
|
|
|
|
#[clap(long)]
|
|
|
|
compile_only: bool,
|
|
|
|
|
2023-05-25 10:50:13 +02:00
|
|
|
/// Don't print warnings.
|
|
|
|
#[clap(long)]
|
|
|
|
no_warnings: bool,
|
|
|
|
|
2022-12-09 11:33:06 +01:00
|
|
|
/// A colon-separated list of directories to use to resolve `<...>`-style paths
|
|
|
|
#[clap(long, short = 'I', env = "NIX_PATH")]
|
|
|
|
nix_search_path: Option<String>,
|
|
|
|
|
2022-12-09 11:29:08 +01:00
|
|
|
/// Print "raw" (unquoted) output.
|
|
|
|
#[clap(long)]
|
|
|
|
raw: bool,
|
2023-03-17 22:15:03 +01:00
|
|
|
|
|
|
|
/// Strictly evaluate values, traversing them and forcing e.g.
|
|
|
|
/// elements of lists and attribute sets before printing the
|
|
|
|
/// return value.
|
|
|
|
#[clap(long)]
|
|
|
|
strict: bool,
|
2022-12-08 22:19:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Interprets the given code snippet, printing out warnings, errors
|
|
|
|
/// and the result itself. The return value indicates whether
|
|
|
|
/// evaluation succeeded.
|
2022-12-19 10:58:39 +01:00
|
|
|
fn interpret(code: &str, path: Option<PathBuf>, args: &Args, explain: bool) -> bool {
|
refactor(tvix/eval): streamline construction of globals/builtins
Previously the construction of globals (a compiler-only concept) and
builtins (a (now) user-facing API) was intermingled between multiple
different modules, and kind of difficult to understand.
The complexity of this had grown in large part due to the
implementation of `builtins.import`, which required the notorious
"knot-tying" trick using Rc::new_cyclic (see cl/7097) for constructing
the set of globals.
As part of the new `Evaluation` API users should have the ability to
bring their own builtins, and control explicitly whether or not impure
builtins are available (regardless of whether they're compiled in or
not).
To streamline the construction and allow the new API features to work,
this commit restructures things by making these changes:
1. The `tvix_eval::builtins` module is now only responsible for
exporting sets of builtins. It no longer has any knowledge of
whether or not certain sets (e.g. only pure, or pure+impure) are
enabled, and it has no control over which builtins are globally
available (this is now handled in the compiler).
2. The compiler module is now responsible for both constructing the
final attribute set of builtins from the set of builtins supplied
by a user, as well as for populating its globals (that is
identifiers which are available at the top-level scope).
3. The `Evaluation` API now carries a `builtins` field which is
populated with the pure builtins by default, and can be extended by
users.
4. The `import` feature has been moved into the compiler, as a
special case. In general, builtins no longer have the ability to
reference the "fix point" of the globals set.
This should not change any functionality, and in fact preserves minor
differences between Tvix/Nix that we already had (such as
`builtins.builtins` not existing).
Change-Id: Icdf5dd50eb81eb9260d89269d6e08b1e67811a2c
Reviewed-on: https://cl.tvl.fyi/c/depot/+/7738
Reviewed-by: sterni <sternenseemann@systemli.org>
Autosubmit: tazjin <tazjin@tvl.su>
Tested-by: BuildkiteCI
Reviewed-by: flokli <flokli@flokli.de>
2023-01-03 20:30:49 +01:00
|
|
|
let mut eval = tvix_eval::Evaluation::new_impure(code, path);
|
2023-01-14 13:45:22 +01:00
|
|
|
|
2023-03-17 22:15:03 +01:00
|
|
|
eval.strict = args.strict;
|
2023-05-14 20:21:27 +02:00
|
|
|
|
2023-06-09 17:22:25 +02:00
|
|
|
let blob_service = Arc::new(MemoryBlobService::default());
|
|
|
|
let directory_service = Arc::new(MemoryDirectoryService::default());
|
2023-06-12 15:04:56 +02:00
|
|
|
let path_info_service = Arc::new(MemoryPathInfoService::new(
|
|
|
|
blob_service.clone(),
|
|
|
|
directory_service.clone(),
|
|
|
|
));
|
2023-05-14 20:21:27 +02:00
|
|
|
|
2023-11-03 12:34:37 +01:00
|
|
|
let known_paths: Rc<RefCell<KnownPaths>> = Default::default();
|
|
|
|
add_derivation_builtins(&mut eval, known_paths.clone());
|
2023-11-03 13:03:19 +01:00
|
|
|
configure_nix_path(&mut eval, &args.nix_search_path);
|
refactor(tvix/store/blobsvc): make BlobStore async
We previously kept the trait of a BlobService sync.
This however had some annoying consequences:
- It became more and more complicated to track when we're in a context
with an async runtime in the context or not, producing bugs like
https://b.tvl.fyi/issues/304
- The sync trait shielded away async clients from async worloads,
requiring manual block_on code inside the gRPC client code, and
spawn_blocking calls in consumers of the trait, even if they were
async (like the gRPC server)
- We had to write our own custom glue code (SyncReadIntoAsyncRead)
to convert a sync io::Read into a tokio::io::AsyncRead, which already
existed in tokio internally, but upstream ia hesitant to expose.
This now makes the BlobService trait async (via the async_trait macro,
like we already do in various gRPC parts), and replaces the sync readers
and writers with their async counterparts.
Tests interacting with a BlobService now need to have an async runtime
available, the easiest way for this is to mark the test functions
with the tokio::test macro, allowing us to directly .await in the test
function.
In places where we don't have an async runtime available from context
(like tvix-cli), we can pass one down explicitly.
Now that we don't provide a sync interface anymore, the (sync) FUSE
library now holds a pointer to a tokio runtime handle, and needs to at
least have 2 threads available when talking to a blob service (which is
why some of the tests now use the multi_thread flavor).
The FUSE tests got a bit more verbose, as we couldn't use the
setup_and_mount function accepting a callback anymore. We can hopefully
move some of the test fixture setup to rstest in the future to make this
less repetitive.
Co-Authored-By: Connor Brewster <cbrewster@hey.com>
Change-Id: Ia0501b606e32c852d0108de9c9016b21c94a3c05
Reviewed-on: https://cl.tvl.fyi/c/depot/+/9329
Reviewed-by: Connor Brewster <cbrewster@hey.com>
Tested-by: BuildkiteCI
Reviewed-by: raitobezarius <tvl@lahfa.xyz>
2023-09-13 14:20:21 +02:00
|
|
|
|
2023-11-03 12:34:37 +01:00
|
|
|
let tokio_runtime = tokio::runtime::Runtime::new().unwrap();
|
|
|
|
eval.io_handle = Box::new(tvix_glue::tvix_io::TvixIO::new(
|
2023-05-17 17:33:22 +02:00
|
|
|
known_paths.clone(),
|
refactor(tvix/store/blobsvc): make BlobStore async
We previously kept the trait of a BlobService sync.
This however had some annoying consequences:
- It became more and more complicated to track when we're in a context
with an async runtime in the context or not, producing bugs like
https://b.tvl.fyi/issues/304
- The sync trait shielded away async clients from async worloads,
requiring manual block_on code inside the gRPC client code, and
spawn_blocking calls in consumers of the trait, even if they were
async (like the gRPC server)
- We had to write our own custom glue code (SyncReadIntoAsyncRead)
to convert a sync io::Read into a tokio::io::AsyncRead, which already
existed in tokio internally, but upstream ia hesitant to expose.
This now makes the BlobService trait async (via the async_trait macro,
like we already do in various gRPC parts), and replaces the sync readers
and writers with their async counterparts.
Tests interacting with a BlobService now need to have an async runtime
available, the easiest way for this is to mark the test functions
with the tokio::test macro, allowing us to directly .await in the test
function.
In places where we don't have an async runtime available from context
(like tvix-cli), we can pass one down explicitly.
Now that we don't provide a sync interface anymore, the (sync) FUSE
library now holds a pointer to a tokio runtime handle, and needs to at
least have 2 threads available when talking to a blob service (which is
why some of the tests now use the multi_thread flavor).
The FUSE tests got a bit more verbose, as we couldn't use the
setup_and_mount function accepting a callback anymore. We can hopefully
move some of the test fixture setup to rstest in the future to make this
less repetitive.
Co-Authored-By: Connor Brewster <cbrewster@hey.com>
Change-Id: Ia0501b606e32c852d0108de9c9016b21c94a3c05
Reviewed-on: https://cl.tvl.fyi/c/depot/+/9329
Reviewed-by: Connor Brewster <cbrewster@hey.com>
Tested-by: BuildkiteCI
Reviewed-by: raitobezarius <tvl@lahfa.xyz>
2023-09-13 14:20:21 +02:00
|
|
|
TvixStoreIO::new(
|
|
|
|
blob_service,
|
|
|
|
directory_service,
|
|
|
|
path_info_service,
|
|
|
|
tokio_runtime.handle().clone(),
|
|
|
|
),
|
2023-05-17 17:33:22 +02:00
|
|
|
));
|
2023-03-04 01:43:59 +01:00
|
|
|
|
2022-12-09 11:16:01 +01:00
|
|
|
let source_map = eval.source_map();
|
2022-12-09 11:49:36 +01:00
|
|
|
let result = {
|
|
|
|
let mut compiler_observer =
|
|
|
|
DisassemblingObserver::new(source_map.clone(), std::io::stderr());
|
|
|
|
if args.dump_bytecode {
|
|
|
|
eval.compiler_observer = Some(&mut compiler_observer);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut runtime_observer = TracingObserver::new(std::io::stderr());
|
|
|
|
if args.trace_runtime {
|
|
|
|
eval.runtime_observer = Some(&mut runtime_observer);
|
|
|
|
}
|
|
|
|
|
|
|
|
eval.evaluate()
|
|
|
|
};
|
|
|
|
|
|
|
|
if args.display_ast {
|
|
|
|
if let Some(ref expr) = result.expr {
|
|
|
|
eprintln!("AST: {}", tvix_eval::pretty_print_expr(expr));
|
|
|
|
}
|
|
|
|
}
|
2022-12-08 22:19:22 +01:00
|
|
|
|
|
|
|
for error in &result.errors {
|
|
|
|
error.fancy_format_stderr(&source_map);
|
|
|
|
}
|
|
|
|
|
2023-05-25 10:50:13 +02:00
|
|
|
if !args.no_warnings {
|
|
|
|
for warning in &result.warnings {
|
|
|
|
warning.fancy_format_stderr(&source_map);
|
|
|
|
}
|
2022-12-08 22:19:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(value) = result.value.as_ref() {
|
2022-12-19 10:58:39 +01:00
|
|
|
if explain {
|
|
|
|
println!("=> {}", value.explain());
|
|
|
|
} else {
|
|
|
|
println_result(value, args.raw);
|
|
|
|
}
|
2022-12-08 22:19:22 +01:00
|
|
|
}
|
2022-10-11 01:41:42 +02:00
|
|
|
|
2022-12-08 22:19:22 +01:00
|
|
|
// inform the caller about any errors
|
|
|
|
result.errors.is_empty()
|
2022-09-18 21:59:59 +02:00
|
|
|
}
|
|
|
|
|
2023-01-05 10:56:19 +01:00
|
|
|
/// Interpret the given code snippet, but only run the Tvix compiler
|
|
|
|
/// on it and return errors and warnings.
|
|
|
|
fn lint(code: &str, path: Option<PathBuf>, args: &Args) -> bool {
|
|
|
|
let mut eval = tvix_eval::Evaluation::new_impure(code, path);
|
2023-03-17 22:15:03 +01:00
|
|
|
eval.strict = args.strict;
|
|
|
|
|
2023-01-05 10:56:19 +01:00
|
|
|
let source_map = eval.source_map();
|
|
|
|
|
|
|
|
let mut compiler_observer = DisassemblingObserver::new(source_map.clone(), std::io::stderr());
|
|
|
|
|
|
|
|
if args.dump_bytecode {
|
|
|
|
eval.compiler_observer = Some(&mut compiler_observer);
|
|
|
|
}
|
|
|
|
|
|
|
|
if args.trace_runtime {
|
|
|
|
eprintln!("warning: --trace-runtime has no effect with --compile-only!");
|
|
|
|
}
|
|
|
|
|
|
|
|
let result = eval.compile_only();
|
|
|
|
|
|
|
|
if args.display_ast {
|
|
|
|
if let Some(ref expr) = result.expr {
|
|
|
|
eprintln!("AST: {}", tvix_eval::pretty_print_expr(expr));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for error in &result.errors {
|
|
|
|
error.fancy_format_stderr(&source_map);
|
|
|
|
}
|
|
|
|
|
|
|
|
for warning in &result.warnings {
|
|
|
|
warning.fancy_format_stderr(&source_map);
|
|
|
|
}
|
|
|
|
|
|
|
|
// inform the caller about any errors
|
|
|
|
result.errors.is_empty()
|
|
|
|
}
|
|
|
|
|
2022-08-04 15:29:38 +02:00
|
|
|
fn main() {
|
2022-09-18 21:59:59 +02:00
|
|
|
let args = Args::parse();
|
2022-08-04 15:43:51 +02:00
|
|
|
|
2022-12-09 11:29:08 +01:00
|
|
|
if let Some(file) = &args.script {
|
|
|
|
run_file(file.clone(), &args)
|
|
|
|
} else if let Some(expr) = &args.expr {
|
2022-12-19 10:58:39 +01:00
|
|
|
if !interpret(expr, None, &args, false) {
|
2022-12-08 22:19:22 +01:00
|
|
|
std::process::exit(1);
|
2022-10-11 01:41:42 +02:00
|
|
|
}
|
2022-08-04 15:43:51 +02:00
|
|
|
} else {
|
2022-12-09 11:29:08 +01:00
|
|
|
run_prompt(&args)
|
2022-08-04 15:43:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-09 11:29:08 +01:00
|
|
|
fn run_file(mut path: PathBuf, args: &Args) {
|
2022-10-08 20:22:26 +02:00
|
|
|
if path.is_dir() {
|
|
|
|
path.push("default.nix");
|
|
|
|
}
|
|
|
|
let contents = fs::read_to_string(&path).expect("failed to read the input file");
|
2022-12-08 22:19:22 +01:00
|
|
|
|
2023-01-05 10:56:19 +01:00
|
|
|
let success = if args.compile_only {
|
|
|
|
lint(&contents, Some(path), args)
|
|
|
|
} else {
|
|
|
|
interpret(&contents, Some(path), args, false)
|
|
|
|
};
|
|
|
|
|
|
|
|
if !success {
|
2022-12-08 22:19:22 +01:00
|
|
|
std::process::exit(1);
|
2022-08-11 23:27:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-25 23:47:39 +01:00
|
|
|
fn println_result(result: &Value, raw: bool) {
|
|
|
|
if raw {
|
|
|
|
println!("{}", result.to_str().unwrap().as_str())
|
|
|
|
} else {
|
|
|
|
println!("=> {} :: {}", result, result.type_of())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-11 23:27:02 +02:00
|
|
|
fn state_dir() -> Option<PathBuf> {
|
|
|
|
let mut path = dirs::data_dir();
|
2022-08-14 01:51:09 +02:00
|
|
|
if let Some(p) = path.as_mut() {
|
|
|
|
p.push("tvix")
|
|
|
|
}
|
2022-08-11 23:27:02 +02:00
|
|
|
path
|
2022-08-04 15:43:51 +02:00
|
|
|
}
|
|
|
|
|
2022-12-09 11:29:08 +01:00
|
|
|
fn run_prompt(args: &Args) {
|
2022-08-11 23:27:02 +02:00
|
|
|
let mut rl = Editor::<()>::new().expect("should be able to launch rustyline");
|
|
|
|
|
2023-01-05 10:56:19 +01:00
|
|
|
if args.compile_only {
|
|
|
|
eprintln!("warning: `--compile-only` has no effect on REPL usage!");
|
|
|
|
}
|
|
|
|
|
2022-08-11 23:27:02 +02:00
|
|
|
let history_path = match state_dir() {
|
2022-08-14 19:08:15 +02:00
|
|
|
// Attempt to set up these paths, but do not hard fail if it
|
|
|
|
// doesn't work.
|
2022-08-11 23:27:02 +02:00
|
|
|
Some(mut path) => {
|
2022-08-26 18:41:56 +02:00
|
|
|
let _ = std::fs::create_dir_all(&path);
|
2022-08-11 23:27:02 +02:00
|
|
|
path.push("history.txt");
|
2022-08-26 18:41:56 +02:00
|
|
|
let _ = rl.load_history(&path);
|
2022-08-11 23:27:02 +02:00
|
|
|
Some(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
None => None,
|
|
|
|
};
|
2022-08-04 15:43:51 +02:00
|
|
|
|
|
|
|
loop {
|
2022-08-11 23:27:02 +02:00
|
|
|
let readline = rl.readline("tvix-repl> ");
|
|
|
|
match readline {
|
|
|
|
Ok(line) => {
|
|
|
|
if line.is_empty() {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2022-09-05 04:12:36 +02:00
|
|
|
rl.add_history_entry(&line);
|
2022-12-19 10:58:39 +01:00
|
|
|
|
|
|
|
if let Some(without_prefix) = line.strip_prefix(":d ") {
|
|
|
|
interpret(without_prefix, None, args, true);
|
|
|
|
} else {
|
|
|
|
interpret(&line, None, args, false);
|
|
|
|
}
|
2022-08-11 23:27:02 +02:00
|
|
|
}
|
|
|
|
Err(ReadlineError::Interrupted) | Err(ReadlineError::Eof) => break,
|
|
|
|
|
|
|
|
Err(err) => {
|
|
|
|
eprintln!("error: {}", err);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2022-08-04 15:43:51 +02:00
|
|
|
}
|
|
|
|
|
2022-08-11 23:27:02 +02:00
|
|
|
if let Some(path) = history_path {
|
|
|
|
rl.save_history(&path).unwrap();
|
2022-08-04 15:43:51 +02:00
|
|
|
}
|
2022-08-04 15:29:38 +02:00
|
|
|
}
|