macros.rs 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. #[macro_export]
  2. macro_rules! dispatcher {
  3. {
  4. $($ns:ident {
  5. $($command:ident {
  6. $handler:expr,
  7. [$($tag:tt)+],
  8. $min_args:expr,
  9. }),+$(,)?
  10. }),+$(,)?
  11. }=> {
  12. $($(
  13. #[allow(non_snake_case, non_camel_case_types)]
  14. pub mod $command {
  15. use super::*;
  16. use async_trait::async_trait;
  17. pub struct Command {
  18. pub tags: &'static [&'static str],
  19. pub min_args: i32,
  20. }
  21. impl Command {
  22. pub fn new() -> Self {
  23. Self {
  24. tags: &[$($tag,)+],
  25. min_args: $min_args,
  26. }
  27. }
  28. }
  29. #[async_trait]
  30. impl ExecutableCommand for Command {
  31. async fn execute(&self, conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
  32. $handler(conn, args).await
  33. }
  34. fn check_number_args(&self, n: usize) -> bool {
  35. if ($min_args >= 0) {
  36. n == ($min_args as i32).try_into().unwrap()
  37. } else {
  38. let s: usize = ($min_args as i32).abs().try_into().unwrap();
  39. n >= s
  40. }
  41. }
  42. fn group(&self) -> &'static str {
  43. stringify!($ns)
  44. }
  45. fn name(&self) -> &'static str {
  46. stringify!($command)
  47. }
  48. }
  49. }
  50. )+)+
  51. use std::ops::Deref;
  52. use async_trait::async_trait;
  53. #[async_trait]
  54. pub trait ExecutableCommand {
  55. async fn execute(&self, conn: &Connection, args: &[Bytes]) -> Result<Value, Error>;
  56. fn check_number_args(&self, n: usize) -> bool;
  57. fn group(&self) -> &'static str;
  58. fn name(&self) -> &'static str;
  59. }
  60. #[allow(non_snake_case, non_camel_case_types)]
  61. pub enum Dispatcher {
  62. $($(
  63. $command($command::Command),
  64. )+)+
  65. }
  66. impl Dispatcher {
  67. pub fn new(args: &[Bytes]) -> Result<Self, Error> {
  68. let command = unsafe { std::str::from_utf8_unchecked(&args[0]) };
  69. let command = match command.to_lowercase().as_str() {
  70. $($(
  71. stringify!($command) => Ok(Self::$command($command::Command::new())),
  72. )+)+
  73. _ => Err(Error::CommandNotFound(command.into())),
  74. }?;
  75. if ! command.check_number_args(args.len()) {
  76. Err(Error::InvalidArgsCount(command.name().into()))
  77. } else {
  78. Ok(command)
  79. }
  80. }
  81. }
  82. impl Deref for Dispatcher {
  83. type Target = dyn ExecutableCommand + Sync + Send;
  84. fn deref(&self) -> &(dyn ExecutableCommand + Sync + Send + 'static) {
  85. match self {
  86. $($(
  87. Self::$command(v) => v as &(dyn ExecutableCommand + Sync + Send),
  88. )+)+
  89. }
  90. }
  91. }
  92. }
  93. }
  94. #[macro_export]
  95. macro_rules! value_try_from {
  96. {$type: ty, $value: expr} => {
  97. impl From<$type> for Value {
  98. fn from(value: $type) -> Value {
  99. $value(value.into())
  100. }
  101. }
  102. value_vec_try_from!($type);
  103. }
  104. }
  105. #[macro_export]
  106. macro_rules! value_vec_try_from {
  107. {$type: ty} => {
  108. impl From<Vec<$type>> for Value {
  109. fn from(value: Vec<$type>) -> Value {
  110. Value::Array(value.iter().map(|x| (*x).into()).collect())
  111. }
  112. }
  113. }
  114. }
  115. #[macro_export]
  116. macro_rules! option {
  117. {$type: expr} => {
  118. if let Some(val) = $type {
  119. val.into()
  120. } else {
  121. Value::Null
  122. }
  123. }
  124. }
  125. #[macro_export]
  126. macro_rules! check_arg {
  127. {$args: tt, $pos: tt, $command: tt} => {{
  128. match $args.get($pos) {
  129. Some(bytes) => {
  130. let command = unsafe { std::str::from_utf8_unchecked(&bytes) };
  131. command.to_uppercase() == $command
  132. },
  133. None => false,
  134. }
  135. }}
  136. }
  137. #[macro_export]
  138. macro_rules! bytes {
  139. ($content:tt) => {
  140. Bytes::from(&$content[..])
  141. };
  142. }