Przeglądaj źródła

Added `ltrim` support

Cesar Rodas 3 lat temu
rodzic
commit
0d848b7472
2 zmienionych plików z 61 dodań i 0 usunięć
  1. 56 0
      src/cmd/list.rs
  2. 5 0
      src/dispatcher.rs

+ 56 - 0
src/cmd/list.rs

@@ -396,6 +396,38 @@ pub async fn lset(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
     )
     )
 }
 }
 
 
+pub async fn ltrim(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 x = x.write();
+
+                if start < 0 {
+                    start += x.len() as i64;
+                }
+
+                if end < 0 {
+                    end += x.len() as i64;
+                }
+
+                let mut i = 0;
+                x.retain(|_| {
+                    let retain = i >= start && i <= end;
+                    i += 1;
+                    retain
+                });
+
+                Ok(Value::OK)
+            }
+            _ => Err(Error::WrongType),
+        },
+        || Ok(Value::OK),
+    )
+}
+
 pub async fn rpop(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
 pub async fn rpop(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
     let count = if args.len() > 2 {
     let count = if args.len() > 2 {
         bytes_to_number(&args[2])?
         bytes_to_number(&args[2])?
@@ -1111,6 +1143,30 @@ mod test {
     }
     }
 
 
     #[tokio::test]
     #[tokio::test]
+    async fn ltrim() {
+        let c = create_connection();
+
+        assert_eq!(
+            Ok(Value::Integer(5)),
+            run_command(&c, &["rpush", "foo", "1", "2", "3", "4", "5"]).await
+        );
+
+        assert_eq!(
+            Ok(Value::OK),
+            run_command(&c, &["ltrim", "foo", "1", "-2"]).await
+        );
+
+        assert_eq!(
+            Ok(Value::Array(vec![
+                Value::Blob("2".into()),
+                Value::Blob("3".into()),
+                Value::Blob("4".into()),
+            ])),
+            run_command(&c, &["lrange", "foo", "0", "-1"]).await
+        );
+    }
+
+    #[tokio::test]
     async fn rpop() {
     async fn rpop() {
         let c = create_connection();
         let c = create_connection();
 
 

+ 5 - 0
src/dispatcher.rs

@@ -83,6 +83,11 @@ dispatcher! {
             [""],
             [""],
             4,
             4,
         },
         },
+        ltrim {
+            cmd::list::ltrim,
+            [""],
+            4,
+        },
         rpop {
         rpop {
             cmd::list::rpop,
             cmd::list::rpop,
             [""],
             [""],