|
@@ -4,6 +4,7 @@ macro_rules! dispatcher {
|
|
|
$($command:ident {
|
|
|
$handler:ident,
|
|
|
[$($tag:tt)+],
|
|
|
+ $min_args:expr,
|
|
|
},)+$(,)?
|
|
|
}=> {
|
|
|
$(
|
|
@@ -13,19 +14,30 @@ macro_rules! dispatcher {
|
|
|
|
|
|
pub struct Command {
|
|
|
pub tags: &'static [&'static str],
|
|
|
+ pub min_args: i32,
|
|
|
}
|
|
|
|
|
|
impl Command {
|
|
|
pub fn new() -> Self {
|
|
|
Self {
|
|
|
tags: &[$($tag,)+],
|
|
|
+ min_args: $min_args,
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
impl ExecutableCommand for Command {
|
|
|
- fn execute(&self, args: &[Value]) -> Result<Value, String> {
|
|
|
- $handler(args)
|
|
|
+ fn execute(&self, db: &Db, args: &[Value]) -> Result<Value, Error> {
|
|
|
+ $handler(db, args)
|
|
|
+ }
|
|
|
+
|
|
|
+ fn check_number_args(&self, n: usize) -> bool {
|
|
|
+ if ($min_args >= 0) {
|
|
|
+ n == ($min_args as i32).try_into().unwrap()
|
|
|
+ } else {
|
|
|
+ let s: usize = ($min_args as i32).abs().try_into().unwrap();
|
|
|
+ n >= s
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
fn name(&self) -> &'static str {
|
|
@@ -37,7 +49,9 @@ macro_rules! dispatcher {
|
|
|
use std::ops::Deref;
|
|
|
|
|
|
pub trait ExecutableCommand {
|
|
|
- fn execute(&self, args: &[Value]) -> Result<Value, String>;
|
|
|
+ fn execute(&self, db: &Db, args: &[Value]) -> Result<Value, Error>;
|
|
|
+
|
|
|
+ fn check_number_args(&self, n: usize) -> bool;
|
|
|
|
|
|
fn name(&self) -> &'static str;
|
|
|
}
|
|
@@ -50,18 +64,24 @@ macro_rules! dispatcher {
|
|
|
}
|
|
|
|
|
|
impl Dispatcher {
|
|
|
- pub fn new(command: &Value) -> Result<Self, String> {
|
|
|
- let command = match command {
|
|
|
+ pub fn new(args: &[Value]) -> Result<Self, Error> {
|
|
|
+ let command = match &args[0] {
|
|
|
Value::String(x) => Ok(x.as_str()),
|
|
|
Value::Blob(x) => Ok(unsafe { std::str::from_utf8_unchecked(&x) }),
|
|
|
- _ => Err("Invalid type"),
|
|
|
+ _ => Err(Error::ProtocolError("$".to_string(), "*".to_string())),
|
|
|
}?;
|
|
|
|
|
|
- match command.to_lowercase().as_str() {
|
|
|
+ let command = match command.to_lowercase().as_str() {
|
|
|
$(
|
|
|
stringify!($command) => Ok(Self::$command($command::Command::new())),
|
|
|
)+
|
|
|
- _ => Err(format!("Command ({}) not found", command)),
|
|
|
+ _ => Err(Error::CommandNotFound(command.into())),
|
|
|
+ }?;
|
|
|
+
|
|
|
+ if ! command.check_number_args(args.len()) {
|
|
|
+ Err(Error::InvalidArgsCount(command.name().into()))
|
|
|
+ } else {
|
|
|
+ Ok(command)
|
|
|
}
|
|
|
}
|
|
|
}
|