Cesar Rodas 3 роки тому
батько
коміт
f881ccce3c
3 змінених файлів з 89 додано та 0 видалено
  1. 80 0
      src/cmd/list.rs
  2. 5 0
      src/dispatcher.rs
  3. 4 0
      src/error.rs

+ 80 - 0
src/cmd/list.rs

@@ -371,6 +371,31 @@ pub async fn lrem(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
     )
 }
 
+pub async fn lset(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
+    conn.db().get_map_or(
+        &args[1],
+        |v| match v {
+            Value::List(x) => {
+                let mut index: i64 = bytes_to_number(&args[2])?;
+                let mut x = x.write();
+
+                if index < 0 {
+                    index += x.len() as i64;
+                }
+
+                if let Some(x) = x.get_mut(index as usize) {
+                    *x = checksum::Value::new(args[3].clone());
+                    Ok(Value::OK)
+                } else {
+                    Err(Error::OutOfRange)
+                }
+            }
+            _ => Err(Error::WrongType),
+        },
+        || Err(Error::NotFound),
+    )
+}
+
 pub async fn rpop(conn: &Connection, args: &[Bytes]) -> Result<Value, Error> {
     let count = if args.len() > 2 {
         bytes_to_number(&args[2])?
@@ -1031,6 +1056,61 @@ mod test {
     }
 
     #[tokio::test]
+    async fn lset() {
+        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, &["lset", "foo", "-1", "6"]).await,
+        );
+
+        assert_eq!(
+            Ok(Value::OK),
+            run_command(&c, &["lset", "foo", "-2", "7"]).await,
+        );
+
+        assert_eq!(
+            Ok(Value::OK),
+            run_command(&c, &["lset", "foo", "0", "8"]).await,
+        );
+
+        assert_eq!(
+            Err(Error::OutOfRange),
+            run_command(&c, &["lset", "foo", "55", "8"]).await,
+        );
+
+        assert_eq!(
+            Err(Error::OutOfRange),
+            run_command(&c, &["lset", "foo", "-55", "8"]).await,
+        );
+
+        assert_eq!(
+            Err(Error::NotFound),
+            run_command(&c, &["lset", "key_not_exists", "-55", "8"]).await,
+        );
+
+        assert_eq!(
+            Ok(Value::Blob("6".into())),
+            run_command(&c, &["rpop", "foo"]).await
+        );
+
+        assert_eq!(
+            Ok(Value::Blob("7".into())),
+            run_command(&c, &["rpop", "foo"]).await
+        );
+
+        assert_eq!(
+            Ok(Value::Blob("8".into())),
+            run_command(&c, &["lpop", "foo"]).await
+        );
+    }
+
+    #[tokio::test]
     async fn rpop() {
         let c = create_connection();
 

+ 5 - 0
src/dispatcher.rs

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

+ 4 - 0
src/error.rs

@@ -6,6 +6,8 @@ pub enum Error {
     InvalidArgsCount(String),
     Protocol(String, String),
     WrongArgument(String, String),
+    NotFound,
+    OutOfRange,
     Syntax,
     NotANumber,
     WrongType,
@@ -23,7 +25,9 @@ impl From<Error> for Value {
             Error::InvalidArgsCount(x) => format!("wrong number of arguments for '{}' command", x),
             Error::Protocol(x, y) => format!("Protocol error: expected '{}', got '{}'", x, y),
             Error::NotANumber => "value is not an integer or out of range".to_owned(),
+            Error::OutOfRange => "index out of range".to_owned(),
             Error::Syntax => "syntax error".to_owned(),
+            Error::NotFound => "no such key".to_owned(),
             Error::WrongArgument(x, y) => format!(
                 "Unknown subcommand or wrong number of arguments for '{}'. Try {} HELP.",
                 y, x