timestamp.hpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. /*************************************************************************
  2. *
  3. * Copyright 2016 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_TIMESTAMP_HPP
  19. #define REALM_TIMESTAMP_HPP
  20. #include <cstdint>
  21. #include <ostream>
  22. #include <chrono>
  23. #include <ctime>
  24. #include <realm/util/assert.hpp>
  25. #include <realm/null.hpp>
  26. namespace realm {
  27. class Timestamp {
  28. public:
  29. // Construct from the number of seconds and nanoseconds since the UNIX epoch: 00:00:00 UTC on 1 January 1970
  30. //
  31. // To split a native nanosecond representation, only division and modulo are necessary:
  32. //
  33. // s = native_nano / nanoseconds_per_second
  34. // n = native_nano % nanoseconds_per_second
  35. // Timestamp ts(s, n);
  36. //
  37. // To convert back into native nanosecond representation, simple multiply and add:
  38. //
  39. // native_nano = ts.s * nanoseconds_per_second + ts.n
  40. //
  41. // Specifically this allows the nanosecond part to become negative (only) for Timestamps before the UNIX epoch.
  42. // Usually this will not need special attention, but for reference, valid Timestamps will have one of the
  43. // following sign combinations:
  44. //
  45. // s | n
  46. // -----
  47. // + | +
  48. // + | 0
  49. // 0 | +
  50. // 0 | 0
  51. // 0 | -
  52. // - | 0
  53. // - | -
  54. //
  55. // Examples:
  56. // The UNIX epoch is constructed by Timestamp(0, 0)
  57. // Relative times are constructed as follows:
  58. // +1 second is constructed by Timestamp(1, 0)
  59. // +1 nanosecond is constructed by Timestamp(0, 1)
  60. // +1.1 seconds (1100 milliseconds after the epoch) is constructed by Timestamp(1, 100000000)
  61. // -1.1 seconds (1100 milliseconds before the epoch) is constructed by Timestamp(-1, -100000000)
  62. //
  63. Timestamp(int64_t seconds, int32_t nanoseconds)
  64. : m_seconds(seconds)
  65. , m_nanoseconds(nanoseconds)
  66. , m_is_null(false)
  67. {
  68. REALM_ASSERT_EX(-nanoseconds_per_second < nanoseconds && nanoseconds < nanoseconds_per_second, nanoseconds);
  69. const bool both_non_negative = seconds >= 0 && nanoseconds >= 0;
  70. const bool both_non_positive = seconds <= 0 && nanoseconds <= 0;
  71. REALM_ASSERT_EX(both_non_negative || both_non_positive, both_non_negative, both_non_positive);
  72. }
  73. Timestamp(realm::null)
  74. : m_is_null(true)
  75. {
  76. }
  77. template <typename C = std::chrono::system_clock, typename D = typename C::duration>
  78. Timestamp(std::chrono::time_point<C, D> tp)
  79. : m_is_null(false)
  80. {
  81. int64_t native_nano = std::chrono::duration_cast<std::chrono::nanoseconds>(tp.time_since_epoch()).count();
  82. m_seconds = native_nano / nanoseconds_per_second;
  83. m_nanoseconds = static_cast<int32_t>(native_nano % nanoseconds_per_second);
  84. }
  85. Timestamp()
  86. : Timestamp(null{})
  87. {
  88. }
  89. bool is_null() const
  90. {
  91. return m_is_null;
  92. }
  93. int64_t get_seconds() const noexcept
  94. {
  95. REALM_ASSERT(!m_is_null);
  96. return m_seconds;
  97. }
  98. int32_t get_nanoseconds() const noexcept
  99. {
  100. REALM_ASSERT(!m_is_null);
  101. return m_nanoseconds;
  102. }
  103. template <typename C = std::chrono::system_clock, typename D = typename C::duration>
  104. std::chrono::time_point<C, D> get_time_point() const
  105. {
  106. REALM_ASSERT(!m_is_null);
  107. int64_t native_nano = m_seconds * nanoseconds_per_second + m_nanoseconds;
  108. auto duration = std::chrono::duration_cast<D>(std::chrono::duration<int64_t, std::nano>{native_nano});
  109. return std::chrono::time_point<C, D>(duration);
  110. }
  111. template <typename C = std::chrono::system_clock, typename D = typename C::duration>
  112. explicit operator std::chrono::time_point<C, D>() const
  113. {
  114. return get_time_point();
  115. }
  116. bool operator==(const Timestamp& rhs) const
  117. {
  118. if (is_null() && rhs.is_null())
  119. return true;
  120. if (is_null() != rhs.is_null())
  121. return false;
  122. return m_seconds == rhs.m_seconds && m_nanoseconds == rhs.m_nanoseconds;
  123. }
  124. bool operator!=(const Timestamp& rhs) const
  125. {
  126. return !(*this == rhs);
  127. }
  128. bool operator>(const Timestamp& rhs) const
  129. {
  130. if (is_null()) {
  131. return false;
  132. }
  133. if (rhs.is_null()) {
  134. return true;
  135. }
  136. return (m_seconds > rhs.m_seconds) || (m_seconds == rhs.m_seconds && m_nanoseconds > rhs.m_nanoseconds);
  137. }
  138. bool operator<(const Timestamp& rhs) const
  139. {
  140. if (rhs.is_null()) {
  141. return false;
  142. }
  143. if (is_null()) {
  144. return true;
  145. }
  146. return (m_seconds < rhs.m_seconds) || (m_seconds == rhs.m_seconds && m_nanoseconds < rhs.m_nanoseconds);
  147. }
  148. bool operator<=(const Timestamp& rhs) const
  149. {
  150. if (is_null()) {
  151. return true;
  152. }
  153. if (rhs.is_null()) {
  154. return false;
  155. }
  156. return *this < rhs || *this == rhs;
  157. }
  158. bool operator>=(const Timestamp& rhs) const
  159. {
  160. if (rhs.is_null()) {
  161. return true;
  162. }
  163. if (is_null()) {
  164. return false;
  165. }
  166. return *this > rhs || *this == rhs;
  167. }
  168. Timestamp& operator=(const Timestamp& rhs) = default;
  169. size_t hash() const noexcept;
  170. template <class Ch, class Tr>
  171. friend std::basic_ostream<Ch, Tr>& operator<<(std::basic_ostream<Ch, Tr>& out, const Timestamp&);
  172. static constexpr int32_t nanoseconds_per_second = 1000000000;
  173. private:
  174. int64_t m_seconds;
  175. int32_t m_nanoseconds;
  176. bool m_is_null;
  177. };
  178. // LCOV_EXCL_START
  179. template <class C, class T>
  180. inline std::basic_ostream<C, T>& operator<<(std::basic_ostream<C, T>& out, const Timestamp& d)
  181. {
  182. auto seconds = time_t(d.get_seconds());
  183. struct tm buf;
  184. #ifdef _MSC_VER
  185. bool success = gmtime_s(&buf, &seconds) == 0;
  186. #else
  187. bool success = gmtime_r(&seconds, &buf) != nullptr;
  188. #endif
  189. if (success) {
  190. // We need a buffer for formatting dates.
  191. // Max size is 20 bytes (incl terminating zero) "YYYY-MM-DD HH:MM:SS"\0
  192. char buffer[30];
  193. if (strftime(buffer, sizeof(buffer), "%Y-%m-%d %H:%M:%S", &buf)) {
  194. out << buffer;
  195. }
  196. }
  197. return out;
  198. }
  199. // LCOV_EXCL_STOP
  200. inline size_t Timestamp::hash() const noexcept
  201. {
  202. return size_t(m_seconds) ^ size_t(m_nanoseconds);
  203. }
  204. } // namespace realm
  205. namespace std {
  206. template <>
  207. struct numeric_limits<realm::Timestamp> {
  208. static constexpr bool is_integer = false;
  209. static realm::Timestamp min()
  210. {
  211. return realm::Timestamp(numeric_limits<int64_t>::min(), 0);
  212. }
  213. static realm::Timestamp lowest()
  214. {
  215. return realm::Timestamp(numeric_limits<int64_t>::lowest(), 0);
  216. }
  217. static realm::Timestamp max()
  218. {
  219. return realm::Timestamp(numeric_limits<int64_t>::max(), 0);
  220. }
  221. };
  222. }
  223. #endif // REALM_TIMESTAMP_HPP