2019-05-28 06:45:18 +00:00
|
|
|
crate mod ast;
|
2019-05-15 16:12:38 +00:00
|
|
|
crate mod completer;
|
2019-05-29 15:26:45 +00:00
|
|
|
crate mod lexer;
|
2019-05-26 06:54:41 +00:00
|
|
|
crate mod parser;
|
|
|
|
crate mod registry;
|
2019-05-30 04:19:46 +00:00
|
|
|
crate mod span;
|
2019-05-11 04:45:57 +00:00
|
|
|
|
2019-05-28 06:45:18 +00:00
|
|
|
crate use ast::{ParsedCommand, Pipeline};
|
2019-06-02 16:28:40 +00:00
|
|
|
crate use registry::{Args, CommandConfig};
|
2019-05-26 06:54:41 +00:00
|
|
|
|
|
|
|
use crate::errors::ShellError;
|
2019-05-30 04:19:46 +00:00
|
|
|
use lexer::Lexer;
|
2019-06-04 21:42:31 +00:00
|
|
|
use log::trace;
|
2019-05-26 06:54:41 +00:00
|
|
|
use parser::PipelineParser;
|
|
|
|
|
2019-06-02 16:28:40 +00:00
|
|
|
pub fn parse(input: &str) -> Result<Pipeline, ShellError> {
|
2019-06-04 21:42:31 +00:00
|
|
|
let _ = pretty_env_logger::try_init();
|
|
|
|
|
2019-05-26 06:54:41 +00:00
|
|
|
let parser = PipelineParser::new();
|
2019-05-31 00:53:54 +00:00
|
|
|
let tokens = Lexer::new(input, false);
|
2019-05-26 06:54:41 +00:00
|
|
|
|
2019-06-04 21:42:31 +00:00
|
|
|
trace!(
|
|
|
|
"Tokens: {:?}",
|
|
|
|
tokens.clone().collect::<Result<Vec<_>, _>>()
|
|
|
|
);
|
|
|
|
|
2019-05-30 04:19:46 +00:00
|
|
|
match parser.parse(tokens) {
|
|
|
|
Ok(val) => Ok(val),
|
|
|
|
Err(err) => Err(ShellError::parse_error(err, input.to_string())),
|
|
|
|
}
|
2019-05-26 06:54:41 +00:00
|
|
|
}
|
2019-06-02 16:28:40 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
2019-06-04 21:42:31 +00:00
|
|
|
use crate::parser::ast::{bare, flag, short, unit, var, Expression, Operator, Pipeline};
|
2019-06-02 16:28:40 +00:00
|
|
|
use pretty_assertions::assert_eq;
|
|
|
|
|
|
|
|
fn assert_parse(source: &str, expected: Pipeline) {
|
2019-06-04 21:42:31 +00:00
|
|
|
let parsed = match parse(source) {
|
|
|
|
Ok(p) => p,
|
|
|
|
Err(ShellError::Diagnostic(diag, source)) => {
|
|
|
|
use language_reporting::termcolor;
|
|
|
|
|
|
|
|
let writer = termcolor::StandardStream::stdout(termcolor::ColorChoice::Auto);
|
|
|
|
let files = crate::parser::span::Files::new(source);
|
|
|
|
|
|
|
|
language_reporting::emit(
|
|
|
|
&mut writer.lock(),
|
|
|
|
&files,
|
|
|
|
&diag.diagnostic,
|
|
|
|
&language_reporting::DefaultConfig,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
panic!("Test failed")
|
|
|
|
}
|
|
|
|
Err(err) => panic!("Something went wrong during parse: {:#?}", err),
|
|
|
|
};
|
|
|
|
|
2019-06-02 16:28:40 +00:00
|
|
|
let printed = parsed.print();
|
2019-06-04 21:42:31 +00:00
|
|
|
|
2019-06-02 16:28:40 +00:00
|
|
|
assert_eq!(parsed, expected);
|
|
|
|
assert_eq!(source, printed);
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! commands {
|
|
|
|
( $( ( $name:tt $( $command:expr)* ) )|* ) => {
|
|
|
|
Pipeline::new(vec![
|
|
|
|
$(
|
|
|
|
command!($name $($command)*)
|
|
|
|
),*
|
|
|
|
])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! command {
|
|
|
|
($name:ident $( $command:expr )*) => {
|
2019-06-04 21:42:31 +00:00
|
|
|
Expression::call(Expression::bare(stringify!($name)), vec![ $($command.into()),* ])
|
2019-06-02 16:28:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
($name:ident $( $command:expr )*) => {
|
2019-06-04 21:42:31 +00:00
|
|
|
Expression::call(Expression::bare(stringify!($name)), vec![ $($command.into()),* ])
|
2019-06-02 16:28:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
($name:tt $( $command:expr )*) => {
|
2019-06-04 21:42:31 +00:00
|
|
|
Expression::call(Expression::bare($name), vec![ $($command.into()),* ])
|
2019-06-02 16:28:40 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_simple_command() {
|
|
|
|
assert_parse("ls", commands![(ls)]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_command_with_args() {
|
|
|
|
assert_parse(
|
|
|
|
r#"open Cargo.toml | select package.authors | split-row " ""#,
|
|
|
|
commands![
|
|
|
|
(open bare("Cargo.toml"))
|
|
|
|
| (select bare("package.authors"))
|
|
|
|
| ("split-row" " ")
|
|
|
|
],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(r#"git add ."#, commands![("git" bare("add") bare("."))]);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
"open Cargo.toml | select package.version | echo $it",
|
|
|
|
commands![
|
|
|
|
(open bare("Cargo.toml"))
|
|
|
|
| (select bare("package.version"))
|
|
|
|
| (echo var("it"))
|
|
|
|
],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
"open Cargo.toml --raw",
|
|
|
|
commands![(open bare("Cargo.toml") flag("raw"))],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
"open Cargo.toml -r",
|
|
|
|
commands![(open bare("Cargo.toml") short("r"))],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
"open Cargo.toml | from-toml | to-toml",
|
|
|
|
commands![(open bare("Cargo.toml")) | ("from-toml") | ("to-toml")],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
r#"config --get "ignore dups" | format-list"#,
|
|
|
|
commands![(config flag("get") "ignore dups") | ("format-list")],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
"open Cargo.toml | from-toml | select dependencies | column serde",
|
|
|
|
commands![
|
|
|
|
(open bare("Cargo.toml"))
|
|
|
|
| ("from-toml")
|
|
|
|
| (select bare("dependencies"))
|
|
|
|
| (column bare("serde"))
|
|
|
|
],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
"config --set tabs 2",
|
|
|
|
commands![(config flag("set") bare("tabs") 2)],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
r#"ls | skip 1 | first 2 | select "file name" | rm $it"#,
|
|
|
|
commands![
|
|
|
|
(ls)
|
|
|
|
| (skip 1)
|
|
|
|
| (first 2)
|
|
|
|
| (select "file name")
|
|
|
|
| (rm var("it"))
|
|
|
|
],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
r#"git branch --merged | split-row "`n" | where $it != "* master""#,
|
|
|
|
commands![
|
|
|
|
// TODO: Handle escapes correctly. Should we do ` escape because of paths?
|
|
|
|
(git bare("branch") flag("merged")) | ("split-row" "`n") | (where binary(var("it"), "!=", "* master"))
|
|
|
|
],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
r#"open input2.json | from-json | select glossary.GlossDiv.GlossList.GlossEntry.GlossDef.GlossSeeAlso | where $it > "GML""#,
|
|
|
|
commands![
|
|
|
|
(open bare("input2.json"))
|
|
|
|
| ("from-json")
|
|
|
|
| (select bare("glossary.GlossDiv.GlossList.GlossEntry.GlossDef.GlossSeeAlso"))
|
|
|
|
| (where binary(var("it"), ">", "GML"))
|
|
|
|
]
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
r"cd ..\.cargo\",
|
|
|
|
commands![
|
|
|
|
(cd bare(r"..\.cargo\"))
|
|
|
|
],
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_parse(
|
|
|
|
"ls | where size < 1KB",
|
|
|
|
commands![
|
|
|
|
(ls) | (where binary(bare("size"), "<", unit(1, "KB")))
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
2019-06-04 21:42:31 +00:00
|
|
|
|
|
|
|
fn binary(
|
|
|
|
left: impl Into<Expression>,
|
|
|
|
op: impl Into<Operator>,
|
|
|
|
right: impl Into<Expression>,
|
|
|
|
) -> Expression {
|
|
|
|
Expression::binary(left, op, right)
|
|
|
|
}
|
2019-06-02 16:28:40 +00:00
|
|
|
}
|