Cache-Entries are purged after a max lifetime
This commit is contained in:
parent
c18893c151
commit
c1551121c8
@ -1,4 +1,5 @@
|
||||
#include "Cache.h"
|
||||
#include "PeriodicTask.h"
|
||||
|
||||
using std::unique_ptr;
|
||||
using std::make_unique;
|
||||
@ -10,8 +11,14 @@ namespace blockstore {
|
||||
namespace caching {
|
||||
|
||||
constexpr uint32_t Cache::MAX_ENTRIES;
|
||||
constexpr double Cache::PURGE_LIFETIME_SEC;
|
||||
constexpr double Cache::PURGE_INTERVAL;
|
||||
constexpr double Cache::MAX_LIFETIME_SEC;
|
||||
|
||||
Cache::Cache(): _cachedBlocks() {
|
||||
Cache::Cache(): _cachedBlocks(), _timeoutFlusher(nullptr) {
|
||||
//Don't initialize timeoutFlusher in the initializer list,
|
||||
//because it then might already call Cache::popOldEntries() before Cache is done constructing
|
||||
_timeoutFlusher = make_unique<PeriodicTask>(std::bind(&Cache::_popOldEntries, this), PURGE_INTERVAL);
|
||||
}
|
||||
|
||||
Cache::~Cache() {
|
||||
@ -38,5 +45,14 @@ void Cache::push(unique_ptr<Block> block) {
|
||||
_cachedBlocks.push(key, make_unique<CacheEntry>(std::move(block)));
|
||||
}
|
||||
|
||||
void Cache::_popOldEntries() {
|
||||
lock_guard<mutex> lock(_mutex);
|
||||
while(_cachedBlocks.size() > 0 && _cachedBlocks.peek().ageSeconds() > PURGE_LIFETIME_SEC) {
|
||||
double age = _cachedBlocks.peek().ageSeconds();
|
||||
printf("Removing block with age: %f\n", age);
|
||||
_cachedBlocks.pop();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
@ -10,6 +10,7 @@
|
||||
|
||||
namespace blockstore {
|
||||
namespace caching {
|
||||
class PeriodicTask;
|
||||
|
||||
//TODO Test
|
||||
//TODO Also throw blocks out after a timeout
|
||||
@ -17,6 +18,10 @@ namespace caching {
|
||||
class Cache {
|
||||
public:
|
||||
static constexpr uint32_t MAX_ENTRIES = 1000;
|
||||
//TODO Experiment with good values
|
||||
static constexpr double PURGE_LIFETIME_SEC = 0.5; //When an entry has this age, it will be purged from the cache
|
||||
static constexpr double PURGE_INTERVAL = 0.5; // With this interval, we check for entries to purge
|
||||
static constexpr double MAX_LIFETIME_SEC = PURGE_LIFETIME_SEC + PURGE_INTERVAL; // This is the oldest age an entry can reach (given purging works in an ideal world, i.e. with the ideal interval and in zero time)
|
||||
|
||||
Cache();
|
||||
virtual ~Cache();
|
||||
@ -25,8 +30,12 @@ public:
|
||||
std::unique_ptr<Block> pop(const Key &key);
|
||||
|
||||
private:
|
||||
void _popOldEntries();
|
||||
|
||||
|
||||
mutable std::mutex _mutex;
|
||||
QueueMap<Key, CacheEntry> _cachedBlocks;
|
||||
std::unique_ptr<PeriodicTask> _timeoutFlusher;
|
||||
};
|
||||
|
||||
}
|
||||
|
@ -5,6 +5,7 @@
|
||||
#include <ctime>
|
||||
#include <memory>
|
||||
#include <messmer/cpp-utils/macros.h>
|
||||
#include <boost/date_time/posix_time/posix_time_types.hpp>
|
||||
|
||||
namespace blockstore {
|
||||
class Block;
|
||||
@ -12,13 +13,13 @@ namespace caching {
|
||||
|
||||
class CacheEntry {
|
||||
public:
|
||||
CacheEntry(std::unique_ptr<Block> block): _lastAccess(time(nullptr)), _block(std::move(block)) {
|
||||
CacheEntry(std::unique_ptr<Block> block): _lastAccess(currentTime()), _block(std::move(block)) {
|
||||
}
|
||||
|
||||
CacheEntry(CacheEntry &&) = default;
|
||||
|
||||
double ageSeconds() const {
|
||||
return difftime(time(nullptr), _lastAccess);
|
||||
return ((double)(currentTime() - _lastAccess).total_nanoseconds()) / ((double)1000000000);
|
||||
}
|
||||
|
||||
std::unique_ptr<Block> releaseBlock() {
|
||||
@ -34,10 +35,14 @@ public:
|
||||
}
|
||||
|
||||
private:
|
||||
time_t _lastAccess;
|
||||
boost::posix_time::ptime _lastAccess;
|
||||
std::unique_ptr<Block> _block;
|
||||
const CacheEntry *_nextEntry;
|
||||
|
||||
static boost::posix_time::ptime currentTime() {
|
||||
return boost::posix_time::microsec_clock::local_time();
|
||||
}
|
||||
|
||||
DISALLOW_COPY_AND_ASSIGN(CacheEntry);
|
||||
};
|
||||
|
||||
|
32
implementations/caching/PeriodicTask.cpp
Normal file
32
implementations/caching/PeriodicTask.cpp
Normal file
@ -0,0 +1,32 @@
|
||||
#include "PeriodicTask.h"
|
||||
|
||||
using std::function;
|
||||
using std::cerr;
|
||||
using std::endl;
|
||||
|
||||
namespace blockstore {
|
||||
namespace caching {
|
||||
|
||||
PeriodicTask::PeriodicTask(function<void ()> task, double intervalSec) : _thread(), _task(task), _intervalSec(intervalSec) {
|
||||
_thread = boost::thread([this]() {
|
||||
boost::chrono::nanoseconds interval((uint64_t)(UINT64_C(1000000000) * _intervalSec));
|
||||
try {
|
||||
while(true) {
|
||||
boost::this_thread::sleep_for(interval);
|
||||
_task();
|
||||
}
|
||||
} catch (const boost::thread_interrupted &e) {
|
||||
//Do nothing, exit thread.
|
||||
} catch (...) {
|
||||
cerr << "PeriodicTask crashed" << endl;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
PeriodicTask::~PeriodicTask() {
|
||||
_thread.interrupt();
|
||||
_thread.join();
|
||||
}
|
||||
|
||||
}
|
||||
}
|
25
implementations/caching/PeriodicTask.h
Normal file
25
implementations/caching/PeriodicTask.h
Normal file
@ -0,0 +1,25 @@
|
||||
#ifndef MESSMER_BLOCKSTORE_IMPLEMENTATIONS_CACHING_PERIODICTASK_H_
|
||||
#define MESSMER_BLOCKSTORE_IMPLEMENTATIONS_CACHING_PERIODICTASK_H_
|
||||
|
||||
#include <functional>
|
||||
#include <boost/thread.hpp>
|
||||
|
||||
namespace blockstore {
|
||||
namespace caching {
|
||||
|
||||
//TODO Test cases
|
||||
class PeriodicTask {
|
||||
public:
|
||||
PeriodicTask(std::function<void ()> task, double intervalSec);
|
||||
virtual ~PeriodicTask();
|
||||
|
||||
private:
|
||||
boost::thread _thread;
|
||||
std::function<void ()> _task;
|
||||
double _intervalSec;
|
||||
};
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
#endif
|
@ -41,6 +41,10 @@ public:
|
||||
return pop(*_sentinel.next->key);
|
||||
}
|
||||
|
||||
const Value &peek() {
|
||||
return *_sentinel.next->value;
|
||||
}
|
||||
|
||||
uint32_t size() {
|
||||
return _entries.size();
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user