2 * Heirloom mailx - a mail user agent derived from Berkeley Mail.
4 * Copyright (c) 2000-2004 Gunnar Ritter, Freiburg i. Br., Germany.
7 * Copyright (c) 1980, 1993
8 * The Regents of the University of California. All rights reserved.
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
18 * 3. All advertising materials mentioning features or use of this software
19 * must display the following acknowledgement:
20 * This product includes software developed by the University of
21 * California, Berkeley and its contributors.
22 * 4. Neither the name of the University nor the names of its contributors
23 * may be used to endorse or promote products derived from this software
24 * without specific prior written permission.
26 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
27 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
28 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
29 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
30 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
31 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
32 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
33 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
34 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
35 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
41 static char sccsid
[] = "@(#)names.c 2.22 (gritter) 3/4/06";
46 * Mail -- a mail program
58 static struct name
*tailof(struct name
*name
);
59 static struct name
*extract1(char *line
, enum gfield ntype
, char *separators
,
61 static char *yankword(char *ap
, char *wbuf
, char *separators
, int copypfx
);
62 static int same_name(char *n1
, char *n2
);
63 static struct name
*gexpand(struct name
*nlist
, struct grouphead
*gh
,
64 int metoo
, int ntype
);
65 static struct name
*put(struct name
*list
, struct name
*node
);
66 static struct name
*delname(struct name
*np
, char *name
);
69 * Allocate a single element of a name list,
70 * initialize its name field to the passed
74 nalloc(char *str
, enum gfield ntype
)
80 np
= (struct name
*)salloc(sizeof *np
);
85 np
->n_name
= savestr(skin(str
));
86 if (strcmp(np
->n_name
, str
)) {
89 mime_fromhdr(&in
, &out
, TD_ISPR
|TD_ICONV
);
90 np
->n_fullname
= savestr(out
.s
);
93 np
->n_fullname
= np
->n_name
;
94 } else if (ntype
& GSKIN
)
95 np
->n_fullname
= np
->n_name
= savestr(skin(str
));
97 np
->n_fullname
= np
->n_name
= savestr(str
);
102 * Find the tail of a list and return it.
105 tailof(struct name
*name
)
112 while (np
->n_flink
!= NULL
)
118 * Extract a list of names from a line,
119 * and make a list of names from it.
120 * Return the list or NULL if none found.
123 extract(char *line
, enum gfield ntype
)
125 return extract1(line
, ntype
, " \t,(", 0);
129 sextract(char *line
, enum gfield ntype
)
131 if (line
&& strpbrk(line
, ",\"\\(<"))
132 return extract1(line
, ntype
, ",", 1);
134 return extract(line
, ntype
);
138 extract1(char *line
, enum gfield ntype
, char *separators
, int copypfx
)
141 struct name
*top
, *np
, *t
;
143 if (line
== NULL
|| *line
== '\0')
148 nbuf
= ac_alloc(strlen(line
) + 1);
149 while ((cp
= yankword(cp
, nbuf
, separators
, copypfx
)) != NULL
) {
150 t
= nalloc(nbuf
, ntype
);
163 * Turn a list of names into a string of the same names.
166 detract(struct name
*np
, enum gfield ntype
)
173 comma
= ntype
& GCOMMA
;
178 if ((debug
|| value("debug")) && comma
)
179 fprintf(stderr
, catgets(catd
, CATSET
, 145,
180 "detract asked to insert commas\n"));
181 for (p
= np
; p
!= NULL
; p
= p
->n_flink
) {
182 if (ntype
&& (p
->n_type
& GMASK
) != ntype
)
184 s
+= strlen(p
->n_fullname
) + 1;
193 for (p
= np
; p
!= NULL
; p
= p
->n_flink
) {
194 if (ntype
&& (p
->n_type
& GMASK
) != ntype
)
196 cp
= sstpcpy(cp
, p
->n_fullname
);
197 if (comma
&& p
->n_flink
!= NULL
)
202 if (comma
&& *--cp
== ',')
208 * Grab a single word (liberal word)
209 * Throw away things between ()'s, and take anything between <>.
212 yankword(char *ap
, char *wbuf
, char *separators
, int copypfx
)
218 while (blankchar(*cp
& 0377) || *cp
== ',')
221 if ((cp
= nexttoken(cp
)) == NULL
)
227 while (*cp
&& (*wp
++ = *cp
++) != '>');
231 while (*cp
&& (incomm
|| !strchr(separators
, *cp
))) {
233 if (cp
== ap
|| *(cp
- 1) != '\\') {
239 } else if (cp
!= ap
) {
253 * For each recipient in the passed name list with a /
254 * in the name, append the message to the end of the named file
255 * and remove him from the recipient list.
257 * Recipients whose name begins with | are piped through the given
258 * program and removed.
262 outof(struct name
*names
, FILE *fo
, struct header
*hp
)
265 struct name
*np
, *top
;
276 if (!is_fileaddr(np
->n_name
) && np
->n_name
[0] != '|') {
280 ispipe
= np
->n_name
[0] == '|';
282 fname
= np
->n_name
+1;
284 fname
= expand(np
->n_name
);
287 * See if we have copied the complete message out yet.
294 if ((fout
= Ftemp(&tempEdit
, "Re", "w", 0600, 1))
296 perror(catgets(catd
, CATSET
, 146,
297 "temporary edit file"));
301 image
= open(tempEdit
, O_RDWR
);
305 perror(catgets(catd
, CATSET
, 147,
306 "temporary edit file"));
311 fcntl(image
, F_SETFD
, FD_CLOEXEC
);
312 fprintf(fout
, "From %s %s", myname
, date
);
314 while (lastc
= c
, (c
= getc(fo
)) != EOF
)
322 perror(catgets(catd
, CATSET
, 148,
323 "temporary edit file"));
328 * Now either copy "image" to the desired file
329 * or give it as the standard input to the desired
330 * program as appropriate.
340 * We can't really reuse the same image file,
341 * because multiple piped recipients will
342 * share the same lseek location and trample
345 if ((shell
= value("SHELL")) == NULL
)
348 sigaddset(&nset
, SIGHUP
);
349 sigaddset(&nset
, SIGINT
);
350 sigaddset(&nset
, SIGQUIT
);
351 pid
= start_command(shell
, &nset
,
352 image
, -1, "-c", fname
, NULL
);
360 if ((fout
= Zopen(fname
, "a", NULL
)) == NULL
) {
365 if ((f
= dup(image
)) < 0) {
369 fin
= Fdopen(f
, "r");
371 fprintf(stderr
, catgets(catd
, CATSET
, 149,
372 "Can't reopen image\n"));
378 while ((c
= getc(fin
)) != EOF
)
381 senderr
++, perror(fname
);
387 * In days of old we removed the entry from the
388 * the list; now for sake of header expansion
389 * we leave it in and mark it as deleted.
402 * Determine if the passed address is a local "send to file" address.
403 * If any of the network metacharacters precedes any slashes, it can't
404 * be a filename. We cheat with .'s to allow path names like ./...
407 is_fileaddr(char *name
)
411 if (strchr(name
, '@') != NULL
)
415 for (cp
= name
; *cp
; cp
++) {
416 if (*cp
== '!' || *cp
== '%')
425 same_name(char *n1
, char *n2
)
429 if (value("allnet") != NULL
) {
437 } while (c1
!= '\0' && c2
!= '\0' && c1
!= '@' && c2
!= '@');
440 return asccasecmp(n1
, n2
) == 0;
444 * Map all of the aliased users in the invoker's mailrc
445 * file and insert them into the list.
446 * Changed after all these months of service to recursively
447 * expand names (2/14/80).
451 usermap(struct name
*names
)
453 struct name
*new, *np
, *cp
;
454 struct grouphead
*gh
;
459 metoo
= (value("metoo") != NULL
);
461 if (np
->n_name
[0] == '\\') {
467 gh
= findgroup(np
->n_name
);
470 new = gexpand(new, gh
, metoo
, np
->n_type
);
479 * Recursively expand a group name. We limit the expansion to some
480 * fixed level to keep things from going haywire.
481 * Direct recursion is not expanded for convenience.
485 gexpand(struct name
*nlist
, struct grouphead
*gh
, int metoo
, int ntype
)
488 struct grouphead
*ngh
;
493 if (depth
> MAXEXP
) {
494 printf(catgets(catd
, CATSET
, 150,
495 "Expanding alias to depth larger than %d\n"), MAXEXP
);
499 for (gp
= gh
->g_list
; gp
!= NULL
; gp
= gp
->ge_link
) {
503 if (strcmp(cp
, gh
->g_name
) == 0)
505 if ((ngh
= findgroup(cp
)) != NULL
) {
506 nlist
= gexpand(nlist
, ngh
, metoo
, ntype
);
510 np
= nalloc(cp
, ntype
|GFULL
);
512 * At this point should allow to expand
513 * to self if only person in group
515 if (gp
== gh
->g_list
&& gp
->ge_link
== NULL
)
517 if (!metoo
&& same_name(cp
, myname
))
520 nlist
= put(nlist
, np
);
527 * Concatenate the two passed name lists, return the result.
530 cat(struct name
*n1
, struct name
*n2
)
545 * Unpack the name list onto a vector of strings.
546 * Return an error if the name list won't fit.
549 unpack(struct name
*np
)
553 int t
, extra
, metoo
, verbose
;
556 if ((t
= count(n
)) == 0)
557 panic(catgets(catd
, CATSET
, 151, "No names to unpack"));
559 * Compute the number of extra arguments we will need.
560 * We need at least two extra -- one for "mail" and one for
561 * the terminating 0 pointer. Additional spots may be needed
562 * to pass along -f to the host mailer.
566 metoo
= value("metoo") != NULL
;
569 verbose
= value("verbose") != NULL
;
573 top
= (char **)salloc((t
+ extra
) * sizeof *top
);
581 for (; n
!= NULL
; n
= n
->n_flink
)
582 if ((n
->n_type
& GDEL
) == 0)
589 * Remove all of the duplicates from the passed name list by
590 * insertion sorting them, then checking for dups.
591 * Return the head of the new list.
594 elide(struct name
*names
)
596 struct name
*np
, *t
, *new;
609 while (asccasecmp(t
->n_name
, np
->n_name
) < 0) {
610 if (t
->n_flink
== NULL
)
616 * If we ran out of t's, put the new entry after
617 * the current value of t.
620 if (asccasecmp(t
->n_name
, np
->n_name
) < 0) {
630 * Otherwise, put the new entry in front of the
631 * current t. If at the front of the list,
632 * the new guy becomes the new head of the list.
646 * The normal case -- we are inserting into the
647 * middle of the list.
653 x
->n_blink
= t
->n_blink
;
654 t
->n_blink
->n_flink
= x
;
659 * Now the list headed up by new is sorted.
660 * Go through it and remove duplicates.
666 while (t
->n_flink
!= NULL
&&
667 asccasecmp(np
->n_name
, t
->n_flink
->n_name
) == 0)
669 if (t
== np
|| t
== NULL
) {
675 * Now t points to the last entry with the same name
676 * as np. Make np point beyond t.
679 np
->n_flink
= t
->n_flink
;
680 if (t
->n_flink
!= NULL
)
681 t
->n_flink
->n_blink
= np
;
688 * Put another node onto a list of names and return
692 put(struct name
*list
, struct name
*node
)
694 node
->n_flink
= list
;
695 node
->n_blink
= NULL
;
697 list
->n_blink
= node
;
702 * Determine the number of undeleted elements in
703 * a name list and return it.
706 count(struct name
*np
)
710 for (c
= 0; np
!= NULL
; np
= np
->n_flink
)
711 if ((np
->n_type
& GDEL
) == 0)
717 * Delete the given name from a namelist.
720 delname(struct name
*np
, char *name
)
724 for (p
= np
; p
!= NULL
; p
= p
->n_flink
)
725 if (same_name(p
->n_name
, name
)) {
726 if (p
->n_blink
== NULL
) {
727 if (p
->n_flink
!= NULL
)
728 p
->n_flink
->n_blink
= NULL
;
732 if (p
->n_flink
== NULL
) {
733 if (p
->n_blink
!= NULL
)
734 p
->n_blink
->n_flink
= NULL
;
737 p
->n_blink
->n_flink
= p
->n_flink
;
738 p
->n_flink
->n_blink
= p
->n_blink
;
744 * Pretty print a name list
745 * Uncomment it if you need it.
750 prettyprint(struct name *name)
756 fprintf(stderr, "%s(%d) ", np->n_name, np->n_type);
759 fprintf(stderr, "\n");
764 delete_alternates(struct name
*np
)
769 np
= delname(np
, myname
);
771 for (ap
= altnames
; *ap
; ap
++)
772 np
= delname(np
, *ap
);
773 if ((xp
= sextract(value("from"), GEXTRA
|GSKIN
)) != NULL
)
775 np
= delname(np
, xp
->n_name
);
778 if ((xp
= sextract(value("replyto"), GEXTRA
|GSKIN
)) != NULL
)
780 np
= delname(np
, xp
->n_name
);
783 if ((xp
= sextract(value("sender"), GEXTRA
|GSKIN
)) != NULL
)
785 np
= delname(np
, xp
->n_name
);
792 is_myname(char *name
)
797 if (same_name(myname
, name
))
800 for (ap
= altnames
; *ap
; ap
++)
801 if (same_name(*ap
, name
))
803 if ((xp
= sextract(value("from"), GEXTRA
|GSKIN
)) != NULL
)
805 if (same_name(xp
->n_name
, name
))
809 if ((xp
= sextract(value("replyto"), GEXTRA
|GSKIN
)) != NULL
)
811 if (same_name(xp
->n_name
, name
))
815 if ((xp
= sextract(value("sender"), GEXTRA
|GSKIN
)) != NULL
)
817 if (same_name(xp
->n_name
, name
))