2020-06-19 01:39:08 -04:00
|
|
|
extern crate nom;
|
2020-06-18 21:00:42 -04:00
|
|
|
extern crate rand;
|
|
|
|
|
2020-06-19 01:39:08 -04:00
|
|
|
pub mod parser;
|
|
|
|
|
2020-06-18 21:00:42 -04:00
|
|
|
use rand::{seq::SliceRandom, thread_rng};
|
|
|
|
use std::collections::HashMap;
|
|
|
|
|
|
|
|
pub trait Purrchance {
|
|
|
|
fn eval(&self, g: &Grammar) -> Option<String>;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum Symbol {
|
|
|
|
Terminal(String),
|
|
|
|
NonTerminal(String),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Purrchance for Symbol {
|
|
|
|
fn eval(&self, g: &Grammar) -> Option<String> {
|
|
|
|
match self {
|
2020-06-19 01:39:08 -04:00
|
|
|
Symbol::Terminal(s) => Some(String::from(s)),
|
2020-06-18 21:00:42 -04:00
|
|
|
Symbol::NonTerminal(label) => g.0.get(label)?.eval(g),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Expr(Vec<Symbol>);
|
|
|
|
|
|
|
|
impl Purrchance for Expr {
|
|
|
|
fn eval(&self, g: &Grammar) -> Option<String> {
|
|
|
|
Some(self.0.iter().map(|sym| sym.eval(g)).collect::<Option<Vec<String>>>()?.join(""))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-18 21:21:39 -04:00
|
|
|
pub struct List(Vec<(Expr, f64)>);
|
2020-06-18 21:00:42 -04:00
|
|
|
|
|
|
|
impl Purrchance for List {
|
|
|
|
fn eval(&self, g: &Grammar) -> Option<String> {
|
2020-06-18 21:21:39 -04:00
|
|
|
self.0.choose_weighted(&mut thread_rng(), |item| item.1).ok()?.0.eval(g)
|
2020-06-18 21:00:42 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Grammar(HashMap<String,List>);
|
|
|
|
|
2020-06-18 19:38:55 -04:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2020-06-18 21:00:42 -04:00
|
|
|
use super::*;
|
2020-06-19 01:39:08 -04:00
|
|
|
use parser::*;
|
2020-06-18 21:00:42 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_terminal() {
|
|
|
|
let sym = Symbol::Terminal("hello world".to_string());
|
|
|
|
let g = Grammar(HashMap::new());
|
|
|
|
assert_eq!(sym.eval(&g), Some("hello world".to_string()));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_terminals_expr() {
|
|
|
|
let sym1 = Symbol::Terminal("hell".to_string());
|
|
|
|
let sym2 = Symbol::Terminal("o world".to_string());
|
|
|
|
let expr = Expr(vec![sym1, sym2]);
|
|
|
|
let g = Grammar(HashMap::new());
|
|
|
|
assert_eq!(expr.eval(&g), Some("hello world".to_string()));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_single_terminals_expr_list() {
|
|
|
|
let sym1 = Symbol::Terminal("hell".to_string());
|
|
|
|
let sym2 = Symbol::Terminal("o world".to_string());
|
|
|
|
let expr = Expr(vec![sym1, sym2]);
|
2020-06-18 21:21:39 -04:00
|
|
|
let list = List(vec![(expr, 1.0)]);
|
2020-06-18 21:00:42 -04:00
|
|
|
let g = Grammar(HashMap::new());
|
|
|
|
assert_eq!(list.eval(&g), Some("hello world".to_string()));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_multiple_terminals_expr_list() {
|
|
|
|
let sym1 = Symbol::Terminal("hello".to_string());
|
|
|
|
let sym2 = Symbol::Terminal("goodbye".to_string());
|
2020-06-18 21:21:39 -04:00
|
|
|
let list = List(vec![(Expr(vec![sym1]), 1.0), (Expr(vec![sym2]), 1.0)]);
|
2020-06-18 21:00:42 -04:00
|
|
|
let g = Grammar(HashMap::new());
|
|
|
|
assert!(vec![Some("hello".to_string()), Some("goodbye".to_string())].contains(&list.eval(&g)));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_empty_list() {
|
|
|
|
let list = List(vec![]);
|
|
|
|
let g = Grammar(HashMap::new());
|
|
|
|
assert_eq!(list.eval(&g), None);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_valid_nonterminal() {
|
|
|
|
let term = Symbol::Terminal("hello world".to_string());
|
2020-06-18 21:21:39 -04:00
|
|
|
let list = List(vec![(Expr(vec![term]), 1.0)]);
|
2020-06-18 21:00:42 -04:00
|
|
|
let mut g = Grammar(HashMap::new());
|
|
|
|
g.0.insert("output".to_string(), list);
|
|
|
|
let nt = Symbol::NonTerminal("output".to_string());
|
|
|
|
assert_eq!(nt.eval(&g), Some("hello world".to_string()));
|
|
|
|
}
|
|
|
|
|
2020-06-18 19:38:55 -04:00
|
|
|
#[test]
|
2020-06-18 21:00:42 -04:00
|
|
|
fn eval_missing_nonterminal() {
|
|
|
|
let term = Symbol::Terminal("hello world".to_string());
|
2020-06-18 21:21:39 -04:00
|
|
|
let list = List(vec![(Expr(vec![term]), 1.0)]);
|
2020-06-18 21:00:42 -04:00
|
|
|
let mut g = Grammar(HashMap::new());
|
|
|
|
g.0.insert("output".to_string(), list);
|
|
|
|
let nt = Symbol::NonTerminal("missing".to_string());
|
|
|
|
assert_eq!(nt.eval(&g), None);
|
2020-06-18 19:38:55 -04:00
|
|
|
}
|
2020-06-19 01:39:08 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_loaded_grammar() {
|
|
|
|
let g = load_grammar("test\n foo\n");
|
|
|
|
let nt = Symbol::NonTerminal(String::from("test"));
|
|
|
|
assert_eq!(nt.eval(&g), Some(String::from("foo")));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_loaded_grammar_comments() {
|
|
|
|
let g = load_grammar("// testing
|
|
|
|
test
|
|
|
|
foo // blah blah
|
|
|
|
// isn't this fun?");
|
|
|
|
let nt = Symbol::NonTerminal(String::from("test"));
|
|
|
|
assert_eq!(nt.eval(&g), Some(String::from("foo")));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_loaded_grammar_comments_weights() {
|
|
|
|
let g = load_grammar("// testing
|
|
|
|
test
|
|
|
|
foo ^100
|
|
|
|
// isn't this fun?");
|
|
|
|
let nt = Symbol::NonTerminal(String::from("test"));
|
|
|
|
assert_eq!(nt.eval(&g), Some(String::from("foo")));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn eval_loaded_grammar_comments_fraction_weights_tabs() {
|
|
|
|
let g = load_grammar("
|
|
|
|
test
|
|
|
|
foo ^1000000
|
|
|
|
bar ^1/1000000
|
|
|
|
");
|
|
|
|
let nt = Symbol::NonTerminal(String::from("test"));
|
|
|
|
assert_eq!(nt.eval(&g), Some(String::from("foo")));
|
|
|
|
}
|
2020-06-18 19:38:55 -04:00
|
|
|
}
|