| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  | use std::{cmp::Reverse, fmt::Debug};
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use bytes::Bytes;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub struct Key<T: AsRef<[u8]>>(T, u64);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub type KeySlice<'a> = Key<&'a [u8]>;
 | 
					
						
							|  |  |  | pub type KeyVec = Key<Vec<u8>>;
 | 
					
						
							|  |  |  | pub type KeyBytes = Key<Bytes>;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  | /// For testing purpose, should not use anywhere in your implementation.
 | 
					
						
							|  |  |  | pub const TS_ENABLED: bool = true;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  | /// Temporary, should remove after implementing full week 3 day 1 + 2.
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  | pub const TS_DEFAULT: u64 = 0;
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | pub const TS_MAX: u64 = std::u64::MAX;
 | 
					
						
							|  |  |  | pub const TS_MIN: u64 = std::u64::MIN;
 | 
					
						
							|  |  |  | pub const TS_RANGE_BEGIN: u64 = std::u64::MAX;
 | 
					
						
							|  |  |  | pub const TS_RANGE_END: u64 = std::u64::MIN;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]>> Key<T> {
 | 
					
						
							|  |  |  |     pub fn into_inner(self) -> T {
 | 
					
						
							|  |  |  |         self.0
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn key_len(&self) -> usize {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |         self.0.as_ref().len()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn raw_len(&self) -> usize {
 | 
					
						
							|  |  |  |         self.0.as_ref().len() + std::mem::size_of::<u64>()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     pub fn is_empty(&self) -> bool {
 | 
					
						
							|  |  |  |         self.0.as_ref().is_empty()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl Key<Vec<u8>> {
 | 
					
						
							|  |  |  |     pub fn new() -> Self {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |         Self(Vec::new(), TS_DEFAULT)
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     /// Create a `KeyVec` from a `Vec<u8>` and a ts. Will be removed in week 3.
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     pub fn from_vec_with_ts(key: Vec<u8>, ts: u64) -> Self {
 | 
					
						
							|  |  |  |         Self(key, ts)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Clears the key and set ts to 0.
 | 
					
						
							|  |  |  |     pub fn clear(&mut self) {
 | 
					
						
							|  |  |  |         self.0.clear()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Append a slice to the end of the key
 | 
					
						
							|  |  |  |     pub fn append(&mut self, data: &[u8]) {
 | 
					
						
							|  |  |  |         self.0.extend(data)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn set_ts(&mut self, ts: u64) {
 | 
					
						
							|  |  |  |         self.1 = ts;
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Set the key from a slice without re-allocating.
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     pub fn set_from_slice(&mut self, key_slice: KeySlice) {
 | 
					
						
							|  |  |  |         self.0.clear();
 | 
					
						
							|  |  |  |         self.0.extend(key_slice.0);
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |         self.1 = key_slice.1;
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn as_key_slice(&self) -> KeySlice {
 | 
					
						
							|  |  |  |         Key(self.0.as_slice(), self.1)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn into_key_bytes(self) -> KeyBytes {
 | 
					
						
							|  |  |  |         Key(self.0.into(), self.1)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn key_ref(&self) -> &[u8] {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |         self.0.as_ref()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn ts(&self) -> u64 {
 | 
					
						
							|  |  |  |         self.1
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     pub fn for_testing_key_ref(&self) -> &[u8] {
 | 
					
						
							|  |  |  |         self.0.as_ref()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn for_testing_from_vec_no_ts(key: Vec<u8>) -> Self {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |         Self(key, TS_DEFAULT)
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl Key<Bytes> {
 | 
					
						
							|  |  |  |     pub fn as_key_slice(&self) -> KeySlice {
 | 
					
						
							|  |  |  |         Key(&self.0, self.1)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     /// Create a `KeyBytes` from a `Bytes` and a ts.
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     pub fn from_bytes_with_ts(bytes: Bytes, ts: u64) -> KeyBytes {
 | 
					
						
							|  |  |  |         Key(bytes, ts)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn key_ref(&self) -> &[u8] {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |         self.0.as_ref()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn ts(&self) -> u64 {
 | 
					
						
							|  |  |  |         self.1
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     pub fn for_testing_from_bytes_no_ts(bytes: Bytes) -> KeyBytes {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |         Key(bytes, TS_DEFAULT)
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn for_testing_key_ref(&self) -> &[u8] {
 | 
					
						
							|  |  |  |         self.0.as_ref()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<'a> Key<&'a [u8]> {
 | 
					
						
							|  |  |  |     pub fn to_key_vec(self) -> KeyVec {
 | 
					
						
							|  |  |  |         Key(self.0.to_vec(), self.1)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Create a key slice from a slice. Will be removed in week 3.
 | 
					
						
							|  |  |  |     pub fn from_slice(slice: &'a [u8], ts: u64) -> Self {
 | 
					
						
							|  |  |  |         Self(slice, ts)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn key_ref(self) -> &'a [u8] {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |         self.0
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |     pub fn ts(&self) -> u64 {
 | 
					
						
							|  |  |  |         self.1
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     pub fn for_testing_key_ref(self) -> &'a [u8] {
 | 
					
						
							|  |  |  |         self.0
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn for_testing_from_slice_no_ts(slice: &'a [u8]) -> Self {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |         Self(slice, TS_DEFAULT)
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]> + Debug> Debug for Key<T> {
 | 
					
						
							|  |  |  |     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
 | 
					
						
							|  |  |  |         self.0.fmt(f)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]> + Default> Default for Key<T> {
 | 
					
						
							|  |  |  |     fn default() -> Self {
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:27:16 +08:00
										 |  |  |         Self(T::default(), TS_DEFAULT)
 | 
					
						
							| 
									
										
										
										
											2024-01-25 12:07:53 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]> + PartialEq> PartialEq for Key<T> {
 | 
					
						
							|  |  |  |     fn eq(&self, other: &Self) -> bool {
 | 
					
						
							|  |  |  |         (self.0.as_ref(), self.1).eq(&(other.0.as_ref(), other.1))
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]> + Eq> Eq for Key<T> {}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]> + Clone> Clone for Key<T> {
 | 
					
						
							|  |  |  |     fn clone(&self) -> Self {
 | 
					
						
							|  |  |  |         Self(self.0.clone(), self.1)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]> + Copy> Copy for Key<T> {}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]> + PartialOrd> PartialOrd for Key<T> {
 | 
					
						
							|  |  |  |     fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
 | 
					
						
							|  |  |  |         (self.0.as_ref(), Reverse(self.1)).partial_cmp(&(other.0.as_ref(), Reverse(other.1)))
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | impl<T: AsRef<[u8]> + Ord> Ord for Key<T> {
 | 
					
						
							|  |  |  |     fn cmp(&self, other: &Self) -> std::cmp::Ordering {
 | 
					
						
							|  |  |  |         (self.0.as_ref(), Reverse(self.1)).cmp(&(other.0.as_ref(), Reverse(other.1)))
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 |