2019-05-15 22:58:44 +00:00
|
|
|
#[allow(unused)]
|
2019-05-15 18:14:51 +00:00
|
|
|
use crate::prelude::*;
|
2019-06-22 03:43:37 +00:00
|
|
|
|
2019-06-24 00:55:31 +00:00
|
|
|
use crate::parser::{Span, Spanned};
|
2019-06-30 06:14:40 +00:00
|
|
|
use ansi_term::Color;
|
2019-05-10 16:59:12 +00:00
|
|
|
use derive_new::new;
|
2019-06-07 22:35:07 +00:00
|
|
|
use language_reporting::{Diagnostic, Label, Severity};
|
2019-06-01 05:50:16 +00:00
|
|
|
use serde::{Deserialize, Deserializer, Serialize, Serializer};
|
2019-05-10 16:59:12 +00:00
|
|
|
|
2019-06-24 00:55:31 +00:00
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Ord, PartialOrd, Serialize, Deserialize)]
|
|
|
|
pub enum Description {
|
|
|
|
Source(Spanned<String>),
|
|
|
|
Synthetic(String),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Description {
|
|
|
|
pub fn from(item: Spanned<impl Into<String>>) -> Description {
|
|
|
|
match item {
|
|
|
|
Spanned {
|
|
|
|
span: Span { start: 0, end: 0 },
|
|
|
|
item,
|
|
|
|
} => Description::Synthetic(item.into()),
|
|
|
|
Spanned { span, item } => Description::Source(Spanned::from_item(item.into(), span)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Description {
|
|
|
|
fn into_label(self) -> Result<Label<Span>, String> {
|
|
|
|
match self {
|
|
|
|
Description::Source(s) => Ok(Label::new_primary(s.span).with_message(s.item)),
|
|
|
|
Description::Synthetic(s) => Err(s),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-30 06:14:40 +00:00
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Ord, PartialOrd, Serialize, Deserialize)]
|
|
|
|
pub enum ArgumentError {
|
|
|
|
MissingMandatoryFlag(String),
|
|
|
|
MissingMandatoryPositional(String),
|
|
|
|
MissingValueForName(String),
|
|
|
|
}
|
|
|
|
|
2019-07-03 20:31:15 +00:00
|
|
|
pub fn labelled(
|
|
|
|
span: impl Into<Option<Span>>,
|
|
|
|
heading: &'a str,
|
|
|
|
span_message: &'a str,
|
|
|
|
) -> impl FnOnce(ShellError) -> ShellError + 'a {
|
|
|
|
let span = span.into();
|
|
|
|
|
2019-07-08 16:44:53 +00:00
|
|
|
move |_| ShellError::maybe_labeled_error(heading, span_message, span)
|
2019-07-03 20:31:15 +00:00
|
|
|
}
|
|
|
|
|
2019-06-01 05:50:16 +00:00
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Ord, PartialOrd, Serialize, Deserialize)]
|
2019-05-30 04:19:46 +00:00
|
|
|
pub enum ShellError {
|
|
|
|
String(StringError),
|
2019-06-29 08:55:42 +00:00
|
|
|
TypeError {
|
|
|
|
expected: String,
|
|
|
|
actual: Spanned<Option<String>>,
|
|
|
|
},
|
2019-06-24 00:55:31 +00:00
|
|
|
MissingProperty {
|
|
|
|
subpath: Description,
|
|
|
|
expr: Description,
|
|
|
|
},
|
2019-06-30 06:14:40 +00:00
|
|
|
ArgumentError {
|
2019-07-03 20:31:15 +00:00
|
|
|
command: String,
|
2019-06-30 06:14:40 +00:00
|
|
|
error: ArgumentError,
|
|
|
|
span: Span,
|
|
|
|
},
|
2019-06-07 22:35:07 +00:00
|
|
|
Diagnostic(ShellDiagnostic),
|
2019-06-24 00:55:31 +00:00
|
|
|
CoerceError {
|
|
|
|
left: Spanned<String>,
|
|
|
|
right: Spanned<String>,
|
|
|
|
},
|
2019-05-13 17:30:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl ShellError {
|
2019-06-29 08:55:42 +00:00
|
|
|
crate fn type_error(
|
|
|
|
expected: impl Into<String>,
|
|
|
|
actual: Spanned<impl Into<String>>,
|
|
|
|
) -> ShellError {
|
|
|
|
ShellError::TypeError {
|
|
|
|
expected: expected.into(),
|
|
|
|
actual: actual.map(|i| Some(i.into())),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-30 04:19:46 +00:00
|
|
|
crate fn parse_error(
|
2019-06-22 03:43:37 +00:00
|
|
|
error: nom::Err<(nom_locate::LocatedSpan<&str>, nom::error::ErrorKind)>,
|
2019-05-30 04:19:46 +00:00
|
|
|
) -> ShellError {
|
|
|
|
use language_reporting::*;
|
|
|
|
|
|
|
|
match error {
|
2019-06-22 03:43:37 +00:00
|
|
|
nom::Err::Incomplete(_) => unreachable!(),
|
|
|
|
nom::Err::Failure(span) | nom::Err::Error(span) => {
|
2019-06-29 08:55:42 +00:00
|
|
|
let diagnostic = Diagnostic::new(Severity::Error, format!("Parse Error"))
|
|
|
|
.with_label(Label::new_primary(Span::from(span.0)));
|
2019-05-30 04:19:46 +00:00
|
|
|
|
2019-06-07 22:35:07 +00:00
|
|
|
ShellError::diagnostic(diagnostic)
|
2019-06-22 03:43:37 +00:00
|
|
|
// nom::Context::Code(span, kind) => {
|
|
|
|
// let diagnostic =
|
|
|
|
// Diagnostic::new(Severity::Error, format!("{}", kind.description()))
|
|
|
|
// .with_label(Label::new_primary(Span::from(span)));
|
|
|
|
|
|
|
|
// ShellError::diagnostic(diagnostic)
|
|
|
|
// }
|
2019-06-29 08:55:42 +00:00
|
|
|
} // ParseError::UnrecognizedToken {
|
|
|
|
// token: (start, SpannedToken { token, .. }, end),
|
|
|
|
// expected,
|
|
|
|
// } => {
|
|
|
|
// let diagnostic = Diagnostic::new(
|
|
|
|
// Severity::Error,
|
|
|
|
// format!("Unexpected {:?}, expected {:?}", token, expected),
|
|
|
|
// )
|
|
|
|
// .with_label(Label::new_primary(Span::from((start, end))));
|
|
|
|
|
|
|
|
// ShellError::diagnostic(diagnostic)
|
|
|
|
// }
|
|
|
|
// ParseError::User { error } => error,
|
|
|
|
// other => ShellError::string(format!("{:?}", other)),
|
2019-05-30 04:19:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-07 22:35:07 +00:00
|
|
|
crate fn diagnostic(diagnostic: Diagnostic<Span>) -> ShellError {
|
|
|
|
ShellError::Diagnostic(ShellDiagnostic { diagnostic })
|
|
|
|
}
|
|
|
|
|
2019-06-24 00:55:31 +00:00
|
|
|
crate fn to_diagnostic(self) -> Diagnostic<Span> {
|
|
|
|
match self {
|
|
|
|
ShellError::String(StringError { title, .. }) => {
|
|
|
|
Diagnostic::new(Severity::Error, title)
|
|
|
|
}
|
2019-07-03 20:31:15 +00:00
|
|
|
ShellError::ArgumentError {
|
|
|
|
command,
|
|
|
|
error,
|
|
|
|
span,
|
|
|
|
} => match error {
|
2019-06-30 06:14:40 +00:00
|
|
|
ArgumentError::MissingMandatoryFlag(name) => Diagnostic::new(
|
|
|
|
Severity::Error,
|
|
|
|
format!(
|
2019-07-03 20:31:15 +00:00
|
|
|
"{} requires {}{}",
|
|
|
|
Color::Cyan.paint(command),
|
|
|
|
Color::Black.bold().paint("--"),
|
|
|
|
Color::Black.bold().paint(name)
|
2019-06-30 06:14:40 +00:00
|
|
|
),
|
|
|
|
)
|
|
|
|
.with_label(Label::new_primary(span)),
|
|
|
|
ArgumentError::MissingMandatoryPositional(name) => Diagnostic::new(
|
|
|
|
Severity::Error,
|
2019-07-03 20:31:15 +00:00
|
|
|
format!(
|
|
|
|
"{} requires {}",
|
|
|
|
Color::Cyan.paint(command),
|
|
|
|
Color::Green.bold().paint(name)
|
|
|
|
),
|
2019-06-30 06:14:40 +00:00
|
|
|
)
|
|
|
|
.with_label(Label::new_primary(span)),
|
|
|
|
|
|
|
|
ArgumentError::MissingValueForName(name) => Diagnostic::new(
|
|
|
|
Severity::Error,
|
|
|
|
format!(
|
2019-07-03 20:31:15 +00:00
|
|
|
"{} is missing value for flag {}{}",
|
|
|
|
Color::Cyan.paint(command),
|
|
|
|
Color::Black.bold().paint("--"),
|
|
|
|
Color::Black.bold().paint(name)
|
2019-06-30 06:14:40 +00:00
|
|
|
),
|
|
|
|
)
|
|
|
|
.with_label(Label::new_primary(span)),
|
|
|
|
},
|
2019-06-29 08:55:42 +00:00
|
|
|
ShellError::TypeError {
|
|
|
|
expected,
|
|
|
|
actual:
|
|
|
|
Spanned {
|
|
|
|
item: Some(actual),
|
|
|
|
span,
|
|
|
|
},
|
|
|
|
} => Diagnostic::new(Severity::Error, "Type Error").with_label(
|
|
|
|
Label::new_primary(span)
|
|
|
|
.with_message(format!("Expected {}, found {}", expected, actual)),
|
|
|
|
),
|
|
|
|
|
|
|
|
ShellError::TypeError {
|
|
|
|
expected,
|
|
|
|
actual: Spanned { item: None, span },
|
|
|
|
} => Diagnostic::new(Severity::Error, "Type Error")
|
|
|
|
.with_label(Label::new_primary(span).with_message(expected)),
|
2019-06-24 00:55:31 +00:00
|
|
|
|
|
|
|
ShellError::MissingProperty { subpath, expr } => {
|
|
|
|
let subpath = subpath.into_label();
|
|
|
|
let expr = expr.into_label();
|
|
|
|
|
|
|
|
let mut diag = Diagnostic::new(Severity::Error, "Missing property");
|
|
|
|
|
|
|
|
match subpath {
|
|
|
|
Ok(label) => diag = diag.with_label(label),
|
|
|
|
Err(ty) => diag.message = format!("Missing property (for {})", ty),
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Ok(label) = expr {
|
|
|
|
diag = diag.with_label(label);
|
|
|
|
}
|
|
|
|
|
|
|
|
diag
|
|
|
|
}
|
|
|
|
|
|
|
|
ShellError::Diagnostic(diag) => diag.diagnostic,
|
|
|
|
ShellError::CoerceError { left, right } => {
|
|
|
|
Diagnostic::new(Severity::Error, "Coercion error")
|
|
|
|
.with_label(Label::new_primary(left.span).with_message(left.item))
|
|
|
|
.with_label(Label::new_secondary(right.span).with_message(right.item))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-07 22:35:07 +00:00
|
|
|
crate fn labeled_error(
|
|
|
|
msg: impl Into<String>,
|
|
|
|
label: impl Into<String>,
|
|
|
|
span: Span,
|
|
|
|
) -> ShellError {
|
|
|
|
ShellError::diagnostic(
|
|
|
|
Diagnostic::new(Severity::Error, msg.into())
|
|
|
|
.with_label(Label::new_primary(span).with_message(label.into())),
|
|
|
|
)
|
2019-05-30 04:19:46 +00:00
|
|
|
}
|
|
|
|
|
2019-06-15 18:36:17 +00:00
|
|
|
crate fn maybe_labeled_error(
|
|
|
|
msg: impl Into<String>,
|
|
|
|
label: impl Into<String>,
|
|
|
|
span: Option<Span>,
|
|
|
|
) -> ShellError {
|
|
|
|
match span {
|
|
|
|
Some(span) => ShellError::diagnostic(
|
|
|
|
Diagnostic::new(Severity::Error, msg.into())
|
|
|
|
.with_label(Label::new_primary(span).with_message(label.into())),
|
|
|
|
),
|
|
|
|
None => ShellError::string(msg),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-27 04:56:48 +00:00
|
|
|
pub fn string(title: impl Into<String>) -> ShellError {
|
2019-05-30 04:19:46 +00:00
|
|
|
ShellError::String(StringError::new(title.into(), Value::nothing()))
|
2019-05-13 17:30:51 +00:00
|
|
|
}
|
2019-05-15 18:14:51 +00:00
|
|
|
|
2019-06-04 21:42:31 +00:00
|
|
|
crate fn unimplemented(title: impl Into<String>) -> ShellError {
|
|
|
|
ShellError::string(&format!("Unimplemented: {}", title.into()))
|
|
|
|
}
|
|
|
|
|
2019-06-22 03:43:37 +00:00
|
|
|
crate fn unexpected(title: impl Into<String>) -> ShellError {
|
|
|
|
ShellError::string(&format!("Unexpected: {}", title.into()))
|
|
|
|
}
|
2019-05-30 04:19:46 +00:00
|
|
|
}
|
2019-05-16 21:43:36 +00:00
|
|
|
|
2019-05-30 04:19:46 +00:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct ShellDiagnostic {
|
|
|
|
crate diagnostic: Diagnostic<Span>,
|
|
|
|
}
|
|
|
|
|
2019-06-04 21:42:31 +00:00
|
|
|
impl ShellDiagnostic {
|
2019-06-06 06:34:59 +00:00
|
|
|
#[allow(unused)]
|
2019-06-04 21:42:31 +00:00
|
|
|
crate fn simple_diagnostic(
|
|
|
|
span: impl Into<Span>,
|
|
|
|
source: impl Into<String>,
|
|
|
|
) -> ShellDiagnostic {
|
|
|
|
use language_reporting::*;
|
|
|
|
|
|
|
|
let span = span.into();
|
|
|
|
let source = source.into();
|
|
|
|
|
|
|
|
let diagnostic =
|
|
|
|
Diagnostic::new(Severity::Error, "Parse error").with_label(Label::new_primary(span));
|
|
|
|
|
|
|
|
ShellDiagnostic { diagnostic }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-30 04:19:46 +00:00
|
|
|
impl PartialEq for ShellDiagnostic {
|
|
|
|
fn eq(&self, _other: &ShellDiagnostic) -> bool {
|
|
|
|
false
|
2019-05-16 21:43:36 +00:00
|
|
|
}
|
2019-05-10 16:59:12 +00:00
|
|
|
}
|
|
|
|
|
2019-05-30 04:19:46 +00:00
|
|
|
impl Eq for ShellDiagnostic {}
|
|
|
|
|
|
|
|
impl std::cmp::PartialOrd for ShellDiagnostic {
|
|
|
|
fn partial_cmp(&self, _other: &Self) -> Option<std::cmp::Ordering> {
|
|
|
|
Some(std::cmp::Ordering::Less)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl std::cmp::Ord for ShellDiagnostic {
|
|
|
|
fn cmp(&self, _other: &Self) -> std::cmp::Ordering {
|
|
|
|
std::cmp::Ordering::Less
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Serialize for ShellDiagnostic {
|
|
|
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
|
|
|
where
|
|
|
|
S: Serializer,
|
|
|
|
{
|
|
|
|
"<diagnostic>".serialize(serializer)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-01 05:50:16 +00:00
|
|
|
impl Deserialize<'de> for ShellDiagnostic {
|
|
|
|
fn deserialize<D>(_deserializer: D) -> Result<Self, D::Error>
|
|
|
|
where
|
|
|
|
D: Deserializer<'de>,
|
|
|
|
{
|
|
|
|
Ok(ShellDiagnostic {
|
|
|
|
diagnostic: Diagnostic::new(
|
|
|
|
language_reporting::Severity::Error,
|
|
|
|
"deserialize not implemented for ShellDiagnostic",
|
|
|
|
),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[derive(Debug, Ord, PartialOrd, Eq, PartialEq, new, Clone, Serialize, Deserialize)]
|
2019-05-30 04:19:46 +00:00
|
|
|
pub struct StringError {
|
|
|
|
title: String,
|
|
|
|
error: Value,
|
|
|
|
}
|
|
|
|
|
2019-05-10 16:59:12 +00:00
|
|
|
impl std::fmt::Display for ShellError {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
|
2019-05-30 04:19:46 +00:00
|
|
|
match self {
|
|
|
|
ShellError::String(s) => write!(f, "{}", &s.title),
|
2019-06-03 05:11:21 +00:00
|
|
|
ShellError::TypeError { .. } => write!(f, "TypeError"),
|
|
|
|
ShellError::MissingProperty { .. } => write!(f, "MissingProperty"),
|
2019-06-30 06:14:40 +00:00
|
|
|
ShellError::ArgumentError { .. } => write!(f, "ArgumentError"),
|
2019-06-07 22:35:07 +00:00
|
|
|
ShellError::Diagnostic(_) => write!(f, "<diagnostic>"),
|
2019-06-24 00:55:31 +00:00
|
|
|
ShellError::CoerceError { .. } => write!(f, "CoerceError"),
|
2019-05-30 04:19:46 +00:00
|
|
|
}
|
2019-05-10 16:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl std::error::Error for ShellError {}
|
|
|
|
|
|
|
|
impl std::convert::From<std::io::Error> for ShellError {
|
|
|
|
fn from(input: std::io::Error) -> ShellError {
|
2019-05-30 04:19:46 +00:00
|
|
|
ShellError::String(StringError {
|
2019-05-10 16:59:12 +00:00
|
|
|
title: format!("{}", input),
|
2019-05-13 17:30:51 +00:00
|
|
|
error: Value::nothing(),
|
2019-05-30 04:19:46 +00:00
|
|
|
})
|
2019-05-10 16:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-24 04:34:43 +00:00
|
|
|
|
|
|
|
impl std::convert::From<futures_sink::VecSinkError> for ShellError {
|
|
|
|
fn from(_input: futures_sink::VecSinkError) -> ShellError {
|
2019-05-30 04:19:46 +00:00
|
|
|
ShellError::String(StringError {
|
2019-05-24 04:34:43 +00:00
|
|
|
title: format!("Unexpected Vec Sink Error"),
|
|
|
|
error: Value::nothing(),
|
2019-05-30 04:19:46 +00:00
|
|
|
})
|
2019-05-24 04:34:43 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-24 07:29:16 +00:00
|
|
|
|
|
|
|
impl std::convert::From<subprocess::PopenError> for ShellError {
|
|
|
|
fn from(input: subprocess::PopenError) -> ShellError {
|
2019-05-30 04:19:46 +00:00
|
|
|
ShellError::String(StringError {
|
2019-05-24 07:29:16 +00:00
|
|
|
title: format!("{}", input),
|
|
|
|
error: Value::nothing(),
|
2019-05-30 04:19:46 +00:00
|
|
|
})
|
2019-05-24 07:29:16 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-26 06:54:41 +00:00
|
|
|
|
2019-06-22 03:43:37 +00:00
|
|
|
// impl std::convert::From<nom::Err<(&str, nom::ErrorKind)>> for ShellError {
|
|
|
|
// fn from(input: nom::Err<(&str, nom::ErrorKind)>) -> ShellError {
|
|
|
|
// ShellError::String(StringError {
|
|
|
|
// title: format!("{:?}", input),
|
|
|
|
// error: Value::nothing(),
|
|
|
|
// })
|
|
|
|
// }
|
|
|
|
// }
|
2019-06-01 05:50:16 +00:00
|
|
|
|
|
|
|
impl std::convert::From<toml::ser::Error> for ShellError {
|
|
|
|
fn from(input: toml::ser::Error) -> ShellError {
|
|
|
|
ShellError::String(StringError {
|
|
|
|
title: format!("{:?}", input),
|
|
|
|
error: Value::nothing(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|