275 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			275 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#ifndef OSMIUM_OSM_TIMESTAMP_HPP
 | 
						|
#define OSMIUM_OSM_TIMESTAMP_HPP
 | 
						|
 | 
						|
/*
 | 
						|
 | 
						|
This file is part of Osmium (http://osmcode.org/libosmium).
 | 
						|
 | 
						|
Copyright 2013-2016 Jochen Topf <jochen@topf.org> and others (see README).
 | 
						|
 | 
						|
Boost Software License - Version 1.0 - August 17th, 2003
 | 
						|
 | 
						|
Permission is hereby granted, free of charge, to any person or organization
 | 
						|
obtaining a copy of the software and accompanying documentation covered by
 | 
						|
this license (the "Software") to use, reproduce, display, distribute,
 | 
						|
execute, and transmit the Software, and to prepare derivative works of the
 | 
						|
Software, and to permit third-parties to whom the Software is furnished to
 | 
						|
do so, all subject to the following:
 | 
						|
 | 
						|
The copyright notices in the Software and this entire statement, including
 | 
						|
the above license grant, this restriction and the following disclaimer,
 | 
						|
must be included in all copies of the Software, in whole or in part, and
 | 
						|
all derivative works of the Software, unless such copies or derivative
 | 
						|
works are solely in the form of machine-executable object code generated by
 | 
						|
a source language processor.
 | 
						|
 | 
						|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
						|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
						|
FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
 | 
						|
SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
 | 
						|
FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
 | 
						|
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 | 
						|
DEALINGS IN THE SOFTWARE.
 | 
						|
 | 
						|
*/
 | 
						|
 | 
						|
#include <cassert>
 | 
						|
#include <cstdint>
 | 
						|
#include <ctime>
 | 
						|
#include <iosfwd>
 | 
						|
#include <limits>
 | 
						|
#include <stdexcept>
 | 
						|
#include <string>
 | 
						|
 | 
						|
#include <osmium/util/compatibility.hpp>
 | 
						|
#include <osmium/util/minmax.hpp> // IWYU pragma: keep
 | 
						|
 | 
						|
