aggregate_ops.hpp 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. /*************************************************************************
  2. *
  3. * Copyright 2021 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. #ifndef REALM_AGGREGATE_OPS_HPP
  19. #define REALM_AGGREGATE_OPS_HPP
  20. #include <realm/column_type_traits.hpp>
  21. #include <realm/mixed.hpp>
  22. #include <algorithm>
  23. namespace realm::aggregate_operations {
  24. template <class T>
  25. inline bool valid_for_agg(T)
  26. {
  27. return true;
  28. }
  29. template <class T>
  30. inline bool valid_for_agg(util::Optional<T> val)
  31. {
  32. return !!val;
  33. }
  34. template <>
  35. inline bool valid_for_agg(Timestamp val)
  36. {
  37. return !val.is_null();
  38. }
  39. inline bool valid_for_agg(StringData val)
  40. {
  41. return !val.is_null();
  42. }
  43. inline bool valid_for_agg(BinaryData val)
  44. {
  45. return !val.is_null();
  46. }
  47. template <>
  48. inline bool valid_for_agg(float val)
  49. {
  50. return !null::is_null_float(val) && !std::isnan(val);
  51. }
  52. template <>
  53. inline bool valid_for_agg(double val)
  54. {
  55. return !null::is_null_float(val) && !std::isnan(val);
  56. }
  57. template <>
  58. inline bool valid_for_agg(Decimal128 val)
  59. {
  60. return !val.is_null() && !val.is_nan();
  61. }
  62. template <>
  63. inline bool valid_for_agg(Mixed val)
  64. {
  65. return !val.is_null() && (val.get_type() != type_Decimal || !val.get_decimal().is_nan());
  66. }
  67. template <typename T, typename Compare>
  68. class MinMaxAggregateOperator {
  69. public:
  70. bool accumulate(T value)
  71. {
  72. if (valid_for_agg(value) && (!m_result || Compare()(value, *m_result))) {
  73. m_result = value;
  74. return true;
  75. }
  76. return false;
  77. }
  78. bool accumulate(util::Optional<T> value)
  79. {
  80. if (value) {
  81. return accumulate(*value);
  82. }
  83. return false;
  84. }
  85. template <typename Type = T>
  86. std::enable_if_t<!std::is_same_v<Type, Mixed>, bool> accumulate(const Mixed& value)
  87. {
  88. if (!value.is_null()) {
  89. return accumulate(value.get<T>());
  90. }
  91. return false;
  92. }
  93. bool is_null() const
  94. {
  95. return !m_result;
  96. }
  97. T result() const
  98. {
  99. REALM_ASSERT(m_result);
  100. return *m_result;
  101. }
  102. private:
  103. util::Optional<T> m_result;
  104. };
  105. template <typename T>
  106. class Minimum : public MinMaxAggregateOperator<T, std::less<>> {
  107. public:
  108. static const char* description()
  109. {
  110. return "@min";
  111. }
  112. };
  113. template <typename T>
  114. class Maximum : public MinMaxAggregateOperator<T, std::greater<>> {
  115. public:
  116. static const char* description()
  117. {
  118. return "@max";
  119. }
  120. };
  121. template <typename T>
  122. class Sum {
  123. public:
  124. using ResultType = typename realm::ColumnSumType<T>;
  125. bool accumulate(T value)
  126. {
  127. if constexpr (std::is_same_v<T, Mixed>) {
  128. if (value.accumulate_numeric_to(m_result)) {
  129. ++m_count;
  130. return true;
  131. }
  132. }
  133. else if constexpr (std::is_integral_v<T> && std::is_signed_v<T>) {
  134. m_result = std::make_unsigned_t<T>(m_result) + value;
  135. ++m_count;
  136. return true;
  137. }
  138. else {
  139. if (valid_for_agg(value)) {
  140. m_result += value;
  141. ++m_count;
  142. return true;
  143. }
  144. }
  145. return false;
  146. }
  147. bool accumulate(const util::Optional<T>& value)
  148. {
  149. if (value) {
  150. return accumulate(*value);
  151. }
  152. return false;
  153. }
  154. template <typename Type = T>
  155. std::enable_if_t<!std::is_same_v<Type, Mixed>, bool> accumulate(const Mixed& value)
  156. {
  157. if (!value.is_null()) {
  158. return accumulate(value.get<Type>());
  159. }
  160. return false;
  161. }
  162. bool is_null() const
  163. {
  164. return false;
  165. }
  166. ResultType result() const
  167. {
  168. return m_result;
  169. }
  170. size_t items_counted() const
  171. {
  172. return m_count;
  173. }
  174. static const char* description()
  175. {
  176. return "@sum";
  177. }
  178. private:
  179. ResultType m_result = {};
  180. size_t m_count = 0;
  181. };
  182. template <typename T>
  183. class Average {
  184. public:
  185. using ResultType = typename std::conditional<realm::is_any_v<T, Decimal128, Mixed>, Decimal128, double>::type;
  186. bool accumulate(T value)
  187. {
  188. if constexpr (std::is_same_v<T, Mixed>) {
  189. if (value.accumulate_numeric_to(m_result)) {
  190. m_count++;
  191. return true;
  192. }
  193. }
  194. else {
  195. if (valid_for_agg(value)) {
  196. m_count++;
  197. m_result += value;
  198. return true;
  199. }
  200. }
  201. return false;
  202. }
  203. bool accumulate(const util::Optional<T>& value)
  204. {
  205. if (value) {
  206. return accumulate(*value);
  207. }
  208. return false;
  209. }
  210. template <typename Type = T>
  211. std::enable_if_t<!std::is_same_v<Type, Mixed>, bool> accumulate(const Mixed& value)
  212. {
  213. if (!value.is_null()) {
  214. return accumulate(value.get<Type>());
  215. }
  216. return false;
  217. }
  218. bool is_null() const
  219. {
  220. return m_count == 0;
  221. }
  222. ResultType result() const
  223. {
  224. REALM_ASSERT_EX(m_count > 0, m_count);
  225. return m_result / m_count;
  226. }
  227. static const char* description()
  228. {
  229. return "@avg";
  230. }
  231. size_t items_counted() const
  232. {
  233. return m_count;
  234. }
  235. private:
  236. size_t m_count = 0;
  237. ResultType m_result = {};
  238. };
  239. } // namespace realm::aggregate_operations
  240. #endif // REALM_AGGREGATE_OPS_HPP