From fd1c6430a82e1108f4fac6803fd9e63179d0c4d2 Mon Sep 17 00:00:00 2001 From: Mike Gerwitz Date: Thu, 28 Apr 2022 22:13:51 -0400 Subject: [PATCH] tamer: diagnose::report::SectionSourceLine: {Option=>Column} If a column isn't present, it degrades to displaying labels like footnotes anyway, so this simplifies the system rather than catering to a rare case. With that said, this does lose functionality, since it does not render the source line at all, even though we _could_ do so. I may re-introduce that rendering after some further refactoring, specifically for gutters. DEV-12151 --- tamer/src/diagnose/report.rs | 67 ++++++++----------- tamer/src/diagnose/report/test/integration.rs | 11 --- 2 files changed, 29 insertions(+), 49 deletions(-) diff --git a/tamer/src/diagnose/report.rs b/tamer/src/diagnose/report.rs index c3f8871b..9551f80d 100644 --- a/tamer/src/diagnose/report.rs +++ b/tamer/src/diagnose/report.rs @@ -304,22 +304,23 @@ where let nlines = src.len(); - body.extend(src.into_iter().enumerate().map(|(i, srcline)| { - let col = srcline.column(); + body.extend(src.into_iter().enumerate().filter_map( + |(i, srcline)| { + let label = + if i == nlines - 1 { olabel.take() } else { None }; - SectionLine::SourceLine(SectionSourceLine { - src: srcline, - mark: LineMark { - col, - level, - label: if i == nlines - 1 { - olabel.take() - } else { - None - }, - }, - }) - })); + if let Some(col) = srcline.column() { + Some(SectionLine::SourceLine(SectionSourceLine { + src: srcline, + mark: LineMark { col, level, label }, + })) + } else { + label.map(|l| { + SectionLine::Footnote(SpanLabel(level, l)) + }) + } + }, + )); (span, level) } @@ -541,7 +542,7 @@ impl<'d> Display for SectionSourceLine<'d> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, " |\n")?; write!(f, " | {src}\n", src = self.src)?; - write!(f, "{}", self.mark) + write!(f, " |{}", self.mark) } } @@ -550,34 +551,24 @@ impl<'d> Display for SectionSourceLine<'d> { #[derive(Debug, PartialEq, Eq)] struct LineMark<'d> { level: Level, - col: Option, + col: Column, label: Option>, } impl<'d> Display for LineMark<'d> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { - write!(f, " |")?; + let Self { col, level, .. } = self; - if let Some(col) = self.col { - let underline = self - .level - .mark_char() - .to_string() - .repeat((col.end().get() - col.start().get()) as usize + 1); + let underline = level + .mark_char() + .to_string() + .repeat((col.end().get() - col.start().get()) as usize + 1); - let lpad = col.start().get() as usize - 1; + let lpad = col.start().get() as usize - 1; - write!(f, " {:lpad$}{underline}", "")?; - } else { - write!(f, "\n")?; - } + write!(f, " {:lpad$}{underline}", "")?; if let Some(label) = self.label.as_ref() { - if self.col.is_none() { - // Render as a footnote. - write!(f, " =")?; - } - // TODO: If the span is at the end of a long line, // this is more likely to wrap on the user's terminal and be // unpleasant to read. @@ -778,7 +769,7 @@ mod test { src: src_lines[0].clone(), mark: LineMark { level: Level::Note, - col: Some(col_1), + col: col_1, // Label goes on the last source line. label: None, } @@ -787,7 +778,7 @@ mod test { src: src_lines[1].clone(), mark: LineMark { level: Level::Note, - col: Some(col_2), + col: col_2, // Label at last source line label: Some("test label".into()), } @@ -902,7 +893,7 @@ mod test { ), mark: LineMark { level: Level::Help, - col: None, + col: Column::Before(1.unwrap_into()), label: Some("kept label".into()) } }) @@ -926,7 +917,7 @@ mod test { ), mark: LineMark { level: Level::Help, - col: None, + col: Column::Before(1.unwrap_into()), label: None, } }) diff --git a/tamer/src/diagnose/report/test/integration.rs b/tamer/src/diagnose/report/test/integration.rs index d0179c82..92bb46d1 100644 --- a/tamer/src/diagnose/report/test/integration.rs +++ b/tamer/src/diagnose/report/test/integration.rs @@ -436,23 +436,12 @@ fn fallback_when_column_fails_to_resolve() { let span = ctx.span(4, 2); - let lossy = String::from_utf8_lossy(FILE_INVALID_UTF8); - - // It's not ideal that the help appears first, - // but this should only happen under very exceptional - // circumstances so it's not worth trying to resolve. - // If you're reading this and it's trivial to swap these with the - // current state of the system, - // go for it. assert_report!( "column resolution failure", vec![span.error("an error we do not want to suppress"),], format!("\ error: column resolution failure --> invalid/utf8:1 bytes 4--6 - | - | {lossy} - | = error: an error we do not want to suppress = help: unable to calculate columns because the line is not a valid UTF-8 string = help: you have been provided with 0-indexed line-relative inclusive byte offsets