From 1eceb5e36002cfca610f18d78e2b99336c04fbe5 Mon Sep 17 00:00:00 2001 From: Michael Aaron Murphy <mmstickman@gmail.com> Date: Sat, 26 Aug 2017 14:53:15 -0400 Subject: [PATCH] Complete Assignment Type Checking --- Cargo.lock | 22 +++--- Cargo.toml | 2 +- src/parser/types/parse.rs | 2 + src/shell/assignments.rs | 161 +++++++++++++++++++++++++++++++------- 4 files changed, 148 insertions(+), 39 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index af47e2d1..74a3b43c 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -5,19 +5,19 @@ dependencies = [ "ansi_term 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)", "app_dirs 1.1.1 (registry+https://github.com/rust-lang/crates.io-index)", "bitflags 0.9.1 (registry+https://github.com/rust-lang/crates.io-index)", - "calculate 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "calculate 0.3.0 (git+https://github.com/redox-os/calc.git)", "fnv 1.0.5 (registry+https://github.com/rust-lang/crates.io-index)", "glob 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)", "lazy_static 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", "libc 0.2.29 (registry+https://github.com/rust-lang/crates.io-index)", "libloading 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", - "liner 0.4.0 (git+https://github.com/MovingtoMars/liner/)", + "liner 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", "nix 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)", "permutate 0.3.2 (registry+https://github.com/rust-lang/crates.io-index)", "redox_syscall 0.1.30 (registry+https://github.com/rust-lang/crates.io-index)", "regex 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", "smallstring 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "smallvec 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", + "smallvec 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", "unicode-segmentation 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)", "users 0.5.3 (registry+https://github.com/rust-lang/crates.io-index)", "version_check 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)", @@ -64,8 +64,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" [[package]] name = "calculate" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" +version = "0.3.0" +source = "git+https://github.com/redox-os/calc.git#ca15759a3722960ff8ba50e709804348c2807ba1" [[package]] name = "cfg-if" @@ -113,8 +113,8 @@ dependencies = [ [[package]] name = "liner" -version = "0.4.0" -source = "git+https://github.com/MovingtoMars/liner/#f5948dadb4c2abaa39bf9d6c887b2bbc899c06bf" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" dependencies = [ "bytecount 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)", "termion 1.5.1 (registry+https://github.com/rust-lang/crates.io-index)", @@ -208,7 +208,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" [[package]] name = "smallvec" -version = "0.4.2" +version = "0.4.3" source = "registry+https://github.com/rust-lang/crates.io-index" [[package]] @@ -293,7 +293,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" "checksum bitflags 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)" = "aad18937a628ec6abcd26d1489012cc0e18c21798210f491af69ded9b881106d" "checksum bitflags 0.9.1 (registry+https://github.com/rust-lang/crates.io-index)" = "4efd02e230a02e18f92fc2735f44597385ed02ad8f831e7c1c1156ee5e1ab3a5" "checksum bytecount 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)" = "4bbeb7c30341fce29f6078b4bdf876ea4779600866e98f5b2d203a534f195050" -"checksum calculate 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "1b593017444d7b7d0d1cacfea9dd28a507e42cfd30c5366130d14e1e16eb33e0" +"checksum calculate 0.3.0 (git+https://github.com/redox-os/calc.git)" = "<none>" "checksum cfg-if 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "d4c819a1287eb618df47cc647173c5c4c66ba19d888a6e50d605672aed3140de" "checksum fnv 1.0.5 (registry+https://github.com/rust-lang/crates.io-index)" = "6cc484842f1e2884faf56f529f960cc12ad8c71ce96cc7abba0a067c98fee344" "checksum glob 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)" = "8be18de09a56b60ed0edf84bc9df007e30040691af7acd1c41874faac5895bfb" @@ -301,7 +301,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" "checksum lazy_static 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)" = "3b37545ab726dd833ec6420aaba8231c5b320814b9029ad585555d2a03e94fbf" "checksum libc 0.2.29 (registry+https://github.com/rust-lang/crates.io-index)" = "8a014d9226c2cc402676fbe9ea2e15dd5222cd1dd57f576b5b283178c944a264" "checksum libloading 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)" = "be99f814beb3e9503a786a592c909692bb6d4fc5a695f6ed7987223acfbd5194" -"checksum liner 0.4.0 (git+https://github.com/MovingtoMars/liner/)" = "<none>" +"checksum liner 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)" = "be5a42003ac3b83b02a169e11741f2946327e52bf1118591b61750610769c3f1" "checksum memchr 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)" = "1dbccc0e46f1ea47b9f17e6d67c5a96bd27030519c519c9c91327e31275a47b4" "checksum nix 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)" = "47e49f6982987135c5e9620ab317623e723bd06738fd85377e8d55f57c8b6487" "checksum ole32-sys 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "5d2c49021782e5233cd243168edfa8037574afed4eba4bbaf538b3d8d1789d8c" @@ -313,7 +313,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" "checksum shell32-sys 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "72f20b8f3c060374edb8046591ba28f62448c369ccbdc7b02075103fb3a9e38d" "checksum smallstring 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "30950abdb5b38f56a0e181ae56ed64a539b64fa77ea6325147203dc7faeb087f" "checksum smallvec 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "4f8266519bc1d17d0b5b16f6c21295625d562841c708f6376f49028a43e9c11e" -"checksum smallvec 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)" = "b5864faef64ccadecaafebd8d57ae1b27ce8190c8c491d4a284400bb6fa639ae" +"checksum smallvec 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)" = "8fcd03faf178110ab0334d74ca9631d77f94c8c11cc77fcb59538abf0025695d" "checksum termion 1.5.1 (registry+https://github.com/rust-lang/crates.io-index)" = "689a3bdfaab439fd92bc87df5c4c78417d3cbe537487274e9b0b2dce76e92096" "checksum thread_local 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)" = "1697c4b57aeeb7a536b647165a2825faddffb1d3bad386d507709bd51a90bb14" "checksum unicode-segmentation 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "a8083c594e02b8ae1654ae26f0ade5158b119bd88ad0e8227a5d8fcd72407946" diff --git a/Cargo.toml b/Cargo.toml index 6aa6b694..5aad4008 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -35,7 +35,7 @@ glob = "0.2" # Provides a macro for lazily-evalulated statics lazy_static = "0.2" # Provides the line editor / prompt for the shell -liner = { git = "https://github.com/MovingtoMars/liner/" } +liner = "0.4" # Provides permutations of strings in brace expansions permutate = "0.3" # Enables strings to be stored inline on the stack, when possible. diff --git a/src/parser/types/parse.rs b/src/parser/types/parse.rs index 61ee31ab..b3fe44b3 100644 --- a/src/parser/types/parse.rs +++ b/src/parser/types/parse.rs @@ -9,12 +9,14 @@ pub struct TypeArg<'a> { #[derive(Debug, PartialEq)] pub enum TypeError<'a> { Invalid(&'a str), + BadValue(TypePrimitive), } impl<'a> Display for TypeError<'a> { fn fmt(&self, f: &mut Formatter) -> fmt::Result { match *self { TypeError::Invalid(parm) => write!(f, "invalid type supplied: {}", parm), + TypeError::BadValue(expected) => write!(f, "expected {}", expected), } } } diff --git a/src/shell/assignments.rs b/src/shell/assignments.rs index 355fc8f6..611c59b6 100644 --- a/src/shell/assignments.rs +++ b/src/shell/assignments.rs @@ -6,9 +6,8 @@ use super::status::*; use parser::expand_string; use parser::types::assignments::*; use parser::types::parse::*; -// use parser::assignments::{Binding, Operator, Value}; - -use types::{ArrayVariableContext, VariableContext}; +use smallvec::SmallVec; +use types::{Array, ArrayVariableContext, VariableContext}; fn print_vars(list: &VariableContext) { let stdout = io::stdout(); @@ -66,9 +65,22 @@ impl<'a> VariableStore for Shell<'a> { for action in assignment_actions { match action { Ok(Action::UpdateArray(key, Operator::Equal, expression)) => { - // TODO: Handle different array types accordingly. - let value = expand_string(expression, self, false); - self.variables.set_array(key.name, value); + let values = expand_string(expression, self, false); + let use_original = match array_is_valid(&values, key.kind) { + Ok(Some(normalized)) => { + self.variables.set_array(key.name, normalized); + false + } + Ok(None) => true, + Err(why) => { + eprintln!("ion: assignment error: {}", why); + return FAILURE; + } + }; + + if use_original { + self.variables.set_array(key.name, values); + } } Ok(Action::UpdateArray(..)) => { eprintln!("ion: arithmetic operators on array expressions aren't supported yet."); @@ -76,6 +88,14 @@ impl<'a> VariableStore for Shell<'a> { } Ok(Action::UpdateString(key, operator, expression)) => { let value = expand_string(expression, self, false).join(" "); + let value = match string_is_valid(&value, key.kind) { + Ok(value) => value, + Err(why) => { + eprintln!("ion: assignment error: {}", why); + return FAILURE; + } + }; + if !integer_math(self, key, operator, &value) { return FAILURE; } @@ -106,8 +126,22 @@ impl<'a> VariableStore for Shell<'a> { for action in assignment_actions { match action { Ok(Action::UpdateArray(key, Operator::Equal, expression)) => { - let value = expand_string(expression, self, false); - env::set_var(key.name, &value.join(" ")); + let values = expand_string(expression, self, false); + let use_original = match array_is_valid(&values, key.kind) { + Ok(Some(normalized)) => { + env::set_var(key.name, normalized.join(" ")); + false + } + Ok(None) => true, + Err(why) => { + eprintln!("ion: assignment error: {}", why); + return FAILURE; + } + }; + + if use_original { + env::set_var(key.name, values.join(" ")); + } } Ok(Action::UpdateArray(..)) => { eprintln!("ion: arithmetic operators on array expressions aren't supported yet."); @@ -115,7 +149,15 @@ impl<'a> VariableStore for Shell<'a> { } Ok(Action::UpdateString(key, operator, expression)) => { let value = expand_string(expression, self, false).join(" "); - if !integer_math_export(key, operator, &value) { + let value = match string_is_valid(&value, key.kind) { + Ok(value) => value, + Err(why) => { + eprintln!("ion: assignment error: {}", why); + return FAILURE; + } + }; + + if !integer_math_export(key, operator, value) { return FAILURE; } } @@ -147,6 +189,71 @@ impl<'a> VariableStore for Shell<'a> { } } +fn is_boolean(value: &str) -> Result<&str, ()> { + if ["true", "1", "y"].contains(&value) { + Ok("true") + } else if ["false", "0", "n"].contains(&value) { + Ok("false") + } else { + Err(()) + } +} + +fn string_is_valid(value: &str, expected: TypePrimitive) -> Result<&str, TypeError> { + match expected { + TypePrimitive::Any | TypePrimitive::Str => Ok(value), + TypePrimitive::Boolean => is_boolean(value).map_err(|_| TypeError::BadValue(expected)), + TypePrimitive::Integer => { + if value.parse::<i64>().is_ok() { + Ok(value) + } else { + Err(TypeError::BadValue(expected)) + } + } + TypePrimitive::Float => { + if value.parse::<f64>().is_ok() { + Ok(value) + } else { + Err(TypeError::BadValue(expected)) + } + } + _ => unreachable!(), + } +} + +fn array_is_valid(values: &[String], expected: TypePrimitive) -> Result<Option<Array>, TypeError> { + match expected { + TypePrimitive::AnyArray | TypePrimitive::StrArray => Ok(None), + TypePrimitive::BooleanArray => { + let mut output = SmallVec::new(); + for value in values { + let value = is_boolean(value.as_str()) + .map_err(|_| TypeError::BadValue(expected))? + .into(); + output.push(value); + } + Ok(Some(output)) + } + TypePrimitive::IntegerArray => { + for value in values { + if !value.parse::<i64>().is_ok() { + return Err(TypeError::BadValue(expected)); + } + } + Ok(None) + } + TypePrimitive::FloatArray => { + for value in values { + if !value.parse::<f64>().is_ok() { + return Err(TypeError::BadValue(expected)); + } + } + Ok(None) + } + _ => unreachable!(), + } +} + // NOTE: Here there be excessively long functions. fn integer_math(shell: &mut Shell, key: TypeArg, operator: Operator, value: &str) -> bool { @@ -168,9 +275,9 @@ fn integer_math(shell: &mut Shell, key: TypeArg, operator: Operator, value: &str } return false; } else if let TypePrimitive::Integer = key.kind { - match shell.variables.get_var_or_empty(key.name).parse::<u64>() { + match shell.variables.get_var_or_empty(key.name).parse::<i64>() { Ok(lhs) => { - match value.parse::<u64>() { + match value.parse::<i64>() { Ok(rhs) => { let value = (lhs + rhs).to_string(); shell.variables.set_var(key.name, &value); @@ -204,9 +311,9 @@ fn integer_math(shell: &mut Shell, key: TypeArg, operator: Operator, value: &str } return false; } else if let TypePrimitive::Integer = key.kind { - match shell.variables.get_var_or_empty(key.name).parse::<u64>() { + match shell.variables.get_var_or_empty(key.name).parse::<i64>() { Ok(lhs) => { - match value.parse::<u64>() { + match value.parse::<i64>() { Ok(rhs) => { let value = (lhs / rhs).to_string(); shell.variables.set_var(key.name, &value); @@ -240,9 +347,9 @@ fn integer_math(shell: &mut Shell, key: TypeArg, operator: Operator, value: &str } return false; } else if let TypePrimitive::Integer = key.kind { - match shell.variables.get_var_or_empty(key.name).parse::<u64>() { + match shell.variables.get_var_or_empty(key.name).parse::<i64>() { Ok(lhs) => { - match value.parse::<u64>() { + match value.parse::<i64>() { Ok(rhs) => { let value = (lhs - rhs).to_string(); shell.variables.set_var(key.name, &value); @@ -276,9 +383,9 @@ fn integer_math(shell: &mut Shell, key: TypeArg, operator: Operator, value: &str } return false; } else if let TypePrimitive::Integer = key.kind { - match shell.variables.get_var_or_empty(key.name).parse::<u64>() { + match shell.variables.get_var_or_empty(key.name).parse::<i64>() { Ok(lhs) => { - match value.parse::<u64>() { + match value.parse::<i64>() { Ok(rhs) => { let value = (lhs * rhs).to_string(); shell.variables.set_var(key.name, &value); @@ -312,7 +419,7 @@ fn integer_math(shell: &mut Shell, key: TypeArg, operator: Operator, value: &str } return false; } else if let TypePrimitive::Integer = key.kind { - match shell.variables.get_var_or_empty(key.name).parse::<u64>() { + match shell.variables.get_var_or_empty(key.name).parse::<i64>() { Ok(lhs) => { match value.parse::<u32>() { Ok(rhs) => { @@ -357,9 +464,9 @@ fn integer_math_export(key: TypeArg, operator: Operator, value: &str) -> bool { } return false; } else if let TypePrimitive::Integer = key.kind { - match env::var(key.name).unwrap_or("".into()).parse::<u64>() { + match env::var(key.name).unwrap_or("".into()).parse::<i64>() { Ok(lhs) => { - match value.parse::<u64>() { + match value.parse::<i64>() { Ok(rhs) => { let value = (lhs + rhs).to_string(); env::set_var(key.name, &value); @@ -393,9 +500,9 @@ fn integer_math_export(key: TypeArg, operator: Operator, value: &str) -> bool { } return false; } else if let TypePrimitive::Integer = key.kind { - match env::var(key.name).unwrap_or("".into()).parse::<u64>() { + match env::var(key.name).unwrap_or("".into()).parse::<i64>() { Ok(lhs) => { - match value.parse::<u64>() { + match value.parse::<i64>() { Ok(rhs) => { let value = (lhs / rhs).to_string(); env::set_var(key.name, &value); @@ -429,9 +536,9 @@ fn integer_math_export(key: TypeArg, operator: Operator, value: &str) -> bool { } return false; } else if let TypePrimitive::Integer = key.kind { - match env::var(key.name).unwrap_or("".into()).parse::<u64>() { + match env::var(key.name).unwrap_or("".into()).parse::<i64>() { Ok(lhs) => { - match value.parse::<u64>() { + match value.parse::<i64>() { Ok(rhs) => { let value = (lhs - rhs).to_string(); env::set_var(key.name, &value); @@ -465,9 +572,9 @@ fn integer_math_export(key: TypeArg, operator: Operator, value: &str) -> bool { } return false; } else if let TypePrimitive::Integer = key.kind { - match env::var(key.name).unwrap_or("".into()).parse::<u64>() { + match env::var(key.name).unwrap_or("".into()).parse::<i64>() { Ok(lhs) => { - match value.parse::<u64>() { + match value.parse::<i64>() { Ok(rhs) => { let value = (lhs * rhs).to_string(); env::set_var(key.name, &value); @@ -501,7 +608,7 @@ fn integer_math_export(key: TypeArg, operator: Operator, value: &str) -> bool { } return false; } else if let TypePrimitive::Integer = key.kind { - match env::var(key.name).unwrap_or("".into()).parse::<u64>() { + match env::var(key.name).unwrap_or("".into()).parse::<i64>() { Ok(lhs) => { match value.parse::<u32>() { Ok(rhs) => { -- GitLab