Merge commit '00f1a4f432b3d8aad1aa270e91c44c57f03ef407'
[unleashed.git] / usr / src / cmd / mail / gendeliv.c
blob945e53e48a2395238f7fe87a47f57ac18703f77c
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License, Version 1.0 only
6 * (the "License"). You may not use this file except in compliance
7 * with the License.
9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10 * or http://www.opensolaris.org/os/licensing.
11 * See the License for the specific language governing permissions
12 * and limitations under the License.
14 * When distributing Covered Code, include this CDDL HEADER in each
15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16 * If applicable, add the following below this CDDL HEADER, with the
17 * fields enclosed by brackets "[]" replaced with your own identifying
18 * information: Portions Copyright [yyyy] [name of copyright owner]
20 * CDDL HEADER END
22 /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
23 /* All Rights Reserved */
25 #pragma ident "%Z%%M% %I% %E% SMI"
27 #include "mail.h"
29 * generate delivery notification if required.
31 void gendeliv(fp, rc, name)
32 FILE *fp;
33 int rc;
34 char *name;
36 static char pn[] = "gendeliv";
37 register char *p;
38 char buf[1024], cbuf[256], ybuf[10];
39 register int i;
40 int didafflines = 0, didrcvlines = 0, suppress = 0, svopts = 0;
41 time_t ltmp;
42 register struct hdrs *hptr;
43 FILE *outfile;
45 Dout(pn, 0, "at entry, fp = o%lo, rc = %d,name = '%s'\n", (long)fp, rc, name);
46 if (fp == NULL) {
47 /* Want to send Positive delivery notification. Need to */
48 /* put selected header info from orig. msg aside to */
49 /* avoid confusion with header info in Delivery Rpt. */
50 Daffbytecnt = affbytecnt; affbytecnt = 0;
51 Daffcnt = affcnt; affcnt = 0;
52 Drcvbytecnt = rcvbytecnt; rcvbytecnt = 0;
54 hdrlines[H_DAFWDFROM].head = hdrlines[H_AFWDFROM].head;
55 hdrlines[H_DAFWDFROM].tail = hdrlines[H_AFWDFROM].tail;
56 hdrlines[H_AFWDFROM].head = NULL;
57 hdrlines[H_AFWDFROM].tail = NULL;
58 hdrlines[H_DRECEIVED].head = hdrlines[H_RECEIVED].head;
59 hdrlines[H_DRECEIVED].tail = hdrlines[H_RECEIVED].tail;
60 hdrlines[H_RECEIVED].head = NULL;
61 hdrlines[H_RECEIVED].tail = NULL;
62 hdrlines[H_DTCOPY].head = hdrlines[H_TCOPY].head;
63 hdrlines[H_DTCOPY].tail = hdrlines[H_TCOPY].tail;
64 hdrlines[H_TCOPY].head = NULL;
65 hdrlines[H_TCOPY].tail = NULL;
67 pushlist (H_TCOPY, HEAD, Rpath, FALSE);
69 if (rc == 0) {
70 /* Verify that positive delivery notification requested */
71 if (ckdlivopts(H_DTCOPY, &svopts) & NODELIVERY) {
72 Dout(pn, 0, "pos. notif. not requested\n");
73 goto rtrn;
75 } else {
76 /* Verify that negative delivery notification requested */
77 if (ckdlivopts(H_DTCOPY, &svopts) & IGNORE) {
78 Dout(pn, 0, "neg. notif. not requested\n");
79 goto rtrn;
82 if (fp == NULL) {
83 char *pargs[3];
84 pargs[0] = "mail";
85 pargs[1] = Rpath;
86 pargs[2] = 0;
87 if ((outfile = popenvp(pargs[0], pargs, "w", 1)) == NULL) {
88 /* Can't get pipe to mail. Just forget it..... */
89 Dout(pn, 0,"popenvp() failed\n");
90 goto rtrn;
92 } else {
93 outfile = fp;
96 /* get date string into buf for later...*/
97 ltmp = time((time_t)0);
98 strcpy(buf, asctime(gmtime(&ltmp)));
99 /* strip year out of date string, insert 'GMT', and put year back... */
100 p = strrchr(buf,' ');
101 strcpy(ybuf,++p);
102 *p = '\0';
103 strcat(buf,"GMT ");
104 strcat(buf, ybuf);
105 trimnl(buf);
107 fprintf(outfile,"%s 2\n", header[H_RVERS].tag);
108 fprintf(outfile,"%s %s\n", header[H_TCOPY].tag,
109 hdrlines[H_TCOPY].head->value);
110 fprintf(outfile,"%s %s\n", header[H_DATE].tag, buf);
111 dumprcv(ORDINARY, -1,&didrcvlines,&suppress,outfile);
112 dumpaff(ORDINARY, -1,&didafflines,&suppress,outfile);
113 fprintf(outfile,"Original-%s ", header[H_DATE].tag);
114 if ((hptr = hdrlines[H_DATE].head) != NULL) {
115 Dout(pn, 0,"date from H_DATE = '%s'\n", hptr->value);
116 fprintf(outfile,"%s\n", hptr->value);
117 } else {
118 /* If no H_DATE line in original message, use date */
119 /* in last UNIX H_FROM1 or H_FROM line */
120 if ((hptr = hdrlines[H_FROM1].tail) == NULL) {
121 hptr = hdrlines[H_FROM].tail;
123 Dout(pn, 0,"date from H_FROM = '%s'\n", hptr->value);
124 (void) strlcpy(buf, hptr->value, sizeof (buf));
125 /* Find date portion of line. */
126 /* Assumes line is of form - */
127 /* 'name_date_[remote_from_sys|forwarded_by_name]' */
128 if ((p = strchr(buf,' ')) == NULL) {
129 strcpy(buf, "No valid datestamp in original.");
130 } else {
131 (void) strlcpy(buf, p++, sizeof (buf));
132 /* Walk backwards from end of string to 3rd blank, */
133 /* and then check for 'remote from' or 'forwarded by' */
134 /* If either found, truncate there, else use entire */
135 /* string. */
136 p = buf + strlen(buf) - 1;
137 i = 0;
138 while (p > buf) {
139 if (*p == ' ') {
140 if (++i == 3) {
141 break;
144 p--;
146 if ((i != 3) || (p <= buf)) {
147 strcpy(buf, "No valid datestamp in original.");
148 } else {
149 if ((strncmp((p+1),"remote from", 11) == 0) ||
150 (strncmp((p+1),"forwarded by", 12) == 0)) {
151 *p = '\0';
155 fprintf(outfile,"%s\n", buf);
157 if ((hptr = hdrlines[H_SUBJ].head) != NULL) {
158 fprintf(outfile,"Original-%s %s\n",
159 header[H_SUBJ].tag, hptr->value);
161 if ((hptr = hdrlines[H_MSVC].head) != NULL) {
162 if ((strlen(hptr->value) != 4) ||
163 (casncmp("mail", hptr->value, 4) != 0)) {
164 fprintf(outfile,"Original-%s %s\n",
165 header[H_MSVC].tag, hptr->value);
168 if ((hptr = hdrlines[H_MTSID].head) != NULL) {
169 fprintf(outfile,"Confirming-%s <%s>\n",
170 header[H_MTSID].tag, hptr->value);
172 if ((hptr = hdrlines[H_UAID].head) != NULL) {
173 fprintf(outfile,"Confirming-%s <%s>\n",
174 header[H_UAID].tag, hptr->value);
176 cbuf[0] = '\0';
177 if ((hptr = hdrlines[H_DTCOPY].head) != NULL) {
178 /* Pick comment field off of ">To:" line and put into cbuf */
179 getcomment(hptr->value, cbuf);
181 if (rc == 0) {
182 fprintf(outfile,"Delivered-To: %s!%s %s on %s\n",
183 thissys, name, cbuf, buf);
184 } else {
185 (void) strlcpy (buf, name, sizeof (buf));
186 if ((p = strchr(buf,'!')) != NULL) {
187 *p = '\0';
189 fprintf(outfile,"Not-Delivered-To: %s!%s %s due to ",
190 thissys, buf,
191 /* if en-route-to, put comment there, else put it here*/
192 ((p == NULL) ? cbuf : ""));
193 mta_ercode(outfile);
194 if (ckdlivopts(H_DTCOPY, &svopts) & RETURN) {
195 fprintf(outfile," ORIGINAL MESSAGE ATTACHED\n");
198 if (error == E_FRWL) {
199 fprintf(outfile, frwlmsg, program, uval);
200 } else {
201 fprintf(outfile, " (%s: Error # %d '%s'",
202 program,error,errlist[error]);
203 if (error == E_SURG) {
204 fprintf(outfile,", rc = %d)\n",surg_rc);
205 fprintf(outfile,
206 " ======= Surrogate command =======\n");
207 fprintf(outfile," %s\n",
208 ((SURRcmdstr == NULL) ?
209 "" : SURRcmdstr));
210 /* Include stderr from surrogate, if any */
211 if (SURRerrfile) {
212 fprintf(outfile,
213 " ==== Start of stdout & stderr ===\n");
214 rewind (SURRerrfile);
215 while (fgets(buf, sizeof(buf), SURRerrfile) !=
216 NULL) {
217 fprintf(outfile," %s", buf);
219 if (buf[strlen(buf)-1] != '\n') {
220 fprintf(outfile,"\n");
222 fprintf(outfile,
223 " ==== End of stdout & stderr ===\n");
224 } else
225 fprintf(outfile,
226 " ==== stdout & stderr unavailable ===\n");
227 } else {
228 fprintf(outfile,")\n");
231 if (p != NULL) {
232 fprintf(outfile, "En-Route-To: %s %s\n", name, cbuf);
235 if ((hptr = hdrlines[H_DAFWDFROM].head) != NULL) {
236 while (hptr != NULL) {
237 fprintf(outfile,"Original-%s %s\n",
238 header[H_DAFWDFROM].tag, hptr->value);
239 hptr = hptr->next;
242 fprintf(outfile,"%s\n", header[H_EOH].tag);
243 if (fp == NULL) {
244 pclosevp(outfile);
246 Dout(pn, 5, "notification sent.\n");
248 rtrn:
249 /* Restore header info from original message. (see above and also */
250 /* goback()). */
251 clrhdr(H_TCOPY);
252 clrhdr(H_AFWDFROM);
253 clrhdr(H_RECEIVED);
254 affbytecnt = Daffbytecnt; Daffbytecnt = 0;
255 affcnt = Daffcnt; Daffcnt = 0;
256 rcvbytecnt = Drcvbytecnt; Drcvbytecnt = 0;
258 hdrlines[H_AFWDFROM].head = hdrlines[H_DAFWDFROM].head;
259 hdrlines[H_AFWDFROM].tail = hdrlines[H_DAFWDFROM].tail;
260 hdrlines[H_DAFWDFROM].head = NULL;
261 hdrlines[H_DAFWDFROM].tail = NULL;
262 hdrlines[H_RECEIVED].head = hdrlines[H_DRECEIVED].head;
263 hdrlines[H_RECEIVED].tail = hdrlines[H_DRECEIVED].tail;
264 hdrlines[H_DRECEIVED].head = NULL;
265 hdrlines[H_DRECEIVED].tail = NULL;
266 hdrlines[H_TCOPY].head = hdrlines[H_DTCOPY].head;
267 hdrlines[H_TCOPY].tail = hdrlines[H_DTCOPY].tail;
268 hdrlines[H_DTCOPY].head = NULL;
269 hdrlines[H_DTCOPY].tail = NULL;
271 return;