123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245 |
- /*************************************************************************
- *
- * Copyright 2016 Realm Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- *
- **************************************************************************/
- #ifndef REALM_TIMESTAMP_HPP
- #define REALM_TIMESTAMP_HPP
- #include <cstdint>
- #include <ostream>
- #include <chrono>
- #include <ctime>
- #include <realm/util/assert.hpp>
- #include <realm/null.hpp>
- namespace realm {
- class Timestamp {
- public:
- // Construct from the number of seconds and nanoseconds since the UNIX epoch: 00:00:00 UTC on 1 January 1970
- //
- // To split a native nanosecond representation, only division and modulo are necessary:
- //
- // s = native_nano / nanoseconds_per_second
- // n = native_nano % nanoseconds_per_second
- // Timestamp ts(s, n);
- //
- // To convert back into native nanosecond representation, simple multiply and add:
- //
- // native_nano = ts.s * nanoseconds_per_second + ts.n
- //
- // Specifically this allows the nanosecond part to become negative (only) for Timestamps before the UNIX epoch.
- // Usually this will not need special attention, but for reference, valid Timestamps will have one of the
- // following sign combinations:
- //
- // s | n
- // -----
- // + | +
- // + | 0
- // 0 | +
- // 0 | 0
- // 0 | -
- // - | 0
- // - | -
- //
- // Examples:
- // The UNIX epoch is constructed by Timestamp(0, 0)
- // Relative times are constructed as follows:
- // +1 second is constructed by Timestamp(1, 0)
- // +1 nanosecond is constructed by Timestamp(0, 1)
- // +1.1 seconds (1100 milliseconds after the epoch) is constructed by Timestamp(1, 100000000)
- // -1.1 seconds (1100 milliseconds before the epoch) is constructed by Timestamp(-1, -100000000)
- //
- constexpr Timestamp(int64_t seconds, int32_t nanoseconds)
- : m_seconds(seconds)
- , m_nanoseconds(nanoseconds)
- , m_is_null(false)
- {
- REALM_ASSERT_EX(-nanoseconds_per_second < nanoseconds && nanoseconds < nanoseconds_per_second, nanoseconds);
- const bool both_non_negative = seconds >= 0 && nanoseconds >= 0;
- const bool both_non_positive = seconds <= 0 && nanoseconds <= 0;
- REALM_ASSERT_EX(both_non_negative || both_non_positive, both_non_negative, both_non_positive);
- }
- constexpr Timestamp() = default;
- constexpr Timestamp(realm::null) {}
- constexpr Timestamp(const Timestamp&) = default;
- constexpr Timestamp& operator=(const Timestamp&) = default;
- template <typename C = std::chrono::system_clock, typename D = typename C::duration>
- constexpr Timestamp(std::chrono::time_point<C, D> tp)
- : m_is_null(false)
- {
- int64_t native_nano = std::chrono::duration_cast<std::chrono::nanoseconds>(tp.time_since_epoch()).count();
- m_seconds = native_nano / nanoseconds_per_second;
- m_nanoseconds = static_cast<int32_t>(native_nano % nanoseconds_per_second);
- }
- constexpr bool is_null() const
- {
- return m_is_null;
- }
- constexpr int64_t get_seconds() const noexcept
- {
- REALM_ASSERT(!m_is_null);
- return m_seconds;
- }
- constexpr int32_t get_nanoseconds() const noexcept
- {
- REALM_ASSERT(!m_is_null);
- return m_nanoseconds;
- }
- template <typename C = std::chrono::system_clock, typename D = typename C::duration>
- constexpr std::chrono::time_point<C, D> get_time_point() const
- {
- REALM_ASSERT(!m_is_null);
- int64_t native_nano = m_seconds * nanoseconds_per_second + m_nanoseconds;
- auto duration = std::chrono::duration_cast<D>(std::chrono::duration<int64_t, std::nano>{native_nano});
- return std::chrono::time_point<C, D>(duration);
- }
- template <typename C = std::chrono::system_clock, typename D = typename C::duration>
- constexpr explicit operator std::chrono::time_point<C, D>() const
- {
- return get_time_point();
- }
- constexpr bool operator==(const Timestamp& rhs) const
- {
- if (is_null() && rhs.is_null())
- return true;
- if (is_null() != rhs.is_null())
- return false;
- return m_seconds == rhs.m_seconds && m_nanoseconds == rhs.m_nanoseconds;
- }
- constexpr bool operator!=(const Timestamp& rhs) const
- {
- return !(*this == rhs);
- }
- constexpr bool operator>(const Timestamp& rhs) const
- {
- if (is_null()) {
- return false;
- }
- if (rhs.is_null()) {
- return true;
- }
- return (m_seconds > rhs.m_seconds) || (m_seconds == rhs.m_seconds && m_nanoseconds > rhs.m_nanoseconds);
- }
- constexpr bool operator<(const Timestamp& rhs) const
- {
- if (rhs.is_null()) {
- return false;
- }
- if (is_null()) {
- return true;
- }
- return (m_seconds < rhs.m_seconds) || (m_seconds == rhs.m_seconds && m_nanoseconds < rhs.m_nanoseconds);
- }
- constexpr bool operator<=(const Timestamp& rhs) const
- {
- if (is_null()) {
- return true;
- }
- if (rhs.is_null()) {
- return false;
- }
- return *this < rhs || *this == rhs;
- }
- constexpr bool operator>=(const Timestamp& rhs) const
- {
- if (rhs.is_null()) {
- return true;
- }
- if (is_null()) {
- return false;
- }
- return *this > rhs || *this == rhs;
- }
- constexpr size_t hash() const noexcept
- {
- return size_t(m_seconds) ^ size_t(m_nanoseconds);
- }
- template <class Ch, class Tr>
- friend std::basic_ostream<Ch, Tr>& operator<<(std::basic_ostream<Ch, Tr>& out, const Timestamp&);
- static constexpr int32_t nanoseconds_per_second = 1000000000;
- private:
- int64_t m_seconds = 0;
- int32_t m_nanoseconds = 0;
- bool m_is_null = true;
- };
- // LCOV_EXCL_START
- template <class C, class T>
- inline std::basic_ostream<C, T>& operator<<(std::basic_ostream<C, T>& out, const Timestamp& d)
- {
- if (d.is_null()) {
- out << "null";
- return out;
- }
- auto seconds = time_t(d.get_seconds());
- struct tm buf;
- #ifdef _MSC_VER
- bool success = gmtime_s(&buf, &seconds) == 0;
- #else
- bool success = gmtime_r(&seconds, &buf) != nullptr;
- #endif
- if (success) {
- // We need a buffer for formatting dates.
- // Max size is 20 bytes (incl terminating zero) "YYYY-MM-DD HH:MM:SS"\0
- char buffer[30];
- if (strftime(buffer, sizeof(buffer), "%Y-%m-%d %H:%M:%S", &buf)) {
- out << buffer;
- }
- }
- return out;
- }
- // LCOV_EXCL_STOP
- } // namespace realm
- namespace std {
- template <>
- struct numeric_limits<realm::Timestamp> {
- static constexpr bool is_integer = false;
- static constexpr realm::Timestamp min()
- {
- return realm::Timestamp(numeric_limits<int64_t>::min(), 0);
- }
- static constexpr realm::Timestamp lowest()
- {
- return realm::Timestamp(numeric_limits<int64_t>::lowest(), 0);
- }
- static constexpr realm::Timestamp max()
- {
- return realm::Timestamp(numeric_limits<int64_t>::max(), 0);
- }
- };
- }
- #endif // REALM_TIMESTAMP_HPP
|