#[allow(unused)] use crate::prelude::*; use ansi_term::Color; use derive_new::new; use language_reporting::{Diagnostic, Label, Severity}; use serde::{Deserialize, Serialize}; #[derive(Debug, Eq, PartialEq, Clone, Ord, PartialOrd, Serialize, Deserialize)] pub enum Description { Source(Tagged), Synthetic(String), } impl Description { pub fn from(value: Tagged>) -> Description { let value_span = value.span(); let value_tag = value.tag(); match value_span { Span { start: 0, end: 0 } => Description::Synthetic(value.item.into()), _ => Description::Source(Tagged::from_item(value.item.into(), value_tag)), } } } impl Description { fn into_label(self) -> Result, String> { match self { Description::Source(s) => Ok(Label::new_primary(s.span()).with_message(s.item)), Description::Synthetic(s) => Err(s), } } } #[derive(Debug, Eq, PartialEq, Clone, Ord, PartialOrd, Serialize, Deserialize)] pub enum ArgumentError { MissingMandatoryFlag(String), MissingMandatoryPositional(String), MissingValueForName(String), } pub fn labelled( span: impl Into, heading: &'a str, span_message: &'a str, ) -> impl FnOnce(ShellError) -> ShellError + 'a { let span = span.into(); move |_| ShellError::labeled_error(heading, span_message, span) } #[derive(Debug, Eq, PartialEq, Clone, Ord, PartialOrd, Serialize, Deserialize)] pub struct ShellError { error: ProximateShellError, cause: Option>, } impl serde::de::Error for ShellError { fn custom(msg: T) -> Self where T: std::fmt::Display, { ShellError::string(msg.to_string()) } } impl ShellError { crate fn type_error( expected: impl Into, actual: Tagged>, ) -> ShellError { ProximateShellError::TypeError { expected: expected.into(), actual: actual.map(|i| Some(i.into())), } .start() } crate fn syntax_error(problem: Tagged>) -> ShellError { ProximateShellError::SyntaxError { problem: problem.map(|p| p.into()), } .start() } crate fn invalid_command(problem: impl Into) -> ShellError { ProximateShellError::InvalidCommand { command: problem.into(), } .start() } crate fn coerce_error( left: Tagged>, right: Tagged>, ) -> ShellError { ProximateShellError::CoerceError { left: left.map(|l| l.into()), right: right.map(|r| r.into()), } .start() } crate fn missing_property(subpath: Description, expr: Description) -> ShellError { ProximateShellError::MissingProperty { subpath, expr }.start() } crate fn missing_value(span: Option, reason: impl Into) -> ShellError { ProximateShellError::MissingValue { span, reason: reason.into(), } .start() } crate fn argument_error( command: impl Into, kind: ArgumentError, span: Span, ) -> ShellError { ProximateShellError::ArgumentError { command: command.into(), error: kind, span: span, } .start() } crate fn parse_error( error: nom::Err<(nom5_locate::LocatedSpan<&str>, nom::error::ErrorKind)>, ) -> ShellError { use language_reporting::*; match error { nom::Err::Incomplete(_) => unreachable!(), nom::Err::Failure(span) | nom::Err::Error(span) => { let diagnostic = Diagnostic::new(Severity::Error, format!("Parse Error")) .with_label(Label::new_primary(Span::from(span.0))); ShellError::diagnostic(diagnostic) } } } crate fn diagnostic(diagnostic: Diagnostic) -> ShellError { ProximateShellError::Diagnostic(ShellDiagnostic { diagnostic }).start() } crate fn to_diagnostic(self) -> Diagnostic { match self.error { ProximateShellError::String(StringError { title, .. }) => { Diagnostic::new(Severity::Error, title) } ProximateShellError::InvalidCommand { command } => { Diagnostic::new(Severity::Error, "Invalid command") .with_label(Label::new_primary(command.span)) } ProximateShellError::MissingValue { span, reason } => { let mut d = Diagnostic::new( Severity::Bug, format!("Internal Error (missing value) :: {}", reason), ); if let Some(span) = span { d = d.with_label(Label::new_primary(span)); } d } ProximateShellError::ArgumentError { command, error, span, } => match error { ArgumentError::MissingMandatoryFlag(name) => Diagnostic::new( Severity::Error, format!( "{} requires {}{}", Color::Cyan.paint(command), Color::Black.bold().paint("--"), Color::Black.bold().paint(name) ), ) .with_label(Label::new_primary(span)), ArgumentError::MissingMandatoryPositional(name) => Diagnostic::new( Severity::Error, format!( "{} requires {}", Color::Cyan.paint(command), Color::Green.bold().paint(name) ), ) .with_label(Label::new_primary(span)), ArgumentError::MissingValueForName(name) => Diagnostic::new( Severity::Error, format!( "{} is missing value for flag {}{}", Color::Cyan.paint(command), Color::Black.bold().paint("--"), Color::Black.bold().paint(name) ), ) .with_label(Label::new_primary(span)), }, ProximateShellError::TypeError { expected, actual: Tagged { item: Some(actual), tag: Tag { span, .. }, }, } => Diagnostic::new(Severity::Error, "Type Error").with_label( Label::new_primary(span) .with_message(format!("Expected {}, found {}", expected, actual)), ), ProximateShellError::TypeError { expected, actual: Tagged { item: None, tag: Tag { span, .. }, }, } => Diagnostic::new(Severity::Error, "Type Error") .with_label(Label::new_primary(span).with_message(expected)), ProximateShellError::SyntaxError { problem: Tagged { tag: Tag { span, .. }, .. }, } => Diagnostic::new(Severity::Error, "Syntax Error") .with_label(Label::new_primary(span).with_message("Unexpected external command")), ProximateShellError::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 } ProximateShellError::Diagnostic(diag) => diag.diagnostic, ProximateShellError::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)) } } } pub fn labeled_error( msg: impl Into, label: impl Into, span: impl Into, ) -> ShellError { ShellError::diagnostic( Diagnostic::new(Severity::Error, msg.into()) .with_label(Label::new_primary(span.into()).with_message(label.into())), ) } pub fn labeled_error_with_secondary( msg: impl Into, primary_label: impl Into, primary_span: Span, secondary_label: impl Into, secondary_span: Span, ) -> ShellError { ShellError::diagnostic( Diagnostic::new_error(msg.into()) .with_label(Label::new_primary(primary_span).with_message(primary_label.into())) .with_label( Label::new_secondary(secondary_span).with_message(secondary_label.into()), ), ) } pub fn string(title: impl Into) -> ShellError { ProximateShellError::String(StringError::new(title.into(), Value::nothing())).start() } crate fn unimplemented(title: impl Into) -> ShellError { ShellError::string(&format!("Unimplemented: {}", title.into())) } crate fn unexpected(title: impl Into) -> ShellError { ShellError::string(&format!("Unexpected: {}", title.into())) } } #[derive(Debug, Eq, PartialEq, Clone, Ord, PartialOrd, Serialize, Deserialize)] pub enum ProximateShellError { String(StringError), SyntaxError { problem: Tagged, }, InvalidCommand { command: Tag, }, TypeError { expected: String, actual: Tagged>, }, MissingProperty { subpath: Description, expr: Description, }, MissingValue { span: Option, reason: String, }, ArgumentError { command: String, error: ArgumentError, span: Span, }, Diagnostic(ShellDiagnostic), CoerceError { left: Tagged, right: Tagged, }, } impl ProximateShellError { fn start(self) -> ShellError { ShellError { cause: None, error: self, } } } #[derive(Debug, Clone, Serialize, Deserialize)] pub struct ShellDiagnostic { crate diagnostic: Diagnostic, } impl ShellDiagnostic { #[allow(unused)] crate fn simple_diagnostic( span: impl Into, source: impl Into, ) -> 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 } } } impl PartialEq for ShellDiagnostic { fn eq(&self, _other: &ShellDiagnostic) -> bool { false } } impl Eq for ShellDiagnostic {} impl std::cmp::PartialOrd for ShellDiagnostic { fn partial_cmp(&self, _other: &Self) -> Option { Some(std::cmp::Ordering::Less) } } impl std::cmp::Ord for ShellDiagnostic { fn cmp(&self, _other: &Self) -> std::cmp::Ordering { std::cmp::Ordering::Less } } #[derive(Debug, Ord, PartialOrd, Eq, PartialEq, new, Clone, Serialize, Deserialize)] pub struct StringError { title: String, error: Value, } impl std::fmt::Display for ShellError { fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { match &self.error { ProximateShellError::String(s) => write!(f, "{}", &s.title), ProximateShellError::MissingValue { .. } => write!(f, "MissingValue"), ProximateShellError::InvalidCommand { .. } => write!(f, "InvalidCommand"), ProximateShellError::TypeError { .. } => write!(f, "TypeError"), ProximateShellError::SyntaxError { .. } => write!(f, "SyntaxError"), ProximateShellError::MissingProperty { .. } => write!(f, "MissingProperty"), ProximateShellError::ArgumentError { .. } => write!(f, "ArgumentError"), ProximateShellError::Diagnostic(_) => write!(f, ""), ProximateShellError::CoerceError { .. } => write!(f, "CoerceError"), } } } impl std::error::Error for ShellError {} impl std::convert::From for ShellError { fn from(input: std::io::Error) -> ShellError { ProximateShellError::String(StringError { title: format!("{}", input), error: Value::nothing(), }) .start() } } impl std::convert::From for ShellError { fn from(input: subprocess::PopenError) -> ShellError { ProximateShellError::String(StringError { title: format!("{}", input), error: Value::nothing(), }) .start() } } impl std::convert::From for ShellError { fn from(input: toml::ser::Error) -> ShellError { ProximateShellError::String(StringError { title: format!("{:?}", input), error: Value::nothing(), }) .start() } } impl std::convert::From for ShellError { fn from(input: serde_json::Error) -> ShellError { ProximateShellError::String(StringError { title: format!("{:?}", input), error: Value::nothing(), }) .start() } } impl std::convert::From for ShellError { fn from(input: regex::Error) -> ShellError { ProximateShellError::String(StringError { title: format!("{:?}", input), error: Value::nothing(), }) .start() } } pub trait ShellErrorUtils { fn unwrap_error(self, desc: impl Into) -> Result; } impl ShellErrorUtils> for Option> { fn unwrap_error(self, desc: impl Into) -> Result, ShellError> { match self { Some(value) => Ok(value), None => Err(ShellError::missing_value(None, desc.into())), } } }