[BACK]Return to io.c CVS log [TXT][DIR] Up to [local] / src / usr.bin / calendar

Diff for /src/usr.bin/calendar/io.c between version 1.10 and 1.50

version 1.10, 2000/12/07 19:36:37 version 1.50, 2021/10/24 21:24:16
Line 12 
Line 12 
  * 2. Redistributions in binary form must reproduce the above copyright   * 2. Redistributions in binary form must reproduce the above copyright
  *    notice, this list of conditions and the following disclaimer in the   *    notice, this list of conditions and the following disclaimer in the
  *    documentation and/or other materials provided with the distribution.   *    documentation and/or other materials provided with the distribution.
  * 3. All advertising materials mentioning features or use of this software   * 3. Neither the name of the University nor the names of its contributors
  *    must display the following acknowledgement:  
  *      This product includes software developed by the University of  
  *      California, Berkeley and its contributors.  
  * 4. Neither the name of the University nor the names of its contributors  
  *    may be used to endorse or promote products derived from this software   *    may be used to endorse or promote products derived from this software
  *    without specific prior written permission.   *    without specific prior written permission.
  *   *
Line 33 
Line 29 
  * SUCH DAMAGE.   * SUCH DAMAGE.
  */   */
   
 #ifndef lint  
 static const char copyright[] =  
 "@(#) Copyright (c) 1989, 1993\n\  
         The Regents of the University of California.  All rights reserved.\n";  
 #endif /* not lint */  
   
 #ifndef lint  
 #if 0  
 static const char sccsid[] = "@(#)calendar.c  8.3 (Berkeley) 3/25/94";  
 #else  
 static char rcsid[] = "$OpenBSD$";  
 #endif  
 #endif /* not lint */  
   
 #include <sys/param.h>  
 #include <sys/stat.h>  #include <sys/stat.h>
 #include <sys/time.h>  #include <sys/time.h>
 #include <sys/types.h>  #include <sys/types.h>
 #include <sys/uio.h>  #include <sys/uio.h>
 #include <sys/wait.h>  #include <sys/wait.h>
 #include <sys/file.h>  
   
 #include <ctype.h>  #include <ctype.h>
 #include <err.h>  #include <err.h>
 #include <errno.h>  #include <errno.h>
   #include <fcntl.h>
 #include <locale.h>  #include <locale.h>
 #include <pwd.h>  #include <pwd.h>
 #include <stdio.h>  #include <stdio.h>
 #include <stdlib.h>  #include <stdlib.h>
 #include <string.h>  #include <string.h>
 #include <tzfile.h>  
 #include <unistd.h>  #include <unistd.h>
   #include <limits.h>
   
 #include "pathnames.h"  #include "pathnames.h"
 #include "calendar.h"  #include "calendar.h"
   
   
 char *calendarFile = "calendar";  /* default calendar file */  
 char *calendarHome = ".calendar"; /* HOME */  
 char *calendarNoMail = "nomail";  /* don't sent mail if this file exist */  
   
 struct iovec header[] = {  struct iovec header[] = {
         {"From: ", 6},          { "From: ", 6 },
         {NULL, 0},          { NULL, 0 },
         {" (Reminder Service)\nTo: ", 24},          { " (Reminder Service)\nTo: ", 24 },
         {NULL, 0},          { NULL, 0 },
         {"\nSubject: ", 10},          { "\nSubject: ", 10 },
         {NULL, 0},          { NULL, 0 },
         {"'s Calendar\nPrecedence: bulk\n\n",  30},          { "'s Calendar\nPrecedence: bulk\n",  29 },
           { "Auto-Submitted: auto-generated\n\n", 32 },
 };  };
   
   
 int     openf(char *path);  
   
 void  void
 cal()  cal(void)
 {  {
         register int printing;          int ch, l, i, bodun = 0, bodun_maybe = 0, var, printing;
         register char *p;  
         FILE *fp;  
         int ch, l, i;  
         int var;  
         char buf[2048 + 1];  
         struct event *events, *cur_evt, *ev1, *tmp;          struct event *events, *cur_evt, *ev1, *tmp;
           char buf[2048 + 1], *prefix = NULL, *p;
         struct match *m;          struct match *m;
           FILE *fp;
   
         events = NULL;          events = NULL;
         cur_evt = NULL;          cur_evt = NULL;
Line 116 
Line 89 
                 if (strncmp(buf, "LANG=", 5) == 0) {                  if (strncmp(buf, "LANG=", 5) == 0) {
                         (void) setlocale(LC_ALL, buf + 5);                          (void) setlocale(LC_ALL, buf + 5);
                         setnnames();                          setnnames();
                           if (!strcmp(buf + 5, "ru_RU.UTF-8") ||
                               !strcmp(buf + 5, "uk_UA.UTF-8") ||
                               !strcmp(buf + 5, "by_BY.UTF-8")) {
                                   bodun_maybe++;
                                   bodun = 0;
                                   free(prefix);
                                   prefix = NULL;
                           } else
                                   bodun_maybe = 0;
                         continue;                          continue;
                   } else if (strncmp(buf, "CALENDAR=", 9) == 0) {
                           char *ep;
   
                           if (buf[9] == '\0')
                                   calendar = 0;
                           else if (!strcasecmp(buf + 9, "julian")) {
                                   calendar = JULIAN;
                                   errno = 0;
                                   julian = strtoul(buf + 14, &ep, 10);
                                   if (buf[0] == '\0' || *ep != '\0')
                                           julian = 13;
                                   if ((errno == ERANGE && julian == ULONG_MAX) ||
                                       julian > 14)
                                           errx(1, "Julian calendar offset is too large");
                           } else if (!strcasecmp(buf + 9, "gregorian"))
                                   calendar = GREGORIAN;
                           else if (!strcasecmp(buf + 9, "lunar"))
                                   calendar = LUNAR;
                   } else if (bodun_maybe && strncmp(buf, "BODUN=", 6) == 0) {
                           bodun++;
                           free(prefix);
                           if ((prefix = strdup(buf + 6)) == NULL)
                                   err(1, NULL);
                           continue;
                 }                  }
                 /* User defined names for special events */                  /* User defined names for special events */
                 if ((p = strchr(buf, '='))) {                  if ((p = strchr(buf, '='))) {
                         for (i = 0; i < NUMEV; i++) {                          for (i = 0; i < NUMEV; i++) {
                         if (strncasecmp(buf, spev[i].name, spev[i].nlen) == 0 &&                                  if (strncasecmp(buf, spev[i].name,
                             (p - buf == spev[i].nlen) && buf[spev[i].nlen + 1]) {                                      spev[i].nlen) == 0 &&
                                 p++;                                      (p - buf == spev[i].nlen) &&
                                 if (spev[i].uname != NULL)                                      buf[spev[i].nlen + 1]) {
                                           p++;
                                         free(spev[i].uname);                                          free(spev[i].uname);
                                 if ((spev[i].uname = strdup(p)) == NULL)                                          if ((spev[i].uname = strdup(p)) == NULL)
                                         errx(1, "cannot allocate memory");                                                  err(1, NULL);
                                 spev[i].ulen = strlen(p);                                          spev[i].ulen = strlen(p);
                                 i = NUMEV + 1;                                          i = NUMEV + 1;
                                   }
                         }                          }
                         }                          if (i > NUMEV)
                 if (i > NUMEV)                                  continue;
                         continue;  
                 }                  }
                 if (buf[0] != '\t') {                  if (buf[0] != '\t') {
                         printing = (m = isnow(buf)) ? 1 : 0;                          printing = (m = isnow(buf, bodun)) ? 1 : 0;
                         if ((p = strchr(buf, '\t')) == NULL) {                          if ((p = strchr(buf, '\t')) == NULL) {
                                 printing = 0;                                  printing = 0;
                                 continue;                                  continue;
Line 149 
Line 156 
                                 var = 0;                                  var = 0;
                         if (printing) {                          if (printing) {
                                 struct match *foo;                                  struct match *foo;
   
                                 ev1 = NULL;                                  ev1 = NULL;
                                 while (m) {                                  while (m) {
                                 cur_evt = (struct event *) malloc(sizeof(struct event));                                          cur_evt = malloc(sizeof(struct event));
                                 if (cur_evt == NULL)                                          if (cur_evt == NULL)
                                         errx(1, "cannot allocate memory");                                                  err(1, NULL);
   
                                 cur_evt->when = m->when;                                          cur_evt->when = m->when;
                                 snprintf(cur_evt->print_date,                                          snprintf(cur_evt->print_date,
                                     sizeof(cur_evt->print_date), "%s%c",                                              sizeof(cur_evt->print_date), "%s%c",
                                     m->print_date, (var + m->var) ? '*' : ' ');                                              m->print_date, (var + m->var) ? '*' : ' ');
                                 if (ev1) {                                          if (ev1) {
                                         cur_evt->desc = ev1->desc;                                                  cur_evt->desc = ev1->desc;
                                         cur_evt->ldesc = NULL;                                                  cur_evt->ldesc = NULL;
                                 } else {                                          } else {
                                         if ((cur_evt->ldesc = strdup(p)) == NULL)                                                  if (m->bodun && prefix) {
                                                 errx(1, "cannot allocate memory");                                                          if (asprintf(&cur_evt->ldesc,
                                         cur_evt->desc = &(cur_evt->ldesc);                                                              "\t%s %s", prefix,
                                         ev1 = cur_evt;                                                              p + 1) == -1)
                                                                   err(1, NULL);
                                                   } else if ((cur_evt->ldesc =
                                                       strdup(p)) == NULL)
                                                           err(1, NULL);
                                                   cur_evt->desc = &(cur_evt->ldesc);
                                                   ev1 = cur_evt;
                                           }
                                           insert(&events, cur_evt);
                                           foo = m;
                                           m = m->next;
                                           free(foo);
                                 }                                  }
                                 insert(&events, cur_evt);  
                                 foo = m;  
                                 m = m->next;  
                                 free(foo);  
                                 }  
                         }                          }
                   } else if (printing) {
                           if (asprintf(&p, "%s\n%s", ev1->ldesc,
                               buf) == -1)
                                   err(1, NULL);
                           free(ev1->ldesc);
                           ev1->ldesc = p;
                 }                  }
                 else if (printing) {  
                         if ((ev1->ldesc = realloc(ev1->ldesc,  
                             (2 + strlen(ev1->ldesc) + strlen(buf)))) == NULL)  
                                 errx(1, "cannot allocate memory");  
                         strcat(ev1->ldesc, "\n");  
                         strcat(ev1->ldesc, buf);  
                 }  
         }          }
         tmp = events;          tmp = events;
         while (tmp) {          while (tmp) {
Line 192 
Line 204 
         tmp = events;          tmp = events;
         while (tmp) {          while (tmp) {
                 events = tmp;                  events = tmp;
                 if (tmp->ldesc)                  free(tmp->ldesc);
                         free(tmp->ldesc);  
                 tmp = tmp->next;                  tmp = tmp->next;
                 free(events);                  free(events);
         }          }
Line 201 
Line 212 
 }  }
   
 int  int
 getfield(p, endp, flags)  getfield(char *p, char **endp, int *flags)
         char *p, **endp;  
         int *flags;  
 {  {
         int val, var, i;          int val, var, i;
         char *start, savech;          char *start, savech;
   
         for (; !isdigit(*p) && !isalpha(*p) && *p != '*' && *p != '\t'; ++p)          for (; !isdigit((unsigned char)*p) && !isalpha((unsigned char)*p) &&
               *p != '*' && *p != '\t'; ++p)
                 ;                  ;
         if (*p == '*') {                        /* `*' is every month */          if (*p == '*') {                        /* `*' is every month */
                 *flags |= F_ISMONTH;                  *flags |= F_ISMONTH;
                 *endp = p+1;                  *endp = p+1;
                 return (-1);    /* means 'every month' */                  return (-1);    /* means 'every month' */
         }          }
         if (isdigit(*p)) {          if (isdigit((unsigned char)*p)) {
                 val = strtol(p, &p, 10);        /* if 0, it's failure */                  val = strtol(p, &p, 10);        /* if 0, it's failure */
                 for (; !isdigit(*p) && !isalpha(*p) && *p != '*'; ++p)                  for (; !isdigit((unsigned char)*p) &&
                       !isalpha((unsigned char)*p) && *p != '*'; ++p)
                         ;                          ;
                 *endp = p;                  *endp = p;
                 return (val);                  return (val);
         }          }
         for (start = p; isalpha(*++p);)          for (start = p; isalpha((unsigned char)*++p);)
                 ;                  ;
   
         /* Sunday-1 */          /* Sunday-1 */
         if (*p == '+' || *p == '-')          if (*p == '+' || *p == '-')
             for(; isdigit(*++p);)                  for(; isdigit((unsigned char)*++p); )
                 ;                          ;
   
         savech = *p;          savech = *p;
         *p = '\0';          *p = '\0';
Line 239 
Line 250 
   
         /* Day */          /* Day */
         else if ((val = getday(start)) != 0) {          else if ((val = getday(start)) != 0) {
             *flags |= F_ISDAY;                  *flags |= F_ISDAY;
   
             /* variable weekday */                  /* variable weekday */
             if ((var = getdayvar(start)) != 0) {                  if ((var = getdayvar(start)) != 0) {
                 if (var <= 5 && var >= -4)                          if (var <= 5 && var >= -4)
                     val += var * 10;                                  val += var * 10;
 #ifdef DEBUG  #ifdef DEBUG
                 printf("var: %d\n", var);                          printf("var: %d\n", var);
 #endif  #endif
             }                  }
         }          }
   
         /* Try specials (Easter, Paskha, ...) */          /* Try specials (Easter, Paskha, ...) */
Line 266 
Line 277 
                         }                          }
                 }                  }
                 if (i > NUMEV) {                  if (i > NUMEV) {
                         switch(*start) {                          const char *errstr;
   
                           switch (*start) {
                         case '-':                          case '-':
                         case '+':                          case '+':
                            var = atoi(start);                                  var = strtonum(start + 1, 0, 365, &errstr);
                            if (var > 365 || var < -365)                                  if (errstr)
                                    return (0); /* Someone is just being silly */                                          return (0); /* Someone is just being silly */
                            val += (NUMEV + 1) * var;                                  if (*start == '-')
                            /* We add one to the matching event and multiply by                                          var = -var;
                             * (NUMEV + 1) so as not to return 0 if there's a match.                                  val += (NUMEV + 1) * var;
                             * val will overflow if there is an obscenely large                                  /* We add one to the matching event and multiply by
                             * number of special events. */                                   * (NUMEV + 1) so as not to return 0 if there's a match.
                            break;                                   * val will overflow if there is an obscenely large
                                    * number of special events. */
                                   break;
                         }                          }
                 *flags |= F_SPECIAL;                          *flags |= F_SPECIAL;
                 }                  }
                 if (!(*flags & F_SPECIAL)) {                  if (!(*flags & F_SPECIAL)) {
                 /* undefined rest */                          /* undefined rest */
                         *p = savech;                          *p = savech;
                         return (0);                          return (0);
                 }                  }
         }          }
         for (*p = savech; !isdigit(*p) && !isalpha(*p) && *p != '*' && *p != '\t'; ++p)          for (*p = savech; !isdigit((unsigned char)*p) &&
               !isalpha((unsigned char)*p) && *p != '*' && *p != '\t'; ++p)
                 ;                  ;
         *endp = p;          *endp = p;
         return (val);          return (val);
 }  }
   
 char path[MAXPATHLEN];  
   
 int  FILE *
 openf(path)  opencal(void)
         char *path;  
 {  {
           int pdes[2], fdin;
         struct stat st;          struct stat st;
         int fd;  
   
         fd = open(path, O_RDONLY|O_NONBLOCK);  
         if (fd == -1)  
                 return (-1);  
         if (fstat(fd, &st) == -1) {  
                 close(fd);  
                 return (-1);  
         }  
         if ((st.st_mode & S_IFMT) != S_IFREG) {  
                 close (fd);  
                 return (-1);  
         }  
   
         fcntl(fd, F_SETFD, fcntl(fd, F_GETFD, 0) &~ O_NONBLOCK);  
         return (fd);  
 }  
   
 FILE *  
 opencal()  
 {  
         int fd, pdes[2];  
         int fdin;  
         struct stat sbuf;  
   
         /* open up calendar file as stdin */          /* open up calendar file as stdin */
         if ((fdin = openf(calendarFile)) == -1) {          if ((fdin = open(calendarFile, O_RDONLY)) == -1 ||
                 if (doall) {              fstat(fdin, &st) == -1 || !S_ISREG(st.st_mode)) {
                         if (chdir(calendarHome) != 0)                  if (!doall) {
                                 return (NULL);  
                         if (stat(calendarNoMail, &sbuf) == 0)  
                                 return (NULL);  
                         if ((fdin = openf(calendarFile)) == -1)  
                                 return (NULL);  
                 } else {  
                         char *home = getenv("HOME");                          char *home = getenv("HOME");
                         if (home == NULL || *home == '\0')                          if (home == NULL || *home == '\0')
                                 errx(1, "cannot get home directory");                                  errx(1, "cannot get home directory");
                         if (!(chdir(home) == 0 &&                          if (!(chdir(home) == 0 &&
                             chdir(calendarHome) == 0 &&                              chdir(calendarHome) == 0 &&
                             (fdin = openf(calendarFile)) != -1))                              (fdin = open(calendarFile, O_RDONLY)) != -1))
                                 errx(1, "no calendar file: ``%s'' or ``~/%s/%s",                                  errx(1, "no calendar file: \"%s\" or \"~/%s/%s\"",
                                     calendarFile, calendarHome, calendarFile);                                      calendarFile, calendarHome, calendarFile);
                 }                  }
         }          }
         if (pipe(pdes) < 0)  
           if (pipe(pdes) == -1) {
                   close(fdin);
                 return (NULL);                  return (NULL);
           }
         switch (vfork()) {          switch (vfork()) {
         case -1:                        /* error */          case -1:                        /* error */
                 (void)close(pdes[0]);                  (void)close(pdes[0]);
                 (void)close(pdes[1]);                  (void)close(pdes[1]);
                   close(fdin);
                 return (NULL);                  return (NULL);
         case 0:          case 0:
                 dup2(fdin, STDIN_FILENO);                  dup2(fdin, STDIN_FILENO);
Line 360 
Line 349 
                         (void)close(pdes[1]);                          (void)close(pdes[1]);
                 }                  }
                 (void)close(pdes[0]);                  (void)close(pdes[0]);
                 (void)setuid(geteuid());                  /*
                 (void)setgid(getegid());                   * Set stderr to /dev/null.  Necessary so that cron does not
                 execl(_PATH_CPP, "cpp", "-P", "-I.", _PATH_INCLUDE, NULL);                   * wait for cpp to finish if it's running calendar -a.
                    */
                   if (doall) {
                           int fderr;
                           fderr = open(_PATH_DEVNULL, O_WRONLY);
                           if (fderr == -1)
                                   _exit(0);
                           (void)dup2(fderr, STDERR_FILENO);
                           (void)close(fderr);
                   }
                   execl(_PATH_CPP, "cpp", "-traditional", "-undef", "-U__GNUC__",
                       "-P", "-I.", _PATH_INCLUDE, (char *)NULL);
                 warn(_PATH_CPP);                  warn(_PATH_CPP);
                 _exit(1);                  _exit(1);
         }          }
