//! # Hash command handlers use crate::{ check_arg, connection::Connection, error::Error, value::{bytes_to_number, float::Float, Value}, }; use bytes::Bytes; use rand::Rng; use std::collections::{BTreeMap, HashMap, VecDeque}; /// Removes the specified fields from the hash stored at key. Specified fields that do not exist /// within this hash are ignored. If key does not exist, it is treated as an empty hash and this /// command returns 0. pub async fn hdel(conn: &Connection, mut args: VecDeque) -> Result { let mut is_empty = false; let key = args.pop_front().ok_or(Error::Syntax)?; let result = conn.db().get_map(&key, |v| match v { Some(Value::Hash(h)) => { let mut h = h.write(); let mut total: i64 = 0; for key in args.into_iter() { if h.remove(&key).is_some() { total += 1; } } is_empty = h.len() == 0; Ok(total.into()) } None => Ok(0.into()), _ => Err(Error::WrongType), })?; if is_empty { let _ = conn.db().del(&[key]); } else { conn.db().bump_version(&key); } Ok(result) } /// Returns if field is an existing field in the hash stored at key. pub async fn hexists(conn: &Connection, args: VecDeque) -> Result { conn.db().get_map(&args[0], |v| match v { Some(Value::Hash(h)) => Ok(if h.read().get(&args[1]).is_some() { 1.into() } else { 0.into() }), None => Ok(0.into()), _ => Err(Error::WrongType), }) } /// Returns the value associated with field in the hash stored at key. pub async fn hget(conn: &Connection, args: VecDeque) -> Result { conn.db().get_map(&args[0], |v| match v { Some(Value::Hash(h)) => Ok(h .read() .get(&args[1]) .map(|v| Value::new(v)) .unwrap_or_default()), None => Ok(Value::Null), _ => Err(Error::WrongType), }) } /// Returns all fields and values of the hash stored at key. In the returned value, every field /// name is followed by its value, so the length of the reply is twice the size of the hash. pub async fn hgetall(conn: &Connection, args: VecDeque) -> Result { conn.db().get_map(&args[0], |v| match v { Some(Value::Hash(h)) => { let mut ret = vec![]; for (key, value) in h.read().iter() { ret.push(Value::new(key)); ret.push(Value::new(value)); } Ok(ret.into()) } None => Ok(Value::Array(vec![])), _ => Err(Error::WrongType), }) } /// Increment the specified field of a hash stored at key, and representing a number, by the /// specified increment. If the increment value is negative, the result is to have the hash field /// value decremented instead of incremented. If the field does not exist, it is set to 0 before /// performing the operation. pub async fn hincrby_int(conn: &Connection, args: VecDeque) -> Result { let result = conn .db() .hincrby::(&args[0], &args[1], &args[2], "an integer")?; conn.db().bump_version(&args[0]); Ok(result) } /// Increment the specified field of a hash stored at key, and representing a number, by the /// specified increment. If the increment value is negative, the result is to have the hash field /// value decremented instead of incremented. If the field does not exist, it is set to 0 before /// performing the operation. pub async fn hincrby_float(conn: &Connection, args: VecDeque) -> Result { let result = conn .db() .hincrby::(&args[0], &args[1], &args[2], "a float")?; conn.db().bump_version(&args[0]); Ok(result) } /// Returns all field names in the hash stored at key. pub async fn hkeys(conn: &Connection, args: VecDeque) -> Result { conn.db().get_map(&args[0], |v| match v { Some(Value::Hash(h)) => { let mut ret = vec![]; for key in h.read().keys() { ret.push(Value::new(key)); } Ok(ret.into()) } None => Ok(Value::Array(vec![])), _ => Err(Error::WrongType), }) } /// Returns the number of fields contained in the hash stored at key. pub async fn hlen(conn: &Connection, args: VecDeque) -> Result { conn.db().get_map(&args[0], |v| match v { Some(Value::Hash(h)) => Ok(h.read().len().into()), None => Ok(0.into()), _ => Err(Error::WrongType), }) } /// Returns the values associated with the specified fields in the hash stored at key. pub async fn hmget(conn: &Connection, mut args: VecDeque) -> Result { let key = args.pop_front().ok_or(Error::Syntax)?; conn.db().get_map(&key, |v| match v { Some(Value::Hash(h)) => { let h = h.read(); Ok(args .iter() .map(|key| h.get(key).map(|v| Value::new(v)).unwrap_or_default()) .collect::>() .into()) } None => Ok(args .iter() .map(|_| Value::Null) .collect::>() .into()), _ => Err(Error::WrongType), }) } /// Returns random keys (or values) from a hash pub async fn hrandfield(conn: &Connection, args: VecDeque) -> Result { let (count, with_values) = match args.len() { 1 => (None, false), 2 => (Some(bytes_to_number::(&args[1])?), false), 3 => { if !(check_arg!(args, 2, "WITHVALUES")) { return Err(Error::Syntax); } (Some(bytes_to_number::(&args[1])?), true) } _ => return Err(Error::InvalidArgsCount("hrandfield".to_owned())), }; let (count, single, repeat) = match count { Some(count) if count > 0 => (count, false, 1), Some(count) => (count.abs(), false, count.abs()), _ => (1, true, 1), }; conn.db().get_map(&args[0], |v| match v { Some(Value::Hash(h)) => { let mut ret = vec![]; let mut i = 0; let mut rand_sorted = BTreeMap::new(); let mut rng = rand::thread_rng(); let h = h.read(); for _ in 0..repeat { for (key, value) in h.iter() { let rand = rng.gen::(); rand_sorted.insert((rand, i), (key, value)); i += 1; } } i = 0; for val in rand_sorted.values() { if single { return Ok(Value::new(val.0)); } if i == count { break; } ret.push(Value::new(val.0)); if with_values { ret.push(Value::new(val.1)); } i += 1; } Ok(ret.into()) } None => Ok(Value::Array(vec![])), _ => Err(Error::WrongType), }) } /// Sets field in the hash stored at key to value. If key does not exist, a new key holding a hash /// is created. If field already exists in the hash, it is overwritten. pub async fn hmset(conn: &Connection, mut args: VecDeque) -> Result { let key = args.pop_front().ok_or(Error::Syntax)?; if args.len() % 2 == 1 { return Err(Error::InvalidArgsCount("hset".to_owned())); } let result = conn.db().get_map(&key, |v| match v { Some(Value::Hash(h)) => { let mut h = h.write(); loop { if args.is_empty() { break; } let key = args.pop_front().ok_or(Error::Syntax)?; let value = args.pop_front().ok_or(Error::Syntax)?; h.insert(key, value); } Ok(Value::Ok) } None => { #[allow(clippy::mutable_key_type)] let mut h = HashMap::new(); loop { if args.is_empty() { break; } let key = args.pop_front().ok_or(Error::Syntax)?; let value = args.pop_front().ok_or(Error::Syntax)?; h.insert(key, value); } let _len = h.len(); conn.db().set(key.clone(), h.into(), None); Ok(Value::Ok) } _ => Err(Error::WrongType), })?; conn.db().bump_version(&key); Ok(result) } /// Sets field in the hash stored at key to value. If key does not exist, a new key holding a hash /// is created. If field already exists in the hash, it is overwritten. pub async fn hset(conn: &Connection, mut args: VecDeque) -> Result { let key = args.pop_front().ok_or(Error::Syntax)?; if args.len() % 2 == 1 { return Err(Error::InvalidArgsCount("hset".to_owned())); } let result = conn.db().get_map(&key, |v| match v { Some(Value::Hash(h)) => { let mut h = h.write(); let mut e: i64 = 0; loop { if args.is_empty() { break; } let key = args.pop_front().ok_or(Error::Syntax)?; let value = args.pop_front().ok_or(Error::Syntax)?; if h.insert(key, value).is_none() { e += 1; } } Ok(e.into()) } None => { #[allow(clippy::mutable_key_type)] let mut h = HashMap::new(); loop { if args.is_empty() { break; } let key = args.pop_front().ok_or(Error::Syntax)?; let value = args.pop_front().ok_or(Error::Syntax)?; h.insert(key, value); } let len = h.len(); conn.db().set(key.clone(), h.into(), None); Ok(len.into()) } _ => Err(Error::WrongType), })?; conn.db().bump_version(&key); Ok(result) } /// Sets field in the hash stored at key to value, only if field does not yet exist. If key does /// not exist, a new key holding a hash is created. If field already exists, this operation has no /// effect. pub async fn hsetnx(conn: &Connection, mut args: VecDeque) -> Result { let key = args.pop_front().ok_or(Error::Syntax)?; let sub_key = args.pop_front().ok_or(Error::Syntax)?; let value = args.pop_front().ok_or(Error::Syntax)?; let result = conn.db().get_map(&key, |v| match v { Some(Value::Hash(h)) => { let mut h = h.write(); if h.get(&sub_key).is_some() { Ok(0.into()) } else { h.insert(sub_key, value); Ok(1.into()) } } None => { #[allow(clippy::mutable_key_type)] let mut h = HashMap::new(); h.insert(sub_key, value); let len = h.len(); conn.db().set(key.clone(), h.into(), None); Ok(len.into()) } _ => Err(Error::WrongType), })?; if result == Value::Integer(1) { conn.db().bump_version(&key); } Ok(result) } /// Returns the string length of the value associated with field in the hash stored at key. If the /// key or the field do not exist, 0 is returned. pub async fn hstrlen(conn: &Connection, args: VecDeque) -> Result { conn.db().get_map(&args[0], |v| match v { Some(Value::Hash(h)) => Ok(h .read() .get(&args[1]) .map(|v| v.len()) .unwrap_or_default() .into()), None => Ok(0.into()), _ => Err(Error::WrongType), }) } /// Returns all values in the hash stored at key. pub async fn hvals(conn: &Connection, args: VecDeque) -> Result { conn.db().get_map(&args[0], |v| match v { Some(Value::Hash(h)) => { let mut ret = vec![]; for value in h.read().values() { ret.push(Value::new(value)); } Ok(ret.into()) } None => Ok(Value::Array(vec![])), _ => Err(Error::WrongType), }) } #[cfg(test)] mod test { use crate::{ cmd::test::{create_connection, invalid_type, run_command}, value::Value, }; #[tokio::test] async fn hget() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1", "f2", "2", "f3", "3"]).await; assert_eq!(Ok(Value::Integer(3)), r); let r = run_command(&c, &["hget", "foo", "f1"]).await; assert_eq!(Ok(Value::Blob("1".into())), r); } #[tokio::test] async fn hgetall() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1", "f2", "2", "f3", "3"]).await; assert_eq!(Ok(Value::Integer(3)), r); let r = run_command(&c, &["hgetall", "foo"]).await; match r { Ok(Value::Array(x)) => { assert_eq!(6, x.len()); assert!( x[0] == Value::Blob("f1".into()) || x[0] == Value::Blob("f2".into()) || x[0] == Value::Blob("f3".into()) ) } _ => unreachable!(), }; } #[tokio::test] async fn hrandfield() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1", "f2", "2", "f3", "3"]).await; assert_eq!(Ok(Value::Integer(3)), r); let r = run_command(&c, &["hrandfield", "foo"]).await; match r { Ok(Value::Blob(x)) => { let x = String::from_utf8_lossy(&x); assert!(x == *"f1" || x == *"f2" || x == *"f3"); } _ => unreachable!(), }; } #[tokio::test] async fn hmget() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1", "f2", "2", "f3", "3"]).await; assert_eq!(Ok(Value::Integer(3)), r); let r = run_command(&c, &["hmget", "foo", "f1", "f2"]).await; assert_eq!( Ok(Value::Array(vec![ Value::Blob("1".into()), Value::Blob("2".into()), ])), r ); } #[tokio::test] async fn hexists() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1", "f2", "2", "f3", "3"]).await; assert_eq!(Ok(Value::Integer(3)), r); assert_eq!( Ok(Value::Integer(1)), run_command(&c, &["hexists", "foo", "f1"]).await ); assert_eq!( Ok(Value::Integer(1)), run_command(&c, &["hexists", "foo", "f3"]).await ); assert_eq!( Ok(Value::Integer(0)), run_command(&c, &["hexists", "foo", "f4"]).await ); } #[tokio::test] async fn hstrlen() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1", "f2", "2", "f3", "3"]).await; assert_eq!(Ok(Value::Integer(3)), r); let r = run_command(&c, &["hstrlen", "foo", "f1"]).await; assert_eq!(Ok(Value::Integer(1)), r); } #[tokio::test] async fn hlen() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1", "f2", "2", "f3", "3"]).await; assert_eq!(Ok(Value::Integer(3)), r); let r = run_command(&c, &["hset", "foo", "f1", "2", "f4", "2", "f5", "3"]).await; assert_eq!(Ok(Value::Integer(2)), r); let r = run_command(&c, &["hlen", "foo"]).await; assert_eq!(Ok(Value::Integer(5)), r); } #[tokio::test] async fn hkeys() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1"]).await; assert_eq!(Ok(Value::Integer(1)), r); let r = run_command(&c, &["hkeys", "foo"]).await; assert_eq!(Ok(Value::Array(vec![Value::Blob("f1".into()),])), r); } #[tokio::test] async fn hvals() { let c = create_connection(); let r = run_command(&c, &["hset", "foo", "f1", "1"]).await; assert_eq!(Ok(Value::Integer(1)), r); let r = run_command(&c, &["hvals", "foo"]).await; assert_eq!(Ok(Value::Array(vec![Value::Blob("1".into()),])), r); } #[tokio::test] async fn hdel_remove_empty_hash() { let c = create_connection(); assert_eq!( Ok(Value::Integer(2)), run_command(&c, &["hset", "foo", "f1", "1", "f2", "1"]).await ); assert_eq!(Ok(1.into()), run_command(&c, &["hdel", "foo", "f1",]).await); assert_eq!( Ok(Value::Integer(-1)), run_command(&c, &["ttl", "foo"]).await ); assert_eq!(Ok(1.into()), run_command(&c, &["hdel", "foo", "f2",]).await); assert_eq!( Ok(Value::Integer(-2)), run_command(&c, &["ttl", "foo"]).await ); } #[tokio::test] async fn hincrby() { let c = create_connection(); assert_eq!( Ok(Value::Integer(1)), run_command(&c, &["hincrby", "foo", "f1", "1"]).await ); assert_eq!( Ok(Value::Integer(-9)), run_command(&c, &["hincrby", "foo", "f1", "-10"]).await ); assert_eq!( Ok(Value::Blob("-9".into())), run_command(&c, &["hget", "foo", "f1"]).await ); } #[tokio::test] async fn hsetnx() { let c = create_connection(); assert_eq!( Ok(Value::Integer(1)), run_command(&c, &["hsetnx", "foo", "xxx", "1"]).await ); assert_eq!( Ok(Value::Integer(0)), run_command(&c, &["hsetnx", "foo", "xxx", "1"]).await ); assert_eq!( Ok(Value::Integer(1)), run_command(&c, &["hsetnx", "foo", "bar", "1"]).await ); assert_eq!( Ok(Value::Integer(2)), run_command(&c, &["hlen", "foo"]).await ); } #[tokio::test] async fn hlen_non_existing() { let c = create_connection(); assert_eq!( Ok(Value::Integer(0)), run_command(&c, &["hlen", "foo"]).await ); } #[tokio::test] async fn invalid_types() { invalid_type(&["hdel", "key", "bar", "1"]).await; invalid_type(&["hexists", "key", "bar"]).await; invalid_type(&["hget", "key", "bar"]).await; invalid_type(&["hgetall", "key"]).await; invalid_type(&["hincrby", "key", "bar", "1"]).await; invalid_type(&["hincrbyfloat", "key", "bar", "1"]).await; invalid_type(&["hkeys", "key"]).await; invalid_type(&["hlen", "key"]).await; invalid_type(&["hstrlen", "key", "foo"]).await; invalid_type(&["hmget", "key", "1", "2"]).await; invalid_type(&["hrandfield", "key"]).await; invalid_type(&["hset", "key", "bar", "1"]).await; invalid_type(&["hsetnx", "key", "bar", "1"]).await; invalid_type(&["hvals", "key"]).await; } }