2017-06-07 16:15:08 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <mutex>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "utils/scheduler.hpp"
|
|
|
|
|
2018-04-22 14:31:09 +08:00
|
|
|
namespace utils {
|
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
/**
|
|
|
|
* @brief - Provides execution of jobs in job queue on one thread with 'pause'
|
|
|
|
* time between two consecutives starts.
|
|
|
|
*/
|
2017-09-07 20:43:27 +08:00
|
|
|
class Executor {
|
2017-06-07 16:15:08 +08:00
|
|
|
public:
|
|
|
|
template <typename TRep, typename TPeriod>
|
2017-11-09 20:46:37 +08:00
|
|
|
explicit Executor(const std::chrono::duration<TRep, TPeriod> pause) {
|
2017-12-18 17:20:35 +08:00
|
|
|
DCHECK(pause > std::chrono::seconds(0))
|
|
|
|
<< "Duration between executions should be reasonable";
|
2018-02-23 21:35:16 +08:00
|
|
|
scheduler_.Run("Executor", pause, std::bind(&Executor::Execute, this));
|
2017-06-07 16:15:08 +08:00
|
|
|
}
|
|
|
|
|
2017-09-07 20:43:27 +08:00
|
|
|
~Executor() {
|
2017-06-07 16:15:08 +08:00
|
|
|
// Be sure to first stop scheduler because otherwise we might destroy the
|
|
|
|
// mutex before the scheduler and that might cause problems since mutex is
|
|
|
|
// used in Execute method passed to scheduler along with jobs vector.
|
|
|
|
scheduler_.Stop();
|
|
|
|
}
|
|
|
|
|
2017-12-18 17:20:35 +08:00
|
|
|
Executor(Executor &&e) = default;
|
|
|
|
Executor &operator=(Executor &&) = default;
|
|
|
|
Executor(const Executor &e) = delete;
|
|
|
|
Executor &operator=(const Executor &) = delete;
|
|
|
|
|
2017-06-07 16:15:08 +08:00
|
|
|
/**
|
|
|
|
* @brief - Add function to job queue.
|
|
|
|
*/
|
|
|
|
int64_t RegisterJob(const std::function<void()> &f) {
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(update_mutex_);
|
|
|
|
id_job_pairs_.emplace_back(std::make_pair(++count_, f));
|
|
|
|
return id_job_pairs_.back().first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief - Remove id from job queue.
|
|
|
|
*/
|
|
|
|
void UnRegisterJob(const int64_t id) {
|
|
|
|
{
|
|
|
|
// First wait for execute lock and then for the update lock because
|
|
|
|
// execute lock will be unavailable for longer and there is no point in
|
|
|
|
// blocking other threads with update lock.
|
|
|
|
std::unique_lock<std::mutex> execute_lock(execute_mutex_);
|
|
|
|
std::unique_lock<std::mutex> update_lock(update_mutex_);
|
|
|
|
|
|
|
|
for (auto id_job_pair_it = id_job_pairs_.begin();
|
|
|
|
id_job_pair_it != id_job_pairs_.end(); ++id_job_pair_it) {
|
|
|
|
if (id_job_pair_it->first == id) {
|
|
|
|
id_job_pairs_.erase(id_job_pair_it);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
/**
|
|
|
|
* @brief - Execute method executes jobs from id_job_pairs vector.
|
|
|
|
* The reason for doing double locking is the following: we don't want to
|
|
|
|
* block creation of new jobs since that will slow down all of memgraph so we
|
|
|
|
* use a special lock for job update. Execute lock is here so that we can
|
|
|
|
* guarantee that after some job is unregistered it's also stopped.
|
|
|
|
*/
|
|
|
|
void Execute() {
|
|
|
|
std::unique_lock<std::mutex> execute_lock(execute_mutex_);
|
|
|
|
std::vector<std::pair<int, std::function<void()>>> id_job_pairs;
|
|
|
|
|
|
|
|
// Acquire newest current version of jobs but being careful not to access
|
|
|
|
// the vector in corrupt state.
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> update_lock(update_mutex_);
|
|
|
|
id_job_pairs = id_job_pairs_;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto id_job_pair : id_job_pairs) {
|
|
|
|
id_job_pair.second();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t count_{0};
|
|
|
|
std::mutex execute_mutex_;
|
|
|
|
std::mutex update_mutex_;
|
2017-08-17 15:19:58 +08:00
|
|
|
Scheduler scheduler_;
|
2017-06-07 16:15:08 +08:00
|
|
|
std::vector<std::pair<int, std::function<void()>>> id_job_pairs_;
|
|
|
|
};
|
2018-04-22 14:31:09 +08:00
|
|
|
|
|
|
|
} // namespace utils
|