# rattle-items-match Matching is not limited to character strings. I'm trying to make a game AI. 文字列に限らないマッチングです。 ゲームAIを作ろうとしています。 Unstable version. It's a tryal and error process. Specifications will change. 不安定版。 試行錯誤中です。 仕様はコロコロ変わるでしょう。 I named it rattle because it's a lot of detailed work. 細かい作業がいっぱいなのでrattleという名前にしました。 ## Run Take a look at the repository. リポジトリを見てください。 ```shell cargo run --example example ``` ## Specification (仕様) The specifications will gradually solidify. 仕様は少しずつ固めていきます。 You can think that you can't do anything that isn't written here. ここに書かれていないことは何もできないと思ってもらって構いません。 ./examples/example.rs: ```rust extern crate rattle_items_match; use rattle_items_match::{ ActualBuilder as Actual, Condition as Cnd, ConditionsBuilder as Cnds, Controls as Co, ExpectedBuilder as Expected, MachineBuilder as Ma, Operator as Op, RangeIncludesMax, Repeat, }; fn main() { println!("Start."); // `Actual` is sequence only. // 比較対象値は シーケンスのみです。 let ac1 = Actual::default() // " 1" .push(&' ') .push(&' ') .push(&' ') .push(&' ') .push(&'1') .build(); let ac2 = Actual::default() // "\t 1" .push(&'\t') .push(&' ') .push(&' ') .push(&' ') .push(&'1') .build(); let ac3 = Actual::default() // 'x 1' .push(&'x') .push(&' ') .push(&' ') .push(&' ') .push(&'1') .build(); let ac4 = Actual::default().push(&'A').build(); // 'A' let ac5 = Actual::default().push(&'B').push(&'C').build(); // 'BC' let ac6 = Actual::default().push(&'d').push(&'e').build(); // 'de' let ac7 = Actual::default().push(&'f').push(&'g').push(&'h').build(); // 'fgh' let ac8 = Actual::default().push(&'\r').push(&'\n').build(); // "\r\n" let ac9 = Actual::default() // '# Comment あ.' .push(&'#') .push(&' ') .push(&'C') .push(&'o') .push(&'m') .push(&'m') .push(&'e') .push(&'n') .push(&'t') .push(&' ') .push(&'あ') .push(&'.') .build(); let ac10 = Actual::default() // 'No-1_0' - Unquloted key. .push(&'N') .push(&'o') .push(&'-') .push(&'1') .push(&'_') .push(&'0') .build(); // Whitespace characters. let wschar = Cnds::default() .push(&Cnd::Pin('\t')) .push(&Cnd::Pin(' ')) .build(); // Newline. let newline = Cnds::default() .push(&Cnd::Pin('\n')) // LF .push(&Cnd::Seq(vec!['\r', '\n'])) // CR LF .build(); // Digit. let digit = Cnd::RangeIncludesMax(RangeIncludesMax::default().min(&'0').max(&'9').build()); // Alphabet. let upper_case = Cnd::RangeIncludesMax(RangeIncludesMax::default().min(&'A').max(&'Z').build()); let lower_case = Cnd::RangeIncludesMax(RangeIncludesMax::default().min(&'a').max(&'z').build()); let alpha = Cnds::default().push(&upper_case).push(&lower_case).build(); let comment_start_symbol = Cnd::Pin('#'); // # let non_ascii = &Cnd::RangeIncludesMax( RangeIncludesMax::default() .min(&(0x80 as char)) .max(&'\u{D7FF}') .build(), ); let non_eol = Op::Or( Cnds::default() .push(&Cnd::Pin(0x09 as char)) .push(&Cnd::RangeIncludesMax( RangeIncludesMax::default() .min(&(0x20 as char)) .max(&(0x7F as char)) .build(), )) .push(&non_ascii) .build(), ); let ex1 = Expected::default() // "(wschar) 1" .push(&Co::Once(Op::Or(wschar.clone()))) .push(&Co::Once(Op::One(Cnd::Pin(' ')))) .push(&Co::Once(Op::One(Cnd::Pin(' ')))) .push(&Co::Once(Op::One(Cnd::Pin(' ')))) .push(&Co::Once(Op::One(Cnd::Pin('1')))) .build(); let ex2 = Expected::default() // "+(wschar)" .push(&Co::Repeat( Repeat::default() .op(&Op::Or(wschar.clone())) .min(1) .max_not_included(usize::MAX) .build(), )) .push(&Co::Once(Op::One(Cnd::Pin('1')))) .build(); let ex3 = Expected::default() // "(wschar){5,}" .push(&Co::Repeat( Repeat::default() .op(&Op::Or(wschar.clone())) .min(5) .max_not_included(usize::MAX) .build(), )) .push(&Co::Once(Op::One(Cnd::Pin('1')))) .build(); let ex4 = Expected::default() // "(wschar){0,3}" .push(&Co::Repeat( Repeat::default() .op(&Op::Or(wschar.clone())) .min(0) .max_not_included(3) .build(), )) .push(&Co::Once(Op::One(Cnd::Pin('1')))) .build(); let ex5 = Expected::default() // "(wschar){1,}" .push(&Co::Repeat( Repeat::default() .op(&Op::Or(wschar.clone())) .min(1) .max_not_included(usize::MAX) .build(), )) .push(&Co::Once(Op::One(digit.clone()))) .build(); let ex6 = Expected::default() // "(alpha)" .push(&Co::Once(Op::Or(alpha.clone()))) .build(); let ex7 = Expected::default() // "(alpha){1,3}" .push(&Co::Repeat( Repeat::default() .op(&Op::Or(alpha.clone())) .min(1) .max_not_included(3) .build(), )) .build(); let ex8 = Expected::default() // "(alpha){1,}" .push(&Co::Repeat( Repeat::default() .op(&Op::Or(alpha.clone())) .min(1) .max_not_included(usize::MAX) .build(), )) .build(); let ex9 = Expected::default() // "(newline)" .push(&Co::Once(Op::Or(newline.clone()))) .build(); let comment = Expected::default() // "# Comment." .push(&Co::Once(Op::One(comment_start_symbol))) .push(&Co::Repeat( Repeat::default() .op(&non_eol) .min(0) .max_not_included(usize::MAX) .build(), )) .build(); let unquoted_key = Expected::default() // 'No-1_0' - Unquloted key. .push(&Co::Repeat( Repeat::default() .min(1) .max_not_included(usize::MAX) .op(&Op::Or( Cnds::default() .extend(&alpha) // A-Z, a-z. .push(&digit) // 0-9. .push(&Cnd::Pin(0x2D as char)) // - .push(&Cnd::Pin(0x5F as char)) // _ .build(), )) .build(), )) .build(); assert!(Ma::default().actual(&ac1).expected(&ex1).build().exec()); assert!(Ma::default().actual(&ac2).expected(&ex1).build().exec()); assert!(!Ma::default().actual(&ac3).expected(&ex1).build().exec()); assert!(Ma::default().actual(&ac1).expected(&ex2).build().exec()); assert!(!Ma::default().actual(&ac1).expected(&ex3).build().exec()); assert!(!Ma::default().actual(&ac1).expected(&ex4).build().exec()); assert!(Ma::default().actual(&ac1).expected(&ex5).build().exec()); assert!(Ma::default().actual(&ac4).expected(&ex6).build().exec()); assert!(Ma::default().actual(&ac5).expected(&ex7).build().exec()); assert!(Ma::default().actual(&ac6).expected(&ex7).build().exec()); assert!(Ma::default().actual(&ac7).expected(&ex8).build().exec()); assert!(Ma::default().actual(&ac8).expected(&ex9).build().exec()); // コメントのテスト assert!(Ma::default().actual(&ac9).expected(&comment).build().exec()); // コメントではないもののテスト assert!(!Ma::default().actual(&ac5).expected(&comment).build().exec()); // 'No-1_0' - Unquloted key. assert!(Ma::default() .actual(&ac10) .expected(&unquoted_key) .build() .exec()); // Newline is not Unquloted key. assert!(!Ma::default() .actual(&ac8) .expected(&unquoted_key) .build() .exec()); println!("Finished."); } ```