tvl-depot/users/tazjin/rlox/src/main.rs
Vincent Ambo 0d0b43ed88 fix(users/tazjin): rustfmt code with non-default settings
rustfmt only sometimes detects path-based nested config
files (probably some kind of race?), so my users folder uses a
separate formatting check for rustfmt to avoid flaky CI. Enough flakes
around already ...

Change-Id: Ifd862f9974f071b3a256643dd8e56c019116156a
Reviewed-on: https://cl.tvl.fyi/c/depot/+/5242
Reviewed-by: tazjin <tazjin@tvl.su>
Autosubmit: tazjin <tazjin@tvl.su>
Tested-by: BuildkiteCI
2022-02-07 16:58:59 +00:00

71 lines
1.7 KiB
Rust

use std::io::Write;
use std::{env, fs, io, process};
mod bytecode;
mod scanner;
mod treewalk;
/// Trait for making the different interpreters callable in the same
/// way.
pub trait Lox {
type Value: std::fmt::Debug;
type Error: std::fmt::Display;
fn create() -> Self;
fn interpret(&mut self, source: String) -> Result<Self::Value, Vec<Self::Error>>;
}
fn main() {
let mut args = env::args();
if args.len() > 2 {
println!("Usage: rlox [script]");
process::exit(1);
}
match env::var("LOX_INTERPRETER").as_ref().map(String::as_str) {
Ok("treewalk") => pick::<treewalk::interpreter::Interpreter>(args.nth(1)),
_ => pick::<bytecode::Interpreter>(args.nth(1)),
}
}
fn pick<I: Lox>(file_arg: Option<String>) {
if let Some(file) = file_arg {
run_file::<I>(&file);
} else {
run_prompt::<I>();
}
}
// Run Lox code from a file and print results to stdout
fn run_file<I: Lox>(file: &str) {
let contents = fs::read_to_string(file).expect("failed to read the input file");
let mut lox = I::create();
run(&mut lox, contents);
}
// Evaluate Lox code interactively in a shitty REPL.
fn run_prompt<I: Lox>() {
let mut line = String::new();
let mut lox = I::create();
loop {
print!("> ");
io::stdout().flush().unwrap();
io::stdin()
.read_line(&mut line)
.expect("failed to read user input");
run(&mut lox, std::mem::take(&mut line));
line.clear();
}
}
fn run<I: Lox>(lox: &mut I, code: String) {
match lox.interpret(code) {
Ok(result) => println!("=> {:?}", result),
Err(errors) => {
for error in errors {
eprintln!("{}", error);
}
}
}
}