Add !load command and some improvements😃
This commit is contained in:
parent
b6eb1ddf60
commit
6e6316b2bb
4 changed files with 125 additions and 55 deletions
|
@ -3,9 +3,6 @@ use clap::{arg, Parser, ValueEnum};
|
||||||
#[derive(Parser, Debug)]
|
#[derive(Parser, Debug)]
|
||||||
#[command(author, about, long_about = None, version)]
|
#[command(author, about, long_about = None, version)]
|
||||||
pub struct Args {
|
pub struct Args {
|
||||||
/// To be verbose
|
|
||||||
#[arg(short, long)]
|
|
||||||
pub verbose: bool,
|
|
||||||
/// The brainfuck source code file to run (if not will be entered in REPL mode)
|
/// The brainfuck source code file to run (if not will be entered in REPL mode)
|
||||||
#[arg(default_value = None)]
|
#[arg(default_value = None)]
|
||||||
pub source: Option<String>,
|
pub source: Option<String>,
|
||||||
|
|
|
@ -164,3 +164,39 @@ impl BfCommand {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mod error {
|
||||||
|
use std::fmt::{Debug, Formatter};
|
||||||
|
|
||||||
|
struct InterpreterError {
|
||||||
|
message: String,
|
||||||
|
code: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl InterpreterError {
|
||||||
|
fn new(message: String, code: i32) -> Self {
|
||||||
|
Self {
|
||||||
|
message,
|
||||||
|
code,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Display for InterpreterError {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
|
||||||
|
write!(f, "{}", self.message)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Debug for InterpreterError {
|
||||||
|
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
||||||
|
write!(f, "{}, code: {}", self.message, self.code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::error::Error for InterpreterError {
|
||||||
|
fn description(&self) -> &str {
|
||||||
|
&self.message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
13
src/main.rs
13
src/main.rs
|
@ -4,8 +4,10 @@ mod utils;
|
||||||
mod repl;
|
mod repl;
|
||||||
|
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
|
|
||||||
extern crate pretty_env_logger;
|
extern crate pretty_env_logger;
|
||||||
#[macro_use] extern crate log;
|
#[macro_use]
|
||||||
|
extern crate log;
|
||||||
|
|
||||||
use arguments::Args;
|
use arguments::Args;
|
||||||
|
|
||||||
|
@ -16,15 +18,12 @@ fn main() {
|
||||||
let args = Args::parse();
|
let args = Args::parse();
|
||||||
info!("Parsed command line arguments: {:?}", args);
|
info!("Parsed command line arguments: {:?}", args);
|
||||||
|
|
||||||
if args.verbose {
|
|
||||||
info!("Verbose mode enabled");
|
|
||||||
}
|
|
||||||
|
|
||||||
info!("Initializing interpreter");
|
info!("Initializing interpreter");
|
||||||
let mut interpreter = interpreter::Interpreter::new(
|
let mut interpreter = interpreter::Interpreter::new(
|
||||||
args.array_size,
|
args.array_size,
|
||||||
utils::read_brainfuck_code_if_any(&args.source),
|
utils::read_brainfuck_code_if_any(&args.source),
|
||||||
args.features.unwrap_or_else(|| vec![]));
|
args.features.unwrap_or_else(|| vec![])
|
||||||
|
);
|
||||||
|
|
||||||
match args.source {
|
match args.source {
|
||||||
Some(source) => {
|
Some(source) => {
|
||||||
|
@ -40,7 +39,7 @@ fn main() {
|
||||||
std::process::exit(code);
|
std::process::exit(code);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
None => {
|
None => {
|
||||||
repl::start(interpreter)
|
repl::start(interpreter)
|
||||||
}
|
}
|
||||||
|
|
56
src/repl.rs
56
src/repl.rs
|
@ -21,7 +21,7 @@ impl Repl {
|
||||||
let mut input = String::new();
|
let mut input = String::new();
|
||||||
|
|
||||||
match std::io::stdin().read_line(&mut input) {
|
match std::io::stdin().read_line(&mut input) {
|
||||||
Ok(_) => {},
|
Ok(_) => {}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("Failed to read input: {}", e);
|
error!("Failed to read input: {}", e);
|
||||||
std::process::exit(1);
|
std::process::exit(1);
|
||||||
|
@ -47,7 +47,10 @@ impl Repl {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run_repl_cmd(&mut self, input: String) {
|
fn run_repl_cmd(&mut self, input: String) {
|
||||||
match input.trim().get(1..).unwrap() {
|
let mut cmd = input.split_whitespace();
|
||||||
|
match cmd.next() {
|
||||||
|
Some(repl_cmd) => {
|
||||||
|
match repl_cmd {
|
||||||
"fuck" => {
|
"fuck" => {
|
||||||
println!("Bye bye :D");
|
println!("Bye bye :D");
|
||||||
std::process::exit(0);
|
std::process::exit(0);
|
||||||
|
@ -68,21 +71,52 @@ impl Repl {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"save" | "s" => {
|
"save" | "s" => {
|
||||||
/// TODO: Use custom name for file
|
let file_name = cmd.next()
|
||||||
println!("Saving history to file: history.bfr");
|
.unwrap_or("brainfuck_repl_history.bfr");
|
||||||
match std::fs::write("history.bfr", self.history.join("\n")) {
|
|
||||||
|
println!("Saving history to file: {file_name}");
|
||||||
|
match std::fs::write(file_name, self.history.join("\n")) {
|
||||||
Ok(_) => {
|
Ok(_) => {
|
||||||
info!("Successfully saved history to file: history.bfr");
|
info!("Successfully saved history to file: {file_name}");
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("Failed to save history to file: {}", e);
|
error!("Failed to save history to file: {}", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
|
"load" | "l" => {
|
||||||
|
let file_name = cmd.next()
|
||||||
|
.unwrap_or("brainfuck_repl_history.bfr");
|
||||||
|
|
||||||
|
println!("Loading history from file: {file_name}");
|
||||||
|
match std::fs::read_to_string(file_name) {
|
||||||
|
Ok(history) => {
|
||||||
|
info!("Successfully loaded history from file: {file_name}");
|
||||||
|
self.history = history.split("\n")
|
||||||
|
.map(|s| s.to_string())
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
// Run all commands in history
|
||||||
|
for cmd in self.history.iter() {
|
||||||
|
match self.interpreter.run(Some(cmd.clone())) {
|
||||||
|
Ok(_) => {
|
||||||
|
info!("Successfully ran brainfuck source code from REPL");
|
||||||
|
}
|
||||||
|
Err((e, _)) => {
|
||||||
|
error!("Failed to run brainfuck source code from REPL: {}", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
error!("Failed to load history from file: {}", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
"reset" | "r" => {
|
"reset" | "r" => {
|
||||||
println!("Resetting REPL");
|
println!("Resetting REPL");
|
||||||
self.interpreter.reset();
|
self.interpreter.reset();
|
||||||
},
|
}
|
||||||
"help" => {
|
"help" => {
|
||||||
println!("!array, !a: print the current array");
|
println!("!array, !a: print the current array");
|
||||||
println!("!array_size, !as: print the current array size");
|
println!("!array_size, !as: print the current array size");
|
||||||
|
@ -94,7 +128,11 @@ impl Repl {
|
||||||
println!("!help: show this fu*king help message");
|
println!("!help: show this fu*king help message");
|
||||||
println!("!fuck: exit the REPL mode");
|
println!("!fuck: exit the REPL mode");
|
||||||
}
|
}
|
||||||
_ => println!("Unknown command: {}, type !help to show the help", input)
|
_ => println!("Unknown command: {}, type !help to show the help",
|
||||||
|
input)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
None => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue