forked from extern/nushell
180 lines
6.2 KiB
Rust
180 lines
6.2 KiB
Rust
use std::{cell::RefCell, collections::HashMap, rc::Rc};
|
|
|
|
use engine_q::{
|
|
eval_block, NuHighlighter, ParserState, ParserWorkingSet, Signature, Stack, State, SyntaxShape,
|
|
};
|
|
|
|
fn main() -> std::io::Result<()> {
|
|
let parser_state = Rc::new(RefCell::new(ParserState::new()));
|
|
let delta = {
|
|
let parser_state = parser_state.borrow();
|
|
let mut working_set = ParserWorkingSet::new(&*parser_state);
|
|
|
|
let sig =
|
|
Signature::build("where").required("cond", SyntaxShape::RowCondition, "condition");
|
|
working_set.add_decl(sig.into());
|
|
|
|
let sig = Signature::build("if")
|
|
.required("cond", SyntaxShape::RowCondition, "condition")
|
|
.required("then_block", SyntaxShape::Block, "then block")
|
|
.required(
|
|
"else",
|
|
SyntaxShape::Literal(b"else".to_vec()),
|
|
"else keyword",
|
|
)
|
|
.required("else_block", SyntaxShape::Block, "else block");
|
|
working_set.add_decl(sig.into());
|
|
|
|
let sig = Signature::build("let")
|
|
.required("var_name", SyntaxShape::Variable, "variable name")
|
|
.required("=", SyntaxShape::Literal(b"=".to_vec()), "equals sign")
|
|
.required(
|
|
"value",
|
|
SyntaxShape::Expression,
|
|
"the value to set the variable to",
|
|
);
|
|
working_set.add_decl(sig.into());
|
|
|
|
let sig = Signature::build("alias")
|
|
.required("var_name", SyntaxShape::Variable, "variable name")
|
|
.required("=", SyntaxShape::Literal(b"=".to_vec()), "equals sign")
|
|
.required(
|
|
"value",
|
|
SyntaxShape::Expression,
|
|
"the value to set the variable to",
|
|
);
|
|
working_set.add_decl(sig.into());
|
|
|
|
let sig = Signature::build("sum").required(
|
|
"arg",
|
|
SyntaxShape::List(Box::new(SyntaxShape::Number)),
|
|
"list of numbers",
|
|
);
|
|
working_set.add_decl(sig.into());
|
|
|
|
let sig = Signature::build("def")
|
|
.required("def_name", SyntaxShape::String, "definition name")
|
|
.required("params", SyntaxShape::Signature, "parameters")
|
|
.required("block", SyntaxShape::Block, "body of the definition");
|
|
working_set.add_decl(sig.into());
|
|
|
|
// let sig = Signature::build("foo").named("--jazz", SyntaxShape::Int, "jazz!!", Some('j'));
|
|
// working_set.add_decl(sig.into());
|
|
|
|
// let sig = Signature::build("bar")
|
|
// .named("--jazz", SyntaxShape::Int, "jazz!!", Some('j'))
|
|
// .switch("--rock", "rock!!", Some('r'));
|
|
// working_set.add_decl(sig.into());
|
|
|
|
let sig = Signature::build("add");
|
|
working_set.add_decl(sig.into());
|
|
let sig = Signature::build("add it");
|
|
working_set.add_decl(sig.into());
|
|
|
|
let sig = Signature::build("add it together")
|
|
.required("x", SyntaxShape::Int, "x value")
|
|
.required("y", SyntaxShape::Int, "y value");
|
|
working_set.add_decl(sig.into());
|
|
|
|
working_set.render()
|
|
};
|
|
|
|
{
|
|
ParserState::merge_delta(&mut *parser_state.borrow_mut(), delta);
|
|
}
|
|
|
|
if let Some(path) = std::env::args().nth(1) {
|
|
let parser_state = parser_state.borrow();
|
|
let mut working_set = ParserWorkingSet::new(&*parser_state);
|
|
|
|
let file = std::fs::read(&path)?;
|
|
|
|
let (block, err) = working_set.parse_file(&path, &file, false);
|
|
println!("{}", block.len());
|
|
// println!("{:#?}", output);
|
|
// println!("error: {:?}", err);
|
|
|
|
//println!("working set: {:#?}", working_set);
|
|
|
|
// println!("{}", size_of::<Statement>());
|
|
|
|
// let engine = Engine::new();
|
|
// let result = engine.eval_block(&output);
|
|
// println!("{:?}", result);
|
|
|
|
// let mut buffer = String::new();
|
|
// let stdin = std::io::stdin();
|
|
// let mut handle = stdin.lock();
|
|
|
|
// handle.read_to_string(&mut buffer)?;
|
|
|
|
Ok(())
|
|
} else {
|
|
use reedline::{DefaultPrompt, FileBackedHistory, Reedline, Signal};
|
|
|
|
let mut line_editor = Reedline::new()
|
|
.with_history(Box::new(FileBackedHistory::with_file(
|
|
1000,
|
|
"history.txt".into(),
|
|
)?))?
|
|
.with_highlighter(Box::new(NuHighlighter {
|
|
parser_state: parser_state.clone(),
|
|
}));
|
|
|
|
let prompt = DefaultPrompt::new(1);
|
|
let mut current_line = 1;
|
|
|
|
loop {
|
|
let input = line_editor.read_line(&prompt)?;
|
|
match input {
|
|
Signal::Success(s) => {
|
|
if s.trim() == "exit" {
|
|
break;
|
|
}
|
|
// println!("input: '{}'", s);
|
|
|
|
let (block, delta) = {
|
|
let parser_state = parser_state.borrow();
|
|
let mut working_set = ParserWorkingSet::new(&*parser_state);
|
|
let (output, err) = working_set.parse_file(
|
|
&format!("line_{}", current_line),
|
|
s.as_bytes(),
|
|
false,
|
|
);
|
|
println!("{:?}", output);
|
|
if let Some(err) = err {
|
|
println!("Error: {:?}", err);
|
|
}
|
|
// println!("Error: {:?}", err);
|
|
(output, working_set.render())
|
|
};
|
|
|
|
ParserState::merge_delta(&mut *parser_state.borrow_mut(), delta);
|
|
|
|
let mut stack = Stack {
|
|
vars: HashMap::new(),
|
|
};
|
|
let state = State {
|
|
parser_state: &*parser_state.borrow(),
|
|
};
|
|
|
|
let output = eval_block(&state, &mut stack, &block);
|
|
println!("{:#?}", output);
|
|
}
|
|
Signal::CtrlC => {
|
|
println!("Ctrl-c");
|
|
}
|
|
Signal::CtrlD => {
|
|
break;
|
|
}
|
|
Signal::CtrlL => {
|
|
line_editor.clear_screen()?;
|
|
}
|
|
}
|
|
current_line += 1;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|