use crate::pretty::{b, DebugDocBuilder, PrettyDebugWithSource}; use crate::text::Text; use crate::tracable::TracableContext; use derive_new::new; use getset::Getters; use serde::Deserialize; use serde::Serialize; use std::path::{Path, PathBuf}; #[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq, PartialOrd, Ord, Hash)] pub enum AnchorLocation { Url(String), File(String), Source(Text), } pub trait HasTag { fn tag(&self) -> Tag; } #[derive(new, Debug, Clone, Copy, Eq, PartialEq, Ord, PartialOrd, Serialize, Deserialize, Hash)] pub struct Spanned { pub span: Span, pub item: T, } impl Spanned { pub fn map(self, input: impl FnOnce(T) -> U) -> Spanned { let span = self.span; let mapped = input(self.item); mapped.spanned(span) } } impl Spanned { pub fn items<'a, U>( items: impl Iterator>, ) -> impl Iterator { items.map(|item| &item.item[..]) } } impl Spanned { pub fn borrow_spanned(&self) -> Spanned<&str> { let span = self.span; self.item[..].spanned(span) } } pub trait SpannedItem: Sized { fn spanned(self, span: impl Into) -> Spanned { Spanned { item: self, span: span.into(), } } fn spanned_unknown(self) -> Spanned { Spanned { item: self, span: Span::unknown(), } } } impl SpannedItem for T {} impl std::ops::Deref for Spanned { type Target = T; fn deref(&self) -> &T { &self.item } } #[derive(new, Debug, Clone, Eq, PartialEq, Ord, PartialOrd, Serialize, Deserialize, Hash)] pub struct Tagged { pub tag: Tag, pub item: T, } impl Tagged { pub fn borrow_spanned(&self) -> Spanned<&str> { let span = self.tag.span; self.item[..].spanned(span) } pub fn borrow_tagged(&self) -> Tagged<&str> { self.item[..].tagged(self.tag.clone()) } } impl Tagged> { pub fn items(&self) -> impl Iterator { self.item.iter() } } impl HasTag for Tagged { fn tag(&self) -> Tag { self.tag.clone() } } impl AsRef for Tagged { fn as_ref(&self) -> &Path { self.item.as_ref() } } pub trait TaggedItem: Sized { fn tagged(self, tag: impl Into) -> Tagged { Tagged { item: self, tag: tag.into(), } } // For now, this is a temporary facility. In many cases, there are other useful spans that we // could be using, such as the original source spans of JSON or Toml files, but we don't yet // have the infrastructure to make that work. fn tagged_unknown(self) -> Tagged { Tagged { item: self, tag: Tag { span: Span::unknown(), anchor: None, }, } } } impl TaggedItem for T {} impl std::ops::Deref for Tagged { type Target = T; fn deref(&self) -> &T { &self.item } } impl Tagged { pub fn map(self, input: impl FnOnce(T) -> U) -> Tagged { let tag = self.tag(); let mapped = input(self.item); mapped.tagged(tag) } pub fn map_anchored(self, anchor: &Option) -> Tagged { let mut tag = self.tag; tag.anchor = anchor.clone(); Tagged { item: self.item, tag, } } pub fn transpose(&self) -> Tagged<&T> { Tagged { item: &self.item, tag: self.tag.clone(), } } pub fn tag(&self) -> Tag { self.tag.clone() } pub fn span(&self) -> Span { self.tag.span } pub fn anchor(&self) -> Option { self.tag.anchor.clone() } pub fn anchor_name(&self) -> Option { match self.tag.anchor { Some(AnchorLocation::File(ref file)) => Some(file.clone()), Some(AnchorLocation::Url(ref url)) => Some(url.clone()), _ => None, } } pub fn item(&self) -> &T { &self.item } pub fn into_parts(self) -> (T, Tag) { (self.item, self.tag) } } impl From<&Tag> for Tag { fn from(input: &Tag) -> Tag { input.clone() } } impl From> for Span { fn from(input: nom_locate::LocatedSpanEx<&str, T>) -> Span { Span::new(input.offset, input.offset + input.fragment.len()) } } impl From<( nom_locate::LocatedSpanEx, nom_locate::LocatedSpanEx, )> for Span { fn from( input: ( nom_locate::LocatedSpanEx, nom_locate::LocatedSpanEx, ), ) -> Span { Span::new(input.0.offset, input.1.offset) } } impl From<(usize, usize)> for Span { fn from(input: (usize, usize)) -> Span { Span::new(input.0, input.1) } } impl From<&std::ops::Range> for Span { fn from(input: &std::ops::Range) -> Span { Span::new(input.start, input.end) } } #[derive( Debug, Clone, PartialEq, Eq, Ord, PartialOrd, Serialize, Deserialize, Hash, Getters, new, )] pub struct Tag { pub anchor: Option, pub span: Span, } impl From for Tag { fn from(span: Span) -> Self { Tag { anchor: None, span } } } impl From<&Span> for Tag { fn from(span: &Span) -> Self { Tag { anchor: None, span: *span, } } } impl From<(usize, usize, TracableContext)> for Tag { fn from((start, end, _context): (usize, usize, TracableContext)) -> Self { Tag { anchor: None, span: Span::new(start, end), } } } impl From<(usize, usize, AnchorLocation)> for Tag { fn from((start, end, anchor): (usize, usize, AnchorLocation)) -> Self { Tag { anchor: Some(anchor), span: Span::new(start, end), } } } impl From<(usize, usize, Option)> for Tag { fn from((start, end, anchor): (usize, usize, Option)) -> Self { Tag { anchor, span: Span::new(start, end), } } } impl From> for Tag { fn from(input: nom_locate::LocatedSpanEx<&str, TracableContext>) -> Tag { Tag { anchor: None, span: Span::new(input.offset, input.offset + input.fragment.len()), } } } impl From for Span { fn from(tag: Tag) -> Self { tag.span } } impl From<&Tag> for Span { fn from(tag: &Tag) -> Self { tag.span } } impl Tag { pub fn unknown_anchor(span: Span) -> Tag { Tag { anchor: None, span } } pub fn for_char(pos: usize, anchor: AnchorLocation) -> Tag { Tag { anchor: Some(anchor), span: Span::new(pos, pos + 1), } } pub fn unknown_span(anchor: AnchorLocation) -> Tag { Tag { anchor: Some(anchor), span: Span::unknown(), } } pub fn unknown() -> Tag { Tag { anchor: None, span: Span::unknown(), } } pub fn anchor(&self) -> Option { self.anchor.clone() } pub fn until(&self, other: impl Into) -> Tag { let other = other.into(); debug_assert!( self.anchor == other.anchor, "Can only merge two tags with the same anchor" ); Tag { span: Span::new(self.span.start, other.span.end), anchor: self.anchor.clone(), } } pub fn until_option(&self, other: Option>) -> Tag { match other { Some(other) => { let other = other.into(); debug_assert!( self.anchor == other.anchor, "Can only merge two tags with the same anchor" ); Tag { span: Span::new(self.span.start, other.span.end), anchor: self.anchor.clone(), } } None => self.clone(), } } pub fn slice<'a>(&self, source: &'a str) -> &'a str { self.span.slice(source) } pub fn string<'a>(&self, source: &'a str) -> String { self.span.slice(source).to_string() } pub fn tagged_slice<'a>(&self, source: &'a str) -> Tagged<&'a str> { self.span.slice(source).tagged(self) } pub fn tagged_string<'a>(&self, source: &'a str) -> Tagged { self.span.slice(source).to_string().tagged(self) } pub fn anchor_name(&self) -> Option { match self.anchor { Some(AnchorLocation::File(ref file)) => Some(file.clone()), Some(AnchorLocation::Url(ref url)) => Some(url.clone()), _ => None, } } } #[allow(unused)] pub fn tag_for_tagged_list(mut iter: impl Iterator) -> Tag { let first = iter.next(); let first = match first { None => return Tag::unknown(), Some(first) => first, }; let last = iter.last(); match last { None => first, Some(last) => first.until(last), } } #[allow(unused)] pub fn span_for_spanned_list(mut iter: impl Iterator) -> Span { let first = iter.next(); let first = match first { None => return Span::unknown(), Some(first) => first, }; let last = iter.last(); match last { None => first, Some(last) => first.until(last), } } #[derive(Debug, Clone, Copy, PartialEq, Eq, Ord, PartialOrd, Serialize, Deserialize, Hash)] pub struct Span { start: usize, end: usize, } impl From<&Span> for Span { fn from(span: &Span) -> Span { *span } } impl From> for Span { fn from(input: Option) -> Span { match input { None => Span::new(0, 0), Some(span) => span, } } } impl Span { pub fn unknown() -> Span { Span::new(0, 0) } pub fn new(start: usize, end: usize) -> Span { assert!( end >= start, "Can't create a Span whose end < start, start={}, end={}", start, end ); Span { start, end } } pub fn for_char(pos: usize) -> Span { Span { start: pos, end: pos + 1, } } pub fn until(&self, other: impl Into) -> Span { let other = other.into(); Span::new(self.start, other.end) } pub fn until_option(&self, other: Option>) -> Span { match other { Some(other) => { let other = other.into(); Span::new(self.start, other.end) } None => *self, } } pub fn string<'a>(&self, source: &'a str) -> String { self.slice(source).to_string() } pub fn spanned_slice<'a>(&self, source: &'a str) -> Spanned<&'a str> { self.slice(source).spanned(*self) } pub fn spanned_string<'a>(&self, source: &'a str) -> Spanned { self.slice(source).to_string().spanned(*self) } pub fn start(&self) -> usize { self.start } pub fn end(&self) -> usize { self.end } pub fn is_unknown(&self) -> bool { self.start == 0 && self.end == 0 } pub fn slice<'a>(&self, source: &'a str) -> &'a str { &source[self.start..self.end] } } impl language_reporting::ReportingSpan for Span { fn with_start(&self, start: usize) -> Self { if self.end < start { Span::new(start, start) } else { Span::new(start, self.end) } } fn with_end(&self, end: usize) -> Self { if end < self.start { Span::new(end, end) } else { Span::new(self.start, end) } } fn start(&self) -> usize { self.start } fn end(&self) -> usize { self.end } } pub trait HasSpan: PrettyDebugWithSource { fn span(&self) -> Span; } pub trait HasFallibleSpan: PrettyDebugWithSource { fn maybe_span(&self) -> Option; } impl HasFallibleSpan for T { fn maybe_span(&self) -> Option { Some(HasSpan::span(self)) } } impl HasSpan for Spanned where Spanned: PrettyDebugWithSource, { fn span(&self) -> Span { self.span } } impl PrettyDebugWithSource for Option { fn pretty_debug(&self, source: &str) -> DebugDocBuilder { match self { None => b::description("no span"), Some(span) => span.pretty_debug(source), } } } impl HasFallibleSpan for Option { fn maybe_span(&self) -> Option { *self } } impl PrettyDebugWithSource for Span { fn pretty_debug(&self, source: &str) -> DebugDocBuilder { b::typed( "spanned", b::keyword("for") + b::space() + b::description(format!("{:?}", source)), ) } } impl HasSpan for Span { fn span(&self) -> Span { *self } } impl PrettyDebugWithSource for Option> where Spanned: PrettyDebugWithSource, { fn pretty_debug(&self, source: &str) -> DebugDocBuilder { match self { None => b::description("nothing"), Some(v) => v.pretty_debug(source), } } } impl HasFallibleSpan for Option> where Spanned: PrettyDebugWithSource, { fn maybe_span(&self) -> Option { match self { None => None, Some(value) => Some(value.span), } } } impl PrettyDebugWithSource for Option> where Tagged: PrettyDebugWithSource, { fn pretty_debug(&self, source: &str) -> DebugDocBuilder { match self { None => b::description("nothing"), Some(d) => d.pretty_debug(source), } } } impl HasFallibleSpan for Option> where Tagged: PrettyDebugWithSource, { fn maybe_span(&self) -> Option { match self { None => None, Some(value) => Some(value.tag.span), } } } impl HasSpan for Tagged where Tagged: PrettyDebugWithSource, { fn span(&self) -> Span { self.tag.span } }