From aa4921102b16510859e181ff1ac299ca4f080a64 Mon Sep 17 00:00:00 2001 From: Michael Aaron Murphy <mmstickman@gmail.com> Date: Sat, 8 Apr 2017 17:01:18 -0400 Subject: [PATCH] Implement the eval Builtin --- src/builtins/mod.rs | 22 +++++++++++++++++++++- src/parser/loops/for_grammar.rs | 4 ++-- src/parser/quotes.rs | 2 +- src/parser/shell_expand/mod.rs | 6 +----- src/shell/flow.rs | 2 +- src/shell/mod.rs | 4 ++-- 6 files changed, 28 insertions(+), 12 deletions(-) diff --git a/src/builtins/mod.rs b/src/builtins/mod.rs index 412c6303..e6459b14 100644 --- a/src/builtins/mod.rs +++ b/src/builtins/mod.rs @@ -10,7 +10,8 @@ use std::collections::HashMap; use std::io::{self, Write}; use std::process; -use shell::{Shell, ShellHistory}; +use parser::QuoteTerminator; +use shell::{Shell, FlowLogic, ShellHistory}; use status::*; /// Structure which represents a Terminal's command. @@ -155,6 +156,25 @@ impl Builtin { }); /* Misc */ + commands.insert("eval", + Builtin { + name: "eval", + help: "evaluates the evaluated expression", + main: box |args: &[String], shell: &mut Shell| -> i32 { + let evaluated_command = args[1..].join(" "); + let mut buffer = QuoteTerminator::new(evaluated_command); + if buffer.check_termination() { + shell.on_command(&buffer.consume()); + shell.previous_status + } else { + let stderr = io::stderr(); + let mut stderr = stderr.lock(); + let _ = writeln!(stderr, "ion: supplied eval expression was not terminted"); + FAILURE + } + }, + }); + commands.insert("exit", Builtin { name: "exit", diff --git a/src/parser/loops/for_grammar.rs b/src/parser/loops/for_grammar.rs index dc863dbf..4b961f6c 100644 --- a/src/parser/loops/for_grammar.rs +++ b/src/parser/loops/for_grammar.rs @@ -11,12 +11,12 @@ pub enum ForExpression { impl ForExpression { pub fn new(expression: &[String], dir_stack: &DirectoryStack, variables: &Variables) -> ForExpression { - let mut output: Vec<String> = expression.iter() + let output: Vec<String> = expression.iter() .flat_map(|expression| expand_string(expression, variables, dir_stack, true)) .collect(); if output.len() == 1 { - let output = output.drain(..).next().unwrap(); + let output = output.into_iter().next().unwrap(); { let mut bytes_iterator = output.bytes().enumerate(); while let Some((id, byte)) = bytes_iterator.next() { diff --git a/src/parser/quotes.rs b/src/parser/quotes.rs index 5c3f6257..8e19fa12 100644 --- a/src/parser/quotes.rs +++ b/src/parser/quotes.rs @@ -18,7 +18,7 @@ impl QuoteTerminator { self.buffer.push_str(if self.flags & TRIM != 0 { input.trim() } else { &input }); } - pub fn is_terminated(&mut self) -> bool { + pub fn check_termination(&mut self) -> bool { for character in self.buffer.bytes().skip(self.read) { self.read += 1; match character { diff --git a/src/parser/shell_expand/mod.rs b/src/parser/shell_expand/mod.rs index f549ccfe..cbebe709 100644 --- a/src/parser/shell_expand/mod.rs +++ b/src/parser/shell_expand/mod.rs @@ -67,21 +67,18 @@ fn expand_brace(current: &mut String, expanders: &mut Vec<Vec<String>>, } } -#[inline(always)] fn array_expand(elements: &[&str], expand_func: &ExpanderFunctions) -> Vec<String> { elements.iter() .flat_map(|element| expand_string(element, expand_func, false)) .collect() } -#[inline(always)] fn array_nth(elements: &[&str], expand_func: &ExpanderFunctions, id: usize) -> String { elements.iter() .flat_map(|element| expand_string(element, expand_func, false)) .nth(id).unwrap_or_default() } -#[inline(always)] fn array_range(elements: &[&str], expand_func: &ExpanderFunctions, start: usize, end: IndexPosition) -> Vec<String> { match end { IndexPosition::CatchAll => elements.iter() @@ -93,11 +90,10 @@ fn array_range(elements: &[&str], expand_func: &ExpanderFunctions, start: usize, } } -#[inline(always)] fn slice_string(output: &mut String, expanded: &str, index: Index) { match index { Index::None => (), - Index::All => output.push_str(&expanded), + Index::All => output.push_str(expanded), Index::ID(id) => { if let Some(character) = UnicodeSegmentation::graphemes(expanded, true).nth(id) { output.push_str(character); diff --git a/src/shell/flow.rs b/src/shell/flow.rs index 08176786..23bb4b5e 100644 --- a/src/shell/flow.rs +++ b/src/shell/flow.rs @@ -259,7 +259,7 @@ impl<'a> FlowLogic for Shell<'a> { match self.run_pipeline(&mut expression, false) { Some(SUCCESS) => self.execute_statements(success), _ => { - for mut elseif in else_if.into_iter() { + for mut elseif in else_if { if self.run_pipeline(&mut elseif.expression, false) == Some(SUCCESS) { return self.execute_statements(elseif.success); } diff --git a/src/shell/mod.rs b/src/shell/mod.rs index 1d81cace..e3ab8737 100644 --- a/src/shell/mod.rs +++ b/src/shell/mod.rs @@ -145,7 +145,7 @@ impl<'a> Shell<'a> { pub fn terminate_quotes(&mut self, command: String) -> String { let mut buffer = QuoteTerminator::new(command); self.flow_control.level += 1; - while !buffer.is_terminated() { + while !buffer.check_termination() { loop { if let Some(command) = self.readln() { buffer.append(command); @@ -175,7 +175,7 @@ impl<'a> Shell<'a> { let mut lines = command_list.lines().map(|x| x.to_owned()); while let Some(command) = lines.next() { let mut buffer = QuoteTerminator::new(command); - while !buffer.is_terminated() { + while !buffer.check_termination() { loop { if let Some(command) = lines.next() { buffer.append(command); -- GitLab