Skip to content

Commit

Permalink
[PATCH] Do date parsing by hand...
Browse files Browse the repository at this point in the history
...since everything out there is either strange (libc mktime has issues
with timezones) or introduces unnecessary dependencies for people (libcurl).

This goes back to the old date parsing, but moves it out into a file of
its own, and does the "struct tm" to "seconds since epoch" handling by
hand. 

I grepped through the tz-database and it seems there's one "country"
left that has non-60-minute DST: Lord Howe Island.  All others dropped
that before 1970.
  • Loading branch information
Edgar Toernig authored and Linus Torvalds committed Apr 30, 2005
1 parent d167f14 commit ecee9d9
Show file tree
Hide file tree
Showing 4 changed files with 191 additions and 27 deletions.
4 changes: 2 additions & 2 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,8 @@ all: $(PROG)
install: $(PROG) $(SCRIPTS)
install $(PROG) $(SCRIPTS) $(HOME)/bin/

LIB_OBJS=read-cache.o sha1_file.o usage.o object.o commit.o tree.o blob.o tag.o
LIB_OBJS=read-cache.o sha1_file.o usage.o object.o commit.o tree.o blob.o \
tag.o date.o
LIB_FILE=libgit.a
LIB_H=cache.h object.h blob.h tree.h commit.h tag.h

Expand Down Expand Up @@ -91,7 +92,6 @@ git-diff-tree-helper: diff-tree-helper.c
git-tar-tree: tar-tree.c

git-http-pull: LIBS += -lcurl
git-commit-tree: LIBS += -lcurl

# Library objects..
blob.o: $(LIB_H)
Expand Down
3 changes: 3 additions & 0 deletions cache.h
Original file line number Diff line number Diff line change
Expand Up @@ -148,6 +148,9 @@ extern void *read_object_with_reference(const unsigned char *sha1,
unsigned long *size,
unsigned char *sha1_ret);

void parse_date(char *date, char *buf, int bufsize);
void datestamp(char *buf, int bufsize);

static inline void *xmalloc(int size)
{
void *ret = malloc(size);
Expand Down
27 changes: 2 additions & 25 deletions commit-tree.c
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@
#include <string.h>
#include <ctype.h>
#include <time.h>
#include <curl/curl.h>

#define BLOCKING (1ul << 14)

Expand Down Expand Up @@ -81,24 +80,6 @@ static void remove_special(char *p)
}
}

/* Gr. strptime is crap for this; it doesn't have a way to require RFC2822
(i.e. English) day/month names, and it doesn't work correctly with %z. */
static void parse_date(char *date, time_t *now, char *result, int maxlen)
{
char *p;
time_t then;

if ((then = curl_getdate(date, now)) == 0)
return;

/* find the timezone at the end */
p = date + strlen(date);
while (p > date && isdigit(*--p))
;
if ((*p == '+' || *p == '-') && strlen(p) == 5)
snprintf(result, maxlen, "%lu %5.5s", then, p);
}

static void check_valid(unsigned char *sha1, const char *expect)
{
void *buf;
Expand Down Expand Up @@ -132,8 +113,6 @@ int main(int argc, char **argv)
char *audate;
char comment[1000];
struct passwd *pw;
time_t now;
struct tm *tm;
char *buffer;
unsigned int size;

Expand Down Expand Up @@ -163,10 +142,8 @@ int main(int argc, char **argv)
strcat(realemail, ".");
getdomainname(realemail+strlen(realemail), sizeof(realemail)-strlen(realemail)-1);
}
time(&now);
tm = localtime(&now);

strftime(realdate, sizeof(realdate), "%s %z", tm);
datestamp(realdate, sizeof(realdate));
strcpy(date, realdate);

commitgecos = getenv("COMMIT_AUTHOR_NAME") ? : realgecos;
Expand All @@ -175,7 +152,7 @@ int main(int argc, char **argv)
email = getenv("AUTHOR_EMAIL") ? : realemail;
audate = getenv("AUTHOR_DATE");
if (audate)
parse_date(audate, &now, date, sizeof(date));
parse_date(audate, date, sizeof(date));

