use ahash::{HashMap, HashMapExt}; use std::path::PathBuf; use std::str::FromStr; use crate::ast::{CellPath, MatchPattern, PathMember}; use crate::engine::{Block, Closure}; use crate::ShellError; use crate::{Range, Spanned, Value}; use chrono::{DateTime, FixedOffset}; pub trait FromValue: Sized { fn from_value(v: &Value) -> Result; } impl FromValue for Value { fn from_value(v: &Value) -> Result { Ok(v.clone()) } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { match v { Value::Int { val, span } => Ok(Spanned { item: *val, span: *span, }), Value::Filesize { val, span } => Ok(Spanned { item: *val, span: *span, }), Value::Duration { val, span } => Ok(Spanned { item: *val, span: *span, }), v => Err(ShellError::CantConvert { to_type: "integer".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for i64 { fn from_value(v: &Value) -> Result { match v { Value::Int { val, .. } => Ok(*val), Value::Filesize { val, .. } => Ok(*val), Value::Duration { val, .. } => Ok(*val), v => Err(ShellError::CantConvert { to_type: "integer".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { match v { Value::Int { val, span } => Ok(Spanned { item: *val as f64, span: *span, }), Value::Float { val, span } => Ok(Spanned { item: *val, span: *span, }), v => Err(ShellError::CantConvert { to_type: "float".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for f64 { fn from_value(v: &Value) -> Result { match v { Value::Float { val, .. } => Ok(*val), Value::Int { val, .. } => Ok(*val as f64), v => Err(ShellError::CantConvert { to_type: "float".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { match v { Value::Int { val, span } => { if val.is_negative() { Err(ShellError::NeedsPositiveValue(*span)) } else { Ok(Spanned { item: *val as usize, span: *span, }) } } Value::Filesize { val, span } => { if val.is_negative() { Err(ShellError::NeedsPositiveValue(*span)) } else { Ok(Spanned { item: *val as usize, span: *span, }) } } Value::Duration { val, span } => { if val.is_negative() { Err(ShellError::NeedsPositiveValue(*span)) } else { Ok(Spanned { item: *val as usize, span: *span, }) } } v => Err(ShellError::CantConvert { to_type: "non-negative integer".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for usize { fn from_value(v: &Value) -> Result { match v { Value::Int { val, span } => { if val.is_negative() { Err(ShellError::NeedsPositiveValue(*span)) } else { Ok(*val as usize) } } Value::Filesize { val, span } => { if val.is_negative() { Err(ShellError::NeedsPositiveValue(*span)) } else { Ok(*val as usize) } } Value::Duration { val, span } => { if val.is_negative() { Err(ShellError::NeedsPositiveValue(*span)) } else { Ok(*val as usize) } } v => Err(ShellError::CantConvert { to_type: "non-negative integer".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for String { fn from_value(v: &Value) -> Result { // FIXME: we may want to fail a little nicer here match v { Value::CellPath { val, .. } => Ok(val.into_string()), Value::String { val, .. } => Ok(val.clone()), v => Err(ShellError::CantConvert { to_type: "string".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { Ok(Spanned { item: match v { Value::CellPath { val, .. } => val.into_string(), Value::String { val, .. } => val.clone(), v => { return Err(ShellError::CantConvert { to_type: "string".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }) } }, span: v.span()?, }) } } impl FromValue for Vec { fn from_value(v: &Value) -> Result { // FIXME: we may want to fail a little nicer here match v { Value::List { vals, .. } => vals .iter() .map(|val| match val { Value::String { val, .. } => Ok(val.clone()), c => Err(ShellError::CantConvert { to_type: "string".into(), from_type: c.get_type().to_string(), span: c.span()?, help: None, }), }) .collect::, ShellError>>(), v => Err(ShellError::CantConvert { to_type: "string".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Vec> { fn from_value(v: &Value) -> Result { // FIXME: we may want to fail a little nicer here match v { Value::List { vals, .. } => vals .iter() .map(|val| match val { Value::String { val, span } => Ok(Spanned { item: val.clone(), span: *span, }), c => Err(ShellError::CantConvert { to_type: "string".into(), from_type: c.get_type().to_string(), span: c.span()?, help: None, }), }) .collect::>, ShellError>>(), v => Err(ShellError::CantConvert { to_type: "string".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Vec { fn from_value(v: &Value) -> Result { match v { Value::List { vals, .. } => vals .iter() .map(|val| match val { Value::Bool { val, .. } => Ok(*val), c => Err(ShellError::CantConvert { to_type: "bool".into(), from_type: c.get_type().to_string(), span: c.span()?, help: None, }), }) .collect::, ShellError>>(), v => Err(ShellError::CantConvert { to_type: "bool".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for CellPath { fn from_value(v: &Value) -> Result { let span = v.span()?; match v { Value::CellPath { val, .. } => Ok(val.clone()), Value::String { val, .. } => Ok(CellPath { members: vec![PathMember::String { val: val.clone(), span, optional: false, }], }), Value::Int { val, span } => { if val.is_negative() { Err(ShellError::NeedsPositiveValue(*span)) } else { Ok(CellPath { members: vec![PathMember::Int { val: *val as usize, span: *span, optional: false, }], }) } } x => Err(ShellError::CantConvert { to_type: "cell path".into(), from_type: x.get_type().to_string(), span, help: None, }), } } } impl FromValue for bool { fn from_value(v: &Value) -> Result { match v { Value::Bool { val, .. } => Ok(*val), v => Err(ShellError::CantConvert { to_type: "bool".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { match v { Value::Bool { val, span } => Ok(Spanned { item: *val, span: *span, }), v => Err(ShellError::CantConvert { to_type: "bool".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for DateTime { fn from_value(v: &Value) -> Result { match v { Value::Date { val, .. } => Ok(*val), v => Err(ShellError::CantConvert { to_type: "date".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned> { fn from_value(v: &Value) -> Result { match v { Value::Date { val, span } => Ok(Spanned { item: *val, span: *span, }), v => Err(ShellError::CantConvert { to_type: "date".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Range { fn from_value(v: &Value) -> Result { match v { Value::Range { val, .. } => Ok((**val).clone()), v => Err(ShellError::CantConvert { to_type: "range".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { match v { Value::Range { val, span } => Ok(Spanned { item: (**val).clone(), span: *span, }), v => Err(ShellError::CantConvert { to_type: "range".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Vec { fn from_value(v: &Value) -> Result { match v { Value::Binary { val, .. } => Ok(val.clone()), Value::String { val, .. } => Ok(val.bytes().collect()), v => Err(ShellError::CantConvert { to_type: "binary data".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned> { fn from_value(v: &Value) -> Result { match v { Value::Binary { val, span } => Ok(Spanned { item: val.clone(), span: *span, }), Value::String { val, span } => Ok(Spanned { item: val.bytes().collect(), span: *span, }), v => Err(ShellError::CantConvert { to_type: "binary data".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { match v { Value::String { val, span } => Ok(Spanned { item: PathBuf::from_str(val) .map_err(|err| ShellError::FileNotFoundCustom(err.to_string(), *span))?, span: *span, }), v => Err(ShellError::CantConvert { to_type: "range".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Vec { fn from_value(v: &Value) -> Result { // FIXME: we may want to fail a little nicer here match v { Value::List { vals, .. } => Ok(vals.clone()), v => Err(ShellError::CantConvert { to_type: "Vector of values".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } // A record impl FromValue for (Vec, Vec) { fn from_value(v: &Value) -> Result { match v { Value::Record { cols, vals, .. } => Ok((cols.clone(), vals.clone())), v => Err(ShellError::CantConvert { to_type: "Record".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Closure { fn from_value(v: &Value) -> Result { match v { Value::Closure { val, captures, .. } => Ok(Closure { block_id: *val, captures: captures.clone(), }), Value::Block { val, .. } => Ok(Closure { block_id: *val, captures: HashMap::new(), }), v => Err(ShellError::CantConvert { to_type: "Closure".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Block { fn from_value(v: &Value) -> Result { match v { Value::Block { val, .. } => Ok(Block { block_id: *val }), v => Err(ShellError::CantConvert { to_type: "Block".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { match v { Value::Closure { val, captures, span, } => Ok(Spanned { item: Closure { block_id: *val, captures: captures.clone(), }, span: *span, }), v => Err(ShellError::CantConvert { to_type: "Closure".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for Spanned { fn from_value(v: &Value) -> Result { match v { Value::MatchPattern { val, span } => Ok(Spanned { item: *val.clone(), span: *span, }), v => Err(ShellError::CantConvert { to_type: "Match pattern".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } } impl FromValue for MatchPattern { fn from_value(v: &Value) -> Result { match v { Value::MatchPattern { val, .. } => Ok(*val.clone()), v => Err(ShellError::CantConvert { to_type: "Match pattern".into(), from_type: v.get_type().to_string(), span: v.span()?, help: None, }), } } }