From cea5fe8741744bc9f80abc2c2944eb9d5c96c814 Mon Sep 17 00:00:00 2001 From: Konrad Siek Date: Wed, 24 Feb 2021 10:43:38 +0100 Subject: [PATCH 1/3] Making db public to access MappingIter in djanco. --- src/lib.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/lib.rs b/src/lib.rs index e1a2fa1..b05933d 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -9,7 +9,7 @@ extern crate lazy_static; mod helpers; #[allow(dead_code)] -mod db; +pub mod db; #[allow(dead_code)] mod records; #[allow(dead_code)] From c63073816ec47f2b06ed87d1d3aa88fa95f32ad5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Colette=20=F0=9F=8C=A9=EF=B8=8E=E2=80=8D=F0=9F=92=9D=20Ker?= =?UTF-8?q?r?= Date: Sun, 28 Feb 2021 19:20:24 +0100 Subject: [PATCH 2/3] Added variants to iterators across the board which may take ownership This allows the creation of iterators that don't have a lifetime dependency on their parent objects, because they manage the ownership themselves They are not exposed through lib.rs yet --- Cargo.toml | 2 +- src/db.rs | 474 ++++++++++++++++++++++++++++++++++++------------- src/helpers.rs | 18 ++ src/lib.rs | 22 +-- 4 files changed, 381 insertions(+), 135 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index fede9e4..da5f6d5 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -36,4 +36,4 @@ num-derive = "0.3" num-traits = "0.2" # not this one, really, really bad dependencies #sysinfo = "0.15.0" -lazy_static = "1.4.0" \ No newline at end of file +lazy_static = "1.4.0" diff --git a/src/db.rs b/src/db.rs index 12390be..4f18379 100644 --- a/src/db.rs +++ b/src/db.rs @@ -15,7 +15,7 @@ Like store, */ -use std::fs::{File, OpenOptions}; +use std::{fs::{File, OpenOptions}, marker::PhantomData, ops::DerefMut}; use std::io::{Seek, SeekFrom, Read, Write}; use byteorder::*; use std::collections::*; @@ -28,7 +28,6 @@ use crate::LOG; pub (crate) const MAX_BUFFER_LENGTH : u64 = 10 * 1024 * 1024 * 1024; // 10GB - /** Marker trait for readonly datastore records. A record is readonly if it does not support updates, such as CommitInfo - once we have it, there is no way to overwrite its value, unlike for instance project heads. @@ -201,7 +200,7 @@ pub struct Indexer = u64, ID : Id = u64 > name : String, f : File, size : u64, - why_oh_why : std::marker::PhantomData<(T, ID)> + _t : std::marker::PhantomData<(T, ID)> } impl, ID : Id> Indexer { @@ -213,7 +212,7 @@ impl, ID : Id> Indexer { f = OpenOptions::new().read(true).write(true).create(true).open(format!("{}/{}.idx", root, name)).unwrap(); } let size = f.seek(SeekFrom::End(0)).unwrap() / T::SIZE; - return Indexer{ name : name.to_owned(), f, size, why_oh_why : std::marker::PhantomData{} }; + return Indexer{ name : name.to_owned(), f, size, _t : std::marker::PhantomData{} }; } pub fn get(& mut self, id : ID) -> Option { @@ -260,39 +259,62 @@ impl, ID : Id> Indexer { self.f.seek(SeekFrom::End(0)).unwrap(); } - pub fn iter(& mut self) -> IndexerIterator { + pub fn iter<'a>(&'a mut self) -> impl Iterator + 'a { + self.f.seek(SeekFrom::Start(0)).unwrap(); + return IndexerIterator{indexer: self, id : ID::from(0), max_offset: u64::MAX, _a: PhantomData}; + } + + pub fn into_iter<'a>(mut self) -> impl Iterator + 'a where Self: 'a{ + self.f.seek(SeekFrom::Start(0)).unwrap(); + return IndexerIterator{indexer: helpers::Holder(self), id : ID::from(0), max_offset: u64::MAX, _a: PhantomData}; + } + + pub fn savepoint_iter<'a>(&'a mut self, sp : & Savepoint) -> impl Iterator + 'a { + let max_offset = sp.limit_for(&self.name); self.f.seek(SeekFrom::Start(0)).unwrap(); - return IndexerIterator{indexer : self, id : 0, max_offset: u64::MAX}; + return IndexerIterator{indexer : self, id: ID::from(0), max_offset, _a: PhantomData }; } - pub fn savepoint_iter(& mut self, sp : & Savepoint) -> IndexerIterator { - let max_offset = sp.limit_for(& self.name); + pub fn into_savepoint_iter<'a>(mut self, sp : & Savepoint) ->impl Iterator + 'a where Self: 'a { + let max_offset = sp.limit_for(&self.name); self.f.seek(SeekFrom::Start(0)).unwrap(); - return IndexerIterator{indexer : self, id : 0, max_offset }; + return IndexerIterator{indexer: helpers::Holder(self), id: ID::from(0), max_offset, _a: PhantomData }; } } -pub struct IndexerIterator<'a, T : Indexable + Serializable, ID : Id = u64> { - indexer : &'a mut Indexer, - id : u64, - max_offset : u64, +pub struct IndexerIterator<'a, I, T, ID> +where + I: DerefMut>, + T: Indexable + Serializable, + ID: Id, +{ + indexer: I, + id: ID, + max_offset: u64, + _a: PhantomData<&'a I> } -impl<'a, T : Indexable + Serializable, ID : Id> Iterator for IndexerIterator<'a, T, ID> { +impl<'a, I, T, ID> Iterator for IndexerIterator<'a, I, T, ID> +where + I: DerefMut>, + T: Indexable + Serializable, + ID: Id, +{ type Item = (ID, T); - fn next(& mut self) -> Option<(ID, T)> { + fn next(&mut self) -> Option<(ID, T)> { + let uid = >::into(self.id); loop { if self.indexer.f.seek(SeekFrom::Current(0)).unwrap() >= self.max_offset { return None; - } else if self.id == self.indexer.len() as u64 { + } else if uid == self.indexer.len() as u64 { return None; } else { - let id = ID::from(self.id); - let result = T::deserialize(& mut self.indexer.f); - self.id += 1; - return Some((id, result)); + let nid = self.id; + let result = T::deserialize(&mut self.indexer.f); + self.id = ID::from(uid + 1u64); + return Some((nid, result)); } } } @@ -302,14 +324,21 @@ impl<'a, T : Indexable + Serializable, ID : Id> Iterator for IndexerIt Store is an indexed updatable container that keeps history of updates. */ -pub struct Store, ID : Id = u64> { +pub struct Store +where + T: Serializable, + ID: Id +{ pub (crate) indexer : Indexer, pub (crate) f : File, - why_oh_why : std::marker::PhantomData, + _t : std::marker::PhantomData, } -impl, ID : Id> Store { - +impl, ID : Id> Store +where + T: Serializable, + ID: Id +{ pub fn new(root : & str, name : & str, readonly : bool) -> Store { let f; if readonly { @@ -320,7 +349,7 @@ impl, ID : Id> Store { let mut result = Store{ indexer : Indexer::new(root, name, readonly), f, - why_oh_why : std::marker::PhantomData{} + _t : std::marker::PhantomData{} }; LOG!(" {}: indices {}, size {}", name, result.indexer.len(), result.f.seek(SeekFrom::End(0)).unwrap()); return result; @@ -432,23 +461,46 @@ impl, ID : Id> Store { Returns the latest stored value for every id. The ids are guaranteed to be increasing. */ - pub fn iter(& mut self) -> StoreIter { - return StoreIter::new(& mut self. f, & mut self.indexer); + pub fn iter<'a>(&'a mut self) -> impl Iterator + 'a { + StoreIter::new(&mut self.f, self.indexer.iter()) + } + + /** Iterates over the stored values. + + Returns the latest stored value for every id. The ids are guaranteed to be increasing. + */ + pub fn into_iter<'a>(self) -> impl Iterator + 'a where Self: 'a{ + StoreIter::new(helpers::Holder(self.f), self.indexer.into_iter()) + } + + /** Iterates over all stored values. + + Iterates over *all* stored values, returning them in the order they were added to the store. Multiple values may be returned for single id, the last value returned is the valid one. + */ + pub fn iter_all<'a>(&'a mut self) -> impl Iterator + 'a { + self.f.seek(SeekFrom::Start(0)).unwrap(); + return StoreIterAll{ store: self, max_offset : u64::MAX, _a: PhantomData }; } /** Iterates over all stored values. Iterates over *all* stored values, returning them in the order they were added to the store. Multiple values may be returned for single id, the last value returned is the valid one. */ - pub fn iter_all(& mut self) -> StoreIterAll { + pub fn into_iter_all<'a>(mut self) -> impl Iterator + 'a where Self: 'a{ self.f.seek(SeekFrom::Start(0)).unwrap(); - return StoreIterAll{ store : self, max_offset : u64::MAX }; + return StoreIterAll{ store: helpers::Holder(self), max_offset : u64::MAX, _a: PhantomData }; } - pub fn savepoint_iter_all(& mut self, sp : & Savepoint) -> StoreIterAll { + pub fn savepoint_iter_all<'a>(&'a mut self, sp : & Savepoint) -> impl Iterator + 'a { let max_offset = sp.limit_for(& format!("{}.store", self.name())); self.f.seek(SeekFrom::Start(0)).unwrap(); - return StoreIterAll{ store : self, max_offset }; + return StoreIterAll{store: self, max_offset, _a: PhantomData }; + } + + pub fn into_savepoint_iter_all<'a>(mut self, sp: &Savepoint) -> impl Iterator + 'a where Self: 'a{ + let max_offset = sp.limit_for(& format!("{}.store", self.name())); + self.f.seek(SeekFrom::Start(0)).unwrap(); + return StoreIterAll{ store: helpers::Holder(self), max_offset, _a: PhantomData }; } /** Reads the record from a file. @@ -471,31 +523,52 @@ impl, ID : Id> Store { } } + /** Latest store iterator does not support savepoints since the indices can be udpated. */ -pub struct StoreIter<'a, T: Serializable, ID : Id> { - f : &'a mut File, - iiter : IndexerIterator<'a, u64,ID>, - why_oh_why : std::marker::PhantomData, + pub struct StoreIter<'a, F, II, T, ID> + where + F: DerefMut + 'a, + II: Iterator + 'a, + T: Serializable, + ID: Id, +{ + file: F, + iiter: II, + _t: PhantomData, + _a: PhantomData<(&'a F, &'a II)>, } -impl<'a, T : Serializable, ID : Id> StoreIter<'a, T, ID> { - fn new(f : &'a mut File, indexer : &'a mut Indexer) -> StoreIter<'a, T, ID> { - return StoreIter{ - f : f, - iiter : indexer.iter(), - why_oh_why : std::marker::PhantomData{} - }; +impl<'a, F, II, T, ID> StoreIter<'a, F, II, T, ID> +where + F: DerefMut, + II: Iterator + 'a, + T: Serializable, + ID: Id + { + fn new(file: F, iiter: II) -> StoreIter<'a, F, II, T, ID>{ + StoreIter{ + file, + iiter, + _t: PhantomData, + _a: PhantomData, + } } } -impl<'a, T : Serializable, ID : Id> Iterator for StoreIter<'a, T, ID> { +impl<'a, F, II, T, ID> Iterator for StoreIter<'a, F, II, T, ID> +where + F: DerefMut, + II: Iterator + 'a, + T: Serializable, + ID: Id + { type Item = (ID, T); fn next(& mut self) -> Option<(ID, T)> { if let Some((id, offset)) = self.iiter.next() { - self.f.seek(SeekFrom::Start(offset)).unwrap(); - let (store_id, value) = Store::::read_record(self.f).unwrap(); + self.file.seek(SeekFrom::Start(offset)).unwrap(); + let (store_id, value) = Store::::read_record(&mut self.file).unwrap(); assert_eq!(id, store_id, "Corrupted store or its indexing"); return Some((id, value)); } else { @@ -504,23 +577,37 @@ impl<'a, T : Serializable, ID : Id> Iterator for StoreIter<'a, T, ID> } } -pub struct StoreIterAll<'a, T : Serializable, ID : Id> { - store : &'a mut Store, + + +pub struct StoreIterAll<'a, T, S, ID> +where + S: DerefMut> + 'a, + T: Serializable, + ID: Id +{ + store: S, max_offset : u64, + _a: PhantomData<&'a S>, } -impl<'a, T : Serializable, ID : Id> Iterator for StoreIterAll<'a, T, ID> { +impl<'a, T, S, ID> Iterator for StoreIterAll<'a, T, S, ID> +where + S: DerefMut> + 'a, + T: Serializable, + ID: Id +{ type Item = (ID, T); fn next(& mut self) -> Option<(ID, T)> { if self.store.f.seek(SeekFrom::Current(0)).unwrap() >= self.max_offset { return None; } else { - return Store::::read_record(& mut self.store.f); + return Store::::read_record(&mut self.store.f); } } } + /** Linked store implementation. Store is an indexed updatable container that keeps history of updates. @@ -530,7 +617,7 @@ impl<'a, T : Serializable, ID : Id> Iterator for StoreIterAll<'a, T, I pub struct LinkedStore, ID : Id = u64> { pub (crate) indexer : Indexer, pub (crate) f : File, - why_oh_why : std::marker::PhantomData, + _t : std::marker::PhantomData, } impl, ID : Id> LinkedStore { @@ -545,7 +632,7 @@ impl, ID : Id> LinkedStore { let mut result = LinkedStore{ indexer : Indexer::new(root, name, readonly), f, - why_oh_why : std::marker::PhantomData{} + _t : std::marker::PhantomData{} }; LOG!(" {}: indices {}, size {}", name, result.indexer.len(), result.f.seek(SeekFrom::End(0)).unwrap()); return result; @@ -672,32 +759,64 @@ impl, ID : Id> LinkedStore { Returns the latest stored value for every id. The ids are guaranteed to be increasing. */ - pub fn iter(& mut self) -> LinkedStoreIter { - return LinkedStoreIter::new(& mut self. f, & mut self.indexer); + pub fn iter<'a>(&'a mut self) -> impl Iterator + 'a { + LinkedStoreIter::new(&mut self.f, self.indexer.iter()) + } + + /** Iterates over the stored values. + + Returns the latest stored value for every id. The ids are guaranteed to be increasing. + */ + pub fn into_iter<'a>(self) -> impl Iterator + 'a where Self: 'a{ + LinkedStoreIter::new(helpers::Holder(self.f), self.indexer.into_iter()) } /** Iterates over all stored values. Iterates over *all* stored values, returning them in the order they were added to the store. Multiple values may be returned for single id, the last value returned is the valid one. */ - pub fn iter_all(& mut self) -> LinkedStoreIterAll { + pub fn iter_all<'a>(&'a mut self) -> impl Iterator + 'a { self.f.seek(SeekFrom::Start(0)).unwrap(); - return LinkedStoreIterAll{ store : self, max_offset : u64::MAX }; + LinkedStoreIterAll{ store: self, max_offset: u64::MAX } } - pub fn savepoint_iter_all(& mut self, sp : & Savepoint) -> LinkedStoreIterAll { - let max_offset = sp.limit_for(& format!("{}.store",self.name())); + /** Iterates over all stored values. + + Iterates over *all* stored values, returning them in the order they were added to the store. Multiple values may be returned for single id, the last value returned is the valid one. + */ + pub fn into_iter_all<'a>(mut self) -> impl Iterator + 'a where Self: 'a{ self.f.seek(SeekFrom::Start(0)).unwrap(); - return LinkedStoreIterAll{ store : self, max_offset }; + LinkedStoreIterAll{ store: helpers::Holder(self), max_offset: u64::MAX } + } + + pub fn savepoint_iter_all<'a>(&'a mut self, sp: &Savepoint) -> impl Iterator + 'a { + let max_offset = sp.limit_for(&format!("{}.store",self.name())); + self.f.seek(SeekFrom::Start(0)).unwrap(); + LinkedStoreIterAll{ store : self, max_offset } + } + + pub fn into_savepoint_iter_all<'a>(mut self, sp: &Savepoint) -> impl Iterator + 'a where Self: 'a{ + let max_offset = sp.limit_for(&format!("{}.store",self.name())); + self.f.seek(SeekFrom::Start(0)).unwrap(); + LinkedStoreIterAll{ store: helpers::Holder(self), max_offset } + } + + /** Given an id, returns an iterator over all values ever stored for it. + + The values are returned in the reverse order they were added, i.e. latest value first. + */ + pub fn iter_id<'a>(&'a mut self, id : ID) -> impl Iterator + 'a { + let offset = self.indexer.get(id); + LinkedStoreIterId{ store: self, offset, _a: PhantomData} } /** Given an id, returns an iterator over all values ever stored for it. The values are returned in the reverse order they were added, i.e. latest value first. */ - pub fn iter_id(& mut self, id : ID) -> LinkedStoreIterId { + pub fn into_iter_id<'a>(mut self, id : ID) -> impl Iterator + 'a where Self: 'a { let offset = self.indexer.get(id); - return LinkedStoreIterId{ store : self, offset }; + return LinkedStoreIterId{ store: helpers::Holder(self), offset, _a: PhantomData }; } /** Reads the record from a file. @@ -725,29 +844,47 @@ impl, ID : Id> LinkedStore { } } -pub struct LinkedStoreIter<'a, T: Serializable, ID : Id> { - f : &'a mut File, - iiter : IndexerIterator<'a, u64, ID>, - why_oh_why : std::marker::PhantomData, +pub struct LinkedStoreIter +where + T: Serializable, + F: DerefMut, + II: Iterator, + ID: Id, +{ + iiter: II, + file: F, + _t: std::marker::PhantomData, } -impl<'a, T : Serializable, ID : Id> LinkedStoreIter<'a, T, ID> { - fn new(f : &'a mut File, indexer : &'a mut Indexer) -> LinkedStoreIter<'a, T, ID> { +impl LinkedStoreIter +where + T: Serializable, + F: DerefMut, + II: Iterator, + ID: Id, +{ + fn new(file: F, iiter: II) -> LinkedStoreIter { return LinkedStoreIter{ - f : f, - iiter : indexer.iter(), - why_oh_why : std::marker::PhantomData{} + iiter, + file, + _t: std::marker::PhantomData{} }; } } -impl<'a, T : Serializable, ID : Id> Iterator for LinkedStoreIter<'a, T, ID> { +impl Iterator for LinkedStoreIter +where + T: Serializable, + F: DerefMut, + II: Iterator, + ID: Id, +{ type Item = (ID, T); fn next(& mut self) -> Option<(ID, T)> { if let Some((id, offset)) = self.iiter.next() { - self.f.seek(SeekFrom::Start(offset)).unwrap(); - let (store_id, _, value) = LinkedStore::::read_record(self.f).unwrap(); + self.file.seek(SeekFrom::Start(offset)).unwrap(); + let (store_id, _, value) = LinkedStore::::read_record(&mut self.file).unwrap(); assert_eq!(id, store_id, "Corrupted store or its indexing"); return Some((id, value)); } else { @@ -756,40 +893,61 @@ impl<'a, T : Serializable, ID : Id> Iterator for LinkedStoreIter<'a, T } } -pub struct LinkedStoreIterAll<'a, T : Serializable, ID : Id> { - store : &'a mut LinkedStore, - max_offset : u64, + + +pub struct LinkedStoreIterAll +where + T: Serializable, + LS: DerefMut>, + ID: Id +{ + store: LS, + max_offset: u64, } -impl<'a, T : Serializable, ID : Id> Iterator for LinkedStoreIterAll<'a, T, ID> { +impl Iterator for LinkedStoreIterAll +where + T: Serializable, + LS: DerefMut>, + ID: Id +{ type Item = (ID, T); fn next(& mut self) -> Option<(ID, T)> { if self.store.f.seek(SeekFrom::Current(0)).unwrap() >= self.max_offset { return None; } else { - match LinkedStore::::read_record(& mut self.store.f) { + match LinkedStore::::read_record(&mut self.store.f) { Some((id, _, value)) => Some((id, value)), None => None } } } - } - -pub struct LinkedStoreIterId<'a, T : Serializable, ID : Id> { - store : &'a mut LinkedStore, - offset : Option, +pub struct LinkedStoreIterId<'a, T, LS, ID> +where + T: Serializable, + LS: DerefMut>, + ID: Id, +{ + store: LS, + offset: Option, + _a: PhantomData<&'a LS>, } -impl<'a, T : Serializable, ID : Id> Iterator for LinkedStoreIterId<'a, T, ID> { +impl<'a, T, LS, ID> Iterator for LinkedStoreIterId<'a, T, LS, ID> +where + T: Serializable, + LS: DerefMut>, + ID: Id, +{ type Item = T; - fn next(& mut self) -> Option { + fn next(&mut self) -> Option { match self.offset { Some(offset) => { self.store.f.seek(SeekFrom::Start(offset)).unwrap(); - let (_, previous_offset, value) = LinkedStore::::read_record(& mut self.store.f).unwrap(); + let (_, previous_offset, value) = LinkedStore::::read_record(&mut self.store.f).unwrap(); self.offset = previous_offset; return Some(value); }, @@ -798,6 +956,7 @@ impl<'a, T : Serializable, ID : Id> Iterator for LinkedStoreIterId<'a, } } + /** Mapping from values to ids. Unlike store, mapping does not allow updates to added values. @@ -867,11 +1026,11 @@ impl + Eq + Hash + Clone, ID : Id> Mapping + Eq + Hash + Clone, ID : Id> Mapping MappingIter { + pub fn iter<'a>(&'a mut self) -> impl Iterator + 'a { + self.f.seek(SeekFrom::Start(0)).unwrap(); + let size = self.size; + MappingIter{file: &mut self.f, index: 0, size, _t: PhantomData, _a: PhantomData } + } + + pub fn savepoint_iter<'a>(&'a mut self, sp : &Savepoint) -> impl Iterator + 'a { + let max_offset = + sp.limit_for(& format!("{}.mapping", self.name())) / (std::mem::size_of::() as u64); + self.f.seek(SeekFrom::Start(0)).unwrap(); + MappingIter{file: &mut self.f, index : 0, size: max_offset, _t: PhantomData, _a: PhantomData } + } + + pub fn into_iter<'a>(mut self) -> impl Iterator + 'a where Self: 'a{ self.f.seek(SeekFrom::Start(0)).unwrap(); - return MappingIter{f : & mut self.f, index : 0, size : self.size, why_oh_why : std::marker::PhantomData{} }; + let size = self.size; + MappingIter{file: helpers::Holder(self.f), index: 0, size, _t: PhantomData, _a: PhantomData } } - pub fn savepoint_iter(& mut self, sp : & Savepoint) -> MappingIter { - let max_offset = sp.limit_for(& format!("{}.mapping", self.name())); + pub fn into_savepoint_iter<'a>(mut self, sp : &Savepoint) -> impl Iterator + 'a where Self: 'a{ + let max_offset = + sp.limit_for(& format!("{}.mapping", self.name())) / (std::mem::size_of::() as u64); self.f.seek(SeekFrom::Start(0)).unwrap(); - return MappingIter{f : & mut self.f, index : 0, size : max_offset / (std::mem::size_of::() as u64), why_oh_why : std::marker::PhantomData{} }; + MappingIter{file: helpers::Holder(self.f), index : 0, size: max_offset, _t: PhantomData, _a: PhantomData } } } -pub struct MappingIter<'a, T : FixedSizeSerializable + Eq + Hash + Clone, ID : Id = u64> { - f : &'a mut File, - index : u64, - size : u64, - why_oh_why : std::marker::PhantomData<(T, ID)> +pub struct MappingIter<'a, F, T, ID> +where + F: DerefMut, + T: FixedSizeSerializable + Eq + Hash + Clone, + ID: Id +{ + file: F, + index: u64, + size: u64, + _t: std::marker::PhantomData<(T, ID)>, + _a: PhantomData<&'a F> } -impl<'a, T : FixedSizeSerializable + Eq + Hash + Clone, ID : Id> Iterator for MappingIter<'a, T, ID> { +impl Iterator for MappingIter<'_, F, T, ID> +where + F: DerefMut, + T: FixedSizeSerializable + Eq + Hash + Clone, + ID: Id +{ type Item = (ID, T); - fn next(& mut self) -> Option<(ID, T)> { + fn next(&mut self) -> Option<(ID, T)> { if self.index == self.size { return None; } else { - let value = T::deserialize(self.f); + let value = T::deserialize(&mut self.file); let id = ID::from(self.index); self.index += 1; return Some((id, value)); @@ -971,14 +1156,24 @@ impl<'a, T : FixedSizeSerializable + Eq + Hash + Clone, ID : Id> Itera } } + + /** Mapping from values to ids where the values require indexing. */ -pub struct IndirectMapping + Eq + Hash + Clone, ID : Id = u64> { +pub struct IndirectMapping +where + T : Serializable + Eq + Hash + Clone, + ID : Id, +{ pub (crate) store : Store, mapping : HashMap } -impl + Eq + Hash + Clone, ID : Id> IndirectMapping { +impl IndirectMapping +where + T : Serializable + Eq + Hash + Clone, + ID : Id, +{ /** Creates new mapping. */ @@ -1054,12 +1249,20 @@ impl + Eq + Hash + Clone, ID : Id> IndirectMapping StoreIterAll { - return self.store.iter_all(); + pub fn iter<'a>(&'a mut self) -> impl Iterator + 'a { + self.store.iter_all() + } + + pub fn into_iter<'a>(self) -> impl Iterator + 'a where Self: 'a{ + self.store.into_iter_all() } - pub fn savepoint_iter(& mut self, sp : & Savepoint) -> StoreIterAll { - return self.store.savepoint_iter_all(sp); + pub fn savepoint_iter<'a> (&'a mut self, sp: &Savepoint) -> impl Iterator + 'a { + self.store.savepoint_iter_all(sp) + } + + pub fn into_savepoint_iter<'a>(self, sp: &Savepoint) -> impl Iterator + 'a where Self: 'a { + self.store.into_savepoint_iter_all(sp) } } @@ -1086,12 +1289,12 @@ pub trait SplitKind : FixedSizeSerializable + Eq + Debug { */ pub struct SplitKindIter { i : u64, - why_oh_why : std::marker::PhantomData, + _t : std::marker::PhantomData, } impl SplitKindIter { pub fn new() -> SplitKindIter { - return SplitKindIter{i : 0, why_oh_why : std::marker::PhantomData{} }; + return SplitKindIter{i : 0, _t : std::marker::PhantomData{} }; } } @@ -1152,11 +1355,16 @@ impl> Indexable for SplitOffset { /** Split store contains single index, but multiple files that store the data based on its kind. */ -pub struct SplitStore, KIND : SplitKind, ID : Id = u64> { +pub struct SplitStore +where + T: Serializable, + KIND : SplitKind, + ID: Id, +{ name : String, pub (crate) indexer : Indexer, ID>, pub (crate) files : Vec, - why_oh_why : std::marker::PhantomData + _t : std::marker::PhantomData } impl, KIND: SplitKind, ID : Id> SplitStore { @@ -1176,7 +1384,7 @@ impl, KIND: SplitKind, ID : Id> SplitSto name : name.to_owned(), indexer : Indexer::new(root, name, readonly), files, - why_oh_why : std::marker::PhantomData{} + _t : std::marker::PhantomData{} }; LOG!(" {}: indices {}, splits {}", name, result.indexer.len(), result.files.len()); return result; @@ -1306,31 +1514,50 @@ impl, KIND: SplitKind, ID : Id> SplitSto return self.indexer.len(); } - pub fn savepoint_iter(& mut self, sp : & Savepoint) -> SplitStoreIterAll { + pub fn savepoint_iter<'a>(&'a mut self, sp: &Savepoint) -> impl Iterator + 'a { let mut max_offsets = Vec::new(); - let mut i = 0; - for _f in self.files.iter_mut() { - max_offsets.push(sp.limit_for(& format!("{}-{}.store", self.name, i))); - i += 1; + for i in 0..self.files.len() { + max_offsets.push(sp.limit_for(&format!("{}-{}.store", self.name, i))); } self.files[0].seek(SeekFrom::Start(0)).unwrap(); - return SplitStoreIterAll{ store : self, max_offsets, split : 0 } + return SplitStoreIterAll{ store: self, max_offsets, split: 0, _a: PhantomData }; } - // TODO add iterators - + pub fn into_savepoint_iter<'a>(mut self, sp: &Savepoint) -> impl Iterator + 'a where Self: 'a{ + let mut max_offsets = Vec::new(); + for i in 0..self.files.len() { + max_offsets.push(sp.limit_for(&format!("{}-{}.store", self.name, i))); + } + self.files[0].seek(SeekFrom::Start(0)).unwrap(); + + SplitStoreIterAll{ store: helpers::Holder(self), max_offsets, split: 0, _a: PhantomData } + } } -pub struct SplitStoreIterAll<'a, T : Serializable, KIND: SplitKind, ID : Id> { - store: &'a mut SplitStore, + +pub struct SplitStoreIterAll<'a, S, T, KIND, ID> +where + S: DerefMut>, + T: Serializable, + KIND: SplitKind, + ID: Id, +{ + store: S, max_offsets : Vec, split : usize, + _a: PhantomData<&'a S>, } -impl<'a, T : Serializable, KIND: SplitKind, ID : Id> Iterator for SplitStoreIterAll<'a, T, KIND, ID> { +impl<'a, S, T, KIND, ID> Iterator for SplitStoreIterAll<'a, S, T, KIND, ID> +where + S: DerefMut> + 'a, + T: Serializable, + KIND: SplitKind, + ID: Id, +{ type Item = (ID, KIND, T); - fn next(& mut self) -> Option<(ID, KIND, T)> { + fn next(&mut self) -> Option<(ID, KIND, T)> { loop { if self.store.files[self.split].seek(SeekFrom::Current(0)).unwrap() >= self.max_offsets[self.split] { self.split += 1; @@ -1347,6 +1574,7 @@ impl<'a, T : Serializable, KIND: SplitKind, ID : Id> Iter } } + /** Savepoint for the entire datastore. */ diff --git a/src/helpers.rs b/src/helpers.rs index 91937c4..cccac4a 100644 --- a/src/helpers.rs +++ b/src/helpers.rs @@ -1,6 +1,24 @@ use std::time::{SystemTime, UNIX_EPOCH, Duration}; use std::str; +use std::ops::{Deref, DerefMut}; + +/// Used in lieu of &mut T +pub struct Holder(pub T); + +impl Deref for Holder { + type Target = T; + + fn deref(&self) -> &Self::Target { + &self.0 + } +} + +impl DerefMut for Holder { + fn deref_mut(&mut self) -> &mut Self::Target { + &mut self.0 + } +} pub fn pct(value : usize, max : usize) -> String { if max == 0 { diff --git a/src/lib.rs b/src/lib.rs index b05933d..5dfc60d 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -507,8 +507,8 @@ pub struct StoreView<'a, T : db::Serializable, ID : db::Id = u64> { } impl<'a, T : db::Serializable, ID : db::Id > StoreView<'a, T, ID> { - pub fn iter(& mut self, sp : & Savepoint) -> db::StoreIterAll { - return self.guard.savepoint_iter_all(sp); + pub fn iter<'b>(&'b mut self, sp : & Savepoint) -> impl Iterator + 'b { + self.guard.savepoint_iter_all(sp) } } @@ -527,8 +527,8 @@ pub struct LinkedStoreView<'a, T : db::Serializable, ID : db::Id> { } impl<'a, T : db::Serializable, ID : db::Id > LinkedStoreView<'a, T, ID> { - pub fn iter(& mut self, sp : & Savepoint) -> db::LinkedStoreIterAll { - return self.guard.savepoint_iter_all(sp); + pub fn iter<'b>(&'b mut self, sp : & Savepoint) -> impl Iterator + 'b { + self.guard.savepoint_iter_all(sp) } } @@ -547,7 +547,7 @@ pub struct MappingView<'a, T : db::FixedSizeSerializable + Eq + Hash + } impl<'a, T : db::FixedSizeSerializable + Eq + Hash + Clone, ID : db::Id> MappingView<'a, T, ID> { - pub fn iter(& mut self, sp : & Savepoint) -> db::MappingIter { + pub fn iter<'b>(&'b mut self, sp : & Savepoint) -> impl Iterator + 'b { return self.guard.savepoint_iter(sp); } @@ -566,8 +566,8 @@ pub struct IndirectMappingView<'a, T : db::Serializable + Eq + Hash + } impl<'a, T : db::Serializable + Eq + Hash + Clone, ID : db::Id> IndirectMappingView<'a, T, ID> { - pub fn iter(& mut self, sp : & Savepoint) -> db::StoreIterAll { - return self.guard.savepoint_iter(sp); + pub fn iter<'b>(&'b mut self, sp : & Savepoint) -> impl Iterator + 'b { + self.guard.savepoint_iter(sp) } pub fn get(& mut self, id : ID) -> Option { @@ -588,8 +588,8 @@ pub struct SplitStoreView<'a, T : db::Serializable, KIND : db::SplitKi impl<'a, T : db::Serializable, KIND : db::SplitKind, ID : db::Id> SplitStoreView<'a, T, KIND, ID> { - pub fn iter(& mut self, sp : & Savepoint) -> db::SplitStoreIterAll { - return self.guard.savepoint_iter(sp); + pub fn iter<'b>(&'b mut self, sp : & Savepoint) -> impl Iterator + 'b { + self.guard.savepoint_iter(sp) } } @@ -606,8 +606,8 @@ pub struct SavepointsView<'a> { } impl<'a> SavepointsView<'a> { - pub fn iter(& mut self) -> db::LinkedStoreIterAll { - return self.guard.iter_all(); + pub fn iter<'b>(&'b mut self) -> impl Iterator + 'b { + self.guard.iter_all() } } From fb03b4f76791e26423b9c9c021890ad9605163ab Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Colette=20=F0=9F=8C=A9=EF=B8=8E=E2=80=8D=F0=9F=92=9D=20Ker?= =?UTF-8?q?r?= Date: Mon, 1 Mar 2021 17:23:28 +0100 Subject: [PATCH 3/3] Slightly simplify the types by making Holder pull double duty as an enum Before everything had a DerefMut trait constraint on the argument, now things are all Holders Holder now has two enum variants, one for a ref, the other for ownership --- src/db.rs | 155 +++++++++++++++++++++---------------------------- src/helpers.rs | 19 ++++-- 2 files changed, 79 insertions(+), 95 deletions(-) diff --git a/src/db.rs b/src/db.rs index 4f18379..788630c 100644 --- a/src/db.rs +++ b/src/db.rs @@ -15,14 +15,14 @@ Like store, */ -use std::{fs::{File, OpenOptions}, marker::PhantomData, ops::DerefMut}; +use std::{fs::{File, OpenOptions}, marker::PhantomData}; use std::io::{Seek, SeekFrom, Read, Write}; use byteorder::*; use std::collections::*; use std::hash::*; use std::fmt::{Debug}; use std::convert::From; -use crate::helpers; +use crate::helpers::{self, Holder}; use crate::settings::SETTINGS; use crate::LOG; @@ -213,7 +213,7 @@ impl, ID : Id> Indexer { } let size = f.seek(SeekFrom::End(0)).unwrap() / T::SIZE; return Indexer{ name : name.to_owned(), f, size, _t : std::marker::PhantomData{} }; - } + } pub fn get(& mut self, id : ID) -> Option { if id.into() < self.size { @@ -261,43 +261,40 @@ impl, ID : Id> Indexer { pub fn iter<'a>(&'a mut self) -> impl Iterator + 'a { self.f.seek(SeekFrom::Start(0)).unwrap(); - return IndexerIterator{indexer: self, id : ID::from(0), max_offset: u64::MAX, _a: PhantomData}; + return IndexerIterator{indexer: Holder::Ref(self), id : ID::from(0), max_offset: u64::MAX}; } pub fn into_iter<'a>(mut self) -> impl Iterator + 'a where Self: 'a{ self.f.seek(SeekFrom::Start(0)).unwrap(); - return IndexerIterator{indexer: helpers::Holder(self), id : ID::from(0), max_offset: u64::MAX, _a: PhantomData}; + return IndexerIterator{indexer: Holder::Own(self), id : ID::from(0), max_offset: u64::MAX }; } pub fn savepoint_iter<'a>(&'a mut self, sp : & Savepoint) -> impl Iterator + 'a { let max_offset = sp.limit_for(&self.name); self.f.seek(SeekFrom::Start(0)).unwrap(); - return IndexerIterator{indexer : self, id: ID::from(0), max_offset, _a: PhantomData }; + return IndexerIterator{indexer : Holder::Ref(self), id: ID::from(0), max_offset }; } pub fn into_savepoint_iter<'a>(mut self, sp : & Savepoint) ->impl Iterator + 'a where Self: 'a { let max_offset = sp.limit_for(&self.name); self.f.seek(SeekFrom::Start(0)).unwrap(); - return IndexerIterator{indexer: helpers::Holder(self), id: ID::from(0), max_offset, _a: PhantomData }; + return IndexerIterator{indexer: Holder::Own(self), id: ID::from(0), max_offset }; } } -pub struct IndexerIterator<'a, I, T, ID> +pub struct IndexerIterator<'a, T, ID> where - I: DerefMut>, T: Indexable + Serializable, ID: Id, { - indexer: I, + indexer: Holder<'a, Indexer>, id: ID, max_offset: u64, - _a: PhantomData<&'a I> } -impl<'a, I, T, ID> Iterator for IndexerIterator<'a, I, T, ID> +impl<'a, T, ID> Iterator for IndexerIterator<'a, T, ID> where - I: DerefMut>, T: Indexable + Serializable, ID: Id, { @@ -462,7 +459,7 @@ where Returns the latest stored value for every id. The ids are guaranteed to be increasing. */ pub fn iter<'a>(&'a mut self) -> impl Iterator + 'a { - StoreIter::new(&mut self.f, self.indexer.iter()) + StoreIter::new(Holder::Ref(&mut self.f), self.indexer.iter()) } /** Iterates over the stored values. @@ -470,7 +467,7 @@ where Returns the latest stored value for every id. The ids are guaranteed to be increasing. */ pub fn into_iter<'a>(self) -> impl Iterator + 'a where Self: 'a{ - StoreIter::new(helpers::Holder(self.f), self.indexer.into_iter()) + StoreIter::new(Holder::Own(self.f), self.indexer.into_iter()) } /** Iterates over all stored values. @@ -479,7 +476,7 @@ where */ pub fn iter_all<'a>(&'a mut self) -> impl Iterator + 'a { self.f.seek(SeekFrom::Start(0)).unwrap(); - return StoreIterAll{ store: self, max_offset : u64::MAX, _a: PhantomData }; + return StoreIterAll{ store: Holder::Ref(self), max_offset : u64::MAX }; } /** Iterates over all stored values. @@ -488,19 +485,19 @@ where */ pub fn into_iter_all<'a>(mut self) -> impl Iterator + 'a where Self: 'a{ self.f.seek(SeekFrom::Start(0)).unwrap(); - return StoreIterAll{ store: helpers::Holder(self), max_offset : u64::MAX, _a: PhantomData }; + return StoreIterAll{ store: Holder::Own(self), max_offset : u64::MAX }; } pub fn savepoint_iter_all<'a>(&'a mut self, sp : & Savepoint) -> impl Iterator + 'a { let max_offset = sp.limit_for(& format!("{}.store", self.name())); self.f.seek(SeekFrom::Start(0)).unwrap(); - return StoreIterAll{store: self, max_offset, _a: PhantomData }; + return StoreIterAll{store: Holder::Ref(self), max_offset }; } pub fn into_savepoint_iter_all<'a>(mut self, sp: &Savepoint) -> impl Iterator + 'a where Self: 'a{ let max_offset = sp.limit_for(& format!("{}.store", self.name())); self.f.seek(SeekFrom::Start(0)).unwrap(); - return StoreIterAll{ store: helpers::Holder(self), max_offset, _a: PhantomData }; + return StoreIterAll{ store: Holder::Own(self), max_offset }; } /** Reads the record from a file. @@ -526,40 +523,35 @@ where /** Latest store iterator does not support savepoints since the indices can be udpated. */ - pub struct StoreIter<'a, F, II, T, ID> + pub struct StoreIter<'a, I, T, ID> where - F: DerefMut + 'a, - II: Iterator + 'a, + I: Iterator, T: Serializable, ID: Id, { - file: F, - iiter: II, + file: Holder<'a, File>, + iiter:I, _t: PhantomData, - _a: PhantomData<(&'a F, &'a II)>, } -impl<'a, F, II, T, ID> StoreIter<'a, F, II, T, ID> +impl<'a, I, T, ID> StoreIter<'a, I, T, ID> where - F: DerefMut, - II: Iterator + 'a, + I: Iterator, T: Serializable, ID: Id { - fn new(file: F, iiter: II) -> StoreIter<'a, F, II, T, ID>{ + fn new(file: Holder<'a, File>, iiter: I) -> StoreIter<'a, I, T, ID>{ StoreIter{ file, iiter, - _t: PhantomData, - _a: PhantomData, + _t: PhantomData } } } -impl<'a, F, II, T, ID> Iterator for StoreIter<'a, F, II, T, ID> +impl<'a, I, T, ID> Iterator for StoreIter<'a, I, T, ID> where - F: DerefMut, - II: Iterator + 'a, + I: Iterator, T: Serializable, ID: Id { @@ -579,20 +571,17 @@ where -pub struct StoreIterAll<'a, T, S, ID> +pub struct StoreIterAll<'a, T, ID> where - S: DerefMut> + 'a, T: Serializable, ID: Id { - store: S, + store: Holder<'a, Store>, max_offset : u64, - _a: PhantomData<&'a S>, } -impl<'a, T, S, ID> Iterator for StoreIterAll<'a, T, S, ID> +impl<'a, T, ID> Iterator for StoreIterAll<'a, T, ID> where - S: DerefMut> + 'a, T: Serializable, ID: Id { @@ -760,7 +749,7 @@ impl, ID : Id> LinkedStore { Returns the latest stored value for every id. The ids are guaranteed to be increasing. */ pub fn iter<'a>(&'a mut self) -> impl Iterator + 'a { - LinkedStoreIter::new(&mut self.f, self.indexer.iter()) + LinkedStoreIter::new(Holder::Ref(&mut self.f), self.indexer.iter()) } /** Iterates over the stored values. @@ -768,7 +757,7 @@ impl, ID : Id> LinkedStore { Returns the latest stored value for every id. The ids are guaranteed to be increasing. */ pub fn into_iter<'a>(self) -> impl Iterator + 'a where Self: 'a{ - LinkedStoreIter::new(helpers::Holder(self.f), self.indexer.into_iter()) + LinkedStoreIter::new(Holder::Own(self.f), self.indexer.into_iter()) } /** Iterates over all stored values. @@ -777,7 +766,7 @@ impl, ID : Id> LinkedStore { */ pub fn iter_all<'a>(&'a mut self) -> impl Iterator + 'a { self.f.seek(SeekFrom::Start(0)).unwrap(); - LinkedStoreIterAll{ store: self, max_offset: u64::MAX } + LinkedStoreIterAll{ store: Holder::Ref(self), max_offset: u64::MAX } } /** Iterates over all stored values. @@ -786,19 +775,19 @@ impl, ID : Id> LinkedStore { */ pub fn into_iter_all<'a>(mut self) -> impl Iterator + 'a where Self: 'a{ self.f.seek(SeekFrom::Start(0)).unwrap(); - LinkedStoreIterAll{ store: helpers::Holder(self), max_offset: u64::MAX } + LinkedStoreIterAll{ store: Holder::Own(self), max_offset: u64::MAX } } pub fn savepoint_iter_all<'a>(&'a mut self, sp: &Savepoint) -> impl Iterator + 'a { let max_offset = sp.limit_for(&format!("{}.store",self.name())); self.f.seek(SeekFrom::Start(0)).unwrap(); - LinkedStoreIterAll{ store : self, max_offset } + LinkedStoreIterAll{ store : Holder::Ref(self), max_offset } } pub fn into_savepoint_iter_all<'a>(mut self, sp: &Savepoint) -> impl Iterator + 'a where Self: 'a{ let max_offset = sp.limit_for(&format!("{}.store",self.name())); self.f.seek(SeekFrom::Start(0)).unwrap(); - LinkedStoreIterAll{ store: helpers::Holder(self), max_offset } + LinkedStoreIterAll{ store: Holder::Own(self), max_offset } } /** Given an id, returns an iterator over all values ever stored for it. @@ -807,7 +796,7 @@ impl, ID : Id> LinkedStore { */ pub fn iter_id<'a>(&'a mut self, id : ID) -> impl Iterator + 'a { let offset = self.indexer.get(id); - LinkedStoreIterId{ store: self, offset, _a: PhantomData} + LinkedStoreIterId{ store: Holder::Ref(self), offset } } /** Given an id, returns an iterator over all values ever stored for it. @@ -816,7 +805,7 @@ impl, ID : Id> LinkedStore { */ pub fn into_iter_id<'a>(mut self, id : ID) -> impl Iterator + 'a where Self: 'a { let offset = self.indexer.get(id); - return LinkedStoreIterId{ store: helpers::Holder(self), offset, _a: PhantomData }; + return LinkedStoreIterId{ store: Holder::Own(self), offset }; } /** Reads the record from a file. @@ -844,26 +833,24 @@ impl, ID : Id> LinkedStore { } } -pub struct LinkedStoreIter +pub struct LinkedStoreIter<'a, I, T, ID> where + I: Iterator, T: Serializable, - F: DerefMut, - II: Iterator, ID: Id, { - iiter: II, - file: F, + iiter: I, + file: Holder<'a, File>, _t: std::marker::PhantomData, } -impl LinkedStoreIter +impl<'a, I, T, ID> LinkedStoreIter<'a, I, T, ID> where + I: Iterator, T: Serializable, - F: DerefMut, - II: Iterator, ID: Id, { - fn new(file: F, iiter: II) -> LinkedStoreIter { + fn new(file: Holder<'a, File>, iiter: I) -> LinkedStoreIter<'a, I, T, ID> { return LinkedStoreIter{ iiter, file, @@ -872,11 +859,10 @@ where } } -impl Iterator for LinkedStoreIter +impl<'a, I, T, ID> Iterator for LinkedStoreIter<'a, I, T, ID> where + I: Iterator, T: Serializable, - F: DerefMut, - II: Iterator, ID: Id, { type Item = (ID, T); @@ -895,20 +881,18 @@ where -pub struct LinkedStoreIterAll +pub struct LinkedStoreIterAll<'a, T, ID> where T: Serializable, - LS: DerefMut>, ID: Id { - store: LS, + store: Holder<'a, LinkedStore>, max_offset: u64, } -impl Iterator for LinkedStoreIterAll +impl<'a, T, ID> Iterator for LinkedStoreIterAll<'a, T, ID> where T: Serializable, - LS: DerefMut>, ID: Id { type Item = (ID, T); @@ -924,21 +908,18 @@ where } } } -pub struct LinkedStoreIterId<'a, T, LS, ID> +pub struct LinkedStoreIterId<'a, T, ID> where T: Serializable, - LS: DerefMut>, ID: Id, { - store: LS, + store: Holder<'a, LinkedStore>, offset: Option, - _a: PhantomData<&'a LS>, } -impl<'a, T, LS, ID> Iterator for LinkedStoreIterId<'a, T, LS, ID> +impl<'a, T, ID> Iterator for LinkedStoreIterId<'a, T, ID> where T: Serializable, - LS: DerefMut>, ID: Id, { type Item = T; @@ -1030,7 +1011,7 @@ impl + Eq + Hash + Clone, ID : Id> Mapping + Eq + Hash + Clone, ID : Id> Mapping(&'a mut self) -> impl Iterator + 'a { self.f.seek(SeekFrom::Start(0)).unwrap(); let size = self.size; - MappingIter{file: &mut self.f, index: 0, size, _t: PhantomData, _a: PhantomData } + MappingIter{file: Holder::Ref(&mut self.f), index: 0, size, _t: PhantomData } } pub fn savepoint_iter<'a>(&'a mut self, sp : &Savepoint) -> impl Iterator + 'a { let max_offset = sp.limit_for(& format!("{}.mapping", self.name())) / (std::mem::size_of::() as u64); self.f.seek(SeekFrom::Start(0)).unwrap(); - MappingIter{file: &mut self.f, index : 0, size: max_offset, _t: PhantomData, _a: PhantomData } + MappingIter{file: Holder::Ref(&mut self.f), index : 0, size: max_offset, _t: PhantomData } } pub fn into_iter<'a>(mut self) -> impl Iterator + 'a where Self: 'a{ self.f.seek(SeekFrom::Start(0)).unwrap(); let size = self.size; - MappingIter{file: helpers::Holder(self.f), index: 0, size, _t: PhantomData, _a: PhantomData } + MappingIter{file: Holder::Own(self.f), index: 0, size, _t: PhantomData } } pub fn into_savepoint_iter<'a>(mut self, sp : &Savepoint) -> impl Iterator + 'a where Self: 'a{ let max_offset = sp.limit_for(& format!("{}.mapping", self.name())) / (std::mem::size_of::() as u64); self.f.seek(SeekFrom::Start(0)).unwrap(); - MappingIter{file: helpers::Holder(self.f), index : 0, size: max_offset, _t: PhantomData, _a: PhantomData } + MappingIter{file: Holder::Own(self.f), index : 0, size: max_offset, _t: PhantomData } } } -pub struct MappingIter<'a, F, T, ID> +pub struct MappingIter<'a, T, ID> where - F: DerefMut, T: FixedSizeSerializable + Eq + Hash + Clone, ID: Id { - file: F, + file: Holder<'a, File>, index: u64, size: u64, _t: std::marker::PhantomData<(T, ID)>, - _a: PhantomData<&'a F> } -impl Iterator for MappingIter<'_, F, T, ID> +impl Iterator for MappingIter<'_, T, ID> where - F: DerefMut, T: FixedSizeSerializable + Eq + Hash + Clone, ID: Id { @@ -1520,7 +1498,7 @@ impl, KIND: SplitKind, ID : Id> SplitSto max_offsets.push(sp.limit_for(&format!("{}-{}.store", self.name, i))); } self.files[0].seek(SeekFrom::Start(0)).unwrap(); - return SplitStoreIterAll{ store: self, max_offsets, split: 0, _a: PhantomData }; + return SplitStoreIterAll{ store: Holder::Ref(self), max_offsets, split: 0 }; } pub fn into_savepoint_iter<'a>(mut self, sp: &Savepoint) -> impl Iterator + 'a where Self: 'a{ @@ -1530,27 +1508,24 @@ impl, KIND: SplitKind, ID : Id> SplitSto } self.files[0].seek(SeekFrom::Start(0)).unwrap(); - SplitStoreIterAll{ store: helpers::Holder(self), max_offsets, split: 0, _a: PhantomData } + SplitStoreIterAll{ store: Holder::Own(self), max_offsets, split: 0 } } } -pub struct SplitStoreIterAll<'a, S, T, KIND, ID> +pub struct SplitStoreIterAll<'a, T, KIND, ID> where - S: DerefMut>, T: Serializable, KIND: SplitKind, ID: Id, { - store: S, + store: Holder<'a, SplitStore>, max_offsets : Vec, split : usize, - _a: PhantomData<&'a S>, } -impl<'a, S, T, KIND, ID> Iterator for SplitStoreIterAll<'a, S, T, KIND, ID> +impl<'a, T, KIND, ID> Iterator for SplitStoreIterAll<'a, T, KIND, ID> where - S: DerefMut> + 'a, T: Serializable, KIND: SplitKind, ID: Id, diff --git a/src/helpers.rs b/src/helpers.rs index cccac4a..d089d05 100644 --- a/src/helpers.rs +++ b/src/helpers.rs @@ -4,19 +4,28 @@ use std::str; use std::ops::{Deref, DerefMut}; /// Used in lieu of &mut T -pub struct Holder(pub T); +pub enum Holder<'a, T>{ + Ref(&'a mut T), + Own(T), +} -impl Deref for Holder { +impl<'a, T> Deref for Holder<'a, T> { type Target = T; fn deref(&self) -> &Self::Target { - &self.0 + match self { + Holder::Ref(r) => r, + Holder::Own(t) => t, + } } } -impl DerefMut for Holder { +impl<'a, T> DerefMut for Holder<'a, T> { fn deref_mut(&mut self) -> &mut Self::Target { - &mut self.0 + match self { + Holder::Ref(r) => r, + Holder::Own(t) => t, + } } }