The info pages for the GNU date
command contains this example:
For example, with the GNU
date
command you can answer the question "What time is it in New York when a Paris clock shows 6:30am on October 31, 2004?" by using a date beginning with `TZ="Europe/Paris"' as shown in the following shell transcript:$ export TZ="America/New_York" $ date --date='TZ="Europe/Paris" 2004-10-31 06:30' Sun Oct 31 01:30:00 EDT 2004
In this example, the '--date' operand begins with its own 'TZ' setting, so the rest of that operand is processed according to 'Europe/Paris' rules, treating the string
2004-10-31 06:30
as if it were in Paris. However, since the output of thedate
command is processed according to the overall time zone rules, it uses New York time. (Paris was normally six hours ahead of New York in 2004, but this example refers to a brief Halloween period when the gap was five hours.)
I am trying to accomplish essentially the same thing programatically in C without calling the date
program millions of times. Basically I am looking for a way to take an arbitrary date and time in one timezone and convert it to the equivalent date and time in another timezone either directly or via conversion to and from UTC. I don't care about the formats of the input and output time as long as I can manipulate them using standard functions (strftime
/strptime
/mktime
/etc).
The date
program appears to accomplish this using complex routines internal to the coreutils
package, I am looking for a way to do this in C using either standard POSIX/Linux routines or an external library. I looked at zoneinfo quite a bit which seemed promising but I cannot find any libraries to do anything useful with it.
For time:
gmtime
mktime
to renormalizeThe date calculation will be similar but a little more complicated.
I came up with a solution that seems to work on Linux with glibc, I don't know how portable this behavior is, any comments about portability or a better way to go about this would be welcome.
convert_time.c (No error checking for clarity):
The first argument is the source timezone (actually "TZ=Timezone" to pass to putenv), the second argument is the time in the specified format, the last argument is the destination timezone. I am using zoneinfo timezone names which glibc supports using the zoneinfo database.
The results of several DST test corner cases correspond to the results from the equivalent
date
command as well as this site which uses the zoneinfo database:I was actually looking for the same answer you did, and this is the solution I came up with. The following sample program takes the local system time and converts it to any timezone you want.