2021-05-22 16:53:47 +03:00
|
|
|
//! Some infrastructure for fuzzy testing.
|
|
|
|
//!
|
|
|
|
//! We don't normally run fuzzying, so this is hopelessly bitrotten :(
|
2019-09-30 11:58:53 +03:00
|
|
|
|
2020-04-25 00:57:47 +02:00
|
|
|
use std::{
|
|
|
|
convert::TryInto,
|
|
|
|
str::{self, FromStr},
|
|
|
|
};
|
|
|
|
|
2020-08-12 17:03:06 +02:00
|
|
|
use text_edit::Indel;
|
2020-04-25 00:57:47 +02:00
|
|
|
|
|
|
|
use crate::{validation, AstNode, SourceFile, TextRange};
|
2019-03-22 02:05:12 +09:00
|
|
|
|
|
|
|
fn check_file_invariants(file: &SourceFile) {
|
|
|
|
let root = file.syntax();
|
|
|
|
validation::validate_block_structure(root);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn check_parser(text: &str) {
|
|
|
|
let file = SourceFile::parse(text);
|
2019-07-18 19:23:05 +03:00
|
|
|
check_file_invariants(&file.tree());
|
2019-03-22 02:05:12 +09:00
|
|
|
}
|
2019-03-22 02:06:48 +09:00
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct CheckReparse {
|
|
|
|
text: String,
|
2020-05-05 23:15:49 +02:00
|
|
|
edit: Indel,
|
2019-03-22 02:06:48 +09:00
|
|
|
edited_text: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CheckReparse {
|
|
|
|
pub fn from_data(data: &[u8]) -> Option<Self> {
|
2019-07-04 13:26:44 -04:00
|
|
|
const PREFIX: &str = "fn main(){\n\t";
|
|
|
|
const SUFFIX: &str = "\n}";
|
2019-03-22 04:11:21 +09:00
|
|
|
|
2019-03-22 02:06:48 +09:00
|
|
|
let data = str::from_utf8(data).ok()?;
|
|
|
|
let mut lines = data.lines();
|
2019-03-22 04:11:21 +09:00
|
|
|
let delete_start = usize::from_str(lines.next()?).ok()? + PREFIX.len();
|
2019-03-22 02:06:48 +09:00
|
|
|
let delete_len = usize::from_str(lines.next()?).ok()?;
|
|
|
|
let insert = lines.next()?.to_string();
|
|
|
|
let text = lines.collect::<Vec<_>>().join("\n");
|
2019-03-22 04:11:21 +09:00
|
|
|
let text = format!("{}{}{}", PREFIX, text, SUFFIX);
|
2019-03-22 02:06:48 +09:00
|
|
|
text.get(delete_start..delete_start.checked_add(delete_len)?)?; // make sure delete is a valid range
|
2020-04-24 23:40:41 +02:00
|
|
|
let delete =
|
2020-04-25 00:57:47 +02:00
|
|
|
TextRange::at(delete_start.try_into().unwrap(), delete_len.try_into().unwrap());
|
2019-03-22 02:06:48 +09:00
|
|
|
let edited_text =
|
|
|
|
format!("{}{}{}", &text[..delete_start], &insert, &text[delete_start + delete_len..]);
|
2021-03-21 15:33:18 +01:00
|
|
|
let edit = Indel { insert, delete };
|
2019-03-22 02:06:48 +09:00
|
|
|
Some(CheckReparse { text, edit, edited_text })
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn run(&self) {
|
2019-05-28 17:34:28 +03:00
|
|
|
let parse = SourceFile::parse(&self.text);
|
|
|
|
let new_parse = parse.reparse(&self.edit);
|
2019-07-18 19:23:05 +03:00
|
|
|
check_file_invariants(&new_parse.tree());
|
|
|
|
assert_eq!(&new_parse.tree().syntax().text().to_string(), &self.edited_text);
|
2019-03-22 02:06:48 +09:00
|
|
|
let full_reparse = SourceFile::parse(&self.edited_text);
|
2019-05-28 17:34:28 +03:00
|
|
|
for (a, b) in
|
2019-07-18 19:23:05 +03:00
|
|
|
new_parse.tree().syntax().descendants().zip(full_reparse.tree().syntax().descendants())
|
2019-05-28 17:34:28 +03:00
|
|
|
{
|
2019-07-20 12:58:27 +03:00
|
|
|
if (a.kind(), a.text_range()) != (b.kind(), b.text_range()) {
|
2019-07-20 12:48:24 +03:00
|
|
|
eprint!("original:\n{:#?}", parse.tree().syntax());
|
|
|
|
eprint!("reparsed:\n{:#?}", new_parse.tree().syntax());
|
|
|
|
eprint!("full reparse:\n{:#?}", full_reparse.tree().syntax());
|
2019-03-22 03:15:16 +09:00
|
|
|
assert_eq!(
|
|
|
|
format!("{:?}", a),
|
|
|
|
format!("{:?}", b),
|
|
|
|
"different syntax tree produced by the full reparse"
|
|
|
|
);
|
|
|
|
}
|
2019-03-22 02:06:48 +09:00
|
|
|
}
|
2019-03-22 03:15:16 +09:00
|
|
|
// FIXME
|
|
|
|
// assert_eq!(new_file.errors(), full_reparse.errors());
|
2019-03-22 02:06:48 +09:00
|
|
|
}
|
|
|
|
}
|