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

网络编程

开发平台:

Unix_Linux

  1. /*
  2.  * Program: UNIX mail routines, Amiga version
  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: 23 September 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. /* Validate line
  47.  * Accepts: pointer to candidate string to validate as a From header
  48.  *     return pointer to end of date/time field
  49.  *     return pointer to offset from t of time (hours of ``mmm dd hh:mm'')
  50.  *     return pointer to offset from t of time zone (if non-zero)
  51.  * Returns: t,ti,zn set if valid From string, else ti is NIL
  52.  */
  53. #define VALID(s,x,ti,zn) {
  54.   int remote = 0;
  55.   ti = 0;
  56.   if ((*s == 'F') && (s[1] == 'r') && (s[2] == 'o') && (s[3] == 'm') &&
  57.       (s[4] == ' ')) {
  58.     for (x = s + 5; *x && *x != '12'; x++);
  59.     if (*x) {
  60.       if (x[-1] == '15') --x;
  61.       if (x - s >= 41) {
  62. for (zn = -1; x[zn] != ' '; zn--);
  63. if ((x[zn-1] == 'm') && (x[zn-2] == 'o') && (x[zn-3] == 'r') &&
  64.     (x[zn-4] == 'f') && (x[zn-5] == ' ') && (x[zn-6] == 'e') &&
  65.     (x[zn-7] == 't') && (x[zn-8] == 'o') && (x[zn-9] == 'm') &&
  66.     (x[zn-10] == 'e') && (x[zn-11] == 'r') && (x[zn-12] == ' '))
  67.   {
  68.     while (x[zn-13] == ' ') zn--;
  69.     x += zn - 12;
  70.     remote = 1;
  71.   }
  72.       }
  73.       if (x - s >= 27) {
  74. if (x[-5] == ' ') {
  75.   if (x[-8] == ':') zn = 0,ti = -5;
  76.   else if (x[-9] == ' ') ti = zn = -9;
  77.   else if ((x[-11] == ' ') && ((x[-10]=='+') || (x[-10]=='-')))
  78.     ti = zn = -11;
  79. }
  80. else if (x[-4] == ' ') {
  81.   if (x[-9] == ' ') zn = -4,ti = -9;
  82.   else if ( (x[-13] == ' ') && (x[-16] == ' ')
  83. && (x[-20] ==' ') &&
  84. ( ((x[-22] == ' ') && (x[-23] == ',')) ||
  85.   ((x[-23] == ' ') && (x[-24] == ',')) ) ) {
  86.     char weekday[4]={0,}, month[4]={0,}, time[11]={0,};
  87.     char tzone[4]={0,}; 
  88.     char realtime[80];
  89.     int day,year,start=-26;
  90.     if (x[-23] == ' ') x--;
  91.       sscanf(&x[start],"%3c, %d %s %d %s %s",
  92. weekday,&day,month,&year,time,tzone);
  93.       sprintf(realtime,"%s %s %2d %s %d %s",
  94. weekday,month,day,time, 
  95. ( (year < 100) ? year+1900 : year),tzone);
  96.       if (remote)
  97. strcat(realtime," remote from ");
  98.       else
  99. strcat(realtime,"n");
  100.       strncpy(&x[start],realtime,strlen(realtime));
  101.       zn = -2;
  102.       ti = -7;
  103.   }
  104. }
  105. else if (x[-6] == ' ') {
  106.   if ((x[-11] == ' ') && ((x[-5] == '+') || (x[-5] == '-')))
  107.     zn = -6,ti = -11;
  108. }
  109. else if (x[-9] == ' ') {
  110.     if ( ( (x[-12] == ' ') && (x[-16] == ' ') &&
  111.   ( ((x[-18] == ' ') && (x[-19] == ',') )  ||
  112.     ((x[-19] == ' ') && (x[-20] == ',')) )
  113. ||
  114. ((x[-14] == ' ') && (x[-18] == ' ') &&
  115.   ( ((x[-20] == ' ') && (x[-21] == ',') )  ||
  116.     ((x[-21] == ' ') && (x[-22] == ',')) ) ) ) ) {
  117.       char weekday[4]={0,}, month[4]={0,},time[11]={0,};
  118.       int day,year,start=-24;
  119.       char realtime[80];
  120.       if (x[-12] == ' ') x++;
  121.       if (x[-19] == ' ') x++;
  122.       sscanf(&x[start],"%3c, %d %3c %d %s",weekday,
  123.      &day,month,&year,time);
  124.       sprintf(realtime,"%s %s %2d %s %d",weekday,month,day,time,
  125.  ( (year < 100) ? year+1900 : year));
  126.       if (remote)
  127. strcat(realtime," remote from ");
  128.       else
  129. strcat(realtime,"n");
  130.       strncpy(&x[start],realtime,strlen(realtime));
  131.       ti=-5;
  132.       zn=0;
  133.     }
  134. }
  135. if (ti && !((x[ti - 3] == ':') &&
  136.     (x[ti -= ((x[ti - 6] == ':') ? 9 : 6)] == ' ') &&
  137.     (x[ti - 3] == ' ') && (x[ti - 7] == ' ') &&
  138.     (x[ti - 11] == ' '))) ti = 0;
  139.       }
  140.     }
  141.   }
  142. }
  143. /* You are not expected to understand this macro, but read the next page if
  144.  * you are not faint of heart.
  145.  *
  146.  * Known formats to the VALID macro are:
  147.  * From user Wed Dec  2 05:53 1992
  148.  * BSD From user Wed Dec  2 05:53:22 1992
  149.  * SysV From user Wed Dec  2 05:53 PST 1992
  150.  * rn From user Wed Dec  2 05:53:22 PST 1992
  151.  * From user Wed Dec  2 05:53 -0700 1992
  152.  * emacs From user Wed Dec  2 05:53:22 -0700 1992
  153.  * From user Wed Dec  2 05:53 1992 PST
  154.  * From user Wed Dec  2 05:53:22 1992 PST
  155.  * From user Wed Dec  2 05:53 1992 -0700
  156.  * Solaris From user Wed Dec  2 05:53:22 1992 -0700
  157.  *
  158.  * Amiga From user Wed, 6 Dec 92 05:53:22 who did this !!!
  159.  * CHANGED in place to
  160.  * From user Wed Dec  2 05:53:22 1992
  161.  *
  162.  * Plus all of the above with `` remote from xxx'' after it. Thank you very
  163.  * much, smail and Solaris, for making my life considerably more complicated.
  164.  */
  165. /*
  166.  * What?  You want to understand the VALID macro anyway?  Alright, since you
  167.  * insist.  Actually, it isn't really all that difficult, provided that you
  168.  * take it step by step.
  169.  *
  170.  * Line 1 Initializes the return ti value to failure (0);
  171.  * Lines 2-3 Validates that the 1st-5th characters are ``From ''.
  172.  * Lines 4-6 Validates that there is an end of line and points x at it.
  173.  * Lines 7-14 First checks to see if the line is at least 41 characters long.
  174.  * If so, it scans backwards to find the rightmost space.  From
  175.  * that point, it scans backwards to see if the string matches
  176.  * `` remote from''.  If so, it sets x to point to the space at
  177.  * the start of the string.
  178.  * Line 15 Makes sure that there are at least 27 characters in the line.
  179.  * Lines 16-21 Checks if the date/time ends with the year (there is a space
  180.  * five characters back).  If there is a colon three characters
  181.  * further back, there is no timezone field, so zn is set to 0
  182.  * and ti is set in front of the year.  Otherwise, there must
  183.  * either to be a space four characters back for a three-letter
  184.  * timezone, or a space six characters back followed by a + or -
  185.  * for a numeric timezone; in either case, zn and ti become the
  186.  * offset of the space immediately before it.
  187.  * Lines 22-24 Are the failure case for line 14.  If there is a space four
  188.  * characters back, it is a three-letter timezone; there must be a
  189.  * space for the year nine characters back.  zn is the zone
  190.  * offset; ti is the offset of the space.
  191.  * Lines 25-28 Are the failure case for line 20.  If there is a space six
  192.  * characters back, it is a numeric timezone; there must be a
  193.  * space eleven characters back and a + or - five characters back.
  194.  * zn is the zone offset; ti is the offset of the space.
  195.  * Line 29-32 If ti is valid, make sure that the string before ti is of the
  196.  * form www mmm dd hh:mm or www mmm dd hh:mm:ss, otherwise
  197.  * invalidate ti.  There must be a colon three characters back
  198.  * and a space six or nine characters back (depending upon
  199.  * whether or not the character six characters back is a colon).
  200.  * There must be a space three characters further back (in front
  201.  * of the day), one seven characters back (in front of the month),
  202.  * and one eleven characters back (in front of the day of week).
  203.  * ti is set to be the offset of the space before the time.
  204.  *
  205.  * Why a macro?  It gets invoked a *lot* in a tight loop.  On some of the
  206.  * newer pipelined machines it is faster being open-coded than it would be if
  207.  * subroutines are called.
  208.  *
  209.  * Why does it scan backwards from the end of the line, instead of doing the
  210.  * much easier forward scan?  There is no deterministic way to parse the
  211.  * ``user'' field, because it may contain unquoted spaces!  Yes, I tested it to
  212.  * see if unquoted spaces were possible.  They are, and I've encountered enough
  213.  * evil mail to be totally unwilling to trust that ``it will never happen''.
  214.  */
  215. /* Build parameters */
  216. #define KODRETRY 15 /* kiss-of-death retry in seconds */
  217. #define LOCKTIMEOUT 5 /* lock timeout in minutes */
  218. #define CHUNK 16384 /* read-in chunk size */
  219. /* UNIX I/O stream local data */
  220. typedef struct unix_local {
  221.   unsigned int dirty : 1; /* disk copy needs updating */
  222.   int fd; /* mailbox file descriptor */
  223.   int ld; /* lock file descriptor */
  224.   char *lname; /* lock file name */
  225.   off_t filesize; /* file size parsed */
  226.   time_t filetime; /* last file time */
  227.   time_t lastsnarf; /* last snarf time (for mbox driver) */
  228.   char *buf; /* temporary buffer */
  229.   unsigned long buflen; /* current size of temporary buffer */
  230.   char *line; /* returned line */
  231. } UNIXLOCAL;
  232. /* Convenient access to local data */
  233. #define LOCAL ((UNIXLOCAL *) stream->local)
  234. /* UNIX protected file structure */
  235. typedef struct unix_file {
  236.   MAILSTREAM *stream; /* current stream */
  237.   off_t curpos; /* current file position */
  238.   off_t protect; /* protected position */
  239.   off_t filepos; /* current last written file position */
  240.   char *buf; /* overflow buffer */
  241.   size_t buflen; /* current overflow buffer length */
  242.   char *bufpos; /* current buffer position */
  243. } UNIXFILE;
  244. /* Function prototypes */
  245. DRIVER *unix_valid (char *name);
  246. long unix_isvalid_fd (int fd);
  247. void *unix_parameters (long function,void *value);
  248. void unix_scan (MAILSTREAM *stream,char *ref,char *pat,char *contents);
  249. void unix_list (MAILSTREAM *stream,char *ref,char *pat);
  250. void unix_lsub (MAILSTREAM *stream,char *ref,char *pat);
  251. long unix_create (MAILSTREAM *stream,char *mailbox);
  252. long unix_delete (MAILSTREAM *stream,char *mailbox);
  253. long unix_rename (MAILSTREAM *stream,char *old,char *newname);
  254. MAILSTREAM *unix_open (MAILSTREAM *stream);
  255. void unix_close (MAILSTREAM *stream,long options);
  256. char *unix_header (MAILSTREAM *stream,unsigned long msgno,
  257.    unsigned long *length,long flags);
  258. long unix_text (MAILSTREAM *stream,unsigned long msgno,STRING *bs,long flags);
  259. char *unix_text_work (MAILSTREAM *stream,MESSAGECACHE *elt,
  260.       unsigned long *length,long flags);
  261. void unix_flagmsg (MAILSTREAM *stream,MESSAGECACHE *elt);
  262. long unix_ping (MAILSTREAM *stream);
  263. void unix_check (MAILSTREAM *stream);
  264. void unix_check (MAILSTREAM *stream);
  265. void unix_expunge (MAILSTREAM *stream);
  266. long unix_copy (MAILSTREAM *stream,char *sequence,char *mailbox,long options);
  267. long unix_append (MAILSTREAM *stream,char *mailbox,char *flags,char *date,
  268.   STRING *message);
  269. long unix_append_putc (int fd,char *s,long *i,char c);
  270. void unix_abort (MAILSTREAM *stream);
  271. char *unix_file (char *dst,char *name);
  272. int unix_lock (char *file,int flags,int mode,DOTLOCK *lock,int op);
  273. void unix_unlock (int fd,MAILSTREAM *stream,DOTLOCK *lock);
  274. int unix_parse (MAILSTREAM *stream,DOTLOCK *lock,int op);
  275. char *unix_mbxline (MAILSTREAM *stream,STRING *bs,unsigned long *size);
  276. unsigned long unix_pseudo (MAILSTREAM *stream,char *hdr);
  277. unsigned long unix_xstatus (MAILSTREAM *stream,char *status,MESSAGECACHE *elt,
  278.     long flag);
  279. long unix_rewrite (MAILSTREAM *stream,unsigned long *nexp,DOTLOCK *lock);
  280. long unix_extend (MAILSTREAM *stream,unsigned long size);
  281. /* New algorithm */
  282. long unix_rewrite_new (MAILSTREAM *stream,unsigned long *nexp,
  283.        unsigned long *size,unsigned long *recent);
  284. void unix_write (UNIXFILE *f,char *s,unsigned long i);
  285. void unix_phys_write (UNIXFILE *f,char *buf,size_t size);
  286. /* Old algorithm */
  287. long unix_rewrite_old (MAILSTREAM *stream,unsigned long *nexp,
  288.        unsigned long *size,unsigned long *recent);
  289. long unix_write_message (FILE *f,MAILSTREAM *stream,MESSAGECACHE *elt,
  290.  unsigned long *size);
  291. long unix_fwrite (FILE *f,char *s,unsigned long i,unsigned long *size);
  292. long unix_punt_scratch (FILE *f);