typos/crates/varcon-core/src/parser.rs

541 lines
16 KiB
Rust
Raw Normal View History

2023-02-21 12:11:24 -05:00
use winnow::prelude::*;
2020-04-07 20:50:06 -04:00
use crate::*;
#[derive(Clone, PartialEq, Eq, Debug)]
pub struct ClusterIter<'i> {
input: &'i str,
}
impl<'i> ClusterIter<'i> {
pub fn new(input: &'i str) -> Self {
Self { input }
}
}
impl<'i> Iterator for ClusterIter<'i> {
type Item = Cluster;
fn next(&mut self) -> Option<Cluster> {
let i = self.input.trim_start();
let (i, c) = Cluster::parse(i).ok()?;
self.input = i;
Some(c)
}
}
#[cfg(test)]
mod test_cluster_iter {
use super::*;
#[test]
fn test_single() {
let iter = ClusterIter::new(
"# acknowledgment <verified> (level 35)
A Cv: acknowledgment / Av B C: acknowledgement
A Cv: acknowledgments / Av B C: acknowledgements
A Cv: acknowledgment's / Av B C: acknowledgement's
",
);
2021-11-08 12:36:05 -05:00
assert_eq!(iter.count(), 1);
2020-04-07 20:50:06 -04:00
}
#[test]
fn test_multiple() {
let iter = ClusterIter::new(
"# acknowledgment <verified> (level 35)
A Cv: acknowledgment / Av B C: acknowledgement
A Cv: acknowledgments / Av B C: acknowledgements
A Cv: acknowledgment's / Av B C: acknowledgement's
# acknowledgment <verified> (level 35)
A Cv: acknowledgment / Av B C: acknowledgement
A Cv: acknowledgments / Av B C: acknowledgements
A Cv: acknowledgment's / Av B C: acknowledgement's
",
);
2021-11-08 12:36:05 -05:00
assert_eq!(iter.count(), 2);
2020-04-07 20:50:06 -04:00
}
}
impl Cluster {
pub fn parse(input: &str) -> IResult<&str, Self> {
2023-02-21 12:11:24 -05:00
let header = (
"#",
winnow::ascii::space0,
winnow::ascii::not_line_ending,
winnow::ascii::line_ending,
2023-02-21 12:11:24 -05:00
);
let note = winnow::combinator::preceded(
("##", winnow::ascii::space0),
winnow::combinator::terminated(
winnow::ascii::not_line_ending,
winnow::ascii::line_ending,
2020-04-07 20:50:06 -04:00
),
);
2023-02-21 12:11:24 -05:00
let mut cluster = (
2023-02-21 11:41:45 -05:00
winnow::combinator::opt(header),
winnow::combinator::repeat(
1..,
winnow::combinator::terminated(Entry::parse, winnow::ascii::line_ending),
),
winnow::combinator::repeat(0.., note),
2023-02-21 12:11:24 -05:00
);
let (input, (header, entries, notes)): (_, (_, _, Vec<_>)) = cluster.parse_next(input)?;
2020-04-07 20:50:06 -04:00
let header = header.map(|s| s.2.to_owned());
let notes = notes.into_iter().map(|s| s.to_owned()).collect();
let c = Self {
header,
entries,
notes,
};
Ok((input, c))
}
}
#[cfg(test)]
mod test_cluster {
use super::*;
#[test]
fn test_basic() {
let (input, actual) = Cluster::parse(
"# acknowledgment <verified> (level 35)
A Cv: acknowledgment / Av B C: acknowledgement
A Cv: acknowledgments / Av B C: acknowledgements
A Cv: acknowledgment's / Av B C: acknowledgement's
",
)
.unwrap();
assert_eq!(input, "\n");
assert_eq!(
actual.header,
Some("acknowledgment <verified> (level 35)".to_owned())
);
assert_eq!(actual.entries.len(), 3);
assert_eq!(actual.notes.len(), 0);
}
#[test]
fn test_notes() {
let (input, actual) = Cluster::parse(
"# coloration <verified> (level 50)
A B C: coloration / B. Cv: colouration
A B C: colorations / B. Cv: colourations
A B C: coloration's / B. Cv: colouration's
## OED has coloration as the preferred spelling and discolouration as a
## variant for British Engl or some reason
",
)
.unwrap();
assert_eq!(input, "\n");
assert_eq!(
actual.header,
Some("coloration <verified> (level 50)".to_owned())
);
assert_eq!(actual.entries.len(), 3);
assert_eq!(actual.notes.len(), 2);
}
}
impl Entry {
pub fn parse(input: &str) -> IResult<&str, Self> {
let var_sep = (winnow::ascii::space0, '/', winnow::ascii::space0);
2023-03-18 05:11:55 -04:00
let (input, variants) =
winnow::combinator::separated1(Variant::parse, var_sep).parse_next(input)?;
2023-02-21 12:11:24 -05:00
let desc_sep = (winnow::ascii::space0, '|');
2023-02-21 12:11:24 -05:00
let (input, description) =
2023-03-18 05:11:55 -04:00
winnow::combinator::opt((desc_sep, Self::parse_description)).parse_next(input)?;
2023-02-21 12:11:24 -05:00
let comment_sep = (winnow::ascii::space0, '#');
2023-02-21 12:11:24 -05:00
let (input, comment) = winnow::combinator::opt((
2020-04-07 20:50:06 -04:00
comment_sep,
winnow::ascii::space1,
winnow::ascii::not_line_ending,
2023-03-18 05:11:55 -04:00
))
.parse_next(input)?;
2020-04-07 20:50:06 -04:00
let mut e = match description {
Some((_, description)) => description,
None => Self {
variants: Vec::new(),
pos: None,
archaic: false,
note: false,
description: None,
comment: None,
},
};
e.variants = variants;
e.comment = comment.map(|c| c.2.to_owned());
Ok((input, e))
}
fn parse_description(input: &str) -> IResult<&str, Self> {
2023-02-21 12:11:24 -05:00
let (input, (pos, archaic, note, description)) = (
winnow::combinator::opt((winnow::ascii::space1, Pos::parse)),
winnow::combinator::opt((winnow::ascii::space1, "(-)")),
winnow::combinator::opt((winnow::ascii::space1, "--")),
2023-02-21 12:11:24 -05:00
winnow::combinator::opt((
winnow::ascii::space1,
winnow::token::take_till0(('\n', '\r', '#')),
2023-02-21 12:11:24 -05:00
)),
)
.parse_next(input)?;
2020-04-07 20:50:06 -04:00
let variants = Vec::new();
let pos = pos.map(|(_, p)| p);
let archaic = archaic.is_some();
let note = note.is_some();
let description = description.map(|(_, d)| d.to_owned());
let e = Self {
variants,
pos,
archaic,
note,
description,
comment: None,
};
Ok((input, e))
}
}
#[cfg(test)]
mod test_entry {
2021-06-28 14:50:59 -04:00
#![allow(clippy::bool_assert_comparison)]
2020-04-07 20:50:06 -04:00
use super::*;
#[test]
fn test_variant_only() {
// Having nothing after `A` causes an incomplete parse. Shouldn't be a problem for my use
// cases.
let (input, actual) =
Entry::parse("A Cv: acknowledgment's / Av B C: acknowledgement's\n").unwrap();
assert_eq!(input, "\n");
assert_eq!(actual.variants.len(), 2);
assert_eq!(actual.pos, None);
assert_eq!(actual.archaic, false);
assert_eq!(actual.note, false);
assert_eq!(actual.description, None);
}
#[test]
fn test_description() {
// Having nothing after `A` causes an incomplete parse. Shouldn't be a problem for my use
// cases.
let (input, actual) = Entry::parse("A C: prize / B: prise | otherwise\n").unwrap();
assert_eq!(input, "\n");
assert_eq!(actual.variants.len(), 2);
assert_eq!(actual.pos, None);
assert_eq!(actual.archaic, false);
assert_eq!(actual.note, false);
assert_eq!(actual.description, Some("otherwise".to_owned()));
}
#[test]
fn test_pos() {
// Having nothing after `A` causes an incomplete parse. Shouldn't be a problem for my use
// cases.
let (input, actual) = Entry::parse("A B C: practice / AV Cv: practise | <N>\n").unwrap();
assert_eq!(input, "\n");
assert_eq!(actual.variants.len(), 2);
assert_eq!(actual.pos, Some(Pos::Noun));
assert_eq!(actual.archaic, false);
assert_eq!(actual.note, false);
assert_eq!(actual.description, None);
}
#[test]
fn test_archaic() {
// Having nothing after `A` causes an incomplete parse. Shouldn't be a problem for my use
// cases.
let (input, actual) = Entry::parse("A: bark / Av B: barque | (-) ship\n").unwrap();
assert_eq!(input, "\n");
assert_eq!(actual.variants.len(), 2);
assert_eq!(actual.pos, None);
assert_eq!(actual.archaic, true);
assert_eq!(actual.note, false);
assert_eq!(actual.description, Some("ship".to_owned()));
}
#[test]
fn test_note() {
// Having nothing after `A` causes an incomplete parse. Shouldn't be a problem for my use
// cases.
let (input, actual) = Entry::parse("_: cabbies | -- plural\n").unwrap();
assert_eq!(input, "\n");
assert_eq!(actual.variants.len(), 1);
assert_eq!(actual.pos, None);
assert_eq!(actual.archaic, false);
assert_eq!(actual.note, true);
assert_eq!(actual.description, Some("plural".to_owned()));
}
#[test]
fn test_trailing_comment() {
let (input, actual) = Entry::parse(
"A B: accursed / AV B-: accurst # ODE: archaic, M-W: 'or' but can find little evidence of use\n",
)
.unwrap();
assert_eq!(input, "\n");
assert_eq!(actual.variants.len(), 2);
assert_eq!(actual.pos, None);
assert_eq!(actual.archaic, false);
assert_eq!(actual.note, false);
assert_eq!(actual.description, None);
assert_eq!(
actual.comment,
Some("ODE: archaic, M-W: 'or' but can find little evidence of use".to_owned())
);
}
}
impl Variant {
pub fn parse(input: &str) -> IResult<&str, Self> {
let types = winnow::combinator::separated1(Type::parse, winnow::ascii::space1);
let sep = (":", winnow::ascii::space0);
2023-03-18 05:11:55 -04:00
let (input, (types, word)) =
winnow::combinator::separated_pair(types, sep, word).parse_next(input)?;
2020-04-07 20:50:06 -04:00
let v = Self { types, word };
Ok((input, v))
}
}
fn word(input: &str) -> IResult<&str, String> {
winnow::token::take_till1(|item: char| item.is_ascii_whitespace())
2023-02-21 11:41:45 -05:00
.map(|s: &str| s.to_owned().replace('_', " "))
.parse_next(input)
2020-04-07 20:50:06 -04:00
}
#[cfg(test)]
mod test_variant {
use super::*;
#[test]
fn test_valid() {
// Having nothing after `A` causes an incomplete parse. Shouldn't be a problem for my use
// cases.
let (input, actual) = Variant::parse("A Cv: acknowledgment ").unwrap();
assert_eq!(input, " ");
assert_eq!(
actual.types,
vec![
Type {
category: Category::American,
tag: None,
num: None,
},
Type {
category: Category::Canadian,
tag: Some(Tag::Variant),
num: None,
}
]
);
assert_eq!(actual.word, "acknowledgment");
}
#[test]
fn test_extra() {
let (input, actual) =
Variant::parse("A Cv: acknowledgment's / Av B C: acknowledgement's").unwrap();
assert_eq!(input, " / Av B C: acknowledgement's");
assert_eq!(
actual.types,
vec![
Type {
category: Category::American,
tag: None,
num: None,
},
Type {
category: Category::Canadian,
tag: Some(Tag::Variant),
num: None,
}
]
);
assert_eq!(actual.word, "acknowledgment's");
}
#[test]
fn test_underscore() {
let (input, actual) = Variant::parse("_: air_gun\n").unwrap();
assert_eq!(input, "\n");
assert_eq!(
actual.types,
vec![Type {
category: Category::Other,
tag: None,
num: None,
},]
);
assert_eq!(actual.word, "air gun");
}
}
impl Type {
pub fn parse(input: &str) -> IResult<&str, Type> {
let (input, category) = Category::parse(input)?;
2023-03-18 05:11:55 -04:00
let (input, tag) = winnow::combinator::opt(Tag::parse).parse_next(input)?;
let (input, num) = winnow::combinator::opt(winnow::ascii::digit1).parse_next(input)?;
2020-04-07 20:50:06 -04:00
let num = num.map(|s| s.parse().expect("parser ensured its a number"));
let t = Type { category, tag, num };
Ok((input, t))
}
}
#[cfg(test)]
mod test_type {
use super::*;
#[test]
fn test_valid() {
// Having nothing after `A` causes an incomplete parse. Shouldn't be a problem for my use
// cases.
let (input, actual) = Type::parse("A ").unwrap();
assert_eq!(input, " ");
assert_eq!(actual.category, Category::American);
assert_eq!(actual.tag, None);
assert_eq!(actual.num, None);
let (input, actual) = Type::parse("Bv ").unwrap();
assert_eq!(input, " ");
assert_eq!(actual.category, Category::BritishIse);
assert_eq!(actual.tag, Some(Tag::Variant));
assert_eq!(actual.num, None);
}
#[test]
fn test_extra() {
let (input, actual) = Type::parse("Z foobar").unwrap();
assert_eq!(input, " foobar");
assert_eq!(actual.category, Category::BritishIze);
assert_eq!(actual.tag, None);
assert_eq!(actual.num, None);
let (input, actual) = Type::parse("C- foobar").unwrap();
assert_eq!(input, " foobar");
assert_eq!(actual.category, Category::Canadian);
assert_eq!(actual.tag, Some(Tag::Possible));
assert_eq!(actual.num, None);
}
#[test]
fn test_num() {
let (input, actual) = Type::parse("Av1 ").unwrap();
assert_eq!(input, " ");
assert_eq!(actual.category, Category::American);
assert_eq!(actual.tag, Some(Tag::Variant));
assert_eq!(actual.num, Some(1));
}
}
impl Category {
pub fn parse(input: &str) -> IResult<&str, Category> {
let symbols = winnow::token::one_of(['A', 'B', 'Z', 'C', 'D', '_']);
2023-02-21 12:11:24 -05:00
symbols
.map(|c| match c {
'A' => Category::American,
'B' => Category::BritishIse,
'Z' => Category::BritishIze,
'C' => Category::Canadian,
'D' => Category::Australian,
'_' => Category::Other,
_ => unreachable!("parser won't select this option"),
})
.parse_next(input)
2020-04-07 20:50:06 -04:00
}
}
#[cfg(test)]
mod test_category {
use super::*;
#[test]
fn test_valid() {
let (input, actual) = Category::parse("A").unwrap();
assert_eq!(input, "");
assert_eq!(actual, Category::American);
}
#[test]
fn test_extra() {
let (input, actual) = Category::parse("_ foobar").unwrap();
assert_eq!(input, " foobar");
assert_eq!(actual, Category::Other);
}
}
impl Tag {
pub fn parse(input: &str) -> IResult<&str, Tag> {
let symbols = winnow::token::one_of(['.', 'v', 'V', '-', 'x']);
2023-02-21 12:11:24 -05:00
symbols
.map(|c| match c {
'.' => Tag::Eq,
'v' => Tag::Variant,
'V' => Tag::Seldom,
'-' => Tag::Possible,
'x' => Tag::Improper,
_ => unreachable!("parser won't select this option"),
})
.parse_next(input)
2020-04-07 20:50:06 -04:00
}
}
#[cfg(test)]
mod test_tag {
use super::*;
#[test]
fn test_valid() {
let (input, actual) = Tag::parse(".").unwrap();
assert_eq!(input, "");
assert_eq!(actual, Tag::Eq);
}
#[test]
fn test_extra() {
let (input, actual) = Tag::parse("x foobar").unwrap();
assert_eq!(input, " foobar");
assert_eq!(actual, Tag::Improper);
}
}
impl Pos {
pub fn parse(input: &str) -> IResult<&str, Pos> {
2023-02-21 11:41:45 -05:00
winnow::branch::alt((
"<N>".value(Pos::Noun),
"<V>".value(Pos::Verb),
"<Adj>".value(Pos::Adjective),
"<Adv>".value(Pos::Adverb),
2023-03-18 05:11:55 -04:00
))
.parse_next(input)
2020-04-07 20:50:06 -04:00
}
}
#[cfg(test)]
mod test_pos {
use super::*;
#[test]
fn test_valid() {
let (input, actual) = Pos::parse("<N>").unwrap();
assert_eq!(input, "");
assert_eq!(actual, Pos::Noun);
}
#[test]
fn test_extra() {
let (input, actual) = Pos::parse("<Adj> foobar").unwrap();
assert_eq!(input, " foobar");
assert_eq!(actual, Pos::Adjective);
}
}