mirror of
https://github.com/crate-ci/typos.git
synced 2025-01-11 01:01:36 -05:00
566 lines
17 KiB
Rust
566 lines
17 KiB
Rust
use nom::IResult;
|
|
use nom::InputTakeAtPosition;
|
|
use nom::Parser;
|
|
|
|
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
|
|
|
|
",
|
|
);
|
|
assert_eq!(iter.count(), 1);
|
|
}
|
|
|
|
#[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
|
|
|
|
",
|
|
);
|
|
assert_eq!(iter.count(), 2);
|
|
}
|
|
}
|
|
|
|
impl Cluster {
|
|
pub fn parse(input: &str) -> IResult<&str, Self> {
|
|
let header = nom::sequence::tuple((
|
|
nom::bytes::streaming::tag("#"),
|
|
nom::character::streaming::space0,
|
|
nom::character::streaming::not_line_ending,
|
|
nom::character::streaming::line_ending,
|
|
));
|
|
let note = nom::sequence::preceded(
|
|
nom::sequence::pair(
|
|
nom::bytes::streaming::tag("##"),
|
|
nom::character::streaming::space0,
|
|
),
|
|
nom::sequence::terminated(
|
|
nom::character::streaming::not_line_ending,
|
|
nom::character::streaming::line_ending,
|
|
),
|
|
);
|
|
let mut cluster = nom::sequence::tuple((
|
|
nom::combinator::opt(header),
|
|
nom::multi::many1(nom::sequence::terminated(
|
|
Entry::parse,
|
|
nom::character::streaming::line_ending,
|
|
)),
|
|
nom::multi::many0(note),
|
|
));
|
|
let (input, (header, entries, notes)) = (cluster)(input)?;
|
|
|
|
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 = nom::sequence::tuple((
|
|
nom::character::streaming::space0,
|
|
nom::bytes::streaming::tag("/"),
|
|
nom::character::streaming::space0,
|
|
));
|
|
let (input, variants) = nom::multi::separated_list1(var_sep, Variant::parse)(input)?;
|
|
|
|
let desc_sep = nom::sequence::tuple((
|
|
nom::character::streaming::space0,
|
|
nom::bytes::streaming::tag("|"),
|
|
));
|
|
let (input, description) =
|
|
nom::combinator::opt(nom::sequence::tuple((desc_sep, Self::parse_description)))(input)?;
|
|
|
|
let comment_sep = nom::sequence::tuple((
|
|
nom::character::streaming::space0,
|
|
nom::bytes::streaming::tag("#"),
|
|
));
|
|
let (input, comment) = nom::combinator::opt(nom::sequence::tuple((
|
|
comment_sep,
|
|
nom::character::streaming::space1,
|
|
nom::character::streaming::not_line_ending,
|
|
)))(input)?;
|
|
|
|
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> {
|
|
let (input, (pos, archaic, note, description)) = nom::sequence::tuple((
|
|
nom::combinator::opt(nom::sequence::tuple((
|
|
nom::character::streaming::space1,
|
|
Pos::parse,
|
|
))),
|
|
nom::combinator::opt(nom::sequence::tuple((
|
|
nom::character::streaming::space1,
|
|
nom::bytes::streaming::tag("(-)"),
|
|
))),
|
|
nom::combinator::opt(nom::sequence::tuple((
|
|
nom::character::streaming::space1,
|
|
nom::bytes::streaming::tag("--"),
|
|
))),
|
|
nom::combinator::opt(nom::sequence::tuple((
|
|
nom::character::streaming::space1,
|
|
nom::bytes::streaming::take_till(|c| c == '\n' || c == '\r' || c == '#'),
|
|
))),
|
|
))(input)?;
|
|
|
|
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 {
|
|
#![allow(clippy::bool_assert_comparison)]
|
|
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 = nom::multi::separated_list1(nom::character::streaming::space1, Type::parse);
|
|
let sep = nom::sequence::tuple((
|
|
nom::bytes::streaming::tag(":"),
|
|
nom::character::streaming::space0,
|
|
));
|
|
let (input, (types, word)) = nom::sequence::separated_pair(types, sep, word)(input)?;
|
|
let v = Self { types, word };
|
|
Ok((input, v))
|
|
}
|
|
}
|
|
|
|
fn word(input: &str) -> IResult<&str, String> {
|
|
input
|
|
.split_at_position1(
|
|
|item| item.is_ascii_whitespace(),
|
|
nom::error::ErrorKind::Alpha,
|
|
)
|
|
.map(|(i, s)| (i, s.to_owned().replace('_', " ")))
|
|
}
|
|
|
|
#[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)?;
|
|
let (input, tag) = nom::combinator::opt(Tag::parse)(input)?;
|
|
let (input, num) = nom::combinator::opt(nom::character::streaming::digit1)(input)?;
|
|
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 = nom::character::streaming::one_of("ABZCD_");
|
|
nom::combinator::map(symbols, |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"),
|
|
})(input)
|
|
}
|
|
}
|
|
|
|
#[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 = nom::character::streaming::one_of(".vV-x");
|
|
nom::combinator::map(symbols, |c| match c {
|
|
'.' => Tag::Eq,
|
|
'v' => Tag::Variant,
|
|
'V' => Tag::Seldom,
|
|
'-' => Tag::Possible,
|
|
'x' => Tag::Improper,
|
|
_ => unreachable!("parser won't select this option"),
|
|
})(input)
|
|
}
|
|
}
|
|
|
|
#[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> {
|
|
use nom::bytes::streaming::tag;
|
|
let noun = tag("<N>");
|
|
let verb = tag("<V>");
|
|
let adjective = tag("<Adj>");
|
|
let adverb = tag("<Adv>");
|
|
nom::branch::alt((
|
|
noun.map(|_| Pos::Noun),
|
|
verb.map(|_| Pos::Verb),
|
|
adjective.map(|_| Pos::Adjective),
|
|
adverb.map(|_| Pos::Adverb),
|
|
))(input)
|
|
}
|
|
}
|
|
|
|
#[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);
|
|
}
|
|
}
|