summaryrefslogtreecommitdiffstats
path: root/lib
diff options
context:
space:
mode:
authorMatthias Beyer <mail@beyermatthias.de>2018-04-30 21:07:37 +0200
committerGitHub <noreply@github.com>2018-04-30 21:07:37 +0200
commit900d9f0984fc64e6c9d59d853f27c294cba33a1c (patch)
tree9977cc4c0c1b6a95cdfddfe5028ef19f1d90379c /lib
parent41615bcc9706d2a960584846c7b549a3447862a5 (diff)
parent860e47a256d38fd959ea83acd2dd2023dee0ee83 (diff)
Merge pull request #1474 from matthiasbeyer/libimagstore/cache-flush
Add functionality to flush the internal cache
Diffstat (limited to 'lib')
-rw-r--r--lib/core/libimagstore/src/store.rs43
1 files changed, 43 insertions, 0 deletions
diff --git a/lib/core/libimagstore/src/store.rs b/lib/core/libimagstore/src/store.rs
index 30c00af3..7edf3f05 100644
--- a/lib/core/libimagstore/src/store.rs
+++ b/lib/core/libimagstore/src/store.rs
@@ -477,6 +477,49 @@ impl Store {
Ok(())
}
+ /// Flush the store internal cache
+ ///
+ /// This is helpful if a lot of entries are beeing read/written, because the store holds the
+ /// file handles internally. At some point, the OS simply errors with "Too many files open".
+ /// With this function, not-borrowed entries can be flushed back to disk and thus file handles
+ /// are dropped.
+ ///
+ /// After the flushables are dropped, the internal cache is shrinked to fit the number of
+ /// elements still in the cache.
+ ///
+ pub fn flush_cache(&self) -> Result<()> {
+ // We borrow this early so that between the aggregation of the flushables and the actual
+ // flush, there is no borrowing from the store.
+ let mut hsmap = self.entries.write().map_err(|_| SE::from_kind(SEK::LockPoisoned))?;
+ let mut to_flush = vec![];
+
+ for (storeid, se) in hsmap.deref() {
+ if !se.is_borrowed() {
+ to_flush.push(storeid.clone());
+ }
+ }
+
+ for id in to_flush {
+ let _ = hsmap.remove(&id);
+ }
+
+ hsmap.shrink_to_fit();
+
+ Ok(())
+ }
+
+ /// The number of elements in the internal cache
+ pub fn cache_size(&self) -> Result<usize> {
+ let hsmap = self.entries.read().map_err(|_| SE::from_kind(SEK::LockPoisoned))?;
+ Ok(hsmap.iter().count())
+ }
+
+ /// The size of the internal cache
+ pub fn cache_capacity(&self) -> Result<usize> {
+ let hsmap = self.entries.read().map_err(|_| SE::from_kind(SEK::LockPoisoned))?;
+ Ok(hsmap.capacity())
+ }
+
/// Get a copy of a given entry, this cannot be used to mutate the one on disk
///
/// # Return value