From bc7cb1a15a67c859c8c71c4b52bb35fe9425a63d Mon Sep 17 00:00:00 2001 From: rsc Date: Sun, 23 Nov 2003 18:04:47 +0000 Subject: new utilities. the .C files compile but are renamed to avoid building automatically. --- src/cmd/calendar.c | 195 +++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 195 insertions(+) create mode 100644 src/cmd/calendar.c (limited to 'src/cmd/calendar.c') diff --git a/src/cmd/calendar.c b/src/cmd/calendar.c new file mode 100644 index 00000000..6202c5a3 --- /dev/null +++ b/src/cmd/calendar.c @@ -0,0 +1,195 @@ +#include +#include +#include +#include +#include + +typedef struct Date Date; +struct Date { + Reprog *p; /* an RE to match this date */ + Date *next; /* pointer to next in list */ +}; + +enum{ + Secondsperday = 24*60*60 +}; + +Biobuf in; +int debug, matchyear; + +Date *dates(Date**, Tm*); +void upper2lower(char*, char*, int); +void *alloc(unsigned int); + +void +main(int argc, char *argv[]) +{ + int i, fd, ahead; + long now; + char *line; + Tm *tm; + Date *first, *last, *d; + char buf[1024]; + + ahead = 0; + ARGBEGIN{ + case 'y': + matchyear = 1; + break; + case 'd': + debug = 1; + break; + case 'p': + ahead = atoi(ARGF()); + break; + default: + fprint(2, "usage: calendar [-y] [-d] [files ...]\n"); + exits("usage"); + }ARGEND; + + /* make a list of dates */ + now = time(0); + tm = localtime(now); + last = nil; + first = dates(&last, tm); + now += Secondsperday; + tm = localtime(now); + dates(&last, tm); + if(tm->wday == 6){ + now += Secondsperday; + tm = localtime(now); + dates(&last, tm); + } + if(tm->wday == 0){ + now += Secondsperday; + tm = localtime(now); + dates(&last, tm); + } + if(ahead){ + now = time(0); + now += ahead * Secondsperday; + tm = localtime(now); + dates(&last, tm); + } + + for(i=0; inext) + if(regexec(d->p, buf, 0, 0)){ + print("%s\n", line); + break; + } + } + close(fd); + } + exits(""); +} + +char *months[] = +{ + "january", + "february", + "march", + "april", + "may", + "june", + "july", + "august", + "september", + "october", + "november", + "december" +}; + +/* + * Generate two Date structures. First has month followed by day; + * second has day followed by month. Link them into list after + * last, and return the first. + */ +Date* +dates(Date **last, Tm *tm) +{ + Date *first; + Date *nd; + char mo[128], buf[128]; + + if(utflen(months[tm->mon]) > 3) + snprint(mo, sizeof mo, "%3.3s(%s)?", + months[tm->mon], months[tm->mon]+3); + else + snprint(mo, sizeof mo, "%3.3s", months[tm->mon]); + if (matchyear) + snprint(buf, sizeof buf, + "(^| |\t)((%s( |\t)+)|(%d/))%d( |\t|$)(((%d|%d)( |\t|$))|[^0-9]|([0-9]+[^0-9 \t]))", + mo, tm->mon+1, tm->mday, tm->year+1900, tm->year%100); + else + snprint(buf, sizeof buf, + "(^| |\t)((%s( |\t)+)|(%d/))%d( |\t|$)", + mo, tm->mon+1, tm->mday); + if(debug) + print("%s\n", buf); + + first = alloc(sizeof(Date)); + if(*last) + (*last)->next = first; + first->p = regcomp(buf); + + if (matchyear) + snprint(buf, sizeof buf, + "(^| |\t)%d( |\t)+(%s)( |\t|$)(((%d|%d)( |\t|$))|[^0-9]|([0-9]+[^0-9 \t]))", + tm->mday, mo, tm->year+1900, tm->year%100); + else + snprint(buf, sizeof buf, + "(^| |\t)%d( |\t)+(%s)( |\t|$)", + tm->mday, mo); + if(debug) + print("%s\n", buf); + nd = alloc(sizeof(Date)); + nd->p = regcomp(buf); + nd->next = 0; + first->next = nd; + *last = nd; + + return first; +} + +/* + * Copy 'from' to 'to', converting to lower case + */ +void +upper2lower(char *to, char *from, int len) +{ + while(--len>0 && *from!='\0') + *to++ = tolower(*from++); + *to = 0; +} + +/* + * Call malloc and check for errors + */ +void* +alloc(unsigned int n) +{ + void *p; + + p = malloc(n); + if(p == 0){ + fprint(2, "calendar: malloc failed: %r\n"); + exits("malloc"); + } + return p; +} -- cgit v1.2.3