Annotation of 43BSDTahoe/ucb/Mail/head.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1980 Regents of the University of California.
        !             3:  * All rights reserved.
        !             4:  *
        !             5:  * Redistribution and use in source and binary forms are permitted
        !             6:  * provided that this notice is preserved and that due credit is given
        !             7:  * to the University of California at Berkeley. The name of the University
        !             8:  * may not be used to endorse or promote products derived from this
        !             9:  * software without specific prior written permission. This software
        !            10:  * is provided ``as is'' without express or implied warranty.
        !            11:  */
        !            12: 
        !            13: #ifdef notdef
        !            14: static char sccsid[] = "@(#)head.c     5.4 (Berkeley) 2/18/88";
        !            15: #endif /* notdef */
        !            16: 
        !            17: #include "rcv.h"
        !            18: 
        !            19: /*
        !            20:  * Mail -- a mail program
        !            21:  *
        !            22:  * Routines for processing and detecting headlines.
        !            23:  */
        !            24: 
        !            25: /*
        !            26:  * See if the passed line buffer is a mail header.
        !            27:  * Return true if yes.  Note the extreme pains to
        !            28:  * accomodate all funny formats.
        !            29:  */
        !            30: ishead(linebuf)
        !            31:        char linebuf[];
        !            32: {
        !            33:        register char *cp;
        !            34:        struct headline hl;
        !            35:        char parbuf[BUFSIZ];
        !            36: 
        !            37:        cp = linebuf;
        !            38:        if (*cp++ != 'F' || *cp++ != 'r' || *cp++ != 'o' || *cp++ != 'm' ||
        !            39:            *cp++ != ' ')
        !            40:                return (0);
        !            41:        parse(linebuf, &hl, parbuf);
        !            42:        if (hl.l_from == NOSTR || hl.l_date == NOSTR) {
        !            43:                fail(linebuf, "No from or date field");
        !            44:                return (0);
        !            45:        }
        !            46:        if (!isdate(hl.l_date)) {
        !            47:                fail(linebuf, "Date field not legal date");
        !            48:                return (0);
        !            49:        }
        !            50:        /*
        !            51:         * I guess we got it!
        !            52:         */
        !            53:        return (1);
        !            54: }
        !            55: 
        !            56: /*ARGSUSED*/
        !            57: fail(linebuf, reason)
        !            58:        char linebuf[], reason[];
        !            59: {
        !            60: 
        !            61:        /*
        !            62:        if (value("debug") == NOSTR)
        !            63:                return;
        !            64:        fprintf(stderr, "\"%s\"\nnot a header because %s\n", linebuf, reason);
        !            65:        */
        !            66: }
        !            67: 
        !            68: /*
        !            69:  * Split a headline into its useful components.
        !            70:  * Copy the line into dynamic string space, then set
        !            71:  * pointers into the copied line in the passed headline
        !            72:  * structure.  Actually, it scans.
        !            73:  */
        !            74: parse(line, hl, pbuf)
        !            75:        char line[], pbuf[];
        !            76:        register struct headline *hl;
        !            77: {
        !            78:        register char *cp;
        !            79:        char *sp;
        !            80:        char word[LINESIZE];
        !            81: 
        !            82:        hl->l_from = NOSTR;
        !            83:        hl->l_tty = NOSTR;
        !            84:        hl->l_date = NOSTR;
        !            85:        cp = line;
        !            86:        sp = pbuf;
        !            87:        /*
        !            88:         * Skip over "From" first.
        !            89:         */
        !            90:        cp = nextword(cp, word);
        !            91:        cp = nextword(cp, word);
        !            92:        if (*word)
        !            93:                hl->l_from = copyin(word, &sp);
        !            94:        if (cp != NOSTR && cp[0] == 't' && cp[1] == 't' && cp[2] == 'y') {
        !            95:                cp = nextword(cp, word);
        !            96:                hl->l_tty = copyin(word, &sp);
        !            97:        }
        !            98:        if (cp != NOSTR)
        !            99:                hl->l_date = copyin(cp, &sp);
        !           100: }
        !           101: 
        !           102: /*
        !           103:  * Copy the string on the left into the string on the right
        !           104:  * and bump the right (reference) string pointer by the length.
        !           105:  * Thus, dynamically allocate space in the right string, copying
        !           106:  * the left string into it.
        !           107:  */
        !           108: char *
        !           109: copyin(src, space)
        !           110:        register char *src;
        !           111:        char **space;
        !           112: {
        !           113:        register char *cp;
        !           114:        char *top;
        !           115: 
        !           116:        top = cp = *space;
        !           117:        while (*cp++ = *src++)
        !           118:                ;
        !           119:        *space = cp;
        !           120:        return (top);
        !           121: }
        !           122: 
        !           123: /*
        !           124:  * Test to see if the passed string is a ctime(3) generated
        !           125:  * date string as documented in the manual.  The template
        !           126:  * below is used as the criterion of correctness.
        !           127:  * Also, we check for a possible trailing time zone using
        !           128:  * the auxtype template.
        !           129:  */
        !           130: 
        !           131: #define        L       1               /* A lower case char */
        !           132: #define        S       2               /* A space */
        !           133: #define        D       3               /* A digit */
        !           134: #define        O       4               /* An optional digit or space */
        !           135: #define        C       5               /* A colon */
        !           136: #define        N       6               /* A new line */
        !           137: #define U      7               /* An upper case char */
        !           138: 
        !           139: char ctypes[] = { U,L,L,S,U,L,L,S,O,D,S,D,D,C,D,D,C,D,D,S,D,D,D,D,0 };
        !           140: char tmztypes[] = { U,L,L,S,U,L,L,S,O,D,S,D,D,C,D,D,C,D,D,S,U,U,U,S,D,D,D,D,0 };
        !           141: 
        !           142: isdate(date)
        !           143:        char date[];
        !           144: {
        !           145: 
        !           146:        if (cmatch(date, ctypes))
        !           147:                return (1);
        !           148:        return (cmatch(date, tmztypes));
        !           149: }
        !           150: 
        !           151: /*
        !           152:  * Match the given string (cp) against the given template (tp).
        !           153:  * Return 1 if they match, 0 if they don't
        !           154:  */
        !           155: 
        !           156: cmatch(cp, tp)
        !           157:        register char *cp, *tp;
        !           158: {
        !           159: 
        !           160:        while (*cp && *tp)
        !           161:                switch (*tp++) {
        !           162:                case L:
        !           163:                        if (!islower(*cp++))
        !           164:                                return 0;
        !           165:                        break;
        !           166:                case U:
        !           167:                        if (!isupper(*cp++))
        !           168:                                return 0;
        !           169:                        break;
        !           170:                case S:
        !           171:                        if (*cp++ != ' ')
        !           172:                                return 0;
        !           173:                        break;
        !           174:                case D:
        !           175:                        if (!isdigit(*cp++))
        !           176:                                return 0;
        !           177:                        break;
        !           178:                case O:
        !           179:                        if (*cp != ' ' && !isdigit(*cp))
        !           180:                                return 0;
        !           181:                        cp++;
        !           182:                        break;
        !           183:                case C:
        !           184:                        if (*cp++ != ':')
        !           185:                                return 0;
        !           186:                        break;
        !           187:                case N:
        !           188:                        if (*cp++ != '\n')
        !           189:                                return 0;
        !           190:                        break;
        !           191:                }
        !           192:        if (*cp || *tp)
        !           193:                return 0;
        !           194:        return (1);
        !           195: }
        !           196: 
        !           197: /*
        !           198:  * Collect a liberal (space, tab delimited) word into the word buffer
        !           199:  * passed.  Also, return a pointer to the next word following that,
        !           200:  * or NOSTR if none follow.
        !           201:  */
        !           202: char *
        !           203: nextword(wp, wbuf)
        !           204:        register char *wp, *wbuf;
        !           205: {
        !           206:        register c;
        !           207: 
        !           208:        if (wp == NOSTR) {
        !           209:                *wbuf = 0;
        !           210:                return (NOSTR);
        !           211:        }
        !           212:        while ((c = *wp++) && c != ' ' && c != '\t') {
        !           213:                *wbuf++ = c;
        !           214:                if (c == '"') {
        !           215:                        while ((c = *wp++) && c != '"')
        !           216:                                *wbuf++ = c;
        !           217:                        if (c == '"')
        !           218:                                *wbuf++ = c;
        !           219:                        else
        !           220:                                wp--;
        !           221:                }
        !           222:        }
        !           223:        *wbuf = '\0';
        !           224:        for (; c == ' ' || c == '\t'; c = *wp++)
        !           225:                ;
        !           226:        if (c == 0)
        !           227:                return (NOSTR);
        !           228:        return (wp - 1);
        !           229: }
        !           230: 
        !           231: /*
        !           232:  * Is c contained in s?
        !           233:  */
        !           234: any(c, s)
        !           235:        register c;
        !           236:        register char *s;
        !           237: {
        !           238: 
        !           239:        while (*s)
        !           240:                if (*s++ == c)
        !           241:                        return 1;
        !           242:        return 0;
        !           243: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.