#pragma once #include #include #include #include #include "shared.hpp" #include "arguments.hpp" #include "singleton.hpp" namespace ace { class dispatcher { public: dispatcher() : _ready(true) {} virtual bool call(const std::string & name_, const arguments & args_, std::string & result_) { if (_methods.find(name_) == _methods.end()) { // @TODO: Exceptions return false; } return _methods[name_](args_, result_); } bool add(const std::string & name_, std::function func_) { if (_methods.find(name_) != _methods.end()) { // @TODO: Exceptions return false; } _methods[name_] = func_; return true; } bool ready() const { return _ready; } void ready(bool r) { _ready.exchange(r); } protected: std::unordered_map < std::string, std::function > _methods; std::atomic_bool _ready; }; class dispatch : public dispatcher, public singleton { }; struct dispatch_message { dispatch_message(const std::string & command_, const arguments & args_, const uint64_t id_) : command(command_), args(args_), id(id_) {} std::string command; arguments args; uint64_t id; }; struct dispatch_result { dispatch_result() {} dispatch_result(const std::string &res, const uint64_t id_) : message(res), id(id_) {} std::string message; uint64_t id; }; class threaded_dispatcher : public dispatcher { public: threaded_dispatcher() : _worker(&ace::threaded_dispatcher::monitor, this) { } ~threaded_dispatcher() {} bool call(const std::string & name_, const arguments & args_, std::string & result_, bool threaded) { if (_methods.find(name_) == _methods.end()) { // @TODO: Exceptions return false; } if (threaded) { std::lock_guard lock(_messages_lock); _messages.push(dispatch_message(name_, args_, _message_id)); // @TODO: We should provide an interface for this serialization. std::stringstream ss; ss << "[\"result_id\", \"" << _message_id << "\" ]"; result_ = ss.str(); _message_id = _message_id + 1; } else { #ifdef _DEBUG if (_messages.front().command != "fetch_result") { LOG(TRACE) << "dispatch[immediate]:\t[" << name_ << "] { " << args_.get() << " }"; } #endif return dispatcher::call(name_, args_, result_); } return true; } bool call(const std::string & name_, const arguments & args_, std::string & result_) override { return call(name_, args_, result_, false); } void push_result(const dispatch_result & result) { { std::lock_guard lock(_results_lock); _results.push(result); } } void push_result(const std::string & result) { push_result(dispatch_result(result, -1)); } protected: void monitor() { _ready = false; while (true) { { std::lock_guard lock(_messages_lock); while (!_messages.empty()) { if (_ready) { dispatch_result result; result.id = _messages.front().id; result.message.resize(4096); #ifdef _DEBUG if (_messages.front().command != "fetch_result") { LOG(TRACE) << "dispatch[threaded]:\t[" << _messages.front().command << "]"; if (_messages.front().args.size() > 0) { // LOG(TRACE) << "\t{ " << _messages.front().args.get() << " }"; } } #endif dispatcher::call(_messages.front().command, _messages.front().args, result.message); { std::lock_guard lock(_results_lock); _results.push(result); } _messages.pop(); } } } sleep(5); } } std::queue _results; std::mutex _results_lock; std::queue _messages; std::mutex _messages_lock; std::thread _worker; uint64_t _message_id; }; class threaded_dispatch : public threaded_dispatcher, public singleton { }; };