1
0

macros.rs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  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. $key_start:expr,
  10. $key_stop:expr,
  11. $key_step:expr,
  12. $queueable:expr,
  13. }),+$(,)?
  14. }),+$(,)?
  15. }=> {
  16. $($(
  17. #[allow(non_snake_case, non_camel_case_types)]
  18. pub mod $command {
  19. use super::*;
  20. use async_trait::async_trait;
  21. pub struct Command {
  22. pub tags: &'static [&'static str],
  23. pub min_args: i32,
  24. pub key_start: i32,
  25. pub key_stop: i32,
  26. pub key_step: usize,
  27. }
  28. impl Command {
  29. pub fn new() -> Self {
  30. Self {
  31. tags: &[$($tag,)+],
  32. min_args: $min_args,
  33. key_start: $key_start,
  34. key_stop: $key_stop,
  35. key_step: $key_step,
  36. }
  37. }
  38. }
  39. #[async_trait]
  40. impl ExecutableCommand for Command {
  41. async fn execute(&self, conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
  42. if conn.in_transaction() && self.is_queueable() {
  43. conn.queue_command(args);
  44. conn.tx_keys(self.get_keys(args));
  45. Ok(Value::Queued)
  46. } else {
  47. $handler(conn, args).await
  48. }
  49. }
  50. fn is_queueable(&self) -> bool {
  51. $queueable
  52. }
  53. fn get_keys<'a>(&self, args: &'a [Bytes]) -> Vec<&'a Bytes> {
  54. let start = self.key_start;
  55. let stop = if self.key_stop > 0 {
  56. self.key_stop
  57. } else {
  58. (args.len() as i32) + self.key_stop
  59. };
  60. if start == 0 {
  61. return vec![];
  62. }
  63. let mut result = vec![];
  64. for i in (start .. stop+1).step_by(self.key_step) {
  65. result.push(&args[i as usize]);
  66. }
  67. result
  68. }
  69. fn check_number_args(&self, n: usize) -> bool {
  70. if ($min_args >= 0) {
  71. n == ($min_args as i32).try_into().unwrap_or(0)
  72. } else {
  73. let s: usize = ($min_args as i32).abs().try_into().unwrap_or(0);
  74. n >= s
  75. }
  76. }
  77. fn group(&self) -> &'static str {
  78. stringify!($ns)
  79. }
  80. fn name(&self) -> &'static str {
  81. stringify!($command)
  82. }
  83. }
  84. }
  85. )+)+
  86. use async_trait::async_trait;
  87. use std::ops::Deref;
  88. #[async_trait]
  89. pub trait ExecutableCommand {
  90. async fn execute(&self, conn: &Connection, args: &[Bytes]) -> Result<Value, Error>;
  91. fn is_queueable(&self) -> bool;
  92. fn get_keys<'a>(&self, args: &'a [Bytes]) -> Vec<&'a Bytes>;
  93. fn check_number_args(&self, n: usize) -> bool;
  94. fn group(&self) -> &'static str;
  95. fn name(&self) -> &'static str;
  96. }
  97. #[allow(non_snake_case, non_camel_case_types)]
  98. pub enum Dispatcher {
  99. $($(
  100. $command($command::Command),
  101. )+)+
  102. }
  103. impl Dispatcher {
  104. pub fn new(args: &[Bytes]) -> Result<Self, Error> {
  105. let command = String::from_utf8_lossy(&args[0]).to_lowercase();
  106. let command = match command.as_str() {
  107. $($(
  108. stringify!($command) => Ok(Self::$command($command::Command::new())),
  109. )+)+
  110. _ => Err(Error::CommandNotFound(command.into())),
  111. }?;
  112. if ! command.check_number_args(args.len()) {
  113. Err(Error::InvalidArgsCount(command.name().into()))
  114. } else {
  115. Ok(command)
  116. }
  117. }
  118. }
  119. impl Deref for Dispatcher {
  120. type Target = dyn ExecutableCommand + Sync + Send;
  121. fn deref(&self) -> &(dyn ExecutableCommand + Sync + Send + 'static) {
  122. match self {
  123. $($(
  124. Self::$command(v) => v as &(dyn ExecutableCommand + Sync + Send),
  125. )+)+
  126. }
  127. }
  128. }
  129. }
  130. }
  131. #[macro_export]
  132. macro_rules! value_try_from {
  133. {$type: ty, $value: expr} => {
  134. impl From<$type> for Value {
  135. fn from(value: $type) -> Value {
  136. $value(value.into())
  137. }
  138. }
  139. value_vec_try_from!($type);
  140. }
  141. }
  142. #[macro_export]
  143. macro_rules! value_vec_try_from {
  144. {$type: ty} => {
  145. impl From<Vec<$type>> for Value {
  146. fn from(value: Vec<$type>) -> Value {
  147. Value::Array(value.iter().map(|x| (*x).into()).collect())
  148. }
  149. }
  150. }
  151. }
  152. #[macro_export]
  153. macro_rules! option {
  154. {$type: expr} => {
  155. if let Some(val) = $type {
  156. val.into()
  157. } else {
  158. Value::Null
  159. }
  160. }
  161. }
  162. #[macro_export]
  163. macro_rules! check_arg {
  164. {$args: tt, $pos: tt, $command: tt} => {{
  165. match $args.get($pos) {
  166. Some(bytes) => {
  167. String::from_utf8_lossy(&bytes).to_uppercase() == $command
  168. },
  169. None => false,
  170. }
  171. }}
  172. }
  173. #[macro_export]
  174. macro_rules! bytes {
  175. ($content:tt) => {
  176. Bytes::from(&$content[..])
  177. };
  178. }