namespace osmium {
 | 
						|
 | 
						|
    /**
 | 
						|
     * A timestamp. Internal representation is an unsigned 32bit integer
 | 
						|
     * holding seconds since epoch (1970-01-01T00:00:00Z), so this will
 | 
						|
     * overflow in 2106. We can use an unsigned integer here, because the
 | 
						|
     * OpenStreetMap project was started long after 1970, so there will
 | 
						|
     * never be dates before that.
 | 
						|
     */
 | 
						|
    class Timestamp {
 | 
						|
 | 
						|
        // length of ISO timestamp string yyyy-mm-ddThh:mm:ssZ\0
 | 
						|
        static constexpr int timestamp_length = 20 + 1;
 | 
						|
 | 
						|
        // The timestamp format for OSM timestamps in strftime(3) format.
 | 
						|
        // This is the ISO-Format "yyyy-mm-ddThh:mm:ssZ".
 | 
						|
        static const char* timestamp_format() {
 | 
						|
            static const char f[timestamp_length] = "%Y-%m-%dT%H:%M:%SZ";
 | 
						|
            return f;
 | 
						|
        }
 | 
						|
 | 
						|
        uint32_t m_timestamp;
 | 
						|
 | 
						|
    public:
 | 
						|
 | 
						|
        /**
 | 
						|
         * Default construct an invalid Timestamp.
 | 
						|
         */
 | 
						|
        constexpr Timestamp() noexcept :
 | 
						|
            m_timestamp(0) {
 | 
						|
        }
 | 
						|
 | 
						|
        /**
 | 
						|
         * Construct a Timestamp from any integer type containing the seconds
 | 
						|
         * since the epoch. This will not check for overruns, you have to
 | 
						|
         * make sure the value fits into a uint32_t which is used internally
 | 
						|
         * in the Timestamp.
 | 
						|
         *
 | 
						|
         * The constructor is not declared "explicit" so that conversions
 | 
						|
         * like @code node.set_timestamp(123); @endcode work.
 | 
						|
         */
 | 
						|
        template <typename T, typename std::enable_if<std::is_integral<T>::value, int>::type = 0>
 | 
						|
        constexpr Timestamp(T timestamp) noexcept :
 | 
						|
            m_timestamp(uint32_t(timestamp)) {
 | 
						|
        }
 | 
						|
 | 
						|
        /**
 | 
						|
         * Construct timestamp from ISO date/time string in the format
 | 
						|
         * "yyyy-mm-ddThh:mm:ssZ".
 | 
						|
         *
 | 
						|
         * @throws std::invalid_argument if the timestamp can not be parsed.
 | 
						|
         */
 | 
						|
        explicit Timestamp(const char* timestamp) {
 | 
						|
#ifndef _WIN32
 | 
						|
            struct tm tm {
 | 
						|
                0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
 | 
						|
            };
 | 
						|
            if (strptime(timestamp, timestamp_format(), &tm) == nullptr) {
 | 
						|
                throw std::invalid_argument("can't parse timestamp");
 | 
						|
            }
 | 
						|
            m_timestamp = static_cast<uint32_t>(timegm(&tm));
 | 
						|
#else
 | 
						|
            struct tm tm;
 | 
						|
            int n = sscanf(timestamp, "%4d-%2d-%2dT%2d:%2d:%2dZ", &tm.tm_year, &tm.tm_mon, &tm.tm_mday, &tm.tm_hour, &tm.tm_min, &tm.tm_sec);
 | 
						|
            if (n != 6) {
 | 
						|
                throw std::invalid_argument("can't parse timestamp");
 | 
						|
            }
 | 
						|
            tm.tm_year -= 1900;
 | 
						|
            tm.tm_mon--;
 | 
						|
            tm.tm_wday = 0;
 | 
						|
            tm.tm_yday = 0;
 | 
						|
            tm.tm_isdst = 0;
 | 
						|
            m_timestamp = static_cast<uint32_t>(_mkgmtime(&tm));
 | 
						|
#endif
 | 
						|
        }
 | 
						|
 | 
						|
        /**
 | 
						|
         * Construct timestamp from ISO date/time string in the format
 | 
						|
         * "yyyy-mm-ddThh:mm:ssZ".
 | 
						|
         *
 | 
						|
         * @throws std::invalid_argument if the timestamp can not be parsed.
 | 
						|
         */
 | 
						|
        explicit Timestamp(const std::string& timestamp) :
 | 
						|
            Timestamp(timestamp.c_str()) {
 | 
						|
        }
 | 
						|
 | 
						|
        /**
 | 
						|
         * Returns true if this timestamp is valid (ie set to something other
 | 
						|
         * than 0).
 | 
						|
         */
 | 
						|
        bool valid() const noexcept {
 | 
						|
            return m_timestamp != 0;
 | 
						|
        }
 | 
						|
 | 
						|
        /// Explicit conversion into bool.
 | 
						|
        explicit constexpr operator bool() const noexcept {
 | 
						|
            return m_timestamp != 0;
 | 
						|
        }
 | 
						|
 | 
						|
        /// Explicit conversion into time_t.
 | 
						|
        constexpr time_t seconds_since_epoch() const noexcept {
 | 
						|
            return time_t(m_timestamp);
 | 
						|
        }
 | 
						|
 | 
						|
        /// Explicit conversion into uint32_t.
 | 
						|
        explicit constexpr operator uint32_t() const noexcept {
 | 
						|
            return uint32_t(m_timestamp);
 | 
						|
        }
 | 
						|
 | 
						|
        /// Explicit conversion into uint64_t.
 | 
						|
        explicit constexpr operator uint64_t() const noexcept {
 | 
						|
            return uint64_t(m_timestamp);
 | 
						|
        }
 | 
						|
 | 
						|
        /**
 | 
						|
         * Implicit conversion into time_t.
 | 
						|
         *
 | 
						|
         * @deprecated You should call seconds_since_epoch() explicitly instead.
 | 
						|
         */
 | 
						|
        OSMIUM_DEPRECATED constexpr operator time_t() const noexcept {
 | 
						|
            return static_cast<time_t>(m_timestamp);
 | 
						|
        }
 | 
						|
 | 
						|
        template <typename T>
 | 
						|
        void operator+=(T time_difference) noexcept {
 | 
						|
            m_timestamp += time_difference;
 | 
						|
        }
 | 
						|
 | 
						|
        template <typename T>
 | 
						|
        void operator-=(T time_difference) noexcept {
 | 
						|
            m_timestamp -= time_difference;
 | 
						|
        }
 | 
						|
 | 
						|
        /**
 | 
						|
         * Return the timestamp as string in ISO date/time
 | 
						|
         * ("yyyy-mm-ddThh:mm:ssZ") format. If the timestamp is invalid, an
 | 
						|
         * empty string will be returned.
 | 
						|
         */
 | 
						|
        std::string to_iso() const {
 | 
						|
            std::string s;
 | 
						|
 | 
						|
            if (m_timestamp != 0) {
 | 
						|
                struct tm tm;
 | 
						|
                time_t sse = seconds_since_epoch();
 | 
						|
#ifndef NDEBUG
 | 
						|
                auto result =
 | 
						|
#endif
 | 
						|
#ifndef _MSC_VER
 | 
						|
                              gmtime_r(&sse, &tm);
 | 
						|
                assert(result != nullptr);
 | 
						|
#else
 | 
						|
                              gmtime_s(&tm, &sse);
 | 
						|
                assert(result == 0);
 | 
						|
#endif
 | 
						|
 | 
						|
                s.resize(timestamp_length);
 | 
						|
                /* This const_cast is ok, because we know we have enough space
 | 
						|
                in the string for the format we are using (well at least until
 | 
						|
                the year will have 5 digits). And by setting the size
 | 
						|
                afterwards from the result of strftime we make sure thats set
 | 
						|
                right, too. */
 | 
						|
                s.resize(strftime(const_cast<char*>(s.c_str()), timestamp_length, timestamp_format(), &tm));
 | 
						|
            }
 | 
						|
 | 
						|
            return s;
 | 
						|
        }
 | 
						|
 | 
						|
    }; // class Timestamp
 | 
						|
 | 
						|
    /**
 | 
						|
     * A special Timestamp guaranteed to be ordered before any other valid
 | 
						|
     * Timestamp.
 | 
						|
     */
 | 
						|
    inline constexpr Timestamp start_of_time() noexcept {
 | 
						|
        return Timestamp(1);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * A special Timestamp guaranteed to be ordered after any other valid
 | 
						|
     * Timestamp.
 | 
						|
     */
 | 
						|
    inline constexpr Timestamp end_of_time() noexcept {
 | 
						|
        return Timestamp(std::numeric_limits<uint32_t>::max());
 | 
						|
    }
 | 
						|
 | 
						|
    template <typename TChar, typename TTraits>
 | 
						|
    inline std::basic_ostream<TChar, TTraits>& operator<<(std::basic_ostream<TChar, TTraits>& out, Timestamp timestamp) {
 | 
						|
        out << timestamp.to_iso();
 | 
						|
        return out;
 | 
						|
    }
 | 
						|
 | 
						|
    inline bool operator==(const Timestamp& lhs, const Timestamp& rhs) noexcept {
 | 
						|
        return uint32_t(lhs) == uint32_t(rhs);
 | 
						|
    }
 | 
						|
 | 
						|
    inline bool operator!=(const Timestamp& lhs, const Timestamp& rhs) noexcept {
 | 
						|
        return !(lhs == rhs);
 | 
						|
    }
 | 
						|
 | 
						|
    inline bool operator<(const Timestamp& lhs, const Timestamp& rhs) noexcept {
 | 
						|
        return uint32_t(lhs) < uint32_t(rhs);
 | 
						|
    }
 | 
						|
 | 
						|
    inline bool operator>(const Timestamp& lhs, const Timestamp& rhs) noexcept {
 | 
						|
        return rhs < lhs;
 | 
						|
    }
 | 
						|
 | 
						|
    inline bool operator<=(const Timestamp& lhs, const Timestamp& rhs) noexcept {
 | 
						|
        return ! (rhs < lhs);
 | 
						|
    }
 | 
						|
 | 
						|
    inline bool operator>=(const Timestamp& lhs, const Timestamp& rhs) noexcept {
 | 
						|
        return ! (lhs < rhs);
 | 
						|
    }
 | 
						|
 | 
						|
    template <>
 | 
						|
    inline osmium::Timestamp min_op_start_value<osmium::Timestamp>() {
 | 
						|
        return end_of_time();
 | 
						|
    }
 | 
						|
 | 
						|
    template <>
 | 
						|
    inline osmium::Timestamp max_op_start_value<osmium::Timestamp>() {
 | 
						|
        return start_of_time();
 | 
						|
    }
 | 
						|
 | 
						|
} // namespace osmium
 | 
						|
 | 
						|
#endif // OSMIUM_OSM_TIMESTAMP_HPP
 |