QtRestClient  3.0.0
A library for generic JSON-based REST-APIs, with a mechanism to map JSON to Qt objects
qtrestclient_helpertypes.h
1 #ifndef QTRESTCLIENT_HELPERTYPES_H
2 #define QTRESTCLIENT_HELPERTYPES_H
3 
4 #include "QtRestClient/qtrestclient_global.h"
5 
6 #include <variant>
7 #include <optional>
8 
9 #include <QtCore/qcborvalue.h>
10 #include <QtCore/qcbormap.h>
11 #include <QtCore/qcborarray.h>
12 #include <QtCore/qjsonvalue.h>
13 #include <QtCore/qjsonobject.h>
14 #include <QtCore/qjsonarray.h>
15 
16 namespace QtRestClient::__private {
17 
18 template<typename... Ts>
19 struct overload : public Ts... {
20  using Ts::operator()...;
21 };
22 template<class... Ts>
23 overload(Ts...) -> overload<Ts...>;
24 
25 namespace __binder {
26 
27 using DataType = std::variant<std::nullopt_t, QCborValue, QJsonValue>;
28 using FunctionType = std::function<void(int, DataType)>;
29 
30 template <typename... TArgs>
31 struct FnBinder;
32 
33 template <>
34 struct FnBinder<int, DataType> {
35  template <typename TFn>
36  static inline FunctionType bind(TFn &&fn) {
37  return std::forward<TFn>(fn);
38  }
39 };
40 
41 template <>
42 struct FnBinder<> {
43  template <typename TFn>
44  static inline FunctionType bind(TFn &&fn) {
45  return [xFn = std::forward<TFn>(fn)](int, const DataType &) {
46  xFn();
47  };
48  }
49 };
50 
51 template <>
52 struct FnBinder<int> {
53  template <typename TFn>
54  static inline FunctionType bind(TFn &&fn) {
55  return [xFn = std::forward<TFn>(fn)](int code, const DataType &) {
56  xFn(code);
57  };
58  }
59 };
60 
61 template <typename TData>
62 struct FnBinder<TData> {
63  template <typename TFn>
64  static inline FunctionType bind(TFn &&fn) {
65  return FnBinder<int, TData>::bind([xFn = std::forward<TFn>(fn)](int, const TData &value) { // TData is already decayed -> true move
66  xFn(value);
67  });
68  }
69 };
70 
71 template <>
72 struct FnBinder<int, QJsonValue> {
73  template <typename TFn>
74  static inline FunctionType bind(TFn &&fn) {
75  return [xFn = std::forward<TFn>(fn)](int code, const DataType &value) {
76  std::visit(overload {
77  [xFn, code](std::nullopt_t){
78  xFn(code, QJsonValue{QJsonValue::Undefined});
79  },
80  [xFn, code](const QCborValue &){
81  qCWarning(logGlobal) << "CBOR data in JSON callback - discarding";
82  },
83  [xFn, code](const QJsonValue &vValue){
84  xFn(code, vValue);
85  }
86  }, value);
87  };
88  }
89 };
90 
91 template <>
92 struct FnBinder<int, QJsonObject> {
93  template <typename TFn>
94  static inline FunctionType bind(TFn &&fn) {
95  return [xFn = std::forward<TFn>(fn)](int code, const DataType &value) {
96  std::visit(overload {
97  [xFn, code](std::nullopt_t){
98  xFn(code, QJsonObject{});
99  },
100  [xFn, code](const QCborValue &){
101  qCWarning(logGlobal) << "CBOR data in JSON callback - discarding";
102  },
103  [xFn, code](const QJsonValue &vValue){
104  xFn(code, vValue.toObject());
105  }
106  }, value);
107  };
108  }
109 };
110 
111 template <>
112 struct FnBinder<int, QJsonArray> {
113  template <typename TFn>
114  static inline FunctionType bind(TFn &&fn) {
115  return [xFn = std::forward<TFn>(fn)](int code, const DataType &value) {
116  std::visit(overload {
117  [xFn, code](std::nullopt_t){
118  xFn(code, QJsonArray{});
119  },
120  [xFn, code](const QCborValue &){
121  qCWarning(logGlobal) << "CBOR data in JSON callback - discarding";
122  },
123  [xFn, code](const QJsonValue &vValue){
124  xFn(code, vValue.toArray());
125  }
126  }, value);
127  };
128  }
129 };
130 
131 template <>
132 struct FnBinder<int, QCborValue> {
133  template <typename TFn>
134  static inline FunctionType bind(TFn &&fn) {
135  return [xFn = std::forward<TFn>(fn)](int code, const DataType &value) {
136  std::visit(overload {
137  [xFn, code](std::nullopt_t){
138  xFn(code, QCborValue{QCborSimpleType::Undefined});
139  },
140  [xFn, code](const QCborValue &vValue){
141  xFn(code, vValue);
142  },
143  [xFn, code](const QJsonValue &){
144  qCWarning(logGlobal) << "JSON data in CBOR callback - discarding";
145  }
146  }, value);
147  };
148  }
149 };
150 
151 template <>
152 struct FnBinder<int, QCborMap> {
153  template <typename TFn>
154  static inline FunctionType bind(TFn &&fn) {
155  return [xFn = std::forward<TFn>(fn)](int code, const DataType &value) {
156  std::visit(overload {
157  [xFn, code](std::nullopt_t){
158  xFn(code, QCborMap{});
159  },
160  [xFn, code](const QCborValue &vValue){
161  xFn(code, vValue.toMap());
162  },
163  [xFn, code](const QJsonValue &){
164  qCWarning(logGlobal) << "JSON data in CBOR callback - discarding";
165  }
166  }, value);
167  };
168  }
169 };
170 
171 template <>
172 struct FnBinder<int, QCborArray> {
173  template <typename TFn>
174  static inline FunctionType bind(TFn &&fn) {
175  return [xFn = std::forward<TFn>(fn)](int code, const DataType &value) {
176  std::visit(overload {
177  [xFn, code](std::nullopt_t){
178  xFn(code, QCborArray{});
179  },
180  [xFn, code](const QCborValue &vValue){
181  xFn(code, vValue.toArray());
182  },
183  [xFn, code](const QJsonValue &){
184  qCWarning(logGlobal) << "JSON data in CBOR callback - discarding";
185  }
186  }, value);
187  };
188  }
189 };
190 
191 template <typename TCbor, typename TJson>
192 struct FnBinder<int, std::variant<std::nullopt_t, TCbor, TJson>> {
193  template <typename TFn>
194  static inline FunctionType bind(TFn &&fn) {
195  return [xFn = std::forward<TFn>(fn)](int code, const DataType &value) {
196  auto cFn = FnBinder<int, TCbor>::bind(xFn)(code, value);
197  auto jFn = FnBinder<int, TJson>::bind(xFn)(code, value);
198  std::visit(overload {
199  [xFn, code](std::nullopt_t){
200  xFn(code, std::nullopt);
201  },
202  [xcFn = std::move(cFn), code](const QCborValue &vValue){
203  xcFn(code, vValue);
204  },
205  [xjFn = std::move(jFn), code](const QJsonValue &vValue){
206  xjFn(code, vValue);
207  }
208  }, value);
209  };
210  }
211 };
212 
213 template <typename TCbor, typename TJson>
214 struct FnBinder<int, std::variant<TCbor, TJson>> {
215  template <typename TFn>
216  static inline FunctionType bind(TFn &&fn) {
217  return [xFn = std::forward<TFn>(fn)](int code, const DataType &value) {
218  auto cFn = FnBinder<int, TCbor>::bind(xFn)(code, value);
219  auto jFn = FnBinder<int, TJson>::bind(xFn)(code, value);
220  std::visit(overload {
221  [xFn, code](std::nullopt_t){
222  xFn(code, std::variant<TCbor, TJson>{});
223  },
224  [xcFn = std::move(cFn), code](const QCborValue &vValue){
225  xcFn(code, vValue);
226  },
227  [xjFn = std::move(jFn), code](const QJsonValue &vValue){
228  xjFn(code, vValue);
229  }
230  }, value);
231  };
232  }
233 };
234 
235 
236 
237 template <typename T>
238 struct FnInfo : public FnInfo<decltype(&T::operator())> {};
239 
240 template <typename TClass, typename TRet, typename... TArgs>
241 struct FnInfo<TRet(TClass::*)(TArgs...) const>
242 {
243  using Binder = FnBinder<std::decay_t<TArgs>...>;
244 };
245 
246 template <typename TClass, typename TRet, typename... TArgs>
247 struct FnInfo<TRet(TClass::*)(TArgs...)>
248 {
249  using Binder = FnBinder<std::decay_t<TArgs>...>;
250 };
251 
252 template <typename TRet, typename... TArgs>
253 struct FnInfo<TRet(*)(TArgs...)>
254 {
255  using Binder = FnBinder<std::decay_t<TArgs>...>;
256 };
257 
258 template <typename TClass, typename TRet, typename... TBinders, typename... TArgs>
259 struct FnInfo<TRet(TClass::*(TBinders...))(TArgs...) const>
260 {
261  using Binder = FnBinder<std::decay_t<TArgs>...>;
262 };
263 
264 template <typename TClass, typename TRet, typename... TBinders, typename... TArgs>
265 struct FnInfo<TRet(TClass::*(TBinders...))(TArgs...)>
266 {
267  using Binder = FnBinder<std::decay_t<TArgs>...>;
268 };
269 
270 }
271 
272 template <typename TFn>
273 static inline __binder::FunctionType bindCallback(TFn &&fn) {
274  return __binder::FnInfo<std::decay_t<TFn>>::Binder::bind(std::forward<TFn>(fn));
275 }
276 
277 template <typename TFn, typename TError>
278 static inline __binder::FunctionType bindCallback(const std::function<void(QString, int, TError)> &handler, TFn &&tFn, TError errorType) {
279  return bindCallback([handler, xTFn = std::forward<TFn>(tFn), errorType](int code, auto &&data) {
280  handler(xTFn(data, code), code, errorType);
281  });
282 }
283 
284 }
285 
286 #endif // QTRESTCLIENT_HELPERTYPES_H
QJsonValue
QJsonArray
QString
QCborValue
QJsonObject
QCborMap
QCborArray