Line 376 
Line 376 
                 return (stdout);                  return (stdout);
   
         /* set output to a temporary file, so if no output don't send mail */          /* set output to a temporary file, so if no output don't send mail */
         (void)snprintf(path, sizeof(path), "%s/_calXXXXXX", _PATH_TMP);          return(tmpfile());
         if ((fd = mkstemp(path)) < 0)  
                 return (NULL);  
         return (fdopen(fd, "w+"));  
 }  }
   
 void  void
 closecal(fp)  closecal(FILE *fp)
         FILE *fp;  
 {  {
         struct stat sbuf;          struct stat sbuf;
         int nread, pdes[2], status;          int nread, pdes[2], status;
         char buf[1024];          char buf[1024];
           pid_t pid = -1;
   
         if (!doall)          if (!doall)
                 return;                  return;
Line 396 
Line 393 
         (void)rewind(fp);          (void)rewind(fp);
         if (fstat(fileno(fp), &sbuf) || !sbuf.st_size)          if (fstat(fileno(fp), &sbuf) || !sbuf.st_size)
                 goto done;                  goto done;
         if (pipe(pdes) < 0)          if (pipe(pdes) == -1)
                 goto done;                  goto done;
         switch (vfork()) {          switch ((pid = vfork())) {
         case -1:                        /* error */          case -1:                        /* error */
                 (void)close(pdes[0]);                  (void)close(pdes[0]);
                 (void)close(pdes[1]);                  (void)close(pdes[1]);
Line 410 
Line 407 
                         (void)close(pdes[0]);                          (void)close(pdes[0]);
                 }                  }
                 (void)close(pdes[1]);                  (void)close(pdes[1]);
                 (void)setuid(geteuid());  
                 (void)setgid(getegid());  
                 execl(_PATH_SENDMAIL, "sendmail", "-i", "-t", "-F",                  execl(_PATH_SENDMAIL, "sendmail", "-i", "-t", "-F",
                     "\"Reminder Service\"", NULL);                      "\"Reminder Service\"", (char *)NULL);
                 warn(_PATH_SENDMAIL);                  warn(_PATH_SENDMAIL);
                 _exit(1);                  _exit(1);
         }          }
