mirror of
https://github.com/postgres/postgres.git
synced 2025-10-24 01:29:19 +03:00
Fix identify_system_timezone() so that it tests the behavior of the system
timezone setting in the current year and for 100 years back, rather than always examining years 1904-2004. The original coding would have problems distinguishing zones whose behavior diverged only after 2004; which is a situation we will surely face sometime, if it's not out there already. In passing, also prevent selection of the dummy "Factory" timezone, even if that's exactly what the system is using. Reporting time as GMT seems better than that.
This commit is contained in:
@@ -6,7 +6,7 @@
|
|||||||
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
|
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $PostgreSQL: pgsql/src/timezone/pgtz.c,v 1.48.2.1 2008/02/11 19:55:13 mha Exp $
|
* $PostgreSQL: pgsql/src/timezone/pgtz.c,v 1.48.2.2 2008/07/01 03:41:10 tgl Exp $
|
||||||
*
|
*
|
||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
@@ -185,7 +185,7 @@ scan_directory_ci(const char *dirname, const char *fname, int fnamelen,
|
|||||||
#define T_WEEK ((time_t) (60*60*24*7))
|
#define T_WEEK ((time_t) (60*60*24*7))
|
||||||
#define T_MONTH ((time_t) (60*60*24*31))
|
#define T_MONTH ((time_t) (60*60*24*31))
|
||||||
|
|
||||||
#define MAX_TEST_TIMES (52*100) /* 100 years, or 1904..2004 */
|
#define MAX_TEST_TIMES (52*100) /* 100 years */
|
||||||
|
|
||||||
struct tztry
|
struct tztry
|
||||||
{
|
{
|
||||||
@@ -354,6 +354,7 @@ identify_system_timezone(void)
|
|||||||
time_t t;
|
time_t t;
|
||||||
struct tztry tt;
|
struct tztry tt;
|
||||||
struct tm *tm;
|
struct tm *tm;
|
||||||
|
int thisyear;
|
||||||
int bestscore;
|
int bestscore;
|
||||||
char tmptzdir[MAXPGPATH];
|
char tmptzdir[MAXPGPATH];
|
||||||
int std_ofs;
|
int std_ofs;
|
||||||
@@ -366,11 +367,14 @@ identify_system_timezone(void)
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* Set up the list of dates to be probed to see how well our timezone
|
* Set up the list of dates to be probed to see how well our timezone
|
||||||
* matches the system zone. We first probe January and July of 2004; this
|
* matches the system zone. We first probe January and July of the
|
||||||
* serves to quickly eliminate the vast majority of the TZ database
|
* current year; this serves to quickly eliminate the vast majority of the
|
||||||
* entries. If those dates match, we probe every week from 2004 backwards
|
* TZ database entries. If those dates match, we probe every week for 100
|
||||||
* to late 1904. (Weekly resolution is good enough to identify DST
|
* years backwards from the current July. (Weekly resolution is good
|
||||||
* transition rules, since everybody switches on Sundays.) The further
|
* enough to identify DST transition rules, since everybody switches on
|
||||||
|
* Sundays.) This is sufficient to cover most of the Unix time_t range,
|
||||||
|
* and we don't want to look further than that since many systems won't
|
||||||
|
* have sane TZ behavior further back anyway. The further
|
||||||
* back the zone matches, the better we score it. This may seem like a
|
* back the zone matches, the better we score it. This may seem like a
|
||||||
* rather random way of doing things, but experience has shown that
|
* rather random way of doing things, but experience has shown that
|
||||||
* system-supplied timezone definitions are likely to have DST behavior
|
* system-supplied timezone definitions are likely to have DST behavior
|
||||||
@@ -380,9 +384,29 @@ identify_system_timezone(void)
|
|||||||
* (Note: we probe Thursdays, not Sundays, to avoid triggering
|
* (Note: we probe Thursdays, not Sundays, to avoid triggering
|
||||||
* DST-transition bugs in localtime itself.)
|
* DST-transition bugs in localtime itself.)
|
||||||
*/
|
*/
|
||||||
|
tnow = time(NULL);
|
||||||
|
tm = localtime(&tnow);
|
||||||
|
if (!tm)
|
||||||
|
return NULL; /* give up if localtime is broken... */
|
||||||
|
thisyear = tm->tm_year + 1900;
|
||||||
|
|
||||||
|
t = build_time_t(thisyear, 1, 15);
|
||||||
|
/*
|
||||||
|
* Round back to GMT midnight Thursday. This depends on the knowledge
|
||||||
|
* that the time_t origin is Thu Jan 01 1970. (With a different origin
|
||||||
|
* we'd be probing some other day of the week, but it wouldn't matter
|
||||||
|
* anyway unless localtime() had DST-transition bugs.)
|
||||||
|
*/
|
||||||
|
t -= (t % T_WEEK);
|
||||||
|
|
||||||
tt.n_test_times = 0;
|
tt.n_test_times = 0;
|
||||||
tt.test_times[tt.n_test_times++] = build_time_t(2004, 1, 15);
|
tt.test_times[tt.n_test_times++] = t;
|
||||||
tt.test_times[tt.n_test_times++] = t = build_time_t(2004, 7, 15);
|
|
||||||
|
t = build_time_t(thisyear, 7, 15);
|
||||||
|
t -= (t % T_WEEK);
|
||||||
|
|
||||||
|
tt.test_times[tt.n_test_times++] = t;
|
||||||
|
|
||||||
while (tt.n_test_times < MAX_TEST_TIMES)
|
while (tt.n_test_times < MAX_TEST_TIMES)
|
||||||
{
|
{
|
||||||
t -= T_WEEK;
|
t -= T_WEEK;
|
||||||
@@ -397,7 +421,12 @@ identify_system_timezone(void)
|
|||||||
&tt,
|
&tt,
|
||||||
&bestscore, resultbuf);
|
&bestscore, resultbuf);
|
||||||
if (bestscore > 0)
|
if (bestscore > 0)
|
||||||
|
{
|
||||||
|
/* Ignore zic's rather silly "Factory" time zone; use GMT instead */
|
||||||
|
if (strcmp(resultbuf, "Factory") == 0)
|
||||||
|
return NULL;
|
||||||
return resultbuf;
|
return resultbuf;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Couldn't find a match in the database, so next we try constructed zone
|
* Couldn't find a match in the database, so next we try constructed zone
|
||||||
|
Reference in New Issue
Block a user