/* */ #include "TimeA2.h" #include #include "util.h" #include "array_fun.h" namespace aria2 { Time::Time():_good(true) { reset(); } Time::Time(const Time& time) { tv = time.tv; _good = time._good; } Time::Time(time_t sec):_good(true) { setTimeInSec(sec); } Time::Time(const struct timeval& tv):_good(true) { this->tv = tv; } Time::~Time() {} Time& Time::operator=(const Time& time) { if(this != &time) { tv = time.tv; _good = time._good; } return *this; } bool Time::operator<(const Time& time) const { return util::difftv(time.tv, tv) > 0; } void Time::reset() { gettimeofday(&tv, 0); } struct timeval Time::getCurrentTime() const { struct timeval now; gettimeofday(&now, 0); return now; } bool Time::elapsed(time_t sec) const { // Because of gettimeofday called from getCurrentTime() is slow, and most of // the time this function is called before specified time passes, we first do // simple test using time. // Then only when the further test is required, call gettimeofday. time_t now = time(0); if(tv.tv_sec+sec < now) { return true; } else if(tv.tv_sec+sec == now) { return util::difftv(getCurrentTime(), tv) >= static_cast(sec)*1000000; } else { return false; } } bool Time::elapsedInMillis(int64_t millis) const { return util::difftv(getCurrentTime(), tv)/1000 >= millis; } bool Time::isNewer(const Time& time) const { return util::difftv(this->tv, time.tv) > 0; } time_t Time::difference() const { return util::difftvsec(getCurrentTime(), tv); } time_t Time::difference(const struct timeval& now) const { return util::difftvsec(now, tv); } int64_t Time::differenceInMillis() const { return util::difftv(getCurrentTime(), tv)/1000; } int64_t Time::differenceInMillis(const struct timeval& now) const { return util::difftv(now, tv)/1000; } bool Time::isZero() const { return tv.tv_sec == 0 && tv.tv_usec == 0; } int64_t Time::getTimeInMicros() const { return (int64_t)tv.tv_sec*1000*1000+tv.tv_usec; } int64_t Time::getTimeInMillis() const { return (int64_t)tv.tv_sec*1000+tv.tv_usec/1000; } time_t Time::getTime() const { return tv.tv_sec; } void Time::setTimeInSec(time_t sec) { tv.tv_sec = sec; tv.tv_usec = 0; } bool Time::good() const { return _good; } bool Time::bad() const { return !_good; } Time Time::parse(const std::string& datetime, const std::string& format) { struct tm tm; memset(&tm, 0, sizeof(tm)); char* r = strptime(datetime.c_str(), format.c_str(), &tm); if(r != datetime.c_str()+datetime.size()) { return Time::null(); } time_t thetime = timegm(&tm); if(thetime == -1) { if(tm.tm_year >= 2038-1900) { thetime = INT32_MAX; } } return Time(thetime); } Time Time::parseRFC1123(const std::string& datetime) { return parse(datetime, "%a, %d %b %Y %H:%M:%S GMT"); } Time Time::parseRFC850(const std::string& datetime) { return parse(datetime, "%a, %d-%b-%y %H:%M:%S GMT"); } Time Time::parseRFC850Ext(const std::string& datetime) { return parse(datetime, "%a, %d-%b-%Y %H:%M:%S GMT"); } Time Time::parseHTTPDate(const std::string& datetime) { Time (*funcs[])(const std::string&) = { &parseRFC850, &parseRFC1123, &parseRFC850Ext, }; for(Time (**funcsp)(const std::string&) = &funcs[0]; funcsp != &funcs[arrayLength(funcs)]; ++funcsp) { Time t = (*funcsp)(datetime); if(t.good()) { return t; } } return Time::null(); } Time Time::null() { Time t(0); t._good = false; return t; } } // namespace aria2