unix.c
上传用户:ycwykj01
上传日期:2007-01-04
资源大小:1819k
文件大小:68k
源码类别:

网络编程

开发平台:

Unix_Linux

  1. /*
  2.  * Program: UNIX mail routines
  3.  *
  4.  * Author: Mark Crispin
  5.  * Networks and Distributed Computing
  6.  * Computing & Communications
  7.  * University of Washington
  8.  * Administration Building, AG-44
  9.  * Seattle, WA  98195
  10.  * Internet: MRC@CAC.Washington.EDU
  11.  *
  12.  * Date: 20 December 1989
  13.  * Last Edited: 14 October 1999
  14.  *
  15.  * Copyright 1999 by the University of Washington
  16.  *
  17.  *  Permission to use, copy, modify, and distribute this software and its
  18.  * documentation for any purpose and without fee is hereby granted, provided
  19.  * that the above copyright notice appears in all copies and that both the
  20.  * above copyright notice and this permission notice appear in supporting
  21.  * documentation, and that the name of the University of Washington not be
  22.  * used in advertising or publicity pertaining to distribution of the software
  23.  * without specific, written prior permission.  This software is made
  24.  * available "as is", and
  25.  * THE UNIVERSITY OF WASHINGTON DISCLAIMS ALL WARRANTIES, EXPRESS OR IMPLIED,
  26.  * WITH REGARD TO THIS SOFTWARE, INCLUDING WITHOUT LIMITATION ALL IMPLIED
  27.  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, AND IN
  28.  * NO EVENT SHALL THE UNIVERSITY OF WASHINGTON BE LIABLE FOR ANY SPECIAL,
  29.  * INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  30.  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, TORT
  31.  * (INCLUDING NEGLIGENCE) OR STRICT LIABILITY, ARISING OUT OF OR IN CONNECTION
  32.  * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  33.  *
  34.  */
  35. /* DEDICATION
  36.  *
  37.  *  This file is dedicated to my dog, Unix, also known as Yun-chan and
  38.  * Unix J. Terwilliker Jehosophat Aloysius Monstrosity Animal Beast.  Unix
  39.  * passed away at the age of 11 1/2 on September 14, 1996, 12:18 PM PDT, after
  40.  * a two-month bout with cirrhosis of the liver.
  41.  *
  42.  *  He was a dear friend, and I miss him terribly.
  43.  *
  44.  *  Lift a leg, Yunie.  Luv ya forever!!!!
  45.  */
  46. #include <stdio.h>
  47. #include <ctype.h>
  48. #include <errno.h>
  49. extern int errno; /* just in case */
  50. #include <signal.h>
  51. #include "mail.h"
  52. #include "osdep.h"
  53. #include <time.h>
  54. #include <sys/stat.h>
  55. #include "unix.h"
  56. #include "pseudo.h"
  57. #include "fdstring.h"
  58. #include "misc.h"
  59. #include "dummy.h"
  60. /* UNIX mail routines */
  61. /* Driver dispatch used by MAIL */
  62. DRIVER unixdriver = {
  63.   "unix", /* driver name */
  64.   DR_LOCAL|DR_MAIL, /* driver flags */
  65.   (DRIVER *) NIL, /* next driver */
  66.   unix_valid, /* mailbox is valid for us */
  67.   unix_parameters, /* manipulate parameters */
  68.   unix_scan, /* scan mailboxes */
  69.   unix_list, /* list mailboxes */
  70.   unix_lsub, /* list subscribed mailboxes */
  71.   NIL, /* subscribe to mailbox */
  72.   NIL, /* unsubscribe from mailbox */
  73.   unix_create, /* create mailbox */
  74.   unix_delete, /* delete mailbox */
  75.   unix_rename, /* rename mailbox */
  76.   NIL, /* status of mailbox */
  77.   unix_open, /* open mailbox */
  78.   unix_close, /* close mailbox */
  79.   NIL, /* fetch message "fast" attributes */
  80.   NIL, /* fetch message flags */
  81.   NIL, /* fetch overview */
  82.   NIL, /* fetch message envelopes */
  83.   unix_header, /* fetch message header */
  84.   unix_text, /* fetch message text */
  85.   NIL, /* fetch partial message text */
  86.   NIL, /* unique identifier */
  87.   NIL, /* message number */
  88.   NIL, /* modify flags */
  89.   unix_flagmsg, /* per-message modify flags */
  90.   NIL, /* search for message based on criteria */
  91.   NIL, /* sort messages */
  92.   NIL, /* thread messages */
  93.   unix_ping, /* ping mailbox to see if still alive */
  94.   unix_check, /* check for new messages */
  95.   unix_expunge, /* expunge deleted messages */
  96.   unix_copy, /* copy messages to another mailbox */
  97.   unix_append, /* append string message to mailbox */
  98.   NIL /* garbage collect stream */
  99. };
  100. /* prototype stream */
  101. MAILSTREAM unixproto = {&unixdriver};
  102. /* driver parameters */
  103. static long unix_fromwidget = T;
  104. /* UNIX mail validate mailbox
  105.  * Accepts: mailbox name
  106.  * Returns: our driver if name is valid, NIL otherwise
  107.  */
  108. DRIVER *unix_valid (char *name)
  109. {
  110.   int fd;
  111.   DRIVER *ret = NIL;
  112.   char *t,file[MAILTMPLEN];
  113.   struct stat sbuf;
  114.   time_t tp[2];
  115.   errno = EINVAL; /* assume invalid argument */
  116. /* must be non-empty file */
  117.   if ((t = dummy_file (file,name)) && !stat (t,&sbuf)) {
  118.     if (!sbuf.st_size)errno = 0;/* empty file */
  119.     else if ((fd = open (file,O_RDONLY,NIL)) >= 0) {
  120. /* OK if mailbox format good */
  121.       if (unix_isvalid_fd (fd)) ret = &unixdriver;
  122.       else errno = -1; /* invalid format */
  123.       close (fd); /* close the file */
  124.       tp[0] = sbuf.st_atime; /* preserve atime and mtime */
  125.       tp[1] = sbuf.st_mtime;
  126.       utime (file,tp); /* set the times */
  127.     }
  128.   }
  129. /* in case INBOX but not unix format */
  130.   else if ((errno == ENOENT) && ((name[0] == 'I') || (name[0] == 'i')) &&
  131.    ((name[1] == 'N') || (name[1] == 'n')) &&
  132.    ((name[2] == 'B') || (name[2] == 'b')) &&
  133.    ((name[3] == 'O') || (name[3] == 'o')) &&
  134.    ((name[4] == 'X') || (name[4] == 'x')) && !name[5]) errno = -1;
  135.   return ret; /* return what we should */
  136. }
  137. /* UNIX mail test for valid mailbox
  138.  * Accepts: file descriptor
  139.  *     scratch buffer
  140.  * Returns: T if valid, NIL otherwise
  141.  */
  142. long unix_isvalid_fd (int fd)
  143. {
  144.   int zn;
  145.   int ret = NIL;
  146.   char tmp[MAILTMPLEN],*s,*t,c = 'n';
  147.   memset (tmp,'',MAILTMPLEN);
  148.   if (read (fd,tmp,MAILTMPLEN-1) >= 0) {
  149.     for (s = tmp; (*s == 'r') || (*s == 'n') || (*s == ' ') || (*s == 't');)
  150.       c = *s++;
  151.     if (c == 'n') VALID (s,t,ret,zn);
  152.   }
  153.   return ret; /* return what we should */
  154. }
  155. /* UNIX manipulate driver parameters
  156.  * Accepts: function code
  157.  *     function-dependent value
  158.  * Returns: function-dependent return value
  159.  */
  160. void *unix_parameters (long function,void *value)
  161. {
  162.   switch ((int) function) {
  163.   case SET_FROMWIDGET:
  164.     unix_fromwidget = (long) value;
  165.     break;
  166.   case GET_FROMWIDGET:
  167.     value = (void *) unix_fromwidget;
  168.     break;
  169.   default:
  170.     value = NIL; /* error case */
  171.     break;
  172.   }
  173.   return value;
  174. }
  175. /* UNIX mail scan mailboxes
  176.  * Accepts: mail stream
  177.  *     reference
  178.  *     pattern to search
  179.  *     string to scan
  180.  */
  181. void unix_scan (MAILSTREAM *stream,char *ref,char *pat,char *contents)
  182. {
  183.   if (stream) dummy_scan (NIL,ref,pat,contents);
  184. }
  185. /* UNIX mail list mailboxes
  186.  * Accepts: mail stream
  187.  *     reference
  188.  *     pattern to search
  189.  */
  190. void unix_list (MAILSTREAM *stream,char *ref,char *pat)
  191. {
  192.   if (stream) dummy_list (NIL,ref,pat);
  193. }
  194. /* UNIX mail list subscribed mailboxes
  195.  * Accepts: mail stream
  196.  *     reference
  197.  *     pattern to search
  198.  */
  199. void unix_lsub (MAILSTREAM *stream,char *ref,char *pat)
  200. {
  201.   if (stream) dummy_lsub (NIL,ref,pat);
  202. }
  203. /* UNIX mail create mailbox
  204.  * Accepts: MAIL stream
  205.  *     mailbox name to create
  206.  * Returns: T on success, NIL on failure
  207.  */
  208. long unix_create (MAILSTREAM *stream,char *mailbox)
  209. {
  210.   char *s,mbx[MAILTMPLEN],tmp[MAILTMPLEN];
  211.   long ret = NIL;
  212.   int i,fd;
  213.   time_t ti = time (0);
  214.   if (!(s = dummy_file (mbx,mailbox))) {
  215.     sprintf (tmp,"Can't create %.80s: invalid name",mailbox);
  216.     mm_log (tmp,ERROR);
  217.   }
  218. /* create underlying file */
  219.   else if (dummy_create_path (stream,s)) {
  220. /* done if made directory */
  221.     if ((s = strrchr (s,'/')) && !s[1]) return T;
  222.     if ((fd = open (mbx,O_WRONLY,
  223.     (int) mail_parameters (NIL,GET_MBXPROTECTION,NIL))) < 0) {
  224.       sprintf (tmp,"Can't reopen mailbox node %.80s: %s",mbx,strerror (errno));
  225.       mm_log (tmp,ERROR);
  226.       unlink (mbx); /* delete the file */
  227.     }
  228. /* in case a whiner with no life */
  229.     else if (mail_parameters (NIL,GET_USERHASNOLIFE,NIL)) ret = T;
  230.     else { /* initialize header */
  231.       memset (tmp,'',MAILTMPLEN);
  232.       sprintf (tmp,"From %s %sDate: ",pseudo_from,ctime (&ti));
  233.       rfc822_fixed_date (s = tmp + strlen (tmp));
  234. /* write the pseudo-header */
  235.       sprintf (s += strlen (s),
  236.        "nFrom: %s <%s@%s>nSubject: %snX-IMAP: %010lu 0000000000",
  237.        pseudo_name,pseudo_from,mylocalhost (),pseudo_subject,
  238.        (unsigned long) ti);
  239.       for (i = 0; i < NUSERFLAGS; ++i) if (default_user_flag (i))
  240. sprintf (s += strlen (s)," %s",default_user_flag (i));
  241.       sprintf (s += strlen (s),"nStatus: ROnn%snn",pseudo_msg);
  242.       if ((write (fd,tmp,strlen (tmp)) < 0) || close (fd)) {
  243. sprintf (tmp,"Can't initialize mailbox node %.80s: %s",mbx,
  244.  strerror (errno));
  245. mm_log (tmp,ERROR);
  246. unlink (mbx); /* delete the file */
  247.       }
  248.       else ret = T; /* success */
  249.     }
  250.     close (fd); /* close file, set proper protections */
  251.   }
  252.   return ret ? set_mbx_protections (mailbox,mbx) : NIL;
  253. }
  254. /* UNIX mail delete mailbox
  255.  * Accepts: MAIL stream
  256.  *     mailbox name to delete
  257.  * Returns: T on success, NIL on failure
  258.  */
  259. long unix_delete (MAILSTREAM *stream,char *mailbox)
  260. {
  261.   return unix_rename (stream,mailbox,NIL);
  262. }
  263. /* UNIX mail rename mailbox
  264.  * Accepts: MAIL stream
  265.  *     old mailbox name
  266.  *     new mailbox name (or NIL for delete)
  267.  * Returns: T on success, NIL on failure
  268.  */
  269. long unix_rename (MAILSTREAM *stream,char *old,char *newname)
  270. {
  271.   long ret = NIL;
  272.   char c,*s = NIL;
  273.   char tmp[MAILTMPLEN],file[MAILTMPLEN],lock[MAILTMPLEN];
  274.   DOTLOCK lockx;
  275.   int fd,ld;
  276.   long i;
  277.   struct stat sbuf;
  278.   mm_critical (stream); /* get the c-client lock */
  279.   if (newname && !((s = dummy_file (tmp,newname)) && *s))
  280.     sprintf (tmp,"Can't rename mailbox %.80s to %.80s: invalid name",
  281.      old,newname);
  282. /* lock out other c-clients */
  283.   else if ((ld = lockname (lock,dummy_file (file,old),LOCK_EX|LOCK_NB,&i)) < 0)
  284.     sprintf (tmp,"Mailbox %.80s is in use by another process",old);
  285.   else {
  286.     if ((fd = unix_lock (file,O_RDWR,S_IREAD|S_IWRITE,&lockx,LOCK_EX)) < 0)
  287.       sprintf (tmp,"Can't lock mailbox %.80s: %s",old,strerror (errno));
  288.     else {
  289.       if (newname) { /* want rename? */
  290. /* found superior to destination name? */
  291. if (s = strrchr (s,'/')) {
  292.   c = *++s; /* remember first character of inferior */
  293.   *s = ''; /* tie off to get just superior */
  294. /* name doesn't exist, create it */
  295.   if ((stat (tmp,&sbuf) || ((sbuf.st_mode & S_IFMT) != S_IFDIR)) &&
  296.       !dummy_create (stream,tmp)) {
  297.     unix_unlock (fd,NIL,&lockx);
  298.     unix_unlock (ld,NIL,NIL);
  299.     unlink (lock);
  300.     mm_nocritical (stream);
  301.     return ret; /* return success or failure */
  302.   }
  303.   *s = c; /* restore full name */
  304. }
  305. if (rename (file,tmp))
  306.   sprintf (tmp,"Can't rename mailbox %.80s to %.80s: %s",old,newname,
  307.    strerror (errno));
  308. else ret = T; /* set success */
  309.       }
  310.       else if (unlink (file))
  311. sprintf (tmp,"Can't delete mailbox %.80s: %s",old,strerror (errno));
  312.       else ret = T; /* set success */
  313.       unix_unlock (fd,NIL,&lockx);
  314.     }
  315.     unix_unlock (ld,NIL,NIL); /* flush the lock */
  316.     unlink (lock);
  317.   }
  318.   mm_nocritical (stream); /* no longer critical */
  319.   if (!ret) mm_log (tmp,ERROR); /* log error */
  320.   return ret; /* return success or failure */
  321. }
  322. /* UNIX mail open
  323.  * Accepts: Stream to open
  324.  * Returns: Stream on success, NIL on failure
  325.  */
  326. MAILSTREAM *unix_open (MAILSTREAM *stream)
  327. {
  328.   long i;
  329.   int fd;
  330.   char tmp[MAILTMPLEN];
  331.   DOTLOCK lock;
  332.   long retry;
  333. /* return prototype for OP_PROTOTYPE call */
  334.   if (!stream) return user_flags (&unixproto);
  335.   retry = stream->silent ? 1 : KODRETRY;
  336.   if (stream->local) fatal ("unix recycle stream");
  337.   stream->local = memset (fs_get (sizeof (UNIXLOCAL)),0,sizeof (UNIXLOCAL));
  338. /* note if an INBOX or not */
  339.   stream->inbox = !strcmp (ucase (strcpy (tmp,stream->mailbox)),"INBOX");
  340. /* canonicalize the stream mailbox name */
  341.   dummy_file (tmp,stream->mailbox);
  342. /* flush old name */
  343.   fs_give ((void **) &stream->mailbox);
  344. /* save canonical name */
  345.   stream->mailbox = cpystr (tmp);
  346.   LOCAL->fd = LOCAL->ld = -1; /* no file or state locking yet */
  347.   LOCAL->buf = (char *) fs_get ((LOCAL->buflen = CHUNK) + 1);
  348.   stream->sequence++; /* bump sequence number */
  349. /* make lock for read/write access */
  350.   if (!stream->rdonly) while (retry) {
  351. /* try to lock file */
  352.     if ((fd = lockname (tmp,stream->mailbox,LOCK_EX|LOCK_NB,&i)) < 0) {
  353.       if (retry-- == KODRETRY) {/* no, first time through? */
  354. if (i) { /* learned the other guy's PID? */
  355.   kill ((int) i,SIGUSR2);
  356.   sprintf (tmp,"Trying to get mailbox lock from process %ld",i);
  357.   mm_log (tmp,WARN);
  358. }
  359. else retry = 0; /* give up */
  360.       }
  361.       if (!stream->silent) { /* nothing if silent stream */
  362. if (retry) sleep (1); /* wait a second before trying again */
  363. else mm_log ("Mailbox is open by another process, access is readonly",
  364.      WARN);
  365.       }
  366.     }
  367.     else { /* got the lock, nobody else can alter state */
  368.       LOCAL->ld = fd; /* note lock's fd and name */
  369.       LOCAL->lname = cpystr (tmp);
  370. /* make sure mode OK (don't use fchmod()) */
  371.       chmod (LOCAL->lname,(int) mail_parameters (NIL,GET_LOCKPROTECTION,NIL));
  372.       if (stream->silent) i = 0;/* silent streams won't accept KOD */
  373.       else { /* note our PID in the lock */
  374. sprintf (tmp,"%d",getpid ());
  375. write (fd,tmp,(i = strlen (tmp))+1);
  376.       }
  377.       ftruncate (fd,i); /* make sure tied off */
  378.       fsync (fd); /* make sure it's available */
  379.       retry = 0; /* no more need to try */
  380.     }
  381.   }
  382. /* parse mailbox */
  383.   stream->nmsgs = stream->recent = 0;
  384. /* will we be able to get write access? */
  385.   if ((LOCAL->ld >= 0) && access (stream->mailbox,W_OK) && (errno == EACCES)) {
  386.     mm_log ("Can't get write access to mailbox, access is readonly",WARN);
  387.     flock (LOCAL->ld,LOCK_UN); /* release the lock */
  388.     close (LOCAL->ld); /* close the lock file */
  389.     LOCAL->ld = -1; /* no more lock fd */
  390.     unlink (LOCAL->lname); /* delete it */
  391.   }
  392. /* reset UID validity */
  393.   stream->uid_validity = stream->uid_last = 0;
  394.   if (stream->silent && !stream->rdonly && (LOCAL->ld < 0))
  395.     unix_abort (stream); /* abort if can't get RW silent stream */
  396. /* parse mailbox */
  397.   else if (unix_parse (stream,&lock,LOCK_SH)) {
  398.     unix_unlock (LOCAL->fd,stream,&lock);
  399.     mail_unlock (stream);
  400.     mm_nocritical (stream); /* done with critical */
  401.   }
  402.   if (!LOCAL) return NIL; /* failure if stream died */
  403. /* make sure upper level knows readonly */
  404.   stream->rdonly = (LOCAL->ld < 0);
  405. /* notify about empty mailbox */
  406.   if (!(stream->nmsgs || stream->silent)) mm_log ("Mailbox is empty",NIL);
  407.   if (!stream->rdonly) { /* flags stick if readwrite */
  408.     stream->perm_seen = stream->perm_deleted =
  409.       stream->perm_flagged = stream->perm_answered = stream->perm_draft = T;
  410.     if (!stream->uid_nosticky) {/* users with lives get permanent keywords */
  411.       stream->perm_user_flags = 0xffffffff;
  412. /* and maybe can create them too! */
  413.       stream->kwd_create = stream->user_flags[NUSERFLAGS-1] ? NIL : T;
  414.     }
  415.   }
  416.   return stream; /* return stream alive to caller */
  417. }
  418. /* UNIX mail close
  419.  * Accepts: MAIL stream
  420.  *     close options
  421.  */
  422. void unix_close (MAILSTREAM *stream,long options)
  423. {
  424.   int silent = stream->silent;
  425.   stream->silent = T; /* go silent */
  426. /* expunge if requested */
  427.   if (options & CL_EXPUNGE) unix_expunge (stream);
  428. /* else dump final checkpoint */
  429.   else if (LOCAL->dirty) unix_check (stream);
  430.   stream->silent = silent; /* restore old silence state */
  431.   unix_abort (stream); /* now punt the file and local data */
  432. }
  433. /* UNIX mail fetch message header
  434.  * Accepts: MAIL stream
  435.  *     message # to fetch
  436.  *     pointer to returned header text length
  437.  *     option flags
  438.  * Returns: message header in RFC822 format
  439.  */
  440. /* lines to filter from header */
  441. static STRINGLIST *unix_hlines = NIL;
  442. char *unix_header (MAILSTREAM *stream,unsigned long msgno,
  443.    unsigned long *length,long flags)
  444. {
  445.   MESSAGECACHE *elt;
  446.   char *s;
  447.   *length = 0; /* default to empty */
  448.   if (flags & FT_UID) return "";/* UID call "impossible" */
  449.   elt = mail_elt (stream,msgno);/* get cache */
  450.   if (!unix_hlines) { /* once only code */
  451.     STRINGLIST *lines = unix_hlines = mail_newstringlist ();
  452.     lines->text.size = strlen ((char *) (lines->text.data =
  453.  (unsigned char *) "Status"));
  454.     lines = lines->next = mail_newstringlist ();
  455.     lines->text.size = strlen ((char *) (lines->text.data =
  456.  (unsigned char *) "X-Status"));
  457.     lines = lines->next = mail_newstringlist ();
  458.     lines->text.size = strlen ((char *) (lines->text.data =
  459.  (unsigned char *) "X-Keywords"));
  460.     lines = lines->next = mail_newstringlist ();
  461.     lines->text.size = strlen ((char *) (lines->text.data =
  462.  (unsigned char *) "X-UID"));
  463.   }
  464. /* go to header position */
  465.   lseek (LOCAL->fd,elt->private.special.offset +
  466.  elt->private.msg.header.offset,L_SET);
  467.   if (flags & FT_INTERNAL) { /* initial data OK? */
  468.     if (elt->private.msg.header.text.size > LOCAL->buflen) {
  469.       fs_give ((void **) &LOCAL->buf);
  470.       LOCAL->buf = (char *) fs_get ((LOCAL->buflen =
  471.      elt->private.msg.header.text.size) + 1);
  472.     }
  473. /* read message */
  474.     read (LOCAL->fd,LOCAL->buf,elt->private.msg.header.text.size);
  475. /* got text, tie off string */
  476.     LOCAL->buf[*length = elt->private.msg.header.text.size] = '';
  477. /* squeeze out CRs (in case from PC) */
  478.     if (s = strchr (LOCAL->buf,'r')) {
  479.       char *t,*tl;
  480.       for (t = s,tl = LOCAL->buf + *length; t <= tl; t++)
  481. if ((*t != 'r') || (t[1] != 'n')) *s++ = *t;
  482. /* adjust length */
  483.       LOCAL->buf[*length = s - LOCAL->buf - 1] = '';
  484.     }
  485.   }
  486.   else { /* need to make a CRLF version */
  487.     read (LOCAL->fd,s = (char *) fs_get (elt->private.msg.header.text.size+1),
  488.   elt->private.msg.header.text.size);
  489. /* tie off string, and convert to CRLF */
  490.     s[elt->private.msg.header.text.size] = '';
  491.     *length = strcrlfcpy (&LOCAL->buf,&LOCAL->buflen,s,
  492.   elt->private.msg.header.text.size);
  493.     fs_give ((void **) &s); /* free readin buffer */
  494.   }
  495.   *length = mail_filter (LOCAL->buf,*length,unix_hlines,FT_NOT);
  496.   return LOCAL->buf; /* return processed copy */
  497. }
  498. /* UNIX mail fetch message text
  499.  * Accepts: MAIL stream
  500.  *     message # to fetch
  501.  *     pointer to returned stringstruct
  502.  *     option flags
  503.  * Returns: T on success, NIL if failure
  504.  */
  505. long unix_text (MAILSTREAM *stream,unsigned long msgno,STRING *bs,long flags)
  506. {
  507.   char *s;
  508.   unsigned long i;
  509.   MESSAGECACHE *elt;
  510. /* UID call "impossible" */
  511.   if (flags & FT_UID) return NIL;
  512.   elt = mail_elt (stream,msgno);/* get cache element */
  513. /* if message not seen */
  514.   if (!(flags & FT_PEEK) && !elt->seen) {
  515. /* mark message seen and dirty */
  516.     elt->seen = elt->private.dirty = LOCAL->dirty = T;
  517.     mm_flags (stream,msgno);
  518.   }
  519.   s = unix_text_work (stream,elt,&i,flags);
  520.   INIT (bs,mail_string,s,i); /* set up stringstruct */
  521.   return T; /* success */
  522. }
  523. /* UNIX mail fetch message text worker routine
  524.  * Accepts: MAIL stream
  525.  *     message cache element
  526.  *     pointer to returned header text length
  527.  *     option flags
  528.  */
  529. char *unix_text_work (MAILSTREAM *stream,MESSAGECACHE *elt,
  530.       unsigned long *length,long flags)
  531. {
  532.   FDDATA d;
  533.   STRING bs;
  534.   char *s,tmp[CHUNK];
  535. /* go to text position */
  536.   lseek (LOCAL->fd,elt->private.special.offset +
  537.  elt->private.msg.text.offset,L_SET);
  538.   if (flags & FT_INTERNAL) { /* initial data OK? */
  539.     if (elt->private.msg.text.text.size > LOCAL->buflen) {
  540.       fs_give ((void **) &LOCAL->buf);
  541.       LOCAL->buf = (char *) fs_get ((LOCAL->buflen =
  542.      elt->private.msg.text.text.size) + 1);
  543.     }
  544. /* read message */
  545.     read (LOCAL->fd,LOCAL->buf,elt->private.msg.text.text.size);
  546. /* got text, tie off string */
  547.     LOCAL->buf[*length = elt->private.msg.text.text.size] = '';
  548. /* squeeze out CRs (in case from PC) */
  549.     if (s = strchr (LOCAL->buf,'r')) {
  550.       char *t,*tl;
  551.       for (t = s,tl = LOCAL->buf + *length; t <= tl; t++)
  552. if ((*t != 'r') || (t[1] != 'n')) *s++ = *t;
  553. /* adjust length */
  554.       *length = s - LOCAL->buf - 1;
  555.     }
  556.   }
  557.   else { /* need to make a CRLF version */
  558.     if (elt->rfc822_size > LOCAL->buflen) {
  559.       /* excessively conservative, but the right thing is too hard to do */
  560.       fs_give ((void **) &LOCAL->buf);
  561.       LOCAL->buf = (char *) fs_get ((LOCAL->buflen = elt->rfc822_size) + 1);
  562.     }
  563.     d.fd = LOCAL->fd; /* yes, set up file descriptor */
  564.     d.pos = elt->private.special.offset + elt->private.msg.text.offset;
  565.     d.chunk = tmp; /* initial buffer chunk */
  566.     d.chunksize = CHUNK; /* file chunk size */
  567.     INIT (&bs,fd_string,&d,elt->private.msg.text.text.size);
  568.     for (s = LOCAL->buf; SIZE (&bs);) switch (CHR (&bs)) {
  569.     case 'r': /* carriage return seen */
  570.       *s++ = SNX (&bs); /* copy it and any succeeding LF */
  571.       if (SIZE (&bs) && (CHR (&bs) == 'n')) *s++ = SNX (&bs);
  572.       break;
  573.     case 'n':
  574.       *s++ = 'r'; /* insert a CR */
  575.     default:
  576.       *s++ = SNX (&bs); /* copy characters */
  577.     }
  578.     *s = ''; /* tie off buffer */
  579.     *length = s - LOCAL->buf; /* calculate length */
  580.   }
  581.   return LOCAL->buf;
  582. }
  583. /* UNIX per-message modify flag
  584.  * Accepts: MAIL stream
  585.  *     message cache element
  586.  */
  587. void unix_flagmsg (MAILSTREAM *stream,MESSAGECACHE *elt)
  588. {
  589. /* only after finishing */
  590.   if (elt->valid) elt->private.dirty = LOCAL->dirty = T;
  591. }
  592. /* UNIX mail ping mailbox
  593.  * Accepts: MAIL stream
  594.  * Returns: T if stream alive, else NIL
  595.  */
  596. long unix_ping (MAILSTREAM *stream)
  597. {
  598.   DOTLOCK lock;
  599.   struct stat sbuf;
  600. /* big no-op if not readwrite */
  601.   if (LOCAL && (LOCAL->ld >= 0) && !stream->lock) {
  602.     if (stream->rdonly) { /* does he want to give up readwrite? */
  603. /* checkpoint if we changed something */
  604.       if (LOCAL->dirty) unix_check (stream);
  605.       flock (LOCAL->ld,LOCK_UN);/* release readwrite lock */
  606.       close (LOCAL->ld); /* close the readwrite lock file */
  607.       LOCAL->ld = -1; /* no more readwrite lock fd */
  608.       unlink (LOCAL->lname); /* delete the readwrite lock file */
  609.     }
  610.     else { /* get current mailbox size */
  611.       if (LOCAL->fd >= 0) fstat (LOCAL->fd,&sbuf);
  612.       else stat (stream->mailbox,&sbuf);
  613. /* parse if mailbox changed */
  614.       if ((sbuf.st_size != LOCAL->filesize) &&
  615.   unix_parse (stream,&lock,LOCK_SH)) {
  616. /* unlock mailbox */
  617. unix_unlock (LOCAL->fd,stream,&lock);
  618. mail_unlock (stream); /* and stream */
  619. mm_nocritical (stream); /* done with critical */
  620.       }
  621.     }
  622.   }
  623.   return LOCAL ? LONGT : NIL; /* return if still alive */
  624. }
  625. /* UNIX mail check mailbox
  626.  * Accepts: MAIL stream
  627.  */
  628. void unix_check (MAILSTREAM *stream)
  629. {
  630.   DOTLOCK lock;
  631. /* parse and lock mailbox */
  632.   if (LOCAL && (LOCAL->ld >= 0) && !stream->lock &&
  633.       unix_parse (stream,&lock,LOCK_EX)) {
  634. /* any unsaved changes? */
  635.     if (LOCAL->dirty && unix_rewrite (stream,NIL,&lock)) {
  636.       if (!stream->silent) mm_log ("Checkpoint completed",NIL);
  637.     }
  638. /* no checkpoint needed, just unlock */
  639.     else unix_unlock (LOCAL->fd,stream,&lock);
  640.     mail_unlock (stream); /* unlock the stream */
  641.     mm_nocritical (stream); /* done with critical */
  642.   }
  643. }
  644. /* UNIX mail expunge mailbox
  645.  * Accepts: MAIL stream
  646.  */
  647. void unix_expunge (MAILSTREAM *stream)
  648. {
  649.   unsigned long i;
  650.   DOTLOCK lock;
  651.   char *msg = NIL;
  652. /* parse and lock mailbox */
  653.   if (LOCAL && (LOCAL->ld >= 0) && !stream->lock &&
  654.       unix_parse (stream,&lock,LOCK_EX)) {
  655. /* count expunged messages if not dirty */
  656.     if (!LOCAL->dirty) for (i = 1; i <= stream->nmsgs; i++)
  657.       if (mail_elt (stream,i)->deleted) LOCAL->dirty = T;
  658.     if (!LOCAL->dirty) { /* not dirty and no expunged messages */
  659.       unix_unlock (LOCAL->fd,stream,&lock);
  660.       msg = "No messages deleted, so no update needed";
  661.     }
  662.     else if (unix_rewrite (stream,&i,&lock)) {
  663.       if (i) sprintf (msg = LOCAL->buf,"Expunged %lu messages",i);
  664.       else msg = "Mailbox checkpointed, but no messages expunged";
  665.     }
  666. /* rewrite failed */
  667.     else unix_unlock (LOCAL->fd,stream,&lock);
  668.     mail_unlock (stream); /* unlock the stream */
  669.     mm_nocritical (stream); /* done with critical */
  670.     if (msg && !stream->silent) mm_log (msg,NIL);
  671.   }
  672.   else if (!stream->silent) mm_log("Expunge ignored on readonly mailbox",WARN);
  673. }
  674. /* UNIX mail copy message(s)
  675.  * Accepts: MAIL stream
  676.  *     sequence
  677.  *     destination mailbox
  678.  *     copy options
  679.  * Returns: T if copy successful, else NIL
  680.  */
  681. long unix_copy (MAILSTREAM *stream,char *sequence,char *mailbox,long options)
  682. {
  683.   struct stat sbuf;
  684.   int fd;
  685.   char *s,file[MAILTMPLEN];
  686.   DOTLOCK lock;
  687.   time_t tp[2];
  688.   unsigned long i,j;
  689.   MESSAGECACHE *elt;
  690.   long ret = T;
  691.   mailproxycopy_t pc =
  692.     (mailproxycopy_t) mail_parameters (stream,GET_MAILPROXYCOPY,NIL);
  693.   if (!((options & CP_UID) ? mail_uid_sequence (stream,sequence) :
  694. mail_sequence (stream,sequence))) return NIL;
  695. /* make sure valid mailbox */
  696.   if (!unix_valid (mailbox)) switch (errno) {
  697.   case ENOENT: /* no such file? */
  698.     mm_notify (stream,"[TRYCREATE] Must create mailbox before copy",NIL);
  699.     return NIL;
  700.   case 0: /* merely empty file? */
  701.     break;
  702.   case EINVAL:
  703.     if (pc) return (*pc) (stream,sequence,mailbox,options);
  704.     sprintf (LOCAL->buf,"Invalid UNIX-format mailbox name: %.80s",mailbox);
  705.     mm_log (LOCAL->buf,ERROR);
  706.     return NIL;
  707.   default:
  708.     if (pc) return (*pc) (stream,sequence,mailbox,options);
  709.     sprintf (LOCAL->buf,"Not a UNIX-format mailbox: %.80s",mailbox);
  710.     mm_log (LOCAL->buf,ERROR);
  711.     return NIL;
  712.   }
  713.   LOCAL->buf[0] = '';
  714.   mm_critical (stream); /* go critical */
  715.   if ((fd = unix_lock (dummy_file (file,mailbox),O_WRONLY|O_APPEND|O_CREAT,
  716.        S_IREAD|S_IWRITE,&lock,LOCK_EX)) < 0) {
  717.     mm_nocritical (stream); /* done with critical */
  718.     sprintf (LOCAL->buf,"Can't open destination mailbox: %s",strerror (errno));
  719.     mm_log (LOCAL->buf,ERROR); /* log the error */
  720.     return NIL; /* failed */
  721.   }
  722.   fstat (fd,&sbuf); /* get current file size */
  723. /* write all requested messages to mailbox */
  724.   for (i = 1; ret && (i <= stream->nmsgs); i++)
  725.     if ((elt = mail_elt (stream,i))->sequence) {
  726.       lseek (LOCAL->fd,elt->private.special.offset,L_SET);
  727.       read (LOCAL->fd,LOCAL->buf,elt->private.special.text.size);
  728.       if (write (fd,LOCAL->buf,elt->private.special.text.size) < 0) ret = NIL;
  729.       else { /* internal header succeeded */
  730. s = unix_header (stream,i,&j,FT_INTERNAL);
  731. /* header size, sans trailing newline */
  732. if (j && (s[j - 2] == 'n')) j--;
  733. if (write (fd,s,j) < 0) ret = NIL;
  734. else { /* message header succeeded */
  735.   j = unix_xstatus (stream,LOCAL->buf,elt,NIL);
  736.   if (write (fd,LOCAL->buf,j) < 0) ret = NIL;
  737.   else { /* message status succeeded */
  738.     s = unix_text_work (stream,elt,&j,FT_INTERNAL);
  739.     if ((write (fd,s,j) < 0) || (write (fd,"n",1) < 0)) ret = NIL;
  740.   }
  741. }
  742.       }
  743.     }
  744.   if (!ret || fsync (fd)) { /* force out the update */
  745.     sprintf (LOCAL->buf,"Message copy failed: %s",strerror (errno));
  746.     ftruncate (fd,sbuf.st_size);
  747.     ret = NIL;
  748.   }
  749.   tp[0] = sbuf.st_atime; /* preserve atime */
  750.   tp[1] = time (0); /* set mtime to now */
  751.   utime (file,tp); /* set the times */
  752.   unix_unlock (fd,NIL,&lock); /* unlock and close mailbox */
  753.   mm_nocritical (stream); /* release critical */
  754. /* log the error */
  755.   if (!ret) mm_log (LOCAL->buf,ERROR);
  756. /* delete if requested message */
  757.   else if (options & CP_MOVE) for (i = 1; i <= stream->nmsgs; i++)
  758.     if ((elt = mail_elt (stream,i))->sequence)
  759.       elt->deleted = elt->private.dirty = LOCAL->dirty = T;
  760.   return ret;
  761. }
  762. /* UNIX mail append message from stringstruct
  763.  * Accepts: MAIL stream
  764.  *     destination mailbox
  765.  *     initial flags
  766.  *     internal date
  767.  *     stringstruct of messages to append
  768.  * Returns: T if append successful, else NIL
  769.  */
  770. #define BUFLEN 8*MAILTMPLEN
  771. long unix_append (MAILSTREAM *stream,char *mailbox,char *flags,char *date,
  772.   STRING *message)
  773. {
  774.   MESSAGECACHE elt;
  775.   struct stat sbuf;
  776.   int fd,ti,zn;
  777.   long f,i,ok;
  778.   unsigned long j,n,uf,size;
  779.   char c,*x,buf[BUFLEN],tmp[MAILTMPLEN],file[MAILTMPLEN];
  780.   DOTLOCK lock;
  781.   time_t tp[2];
  782. /* default stream to prototype */
  783.   if (!stream) stream = user_flags (&unixproto);
  784. /* get flags */
  785.   f = mail_parse_flags (stream,flags,&uf);
  786. /* parse date */
  787.   if (!date) rfc822_date (date = tmp);
  788.   if (!mail_parse_date (&elt,date)) {
  789.     sprintf (buf,"Bad date in append: %.80s",date);
  790.     mm_log (buf,ERROR);
  791.     return NIL;
  792.   }
  793. /* make sure valid mailbox */
  794.   if (!unix_valid (mailbox)) switch (errno) {
  795.   case ENOENT: /* no such file? */
  796.     if (((mailbox[0] == 'I') || (mailbox[0] == 'i')) &&
  797. ((mailbox[1] == 'N') || (mailbox[1] == 'n')) &&
  798. ((mailbox[2] == 'B') || (mailbox[2] == 'b')) &&
  799. ((mailbox[3] == 'O') || (mailbox[3] == 'o')) &&
  800. ((mailbox[4] == 'X') || (mailbox[4] == 'x')) && !mailbox[5])
  801.       unix_create (NIL,"INBOX");
  802.     else {
  803.       mm_notify (stream,"[TRYCREATE] Must create mailbox before append",NIL);
  804.       return NIL;
  805.     }
  806. /* falls through */
  807.   case 0: /* INBOX ENOENT or empty file? */
  808.     break;
  809.   case EINVAL:
  810.     sprintf (buf,"Invalid UNIX-format mailbox name: %.80s",mailbox);
  811.     mm_log (buf,ERROR);
  812.     return NIL;
  813.   default:
  814.     sprintf (buf,"Not a UNIX-format mailbox: %.80s",mailbox);
  815.     mm_log (buf,ERROR);
  816.     return NIL;
  817.   }
  818.   mm_critical (stream); /* go critical */
  819.   if ((fd = unix_lock (dummy_file (file,mailbox),O_WRONLY|O_APPEND|O_CREAT,
  820.        S_IREAD|S_IWRITE,&lock,LOCK_EX)) < 0) {
  821.     mm_nocritical (stream); /* done with critical */
  822.     sprintf (buf,"Can't open append mailbox: %s",strerror (errno));
  823.     mm_log (buf,ERROR);
  824.     return NIL;
  825.   }
  826.   fstat (fd,&sbuf); /* get current file size */
  827.   sprintf (buf,"From %s@%s ",myusername (),mylocalhost ());
  828. /* user wants to suppress time zones? */
  829.   if (mail_parameters (NIL,GET_NOTIMEZONES,NIL)) {
  830.     time_t when = mail_longdate (&elt);
  831.     strcat (buf,ctime (&when));
  832.   }
  833. /* write the date given */
  834.   else mail_cdate (buf + strlen (buf),&elt);
  835.   sprintf (buf + strlen (buf),"Status: %snX-Status: %s%s%s%snX-Keywords:",
  836.    f&fSEEN ? "R" : "",f&fDELETED ? "D" : "",f&fFLAGGED ? "F" : "",
  837.    f&fANSWERED ? "A" : "",f&fDRAFT ? "T" : "");
  838.   while (uf) /* write user flags */
  839.     sprintf(buf+strlen(buf)," %s",stream->user_flags[find_rightmost_bit(&uf)]);
  840.   strcat (buf,"n"); /* tie off flags */
  841. /* copy text, tossing out CRs */
  842.   for (i = strlen (buf), ok = T, size = SIZE (message); ok && size; ) {
  843.     for (j = 0, c = ''; size && (c != 'n') && (j < MAILTMPLEN); size--)
  844.       if (((c = SNX (message)) != 'r') || !(SIZE (message)) ||
  845.   (CHR (message) != 'n')) tmp[j++] = c;
  846. /* possible "From " line? */
  847.     if ((j > 4) && (tmp[0] == 'F') && (tmp[1] == 'r') && (tmp[2] == 'o') &&
  848. (tmp[3] == 'm') && (tmp[4] == ' ')) {
  849. /* see if need to write a widget */
  850.       if (!(ti = unix_fromwidget || (c != 'n'))) VALID (tmp,x,ti,zn);
  851.       if (ti) ok = unix_append_putc (fd,buf,&i,'>');
  852.     }
  853. /* write the line */
  854.     for (n = 0; ok && (n < j); n++) ok = unix_append_putc (fd,buf,&i,tmp[n]);
  855. /* handle pathologically long line */
  856.     if (ok && (c != 'n') && size) do {
  857.       if (((c = SNX (message)) != 'r') || !(SIZE (message)) ||
  858.   (CHR (message) != 'n')) ok = unix_append_putc (fd,buf,&i,c);
  859.     } while (ok && --size && (c != 'n'));
  860.   }
  861. /* write trailing newline */
  862.   if (!(ok && (ok = unix_append_putc (fd,buf,&i,'n') &&
  863.        (i ? (write (fd,buf,i) >= 0) : T) && !fsync (fd)))) {
  864.     sprintf (buf,"Message append failed: %s",strerror (errno));
  865.     mm_log (buf,ERROR);
  866.     ftruncate (fd,sbuf.st_size);
  867.   }
  868.   tp[0] = sbuf.st_atime; /* preserve atime */
  869.   tp[1] = time (0); /* set mtime to now */
  870.   utime (file,tp); /* set the times */
  871.   unix_unlock (fd,NIL,&lock); /* unlock and close mailbox */
  872.   mm_nocritical (stream); /* release critical */
  873.   return ok; /* return success */
  874. }
  875. /* UNIX mail append character
  876.  * Accepts: file descriptor
  877.  *     output buffer
  878.  *     pointer to current size of output buffer
  879.  *     character to append
  880.  * Returns: T if append successful, else NIL
  881.  */
  882. long unix_append_putc (int fd,char *s,long *i,char c)
  883. {
  884.   s[(*i)++] = c;
  885.   if (*i == BUFLEN) { /* dump if buffer filled */
  886.     if (write (fd,s,*i) < 0) return NIL;
  887.     *i = 0; /* reset */
  888.   }
  889.   return T;
  890. }
  891. /* Internal routines */
  892. /* UNIX mail abort stream
  893.  * Accepts: MAIL stream
  894.  */
  895. void unix_abort (MAILSTREAM *stream)
  896. {
  897.   if (LOCAL) { /* only if a file is open */
  898.     if (LOCAL->fd >= 0) close (LOCAL->fd);
  899.     if (LOCAL->ld >= 0) { /* have a mailbox lock? */
  900.       flock (LOCAL->ld,LOCK_UN);/* yes, release the lock */
  901.       close (LOCAL->ld); /* close the lock file */
  902.       unlink (LOCAL->lname); /* and delete it */
  903.     }
  904.     if (LOCAL->lname) fs_give ((void **) &LOCAL->lname);
  905. /* free local text buffers */
  906.     if (LOCAL->buf) fs_give ((void **) &LOCAL->buf);
  907.     if (LOCAL->line) fs_give ((void **) &LOCAL->line);
  908. /* nuke the local data */
  909.     fs_give ((void **) &stream->local);
  910.     stream->dtb = NIL; /* log out the DTB */
  911.   }
  912. }
  913. /* UNIX open and lock mailbox
  914.  * Accepts: file name to open/lock
  915.  *     file open mode
  916.  *     destination buffer for lock file name
  917.  *     type of locking operation (LOCK_SH or LOCK_EX)
  918.  */
  919. int unix_lock (char *file,int flags,int mode,DOTLOCK *lock,int op)
  920. {
  921.   int fd;
  922.   blocknotify_t bn = (blocknotify_t) mail_parameters (NIL,GET_BLOCKNOTIFY,NIL);
  923.   (*bn) (BLOCK_FILELOCK,NIL);
  924. /* open file */
  925.   if ((fd = open (file,flags,mode)) >= 0) {
  926.     flock (fd,op); /* lock the file */
  927.     dotlock_lock (file,lock,fd);/* make a dot lock file */
  928.   }
  929.   (*bn) (BLOCK_NONE,NIL);
  930.   return fd;
  931. }
  932. /* UNIX unlock and close mailbox
  933.  * Accepts: file descriptor
  934.  *     (optional) mailbox stream to check atime/mtime
  935.  *     (optional) lock file name
  936.  */
  937. void unix_unlock (int fd,MAILSTREAM *stream,DOTLOCK *lock)
  938. {
  939.   struct stat sbuf;
  940.   time_t tp[2];
  941.   fstat (fd,&sbuf); /* get file times */
  942. /* if stream and csh would think new mail */
  943.   if (stream && (sbuf.st_atime <= sbuf.st_mtime)) {
  944.     tp[0] = time (0); /* set atime to now */
  945. /* set mtime to (now - 1) if necessary */
  946.     tp[1] = tp[0] > sbuf.st_mtime ? sbuf.st_mtime : tp[0] - 1;
  947. /* set the times, note change */
  948.     if (!utime (stream->mailbox,tp)) LOCAL->filetime = tp[1];
  949.   }
  950.   flock (fd,LOCK_UN); /* release flock'ers */
  951.   if (!stream) close (fd); /* close the file if no stream */
  952.   dotlock_unlock (lock); /* flush the lock file if any */
  953. }
  954. /* UNIX mail parse and lock mailbox
  955.  * Accepts: MAIL stream
  956.  *     space to write lock file name
  957.  *     type of locking operation
  958.  * Returns: T if parse OK, critical & mailbox is locked shared; NIL if failure
  959.  */
  960. int unix_parse (MAILSTREAM *stream,DOTLOCK *lock,int op)
  961. {
  962.   int zn;
  963.   unsigned long i,j,k,m;
  964.   char c,*s,*t,*u,tmp[MAILTMPLEN],date[30];
  965.   int ti = 0,pseudoseen = NIL;
  966.   unsigned long nmsgs = stream->nmsgs;
  967.   unsigned long prevuid = nmsgs ? mail_elt (stream,nmsgs)->private.uid : 0;
  968.   unsigned long recent = stream->recent;
  969.   unsigned long oldnmsgs = stream->nmsgs;
  970.   short silent = stream->silent;
  971.   struct stat sbuf;
  972.   STRING bs;
  973.   FDDATA d;
  974.   MESSAGECACHE *elt;
  975.   mail_lock (stream); /* guard against recursion or pingers */
  976. /* toss out previous descriptor */
  977.   if (LOCAL->fd >= 0) close (LOCAL->fd);
  978.   mm_critical (stream); /* open and lock mailbox (shared OK) */
  979.   if ((LOCAL->fd = unix_lock (stream->mailbox,(LOCAL->ld >= 0) ?
  980.       O_RDWR : O_RDONLY,NIL,lock,op)) < 0) {
  981.     sprintf (tmp,"Mailbox open failed, aborted: %s",strerror (errno));
  982.     mm_log (tmp,ERROR);
  983.     unix_abort (stream);
  984.     mail_unlock (stream);
  985.     mm_nocritical (stream); /* done with critical */
  986.     return NIL;
  987.   }
  988.   fstat (LOCAL->fd,&sbuf); /* get status */
  989. /* validate change in size */
  990.   if (sbuf.st_size < LOCAL->filesize) {
  991.     sprintf (tmp,"Mailbox shrank from %lu to %lu bytes, aborted",
  992.      (unsigned long) LOCAL->filesize,(unsigned long) sbuf.st_size);
  993.     mm_log (tmp,ERROR); /* this is pretty bad */
  994.     unix_unlock (LOCAL->fd,stream,lock);
  995.     unix_abort (stream);
  996.     mail_unlock (stream);
  997.     mm_nocritical (stream); /* done with critical */
  998.     return NIL;
  999.   }
  1000. /* new data? */
  1001.   else if (i = sbuf.st_size - LOCAL->filesize) {
  1002.     d.fd = LOCAL->fd; /* yes, set up file descriptor */
  1003.     d.pos = LOCAL->filesize; /* get to that position in the file */
  1004.     d.chunk = LOCAL->buf; /* initial buffer chunk */
  1005.     d.chunksize = CHUNK; /* file chunk size */
  1006.     INIT (&bs,fd_string,&d,i); /* initialize stringstruct */
  1007. /* skip leading whitespace for broken MTAs */
  1008.     while (((c = CHR (&bs)) == 'n') || (c == 'r') ||
  1009.    (c == ' ') || (c == 't')) SNX (&bs);
  1010.     if (SIZE (&bs)) { /* read new data */
  1011. /* remember internal header position */
  1012.       j = LOCAL->filesize + GETPOS (&bs);
  1013.       s = unix_mbxline (stream,&bs,&i);
  1014.       t = NIL,zn = 0;
  1015.       if (i) VALID (s,t,ti,zn); /* see if valid From line */
  1016.       if (!ti) { /* someone pulled the rug from under us */
  1017. sprintf(tmp,"Unexpected changes to mailbox (try restarting): %.20s",s);
  1018. mm_log (tmp,ERROR);
  1019. unix_unlock (LOCAL->fd,stream,lock);
  1020. unix_abort (stream);
  1021. mail_unlock (stream);
  1022. mm_nocritical (stream); /* done with critical */
  1023. return NIL;
  1024.       }
  1025.       stream->silent = T; /* quell main program new message events */
  1026.       do { /* found a message */
  1027. /* instantiate first new message */
  1028. mail_exists (stream,++nmsgs);
  1029. (elt = mail_elt (stream,nmsgs))->valid = T;
  1030. recent++; /* assume recent by default */
  1031. elt->recent = T;
  1032. /* note position/size of internal header */
  1033. elt->private.special.offset = j;
  1034. elt->private.msg.header.offset = elt->private.special.text.size = i;
  1035. /* generate plausible IMAPish date string */
  1036. date[2] = date[6] = date[20] = '-'; date[11] = ' ';
  1037. date[14] = date[17] = ':';
  1038. /* dd */
  1039. date[0] = t[ti - 2]; date[1] = t[ti - 1];
  1040. /* mmm */
  1041. date[3] = t[ti - 6]; date[4] = t[ti - 5]; date[5] = t[ti - 4];
  1042. /* hh */
  1043. date[12] = t[ti + 1]; date[13] = t[ti + 2];
  1044. /* mm */
  1045. date[15] = t[ti + 4]; date[16] = t[ti + 5];
  1046. if (t[ti += 6] == ':') {/* ss */
  1047.   date[18] = t[++ti]; date[19] = t[++ti];
  1048.   ti++; /* move to space */
  1049. }
  1050. else date[18] = date[19] = '0';
  1051. /* yy -- advance over timezone if necessary */
  1052. if (zn == ti) ti += (((t[zn+1] == '+') || (t[zn+1] == '-')) ? 6 : 4);
  1053. date[7] = t[ti + 1]; date[8] = t[ti + 2];
  1054. date[9] = t[ti + 3]; date[10] = t[ti + 4];
  1055. /* zzz */
  1056. t = zn ? (t + zn + 1) : "LCL";
  1057. date[21] = *t++; date[22] = *t++; date[23] = *t++;
  1058. if ((date[21] != '+') && (date[21] != '-')) date[24] = '';
  1059. else { /* numeric time zone */
  1060.   date[24] = *t++; date[25] = *t++;
  1061.   date[26] = ''; date[20] = ' ';
  1062. }
  1063. /* set internal date */
  1064. if (!mail_parse_date (elt,date)) {
  1065.   sprintf (tmp,"Unable to parse internal date: %s",date);
  1066.   mm_log (tmp,WARN);
  1067. }
  1068. do { /* look for message body */
  1069.   s = t = unix_mbxline (stream,&bs,&i);
  1070.   if (i) switch (*s) { /* check header lines */
  1071.   case 'X': /* possible X-???: line */
  1072.     if (s[1] == '-') { /* must be immediately followed by hyphen */
  1073. /* X-Status: becomes Status: in S case */
  1074.       if (s[2] == 'S' && s[3] == 't' && s[4] == 'a' && s[5] == 't' &&
  1075.   s[6] == 'u' && s[7] == 's' && s[8] == ':') s += 2;
  1076. /* possible X-Keywords */
  1077.       else if (s[2] == 'K' && s[3] == 'e' && s[4] == 'y' &&
  1078.        s[5] == 'w' && s[6] == 'o' && s[7] == 'r' &&
  1079.        s[8] == 'd' && s[9] == 's' && s[10] == ':') {
  1080. char uf[MAILTMPLEN];
  1081. s += 11; /* flush leading whitespace */
  1082. while (*s && (*s != 'n') && (*s != 'r')) {
  1083.   while (*s == ' ') s++;
  1084. /* find end of keyword */
  1085.   if (!(u = strpbrk (s," nr"))) u = s + strlen (s);
  1086. /* got a keyword? */
  1087.   if ((k = (u - s)) && (k < MAILTMPLEN)) {
  1088. /* copy keyword */
  1089.     strncpy (uf,s,k);
  1090.     uf[k] = ''; /* make sure tied off */
  1091.     ucase (uf); /* coerce upper case */
  1092.     for (j = 0; (j<NUSERFLAGS) && stream->user_flags[j]; ++j)
  1093.       if (!strcmp(uf,
  1094.   ucase (strcpy(tmp,stream->user_flags[j])))) {
  1095. elt->user_flags |= ((long) 1) << j;
  1096. break;
  1097.       }
  1098. /* need to create it? */
  1099.     if (!stream->rdonly && (j < NUSERFLAGS) &&
  1100. !stream->user_flags[j]) {
  1101. /* set the bit */
  1102.       *uf |= 1 << j;
  1103.       stream->user_flags[j] = (char *) fs_get (k + 1);
  1104.       strncpy (stream->user_flags[j],s,k);
  1105.       stream->user_flags[j][k] = '';
  1106. /* if now out of user flags */
  1107.       if (j == NUSERFLAGS - 1) stream->kwd_create = NIL;
  1108.     }
  1109.   }
  1110.   s = u; /* advance to next keyword */
  1111. }
  1112. break;
  1113.       }
  1114. /* possible X-IMAP */
  1115.       else if ((nmsgs == 1) && !stream->uid_validity &&
  1116.        (s[2] == 'I') && (s[3] == 'M') && (s[4] == 'A') &&
  1117.        (s[5] == 'P') && (s[6] == ':')) {
  1118. s += 7; /* advance to data */
  1119. /* flush whitespace */
  1120. while (*s == ' ') s++;
  1121. j = 0; /* slurp UID validity */
  1122. /* found a digit? */
  1123. while (isdigit (*s)) {
  1124.   j *= 10; /* yes, add it in */
  1125.   j += *s++ - '0';
  1126. }
  1127. if (!j) break; /* punt if invalid UID validity */
  1128. stream->uid_validity = j;
  1129. /* flush whitespace */
  1130. while (*s == ' ') s++;
  1131. /* must have UID last too */
  1132. if (isdigit (*s)) {
  1133.   j = 0; /* slurp UID last */
  1134.   while (isdigit (*s)) {
  1135.     j *= 10; /* yes, add it in */
  1136.     j += *s++ - '0';
  1137.   }
  1138.   stream->uid_last = j;
  1139. /* process keywords */
  1140.   for (j = 0; (*s != 'n') && (*s != 'r'); j++) {
  1141. /* flush leading whitespace */
  1142.     while (*s == ' ') s++;
  1143.     u = strpbrk (s," nr");
  1144. /* got a keyword? */
  1145.     if ((k = (u - s)) && j < NUSERFLAGS) {
  1146.       if (stream->user_flags[j])
  1147. fs_give ((void **) &stream->user_flags[j]);
  1148.       stream->user_flags[j] = (char *) fs_get (k + 1);
  1149.       strncpy (stream->user_flags[j],s,k);
  1150.       stream->user_flags[j][k] = '';
  1151.     }
  1152.     s = u; /* advance to next keyword */
  1153.   }
  1154. /* pseudo-header seen */
  1155.   pseudoseen = T;
  1156. }
  1157. break;
  1158.       }
  1159. /* possible X-UID */
  1160.       else if (s[2] == 'U' && s[3] == 'I' && s[4] == 'D' &&
  1161.        s[5] == ':') {
  1162. /* only believe if have a UID validity */
  1163. if (stream->uid_validity && (nmsgs > 1)) {
  1164.   s += 6; /* advance to UID value */
  1165. /* flush whitespace */
  1166.   while (*s == ' ') s++;
  1167.   j = 0;
  1168. /* found a digit? */
  1169.   while (isdigit (*s)) {
  1170.     j *= 10; /* yes, add it in */
  1171.     j += *s++ - '0';
  1172.   }
  1173. /* flush remainder of line */
  1174.   while (*s != 'n') s++;
  1175. /* make sure not duplicated */
  1176.   if (elt->private.uid)
  1177.     sprintf (tmp,"Message %lu UID %lu already has UID %lu",
  1178.      pseudoseen ? elt->msgno - 1 : elt->msgno,
  1179.      j,elt->private.uid);
  1180. /* make sure UID doesn't go backwards */
  1181.   else if (j <= prevuid)
  1182.     sprintf (tmp,"Message %lu UID %lu less than %lu",
  1183.      pseudoseen ? elt->msgno - 1 : elt->msgno,
  1184.      j,prevuid + 1);
  1185. /* or skip by mailbox's recorded last */
  1186.   else if (j > stream->uid_last)
  1187.     sprintf (tmp,"Message %lu UID %lu greater than last %lu",
  1188.      pseudoseen ? elt->msgno - 1 : elt->msgno,
  1189.      j,stream->uid_last);
  1190.   else { /* normal UID case */
  1191.     prevuid = elt->private.uid = j;
  1192.     break; /* exit this cruft */
  1193.   }
  1194.   mm_log (tmp,WARN);
  1195. /* invalidate UID validity */
  1196.   stream->uid_validity = 0;
  1197.   elt->private.uid = 0;
  1198. }
  1199. break;
  1200.       }
  1201.     }
  1202. /* otherwise fall into S case */
  1203.   case 'S': /* possible Status: line */
  1204.     if (s[0] == 'S' && s[1] == 't' && s[2] == 'a' && s[3] == 't' &&
  1205. s[4] == 'u' && s[5] == 's' && s[6] == ':') {
  1206.       s += 6; /* advance to status flags */
  1207.       do switch (*s++) {/* parse flags */
  1208.       case 'R': /* message read */
  1209. elt->seen = T;
  1210. break;
  1211.       case 'O': /* message old */
  1212. if (elt->recent) {
  1213.   elt->recent = NIL;
  1214.   recent--; /* it really wasn't recent */
  1215. }
  1216. break;
  1217.       case 'D': /* message deleted */
  1218. elt->deleted = T;
  1219. break;
  1220.       case 'F': /* message flagged */
  1221. elt->flagged = T;
  1222. break;
  1223.       case 'A': /* message answered */
  1224. elt->answered = T;
  1225. break;
  1226.       case 'T': /* message is a draft */
  1227. elt->draft = T;
  1228. break;
  1229.       default: /* some other crap */
  1230. break;
  1231.       } while (*s && (*s != 'n') && (*s != 'r'));
  1232.       break; /* all done */
  1233.     }
  1234. /* otherwise fall into default case */
  1235.   default: /* ordinary header line */
  1236. /* line length in LF format newline */
  1237.     k =  i - (((i >= 2) && (s[i - 2] == 'r')) ? 1 : 0);
  1238. /* "internal" header size */
  1239.     elt->private.data += k;
  1240. /* message size */
  1241.     elt->rfc822_size += k + 1;
  1242.     break;
  1243.   }
  1244. } while (i && (*t != 'n') && (*t != 'r'));
  1245. /* "internal" header sans trailing newline */
  1246. if (i) elt->private.data--;
  1247. /* assign a UID if none found */
  1248. if (((nmsgs > 1) || !pseudoseen) && !elt->private.uid) {
  1249.   prevuid = elt->private.uid = ++stream->uid_last;
  1250.   elt->private.dirty = T;
  1251. }
  1252. else elt->private.dirty = elt->recent;
  1253. /* note size of header, location of text */
  1254. elt->private.msg.header.text.size = 
  1255.   (elt->private.msg.text.offset =
  1256.    (LOCAL->filesize + GETPOS (&bs)) - elt->private.special.offset) -
  1257.      elt->private.special.text.size;
  1258. k = m = 0; /* no previous line size yet */
  1259. /* note current position */
  1260. j = LOCAL->filesize + GETPOS (&bs);
  1261. if (i) do { /* look for next message */
  1262.   s = unix_mbxline (stream,&bs,&i);
  1263.   if (i) { /* got new data? */
  1264.     VALID (s,t,ti,zn); /* yes, parse line */
  1265.     if (!ti) { /* not a header line, add it to message */
  1266.       elt->rfc822_size += 
  1267. k = i + (m = (((i < 2) || s[i - 2] != 'r') ? 1 : 0));
  1268. /* update current position */
  1269.       j = LOCAL->filesize + GETPOS (&bs);
  1270.     }
  1271.   }
  1272. } while (i && !ti); /* until found a header */
  1273. elt->private.msg.text.text.size = j -
  1274.   (elt->private.special.offset + elt->private.msg.text.offset);
  1275. if (k == 2) { /* last line was blank? */
  1276.   elt->private.msg.text.text.size -= (m ? 1 : 2);
  1277.   elt->rfc822_size -= 2;
  1278. }
  1279.       } while (i); /* until end of buffer */
  1280.       if (pseudoseen) { /* flush pseudo-message if present */
  1281. /* decrement recent count */
  1282. if (mail_elt (stream,1)->recent) recent--;
  1283. /* and the exists count */
  1284. mail_exists (stream,nmsgs--);
  1285. mail_expunged(stream,1);/* fake an expunge of that message */
  1286.       }
  1287. /* need to start a new UID validity? */
  1288.       if (!stream->uid_validity) {
  1289. stream->uid_validity = time (0);
  1290. /* in case a whiner with no life */
  1291. if (mail_parameters (NIL,GET_USERHASNOLIFE,NIL))
  1292.   stream->uid_nosticky = T;
  1293. else LOCAL->dirty = T; /* make dirty to create pseudo-message */
  1294.       }
  1295.       stream->nmsgs = oldnmsgs; /* whack it back down */
  1296.       stream->silent = silent; /* restore old silent setting */
  1297. /* notify upper level of new mailbox sizes */
  1298.       mail_exists (stream,nmsgs);
  1299.       mail_recent (stream,recent);
  1300. /* mark dirty so O flags are set */
  1301.       if (recent) LOCAL->dirty = T;
  1302.     }
  1303.   }
  1304. /* no change, don't babble if never got time */
  1305.   else if (LOCAL->filetime && LOCAL->filetime != sbuf.st_mtime)
  1306.     mm_log ("New mailbox modification time but apparently no changes",WARN);
  1307. /* update parsed file size and time */
  1308.   LOCAL->filesize = sbuf.st_size;
  1309.   LOCAL->filetime = sbuf.st_mtime;
  1310.   return T; /* return the winnage */
  1311. }
  1312. /* UNIX read line from mailbox
  1313.  * Accepts: mail stream
  1314.  *     stringstruct
  1315.  *     pointer to line size
  1316.  * Returns: pointer to input line
  1317.  */
  1318. char *unix_mbxline (MAILSTREAM *stream,STRING *bs,unsigned long *size)
  1319. {
  1320.   unsigned long i,j,k,m;
  1321.   char *s,*t,*te,p1[CHUNK];
  1322.   char *ret = "";
  1323. /* flush old buffer */
  1324.   if (LOCAL->line) fs_give ((void **) &LOCAL->line);
  1325. /* if buffer needs refreshing */
  1326.   if (!bs->cursize) SETPOS (bs,GETPOS (bs));
  1327.   if (SIZE (bs)) { /* find newline */
  1328. /* end of fast scan */
  1329.     te = (t = (s = bs->curpos) + bs->cursize) - 12;
  1330.     while (s < te) if ((*s++ == 'n') || (*s++ == 'n') || (*s++ == 'n') ||
  1331.        (*s++ == 'n') || (*s++ == 'n') || (*s++ == 'n') ||
  1332.        (*s++ == 'n') || (*s++ == 'n') || (*s++ == 'n') ||
  1333.        (*s++ == 'n') || (*s++ == 'n') || (*s++ == 'n')) {
  1334.       --s; /* back up */
  1335.       break; /* exit loop */
  1336.     }
  1337. /* final character-at-a-time scan */
  1338.     while ((s < t) && (*s != 'n')) ++s;
  1339. /* difficult case if line spans buffer */
  1340.     if ((i = s - bs->curpos) == bs->cursize) {
  1341.       memcpy (p1,bs->curpos,i); /* remember what we have so far */
  1342. /* load next buffer */
  1343.       SETPOS (bs,k = GETPOS (bs) + i);
  1344. /* end of fast scan */
  1345.       te = (t = (s = bs->curpos) + bs->cursize) - 12;
  1346. /* fast scan in overlap buffer */
  1347.       while (s < te) if ((*s++ == 'n') || (*s++ == 'n') || (*s++ == 'n') ||
  1348.  (*s++ == 'n') || (*s++ == 'n') || (*s++ == 'n') ||
  1349.  (*s++ == 'n') || (*s++ == 'n') || (*s++ == 'n') ||
  1350.  (*s++ == 'n') || (*s++ == 'n') || (*s++ == 'n')) {
  1351. --s; /* back up */
  1352. break; /* exit loop */
  1353.       }
  1354. /* final character-at-a-time scan */
  1355.       while ((s < t) && (*s != 'n')) ++s;
  1356. /* huge line? */
  1357.       if ((j = s - bs->curpos) == bs->cursize) {
  1358. SETPOS (bs,GETPOS (bs) + j);
  1359. /* look for end of line (s-l-o-w!!) */
  1360. for (m = SIZE (bs); m && (SNX (bs) != 'n'); --m,++j);
  1361. SETPOS (bs,k); /* go back to where it started */
  1362.       }
  1363. /* got size of data, make buffer for return */
  1364.       ret = LOCAL->line = (char *) fs_get (i + j + 2);
  1365.       memcpy (ret,p1,i); /* copy first chunk */
  1366.       while (j) { /* copy remainder */
  1367. if (!bs->cursize) SETPOS (bs,GETPOS (bs));
  1368. memcpy (ret + i,bs->curpos,k = min (j,bs->cursize));
  1369. i += k; /* account for this much read in */
  1370. j -= k;
  1371. bs->curpos += k; /* increment new position */
  1372. bs->cursize -= k; /* eat that many bytes */
  1373.       }
  1374.       if (SIZE (bs)) SNX (bs); /* skip over newline if one seen */
  1375.       ret[i++] = 'n'; /* make sure newline at end */
  1376.       ret[i] = ''; /* makes debugging easier */
  1377.     }
  1378.     else { /* this is easy */
  1379.       ret = bs->curpos; /* string it at this position */
  1380.       bs->curpos += ++i; /* increment new position */
  1381.       bs->cursize -= i; /* eat that many bytes */
  1382.     }
  1383.     *size = i; /* return that to user */
  1384.   }
  1385.   else *size = 0; /* end of data, return empty */
  1386.   return ret;
  1387. }
  1388. /* UNIX make pseudo-header
  1389.  * Accepts: MAIL stream
  1390.  *     buffer to write pseudo-header
  1391.  * Returns: length of pseudo-header
  1392.  */
  1393. unsigned long unix_pseudo (MAILSTREAM *stream,char *hdr)
  1394. {
  1395.   int i;
  1396.   char *s,tmp[MAILTMPLEN];
  1397.   time_t now = time (0);
  1398.   rfc822_fixed_date (tmp);
  1399.   sprintf (hdr,"From %s %.24snDate: %snFrom: %s <%s@%.80s>nSubject: %snMessage-ID: <%lu@%.80s>nX-IMAP: %010lu %010lu",
  1400.    pseudo_from,ctime (&now),
  1401.    tmp,pseudo_name,pseudo_from,mylocalhost (),pseudo_subject,
  1402.    (unsigned long) now,mylocalhost (),stream->uid_validity,
  1403.    stream->uid_last);
  1404.   for (s = hdr,i = 0; i < NUSERFLAGS; ++i) if (stream->user_flags[i])
  1405.     sprintf (s += strlen (s)," %s",stream->user_flags[i]);
  1406.   sprintf (s += strlen (s),"nStatus: ROnn%snn",pseudo_msg);
  1407.   return strlen (hdr); /* return header length */
  1408. }
  1409. /* UNIX make status string
  1410.  * Accepts: MAIL stream
  1411.  *     destination string to write
  1412.  *     message cache entry
  1413.  *     non-zero flag to write UID as well
  1414.  * Returns: length of string
  1415.  */
  1416. unsigned long unix_xstatus (MAILSTREAM *stream,char *status,MESSAGECACHE *elt,
  1417.     long flag)
  1418. {
  1419.   char *t;
  1420.   char *s = status;
  1421.   unsigned long n = elt->user_flags;
  1422.   /* This used to be an sprintf(), but thanks to certain cretinous C libraries
  1423.      with horribly slow implementations of sprintf() I had to change it to this
  1424.      mess.  At least it should be fast. */
  1425.   *s++ = 'S'; *s++ = 't'; *s++ = 'a'; *s++ = 't'; *s++ = 'u'; *s++ = 's';
  1426.   *s++ = ':'; *s++ = ' ';
  1427.   if (elt->seen) *s++ = 'R';
  1428.   *s++ = 'O'; *s++ = 'n';
  1429.   *s++ = 'X'; *s++ = '-'; *s++ = 'S'; *s++ = 't'; *s++ = 'a'; *s++ = 't';
  1430.   *s++ = 'u'; *s++ = 's'; *s++ = ':'; *s++ = ' ';
  1431.   if (elt->deleted) *s++ = 'D';
  1432.   if (elt->flagged) *s++ = 'F';
  1433.   if (elt->answered) *s++ = 'A';
  1434.   if (elt->draft) *s++ = 'T';
  1435.   *s++ = 'n';
  1436.   if (!stream->uid_nosticky) { /* cretins with no life can't use this */
  1437.     *s++ = 'X'; *s++ = '-'; *s++ = 'K'; *s++ = 'e'; *s++ = 'y'; *s++ = 'w';
  1438.     *s++ = 'o'; *s++ = 'r'; *s++ = 'd'; *s++ = 's'; *s++ = ':';
  1439.     if (n) do {
  1440.       *s++ = ' ';
  1441.       for (t = stream->user_flags[find_rightmost_bit (&n)]; *t; *s++ = *t++);
  1442.     } while (n);
  1443. /* pad X-Keywords to make size constant */
  1444.     else for (n = 50 - (s - status); n > 0; --n) *s++ = ' ';
  1445.     *s++ = 'n';
  1446.     if (flag) { /* want to include UID? */
  1447.       char stack[64];
  1448.       char *p = stack;
  1449.       n = elt->private.uid; /* push UID digits on the stack */
  1450.       do *p++ = (char) (n % 10) + '0';
  1451.       while (n /= 10);
  1452.       *s++ = 'X'; *s++ = '-'; *s++ = 'U'; *s++ = 'I'; *s++ = 'D'; *s++ = ':';
  1453.       *s++ = ' ';
  1454. /* pop UID from stack */
  1455.       while (p > stack) *s++ = *--p;
  1456.       *s++ = 'n';
  1457.     }
  1458.   }
  1459.   *s++ = 'n'; *s = ''; /* end of extended message status */
  1460.   return s - status; /* return size of resulting string */
  1461. }
  1462. /* Rewrite mailbox file
  1463.  * Accepts: MAIL stream, must be critical and locked
  1464.  *     return pointer to number of expunged messages if want expunge
  1465.  *     lock file name
  1466.  * Returns: T if success and mailbox unlocked, NIL if failure
  1467.  */
  1468. int unix_old_algorithm = NIL; /* non-zero to use old algorithm */
  1469. long unix_rewrite (MAILSTREAM *stream,unsigned long *nexp,DOTLOCK *lock)
  1470. {
  1471.   time_t tp[2];
  1472.   unsigned long size = 0;
  1473.   unsigned long recent = stream->recent;
  1474.   long ret;
  1475.   if (nexp) *nexp = 0; /* initially nothing expunged */
  1476.   if (ret = unix_old_algorithm ? /* rewrite using selected algorithm */
  1477.       unix_rewrite_old (stream,nexp,&size,&recent) :
  1478.       unix_rewrite_new (stream,nexp,&size,&recent)) {
  1479. /* make sure tied off */
  1480.     ftruncate (LOCAL->fd,LOCAL->filesize = size);
  1481.     fsync (LOCAL->fd); /* make sure the updates take */
  1482.     LOCAL->dirty = NIL; /* no longer dirty */
  1483.    /* notify upper level of new mailbox sizes */
  1484.     mail_exists (stream,stream->nmsgs);
  1485.     mail_recent (stream,recent);
  1486. /* set atime to now, mtime a second earlier */
  1487.     tp[1] = (tp[0] = time (0)) - 1;
  1488. /* set the times, note change */
  1489.     if (!utime (stream->mailbox,tp)) LOCAL->filetime = tp[1];
  1490.     close (LOCAL->fd); /* close and reopen file */
  1491.     if ((LOCAL->fd = open (stream->mailbox,O_RDWR,NIL)) < 0) {
  1492.       sprintf (LOCAL->buf,"Mailbox open failed, aborted: %s",strerror (errno));
  1493.       mm_log (LOCAL->buf,ERROR);
  1494.       unix_abort (stream);
  1495.     }
  1496.     dotlock_unlock (lock); /* flush the lock file */
  1497.   }
  1498.   return ret; /* return state from algorithm */
  1499. }
  1500. /* Extend UNIX mailbox file
  1501.  * Accepts: MAIL stream
  1502.  *     new desired size
  1503.  * Return: T if success, else NIL
  1504.  */
  1505. long unix_extend (MAILSTREAM *stream,unsigned long size)
  1506. {
  1507.   unsigned long i = (size > LOCAL->filesize) ? size - LOCAL->filesize : 0;
  1508.   if (i) { /* does the mailbox need to grow? */
  1509.     if (i > LOCAL->buflen) { /* make sure have enough space */
  1510. /* this user won the lottery all right */
  1511.       fs_give ((void **) &LOCAL->buf);
  1512.       LOCAL->buf = (char *) fs_get ((LOCAL->buflen = i) + 1);
  1513.     }
  1514.     memset (LOCAL->buf,'',i); /* get a block of nulls */
  1515.     while (T) { /* until write successful or punt */
  1516.       lseek (LOCAL->fd,LOCAL->filesize,L_SET);
  1517.       if ((write (LOCAL->fd,LOCAL->buf,i) >= 0) && !fsync (LOCAL->fd)) break;
  1518.       else {
  1519. long e = errno; /* note error before doing ftruncate */
  1520. ftruncate (LOCAL->fd,LOCAL->filesize);
  1521. if (mm_diskerror (stream,e,NIL)) {
  1522.   fsync (LOCAL->fd); /* user chose to punt */
  1523.   sprintf (LOCAL->buf,"Unable to extend mailbox: %s",strerror (e));
  1524.   mm_log (LOCAL->buf,ERROR);
  1525.   return NIL;
  1526. }
  1527.       }
  1528.     }
  1529.   }
  1530.   return LONGT;
  1531. }
  1532. /* Rewrite mailbox file (new, fast, dangerous algorithm)
  1533.  * Accepts: MAIL stream, must be critical and locked
  1534.  *     return pointer to number of expunged messages if want expunge
  1535.  *     return pointer to size of mailbox
  1536.  *     return pointer to number of recent messages
  1537.  * Returns: T if success, NIL if failure
  1538.  */
  1539. #define OVERFLOWBUFLEN 8192 /* initial overflow buffer length */
  1540. long unix_rewrite_new (MAILSTREAM *stream,unsigned long *nexp,
  1541.        unsigned long *size,unsigned long *recent)
  1542. {
  1543.   MESSAGECACHE *elt;
  1544.   UNIXFILE f;
  1545.   char *s;
  1546.   unsigned long i,j;
  1547. /* size of mailbox starts with pseudo */
  1548.   *size = stream->uid_nosticky ? 0 : unix_pseudo (stream,LOCAL->buf);
  1549. /* calculate size of mailbox after rewrite */
  1550.   for (i = 1; i <= stream->nmsgs; i++)
  1551.     if (!(elt = mail_elt (stream,i))->deleted || !nexp) {
  1552. /* add RFC822 size of this message */
  1553.       *size += elt->private.special.text.size + elt->private.data +
  1554. unix_xstatus (stream,LOCAL->buf,elt,T) +
  1555.   elt->private.msg.text.text.size + 1;
  1556.     }
  1557. /* extend the file as necessary */
  1558.   if (!unix_extend (stream,*size)) return NIL;
  1559.   /* Set up buffered I/O file structure
  1560.    * curpos current position being written through buffering
  1561.    * filepos current position being written physically to the disk
  1562.    * bufpos current position being written in the buffer
  1563.    * protect current maximum position that can be written to the disk
  1564.    * before buffering is forced
  1565.    * The code tries to buffer so that that disk is written in multiples of
  1566.    * OVERBLOWBUFLEN bytes.
  1567.    */
  1568.   f.stream = stream; /* note mail stream */
  1569.   f.curpos = f.filepos = 0; /* start of file */
  1570.   f.protect = stream->nmsgs ? /* initial protection pointer */
  1571.     mail_elt (stream,1)->private.special.offset : 8192;
  1572.   f.bufpos = f.buf = (char *) fs_get (f.buflen = OVERFLOWBUFLEN);
  1573.   if (!stream->uid_nosticky) /* write pseudo-header */
  1574.     unix_write (&f,LOCAL->buf,unix_pseudo (stream,LOCAL->buf));
  1575. /* loop through all messages */
  1576.   for (i = 1; i <= stream->nmsgs;) {
  1577.     elt = mail_elt (stream,i); /* get cache */
  1578.     if (nexp && elt->deleted) { /* expunge this message? */
  1579.       if (elt->recent)--*recent;/* one less recent message */
  1580.       mail_expunged (stream,i); /* notify upper levels */
  1581.       ++*nexp; /* count up one more expunged message */
  1582.     }
  1583.     else { /* preserve this message */
  1584.       i++; /* advance to next message */
  1585. /* need to rewrite message? */
  1586.       if (elt->private.dirty || (f.curpos != elt->private.special.offset) ||
  1587.   (elt->private.msg.header.text.size !=
  1588.    (elt->private.data + unix_xstatus (stream,LOCAL->buf,elt,T)))) {
  1589. unsigned long newoffset = f.curpos;
  1590. /* yes, seek to internal header */
  1591. lseek (LOCAL->fd,elt->private.special.offset,L_SET);
  1592. read (LOCAL->fd,LOCAL->buf,elt->private.special.text.size);
  1593. /* see if need to squeeze out a CR */
  1594. if (LOCAL->buf[elt->private.special.text.size - 2] == 'r') {
  1595.   LOCAL->buf[--elt->private.special.text.size - 1] = 'n';
  1596.   --*size; /* squeezed out a CR from PC */
  1597. }
  1598. /* protection pointer moves to RFC822 header */
  1599. f.protect = elt->private.special.offset +
  1600.   elt->private.msg.header.offset;
  1601. /* write internal header */
  1602. unix_write (&f,LOCAL->buf,elt->private.special.text.size);
  1603. /* get RFC822 header */
  1604. s = unix_header (stream,elt->msgno,&j,FT_INTERNAL);
  1605. /* in case this got decremented */
  1606. elt->private.msg.header.offset = elt->private.special.text.size;
  1607. /* header size, sans trailing newline */
  1608. if ((j < 2) || (s[j - 2] == 'n')) j--;
  1609. if (j != elt->private.data) fatal ("header size inconsistant");
  1610. /* protection pointer moves to RFC822 text */
  1611. f.protect = elt->private.special.offset + elt->private.msg.text.offset;
  1612. unix_write (&f,s,j); /* write RFC822 header */
  1613. /* write status and UID */
  1614. unix_write (&f,LOCAL->buf,j = unix_xstatus (stream,LOCAL->buf,elt,T));
  1615. /* new file header size */
  1616. elt->private.msg.header.text.size = elt->private.data + j;
  1617. /* did text move? */
  1618. if (f.curpos != f.protect) {
  1619. /* get message text */
  1620.   s = unix_text_work (stream,elt,&j,FT_INTERNAL);
  1621. /* this can happen if CRs were squeezed */
  1622.   if (j < elt->private.msg.text.text.size) {
  1623. /* so fix up counts */
  1624.     *size -= elt->private.msg.text.text.size - j;
  1625.     elt->private.msg.text.text.size = j;
  1626.   }
  1627. /* can't happen it says here */
  1628.   else if (j > elt->private.msg.text.text.size)
  1629.     fatal ("text size inconsistant");
  1630. /* new text offset, status/UID may change it */
  1631.   elt->private.msg.text.offset = f.curpos - newoffset;
  1632. /* protection pointer moves to next message */
  1633.   f.protect = (i <= stream->nmsgs) ?
  1634.     mail_elt (stream,i)->private.special.offset : (f.curpos + j + 1);
  1635.   unix_write (&f,s,j); /* write text */
  1636. /* write trailing newline */
  1637.   unix_write (&f,"n",1);
  1638. }
  1639. else { /* tie off header and status */
  1640.   unix_write (&f,NIL,NIL);
  1641.   f.curpos = f.protect =/* restart everything at end of message */
  1642.     f.filepos += elt->private.msg.text.text.size + 1;
  1643. }
  1644. /* new internal header offset */
  1645. elt->private.special.offset = newoffset;
  1646. elt->private.dirty =NIL;/* message is now clean */
  1647.       }
  1648.       else { /* no need to rewrite */
  1649. unix_write (&f,NIL,NIL);/* tie off previous message if needed */
  1650. f.curpos = f.protect = /* restart everything at end of message */
  1651.   f.filepos += elt->private.special.text.size +
  1652.     elt->private.msg.header.text.size +
  1653.       elt->private.msg.text.text.size + 1;
  1654.       }
  1655.     }
  1656.   }
  1657.   unix_write (&f,NIL,NIL); /* tie off final message */
  1658.   if (*size != f.filepos) fatal ("file size inconsistant");
  1659.   fs_give ((void **) &f.buf); /* free buffer */
  1660.   return LONGT; /* looks good */
  1661. }
  1662. /* Write data to buffered file
  1663.  * Accepts: buffered file pointer
  1664.  *     file data or NIL to indicate "flush buffer"
  1665.  *     date size (ignored for "flush buffer")
  1666.  * Does not return until success
  1667.  */
  1668. void unix_write (UNIXFILE *f,char *buf,unsigned long size)
  1669. {
  1670.   unsigned long i,j,k;
  1671.   if (buf) { /* doing buffered write? */
  1672.     i = f->bufpos - f->buf; /* yes, get size of current buffer data */
  1673. /* yes, have space in current buffer chunk? */
  1674.     if (j = i ? ((f->buflen - i) % OVERFLOWBUFLEN) : f->buflen) {
  1675. /* yes, fill up buffer as much as we can */
  1676.       memcpy (f->bufpos,buf,k = min (j,size));
  1677.       f->bufpos += k; /* new buffer position */
  1678.       f->curpos += k; /* new current position */
  1679.       if (j -= k) return; /* all done if still have buffer free space */
  1680.       buf += k; /* full, get new unwritten data pointer */
  1681.       size -= k; /* new data size */
  1682.       i += k; /* new buffer data size */
  1683.     }
  1684.     /* This chunk of the buffer is full.  See if can make some space by
  1685.      * writing to the disk, if there's enough unprotected space to do so.
  1686.      * Try to fill out any unaligned chunk, along with any subsequent full
  1687.      * chunks that will fit in unprotected space.
  1688.      */
  1689. /* any unprotected space we can write to? */
  1690.     if (j = min (i,f->protect - f->filepos)) {
  1691. /* yes, filepos not at chunk boundary? */
  1692.       if ((k = f->filepos % OVERFLOWBUFLEN) && ((k = OVERFLOWBUFLEN - k) < j))
  1693. j -= k; /* yes, and can write out partial chunk */
  1694.       else k = 0; /* no partial chunk to write */
  1695. /* if at least a chunk free, write that too */
  1696.       if (j > OVERFLOWBUFLEN) k += j - (j % OVERFLOWBUFLEN);
  1697.       if (k) { /* write data if there is anything we can */
  1698. unix_phys_write (f,f->buf,k);
  1699. /* slide buffer */
  1700. if (i -= k) memmove (f->buf,f->buf + k,i);
  1701. f->bufpos = f->buf + i; /* new end of buffer */
  1702.       }
  1703.     }
  1704.     /* Have flushed the buffer as best as possible.  All done if no more
  1705.      * data to write.  Otherwise, if the buffer is empty AND if the unwritten
  1706.      * data is larger than a chunk AND the unprotected space is also larger
  1707.      * than a chunk, then write as many chunks as we can directly from the
  1708.      * data.  Buffer the rest, expanding the buffer as needed.
  1709.      */
  1710.     if (size) { /* have more data that we need to buffer? */
  1711. /* can write any of it to disk instead? */
  1712.       if ((f->bufpos == f->buf) && 
  1713.   ((j = min (f->protect - f->filepos,size)) > OVERFLOWBUFLEN)) {
  1714. /* write as much as we can right now */
  1715. unix_phys_write (f,buf,j -= (j % OVERFLOWBUFLEN));
  1716. buf += j; /* new data pointer */
  1717. size -= j; /* new data size */
  1718. f->curpos += j; /* advance current pointer */
  1719.       }
  1720.       if (size) { /* still have data that we need to buffer? */
  1721. /* yes, need to expand the buffer? */
  1722. if ((i = ((f->bufpos + size) - f->buf)) > f->buflen) {
  1723. /* note current position in buffer */
  1724.   j = f->bufpos - f->buf;
  1725.   i += OVERFLOWBUFLEN; /* yes, grow another chunk */
  1726.   fs_resize ((void **) &f->buf,f->buflen = i - (i % OVERFLOWBUFLEN));
  1727. /* in case buffer relocated */
  1728.   f->bufpos = f->buf + j;
  1729. }
  1730. /* buffer remaining data */
  1731. memcpy (f->bufpos,buf,size);
  1732. f->bufpos += size; /* new end of buffer */
  1733. f->curpos += size; /* advance current pointer */
  1734.       }
  1735.     }
  1736.   }
  1737.   else { /* flush buffer to disk */
  1738.     unix_phys_write (f,f->buf,i = f->bufpos - f->buf);
  1739.     f->bufpos = f->buf; /* reset buffer */
  1740. /* update positions */
  1741.     f->curpos = f->protect = f->filepos;
  1742.   }
  1743. }
  1744. /* Physical disk write
  1745.  * Accepts: buffered file pointer
  1746.  *     buffer address
  1747.  *     buffer size
  1748.  * Does not return until success
  1749.  */
  1750. void unix_phys_write (UNIXFILE *f,char *buf,size_t size)
  1751. {
  1752.   MAILSTREAM *stream = f->stream;
  1753. /* write data at desired position */
  1754.   while (size && ((lseek (LOCAL->fd,f->filepos,L_SET) < 0) ||
  1755.   (write (LOCAL->fd,buf,size) < 0))) {
  1756.     int e;
  1757.     char tmp[MAILTMPLEN];
  1758.     sprintf (tmp,"Unable to write to mailbox: %s",strerror (e = errno));
  1759.     mm_log (tmp,ERROR);
  1760.     mm_diskerror (NIL,e,T); /* serious problem, must retry */
  1761.   }
  1762.   f->filepos += size; /* update file position */
  1763. }
  1764. /* Rewrite mailbox file (old, slow, safe algorithm)
  1765.  * Accepts: MAIL stream, must be critical and locked
  1766.  *     return pointer to number of expunged messages if want expunge
  1767.  *     return pointer to size of mailbox
  1768.  *     return pointer to number of recent messages
  1769.  * Returns: T if success, NIL if failure
  1770.  */
  1771. long unix_rewrite_old (MAILSTREAM *stream,unsigned long *nexp,
  1772.        unsigned long *size,unsigned long *recent)
  1773. {
  1774.   unsigned long i,j;
  1775.   int e,retry;
  1776.   struct stat sbuf;
  1777.   FILE *f;
  1778.   MESSAGECACHE *elt;
  1779. /* open scratch file */
  1780.   if (!(f = tmpfile ())) return unix_punt_scratch (NIL);
  1781.   if (!(stream->uid_nosticky || /* write pseudo-header */
  1782. unix_fwrite (f,LOCAL->buf,unix_pseudo (stream,LOCAL->buf),size)))
  1783.     return unix_punt_scratch (f);
  1784.   if (nexp) { /* expunging */
  1785.     for (i = 1; i <= stream->nmsgs; i++)
  1786.       if (!(elt = mail_elt (stream,i))->deleted &&
  1787.   !unix_write_message (f,stream,elt,size))
  1788. return unix_punt_scratch (f);
  1789.   }
  1790.   else for (i = 1; i <= stream->nmsgs; i++)
  1791.     if (!unix_write_message (f,stream,mail_elt (stream,i),size))
  1792.       return unix_punt_scratch (f);
  1793. /* write remaining data */
  1794.   if (fflush (f) || fstat (fileno (f),&sbuf)) return unix_punt_scratch (f);
  1795.   if (*size != sbuf.st_size) { /* make damn sure stdio isn't lying */
  1796.     char tmp[MAILTMPLEN];
  1797.     sprintf (tmp,"Checkpoint file size mismatch (%lu != %lu)",
  1798.      (unsigned long) size,(unsigned long) sbuf.st_size);
  1799.     mm_log (tmp,ERROR);
  1800.     fclose (f); /* flush the output file */
  1801.     return NIL;
  1802.   }
  1803. /* extend the file as necessary */
  1804.   if (!unix_extend (stream,*size)) return NIL;
  1805. /* update the cache */
  1806.   for (i = 1; i <= stream->nmsgs;) {
  1807.     elt = mail_elt (stream,i); /* get cache */
  1808.     if (nexp && elt->deleted) { /* expunge this message? */
  1809.       if (elt->recent)--*recent;/* one less recent message */
  1810.       mail_expunged (stream,i); /* notify upper levels */
  1811.       ++*nexp; /* count up one more expunged message */
  1812.     }
  1813.     else { /* update file pointers from kludgey places */
  1814.       elt->private.special.offset = elt->private.msg.full.offset;
  1815.       elt->private.msg.text.offset = elt->private.msg.full.text.size;
  1816. /* in case header grew */
  1817.       elt->private.msg.header.text.size = elt->private.msg.text.offset -
  1818. elt->private.msg.header.offset;
  1819. /* stomp on these two kludges */
  1820.       elt->private.msg.full.offset = elt->private.msg.full.text.size = 0;
  1821.       i++; /* preserved message */
  1822.     }
  1823.   }
  1824.   do { /* restart point if failure */
  1825.     retry = NIL; /* no need to retry yet */
  1826.     fseek (f,0,L_SET); /* rewind files */
  1827.     lseek (LOCAL->fd,0,L_SET);
  1828.     for (i = *size; i; i -= j)
  1829.       if (!((j = fread (LOCAL->buf,1,min ((long) CHUNK,i),f)) &&
  1830.     (write (LOCAL->fd,LOCAL->buf,j) >= 0))) {
  1831. sprintf (LOCAL->buf,"Mailbox rewrite error: %s",strerror (e = errno));
  1832. mm_notify (stream,LOCAL->buf,WARN);
  1833. mm_diskerror (stream,e,T);
  1834. retry = T; /* must retry */
  1835. break;
  1836.       }
  1837.   } while (retry); /* in case need to retry */
  1838.   fclose (f); /* finished with scratch file */
  1839.   return LONGT;
  1840. }
  1841. /* Write message (old algorithm)
  1842.  * Accepts: destination file
  1843.  *     MAIL stream
  1844.  *     message number
  1845.  *     pointer to current filesize tally
  1846.  * Returns: T if success, NIL if failure
  1847.  */
  1848. long unix_write_message (FILE *f,MAILSTREAM *stream,MESSAGECACHE *elt,
  1849.  unsigned long *size)
  1850. {
  1851.   char *s;
  1852.   unsigned long i;
  1853. /* (kludge alert) note new message offset */
  1854.   elt->private.msg.full.offset = *size;
  1855. /* internal header */
  1856.   lseek (LOCAL->fd,elt->private.special.offset,L_SET);
  1857.   read (LOCAL->fd,LOCAL->buf,elt->private.special.text.size);
  1858.   if (unix_fwrite (f,LOCAL->buf,elt->private.special.text.size,size)) {
  1859. /* get header */
  1860.     s = unix_header (stream,elt->msgno,&i,FT_INTERNAL);
  1861. /* header size, sans trailing newline */
  1862.     if (i && (s[i - 2] == 'n')) i--;
  1863. /* write header */
  1864.     if (unix_fwrite (f,s,i,size) &&
  1865. unix_fwrite (f,LOCAL->buf,unix_xstatus(stream,LOCAL->buf,elt,T),size)){
  1866. /* (kludge alert) note new text offset */
  1867.       elt->private.msg.full.text.size = *size - elt->private.msg.full.offset;
  1868. /* get text */
  1869.       s = unix_text_work (stream,elt,&i,FT_INTERNAL);
  1870. /* write text and trailing newline */
  1871.       if (unix_fwrite (f,s,i,size) && unix_fwrite (f,"n",1,size)) return T;
  1872.     }
  1873.   }
  1874.   return NIL; /* failed */
  1875. }
  1876. /* Safely write buffer (old algorithm)
  1877.  * Accepts: destination file
  1878.  *     buffer pointer
  1879.  *     number of octets
  1880.  *     pointer to current filesize tally
  1881.  * Returns: T if successful, NIL if failure
  1882.  */
  1883. long unix_fwrite (FILE *f,char *s,unsigned long i,unsigned long *size)
  1884. {
  1885.   unsigned long j;
  1886.   while (i && ((j = fwrite (s,1,i,f)) || (errno == EINTR))) {
  1887.     *size += j;
  1888.     s += j;
  1889.     i -= j;
  1890.   }
  1891.   return i ? NIL : T; /* success if wrote all requested data */
  1892. }
  1893. /* Punt scratch file (old algorithm)
  1894.  * Accepts: file pointer
  1895.  * Returns: NIL, always
  1896.  */
  1897. long unix_punt_scratch (FILE *f)
  1898. {
  1899.   char tmp[MAILTMPLEN];
  1900.   sprintf (tmp,"Checkpoint file failure: %s",strerror (errno));
  1901.   mm_log (tmp,ERROR);
  1902.   if (f) fclose (f); /* flush the output file */
  1903.   return NIL;
  1904. }