CantConvert improvements (#4926)

* CantConvert improvements

* cargo fmt
This commit is contained in:
Jonathan Moore 2022-03-24 07:04:31 -05:00 committed by GitHub
parent 5d5b02d8dc
commit ea7c8c237e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
11 changed files with 45 additions and 20 deletions

View file

@ -136,10 +136,11 @@ fn string_to_boolean(s: &str, span: Span) -> Result<bool, ShellError> {
let val = o.parse::<f64>(); let val = o.parse::<f64>();
match val { match val {
Ok(f) => Ok(f.abs() >= f64::EPSILON), Ok(f) => Ok(f.abs() >= f64::EPSILON),
Err(_) => Err(ShellError::CantConvert( Err(_) => Err(ShellError::CantConvertWithHelp(
"boolean".to_string(), "boolean".to_string(),
"string".to_string(), "string".to_string(),
span, span,
r#"the strings "true" and "false" can be converted into a bool"#.to_string(),
)), )),
} }
} }

View file

@ -260,10 +260,11 @@ fn action(
Ok(d) => Value::Date { val: d, span: head }, Ok(d) => Value::Date { val: d, span: head },
Err(reason) => { Err(reason) => {
return Value::Error { return Value::Error {
error: ShellError::CantConvert( error: ShellError::CantConvertWithHelp(
format!("could not parse as datetime using format '{}'", dt.0), format!("could not parse as datetime using format '{}'", dt.0),
reason.to_string(), reason.to_string(),
head, head,
"you can use `into datetime` without a format string to enable flexible parsing".to_string()
), ),
} }
} }

View file

@ -151,10 +151,11 @@ fn string_to_duration(s: &str, span: Span) -> Result<i64, ShellError> {
} }
} }
Err(ShellError::CantConvert( Err(ShellError::CantConvertWithHelp(
"duration".to_string(), "duration".to_string(),
"string".to_string(), "string".to_string(),
span, span,
"supported units are ns, us, ms, sec, min, hr, day, and wk".to_string(),
)) ))
} }

View file

