utils.rs 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. use crate::error::Error;
  2. use bytes::Bytes;
  3. use std::convert::{TryFrom, TryInto};
  4. use tokio::time::{Duration, Instant};
  5. pub(crate) fn far_future() -> Instant {
  6. Instant::now() + Duration::from_secs(86400 * 365 * 30)
  7. }
  8. /// Override database entries
  9. #[derive(PartialEq, Debug, Clone, Copy)]
  10. pub enum Override {
  11. /// Allow override
  12. Yes,
  13. /// Do not allow override, only new entries
  14. No,
  15. /// Allow only override
  16. Only,
  17. }
  18. impl From<bool> for Override {
  19. fn from(v: bool) -> Override {
  20. if v {
  21. Override::Yes
  22. } else {
  23. Override::No
  24. }
  25. }
  26. }
  27. impl Default for Override {
  28. fn default() -> Self {
  29. Self::Yes
  30. }
  31. }
  32. /// Override database entries
  33. #[derive(PartialEq, Debug, Clone, Copy, Default)]
  34. pub struct ExpirationOpts {
  35. /// Set expiry only when the key has no expiry
  36. pub if_none: bool,
  37. /// Set expiry only when the key has an existing expiry
  38. pub replace_only: bool,
  39. /// Set expiry only when the new expiry is greater than current one
  40. pub greater_than: bool,
  41. /// Set expiry only when the new expiry is less than current one
  42. pub lower_than: bool,
  43. }
  44. impl TryFrom<Vec<Bytes>> for ExpirationOpts {
  45. type Error = Error;
  46. fn try_from(args: Vec<Bytes>) -> Result<Self, Self::Error> {
  47. args.as_slice().try_into()
  48. }
  49. }
  50. impl TryFrom<&[Bytes]> for ExpirationOpts {
  51. type Error = Error;
  52. fn try_from(args: &[Bytes]) -> Result<Self, Self::Error> {
  53. let mut expiration_opts = Self::default();
  54. for arg in args.iter() {
  55. match String::from_utf8_lossy(arg).to_uppercase().as_str() {
  56. "NX" => expiration_opts.if_none = true,
  57. "XX" => expiration_opts.replace_only = true,
  58. "GT" => expiration_opts.greater_than = true,
  59. "LT" => expiration_opts.lower_than = true,
  60. invalid => return Err(Error::UnsupportedOption(invalid.to_owned())),
  61. }
  62. }
  63. Ok(expiration_opts)
  64. }
  65. }
  66. #[cfg(test)]
  67. mod test {
  68. use super::*;
  69. use std::convert::TryInto;
  70. #[test]
  71. fn parsing_expiration_1() {
  72. let opts = vec![
  73. Bytes::copy_from_slice(b"nx"),
  74. Bytes::copy_from_slice(b"Xx"),
  75. Bytes::copy_from_slice(b"GT"),
  76. Bytes::copy_from_slice(b"lT"),
  77. ];
  78. let x: ExpirationOpts = opts.as_slice().try_into().unwrap();
  79. assert!(x.if_none);
  80. assert!(x.replace_only);
  81. assert!(x.greater_than);
  82. assert!(x.lower_than);
  83. }
  84. #[test]
  85. fn parsing_expiration_2() {
  86. let opts = vec![Bytes::copy_from_slice(b"nx")];
  87. let x: ExpirationOpts = opts.as_slice().try_into().unwrap();
  88. assert!(x.if_none);
  89. assert!(!x.replace_only);
  90. assert!(!x.greater_than);
  91. assert!(!x.lower_than);
  92. }
  93. #[test]
  94. fn parsing_expiration_3() {
  95. let opts = vec![Bytes::copy_from_slice(b"xxx")];
  96. let x: Result<ExpirationOpts, _> = opts.as_slice().try_into();
  97. assert!(x.is_err());
  98. }
  99. }