|
@@ -1,6 +1,29 @@
|
|
|
use crate::value;
|
|
|
use bytes::Bytes;
|
|
|
-use crc32fast::Hasher;
|
|
|
+use crc32fast::Hasher as Crc32Hasher;
|
|
|
+use std::hash::{Hash, Hasher};
|
|
|
+
|
|
|
+fn calculate_checksum(bytes: &Bytes) -> Option<u32> {
|
|
|
+ if bytes.len() < 1024 {
|
|
|
+ None
|
|
|
+ } else {
|
|
|
+ let mut hasher = Crc32Hasher::new();
|
|
|
+ hasher.update(bytes);
|
|
|
+ Some(hasher.finalize())
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+pub struct Ref<'a> {
|
|
|
+ bytes: &'a Bytes,
|
|
|
+ checksum: Option<u32>,
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a> Ref<'a> {
|
|
|
+ pub fn new(bytes: &'a Bytes) -> Self {
|
|
|
+ let checksum = calculate_checksum(bytes);
|
|
|
+ Self { bytes, checksum }
|
|
|
+ }
|
|
|
+}
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
pub struct Value {
|
|
@@ -10,7 +33,7 @@ pub struct Value {
|
|
|
|
|
|
impl Value {
|
|
|
pub fn new(bytes: Bytes) -> Self {
|
|
|
- let checksum = Self::calculate_checksum(&bytes);
|
|
|
+ let checksum = calculate_checksum(&bytes);
|
|
|
Self { bytes, checksum }
|
|
|
}
|
|
|
|
|
@@ -21,15 +44,11 @@ impl Value {
|
|
|
pub fn has_checksum(&self) -> bool {
|
|
|
self.checksum.is_some()
|
|
|
}
|
|
|
+}
|
|
|
|
|
|
- fn calculate_checksum(bytes: &Bytes) -> Option<u32> {
|
|
|
- if bytes.len() < 1024 {
|
|
|
- None
|
|
|
- } else {
|
|
|
- let mut hasher = Hasher::new();
|
|
|
- hasher.update(bytes);
|
|
|
- Some(hasher.finalize())
|
|
|
- }
|
|
|
+impl Hash for Value {
|
|
|
+ fn hash<H: Hasher>(&self, state: &mut H) {
|
|
|
+ self.bytes.hash(state);
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -45,6 +64,20 @@ impl PartialEq for Value {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+impl Eq for Value {}
|
|
|
+
|
|
|
+impl<'a> PartialEq<Ref<'a>> for Value {
|
|
|
+ fn eq(&self, other: &Ref) -> bool {
|
|
|
+ if self.checksum == other.checksum && self.bytes.len() == other.bytes.len() {
|
|
|
+ // The data have the same checksum now perform a more extensive
|
|
|
+ // comparision
|
|
|
+ return self.bytes.eq(&other.bytes);
|
|
|
+ }
|
|
|
+
|
|
|
+ false
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
#[cfg(test)]
|
|
|
mod test {
|
|
|
use super::*;
|