350 lines
9.6 KiB
Rust
350 lines
9.6 KiB
Rust
use std::collections::HashMap;
|
|
|
|
use rottlib::diagnostics::{Diagnostic, Severity};
|
|
use rottlib::lexer::{TokenPosition, TokenSpan, TokenizedFile};
|
|
|
|
#[derive(Debug)]
|
|
struct ExpectedLabel {
|
|
span: TokenSpan,
|
|
message: &'static str,
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
struct ExpectedDiagnostic<'a> {
|
|
headline: &'static str,
|
|
severity: Severity,
|
|
code: Option<&'static str>,
|
|
primary_label: Option<ExpectedLabel>,
|
|
secondary_labels: &'a [ExpectedLabel],
|
|
help: Option<&'static str>,
|
|
notes: &'a [&'static str],
|
|
}
|
|
|
|
#[track_caller]
|
|
fn assert_diagnostic(actual: &Diagnostic, expected: &ExpectedDiagnostic<'_>) {
|
|
assert_eq!(actual.headline(), expected.headline);
|
|
assert_eq!(actual.severity(), expected.severity);
|
|
assert_eq!(actual.code(), expected.code);
|
|
assert_eq!(actual.help(), expected.help);
|
|
|
|
match (actual.primary_label(), expected.primary_label.as_ref()) {
|
|
(None, None) => {}
|
|
(Some(actual), Some(expected)) => {
|
|
assert_eq!(actual.span, expected.span);
|
|
assert_eq!(actual.message, expected.message);
|
|
}
|
|
_ => panic!("primary label mismatch"),
|
|
}
|
|
|
|
let actual_secondary = actual.secondary_labels();
|
|
assert_eq!(actual_secondary.len(), expected.secondary_labels.len());
|
|
|
|
for (actual, expected) in actual_secondary
|
|
.iter()
|
|
.zip(expected.secondary_labels.iter())
|
|
{
|
|
assert_eq!(actual.span, expected.span);
|
|
assert_eq!(actual.message, expected.message);
|
|
}
|
|
|
|
let actual_notes = actual.notes();
|
|
assert_eq!(actual_notes.len(), expected.notes.len());
|
|
|
|
for (actual, expected) in actual_notes.iter().zip(expected.notes.iter()) {
|
|
assert_eq!(actual, expected);
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone, Copy)]
|
|
struct Fixture {
|
|
label: &'static str,
|
|
source: &'static str,
|
|
}
|
|
|
|
type FixtureRun = Vec<Diagnostic>;
|
|
|
|
struct FixtureRuns {
|
|
runs: HashMap<&'static str, FixtureRun>,
|
|
}
|
|
|
|
impl FixtureRuns {
|
|
#[track_caller]
|
|
fn get(&self, label: &str) -> Option<Vec<Diagnostic>> {
|
|
self.runs.get(label).cloned()
|
|
}
|
|
|
|
#[track_caller]
|
|
fn get_any(&self, label: &str) -> Diagnostic {
|
|
self.runs
|
|
.get(label)
|
|
.map(|fixture_run| fixture_run[0].clone())
|
|
.unwrap_or_else(|| panic!("no fixture run for `{label}`"))
|
|
}
|
|
|
|
#[track_caller]
|
|
fn get_by_code(&self, label: &str, code: &str) -> Diagnostic {
|
|
self.runs
|
|
.get(label)
|
|
.unwrap_or_else(|| panic!("no fixture run for `{label}`"))
|
|
.iter()
|
|
.find(|diagnostic| diagnostic.code() == Some(code))
|
|
.unwrap_or_else(|| panic!("no `{code}` diagnostic in fixture `{label}`"))
|
|
.clone()
|
|
}
|
|
}
|
|
|
|
const fn span(position: usize) -> TokenSpan {
|
|
TokenSpan {
|
|
start: TokenPosition(position),
|
|
end: TokenPosition(position),
|
|
}
|
|
}
|
|
|
|
const LEXER_FIXTURES: &[Fixture] = &[
|
|
Fixture {
|
|
label: "files/L0001_01.uc",
|
|
source: "`",
|
|
},
|
|
Fixture {
|
|
label: "files/L0002_01.uc",
|
|
source: "]",
|
|
},
|
|
Fixture {
|
|
label: "files/L0003_01.uc",
|
|
source: "{\n foo(\n}\n",
|
|
},
|
|
Fixture {
|
|
label: "files/L0004_01.uc",
|
|
source: "(]",
|
|
},
|
|
Fixture {
|
|
label: "files/L0005_01.uc",
|
|
source: "foo(",
|
|
},
|
|
Fixture {
|
|
label: "files/L_mixed_01.uc",
|
|
source: "([)]",
|
|
},
|
|
];
|
|
|
|
fn run_fixture(fixture: &'static Fixture) -> FixtureRun {
|
|
let file = TokenizedFile::tokenize(fixture.source);
|
|
file.diagnostics().to_vec()
|
|
}
|
|
|
|
fn run_fixtures(fixtures: &'static [Fixture]) -> FixtureRuns {
|
|
let mut runs = HashMap::new();
|
|
|
|
for fixture in fixtures {
|
|
runs.insert(fixture.label, run_fixture(fixture));
|
|
}
|
|
|
|
FixtureRuns { runs }
|
|
}
|
|
|
|
#[test]
|
|
fn check_lexer_diagnostic_counts() {
|
|
let runs = run_fixtures(LEXER_FIXTURES);
|
|
|
|
assert_eq!(runs.get("files/L0001_01.uc").unwrap().len(), 1);
|
|
assert_eq!(runs.get("files/L0002_01.uc").unwrap().len(), 1);
|
|
assert_eq!(runs.get("files/L0003_01.uc").unwrap().len(), 1);
|
|
assert_eq!(runs.get("files/L0004_01.uc").unwrap().len(), 1);
|
|
assert_eq!(runs.get("files/L0005_01.uc").unwrap().len(), 1);
|
|
assert_eq!(runs.get("files/L_mixed_01.uc").unwrap().len(), 2);
|
|
}
|
|
|
|
#[test]
|
|
fn check_l0001_invalid_token() {
|
|
let runs = run_fixtures(LEXER_FIXTURES);
|
|
|
|
assert_diagnostic(
|
|
&runs.get_any("files/L0001_01.uc"),
|
|
&ExpectedDiagnostic {
|
|
headline: "invalid token: backtick",
|
|
severity: Severity::Error,
|
|
code: Some("L0001"),
|
|
primary_label: Some(ExpectedLabel {
|
|
span: span(0),
|
|
message: "invalid token: backtick",
|
|
}),
|
|
secondary_labels: &[],
|
|
help: None,
|
|
notes: &[],
|
|
},
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn check_l0002_unexpected_closing_delimiter() {
|
|
let runs = run_fixtures(LEXER_FIXTURES);
|
|
|
|
assert_diagnostic(
|
|
&runs.get_any("files/L0002_01.uc"),
|
|
&ExpectedDiagnostic {
|
|
headline: "unexpected closing delimiter: `]`",
|
|
severity: Severity::Error,
|
|
code: Some("L0002"),
|
|
primary_label: Some(ExpectedLabel {
|
|
span: span(0),
|
|
message: "unexpected closing delimiter",
|
|
}),
|
|
secondary_labels: &[],
|
|
help: None,
|
|
notes: &[],
|
|
},
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn check_l0003_unclosed_delimiter_before_later_close() {
|
|
let runs = run_fixtures(LEXER_FIXTURES);
|
|
|
|
assert_diagnostic(
|
|
&runs.get_any("files/L0003_01.uc"),
|
|
&ExpectedDiagnostic {
|
|
headline: "unclosed delimiter before `}`",
|
|
severity: Severity::Error,
|
|
code: Some("L0003"),
|
|
primary_label: Some(ExpectedLabel {
|
|
span: span(4),
|
|
message: "this `(` is not closed before `}`",
|
|
}),
|
|
secondary_labels: &[
|
|
ExpectedLabel {
|
|
span: span(6),
|
|
message: "this `}` is matched with the earlier `{`",
|
|
},
|
|
ExpectedLabel {
|
|
span: span(0),
|
|
message: "this `{` is likely the intended match",
|
|
},
|
|
],
|
|
help: None,
|
|
notes: &[],
|
|
},
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn check_l0004_mismatched_closing_delimiter() {
|
|
let runs = run_fixtures(LEXER_FIXTURES);
|
|
|
|
assert_diagnostic(
|
|
&runs.get_any("files/L0004_01.uc"),
|
|
&ExpectedDiagnostic {
|
|
headline: "mismatched closing delimiter: `]`",
|
|
severity: Severity::Error,
|
|
code: Some("L0004"),
|
|
primary_label: Some(ExpectedLabel {
|
|
span: span(1),
|
|
message: "closing delimiter does not match `(`",
|
|
}),
|
|
secondary_labels: &[ExpectedLabel {
|
|
span: span(0),
|
|
message: "`(` opened here",
|
|
}],
|
|
help: None,
|
|
notes: &[],
|
|
},
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn check_l0005_unclosed_delimiter_at_eof() {
|
|
let runs = run_fixtures(LEXER_FIXTURES);
|
|
|
|
assert_diagnostic(
|
|
&runs.get_any("files/L0005_01.uc"),
|
|
&ExpectedDiagnostic {
|
|
headline: "unclosed delimiter: `(`",
|
|
severity: Severity::Error,
|
|
code: Some("L0005"),
|
|
primary_label: Some(ExpectedLabel {
|
|
span: span(1),
|
|
message: "this `(` was never closed",
|
|
}),
|
|
secondary_labels: &[],
|
|
help: None,
|
|
notes: &[],
|
|
},
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn check_mixed_recovery_diagnostics() {
|
|
let runs = run_fixtures(LEXER_FIXTURES);
|
|
|
|
assert_diagnostic(
|
|
&runs.get_by_code("files/L_mixed_01.uc", "L0003"),
|
|
&ExpectedDiagnostic {
|
|
headline: "unclosed delimiter before `)`",
|
|
severity: Severity::Error,
|
|
code: Some("L0003"),
|
|
primary_label: Some(ExpectedLabel {
|
|
span: span(1),
|
|
message: "this `[` is not closed before `)`",
|
|
}),
|
|
secondary_labels: &[
|
|
ExpectedLabel {
|
|
span: span(2),
|
|
message: "this `)` is matched with the earlier `(`",
|
|
},
|
|
ExpectedLabel {
|
|
span: span(0),
|
|
message: "this `(` is likely the intended match",
|
|
},
|
|
],
|
|
help: None,
|
|
notes: &[],
|
|
},
|
|
);
|
|
|
|
assert_diagnostic(
|
|
&runs.get_by_code("files/L_mixed_01.uc", "L0002"),
|
|
&ExpectedDiagnostic {
|
|
headline: "unexpected closing delimiter: `]`",
|
|
severity: Severity::Error,
|
|
code: Some("L0002"),
|
|
primary_label: Some(ExpectedLabel {
|
|
span: span(3),
|
|
message: "unexpected closing delimiter",
|
|
}),
|
|
secondary_labels: &[],
|
|
help: None,
|
|
notes: &[],
|
|
},
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn check_recovered_delimiter_matches_are_stored() {
|
|
let file = TokenizedFile::tokenize("{\n foo(\n}\n");
|
|
|
|
assert_eq!(
|
|
file.matching_delimiter(TokenPosition(0)),
|
|
Some(TokenPosition(6))
|
|
);
|
|
assert_eq!(
|
|
file.matching_delimiter(TokenPosition(6)),
|
|
Some(TokenPosition(0))
|
|
);
|
|
assert_eq!(file.matching_delimiter(TokenPosition(4)), None);
|
|
}
|
|
|
|
#[test]
|
|
fn check_mixed_recovery_delimiter_matches_are_stored() {
|
|
let file = TokenizedFile::tokenize("([)]");
|
|
|
|
assert_eq!(
|
|
file.matching_delimiter(TokenPosition(0)),
|
|
Some(TokenPosition(2))
|
|
);
|
|
assert_eq!(
|
|
file.matching_delimiter(TokenPosition(2)),
|
|
Some(TokenPosition(0))
|
|
);
|
|
|
|
assert_eq!(file.matching_delimiter(TokenPosition(1)), None);
|
|
assert_eq!(file.matching_delimiter(TokenPosition(3)), None);
|
|
} |