Files
ValueScript/vstc/src/run_command.rs
2023-03-21 09:51:25 +11:00

139 lines
3.4 KiB
Rust

use std::rc::Rc;
use std::{ffi::OsStr, path::Path, process::exit};
use valuescript_compiler::{assemble, compile, parse_module};
use valuescript_vm::VirtualMachine;
use crate::resolve_entry_path::resolve_entry_path;
use super::handle_diagnostics_cli::handle_diagnostics_cli;
pub fn run_command(args: &Vec<String>) {
if args.len() < 3 {
println!("ERROR: Unrecognized command\n");
show_help();
exit(1);
}
let mut argpos = 2;
if args[argpos] == "-h" || args[argpos] == "--help" {
show_help();
return;
}
let format = match args[argpos].chars().next() {
Some('-') => {
let res = format_from_option(&args[argpos]);
argpos += 1;
res
}
_ => format_from_path(&args[argpos]),
};
let file_path = &args[argpos];
argpos += 1;
let bytecode = to_bytecode(format, file_path);
let mut vm = VirtualMachine::new();
match vm.run(&bytecode, &args[argpos..]) {
Ok(result) => {
println!("{}", result);
}
Err(err) => {
println!("Uncaught exception: {}", err);
exit(1);
}
}
}
enum RunFormat {
TypeScript,
Assembly,
Bytecode,
}
fn format_from_option(option: &String) -> RunFormat {
return match option.as_str() {
"--typescript" => RunFormat::TypeScript,
"--assembly" => RunFormat::Assembly,
"--bytecode" => RunFormat::Bytecode,
_ => std::panic!("Unrecognized option {}", option),
};
}
fn format_from_path(file_path: &String) -> RunFormat {
let ext = Path::new(&file_path)
.extension()
.and_then(OsStr::to_str)
.unwrap_or("");
return match ext {
"ts" => RunFormat::TypeScript,
"mts" => RunFormat::TypeScript,
"js" => RunFormat::TypeScript,
"mjs" => RunFormat::TypeScript,
"vsm" => RunFormat::Assembly,
"vsb" => RunFormat::Bytecode,
_ => std::panic!("Unrecognized file extension \"{}\"", ext),
};
}
fn to_bytecode(format: RunFormat, file_path: &String) -> Rc<Vec<u8>> {
match format {
RunFormat::TypeScript => {
let resolved_entry_path = resolve_entry_path(file_path);
let compile_result = compile(resolved_entry_path, |path| {
std::fs::read_to_string(path).map_err(|err| err.to_string())
});
for (path, diagnostics) in compile_result.diagnostics.iter() {
handle_diagnostics_cli(&path.path, diagnostics);
}
assemble(
&compile_result
.module
.expect("Should have exited if module is None"),
)
}
RunFormat::Assembly => {
let file_content = std::fs::read_to_string(file_path)
.unwrap_or_else(|_| panic!("Failed to read file {}", file_path));
let module = parse_module(&file_content);
assemble(&module)
}
RunFormat::Bytecode => Rc::new(
std::fs::read(file_path).unwrap_or_else(|_| panic!("Failed to read file {}", file_path)),
),
}
}
fn show_help() {
println!("vstc run");
println!();
println!("Run a ValueScript program");
println!();
println!("USAGE:");
println!(" vstc run [OPTIONS] <file>");
println!();
println!("OPTIONS:");
println!(" --assembly");
println!(" Interpret <file> as assembly");
println!();
println!(" --bytecode");
println!(" Interpret <file> as bytecode");
println!();
println!(" --typescript");
println!(" Interpret <file> as typescript");
println!();
println!("NOTE:");
println!(" <file> will be interpreted based on file extension if not otherwise specified");
}