NAME
Time::UTC::Now - determine current time in UTC correctly
SYNOPSIS
use Time::UTC::Now qw(now_utc_rat now_utc_sna now_utc_flt);
($day, $secs, $bound) = now_utc_rat;
($day, $secs, $bound) = now_utc_rat(1);
($day, $secs, $bound) = now_utc_sna;
($day, $secs, $bound) = now_utc_sna(1);
($day, $secs, $bound) = now_utc_flt;
($day, $secs, $bound) = now_utc_flt(1);
use Time::UTC::Now qw(utc_day_to_cjdn);
$cjdn = utc_day_to_cjdn($day);
DESCRIPTION
This module is one answer to the question "what time is it?". It determines the current time on the UTC scale, handling leap seconds correctly, and puts a bound on how inaccurate it could be. It is the rigorously correct approach to determining civil time. It is designed to interoperate with Time::UTC, which knows all about the UTC time scale.
UTC (Coordinated Universal Time) is a time scale derived from International Atomic Time (TAI). UTC divides time up into days, and each day into seconds. The seconds are atomically-realised SI seconds, of uniform length. Most UTC days are exactly 86400 seconds long, but occasionally there is a day of length 86401 s or (theoretically) 86399 s. These leap seconds are used to keep the UTC day approximately synchronised with the non-uniform rotation of the Earth. (Prior to 1972 a different mechanism was used for UTC, but that's not an issue here.)
Because UTC days have differing lengths, instants on the UTC scale are identified here by the combination of a day number and a number of seconds since midnight within the day. In this module the day number is the integral number of days since 1958-01-01, which is the epoch of the TAI scale which underlies UTC. This is the convention used by the Time::UTC
module. That module has some functions to format these numbers for display. For a more general solution, use the utc_day_to_cjdn
function to translate to a standard Chronological Julian Day Number, which can be used as input to a calendar module.
FUNCTIONS
- now_utc_rat[(DEMAND_ACCURACY)]
-
Returns a list of three values. The first two values identify a current UTC instant, in the form of a day number (number of days since the TAI epoch) and a number of seconds since midnight within the day. The third value is an inaccuracy bound, as a number of seconds, or
undef
if no accurate answer could be determined.If an inaccuracy bound is returned then this function is claiming to have answered correctly, to within the specified margin. That is, some instant during the execution of
now_utc_rat
is within the specified margin of the instant identified. (This semantic differs from older current-time interfaces that are content to return an instant that has already passed.)The inaccuracy bound is measured in UTC seconds; that is, in SI seconds on the Terran geoid as realised by atomic clocks. This differs from SI seconds at the computer's location, but the difference is only apparent if the computer hardware is significantly time dilated with respect to the geoid.
If
undef
is returned instead of an inaccuracy bound then this function could not find a trustable answer. Either the clock available was not properly synchronised or its accuracy could not be established. Whatever time could be found is returned, but this function makes no claim that it is accurate. It should be treated with suspicion. In practice, clocks of this nature are especially likely to misbehave around leap seconds.The function
die
s if it could not find a plausible time at all. If DEMAND_ACCURACY is supplied and true then it will also die if it could not find an accurate answer, instead of returning withundef
for the inaccuracy bound.All three return values are in the form of
Math::BigRat
objects. This retains full resolution, is future-proof, and is easy to manipulate, but beware thatMath::BigRat
is currently rather slow. If performance is a problem then consider using one of the functions below that return the results in other formats. - now_utc_sna[(DEMAND_ACCURACY)]
-
This performs exactly the same operation as
now_utc_rat
, but returns the results in a different form. The day number is returned as a Perl integer. The time since midnight and the inaccuracy bound (if present) are each returned in the form of a three-element array, giving a high-resolution fixed-point number of seconds. The first element is the integral number of whole seconds, the second is an integral number of nanoseconds in the range [0, 1000000000), and the third is an integral number of attoseconds in the same range.This form of return value is fairly efficient. It is convenient for decimal output, but awkward to do arithmetic with. Its resolution is adequate for the foreseeable future, but could in principle be obsoleted some day.
It is presumed that native integer formats will grow fast enough to always represent the day number fully; if not, 31 bits will overflow late in the sixth megayear of the Common Era. (Average day length by then is projected to be around 86520 s, posing more serious problems for UTC.)
The inaccuracy bound describes the actual time represented in the return values, not an internal value that was rounded to generate the return values.
- now_utc_flt[(DEMAND_ACCURACY)]
-
This performs exactly the same operation as
now_utc_rat
, but returns all the results as Perl numbers (the day number as an integer, with the same caveat as fornow_utc_sna
). This form of return value is very efficient and easy to manipulate. However, its resolution is limited, rendering it obsolete in the near future unless floating point number formats get bigger.The inaccuracy bound describes the actual time represented in the return values, not an internal value that was rounded to generate the return values.
- utc_day_to_cjdn(DAY)
-
This function takes a number of days since the TAI epoch and returns the corresponding Chronological Julian Day Number (a number of days since -4713-11-24). CJDN is a standard day numbering that is useful as an interchange format between implementations of different calendars. There is no bound on the permissible day numbers.
TECHNIQUES
There are several interfaces available to determine the time on a computer, and most of them suck. This module will attempt to use the best interface available when it runs. It knows about the following:
- ntp_adjtime()
-
Designed for precision timekeeping, this interface gives some leap second indications and an inaccuracy bound on the time it returns. Both are faulty in their raw form, but they are corrected by this module. (Those interested in the gory details are invited to read the source.) Resolution 1 us.
- gettimeofday()
-
Misbehaves around leap seconds, and does not give an inaccuracy bound. Resolution 1 us.
- Time::Unix::time()
-
Misbehaves around leap seconds, and does not give an inaccuracy bound. Resolution 1 s. The
Time::Unix
module corrects for the varying epochs oftime()
across OSes; nativetime()
is not a suitable fallback.
The author would welcome patches to this module to make use of high-precision interfaces, along the lines of ntp_adjtime()
, on non-Unix operating systems.
SEE ALSO
AUTHOR
Andrew Main (Zefram) <zefram@fysh.org>
COPYRIGHT
Copyright (C) 2006 Andrew Main (Zefram) <zefram@fysh.org>
This module is free software; you can redistribute it and/or modify it under the same terms as Perl itself.