forked from qicosmos/cosmos
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmodern_messagebus.hpp
105 lines (86 loc) · 2.85 KB
/
modern_messagebus.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
#pragma once
#include <string>
#include <map>
#include <functional>
#include "function_traits.hpp"
class message_bus
{
public:
template<typename Function>
void register_handler(std::string const & name, const Function& f)
{
using std::placeholders::_1;
using std::placeholders::_2;
using return_type = typename timax::function_traits<Function>::result_type;
this->invokers_[name] = { std::bind(&invoker<Function>::apply, f, _1, _2) };
}
template <typename T, typename ... Args>
T call(const std::string& name, Args&& ... args)
{
auto it = invokers_.find(name);
if (it == invokers_.end())
return{};
auto args_tuple = std::make_tuple(std::forward<Args>(args)...);
char data[sizeof(std::tuple<Args...>)];
std::tuple<Args...>* tp = new (data) std::tuple<Args...>;
*tp = args_tuple;
T t;
it->second(tp, &t);
return t;
}
template <typename ... Args>
void call_void(const std::string& name, Args&& ... args)
{
auto it = invokers_.find(name);
if (it == invokers_.end())
return;
auto args_tuple = std::make_tuple(std::forward<Args>(args)...);
it->second(&args_tuple, nullptr);
}
private:
template<typename Function>
struct invoker
{
static inline void apply(const Function& func, void* bl, void* result)
{
using tuple_type = typename timax::function_traits<Function>::tuple_type;
const tuple_type* tp = static_cast<tuple_type*>(bl);
call(func, *tp, result);
}
template<typename F, typename ... Args>
static typename std::enable_if<std::is_void<typename std::result_of<F(Args...)>::type>::value>::type
call(const F& f, const std::tuple<Args...>& tp, void*)
{
call_helper(f, std::make_index_sequence<sizeof... (Args)>{}, tp);
}
template<typename F, typename ... Args>
static typename std::enable_if<!std::is_void<typename std::result_of<F(Args...)>::type>::value>::type
call(const F& f, const std::tuple<Args...>& tp, void* result)
{
auto r = call_helper(f, std::make_index_sequence<sizeof... (Args)>{}, tp);
*(decltype(r)*)result = r;
}
template<typename F, size_t... I, typename ... Args>
static auto call_helper(const F& f, const std::index_sequence<I...>&, const std::tuple<Args...>& tup)
{
return f(std::get<I>(tup)...);
}
};
private:
std::map<std::string, std::function<void(void*, void*)>> invokers_;
};
/*
void test_messge_bus()
{
message_bus bus;
bus.register_handler("test", [](int a, int b) { return a + b; });
bus.register_handler("void", [] { std::cout << "void" << std::endl; });
bus.register_handler("str", [](int a, const std::string& b) { return std::to_string(a) + b; });
bus.register_handler("ptr", [](const char* data, int size) { std::cout << data <<" "<<size<< std::endl; });
auto r = bus.call<int>("test", 1, 2);
std::cout << r << std::endl;
auto s = bus.call<std::string>("str", 1, std::string("test"));
bus.call_void("void");
bus.call_void("ptr", "test", 4);
}
*/