mod service;
use std::{
borrow::Cow,
fmt::{self, Display},
ops::{Deref, DerefMut},
};
pub mod reporter;
pub use crate::service::{DiagnosticSender, DiagnosticService};
pub type Error = miette::Error;
pub type Severity = miette::Severity;
pub type Result<T> = std::result::Result<T, OxcDiagnostic>;
use miette::{Diagnostic, SourceCode};
pub use miette::{GraphicalReportHandler, GraphicalTheme, LabeledSpan, Labels, NamedSource};
#[derive(Debug, Default, Clone, PartialEq, Eq)]
pub struct Diagnostics(Vec<OxcDiagnostic>);
impl Diagnostics {
pub fn new() -> Self {
Self(Vec::new())
}
pub fn has_errors(&self) -> bool {
self.0.iter().any(|diagnostic| diagnostic.severity == Severity::Error)
}
pub fn has_warnings(&self) -> bool {
self.0.iter().any(|diagnostic| diagnostic.severity == Severity::Warning)
}
pub fn errors(&self) -> impl Iterator<Item = &OxcDiagnostic> {
self.0.iter().filter(|diagnostic| diagnostic.severity == Severity::Error)
}
pub fn warnings(&self) -> impl Iterator<Item = &OxcDiagnostic> {
self.0.iter().filter(|diagnostic| diagnostic.severity == Severity::Warning)
}
pub fn into_vec(self) -> Vec<OxcDiagnostic> {
self.0
}
pub fn push(&mut self, diagnostic: OxcDiagnostic) {
self.0.push(diagnostic);
}
pub fn extend(&mut self, diagnostics: impl IntoIterator<Item = OxcDiagnostic>) {
self.0.extend(diagnostics);
}
}
impl Deref for Diagnostics {
type Target = Vec<OxcDiagnostic>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl DerefMut for Diagnostics {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.0
}
}
impl From<Vec<OxcDiagnostic>> for Diagnostics {
fn from(diagnostics: Vec<OxcDiagnostic>) -> Self {
Self(diagnostics)
}
}
impl From<OxcDiagnostic> for Diagnostics {
fn from(diagnostic: OxcDiagnostic) -> Self {
Self(vec![diagnostic])
}
}
impl From<Diagnostics> for Vec<OxcDiagnostic> {
fn from(diagnostics: Diagnostics) -> Self {
diagnostics.0
}
}
impl FromIterator<OxcDiagnostic> for Diagnostics {
fn from_iter<T: IntoIterator<Item = OxcDiagnostic>>(iter: T) -> Self {
Self(iter.into_iter().collect())
}
}
impl IntoIterator for Diagnostics {
type Item = OxcDiagnostic;
type IntoIter = std::vec::IntoIter<OxcDiagnostic>;
fn into_iter(self) -> Self::IntoIter {
self.0.into_iter()
}
}
impl<'a> IntoIterator for &'a Diagnostics {
type Item = &'a OxcDiagnostic;
type IntoIter = std::slice::Iter<'a, OxcDiagnostic>;
fn into_iter(self) -> Self::IntoIter {
self.0.iter()
}
}
#[derive(Debug, Clone, Eq, PartialEq)]
#[must_use]
pub struct OxcDiagnostic {
inner: OxcDiagnosticInner,
}
impl Deref for OxcDiagnostic {
type Target = OxcDiagnosticInner;
fn deref(&self) -> &Self::Target {
&self.inner
}
}
impl DerefMut for OxcDiagnostic {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.inner
}
}
#[derive(Debug, Default, Clone, Eq, PartialEq, PartialOrd, Ord)]
pub struct OxcCode {
pub scope: Option<Cow<'static, str>>,
pub number: Option<Cow<'static, str>>,
}
impl OxcCode {
pub fn is_some(&self) -> bool {
self.scope.is_some() || self.number.is_some()
}
}
impl Display for OxcCode {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match (&self.scope, &self.number) {
(Some(scope), Some(number)) => write!(f, "{scope}({number})"),
(Some(scope), None) => scope.fmt(f),
(None, Some(number)) => number.fmt(f),
(None, None) => Ok(()),
}
}
}
#[derive(Debug, Clone, Eq, PartialEq)]
pub struct OxcDiagnosticInner {
pub message: Cow<'static, str>,
pub labels: Labels,
pub help: Option<Cow<'static, str>>,
pub note: Option<Cow<'static, str>>,
pub severity: Severity,
pub code: OxcCode,
pub url: Option<Cow<'static, str>>,
}
impl Display for OxcDiagnostic {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> std::fmt::Result {
self.message.fmt(f)
}
}
impl std::error::Error for OxcDiagnostic {}
impl Diagnostic for OxcDiagnostic {
fn help(&self) -> Option<Cow<'_, str>> {
self.help.as_deref().map(Cow::Borrowed)
}
fn note(&self) -> Option<Cow<'_, str>> {
self.note.as_deref().map(Cow::Borrowed)
}
fn severity(&self) -> Option<Severity> {
Some(self.severity)
}
fn labels(&self) -> Labels {
self.labels.clone()
}
fn code(&self) -> Option<Cow<'_, str>> {
self.code.is_some().then(|| Cow::Owned(self.code.to_string()))
}
fn url(&self) -> Option<Cow<'_, str>> {
self.url.as_deref().map(Cow::Borrowed)
}
}
impl OxcDiagnostic {
pub fn error<T: Into<Cow<'static, str>>>(message: T) -> Self {
Self {
inner: OxcDiagnosticInner {
message: message.into(),
labels: Labels::None,
note: None,
help: None,
severity: Severity::Error,
code: OxcCode::default(),
url: None,
},
}
}
pub fn warn<T: Into<Cow<'static, str>>>(message: T) -> Self {
Self {
inner: OxcDiagnosticInner {
message: message.into(),
labels: Labels::None,
help: None,
note: None,
severity: Severity::Warning,
code: OxcCode::default(),
url: None,
},
}
}
#[inline]
pub fn with_error_code<T: Into<Cow<'static, str>>, U: Into<Cow<'static, str>>>(
self,
scope: T,
number: U,
) -> Self {
self.with_error_code_scope(scope).with_error_code_num(number)
}
#[inline]
pub fn with_error_code_scope<T: Into<Cow<'static, str>>>(mut self, code_scope: T) -> Self {
self.inner.code.scope = match self.inner.code.scope {
Some(scope) => Some(scope),
None => Some(code_scope.into()),
};
debug_assert!(
self.inner.code.scope.as_ref().is_some_and(|s| !s.is_empty()),
"Error code scopes cannot be empty"
);
self
}
#[inline]
pub fn with_error_code_num<T: Into<Cow<'static, str>>>(mut self, code_num: T) -> Self {
self.inner.code.number = match self.inner.code.number {
Some(num) => Some(num),
None => Some(code_num.into()),
};
debug_assert!(
self.inner.code.number.as_ref().is_some_and(|n| !n.is_empty()),
"Error code numbers cannot be empty"
);
self
}
pub fn with_severity(mut self, severity: Severity) -> Self {
self.inner.severity = severity;
self
}
pub fn with_help<T: Into<Cow<'static, str>>>(mut self, help: T) -> Self {
self.inner.help = Some(help.into());
self
}
pub fn with_note<T: Into<Cow<'static, str>>>(mut self, note: T) -> Self {
self.inner.note = Some(note.into());
self
}
pub fn with_label<T: Into<LabeledSpan>>(mut self, label: T) -> Self {
self.inner.labels = Labels::One([label.into()]);
self
}
pub fn with_labels<L: Into<LabeledSpan>, T: IntoIterator<Item = L>>(
mut self,
labels: T,
) -> Self {
self.inner.labels = labels.into_iter().map(Into::into).collect();
self
}
pub fn and_label<T: Into<LabeledSpan>>(mut self, label: T) -> Self {
self.inner.labels.push(label.into());
self
}
pub fn and_labels<L: Into<LabeledSpan>, T: IntoIterator<Item = L>>(
mut self,
labels: T,
) -> Self {
self.inner.labels.extend(labels.into_iter().map(Into::into));
self
}
pub fn with_url<S: Into<Cow<'static, str>>>(mut self, url: S) -> Self {
self.inner.url = Some(url.into());
self
}
pub fn with_source_code<T: SourceCode + Send + Sync + 'static>(self, code: T) -> Error {
Error::from(self).with_source_code(code)
}
pub fn inner_owned(self) -> OxcDiagnosticInner {
self.inner
}
}