@ -206,8 +206,8 @@ fn convert_int(input: &Value, head: Span, radix: u32) -> Value {
}; };
match i64::from_str_radix(&i, radix) { match i64::from_str_radix(&i, radix) {
Ok(n) => Value::Int { val: n, span: head }, Ok(n) => Value::Int { val: n, span: head },
Err(reason) => Value::Error { Err(_reason) => Value::Error {
error: ShellError::CantConvert("".to_string(), reason.to_string(), head), error: ShellError::CantConvert("int".to_string(), "string".to_string(), head),
}, },
} }
} }
@ -218,11 +218,12 @@ fn int_from_string(a_string: &str, span: Span) -> Result<i64, ShellError> {
b if b.starts_with("0b") => { b if b.starts_with("0b") => {
let num = match i64::from_str_radix(b.trim_start_matches("0b"), 2) { let num = match i64::from_str_radix(b.trim_start_matches("0b"), 2) {
Ok(n) => n, Ok(n) => n,
Err(reason) => { Err(_reason) => {
return Err(ShellError::CantConvert( return Err(ShellError::CantConvertWithHelp(
"could not parse as integer".to_string(), "int".to_string(),
reason.to_string(), "string".to_string(),
span, span,
r#"digits following "0b" can only be 0 or 1"#.to_string(),
)) ))
} }
}; };
@ -231,11 +232,13 @@ fn int_from_string(a_string: &str, span: Span) -> Result<i64, ShellError> {
h if h.starts_with("0x") => { h if h.starts_with("0x") => {
let num = match i64::from_str_radix(h.trim_start_matches("0x"), 16) { let num = match i64::from_str_radix(h.trim_start_matches("0x"), 16) {
Ok(n) => n, Ok(n) => n,
Err(reason) => { Err(_reason) => {
return Err(ShellError::CantConvert( return Err(ShellError::CantConvertWithHelp(
"could not parse as int".to_string(), "int".to_string(),
reason.to_string(), "string".to_string(),
span, span,
r#"hexadecimal digits following "0x" should be in 0-9, a-f, or A-F"#
.to_string(),
)) ))
} }
}; };
@ -246,7 +249,7 @@ fn int_from_string(a_string: &str, span: Span) -> Result<i64, ShellError> {
Err(_) => match a_string.parse::<f64>() { Err(_) => match a_string.parse::<f64>() {
Ok(f) => Ok(f as i64), Ok(f) => Ok(f as i64),
_ => Err(ShellError::CantConvert( _ => Err(ShellError::CantConvert(
"into int".to_string(), "int".to_string(),
"string".to_string(), "string".to_string(),
span, span,
)), )),

View file

@ -257,6 +257,14 @@ pub fn action(
span, span,
), ),
}, },
Value::Binary { .. } => Value::Error {
error: ShellError::CantConvertWithHelp(
"string".into(),
"binary".into(),
span,
"try using the `decode` command".into(),
),
},
x => Value::Error { x => Value::Error {
error: ShellError::CantConvert(String::from("string"), x.get_type().to_string(), span), error: ShellError::CantConvert(String::from("string"), x.get_type().to_string(), span),
}, },

View file

@ -193,7 +193,7 @@ fn from_eml(
.with_body_preview(body_preview) .with_body_preview(body_preview)
.parse() .parse()
.map_err(|_| { .map_err(|_| {
ShellError::CantConvert("structured data from eml".into(), "string".into(), head) ShellError::CantConvert("structured eml data".into(), "string".into(), head)
})?; })?;
let mut collected = IndexMap::new(); let mut collected = IndexMap::new();

View file

@ -135,7 +135,7 @@ fn convert_nujson_to_value(value: &nu_json::Value, span: Span) -> Value {
Value::Error { Value::Error {
error: ShellError::CantConvert( error: ShellError::CantConvert(
"i64 sized integer".into(), "i64 sized integer".into(),
"larger than i64".into(), "value larger than i64".into(),
span, span,
), ),
} }
@ -201,7 +201,7 @@ fn convert_string_to_value(string_input: String, span: Span) -> Result<Value, Sh
)) ))
} }
x => Err(ShellError::CantConvert( x => Err(ShellError::CantConvert(
format!("structured data from json ({})", x), format!("structured json data ({})", x),
"string".into(), "string".into(),
span, span,
)), )),

View file

@ -121,7 +121,7 @@ pub fn convert_string_to_value(string_input: String, span: Span) -> Result<Value
Ok(value) => Ok(convert_toml_to_value(&value, span)), Ok(value) => Ok(convert_toml_to_value(&value, span)),
Err(_x) => Err(ShellError::CantConvert( Err(_x) => Err(ShellError::CantConvert(
"structured data from toml".into(), "structured toml data".into(),
"string".into(), "string".into(),
span, span,
)), )),

View file

@ -47,7 +47,8 @@ impl From<ShellError> for LabeledError {
msg, msg,
span: None, span: None,
}, },
ShellError::CantConvert(expected, input, span) => LabeledError { ShellError::CantConvert(expected, input, span)
| ShellError::CantConvertWithHelp(expected, input, span, _) => LabeledError {
label: format!("Can't convert to {}", expected), label: format!("Can't convert to {}", expected),
msg: format!("can't convert {} to {}", expected, input), msg: format!("can't convert {} to {}", expected, input),
span: Some(span), span: Some(span),

View file

@ -112,6 +112,16 @@ pub enum ShellError {
#[diagnostic(code(nu::shell::cant_convert), url(docsrs))] #[diagnostic(code(nu::shell::cant_convert), url(docsrs))]
CantConvert(String, String, #[label("can't convert {1} to {0}")] Span), CantConvert(String, String, #[label("can't convert {1} to {0}")] Span),
// Identical to above, but with help
#[error("Can't convert to {0}.")]
#[diagnostic(code(nu::shell::cant_convert), url(docsrs), help("{3}"))]
CantConvertWithHelp(
String,
String,
#[label("can't convert {1} to {0}")] Span,
String,
),
#[error("{0} is not representable as a string.")] #[error("{0} is not representable as a string.")]
#[diagnostic( #[diagnostic(
code(nu::shell::env_var_not_a_string), code(nu::shell::env_var_not_a_string),

View file

@ -16,7 +16,7 @@ impl Value {
match self { match self {
Value::Int { val, .. } => Ok(*val), Value::Int { val, .. } => Ok(*val),
x => Err(ShellError::CantConvert( x => Err(ShellError::CantConvert(
"rf64".into(), "i64".into(),
x.get_type().to_string(), x.get_type().to_string(),
self.span()?, self.span()?,
)), )),