use std::collections::HashMap; use structopt::StructOpt; pub const DICT: &str = include_str!("../../assets/words.go"); struct Words<'s> { main: HashMap<&'s str, Vec<&'s str>>, american: HashMap<&'s str, Vec<&'s str>>, british: HashMap<&'s str, Vec<&'s str>>, } fn parse_dict(raw: &str) -> Words { let mut bad = HashMap::new(); let mut main = HashMap::new(); let mut american = HashMap::new(); let mut british = HashMap::new(); let mapping = regex::Regex::new(r#"^"(.*)", "(.*)",$"#).unwrap(); let mut current = &mut bad; for line in raw.lines() { let line = line.splitn(2, "//").next().unwrap().trim(); if line.is_empty() || line.starts_with("package") { continue; } else if line.contains("DictMain") { current = &mut main; } else if line.contains("DictAmerican") { current = &mut american; } else if line.contains("DictBritish") { current = &mut british; } else if line.contains('}') { current = &mut bad; } else { let captures = mapping.captures(line); if let Some(captures) = captures { current.insert( captures.get(1).unwrap().as_str(), vec![captures.get(2).unwrap().as_str()], ); } else { eprintln!("Unknown line: {}", line); } } } if !bad.is_empty() { panic!("Failed parsing; found extra words: {:#?}", bad); } Words { main, american, british, } } fn generate(file: &mut W) { writeln!( file, "// This file is code-genned by {}", env!("CARGO_PKG_NAME") ) .unwrap(); writeln!(file, "#![allow(clippy::unreadable_literal)]",).unwrap(); writeln!(file).unwrap(); let Words { main, american, british, } = parse_dict(DICT); let mut main: Vec<_> = main.into_iter().collect(); main.sort_unstable_by(|a, b| { unicase::UniCase::new(a.0) .partial_cmp(&unicase::UniCase::new(b.0)) .unwrap() }); let mut american: Vec<_> = american.into_iter().collect(); american.sort_unstable_by(|a, b| { unicase::UniCase::new(a.0) .partial_cmp(&unicase::UniCase::new(b.0)) .unwrap() }); let mut british: Vec<_> = british.into_iter().collect(); british.sort_unstable_by(|a, b| { unicase::UniCase::new(a.0) .partial_cmp(&unicase::UniCase::new(b.0)) .unwrap() }); writeln!(file, "pub static MAIN_DICTIONARY: &[(&str, &[&str])] = &[").unwrap(); for (typo, corrections) in main.into_iter() { let value = itertools::join(corrections.iter().map(|s| format!("{:?}", s)), ", "); let value = format!("&[{}]", value); let key = format!("{:?}", typo); writeln!(file, " ({}, {}),", key, &value).unwrap(); } writeln!(file, "];").unwrap(); writeln!(file).unwrap(); writeln!( file, "pub static AMERICAN_DICTIONARY: &[(&str, &[&str])] = &[" ) .unwrap(); for (typo, corrections) in american.into_iter() { let value = itertools::join(corrections.iter().map(|s| format!("{:?}", s)), ", "); let value = format!("&[{}]", value); let key = format!("{:?}", typo); writeln!(file, " ({}, {}),", key, &value).unwrap(); } writeln!(file, "];").unwrap(); writeln!(file).unwrap(); writeln!( file, "pub static BRITISH_DICTIONARY: &[(&str, &[&str])] = &[" ) .unwrap(); for (typo, corrections) in british.into_iter() { let value = itertools::join(corrections.iter().map(|s| format!("{:?}", s)), ", "); let value = format!("&[{}]", value); let key = format!("{:?}", typo); writeln!(file, " ({}, {}),", key, &value).unwrap(); } writeln!(file, "];").unwrap(); } #[derive(Debug, StructOpt)] #[structopt(rename_all = "kebab-case")] struct Options { #[structopt(flatten)] codegen: codegenrs::CodeGenArgs, #[structopt(flatten)] rustmft: codegenrs::RustfmtArgs, } fn run() -> Result> { let options = Options::from_args(); let mut content = vec![]; generate(&mut content); let content = String::from_utf8(content)?; let content = options.rustmft.reformat(&content)?; options.codegen.write_str(&content)?; Ok(0) } fn main() { let code = run().unwrap(); std::process::exit(code); }