// The MIT License (MIT) // // Copyright (c) 2017, 2018 Tomasz KamiƄski // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // // 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 AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE // SOFTWARE. #include "tz.h" #include #include //used to count number of conversion int conversions = 0; //to/from impl struct mil_clock { using duration = typename std::common_type::type; using rep = duration::rep; using period = duration::period; using time_point = std::chrono::time_point; static constexpr date::sys_days epoch{date::days{1000}}; template static std::chrono::time_point::type> to_sys(std::chrono::time_point const& tp) { ++conversions; return epoch + tp.time_since_epoch(); } template static std::chrono::time_point::type> from_sys(std::chrono::time_point const& tp) { ++conversions; using res = std::chrono::time_point::type>; return res(tp - epoch); } template static std::chrono::time_point::type> to_local(std::chrono::time_point const& tp) { return date::clock_cast(to_sys(tp)); } template static std::chrono::time_point::type> from_local(std::chrono::time_point const& tp) { return from_sys(date::clock_cast(tp)); } static time_point now() { return from_sys(std::chrono::system_clock::now()); } }; date::sys_days const mil_clock::epoch; // traits example struct s2s_clock { using duration = std::chrono::system_clock::duration; using rep = duration::rep; using period = duration::period; using time_point = std::chrono::time_point; template static std::chrono::time_point to_sys(std::chrono::time_point const& tp) { ++conversions; return std::chrono::time_point(tp.time_since_epoch()); } template static std::chrono::time_point from_sys(std::chrono::time_point const& tp) { ++conversions; return std::chrono::time_point(tp.time_since_epoch()); } static time_point now() { return from_sys(std::chrono::system_clock::now()); } }; namespace date { template<> struct clock_time_conversion { template std::chrono::time_point::type> operator()(std::chrono::time_point const& tp) { ++conversions; using res = std::chrono::time_point::type>; return res(tp.time_since_epoch() - mil_clock::epoch.time_since_epoch()); } }; } int main() { using namespace date; using sys_clock = std::chrono::system_clock; // self { sys_days st(1997_y/dec/12); auto mt = mil_clock::from_sys(st); assert(clock_cast(mt) == mt); } // mil <-> local { local_days lt(1997_y/dec/12); auto mt = mil_clock::from_local(lt); assert(clock_cast(lt) == mt); assert(clock_cast(mt) == lt); } // mil <-> sys { sys_days st(1997_y/dec/12); auto mt = mil_clock::from_sys(st); assert(clock_cast(st) == mt); assert(clock_cast(mt) == st); } // mil <-> utc { sys_days st(1997_y/dec/12); auto mt = mil_clock::from_sys(st); auto ut = utc_clock::from_sys(st); assert(clock_cast(ut) == mt); assert(clock_cast(mt) == ut); } // mil <-> tai { sys_days st(1997_y/dec/12); auto mt = mil_clock::from_sys(st); auto ut = utc_clock::from_sys(st); auto tt = tai_clock::from_utc(ut); assert(clock_cast(mt) == tt); assert(clock_cast(tt) == mt); } // mil <-> gps { sys_days st(1997_y/dec/12); auto mt = mil_clock::from_sys(st); auto ut = utc_clock::from_sys(st); auto gt = gps_clock::from_utc(ut); assert(clock_cast(mt) == gt); assert(clock_cast(gt) == mt); } // s2s -> mil { sys_days st(1997_y/dec/12); auto mt = mil_clock::from_sys(st); auto s2t = s2s_clock::from_sys(st); //direct trait conversion conversions = 0; assert(clock_cast(s2t) == mt); assert(conversions == 1); //uses sys_clock conversions = 0; assert(clock_cast(mt) == s2t); assert(conversions == 2); } }