2021-12-09 15:38:46 +00:00
|
|
|
use crate::Diagnostic;
|
|
|
|
use std::{
|
|
|
|
cmp::{Ordering, Reverse},
|
|
|
|
iter,
|
|
|
|
ops::Range,
|
|
|
|
};
|
|
|
|
use sum_tree::{self, Bias, SumTree};
|
2021-12-09 16:53:08 +00:00
|
|
|
use text::{Anchor, FromAnchor, PointUtf16, ToOffset};
|
2021-12-09 15:38:46 +00:00
|
|
|
|
|
|
|
#[derive(Clone, Default)]
|
|
|
|
pub struct DiagnosticSet {
|
2021-12-09 16:53:08 +00:00
|
|
|
diagnostics: SumTree<DiagnosticEntry<Anchor>>,
|
2021-12-09 15:38:46 +00:00
|
|
|
}
|
|
|
|
|
2021-12-09 16:53:08 +00:00
|
|
|
#[derive(Clone, Debug, PartialEq, Eq)]
|
|
|
|
pub struct DiagnosticEntry<T> {
|
|
|
|
pub range: Range<T>,
|
2021-12-09 15:38:46 +00:00
|
|
|
pub diagnostic: Diagnostic,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub struct Summary {
|
|
|
|
start: Anchor,
|
|
|
|
end: Anchor,
|
|
|
|
min_start: Anchor,
|
|
|
|
max_end: Anchor,
|
|
|
|
count: usize,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DiagnosticSet {
|
2021-12-10 14:58:37 +00:00
|
|
|
pub fn from_sorted_entries<I>(iter: I, buffer: &text::BufferSnapshot) -> Self
|
2021-12-09 15:38:46 +00:00
|
|
|
where
|
2021-12-09 16:53:08 +00:00
|
|
|
I: IntoIterator<Item = DiagnosticEntry<Anchor>>,
|
2021-12-09 15:38:46 +00:00
|
|
|
{
|
|
|
|
Self {
|
|
|
|
diagnostics: SumTree::from_iter(iter, buffer),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-10 14:58:37 +00:00
|
|
|
pub fn new<I>(iter: I, buffer: &text::BufferSnapshot) -> Self
|
2021-12-09 15:38:46 +00:00
|
|
|
where
|
2021-12-09 16:53:08 +00:00
|
|
|
I: IntoIterator<Item = DiagnosticEntry<PointUtf16>>,
|
2021-12-09 15:38:46 +00:00
|
|
|
{
|
|
|
|
let mut entries = iter.into_iter().collect::<Vec<_>>();
|
2021-12-09 16:53:08 +00:00
|
|
|
entries.sort_unstable_by_key(|entry| (entry.range.start, Reverse(entry.range.end)));
|
2021-12-10 08:43:21 +00:00
|
|
|
Self {
|
|
|
|
diagnostics: SumTree::from_iter(
|
|
|
|
entries.into_iter().map(|entry| DiagnosticEntry {
|
|
|
|
range: buffer.anchor_before(entry.range.start)
|
|
|
|
..buffer.anchor_after(entry.range.end),
|
|
|
|
diagnostic: entry.diagnostic,
|
|
|
|
}),
|
|
|
|
buffer,
|
|
|
|
),
|
|
|
|
}
|
2021-12-09 15:38:46 +00:00
|
|
|
}
|
|
|
|
|
2021-12-09 16:53:08 +00:00
|
|
|
pub fn iter(&self) -> impl Iterator<Item = &DiagnosticEntry<Anchor>> {
|
2021-12-09 15:38:46 +00:00
|
|
|
self.diagnostics.iter()
|
|
|
|
}
|
|
|
|
|
2021-12-09 16:53:08 +00:00
|
|
|
pub fn range<'a, T, O>(
|
2021-12-09 15:38:46 +00:00
|
|
|
&'a self,
|
|
|
|
range: Range<T>,
|
2021-12-10 14:58:37 +00:00
|
|
|
buffer: &'a text::BufferSnapshot,
|
2021-12-09 15:38:46 +00:00
|
|
|
inclusive: bool,
|
2021-12-09 16:53:08 +00:00
|
|
|
) -> impl 'a + Iterator<Item = DiagnosticEntry<O>>
|
2021-12-09 15:38:46 +00:00
|
|
|
where
|
|
|
|
T: 'a + ToOffset,
|
2021-12-09 16:53:08 +00:00
|
|
|
O: FromAnchor,
|
2021-12-09 15:38:46 +00:00
|
|
|
{
|
|
|
|
let end_bias = if inclusive { Bias::Right } else { Bias::Left };
|
|
|
|
let range = buffer.anchor_before(range.start)..buffer.anchor_at(range.end, end_bias);
|
|
|
|
let mut cursor = self.diagnostics.filter::<_, ()>(
|
|
|
|
{
|
|
|
|
move |summary: &Summary| {
|
|
|
|
let start_cmp = range.start.cmp(&summary.max_end, buffer).unwrap();
|
|
|
|
let end_cmp = range.end.cmp(&summary.min_start, buffer).unwrap();
|
|
|
|
if inclusive {
|
|
|
|
start_cmp <= Ordering::Equal && end_cmp >= Ordering::Equal
|
|
|
|
} else {
|
|
|
|
start_cmp == Ordering::Less && end_cmp == Ordering::Greater
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
buffer,
|
|
|
|
);
|
|
|
|
|
|
|
|
iter::from_fn({
|
|
|
|
move || {
|
|
|
|
if let Some(diagnostic) = cursor.item() {
|
|
|
|
cursor.next(buffer);
|
2021-12-09 16:53:08 +00:00
|
|
|
Some(diagnostic.resolve(buffer))
|
2021-12-09 15:38:46 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-12-09 16:53:08 +00:00
|
|
|
pub fn group<'a, O: FromAnchor>(
|
|
|
|
&'a self,
|
|
|
|
group_id: usize,
|
2021-12-10 14:58:37 +00:00
|
|
|
buffer: &'a text::BufferSnapshot,
|
2021-12-09 16:53:08 +00:00
|
|
|
) -> impl 'a + Iterator<Item = DiagnosticEntry<O>> {
|
2021-12-09 15:38:46 +00:00
|
|
|
self.iter()
|
|
|
|
.filter(move |entry| entry.diagnostic.group_id == group_id)
|
2021-12-09 16:53:08 +00:00
|
|
|
.map(|entry| entry.resolve(buffer))
|
2021-12-09 15:38:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-09 16:53:08 +00:00
|
|
|
impl sum_tree::Item for DiagnosticEntry<Anchor> {
|
2021-12-09 15:38:46 +00:00
|
|
|
type Summary = Summary;
|
|
|
|
|
|
|
|
fn summary(&self) -> Self::Summary {
|
|
|
|
Summary {
|
|
|
|
start: self.range.start.clone(),
|
|
|
|
end: self.range.end.clone(),
|
|
|
|
min_start: self.range.start.clone(),
|
|
|
|
max_end: self.range.end.clone(),
|
|
|
|
count: 1,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-09 16:53:08 +00:00
|
|
|
impl DiagnosticEntry<Anchor> {
|
2021-12-10 14:58:37 +00:00
|
|
|
pub fn resolve<O: FromAnchor>(&self, buffer: &text::BufferSnapshot) -> DiagnosticEntry<O> {
|
2021-12-09 16:53:08 +00:00
|
|
|
DiagnosticEntry {
|
|
|
|
range: O::from_anchor(&self.range.start, buffer)
|
|
|
|
..O::from_anchor(&self.range.end, buffer),
|
|
|
|
diagnostic: self.diagnostic.clone(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-09 15:38:46 +00:00
|
|
|
impl Default for Summary {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
start: Anchor::min(),
|
|
|
|
end: Anchor::max(),
|
|
|
|
min_start: Anchor::max(),
|
|
|
|
max_end: Anchor::min(),
|
|
|
|
count: 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl sum_tree::Summary for Summary {
|
2021-12-10 14:58:37 +00:00
|
|
|
type Context = text::BufferSnapshot;
|
2021-12-09 15:38:46 +00:00
|
|
|
|
|
|
|
fn add_summary(&mut self, other: &Self, buffer: &Self::Context) {
|
|
|
|
if other
|
|
|
|
.min_start
|
|
|
|
.cmp(&self.min_start, buffer)
|
|
|
|
.unwrap()
|
|
|
|
.is_lt()
|
|
|
|
{
|
|
|
|
self.min_start = other.min_start.clone();
|
|
|
|
}
|
|
|
|
if other.max_end.cmp(&self.max_end, buffer).unwrap().is_gt() {
|
|
|
|
self.max_end = other.max_end.clone();
|
|
|
|
}
|
|
|
|
self.start = other.start.clone();
|
|
|
|
self.end = other.end.clone();
|
|
|
|
self.count += other.count;
|
|
|
|
}
|
|
|
|
}
|