command.rs 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. use crate::error::Error;
  2. use crate::ignore_block::IgnoreBlocks;
  3. use crate::token::{Token, Tokenizer};
  4. pub mod assign;
  5. pub mod close;
  6. pub mod glacier;
  7. pub mod nominate;
  8. pub mod note;
  9. pub mod ping;
  10. pub mod prioritize;
  11. pub mod relabel;
  12. pub mod second;
  13. pub mod shortcut;
  14. pub fn find_command_start(input: &str, bot: &str) -> Option<usize> {
  15. input.to_ascii_lowercase().find(&format!("@{}", bot))
  16. }
  17. #[derive(Debug, PartialEq)]
  18. pub enum Command<'a> {
  19. Relabel(Result<relabel::RelabelCommand, Error<'a>>),
  20. Assign(Result<assign::AssignCommand, Error<'a>>),
  21. Ping(Result<ping::PingCommand, Error<'a>>),
  22. Nominate(Result<nominate::NominateCommand, Error<'a>>),
  23. Prioritize(Result<prioritize::PrioritizeCommand, Error<'a>>),
  24. Second(Result<second::SecondCommand, Error<'a>>),
  25. Glacier(Result<glacier::GlacierCommand, Error<'a>>),
  26. Shortcut(Result<shortcut::ShortcutCommand, Error<'a>>),
  27. Close(Result<close::CloseCommand, Error<'a>>),
  28. Note(Result<note::NoteCommand, Error<'a>>),
  29. }
  30. #[derive(Debug)]
  31. pub struct Input<'a> {
  32. all: &'a str,
  33. parsed: usize,
  34. ignore: IgnoreBlocks,
  35. // A list of possible bot names.
  36. bot: Vec<&'a str>,
  37. }
  38. fn parse_single_command<'a, T, F, M>(
  39. parse: F,
  40. mapper: M,
  41. tokenizer: &Tokenizer<'a>,
  42. ) -> Option<(Tokenizer<'a>, Command<'a>)>
  43. where
  44. F: FnOnce(&mut Tokenizer<'a>) -> Result<Option<T>, Error<'a>>,
  45. M: FnOnce(Result<T, Error<'a>>) -> Command<'a>,
  46. T: std::fmt::Debug,
  47. {
  48. let mut tok = tokenizer.clone();
  49. let res = parse(&mut tok);
  50. log::info!("parsed {:?} command: {:?}", std::any::type_name::<T>(), res);
  51. match res {
  52. Ok(None) => None,
  53. Ok(Some(v)) => Some((tok, mapper(Ok(v)))),
  54. Err(err) => Some((tok, mapper(Err(err)))),
  55. }
  56. }
  57. impl<'a> Input<'a> {
  58. pub fn new(input: &'a str, bot: Vec<&'a str>) -> Input<'a> {
  59. Input {
  60. all: input,
  61. parsed: 0,
  62. ignore: IgnoreBlocks::new(input),
  63. bot,
  64. }
  65. }
  66. fn parse_command(&mut self) -> Option<Command<'a>> {
  67. let mut tok = Tokenizer::new(&self.all[self.parsed..]);
  68. let name_length = if let Ok(Some(Token::Word(bot_name))) = tok.next_token() {
  69. assert!(self
  70. .bot
  71. .iter()
  72. .any(|name| bot_name.eq_ignore_ascii_case(&format!("@{}", name))));
  73. bot_name.len()
  74. } else {
  75. panic!("no bot name?")
  76. };
  77. log::info!("identified potential command");
  78. let mut success = vec![];
  79. let original_tokenizer = tok.clone();
  80. success.extend(parse_single_command(
  81. relabel::RelabelCommand::parse,
  82. Command::Relabel,
  83. &original_tokenizer,
  84. ));
  85. success.extend(parse_single_command(
  86. assign::AssignCommand::parse,
  87. Command::Assign,
  88. &original_tokenizer,
  89. ));
  90. success.extend(parse_single_command(
  91. note::NoteCommand::parse,
  92. Command::Note,
  93. &original_tokenizer,
  94. ));
  95. success.extend(parse_single_command(
  96. ping::PingCommand::parse,
  97. Command::Ping,
  98. &original_tokenizer,
  99. ));
  100. success.extend(parse_single_command(
  101. nominate::NominateCommand::parse,
  102. Command::Nominate,
  103. &original_tokenizer,
  104. ));
  105. success.extend(parse_single_command(
  106. prioritize::PrioritizeCommand::parse,
  107. Command::Prioritize,
  108. &original_tokenizer,
  109. ));
  110. success.extend(parse_single_command(
  111. second::SecondCommand::parse,
  112. Command::Second,
  113. &original_tokenizer,
  114. ));
  115. success.extend(parse_single_command(
  116. glacier::GlacierCommand::parse,
  117. Command::Glacier,
  118. &original_tokenizer,
  119. ));
  120. success.extend(parse_single_command(
  121. shortcut::ShortcutCommand::parse,
  122. Command::Shortcut,
  123. &original_tokenizer,
  124. ));
  125. success.extend(parse_single_command(
  126. close::CloseCommand::parse,
  127. Command::Close,
  128. &original_tokenizer,
  129. ));
  130. if success.len() > 1 {
  131. panic!(
  132. "succeeded parsing {:?} to multiple commands: {:?}",
  133. &self.all[self.parsed..],
  134. success
  135. );
  136. }
  137. if self
  138. .ignore
  139. .overlaps_ignore((self.parsed)..(self.parsed + tok.position()))
  140. .is_some()
  141. {
  142. log::info!("command overlaps ignored block; ignore: {:?}", self.ignore);
  143. return None;
  144. }
  145. let (mut tok, c) = success.pop()?;
  146. // if we errored out while parsing the command do not move the input forwards
  147. self.parsed += if c.is_ok() {
  148. tok.position()
  149. } else {
  150. name_length
  151. };
  152. Some(c)
  153. }
  154. }
  155. impl<'a> Iterator for Input<'a> {
  156. type Item = Command<'a>;
  157. fn next(&mut self) -> Option<Command<'a>> {
  158. loop {
  159. let start = self
  160. .bot
  161. .iter()
  162. .filter_map(|name| find_command_start(&self.all[self.parsed..], name))
  163. .min()?;
  164. self.parsed += start;
  165. if let Some(command) = self.parse_command() {
  166. return Some(command);
  167. }
  168. self.parsed += self.bot.len() + 1;
  169. }
  170. }
  171. }
  172. impl<'a> Command<'a> {
  173. pub fn is_ok(&self) -> bool {
  174. match self {
  175. Command::Relabel(r) => r.is_ok(),
  176. Command::Assign(r) => r.is_ok(),
  177. Command::Ping(r) => r.is_ok(),
  178. Command::Nominate(r) => r.is_ok(),
  179. Command::Prioritize(r) => r.is_ok(),
  180. Command::Second(r) => r.is_ok(),
  181. Command::Glacier(r) => r.is_ok(),
  182. Command::Shortcut(r) => r.is_ok(),
  183. Command::Close(r) => r.is_ok(),
  184. Command::Note(r) => r.is_ok(),
  185. }
  186. }
  187. pub fn is_err(&self) -> bool {
  188. !self.is_ok()
  189. }
  190. }
  191. #[test]
  192. fn errors_outside_command_are_fine() {
  193. let input =
  194. "haha\" unterminated quotes @bot modify labels: +bug. Terminating after the command";
  195. let mut input = Input::new(input, vec!["bot"]);
  196. assert!(input.next().unwrap().is_ok());
  197. }
  198. #[test]
  199. fn code_1() {
  200. let input = "`@bot modify labels: +bug.`";
  201. let mut input = Input::new(input, vec!["bot"]);
  202. assert!(input.next().is_none());
  203. }
  204. #[test]
  205. fn code_2() {
  206. let input = "```
  207. @bot modify labels: +bug.
  208. ```";
  209. let mut input = Input::new(input, vec!["bot"]);
  210. assert!(input.next().is_none());
  211. }
  212. #[test]
  213. fn edit_1() {
  214. let input_old = "@bot modify labels: +bug.";
  215. let mut input_old = Input::new(input_old, vec!["bot"]);
  216. let input_new = "Adding labels: @bot modify labels: +bug. some other text";
  217. let mut input_new = Input::new(input_new, vec!["bot"]);
  218. assert_eq!(input_old.next(), input_new.next());
  219. }
  220. #[test]
  221. fn edit_2() {
  222. let input_old = "@bot modify label: +bug.";
  223. let mut input_old = Input::new(input_old, vec!["bot"]);
  224. let input_new = "@bot modify labels: +bug.";
  225. let mut input_new = Input::new(input_new, vec!["bot"]);
  226. assert_ne!(input_old.next(), input_new.next());
  227. }
  228. #[test]
  229. fn move_input_along() {
  230. let input = "@bot modify labels: +bug. Afterwards, delete the world.";
  231. let mut input = Input::new(input, vec!["bot"]);
  232. assert!(input.next().unwrap().is_ok());
  233. assert_eq!(&input.all[input.parsed..], " Afterwards, delete the world.");
  234. }
  235. #[test]
  236. fn move_input_along_1() {
  237. let input = "@bot modify labels\": +bug. Afterwards, delete the world.";
  238. let mut input = Input::new(input, vec!["bot"]);
  239. assert!(input.next().unwrap().is_err());
  240. // don't move input along if parsing the command fails
  241. assert_eq!(&input.all[..input.parsed], "@bot");
  242. }
  243. #[test]
  244. fn multiname() {
  245. let input = "@rustbot modify labels: +bug. Afterwards, delete the world. @triagebot prioritize";
  246. let mut input = Input::new(input, vec!["triagebot", "rustbot"]);
  247. assert!(dbg!(input.next().unwrap()).is_ok());
  248. assert_eq!(
  249. &input.all[input.parsed..],
  250. " Afterwards, delete the world. @triagebot prioritize"
  251. );
  252. assert!(input.next().unwrap().is_ok());
  253. assert!(input.next().is_none());
  254. }