#![feature(crate_visibility_modifier)] #![feature(in_band_lifetimes)] #[allow(unused)] use crate::prelude::*; use std::borrow::Cow::{self, Borrowed, Owned}; mod commands; mod context; mod env; mod errors; mod format; mod object; mod parser; mod prelude; use crate::commands::command::ReturnValue; crate use crate::commands::command::{Command, CommandAction, CommandBlueprint}; use crate::context::Context; crate use crate::env::{Environment, Host}; crate use crate::errors::ShellError; crate use crate::format::{EntriesListView, GenericView}; use crate::object::Value; use rustyline::completion::{Completer, FilenameCompleter, Pair}; use rustyline::highlight::{Highlighter, MatchingBracketHighlighter}; use rustyline::hint::{Hinter, HistoryHinter}; use ansi_term::Color; use rustyline::error::ReadlineError; use rustyline::{ColorMode, Config, Editor, Helper, self}; use std::collections::VecDeque; use std::error::Error; use std::sync::{Arc, Mutex}; use subprocess::Exec; #[derive(Debug)] pub enum MaybeOwned<'a, T> { Owned(T), Borrowed(&'a T), } impl MaybeOwned<'a, T> { crate fn borrow(&self) -> &T { match self { MaybeOwned::Owned(v) => v, MaybeOwned::Borrowed(v) => v, } } } struct MyHelper(FilenameCompleter, MatchingBracketHighlighter, HistoryHinter); impl Completer for MyHelper { type Candidate = Pair; fn complete( &self, line: &str, pos: usize, ctx: &rustyline::Context<'_>, ) -> Result<(usize, Vec), ReadlineError> { self.0.complete(line, pos, ctx) } } impl Hinter for MyHelper { fn hint(&self, line: &str, pos: usize, ctx: &rustyline::Context<'_>) -> Option { self.2.hint(line, pos, ctx) } } impl Highlighter for MyHelper { fn highlight_prompt<'p>(&self, prompt: &'p str) -> Cow<'p, str> { Owned("\x1b[32m".to_owned() + &prompt[0..prompt.len() - 2] + "\x1b[m> ") } fn highlight_hint<'h>(&self, hint: &'h str) -> Cow<'h, str> { Owned("\x1b[1m".to_owned() + hint + "\x1b[m") } fn highlight<'l>(&self, line: &'l str, pos: usize) -> Cow<'l, str> { self.1.highlight(line, pos) } fn highlight_char(&self, line: &str, pos: usize) -> bool { self.1.highlight_char(line, pos) } } impl Helper for MyHelper {} fn main() -> Result<(), Box> { let config = Config::builder().color_mode(ColorMode::Forced).build(); let h = MyHelper( FilenameCompleter::new(), MatchingBracketHighlighter::new(), HistoryHinter {}, ); let mut rl: Editor = Editor::with_config(config); rl.set_helper(Some(h)); if rl.load_history("history.txt").is_err() { println!("No previous history."); } let context = Arc::new(Mutex::new(Context::basic()?)); { use crate::commands::*; context.lock().unwrap().add_commands(vec![ ("ps", Box::new(ps::Ps)), ("ls", Box::new(ls::Ls)), ("cd", Box::new(cd::Cd)), ("skip", Box::new(skip::Skip)), ("take", Box::new(take::Take)), ("select", Box::new(select::Select)), ("reject", Box::new(reject::Reject)), ("to-array", Box::new(to_array::ToArray)), ("where", Box::new(where_::Where)), ]); } loop { let readline = rl.readline(&format!( "{}> ", context.lock().unwrap().env.cwd().display().to_string() )); match readline { Ok(ref line) if line.trim() == "exit" => { break; } Ok(line) => { let result = crate::parser::shell_parser(&line) .map_err(|e| ShellError::string(format!("{:?}", e)))?; let parsed = result.1; rl.add_history_entry(line.as_ref()); let mut input = VecDeque::new(); for item in parsed { input = process_command( crate::parser::print_items(&item), item.clone(), input, context.clone(), )?; } if input.len() > 0 { if equal_shapes(&input) { format(crate::commands::to_array(input), context.clone()); } else { format(input, context.clone()); } } } Err(ReadlineError::Interrupted) => { println!("CTRL-C"); break; } Err(ReadlineError::Eof) => { println!("CTRL-D"); break; } Err(err) => { println!("Error: {:?}", err); break; } } } rl.save_history("history.txt").unwrap(); Ok(()) } fn process_command( line: String, parsed: Vec, input: VecDeque, context: Arc>, ) -> Result, ShellError> { let command = &parsed[0].name(); let arg_list = parsed[1..].iter().map(|i| i.as_value()).collect(); if command == &"format" { format(input, context); Ok(VecDeque::new()) } else if command == &"format-list" { let view = EntriesListView::from_stream(input); crate::format::print_view(&view, context.clone()); Ok(VecDeque::new()) } else { let mut ctx = context.lock().unwrap(); match ctx.has_command(*command) { true => { // let mut instance = ctx.create_command(command, arg_list)?; let result = ctx.run_command(command, arg_list, input)?; // let result = command.run(input_args)?; let mut next = VecDeque::new(); for v in result { match v { ReturnValue::Action(action) => match action { crate::CommandAction::ChangeCwd(cwd) => ctx.env.cwd = cwd, }, ReturnValue::Value(v) => next.push_back(v), } } Ok(next) } false => { Exec::shell(line).cwd(ctx.env.cwd()).join().unwrap(); Ok(VecDeque::new()) } } } } fn format(input: VecDeque, context: Arc>) { let last = input.len() - 1; for (i, item) in input.iter().enumerate() { let view = GenericView::new(item); crate::format::print_view(&view, context.clone()); if last != i { println!(""); } } } fn equal_shapes(input: &VecDeque) -> bool { let mut items = input.iter(); let item = match items.next() { Some(item) => item, None => return false, }; let desc = item.data_descriptors(); for item in items { if desc != item.data_descriptors() { return false; } } true }