Line 422 
Line 417 
   
         header[1].iov_base = header[3].iov_base = pw->pw_name;          header[1].iov_base = header[3].iov_base = pw->pw_name;
         header[1].iov_len = header[3].iov_len = strlen(pw->pw_name);          header[1].iov_len = header[3].iov_len = strlen(pw->pw_name);
         writev(pdes[1], header, 7);          writev(pdes[1], header, 8);
         while ((nread = read(fileno(fp), buf, sizeof(buf))) > 0)          while ((nread = read(fileno(fp), buf, sizeof(buf))) > 0)
                 (void)write(pdes[1], buf, nread);                  (void)write(pdes[1], buf, nread);
         (void)close(pdes[1]);          (void)close(pdes[1]);
 done:   (void)fclose(fp);  done:   (void)fclose(fp);
         (void)unlink(path);          if (pid != -1) {
         while (wait(&status) >= 0)                  while (waitpid(pid, &status, 0) == -1) {
                 ;                          if (errno != EINTR)
                                   break;
                   }
           }
 }  }
   
   
 void  void
 insert(head, cur_evt)  insert(struct event **head, struct event *cur_evt)
         struct event **head;  
         struct event *cur_evt;  
 {  {
         struct event *tmp, *tmp2;          struct event *tmp, *tmp2;
   

Legend:
Removed from v.1.10  
changed lines
  Added in v.1.50