checkin initial MVCC codebase
Signed-off-by: Alex Chi <iskyzh@gmail.com>
This commit is contained in:
		
							
								
								
									
										123
									
								
								mini-lsm-mvcc/src/iterators/concat_iterator.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								mini-lsm-mvcc/src/iterators/concat_iterator.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,123 @@ | ||||
| use std::sync::Arc; | ||||
|  | ||||
| use anyhow::Result; | ||||
|  | ||||
| use crate::{ | ||||
|     key::KeySlice, | ||||
|     table::{SsTable, SsTableIterator}, | ||||
| }; | ||||
|  | ||||
| use super::StorageIterator; | ||||
|  | ||||
| /// Concat multiple iterators ordered in key order and their key ranges do not overlap. We do not want to create the | ||||
| /// iterators when initializing this iterator to reduce the overhead of seeking. | ||||
| pub struct SstConcatIterator { | ||||
|     current: Option<SsTableIterator>, | ||||
|     next_sst_idx: usize, | ||||
|     sstables: Vec<Arc<SsTable>>, | ||||
| } | ||||
|  | ||||
| impl SstConcatIterator { | ||||
|     fn check_sst_valid(sstables: &[Arc<SsTable>]) { | ||||
|         for sst in sstables { | ||||
|             assert!(sst.first_key() <= sst.last_key()); | ||||
|         } | ||||
|         if !sstables.is_empty() { | ||||
|             for i in 0..(sstables.len() - 1) { | ||||
|                 assert!(sstables[i].last_key() < sstables[i + 1].first_key()); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn create_and_seek_to_first(sstables: Vec<Arc<SsTable>>) -> Result<Self> { | ||||
|         Self::check_sst_valid(&sstables); | ||||
|         if sstables.is_empty() { | ||||
|             return Ok(Self { | ||||
|                 current: None, | ||||
|                 next_sst_idx: 0, | ||||
|                 sstables, | ||||
|             }); | ||||
|         } | ||||
|         let mut iter = Self { | ||||
|             current: Some(SsTableIterator::create_and_seek_to_first( | ||||
|                 sstables[0].clone(), | ||||
|             )?), | ||||
|             next_sst_idx: 1, | ||||
|             sstables, | ||||
|         }; | ||||
|         iter.move_until_valid()?; | ||||
|         Ok(iter) | ||||
|     } | ||||
|  | ||||
|     pub fn create_and_seek_to_key(sstables: Vec<Arc<SsTable>>, key: KeySlice) -> Result<Self> { | ||||
|         Self::check_sst_valid(&sstables); | ||||
|         let idx: usize = sstables | ||||
|             .partition_point(|table| table.first_key().as_key_slice() <= key) | ||||
|             .saturating_sub(1); | ||||
|         if idx >= sstables.len() { | ||||
|             return Ok(Self { | ||||
|                 current: None, | ||||
|                 next_sst_idx: sstables.len(), | ||||
|                 sstables, | ||||
|             }); | ||||
|         } | ||||
|         let mut iter = Self { | ||||
|             current: Some(SsTableIterator::create_and_seek_to_key( | ||||
|                 sstables[idx].clone(), | ||||
|                 key, | ||||
|             )?), | ||||
|             next_sst_idx: idx + 1, | ||||
|             sstables, | ||||
|         }; | ||||
|         iter.move_until_valid()?; | ||||
|         Ok(iter) | ||||
|     } | ||||
|  | ||||
|     fn move_until_valid(&mut self) -> Result<()> { | ||||
|         while let Some(iter) = self.current.as_mut() { | ||||
|             if iter.is_valid() { | ||||
|                 break; | ||||
|             } | ||||
|             if self.next_sst_idx >= self.sstables.len() { | ||||
|                 self.current = None; | ||||
|             } else { | ||||
|                 self.current = Some(SsTableIterator::create_and_seek_to_first( | ||||
|                     self.sstables[self.next_sst_idx].clone(), | ||||
|                 )?); | ||||
|                 self.next_sst_idx += 1; | ||||
|             } | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl StorageIterator for SstConcatIterator { | ||||
|     type KeyType<'a> = KeySlice<'a>; | ||||
|  | ||||
|     fn key(&self) -> KeySlice { | ||||
|         self.current.as_ref().unwrap().key() | ||||
|     } | ||||
|  | ||||
|     fn value(&self) -> &[u8] { | ||||
|         self.current.as_ref().unwrap().value() | ||||
|     } | ||||
|  | ||||
|     fn is_valid(&self) -> bool { | ||||
|         if let Some(current) = &self.current { | ||||
|             assert!(current.is_valid()); | ||||
|             true | ||||
|         } else { | ||||
|             false | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn next(&mut self) -> Result<()> { | ||||
|         self.current.as_mut().unwrap().next()?; | ||||
|         self.move_until_valid()?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn num_active_iterators(&self) -> usize { | ||||
|         1 | ||||
|     } | ||||
| } | ||||
							
								
								
									
										155
									
								
								mini-lsm-mvcc/src/iterators/merge_iterator.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										155
									
								
								mini-lsm-mvcc/src/iterators/merge_iterator.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,155 @@ | ||||
| use std::cmp::{self}; | ||||
| use std::collections::binary_heap::PeekMut; | ||||
| use std::collections::BinaryHeap; | ||||
|  | ||||
| use anyhow::Result; | ||||
|  | ||||
| use crate::key::KeySlice; | ||||
|  | ||||
| use super::StorageIterator; | ||||
|  | ||||
| struct HeapWrapper<I: StorageIterator>(pub usize, pub Box<I>); | ||||
|  | ||||
| impl<I: StorageIterator> PartialEq for HeapWrapper<I> { | ||||
|     fn eq(&self, other: &Self) -> bool { | ||||
|         self.partial_cmp(other).unwrap() == cmp::Ordering::Equal | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<I: StorageIterator> Eq for HeapWrapper<I> {} | ||||
|  | ||||
| impl<I: StorageIterator> PartialOrd for HeapWrapper<I> { | ||||
|     #[allow(clippy::non_canonical_partial_ord_impl)] | ||||
|     fn partial_cmp(&self, other: &Self) -> Option<cmp::Ordering> { | ||||
|         match self.1.key().cmp(&other.1.key()) { | ||||
|             cmp::Ordering::Greater => Some(cmp::Ordering::Greater), | ||||
|             cmp::Ordering::Less => Some(cmp::Ordering::Less), | ||||
|             cmp::Ordering::Equal => self.0.partial_cmp(&other.0), | ||||
|         } | ||||
|         .map(|x| x.reverse()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<I: StorageIterator> Ord for HeapWrapper<I> { | ||||
|     fn cmp(&self, other: &Self) -> cmp::Ordering { | ||||
|         self.partial_cmp(other).unwrap() | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Merge multiple iterators of the same type. If the same key occurs multiple times in some | ||||
| /// iterators, perfer the one with smaller index. | ||||
| pub struct MergeIterator<I: StorageIterator> { | ||||
|     iters: BinaryHeap<HeapWrapper<I>>, | ||||
|     current: Option<HeapWrapper<I>>, | ||||
| } | ||||
|  | ||||
| impl<I: StorageIterator> MergeIterator<I> { | ||||
|     pub fn create(iters: Vec<Box<I>>) -> Self { | ||||
|         if iters.is_empty() { | ||||
|             return Self { | ||||
|                 iters: BinaryHeap::new(), | ||||
|                 current: None, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         let mut heap = BinaryHeap::new(); | ||||
|  | ||||
|         if iters.iter().all(|x| !x.is_valid()) { | ||||
|             // All invalid, select the last one as the current. | ||||
|             let mut iters = iters; | ||||
|             return Self { | ||||
|                 iters: heap, | ||||
|                 current: Some(HeapWrapper(0, iters.pop().unwrap())), | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         for (idx, iter) in iters.into_iter().enumerate() { | ||||
|             if iter.is_valid() { | ||||
|                 heap.push(HeapWrapper(idx, iter)); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         let current = heap.pop().unwrap(); | ||||
|         Self { | ||||
|             iters: heap, | ||||
|             current: Some(current), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<I: 'static + for<'a> StorageIterator<KeyType<'a> = KeySlice<'a>>> StorageIterator | ||||
|     for MergeIterator<I> | ||||
| { | ||||
|     type KeyType<'a> = KeySlice<'a>; | ||||
|  | ||||
|     fn key(&self) -> KeySlice { | ||||
|         self.current.as_ref().unwrap().1.key() | ||||
|     } | ||||
|  | ||||
|     fn value(&self) -> &[u8] { | ||||
|         self.current.as_ref().unwrap().1.value() | ||||
|     } | ||||
|  | ||||
|     fn is_valid(&self) -> bool { | ||||
|         self.current | ||||
|             .as_ref() | ||||
|             .map(|x| x.1.is_valid()) | ||||
|             .unwrap_or(false) | ||||
|     } | ||||
|  | ||||
|     fn next(&mut self) -> Result<()> { | ||||
|         let current = self.current.as_mut().unwrap(); | ||||
|         // Pop the item out of the heap if they have the same value. | ||||
|         while let Some(mut inner_iter) = self.iters.peek_mut() { | ||||
|             debug_assert!( | ||||
|                 inner_iter.1.key() >= current.1.key(), | ||||
|                 "heap invariant violated" | ||||
|             ); | ||||
|             if inner_iter.1.key() == current.1.key() { | ||||
|                 // Case 1: an error occurred when calling `next`. | ||||
|                 if let e @ Err(_) = inner_iter.1.next() { | ||||
|                     PeekMut::pop(inner_iter); | ||||
|                     return e; | ||||
|                 } | ||||
|  | ||||
|                 // Case 2: iter is no longer valid. | ||||
|                 if !inner_iter.1.is_valid() { | ||||
|                     PeekMut::pop(inner_iter); | ||||
|                 } | ||||
|             } else { | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         current.1.next()?; | ||||
|  | ||||
|         // If the current iterator is invalid, pop it out of the heap and select the next one. | ||||
|         if !current.1.is_valid() { | ||||
|             if let Some(iter) = self.iters.pop() { | ||||
|                 *current = iter; | ||||
|             } | ||||
|             return Ok(()); | ||||
|         } | ||||
|  | ||||
|         // Otherwise, compare with heap top and swap if necessary. | ||||
|         if let Some(mut inner_iter) = self.iters.peek_mut() { | ||||
|             if *current < *inner_iter { | ||||
|                 std::mem::swap(&mut *inner_iter, current); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn num_active_iterators(&self) -> usize { | ||||
|         self.iters | ||||
|             .iter() | ||||
|             .map(|x| x.1.num_active_iterators()) | ||||
|             .sum::<usize>() | ||||
|             + self | ||||
|                 .current | ||||
|                 .as_ref() | ||||
|                 .map(|x| x.1.num_active_iterators()) | ||||
|                 .unwrap_or(0) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										137
									
								
								mini-lsm-mvcc/src/iterators/tests/merge_iterator_test.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										137
									
								
								mini-lsm-mvcc/src/iterators/tests/merge_iterator_test.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,137 @@ | ||||
| use super::*; | ||||
| use crate::iterators::merge_iterator::MergeIterator; | ||||
|  | ||||
| fn as_bytes(x: &[u8]) -> Bytes { | ||||
|     Bytes::copy_from_slice(x) | ||||
| } | ||||
|  | ||||
| fn check_iter_result(iter: impl StorageIterator, expected: Vec<(Bytes, Bytes)>) { | ||||
|     let mut iter = iter; | ||||
|     for (k, v) in expected { | ||||
|         assert!(iter.is_valid()); | ||||
|         assert_eq!( | ||||
|             k, | ||||
|             iter.key(), | ||||
|             "expected key: {:?}, actual key: {:?}", | ||||
|             k, | ||||
|             as_bytes(iter.key()), | ||||
|         ); | ||||
|         assert_eq!( | ||||
|             v, | ||||
|             iter.value(), | ||||
|             "expected value: {:?}, actual value: {:?}", | ||||
|             v, | ||||
|             as_bytes(iter.value()), | ||||
|         ); | ||||
|         iter.next().unwrap(); | ||||
|     } | ||||
|     assert!(!iter.is_valid()); | ||||
| } | ||||
|  | ||||
| #[test] | ||||
| fn test_merge_1() { | ||||
|     let i1 = MockIterator::new(vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.1")), | ||||
|         (Bytes::from("b"), Bytes::from("2.1")), | ||||
|         (Bytes::from("c"), Bytes::from("3.1")), | ||||
|     ]); | ||||
|     let i2 = MockIterator::new(vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.2")), | ||||
|         (Bytes::from("b"), Bytes::from("2.2")), | ||||
|         (Bytes::from("c"), Bytes::from("3.2")), | ||||
|         (Bytes::from("d"), Bytes::from("4.2")), | ||||
|     ]); | ||||
|     let i3 = MockIterator::new(vec![ | ||||
|         (Bytes::from("b"), Bytes::from("2.3")), | ||||
|         (Bytes::from("c"), Bytes::from("3.3")), | ||||
|         (Bytes::from("d"), Bytes::from("4.3")), | ||||
|     ]); | ||||
|  | ||||
|     let iter = MergeIterator::create(vec![ | ||||
|         Box::new(i1.clone()), | ||||
|         Box::new(i2.clone()), | ||||
|         Box::new(i3.clone()), | ||||
|     ]); | ||||
|  | ||||
|     check_iter_result( | ||||
|         iter, | ||||
|         vec![ | ||||
|             (Bytes::from("a"), Bytes::from("1.1")), | ||||
|             (Bytes::from("b"), Bytes::from("2.1")), | ||||
|             (Bytes::from("c"), Bytes::from("3.1")), | ||||
|             (Bytes::from("d"), Bytes::from("4.2")), | ||||
|         ], | ||||
|     ); | ||||
|  | ||||
|     let iter = MergeIterator::create(vec![Box::new(i3), Box::new(i1), Box::new(i2)]); | ||||
|  | ||||
|     check_iter_result( | ||||
|         iter, | ||||
|         vec![ | ||||
|             (Bytes::from("a"), Bytes::from("1.1")), | ||||
|             (Bytes::from("b"), Bytes::from("2.3")), | ||||
|             (Bytes::from("c"), Bytes::from("3.3")), | ||||
|             (Bytes::from("d"), Bytes::from("4.3")), | ||||
|         ], | ||||
|     ); | ||||
| } | ||||
|  | ||||
| #[test] | ||||
| fn test_merge_2() { | ||||
|     let i1 = MockIterator::new(vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.1")), | ||||
|         (Bytes::from("b"), Bytes::from("2.1")), | ||||
|         (Bytes::from("c"), Bytes::from("3.1")), | ||||
|     ]); | ||||
|     let i2 = MockIterator::new(vec![ | ||||
|         (Bytes::from("d"), Bytes::from("1.2")), | ||||
|         (Bytes::from("e"), Bytes::from("2.2")), | ||||
|         (Bytes::from("f"), Bytes::from("3.2")), | ||||
|         (Bytes::from("g"), Bytes::from("4.2")), | ||||
|     ]); | ||||
|     let i3 = MockIterator::new(vec![ | ||||
|         (Bytes::from("h"), Bytes::from("1.3")), | ||||
|         (Bytes::from("i"), Bytes::from("2.3")), | ||||
|         (Bytes::from("j"), Bytes::from("3.3")), | ||||
|         (Bytes::from("k"), Bytes::from("4.3")), | ||||
|     ]); | ||||
|     let i4 = MockIterator::new(vec![]); | ||||
|     let result = vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.1")), | ||||
|         (Bytes::from("b"), Bytes::from("2.1")), | ||||
|         (Bytes::from("c"), Bytes::from("3.1")), | ||||
|         (Bytes::from("d"), Bytes::from("1.2")), | ||||
|         (Bytes::from("e"), Bytes::from("2.2")), | ||||
|         (Bytes::from("f"), Bytes::from("3.2")), | ||||
|         (Bytes::from("g"), Bytes::from("4.2")), | ||||
|         (Bytes::from("h"), Bytes::from("1.3")), | ||||
|         (Bytes::from("i"), Bytes::from("2.3")), | ||||
|         (Bytes::from("j"), Bytes::from("3.3")), | ||||
|         (Bytes::from("k"), Bytes::from("4.3")), | ||||
|     ]; | ||||
|  | ||||
|     let iter = MergeIterator::create(vec![ | ||||
|         Box::new(i1.clone()), | ||||
|         Box::new(i2.clone()), | ||||
|         Box::new(i3.clone()), | ||||
|         Box::new(i4.clone()), | ||||
|     ]); | ||||
|     check_iter_result(iter, result.clone()); | ||||
|  | ||||
|     let iter = MergeIterator::create(vec![ | ||||
|         Box::new(i2.clone()), | ||||
|         Box::new(i4.clone()), | ||||
|         Box::new(i3.clone()), | ||||
|         Box::new(i1.clone()), | ||||
|     ]); | ||||
|     check_iter_result(iter, result.clone()); | ||||
|  | ||||
|     let iter = MergeIterator::create(vec![Box::new(i4), Box::new(i3), Box::new(i2), Box::new(i1)]); | ||||
|     check_iter_result(iter, result); | ||||
| } | ||||
|  | ||||
| #[test] | ||||
| fn test_merge_empty() { | ||||
|     let iter = MergeIterator::<MockIterator>::create(vec![]); | ||||
|     check_iter_result(iter, vec![]); | ||||
| } | ||||
							
								
								
									
										129
									
								
								mini-lsm-mvcc/src/iterators/tests/two_merge_iterator_test.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								mini-lsm-mvcc/src/iterators/tests/two_merge_iterator_test.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | ||||
| use super::*; | ||||
| use crate::iterators::two_merge_iterator::TwoMergeIterator; | ||||
|  | ||||
| fn check_iter_result(iter: impl StorageIterator, expected: Vec<(Bytes, Bytes)>) { | ||||
|     let mut iter = iter; | ||||
|     for (k, v) in expected { | ||||
|         assert!(iter.is_valid()); | ||||
|         assert_eq!(iter.key(), k.as_ref()); | ||||
|         assert_eq!(iter.value(), v.as_ref()); | ||||
|         iter.next().unwrap(); | ||||
|     } | ||||
|     assert!(!iter.is_valid()); | ||||
| } | ||||
|  | ||||
| #[test] | ||||
| fn test_merge_1() { | ||||
|     let i1 = MockIterator::new(vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.1")), | ||||
|         (Bytes::from("b"), Bytes::from("2.1")), | ||||
|         (Bytes::from("c"), Bytes::from("3.1")), | ||||
|     ]); | ||||
|     let i2 = MockIterator::new(vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.2")), | ||||
|         (Bytes::from("b"), Bytes::from("2.2")), | ||||
|         (Bytes::from("c"), Bytes::from("3.2")), | ||||
|         (Bytes::from("d"), Bytes::from("4.2")), | ||||
|     ]); | ||||
|     let iter = TwoMergeIterator::create(i1, i2).unwrap(); | ||||
|     check_iter_result( | ||||
|         iter, | ||||
|         vec![ | ||||
|             (Bytes::from("a"), Bytes::from("1.1")), | ||||
|             (Bytes::from("b"), Bytes::from("2.1")), | ||||
|             (Bytes::from("c"), Bytes::from("3.1")), | ||||
|             (Bytes::from("d"), Bytes::from("4.2")), | ||||
|         ], | ||||
|     ) | ||||
| } | ||||
|  | ||||
| #[test] | ||||
| fn test_merge_2() { | ||||
|     let i2 = MockIterator::new(vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.1")), | ||||
|         (Bytes::from("b"), Bytes::from("2.1")), | ||||
|         (Bytes::from("c"), Bytes::from("3.1")), | ||||
|     ]); | ||||
|     let i1 = MockIterator::new(vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.2")), | ||||
|         (Bytes::from("b"), Bytes::from("2.2")), | ||||
|         (Bytes::from("c"), Bytes::from("3.2")), | ||||
|         (Bytes::from("d"), Bytes::from("4.2")), | ||||
|     ]); | ||||
|     let iter = TwoMergeIterator::create(i1, i2).unwrap(); | ||||
|     check_iter_result( | ||||
|         iter, | ||||
|         vec![ | ||||
|             (Bytes::from("a"), Bytes::from("1.2")), | ||||
|             (Bytes::from("b"), Bytes::from("2.2")), | ||||
|             (Bytes::from("c"), Bytes::from("3.2")), | ||||
|             (Bytes::from("d"), Bytes::from("4.2")), | ||||
|         ], | ||||
|     ) | ||||
| } | ||||
|  | ||||
| #[test] | ||||
| fn test_merge_3() { | ||||
|     let i2 = MockIterator::new(vec![ | ||||
|         (Bytes::from("a"), Bytes::from("1.1")), | ||||
|         (Bytes::from("b"), Bytes::from("2.1")), | ||||
|         (Bytes::from("c"), Bytes::from("3.1")), | ||||
|     ]); | ||||
|     let i1 = MockIterator::new(vec![ | ||||
|         (Bytes::from("b"), Bytes::from("2.2")), | ||||
|         (Bytes::from("c"), Bytes::from("3.2")), | ||||
|         (Bytes::from("d"), Bytes::from("4.2")), | ||||
|     ]); | ||||
|     let iter = TwoMergeIterator::create(i1, i2).unwrap(); | ||||
|     check_iter_result( | ||||
|         iter, | ||||
|         vec![ | ||||
|             (Bytes::from("a"), Bytes::from("1.1")), | ||||
|             (Bytes::from("b"), Bytes::from("2.2")), | ||||
|             (Bytes::from("c"), Bytes::from("3.2")), | ||||
|             (Bytes::from("d"), Bytes::from("4.2")), | ||||
|         ], | ||||
|     ) | ||||
| } | ||||
|  | ||||
| #[test] | ||||
| fn test_merge_4() { | ||||
|     let i2 = MockIterator::new(vec![]); | ||||
|     let i1 = MockIterator::new(vec![ | ||||
|         (Bytes::from("b"), Bytes::from("2.2")), | ||||
|         (Bytes::from("c"), Bytes::from("3.2")), | ||||
|         (Bytes::from("d"), Bytes::from("4.2")), | ||||
|     ]); | ||||
|     let iter = TwoMergeIterator::create(i1, i2).unwrap(); | ||||
|     check_iter_result( | ||||
|         iter, | ||||
|         vec![ | ||||
|             (Bytes::from("b"), Bytes::from("2.2")), | ||||
|             (Bytes::from("c"), Bytes::from("3.2")), | ||||
|             (Bytes::from("d"), Bytes::from("4.2")), | ||||
|         ], | ||||
|     ); | ||||
|     let i1 = MockIterator::new(vec![]); | ||||
|     let i2 = MockIterator::new(vec![ | ||||
|         (Bytes::from("b"), Bytes::from("2.2")), | ||||
|         (Bytes::from("c"), Bytes::from("3.2")), | ||||
|         (Bytes::from("d"), Bytes::from("4.2")), | ||||
|     ]); | ||||
|     let iter = TwoMergeIterator::create(i1, i2).unwrap(); | ||||
|     check_iter_result( | ||||
|         iter, | ||||
|         vec![ | ||||
|             (Bytes::from("b"), Bytes::from("2.2")), | ||||
|             (Bytes::from("c"), Bytes::from("3.2")), | ||||
|             (Bytes::from("d"), Bytes::from("4.2")), | ||||
|         ], | ||||
|     ); | ||||
| } | ||||
|  | ||||
| #[test] | ||||
| fn test_merge_5() { | ||||
|     let i2 = MockIterator::new(vec![]); | ||||
|     let i1 = MockIterator::new(vec![]); | ||||
|     let iter = TwoMergeIterator::create(i1, i2).unwrap(); | ||||
|     check_iter_result(iter, vec![]) | ||||
| } | ||||
							
								
								
									
										94
									
								
								mini-lsm-mvcc/src/iterators/two_merge_iterator.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								mini-lsm-mvcc/src/iterators/two_merge_iterator.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,94 @@ | ||||
| use anyhow::Result; | ||||
|  | ||||
| use crate::key::KeySlice; | ||||
|  | ||||
| use super::StorageIterator; | ||||
|  | ||||
| /// Merges two iterators of different types into one. If the two iterators have the same key, only | ||||
| /// produce the key once and prefer the entry from A. | ||||
| pub struct TwoMergeIterator<A: StorageIterator, B: StorageIterator> { | ||||
|     a: A, | ||||
|     b: B, | ||||
|     choose_a: bool, | ||||
| } | ||||
|  | ||||
| impl< | ||||
|         A: 'static + for<'a> StorageIterator<KeyType<'a> = KeySlice<'a>>, | ||||
|         B: 'static + for<'a> StorageIterator<KeyType<'a> = KeySlice<'a>>, | ||||
|     > TwoMergeIterator<A, B> | ||||
| { | ||||
|     fn choose_a(a: &A, b: &B) -> bool { | ||||
|         if !a.is_valid() { | ||||
|             return false; | ||||
|         } | ||||
|         if !b.is_valid() { | ||||
|             return true; | ||||
|         } | ||||
|         a.key() < b.key() | ||||
|     } | ||||
|  | ||||
|     fn skip_b(&mut self) -> Result<()> { | ||||
|         if self.a.is_valid() && self.b.is_valid() && self.b.key() == self.a.key() { | ||||
|             self.b.next()?; | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     pub fn create(a: A, b: B) -> Result<Self> { | ||||
|         let mut iter = Self { | ||||
|             choose_a: false, | ||||
|             a, | ||||
|             b, | ||||
|         }; | ||||
|         iter.skip_b()?; | ||||
|         iter.choose_a = Self::choose_a(&iter.a, &iter.b); | ||||
|         Ok(iter) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl< | ||||
|         A: 'static + for<'a> StorageIterator<KeyType<'a> = KeySlice<'a>>, | ||||
|         B: 'static + for<'a> StorageIterator<KeyType<'a> = KeySlice<'a>>, | ||||
|     > StorageIterator for TwoMergeIterator<A, B> | ||||
| { | ||||
|     type KeyType<'a> = KeySlice<'a>; | ||||
|  | ||||
|     fn key(&self) -> KeySlice { | ||||
|         if self.choose_a { | ||||
|             self.a.key() | ||||
|         } else { | ||||
|             self.b.key() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn value(&self) -> &[u8] { | ||||
|         if self.choose_a { | ||||
|             self.a.value() | ||||
|         } else { | ||||
|             self.b.value() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn is_valid(&self) -> bool { | ||||
|         if self.choose_a { | ||||
|             self.a.is_valid() | ||||
|         } else { | ||||
|             self.b.is_valid() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn next(&mut self) -> Result<()> { | ||||
|         if self.choose_a { | ||||
|             self.a.next()?; | ||||
|         } else { | ||||
|             self.b.next()?; | ||||
|         } | ||||
|         self.skip_b()?; | ||||
|         self.choose_a = Self::choose_a(&self.a, &self.b); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn num_active_iterators(&self) -> usize { | ||||
|         self.a.num_active_iterators() + self.b.num_active_iterators() | ||||
|     } | ||||
| } | ||||
		Reference in New Issue
	
	Block a user
	 Alex Chi
					Alex Chi