From b27e72f5ac588506941b74c1f421235eb59e69d5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Julian=20J=C3=B8rgensen=20Teule?= Date: Tue, 5 Mar 2024 16:13:07 +0100 Subject: Initial commit --- src/commands/mod.rs | 2 ++ src/commands/parser.rs | 77 ++++++++++++++++++++++++++++++++++++++++++++++++++ src/commands/start.rs | 12 ++++++++ 3 files changed, 91 insertions(+) create mode 100644 src/commands/mod.rs create mode 100644 src/commands/parser.rs create mode 100644 src/commands/start.rs (limited to 'src/commands') diff --git a/src/commands/mod.rs b/src/commands/mod.rs new file mode 100644 index 0000000..5a61ce8 --- /dev/null +++ b/src/commands/mod.rs @@ -0,0 +1,2 @@ +mod start; +mod parser; diff --git a/src/commands/parser.rs b/src/commands/parser.rs new file mode 100644 index 0000000..e75840c --- /dev/null +++ b/src/commands/parser.rs @@ -0,0 +1,77 @@ + +pub enum ParseResult<'a, T> { + Success(T, &'a str), + Failure(String), +} + +pub trait Parse<'a> { + type Res; + + fn parse(self, input: &'a str) -> ParseResult<'a, Self::Res>; +} + +pub struct Text(String); + +impl<'a> Parse<'a> for Text { + type Res = Text; + + fn parse(self, input: &'a str) -> ParseResult<'a, Text> { + let mut value = String::new(); + + let mut it = input.chars(); + let mut escape_next = false; + while let Some(c) = it.next() { + if c == '\\' && !escape_next { + escape_next = true; + continue; + } else if c == ';' { + break; + } + value.push(c); + + escape_next = false; + } + + ParseResult::Success(Text(value), it.as_str()) + } +} + +pub struct Branch { + choices: Vec<(&'static str, Option, T)>, +} + +impl<'a, T> Parse<'a> for Branch { + type Res = T; + + fn parse(self, input: &'a str) -> ParseResult<'a, T> { + let choices: Vec<&'static str> = self.choices.iter().map(|x| x.0).collect(); + + for (long, short_op, res) in self.choices { + let rest = if input.starts_with(long) { + Some(&input[long.len()..]) + } else if let Some(short) = short_op { + if input.starts_with(short) { + Some(&input[1..]) + } else { + None + } + } else { None }; + + if let Some(rest) = rest { + return ParseResult::Success(res, rest); + } + } + + return ParseResult::Failure(format!("Expected one of [{}]", choices.join(", "))); + } +} + +impl Branch { + pub fn new() -> Branch { + Branch { choices: Vec::new() } + } + + pub fn add(&mut self, long: &'static str, short: Option, res: T) { + self.choices.push((long, short, res)); + } +} diff --git a/src/commands/start.rs b/src/commands/start.rs new file mode 100644 index 0000000..a122d5c --- /dev/null +++ b/src/commands/start.rs @@ -0,0 +1,12 @@ +use crate::commands::parser::{Text, Parse}; + +pub enum Commands { + Start(Text), +} + +impl<'a> Parse<'a> for Commands { + type Res = Commands; + + fn parse(self +} + -- cgit v1.2.3