aggregate_ops.hpp 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  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 {
  134. if (valid_for_agg(value)) {
  135. m_result += value;
  136. ++m_count;
  137. return true;
  138. }
  139. }
  140. return false;
  141. }
  142. bool accumulate(const util::Optional<T>& value)
  143. {
  144. if (value) {
  145. return accumulate(*value);
  146. }
  147. return false;
  148. }
  149. template <typename Type = T>
  150. std::enable_if_t<!std::is_same_v<Type, Mixed>, bool> accumulate(const Mixed& value)
  151. {
  152. if (!value.is_null()) {
  153. return accumulate(value.get<Type>());
  154. }
  155. return false;
  156. }
  157. bool is_null() const
  158. {
  159. return false;
  160. }
  161. ResultType result() const
  162. {
  163. return m_result;
  164. }
  165. size_t items_counted() const
  166. {
  167. return m_count;
  168. }
  169. static const char* description()
  170. {
  171. return "@sum";
  172. }
  173. private:
  174. ResultType m_result = {};
  175. size_t m_count = 0;
  176. };
  177. template <typename T>
  178. class Average {
  179. public:
  180. using ResultType = typename std::conditional<realm::is_any_v<T, Decimal128, Mixed>, Decimal128, double>::type;
  181. bool accumulate(T value)
  182. {
  183. if constexpr (std::is_same_v<T, Mixed>) {
  184. if (value.accumulate_numeric_to(m_result)) {
  185. m_count++;
  186. return true;
  187. }
  188. }
  189. else {
  190. if (valid_for_agg(value)) {
  191. m_count++;
  192. m_result += value;
  193. return true;
  194. }
  195. }
  196. return false;
  197. }
  198. bool accumulate(const util::Optional<T>& value)
  199. {
  200. if (value) {
  201. return accumulate(*value);
  202. }
  203. return false;
  204. }
  205. template <typename Type = T>
  206. std::enable_if_t<!std::is_same_v<Type, Mixed>, bool> accumulate(const Mixed& value)
  207. {
  208. if (!value.is_null()) {
  209. return accumulate(value.get<Type>());
  210. }
  211. return false;
  212. }
  213. bool is_null() const
  214. {
  215. return m_count == 0;
  216. }
  217. ResultType result() const
  218. {
  219. REALM_ASSERT_EX(m_count > 0, m_count);
  220. return m_result / m_count;
  221. }
  222. static const char* description()
  223. {
  224. return "@avg";
  225. }
  226. size_t items_counted() const
  227. {
  228. return m_count;
  229. }
  230. private:
  231. size_t m_count = 0;
  232. ResultType m_result = {};
  233. };
  234. } // namespace realm::aggregate_operations
  235. #endif // REALM_AGGREGATE_OPS_HPP