From 155933c661c932179edb0ba8b92d8326f2046371 Mon Sep 17 00:00:00 2001 From: jedel1043 Date: Sun, 5 Jan 2025 00:58:24 -0600 Subject: [PATCH] Cleanup CLI to use eyre + refactor patterns --- .vscode/launch.json | 2 - Cargo.lock | 1 + cli/Cargo.toml | 1 + cli/src/main.rs | 310 ++++++++++++++++++----------------- core/parser/src/error/mod.rs | 2 + 5 files changed, 165 insertions(+), 151 deletions(-) diff --git a/.vscode/launch.json b/.vscode/launch.json index 3b441e7c6fc..4cbbdfb70c0 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -51,13 +51,11 @@ { "id": "filePath", "description": "Relative path to the file to run", - "default": "debug/script.js", "type": "promptString" }, { "id": "modulePath", "description": "Relative path to the module root directory", - "default": "debug", "type": "promptString" }, { diff --git a/Cargo.lock b/Cargo.lock index 06aebcc86bd..f43d87975aa 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -345,6 +345,7 @@ dependencies = [ "boa_parser", "boa_runtime", "clap", + "color-eyre", "colored", "dhat", "jemallocator", diff --git a/cli/Cargo.toml b/cli/Cargo.toml index 883a3882f27..d4d5bf6c0be 100644 --- a/cli/Cargo.toml +++ b/cli/Cargo.toml @@ -24,6 +24,7 @@ regex.workspace = true phf = { workspace = true, features = ["macros"] } pollster.workspace = true dhat = { workspace = true, optional = true } +color-eyre.workspace = true [features] default = ["boa_engine/annex-b", "boa_engine/experimental", "boa_engine/intl_bundled"] diff --git a/cli/src/main.rs b/cli/src/main.rs index 8d5312a92b3..8e8d0826df9 100644 --- a/cli/src/main.rs +++ b/cli/src/main.rs @@ -18,15 +18,26 @@ use boa_engine::{ optimizer::OptimizerOptions, script::Script, vm::flowgraph::{Direction, Graph}, - Context, JsError, JsNativeError, JsResult, Source, + Context, JsError, Source, }; +use boa_parser::source::ReadChar; use clap::{Parser, ValueEnum, ValueHint}; +use color_eyre::{ + eyre::{eyre, WrapErr}, + Result, Section, +}; use colored::Colorize; use debug::init_boa_debug_object; use rustyline::{config::Config, error::ReadlineError, EditMode, Editor}; use std::{ - cell::RefCell, collections::VecDeque, eprintln, fs::read, fs::OpenOptions, io, path::PathBuf, - println, rc::Rc, + cell::RefCell, + collections::VecDeque, + eprintln, + fs::OpenOptions, + io, + path::{Path, PathBuf}, + println, + rc::Rc, }; #[cfg(all( @@ -177,19 +188,16 @@ enum FlowgraphDirection { /// /// Returns a error of type String with a error message, /// if the source has a syntax or parsing error. -fn dump(src: &S, args: &Opt, context: &mut Context) -> Result<(), String> -where - S: AsRef<[u8]> + ?Sized, -{ +fn dump(src: Source<'_, R>, args: &Opt, context: &mut Context) -> Result<()> { if let Some(arg) = args.dump_ast { let arg = arg.unwrap_or_default(); - let mut parser = boa_parser::Parser::new(Source::from_bytes(src)); + let mut parser = boa_parser::Parser::new(src); let dump = if args.module { let scope = context.realm().scope().clone(); let module = parser .parse_module(&scope, context.interner_mut()) - .map_err(|e| format!("Uncaught SyntaxError: {e}"))?; + .map_err(|e| eyre!("Uncaught SyntaxError: {e}"))?; match arg { DumpFormat::Json => serde_json::to_string(&module) @@ -202,7 +210,7 @@ where let scope = context.realm().scope().clone(); let mut script = parser .parse_script(&scope, context.interner_mut()) - .map_err(|e| format!("Uncaught SyntaxError: {e}"))?; + .map_err(|e| eyre!("Uncaught SyntaxError: {e}"))?; if args.optimize { context.optimize_statement_list(script.statements_mut()); @@ -223,14 +231,16 @@ where Ok(()) } -fn generate_flowgraph( +fn generate_flowgraph( context: &mut Context, - src: &[u8], + src: Source<'_, R>, format: FlowgraphFormat, direction: Option, -) -> JsResult { - let script = Script::parse(Source::from_bytes(src), None, context)?; - let code = script.codeblock(context)?; +) -> Result { + let script = Script::parse(src, None, context).map_err(|e| e.into_erased(context))?; + let code = script + .codeblock(context) + .map_err(|e| e.into_erased(context))?; let direction = match direction { Some(FlowgraphDirection::TopToBottom) | None => Direction::TopToBottom, @@ -248,96 +258,95 @@ fn generate_flowgraph( Ok(result) } -fn evaluate_files( +fn evaluate_file( + file: &Path, args: &Opt, context: &mut Context, loader: &SimpleModuleLoader, -) -> Result<(), io::Error> { - for file in &args.files { - let buffer = read(file)?; - - if args.has_dump_flag() { - if let Err(e) = dump(&buffer, args, context) { - eprintln!("{e}"); - } - } else if let Some(flowgraph) = args.flowgraph { - match generate_flowgraph( - context, - &buffer, - flowgraph.unwrap_or(FlowgraphFormat::Graphviz), - args.flowgraph_direction, - ) { - Ok(v) => println!("{v}"), - Err(v) => eprintln!("Uncaught {v}"), - } - } else if args.module { - let result: JsResult = (|| { - let module = Module::parse(Source::from_bytes(&buffer), None, context)?; - - loader.insert( - file.canonicalize() - .map_err(|e| JsNativeError::typ().with_message(e.to_string()))?, - module.clone(), - ); +) -> Result<()> { + if args.has_dump_flag() { + return dump(Source::from_filepath(file)?, args, context); + } - let promise = module.load_link_evaluate(context); + if let Some(flowgraph) = args.flowgraph { + let flowgraph = generate_flowgraph( + context, + Source::from_filepath(file)?, + flowgraph.unwrap_or(FlowgraphFormat::Graphviz), + args.flowgraph_direction, + )?; - context.run_jobs(); - Ok(promise.state()) - })(); + println!("{flowgraph}"); - match result { - Ok(PromiseState::Pending) => { - eprintln!("module `{}` didn't execute", file.display()); - } - Ok(PromiseState::Fulfilled(_)) => {} - Ok(PromiseState::Rejected(err)) => { - eprintln!("Uncaught {}", err.display()); - - if let Ok(err) = JsError::from_opaque(err).try_native(context) { - if let Some(cause) = err.cause() { - eprintln!("\tCaused by: {cause}"); - } - } - } - Err(err) => { - eprintln!("Uncaught {err}"); + return Ok(()); + } - if let Ok(err) = err.try_native(context) { - if let Some(cause) = err.cause() { - eprintln!("\tCaused by: {cause}"); - } - } - } - } - } else { - match context.eval(Source::from_bytes(&buffer)) { - Ok(v) => println!("{}", v.display()), - Err(v) => eprintln!("Uncaught {v}"), + if args.module { + let module = Module::parse(Source::from_filepath(file)?, None, context) + .map_err(|e| e.into_erased(context))?; + + loader.insert( + file.canonicalize() + .wrap_err("could not canonicalize input file path")?, + module.clone(), + ); + + let promise = module.load_link_evaluate(context); + context.run_jobs(); + let result = promise.state(); + + return match result { + PromiseState::Pending => Err(eyre!("module didn't execute")), + PromiseState::Fulfilled(_) => Ok(()), + PromiseState::Rejected(err) => { + return Err(JsError::from_opaque(err).into_erased(context).into()) } - context.run_jobs(); - } + }; + } + + match context.eval(Source::from_filepath(file)?) { + Ok(v) => println!("{}", v.display()), + Err(v) => eprintln!("Uncaught {v}"), } + context.run_jobs(); Ok(()) } -fn main() -> Result<(), io::Error> { +fn evaluate_files(args: &Opt, context: &mut Context, loader: &SimpleModuleLoader) { + for file in &args.files { + let Err(err) = evaluate_file(file, args, context, loader) + .wrap_err_with(|| eyre!("could not evaluate file `{}`", file.display())) + else { + continue; + }; + + eprintln!("{err:?}"); + } +} + +fn main() -> Result<()> { + color_eyre::config::HookBuilder::default() + .display_location_section(false) + .display_env_section(false) + .install()?; + + if std::env::var("RUST_BACKTRACE").is_err() { + std::env::set_var("RUST_BACKTRACE", "0"); + } + #[cfg(feature = "dhat")] let _profiler = dhat::Profiler::new_heap(); let args = Opt::parse(); let queue = Rc::new(Jobs::default()); - let loader = Rc::new( - SimpleModuleLoader::new(&args.root) - .map_err(|e| io::Error::new(io::ErrorKind::Other, e.to_string()))?, - ); + let loader = Rc::new(SimpleModuleLoader::new(&args.root).map_err(|e| eyre!(e.to_string()))?); let mut context = ContextBuilder::new() .job_queue(queue) .module_loader(loader.clone()) .build() - .expect("cannot fail with default global object"); + .map_err(|e| eyre!(e.to_string()))?; // Strict mode context.strict(args.strict); @@ -358,83 +367,86 @@ fn main() -> Result<(), io::Error> { optimizer_options.set(OptimizerOptions::OPTIMIZE_ALL, args.optimize); context.set_optimizer_options(optimizer_options); - if args.files.is_empty() { - let config = Config::builder() - .keyseq_timeout(Some(1)) - .edit_mode(if args.vi_mode { - EditMode::Vi - } else { - EditMode::Emacs - }) - .build(); - - let mut editor = - Editor::with_config(config).map_err(|e| io::Error::new(io::ErrorKind::Other, e))?; - // Check if the history file exists. If it does, create it. - OpenOptions::new() - .read(true) - .write(true) - .create(true) - .truncate(false) - .open(CLI_HISTORY)?; - editor.load_history(CLI_HISTORY).map_err(|err| match err { - ReadlineError::Io(e) => e, - e => io::Error::new(io::ErrorKind::Other, e), - })?; - let readline = ">> "; - editor.set_helper(Some(helper::RLHelper::new(readline))); + if !args.files.is_empty() { + evaluate_files(&args, &mut context, &loader); + return Ok(()); + } - loop { - match editor.readline(readline) { - Ok(line) if line == ".exit" => break, - Err(ReadlineError::Interrupted | ReadlineError::Eof) => break, - - Ok(line) => { - editor - .add_history_entry(&line) - .map_err(|e| io::Error::new(io::ErrorKind::Other, e))?; - - if args.has_dump_flag() { - if let Err(e) = dump(&line, &args, &mut context) { - eprintln!("{e}"); - } - } else if let Some(flowgraph) = args.flowgraph { - match generate_flowgraph( - &mut context, - line.trim_end().as_bytes(), - flowgraph.unwrap_or(FlowgraphFormat::Graphviz), - args.flowgraph_direction, - ) { - Ok(v) => println!("{v}"), - Err(v) => eprintln!("Uncaught {v}"), + let config = Config::builder() + .keyseq_timeout(Some(1)) + .edit_mode(if args.vi_mode { + EditMode::Vi + } else { + EditMode::Emacs + }) + .build(); + + let mut editor = + Editor::with_config(config).wrap_err("failed to set the editor configuration")?; + // Check if the history file exists. If it doesn't, create it. + OpenOptions::new() + .read(true) + .write(true) + .create(true) + .truncate(false) + .open(CLI_HISTORY)?; + editor + .load_history(CLI_HISTORY) + .wrap_err("failed to read history file `.boa_history`")?; + let readline = ">> "; + editor.set_helper(Some(helper::RLHelper::new(readline))); + + loop { + match editor.readline(readline) { + Ok(line) if line == ".exit" => break, + Err(ReadlineError::Interrupted | ReadlineError::Eof) => break, + + Ok(line) => { + editor + .add_history_entry(&line) + .map_err(|e| io::Error::new(io::ErrorKind::Other, e))?; + + if args.has_dump_flag() { + if let Err(e) = dump(Source::from_bytes(&line), &args, &mut context) { + eprintln!("{e:?}"); + } + } else if let Some(flowgraph) = args.flowgraph { + match generate_flowgraph( + &mut context, + Source::from_bytes(line.trim_end()), + flowgraph.unwrap_or(FlowgraphFormat::Graphviz), + args.flowgraph_direction, + ) { + Ok(v) => println!("{v}"), + Err(v) => eprintln!("{v:?}"), + } + } else { + match context.eval(Source::from_bytes(line.trim_end())) { + Ok(v) => { + println!("{}", v.display()); } - } else { - match context.eval(Source::from_bytes(line.trim_end())) { - Ok(v) => { - println!("{}", v.display()); - } - Err(v) => { - eprintln!("{}: {}", "Uncaught".red(), v.to_string().red()); - } + Err(v) => { + eprintln!("{}: {}", "Uncaught".red(), v.to_string().red()); } - context.run_jobs(); } + context.run_jobs(); } + } - Err(err) => { - eprintln!("Unknown error: {err:?}"); - break; - } + Err(err) => { + let final_error = eyre!("could not read the next line of the input"); + let final_error = if let Err(e) = editor.save_history(CLI_HISTORY) { + final_error.error(e) + } else { + final_error + }; + return Err(final_error.error(err)); } } - - editor - .save_history(CLI_HISTORY) - .expect("could not save CLI history"); - } else { - evaluate_files(&args, &mut context, &loader)?; } + editor.save_history(CLI_HISTORY)?; + Ok(()) } diff --git a/core/parser/src/error/mod.rs b/core/parser/src/error/mod.rs index 727c247bbeb..502498234c5 100644 --- a/core/parser/src/error/mod.rs +++ b/core/parser/src/error/mod.rs @@ -231,3 +231,5 @@ impl fmt::Display for Error { } } } + +impl std::error::Error for Error {}