Ver código fonte

Added list commands

Added commands:

* llen
* lpop
* lpush
* lpushx
* lrange
* rpush
* rpushx
Cesar Rodas 3 anos atrás
pai
commit
554ead0ee2
3 arquivos alterados com 478 adições e 6 exclusões
  1. 453 0
      src/cmd/list.rs
  2. 25 0
      src/dispatcher.rs
  3. 0 6
      src/value/mod.rs

+ 453 - 0
src/cmd/list.rs

@@ -0,0 +1,453 @@
+use crate::{
+    check_arg, connection::Connection, error::Error, value::bytes_to_number, value::checksum,
+    value::Value,
+};
+use bytes::Bytes;
+use std::collections::LinkedList;
+
+fn remove_element(
+    conn: &Connection,
+    key: &Bytes,
+    count: usize,
+    front: bool,
+) -> Result<Value, Error> {
+    conn.db().get_map_or(
+        &key,
+        |v| match v {
+            Value::List(x) => {
+                let mut x = x.write();
+
+                if count == 0 {
+                    // Return a single element
+                    return Ok((if front { x.pop_front() } else { x.pop_back() })
+                        .map_or(Value::Null, |x| x.clone_value()));
+                }
+
+                let mut ret = vec![None; count];
+
+                for i in 0..count {
+                    if front {
+                        ret[i] = x.pop_front();
+                    } else {
+                        ret[i] = x.pop_back();
+                    }
+                }
+
+                let ret: Vec<Value> = ret
+                    .iter()
+                    .filter(|v| v.is_some())
+                    .map(|x| x.as_ref().unwrap().clone_value())
+                    .collect();
+
+                Ok(if ret.len() == 0 {
+                    Value::Null
+                } else {
+                    ret.into()
+                })
+            }
+            _ => Err(Error::WrongType),
+        },
+        || Ok(Value::Null),
+    )
+}
+
+
+pub fn llen(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
+    conn.db().get_map_or(
+        &args[1],
+        |v| match v {
+            Value::List(x) => Ok((x.read().len() as i64).into()),
+            _ => Err(Error::WrongType),
+        },
+        || Ok(0.into()),
+    )
+}
+
+pub fn lpop(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
+    let count = if args.len() > 2 {
+        bytes_to_number(&args[2])?
+    } else {
+        0
+    };
+
+    remove_element(conn, &args[1], count, true)
+}
+
+pub fn lpush(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
+    let is_push_x = check_arg!(args, 0, "LPUSHX");
+
+    conn.db().get_map_or(
+        &args[1],
+        |v| match v {
+            Value::List(x) => {
+                let mut x = x.write();
+                for val in args.iter().skip(2) {
+                    x.push_front(checksum::Value::new(val.clone()));
+                }
+                Ok((x.len() as i64).into())
+            }
+            _ => Err(Error::WrongType),
+        },
+        || {
+            if is_push_x {
+                return Ok(0.into());
+            }
+            let mut h = LinkedList::new();
+
+            for val in args.iter().skip(2) {
+                h.push_front(checksum::Value::new(val.clone()));
+            }
+
+            let len = h.len() as i64;
+            conn.db().set(&args[1], h.into(), None);
+            Ok(len.into())
+        },
+    )
+}
+
+pub fn lrange(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
+    conn.db().get_map_or(
+        &args[1],
+        |v| match v {
+            Value::List(x) => {
+                let mut start: i64 = bytes_to_number(&args[2])?;
+                let mut end: i64 = bytes_to_number(&args[3])?;
+                let mut ret = vec![];
+                let x = x.read();
+
+                if start < 0 {
+                    start += x.len() as i64;
+                }
+
+                if end < 0 {
+                    end += x.len() as i64;
+                }
+
+                for (i, val) in x.iter().enumerate() {
+                    if i >= start as usize && i <= end as usize {
+                        ret.push(val.clone_value());
+                    }
+                }
+                Ok(ret.into())
+            }
+            _ => Err(Error::WrongType),
+        },
+        || Ok(Value::Array(vec![])),
+    )
+}
+
+pub fn rpush(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
+    let is_push_x = check_arg!(args, 0, "RPUSHX");
+
+    conn.db().get_map_or(
+        &args[1],
+        |v| match v {
+            Value::List(x) => {
+                let mut x = x.write();
+                for val in args.iter().skip(2) {
+                    x.push_back(checksum::Value::new(val.clone()));
+                }
+                Ok((x.len() as i64).into())
+            }
+            _ => Err(Error::WrongType),
+        },
+        || {
+            if is_push_x {
+                return Ok(0.into());
+            }
+            let mut h = LinkedList::new();
+
+            for val in args.iter().skip(2) {
+                h.push_back(checksum::Value::new(val.clone()));
+            }
+
+            let len = h.len() as i64;
+            conn.db().set(&args[1], h.into(), None);
+            Ok(len.into())
+        },
+    )
+}
+
+#[cfg(test)]
+mod test {
+    use crate::{
+        cmd::test::{create_connection, run_command},
+        value::Value,
+    };
+
+    #[test]
+    fn llen() {
+        let c = create_connection();
+
+        assert_eq!(
+            run_command(&c, &["lpush", "foo", "1", "2", "3", "4", "5"]),
+            run_command(&c, &["llen", "foo"])
+        );
+
+        assert_eq!(Ok(Value::Integer(0)), run_command(&c, &["llen", "foobar"]));
+    }
+
+    #[test]
+    fn lpop() {
+        let c = create_connection();
+
+        assert_eq!(
+            Ok(Value::Integer(5)),
+            run_command(&c, &["lpush", "foo", "1", "2", "3", "4", "5"]),
+        );
+
+        assert_eq!(
+            Ok(Value::Blob("5".into())),
+            run_command(&c, &["lpop", "foo"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("4".into())
+            ])),
+            run_command(&c, &["lpop", "foo", "1"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("3".into()),
+                Value::Blob("2".into()),
+                Value::Blob("1".into()),
+            ])),
+            run_command(&c, &["lpop", "foo", "55"])
+        );
+
+        assert_eq!(
+            Ok(Value::Null),
+            run_command(&c, &["lpop", "foo", "55"])
+        );
+
+        assert_eq!(
+            Ok(Value::Null),
+            run_command(&c, &["lpop", "foo"])
+        );
+
+        assert_eq!(Ok(Value::Integer(0)), run_command(&c, &["llen", "foobar"]));
+    }
+
+    #[test]
+    fn lpush() {
+        let c = create_connection();
+
+        assert_eq!(
+            Ok(Value::Integer(5)),
+            run_command(&c, &["lpush", "foo", "1", "2", "3", "4", "5"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("5".into()),
+                Value::Blob("4".into()),
+                Value::Blob("3".into()),
+                Value::Blob("2".into()),
+                Value::Blob("1".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+
+        assert_eq!(
+            Ok(Value::Integer(10)),
+            run_command(&c, &["lpush", "foo", "6", "7", "8", "9", "10"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("10".into()),
+                Value::Blob("9".into()),
+                Value::Blob("8".into()),
+                Value::Blob("7".into()),
+                Value::Blob("6".into()),
+                Value::Blob("5".into()),
+                Value::Blob("4".into()),
+                Value::Blob("3".into()),
+                Value::Blob("2".into()),
+                Value::Blob("1".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+    }
+
+    #[test]
+    fn lpush_simple() {
+        let c = create_connection();
+
+        assert_eq!(
+            Ok(Value::Integer(1)),
+            run_command(&c, &["lpush", "foo", "world"])
+        );
+
+        assert_eq!(
+            Ok(Value::Integer(2)),
+            run_command(&c, &["lpush", "foo", "hello"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("hello".into()),
+                Value::Blob("world".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+    }
+
+    #[test]
+    fn rpush_simple() {
+        let c = create_connection();
+
+        assert_eq!(
+            Ok(Value::Integer(1)),
+            run_command(&c, &["rpush", "foo", "world"])
+        );
+
+        assert_eq!(
+            Ok(Value::Integer(2)),
+            run_command(&c, &["rpush", "foo", "hello"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("world".into()),
+                Value::Blob("hello".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+    }
+
+    #[test]
+    fn lrange() {
+        let c = create_connection();
+
+        assert_eq!(
+            Ok(Value::Integer(5)),
+            run_command(&c, &["rpush", "foo", "1", "2", "3", "4", "5"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("1".into()),
+                Value::Blob("2".into()),
+                Value::Blob("3".into()),
+                Value::Blob("4".into()),
+                Value::Blob("5".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("1".into()),
+                Value::Blob("2".into()),
+                Value::Blob("3".into()),
+                Value::Blob("4".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-2"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("4".into()),
+                Value::Blob("5".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "-2", "-1"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![Value::Blob("3".into()),])),
+            run_command(&c, &["lrange", "foo", "-3", "-3"])
+        );
+    }
+
+    #[test]
+    fn rpush() {
+        let c = create_connection();
+
+        assert_eq!(
+            Ok(Value::Integer(5)),
+            run_command(&c, &["rpush", "foo", "1", "2", "3", "4", "5"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("1".into()),
+                Value::Blob("2".into()),
+                Value::Blob("3".into()),
+                Value::Blob("4".into()),
+                Value::Blob("5".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+
+        assert_eq!(
+            Ok(Value::Integer(10)),
+            run_command(&c, &["rpush", "foo", "6", "7", "8", "9", "10"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("1".into()),
+                Value::Blob("2".into()),
+                Value::Blob("3".into()),
+                Value::Blob("4".into()),
+                Value::Blob("5".into()),
+                Value::Blob("6".into()),
+                Value::Blob("7".into()),
+                Value::Blob("8".into()),
+                Value::Blob("9".into()),
+                Value::Blob("10".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+    }
+
+    #[test]
+    fn rpushx() {
+        let c = create_connection();
+
+        assert_eq!(
+            Ok(Value::Integer(5)),
+            run_command(&c, &["rpush", "foo", "1", "2", "3", "4", "5"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("1".into()),
+                Value::Blob("2".into()),
+                Value::Blob("3".into()),
+                Value::Blob("4".into()),
+                Value::Blob("5".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+
+        assert_eq!(
+            Ok(Value::Integer(10)),
+            run_command(&c, &["rpushx", "foo", "6", "7", "8", "9", "10"])
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("1".into()),
+                Value::Blob("2".into()),
+                Value::Blob("3".into()),
+                Value::Blob("4".into()),
+                Value::Blob("5".into()),
+                Value::Blob("6".into()),
+                Value::Blob("7".into()),
+                Value::Blob("8".into()),
+                Value::Blob("9".into()),
+                Value::Blob("10".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"])
+        );
+
+        assert_eq!(
+            Ok(Value::Integer(0)),
+            run_command(&c, &["rpushx", "foobar", "6", "7", "8", "9", "10"])
+        );
+    }
+}

+ 25 - 0
src/dispatcher.rs

@@ -23,6 +23,26 @@ fn do_command(_conn: &Connection, _args: &[Bytes]) -> Result<Value, Error> {
 
 dispatcher! {
     list {
+        llen {
+            cmd::list::llen,
+            [""],
+            2,
+        },
+        lpop {
+            cmd::list::lpop,
+            [""],
+            -2,
+        },
+        lpush {
+            cmd::list::lpush,
+            [""],
+            -3,
+        },
+        lpushx {
+            cmd::list::lpush,
+            [""],
+            -3,
+        },
         lrange {
             cmd::list::lrange,
             [""],
@@ -33,6 +53,11 @@ dispatcher! {
             [""],
             -3,
         },
+        rpushx {
+            cmd::list::rpush,
+            [""],
+            -3,
+        },
     },
     hash {
         hdel {

+ 0 - 6
src/value/mod.rs

@@ -11,12 +11,6 @@ use std::{
 };
 
 #[derive(Debug, PartialEq, Clone)]
-/// Stores binary data as Bytes but has built-in verification to make
-/// comparisions faster. The verification (CRC32) is a quicker process to
-/// compare two objects, which will return MAYBE or NOT.
-pub struct BytesWithChecksum((Bytes, u32));
-
-#[derive(Debug, PartialEq, Clone)]
 pub enum Value {
     Hash(locked::Value<HashMap<Bytes, Bytes>>),
     List(locked::Value<LinkedList<checksum::Value>>),