remove_special(gecos); remove_special(realgecos); remove_special(commitgecos);
remove_special(email); remove_special(realemail); remove_special(commitemail);
Expand Down
184 changes: 184 additions & 0 deletions date.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,184 @@
/*
* GIT - The information manager from hell
*
* Copyright (C) Linus Torvalds, 2005
*/

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <time.h>

static time_t my_mktime(struct tm *tm)
{
static const int mdays[] = {
0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334
};
int year = tm->tm_year - 70;
int month = tm->tm_mon;
int day = tm->tm_mday;

if (year < 0 || year > 129) /* algo only works for 1970-2099 */
return -1;
if (month < 0 || month > 11) /* array bounds */
return -1;
if (month < 2 || (year + 2) % 4)
day--;
return (year * 365 + (year + 1) / 4 + mdays[month] + day) * 24*60*60UL +
tm->tm_hour * 60*60 + tm->tm_min * 60 + tm->tm_sec;
}

static const char *month_names[] = {
"Jan", "Feb", "Mar", "Apr", "May", "Jun",
"Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
};

static const char *weekday_names[] = {
"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"
};


static char *skipfws(char *str)
{
while (isspace(*str))
str++;
return str;
}


/* Gr. strptime is crap for this; it doesn't have a way to require RFC2822
(i.e. English) day/month names, and it doesn't work correctly with %z. */
void parse_date(char *date, char *result, int maxlen)
{
struct tm tm;
char *p, *tz;
int i, offset;
time_t then;

memset(&tm, 0, sizeof(tm));

/* Skip day-name */
p = skipfws(date);
if (!isdigit(*p)) {
for (i=0; i<7; i++) {
if (!strncmp(p,weekday_names[i],3) && p[3] == ',') {
p = skipfws(p+4);
goto day;
}
}
return;
}

/* day */
day:
tm.tm_mday = strtoul(p, &p, 10);

if (tm.tm_mday < 1 || tm.tm_mday > 31)
return;

if (!isspace(*p))
return;

p = skipfws(p);

/* month */

for (i=0; i<12; i++) {
if (!strncmp(p, month_names[i], 3) && isspace(p[3])) {
tm.tm_mon = i;
p = skipfws(p+strlen(month_names[i]));
goto year;
}
}
return; /* Error -- bad month */

/* year */
year:
tm.tm_year = strtoul(p, &p, 10);

if (!tm.tm_year && !isspace(*p))
return;

if (tm.tm_year > 1900)
tm.tm_year -= 1900;

p=skipfws(p);

/* hour */
if (!isdigit(*p))
return;
tm.tm_hour = strtoul(p, &p, 10);

if (tm.tm_hour > 23)
return;

if (*p != ':')
return; /* Error -- bad time */
p++;

/* minute */
if (!isdigit(*p))
return;
tm.tm_min = strtoul(p, &p, 10);

if (tm.tm_min > 59)
return;

if (*p != ':')
goto zone;
p++;

/* second */
if (!isdigit(*p))
return;
tm.tm_sec = strtoul(p, &p, 10);

if (tm.tm_sec > 60)
return;

zone:
if (!isspace(*p))
return;

p = skipfws(p);

if (*p == '-')
offset = -60;
else if (*p == '+')
offset = 60;
else
return;

if (!isdigit(p[1]) || !isdigit(p[2]) || !isdigit(p[3]) || !isdigit(p[4]))
return;

tz = p;
i = strtoul(p+1, NULL, 10);
offset *= ((i % 100) + ((i / 100) * 60));

p = skipfws(p + 5);
if (*p && *p != '(') /* trailing comment like (EDT) is ok */
return;

then = my_mktime(&tm); /* mktime uses local timezone */
if (then == -1)
return;

then -= offset;

snprintf(result, maxlen, "%lu %5.5s", then, tz);
}

void datestamp(char *buf, int bufsize)
{
time_t now;
int offset;

time(&now);

offset = my_mktime(localtime(&now)) - now;
offset /= 60;

snprintf(buf, bufsize, "%lu %+05d", now, offset/60*100 + offset%60);
}

0 comments on commit ecee9d9

Please sign in to comment.