All: clean cmd_tab entry names, prefix with c_
[s-mailx.git] / sendout.c
blob4eef6080f699cc9a5116a22c0e0f841cac2e11a5
1 /*@ S-nail - a mail user agent derived from Berkeley Mail.
2 *@ Mail to others.
4 * Copyright (c) 2000-2004 Gunnar Ritter, Freiburg i. Br., Germany.
5 * Copyright (c) 2012 - 2014 Steffen "Daode" Nurpmeso <sdaoden@users.sf.net>.
6 */
7 /*
8 * Copyright (c) 1980, 1993
9 * The Regents of the University of California. All rights reserved.
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
13 * are met:
14 * 1. Redistributions of source code must retain the above copyright
15 * notice, this list of conditions and the following disclaimer.
16 * 2. Redistributions in binary form must reproduce the above copyright
17 * notice, this list of conditions and the following disclaimer in the
18 * documentation and/or other materials provided with the distribution.
19 * 3. All advertising materials mentioning features or use of this software
20 * must display the following acknowledgement:
21 * This product includes software developed by the University of
22 * California, Berkeley and its contributors.
23 * 4. Neither the name of the University nor the names of its contributors
24 * may be used to endorse or promote products derived from this software
25 * without specific prior written permission.
27 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
28 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
29 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
30 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
31 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
32 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
33 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
34 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
35 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
36 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 * SUCH DAMAGE.
40 #ifndef HAVE_AMALGAMATION
41 # include "nail.h"
42 #endif
44 #include <fcntl.h>
46 #define INFIX_BUF \
47 ((1024 / B64_ENCODE_INPUT_PER_LINE) * B64_ENCODE_INPUT_PER_LINE)
49 static char * _sendout_boundary;
50 static bool_t _sendout_error;
52 static enum okay _putname(char const *line, enum gfield w,
53 enum sendaction action, size_t *gotcha,
54 char const *prefix, FILE *fo, struct name **xp);
56 /* Get an encoding flag based on the given string */
57 static char const * _get_encoding(const enum conversion convert);
59 /* Write an attachment to the file buffer, converting to MIME */
60 static int _attach_file(struct attachment *ap, FILE *fo);
61 static int __attach_file(struct attachment *ap, FILE *fo);
63 static char const ** _prepare_mta_args(struct name *to, struct header *hp);
65 /* Fix the header by glopping all of the expanded names from the distribution
66 * list into the appropriate fields */
67 static struct name * fixhead(struct header *hp, struct name *tolist);
69 /* Put the signature file at fo. TODO layer rewrite: *integrate in body*!! */
70 static int put_signature(FILE *fo, int convert);
72 /* Attach a message to the file buffer */
73 static int attach_message(struct attachment *ap, FILE *fo);
75 /* Generate the body of a MIME multipart message */
76 static int make_multipart(struct header *hp, int convert, FILE *fi,
77 FILE *fo, char const *contenttype, char const *charset);
79 /* Prepend a header in front of the collected stuff and return the new file */
80 static FILE * infix(struct header *hp, FILE *fi);
82 /* Save the outgoing mail on the passed file */
83 static int savemail(char const *name, FILE *fi);
85 /* Send mail to a bunch of user names. The interface is through mail() */
86 static int sendmail_internal(void *v, int recipient_record);
88 static enum okay transfer(struct name *to, FILE *input, struct header *hp);
90 /* Start the MTA mailing to namelist and stdin redirected to input */
91 static enum okay start_mta(struct name *to, FILE *input, struct header *hp);
93 /* Record outgoing mail if instructed to do so; in *record* unless to is set */
94 static enum okay mightrecord(FILE *fp, struct name *to);
96 /* Create a Message-Id: header field. Use either host name or from address */
97 static void message_id(FILE *fo, struct header *hp);
99 /* Format the given header line to not exceed 72 characters */
100 static int fmt(char const *str, struct name *np, FILE *fo, int comma,
101 int dropinvalid, int domime);
103 /* Rewrite a message for resending, adding the Resent-Headers */
104 static int infix_resend(FILE *fi, FILE *fo, struct message *mp,
105 struct name *to, int add_resent);
107 static enum okay
108 _putname(char const *line, enum gfield w, enum sendaction action,
109 size_t *gotcha, char const *prefix, FILE *fo, struct name **xp)
111 struct name *np;
112 enum okay rv = STOP;
113 NYD_ENTER;
115 np = lextract(line, GEXTRA | GFULL);
116 if (xp)
117 *xp = np;
118 if (np == NULL)
120 else if (fmt(prefix, np, fo, w & GCOMMA, 0, (action != SEND_TODISP)))
121 rv = OKAY;
122 else if (gotcha)
123 ++(*gotcha);
124 NYD_LEAVE;
125 return rv;
128 static char const *
129 _get_encoding(enum conversion const convert)
131 char const *rv;
132 NYD_ENTER;
134 switch (convert) {
135 case CONV_7BIT: rv = "7bit"; break;
136 case CONV_8BIT: rv = "8bit"; break;
137 case CONV_TOQP: rv = "quoted-printable"; break;
138 case CONV_TOB64: rv = "base64"; break;
139 default: rv = NULL; break;
141 NYD_LEAVE;
142 return rv;
145 static int
146 _attach_file(struct attachment *ap, FILE *fo)
148 /* TODO of course, the MIME classification needs to performed once
149 * TODO only, not for each and every charset anew ... ;-// */
150 int err = 0;
151 char *charset_iter_orig[2];
152 long offs;
153 NYD_ENTER;
155 /* Is this already in target charset? */
156 if (ap->a_conv == AC_TMPFILE) {
157 err = __attach_file(ap, fo);
158 Fclose(ap->a_tmpf);
159 goto jleave;
162 /* We "consume" *ap*, so directly adjust it as we need it */
163 if (ap->a_conv == AC_FIX_INCS)
164 ap->a_charset = ap->a_input_charset;
166 if ((offs = ftell(fo)) < 0) {
167 err = EIO;
168 goto jleave;
171 charset_iter_recurse(charset_iter_orig);
172 for (charset_iter_reset(NULL); charset_iter_next() != NULL;) {
173 err = __attach_file(ap, fo);
174 if (err == 0 || (err != EILSEQ && err != EINVAL))
175 break;
176 clearerr(fo);
177 if (fseek(fo, offs, SEEK_SET) == -1) {
178 err = EIO;
179 break;
181 if (ap->a_conv != AC_DEFAULT) {
182 err = EILSEQ;
183 break;
185 ap->a_charset = NULL;
187 charset_iter_restore(charset_iter_orig);
188 jleave:
189 NYD_LEAVE;
190 return err;
193 static int
194 __attach_file(struct attachment *ap, FILE *fo) /* XXX linelength */
196 int err = 0, do_iconv;
197 FILE *fi;
198 char const *charset;
199 enum conversion convert;
200 char *buf;
201 size_t bufsize, lncnt, inlen;
202 NYD_ENTER;
204 /* Either charset-converted temporary file, or plain path */
205 if (ap->a_conv == AC_TMPFILE) {
206 fi = ap->a_tmpf;
207 assert(ftell(fi) == 0x0l);
208 } else if ((fi = Fopen(ap->a_name, "r")) == NULL) {
209 err = errno;
210 perror(ap->a_name);
211 goto jleave;
214 /* MIME part header for attachment */
215 { char const *bn = ap->a_name, *ct;
217 if ((ct = strrchr(bn, '/')) != NULL)
218 bn = ++ct;
219 ct = ap->a_content_type;
220 charset = ap->a_charset;
221 convert = mime_classify_file(fi, (char const**)&ct, &charset, &do_iconv);
222 if (charset == NULL || ap->a_conv == AC_FIX_INCS ||
223 ap->a_conv == AC_TMPFILE)
224 do_iconv = 0;
226 if (fprintf(fo, "\n--%s\nContent-Type: %s", _sendout_boundary, ct) == -1)
227 goto jerr_header;
229 if (charset == NULL) {
230 if (putc('\n', fo) == EOF)
231 goto jerr_header;
232 } else if (fprintf(fo, "; charset=%s\n", charset) == -1)
233 goto jerr_header;
235 if (fprintf(fo, "Content-Transfer-Encoding: %s\n"
236 "Content-Disposition: %s;\n filename=\"",
237 _get_encoding(convert), ap->a_content_disposition) == -1)
238 goto jerr_header;
239 if (xmime_write(bn, strlen(bn), fo, CONV_TOHDR, TD_NONE, NULL) < 0)
240 goto jerr_header;
241 if (fwrite("\"\n", sizeof(char), 2, fo) != 2 * sizeof(char))
242 goto jerr_header;
244 if ((bn = ap->a_content_id) != NULL &&
245 fprintf(fo, "Content-ID: %s\n", bn) == -1)
246 goto jerr_header;
248 if ((bn = ap->a_content_description) != NULL &&
249 fprintf(fo, "Content-Description: %s\n", bn) == -1)
250 goto jerr_header;
252 if (putc('\n', fo) == EOF) {
253 jerr_header:
254 err = errno;
255 goto jerr_fclose;
259 #ifdef HAVE_ICONV
260 if (iconvd != (iconv_t)-1)
261 n_iconv_close(iconvd);
262 if (do_iconv) {
263 char const *tcs = charset_get_lc();
264 if (asccasecmp(charset, tcs) &&
265 (iconvd = n_iconv_open(charset, tcs)) == (iconv_t)-1 &&
266 (err = errno) != 0) {
267 if (err == EINVAL)
268 fprintf(stderr, tr(179, "Cannot convert from %s to %s\n"),
269 tcs, charset);
270 else
271 perror("iconv_open");
272 goto jerr_fclose;
275 #endif
277 bufsize = INFIX_BUF;
278 buf = smalloc(bufsize);
279 if (convert == CONV_TOQP
280 #ifdef HAVE_ICONV
281 || iconvd != (iconv_t)-1
282 #endif
284 lncnt = fsize(fi);
285 for (;;) {
286 if (convert == CONV_TOQP
287 #ifdef HAVE_ICONV
288 || iconvd != (iconv_t)-1
289 #endif
291 if (fgetline(&buf, &bufsize, &lncnt, &inlen, fi, 0) == NULL)
292 break;
293 } else if ((inlen = fread(buf, sizeof *buf, bufsize, fi)) == 0)
294 break;
295 if (xmime_write(buf, inlen, fo, convert, TD_ICONV, NULL) < 0) {
296 err = errno;
297 goto jerr;
300 if (ferror(fi))
301 err = EDOM;
302 jerr:
303 free(buf);
304 jerr_fclose:
305 if (ap->a_conv != AC_TMPFILE)
306 Fclose(fi);
307 jleave:
308 NYD_LEAVE;
309 return err;
312 static char const **
313 _prepare_mta_args(struct name *to, struct header *hp)
315 size_t j, i;
316 char const **args;
317 NYD_ENTER;
319 i = 4 + smopts_count + 2 + count(to) + 1;
320 args = salloc(i * sizeof(char*));
322 args[0] = ok_vlook(sendmail_progname);
323 if (args[0] == NULL || *args[0] == '\0')
324 args[0] = SENDMAIL_PROGNAME;
326 args[1] = "-i";
327 i = 2;
328 if (ok_blook(metoo))
329 args[i++] = "-m";
330 if (options & OPT_VERBOSE)
331 args[i++] = "-v";
333 for (j = 0; j < smopts_count; ++j, ++i)
334 args[i] = smopts[j];
336 /* -r option? We may only pass skinned addresses, which is why we do
337 * not simply call myorigin() (TODO myorigin shouldn't fullname!) */
338 if (options & OPT_r_FLAG) {
339 char const *froma;
341 if (option_r_arg[0] != '\0')
342 froma = option_r_arg;
343 else if (hp->h_from != NULL)
344 froma = hp->h_from->n_name;
345 else
346 froma = myorigin(hp);
347 if (froma != NULL) {
348 args[i++] = "-r";
349 args[i++] = froma;
353 /* Receivers follow */
354 for (; to != NULL; to = to->n_flink)
355 if (!(to->n_type & GDEL))
356 args[i++] = to->n_name;
357 args[i] = NULL;
358 NYD_LEAVE;
359 return args;
362 static struct name *
363 fixhead(struct header *hp, struct name *tolist)
365 struct name **npp, *np;
366 NYD_ENTER;
368 tolist = elide(tolist);
370 hp->h_to = hp->h_cc = hp->h_bcc = NULL;
371 for (np = tolist; np != NULL; np = np->n_flink) {
372 switch (np->n_type & (GDEL | GMASK)) {
373 case GTO: npp = &hp->h_to; break;
374 case GCC: npp = &hp->h_cc; break;
375 case GBCC: npp = &hp->h_bcc; break;
376 default: continue;
378 *npp = cat(*npp, ndup(np, np->n_type | GFULL));
380 NYD_LEAVE;
381 return tolist;
384 static int
385 put_signature(FILE *fo, int convert)
387 char buf[INFIX_BUF], *sig, c = '\n';
388 FILE *fsig;
389 size_t sz;
390 int rv;
391 NYD_ENTER;
393 if ((sig = ok_vlook(signature)) == NULL || *sig == '\0') {
394 rv = 0;
395 goto jleave;
397 rv = -1;
399 if ((sig = file_expand(sig)) == NULL)
400 goto jleave;
402 if ((fsig = Fopen(sig, "r")) == NULL) {
403 perror(sig);
404 goto jleave;
406 while ((sz = fread(buf, sizeof *buf, INFIX_BUF, fsig)) != 0) {
407 c = buf[sz - 1];
408 if (xmime_write(buf, sz, fo, convert, TD_NONE, NULL) < 0)
409 goto jerr;
411 if (ferror(fsig)) {
412 jerr:
413 perror(sig);
414 Fclose(fsig);
415 goto jleave;
417 Fclose(fsig);
418 if (c != '\n')
419 putc('\n', fo);
421 rv = 0;
422 jleave:
423 NYD_LEAVE;
424 return rv;
427 static int
428 attach_message(struct attachment *ap, FILE *fo)
430 struct message *mp;
431 char const *ccp;
432 int rv;
433 NYD_ENTER;
435 fprintf(fo, "\n--%s\nContent-Type: message/rfc822\n"
436 "Content-Disposition: inline\n", _sendout_boundary);
437 if ((ccp = ap->a_content_description) != NULL)
438 fprintf(fo, "Content-Description: %s\n", ccp);
439 fputc('\n', fo);
441 mp = &message[ap->a_msgno - 1];
442 touch(mp);
443 rv = (sendmp(mp, fo, 0, NULL, SEND_RFC822, NULL) < 0) ? -1 : 0;
444 NYD_LEAVE;
445 return rv;
448 static int
449 make_multipart(struct header *hp, int convert, FILE *fi, FILE *fo,
450 char const *contenttype, char const *charset)
452 struct attachment *att;
453 int rv = -1;
454 NYD_ENTER;
456 fputs("This is a multi-part message in MIME format.\n", fo);
457 if (fsize(fi) != 0) {
458 char *buf;
459 size_t sz, bufsize, cnt;
461 fprintf(fo, "\n--%s\n", _sendout_boundary);
462 fprintf(fo, "Content-Type: %s", contenttype);
463 if (charset != NULL)
464 fprintf(fo, "; charset=%s", charset);
465 fprintf(fo, "\nContent-Transfer-Encoding: %s\n"
466 "Content-Disposition: inline\n\n", _get_encoding(convert));
468 buf = smalloc(bufsize = INFIX_BUF);
469 if (convert == CONV_TOQP
470 #ifdef HAVE_ICONV
471 || iconvd != (iconv_t)-1
472 #endif
474 fflush(fi);
475 cnt = fsize(fi);
477 for (;;) {
478 if (convert == CONV_TOQP
479 #ifdef HAVE_ICONV
480 || iconvd != (iconv_t)-1
481 #endif
483 if (fgetline(&buf, &bufsize, &cnt, &sz, fi, 0) == NULL)
484 break;
485 } else if ((sz = fread(buf, sizeof *buf, bufsize, fi)) == 0)
486 break;
488 if (xmime_write(buf, sz, fo, convert, TD_ICONV, NULL) < 0) {
489 free(buf);
490 goto jleave;
493 free(buf);
495 if (ferror(fi))
496 goto jleave;
497 if (charset != NULL)
498 put_signature(fo, convert);
501 for (att = hp->h_attach; att != NULL; att = att->a_flink) {
502 if (att->a_msgno) {
503 if (attach_message(att, fo) != 0)
504 goto jleave;
505 } else if (_attach_file(att, fo) != 0)
506 goto jleave;
509 /* the final boundary with two attached dashes */
510 fprintf(fo, "\n--%s--\n", _sendout_boundary);
511 rv = 0;
512 jleave:
513 NYD_LEAVE;
514 return rv;
517 static FILE *
518 infix(struct header *hp, FILE *fi) /* TODO check */
520 FILE *nfo, *nfi = NULL;
521 char *tempMail;
522 char const *contenttype, *charset = NULL;
523 enum conversion convert;
524 int do_iconv = 0, err;
525 #ifdef HAVE_ICONV
526 char const *tcs, *convhdr = NULL;
527 #endif
528 NYD_ENTER;
530 if ((nfo = Ftmp(&tempMail, "infix", OF_WRONLY | OF_HOLDSIGS | OF_REGISTER,
531 0600)) == NULL) {
532 perror(tr(178, "temporary mail file"));
533 goto jleave;
535 if ((nfi = Fopen(tempMail, "r")) == NULL) {
536 perror(tempMail);
537 Fclose(nfo);
539 Ftmp_release(&tempMail);
540 if (nfi == NULL)
541 goto jleave;
543 contenttype = "text/plain"; /* XXX mail body - always text/plain, want XX? */
544 convert = mime_classify_file(fi, &contenttype, &charset, &do_iconv);
546 #ifdef HAVE_ICONV
547 tcs = charset_get_lc();
548 if ((convhdr = need_hdrconv(hp, GTO | GSUBJECT | GCC | GBCC | GIDENT))) {
549 if (iconvd != (iconv_t)-1) /* XXX */
550 n_iconv_close(iconvd);
551 if (asccasecmp(convhdr, tcs) != 0 &&
552 (iconvd = n_iconv_open(convhdr, tcs)) == (iconv_t)-1 &&
553 (err = errno) != 0)
554 goto jiconv_err;
556 #endif
557 if (puthead(hp, nfo,
558 (GTO | GSUBJECT | GCC | GBCC | GNL | GCOMMA | GUA | GMIME | GMSGID |
559 GIDENT | GREF | GDATE), SEND_MBOX, convert, contenttype, charset))
560 goto jerr;
561 #ifdef HAVE_ICONV
562 if (iconvd != (iconv_t)-1)
563 n_iconv_close(iconvd);
564 #endif
566 #ifdef HAVE_ICONV
567 if (do_iconv && charset != NULL) { /*TODO charset->mime_classify_file*/
568 if (asccasecmp(charset, tcs) != 0 &&
569 (iconvd = n_iconv_open(charset, tcs)) == (iconv_t)-1 &&
570 (err = errno) != 0) {
571 jiconv_err:
572 if (err == EINVAL)
573 fprintf(stderr, tr(179, "Cannot convert from %s to %s\n"),
574 tcs, charset);
575 else
576 perror("iconv_open");
577 goto jerr;
580 #endif
582 if (hp->h_attach != NULL) {
583 if (make_multipart(hp, convert, fi, nfo, contenttype, charset) != 0)
584 goto jerr;
585 } else {
586 size_t sz, bufsize, cnt;
587 char *buf;
589 if (convert == CONV_TOQP
590 #ifdef HAVE_ICONV
591 || iconvd != (iconv_t)-1
592 #endif
594 fflush(fi);
595 cnt = fsize(fi);
597 buf = smalloc(bufsize = INFIX_BUF);
598 for (err = 0;;) {
599 if (convert == CONV_TOQP
600 #ifdef HAVE_ICONV
601 || iconvd != (iconv_t)-1
602 #endif
604 if (fgetline(&buf, &bufsize, &cnt, &sz, fi, 0) == NULL)
605 break;
606 } else if ((sz = fread(buf, sizeof *buf, bufsize, fi)) == 0)
607 break;
608 if (xmime_write(buf, sz, nfo, convert, TD_ICONV, NULL) < 0) {
609 err = 1;
610 break;
613 free(buf);
615 if (err || ferror(fi)) {
616 jerr:
617 Fclose(nfo);
618 Fclose(nfi);
619 #ifdef HAVE_ICONV
620 if (iconvd != (iconv_t)-1)
621 n_iconv_close(iconvd);
622 #endif
623 nfi = NULL;
624 goto jleave;
626 if (charset != NULL)
627 put_signature(nfo, convert); /* XXX if (text/) !! */
630 #ifdef HAVE_ICONV
631 if (iconvd != (iconv_t)-1)
632 n_iconv_close(iconvd);
633 #endif
635 fflush(nfo);
636 if ((err = ferror(nfo)))
637 perror(tr(180, "temporary mail file"));
638 Fclose(nfo);
639 if (!err) {
640 fflush_rewind(nfi);
641 Fclose(fi);
642 } else {
643 Fclose(nfi);
644 nfi = NULL;
646 jleave:
647 NYD_LEAVE;
648 return nfi;
651 static int
652 savemail(char const *name, FILE *fi)
654 FILE *fo;
655 char *buf;
656 size_t bufsize, buflen, cnt;
657 int prependnl = 0, rv = -1;
658 NYD_ENTER;
660 buf = smalloc(bufsize = LINESIZE);
662 if ((fo = Zopen(name, "a+", NULL)) == NULL) {
663 if ((fo = Zopen(name, "wx", NULL)) == NULL) {
664 perror(name);
665 goto jleave;
667 } else {
668 if (fseek(fo, -2L, SEEK_END) == 0) {
669 switch (fread(buf, sizeof *buf, 2, fo)) {
670 case 2:
671 if (buf[1] != '\n') {
672 prependnl = 1;
673 break;
675 /* FALLTHRU */
676 case 1:
677 if (buf[0] != '\n')
678 prependnl = 1;
679 break;
680 default:
681 if (ferror(fo)) {
682 perror(name);
683 goto jleave;
686 if (prependnl) {
687 putc('\n', fo);
689 fflush(fo);
693 fprintf(fo, "From %s %s", myname, time_current.tc_ctime);
694 fflush_rewind(fi);
695 cnt = fsize(fi);
696 buflen = 0;
697 while (fgetline(&buf, &bufsize, &cnt, &buflen, fi, 0) != NULL) {
698 #ifdef HAVE_DEBUG /* TODO assert legacy */
699 assert(!is_head(buf, buflen));
700 #else
701 if (is_head(buf, buflen))
702 putc('>', fo);
703 #endif
704 fwrite(buf, sizeof *buf, buflen, fo);
706 if (buflen && *(buf + buflen - 1) != '\n')
707 putc('\n', fo);
708 putc('\n', fo);
709 fflush(fo);
711 rv = 0;
712 if (ferror(fo)) {
713 perror(name);
714 rv = -1;
716 if (Fclose(fo) != 0)
717 rv = -1;
718 fflush_rewind(fi);
719 jleave:
720 free(buf);
721 NYD_LEAVE;
722 return rv;
725 static int
726 sendmail_internal(void *v, int recipient_record)
728 struct header head;
729 char *str = v;
730 int rv;
731 NYD_ENTER;
733 memset(&head, 0, sizeof head);
734 head.h_to = lextract(str, GTO | GFULL);
735 rv = mail1(&head, 0, NULL, NULL, recipient_record, 0);
736 NYD_LEAVE;
737 return rv;
740 static enum okay
741 transfer(struct name *to, FILE *input, struct header *hp)
743 char o[LINESIZE], *cp;
744 struct name *np;
745 int cnt = 0;
746 enum okay rv = OKAY;
747 NYD_ENTER;
749 np = to;
750 while (np) {
751 snprintf(o, sizeof o, "smime-encrypt-%s", np->n_name);/* XXX */
752 if ((cp = vok_vlook(o)) != NULL) {
753 #ifdef HAVE_SSL
754 struct name *nt;
755 FILE *ef;
756 if ((ef = smime_encrypt(input, cp, np->n_name)) != 0) {
757 nt = ndup(np, np->n_type & ~(GFULL | GSKIN));
758 rv = start_mta(nt, ef, hp);
759 Fclose(ef);
760 } else {
761 #else
762 fprintf(stderr, tr(225, "No SSL support compiled in.\n"));
763 rv = STOP;
764 #endif
765 fprintf(stderr, tr(38, "Message not sent to <%s>\n"), np->n_name);
766 _sendout_error = TRU1;
767 #ifdef HAVE_SSL
769 #endif
770 rewind(input);
772 if (np->n_flink)
773 np->n_flink->n_blink = np->n_blink;
774 if (np->n_blink)
775 np->n_blink->n_flink = np->n_flink;
776 if (np == to)
777 to = np->n_flink;
778 np = np->n_flink;
779 } else {
780 ++cnt;
781 np = np->n_flink;
785 if (cnt)
786 if (ok_blook(smime_force_encryption) || start_mta(to, input, hp) != OKAY)
787 rv = STOP;
788 NYD_LEAVE;
789 return rv;
792 static enum okay
793 start_mta(struct name *to, FILE *input, struct header *hp)
795 #ifdef HAVE_SMTP
796 char *user = NULL, *password = NULL, *skinned = NULL;
797 #endif
798 char const **args = NULL, **t, *mta;
799 char *smtp;
800 pid_t pid;
801 sigset_t nset;
802 enum okay rv = STOP;
803 NYD_ENTER;
804 UNUSED(hp);
806 if ((smtp = ok_vlook(smtp)) == NULL) {
807 if ((mta = ok_vlook(sendmail)) != NULL) {
808 if ((mta = file_expand(mta)) == NULL)
809 goto jstop;
810 } else
811 mta = SENDMAIL;
813 args = _prepare_mta_args(to, hp);
814 if (options & OPT_DEBUG) {
815 printf(tr(181, "Sendmail arguments:"));
816 for (t = args; *t != NULL; ++t)
817 printf(" \"%s\"", *t);
818 printf("\n");
819 rv = OKAY;
820 goto jleave;
822 } else {
823 mta = NULL; /* Silence cc */
824 #ifndef HAVE_SMTP
825 fputs(tr(194, "No SMTP support compiled in.\n"), stderr);
826 goto jstop;
827 #else
828 skinned = skin(myorigin(hp));
829 if ((user = smtp_auth_var("-user", skinned)) != NULL &&
830 (password = smtp_auth_var("-password", skinned)) == NULL)
831 password = getpassword(NULL);
832 #endif
835 /* Fork, set up the temporary mail file as standard input for "mail", and
836 * exec with the user list we generated far above */
837 if ((pid = fork()) == -1) {
838 perror("fork");
839 jstop:
840 savedeadletter(input, 0);
841 _sendout_error = TRU1;
842 goto jleave;
844 if (pid == 0) {
845 sigemptyset(&nset);
846 sigaddset(&nset, SIGHUP);
847 sigaddset(&nset, SIGINT);
848 sigaddset(&nset, SIGQUIT);
849 sigaddset(&nset, SIGTSTP);
850 sigaddset(&nset, SIGTTIN);
851 sigaddset(&nset, SIGTTOU);
852 freopen("/dev/null", "r", stdin);
853 #ifdef HAVE_SMTP
854 if (smtp != NULL) {
855 prepare_child(&nset, 0, 1);
856 if (smtp_mta(smtp, to, input, hp, user, password, skinned) == 0)
857 _exit(0);
858 } else {
859 #endif
860 prepare_child(&nset, fileno(input), -1);
861 /* If *record* is set then savemail() will move the file position;
862 * it'll call rewind(), but that may optimize away the systemcall if
863 * possible, and since dup2() shares the position with the original FD
864 * the MTA may end up reading nothing */
865 lseek(0, 0, SEEK_SET);
866 execv(mta, UNCONST(args));
867 perror(mta);
868 #ifdef HAVE_SMTP
870 #endif
871 savedeadletter(input, 1);
872 fputs(tr(182, ". . . message not sent.\n"), stderr);
873 _exit(1);
875 if ((options & (OPT_DEBUG | OPT_VERBOSE | OPT_BATCH_FLAG)) ||
876 ok_blook(sendwait)) {
877 if (wait_child(pid, NULL))
878 rv = OKAY;
879 else
880 _sendout_error = TRU1;
881 } else {
882 rv = OKAY;
883 free_child(pid);
885 jleave:
886 NYD_LEAVE;
887 return rv;
890 static enum okay
891 mightrecord(FILE *fp, struct name *to)
893 char *cp, *cq;
894 char const *ep;
895 enum okay rv = OKAY;
896 NYD_ENTER;
898 if (to != NULL) {
899 cp = savestr(skinned_name(to));
900 for (cq = cp; *cq != '\0' && *cq != '@'; ++cq)
902 *cq = '\0';
903 } else
904 cp = ok_vlook(record);
906 if (cp != NULL) {
907 if ((ep = expand(cp)) == NULL) {
908 ep = "NULL";
909 goto jbail;
912 if (ok_blook(outfolder) && *ep != '/' && *ep != '+' &&
913 which_protocol(ep) == PROTO_FILE) {
914 size_t i = strlen(cp);
915 cq = salloc(i + 1 +1);
916 cq[0] = '+';
917 memcpy(cq + 1, cp, i + 1);
918 cp = cq;
919 if ((ep = expand(cp)) == NULL) { /* TODO file_expand() possible? */
920 ep = "NULL";
921 goto jbail;
925 if (savemail(ep, fp) != 0) {
926 jbail:
927 fprintf(stderr, tr(285, "Failed to save message in %s - "
928 "message not sent\n"), ep);
929 exit_status |= 1;
930 savedeadletter(fp, 1);
931 rv = STOP;
934 NYD_LEAVE;
935 return rv;
938 static void
939 message_id(FILE *fo, struct header *hp)
941 char const *h;
942 size_t rl;
943 struct tm *tmp;
944 NYD_ENTER;
946 if (ok_blook(message_id_disable))
947 goto jleave;
949 if ((h = ok_vlook(hostname)) != NULL)
950 rl = 24;
951 else if ((h = skin(myorigin(hp))) != NULL && strchr(h, '@') != NULL)
952 rl = 16;
953 else
954 /* Up to MTA */
955 goto jleave;
957 tmp = &time_current.tc_gm;
958 fprintf(fo, "Message-ID: <%04d%02d%02d%02d%02d%02d.%s%c%s>\n",
959 tmp->tm_year + 1900, tmp->tm_mon + 1, tmp->tm_mday,
960 tmp->tm_hour, tmp->tm_min, tmp->tm_sec,
961 getrandstring(rl), (rl == 16 ? '%' : '@'), h);
962 jleave:
963 NYD_LEAVE;
966 static int
967 fmt(char const *str, struct name *np, FILE *fo, int flags, int dropinvalid,
968 int domime)
970 enum {
971 m_INIT = 1<<0,
972 m_COMMA = 1<<1,
973 m_NOPF = 1<<2,
974 m_CSEEN = 1<<3
975 } m = (flags & GCOMMA) ? m_COMMA : 0;
976 ssize_t col, len;
977 int rv = 1;
978 NYD_ENTER;
980 col = strlen(str);
981 if (col) {
982 fwrite(str, sizeof *str, col, fo);
983 if (flags & GFILES)
984 goto jstep;
985 if (col == 9 && !asccasecmp(str, "reply-to:")) {
986 m |= m_NOPF;
987 goto jstep;
989 if (ok_blook(add_file_recipients))
990 goto jstep;
991 if ((col == 3 && (!asccasecmp(str, "to:") || !asccasecmp(str, "cc:"))) ||
992 (col == 4 && !asccasecmp(str, "bcc:")) ||
993 (col == 10 && !asccasecmp(str, "Resent-To:")))
994 m |= m_NOPF;
996 jstep:
997 for (; np != NULL; np = np->n_flink) {
998 if ((m & m_NOPF) && is_fileorpipe_addr(np))
999 continue;
1000 if (is_addr_invalid(np, !dropinvalid)) {
1001 if (dropinvalid)
1002 continue;
1003 else
1004 goto jleave;
1006 if ((m & (m_INIT | m_COMMA)) == (m_INIT | m_COMMA)) {
1007 putc(',', fo);
1008 m |= m_CSEEN;
1009 ++col;
1011 len = strlen(np->n_fullname);
1012 ++col; /* The separating space */
1013 if ((m & m_INIT) && col > 1 && col + len > 72) {
1014 fputs("\n ", fo);
1015 col = 1;
1016 m &= ~m_CSEEN;
1017 } else
1018 putc(' ', fo);
1019 m = (m & ~m_CSEEN) | m_INIT;
1020 len = xmime_write(np->n_fullname, len, fo,
1021 (domime ? CONV_TOHDR_A : CONV_NONE), TD_ICONV, NULL);
1022 if (len < 0)
1023 goto jleave;
1024 col += len;
1026 putc('\n', fo);
1027 rv = 0;
1028 jleave:
1029 NYD_LEAVE;
1030 return rv;
1033 static int
1034 infix_resend(FILE *fi, FILE *fo, struct message *mp, struct name *to,
1035 int add_resent)
1037 size_t cnt, c, bufsize = 0;
1038 char *buf = NULL;
1039 char const *cp;
1040 struct name *fromfield = NULL, *senderfield = NULL;
1041 int rv = 1;
1042 NYD_ENTER;
1044 cnt = mp->m_size;
1046 /* Write the Resent-Fields */
1047 if (add_resent) {
1048 fputs("Resent-", fo);
1049 mkdate(fo, "Date");
1050 if ((cp = myaddrs(NULL)) != NULL) {
1051 if (_putname(cp, GCOMMA, SEND_MBOX, NULL, "Resent-From:", fo,
1052 &fromfield))
1053 goto jleave;
1055 if ((cp = ok_vlook(sender)) != NULL) {
1056 if (_putname(cp, GCOMMA, SEND_MBOX, NULL, "Resent-Sender:", fo,
1057 &senderfield))
1058 goto jleave;
1060 if (fmt("Resent-To:", to, fo, 1, 1, 0))
1061 goto jleave;
1062 if ((cp = ok_vlook(stealthmua)) == NULL || !strcmp(cp, "noagent")) {
1063 fputs("Resent-", fo);
1064 message_id(fo, NULL);
1067 if (check_from_and_sender(fromfield, senderfield))
1068 goto jleave;
1070 /* Write the original headers */
1071 while (cnt > 0) {
1072 if (fgetline(&buf, &bufsize, &cnt, &c, fi, 0) == NULL)
1073 break;
1074 /* XXX more checks: The From_ line may be seen when resending */
1075 /* During headers is_head() is actually overkill, so ^From_ is sufficient
1076 * && !is_head(buf, c) */
1077 if (ascncasecmp("status: ", buf, 8) != 0 && strncmp("From ", buf, 5))
1078 fwrite(buf, sizeof *buf, c, fo);
1079 if (cnt > 0 && *buf == '\n')
1080 break;
1083 /* Write the message body */
1084 while (cnt > 0) {
1085 if (fgetline(&buf, &bufsize, &cnt, &c, fi, 0) == NULL)
1086 break;
1087 if (cnt == 0 && *buf == '\n')
1088 break;
1089 fwrite(buf, sizeof *buf, c, fo);
1091 if (buf != NULL)
1092 free(buf);
1093 if (ferror(fo)) {
1094 perror(tr(188, "temporary mail file"));
1095 goto jleave;
1097 rv = 0;
1098 jleave:
1099 NYD_LEAVE;
1100 return rv;
1103 FL int
1104 mail(struct name *to, struct name *cc, struct name *bcc, char *subject,
1105 struct attachment *attach, char *quotefile, int recipient_record)
1107 struct header head;
1108 struct str in, out;
1109 NYD_ENTER;
1111 memset(&head, 0, sizeof head);
1113 /* The given subject may be in RFC1522 format. */
1114 if (subject != NULL) {
1115 in.s = subject;
1116 in.l = strlen(subject);
1117 mime_fromhdr(&in, &out, /* TODO ??? TD_ISPR |*/ TD_ICONV);
1118 head.h_subject = out.s;
1120 if (!(options & OPT_t_FLAG)) {
1121 head.h_to = to;
1122 head.h_cc = cc;
1123 head.h_bcc = bcc;
1125 head.h_attach = attach;
1127 mail1(&head, 0, NULL, quotefile, recipient_record, 0);
1129 if (subject != NULL)
1130 free(out.s);
1131 NYD_LEAVE;
1132 return 0;
1135 FL int
1136 c_sendmail(void *v)
1138 int rv;
1139 NYD_ENTER;
1141 rv = sendmail_internal(v, 0);
1142 NYD_LEAVE;
1143 return rv;
1146 FL int
1147 c_Sendmail(void *v)
1149 int rv;
1150 NYD_ENTER;
1152 rv = sendmail_internal(v, 1);
1153 NYD_LEAVE;
1154 return rv;
1157 FL enum okay
1158 mail1(struct header *hp, int printheaders, struct message *quote,
1159 char *quotefile, int recipient_record, int doprefix)
1161 struct name *to;
1162 FILE *mtf, *nmtf;
1163 int dosign = -1, err;
1164 char const *cp;
1165 enum okay rv = STOP;
1166 NYD_ENTER;
1168 _sendout_error = FAL0;
1170 /* Update some globals we likely need first */
1171 time_current_update(&time_current, TRU1);
1173 /* */
1174 if ((cp = ok_vlook(autocc)) != NULL && *cp != '\0')
1175 hp->h_cc = cat(hp->h_cc, checkaddrs(lextract(cp, GCC | GFULL)));
1176 if ((cp = ok_vlook(autobcc)) != NULL && *cp != '\0')
1177 hp->h_bcc = cat(hp->h_bcc, checkaddrs(lextract(cp, GBCC | GFULL)));
1179 /* Collect user's mail from standard input. Get the result as mtf */
1180 mtf = collect(hp, printheaders, quote, quotefile, doprefix);
1181 if (mtf == NULL)
1182 goto j_leave;
1184 if (options & OPT_INTERACTIVE) {
1185 err = (ok_blook(bsdcompat) || ok_blook(askatend));
1186 if (err == 0)
1187 goto jaskeot;
1188 if (ok_blook(askcc))
1189 ++err, grab_headers(hp, GCC, 1);
1190 if (ok_blook(askbcc))
1191 ++err, grab_headers(hp, GBCC, 1);
1192 if (ok_blook(askattach))
1193 ++err, hp->h_attach = edit_attachments(hp->h_attach);
1194 if (ok_blook(asksign))
1195 ++err, dosign = yorn(tr(35, "Sign this message (y/n)? "));
1196 if (err == 1) {
1197 jaskeot:
1198 printf(tr(183, "EOT\n"));
1199 fflush(stdout);
1203 if (fsize(mtf) == 0) {
1204 if (options & OPT_E_FLAG)
1205 goto jleave;
1206 if (hp->h_subject == NULL)
1207 printf(tr(184, "No message, no subject; hope that's ok\n"));
1208 else if (ok_blook(bsdcompat) || ok_blook(bsdmsgs))
1209 printf(tr(185, "Null message body; hope that's ok\n"));
1212 if (dosign < 0)
1213 dosign = ok_blook(smime_sign);
1214 #ifndef HAVE_SSL
1215 if (dosign) {
1216 fprintf(stderr, tr(225, "No SSL support compiled in.\n"));
1217 goto jleave;
1219 #endif
1221 /* XXX Update time_current again; once collect() offers editing of more
1222 * XXX headers, including Date:, this must only happen if Date: is the
1223 * XXX same that it was before collect() (e.g., postponing etc.).
1224 * XXX But *do* update otherwise because the mail seems to be backdated
1225 * XXX if the user edited some time, which looks odd and it happened
1226 * XXX to me that i got mis-dated response mails due to that... */
1227 time_current_update(&time_current, TRU1);
1229 /* TODO hrmpf; the MIME/send layer rewrite MUST address the init crap:
1230 * TODO setup the header ONCE; note this affects edit.c, collect.c ...,
1231 * TODO but: offer a hook that rebuilds/expands/checks/fixates all
1232 * TODO header fields ONCE, call that ONCE after user editing etc. has
1233 * TODO completed (one edit cycle) */
1235 /* Take the user names from the combined to and cc lists and do all the
1236 * alias processing. The POSIX standard says:
1237 * The names shall be substituted when alias is used as a recipient
1238 * address specified by the user in an outgoing message (that is,
1239 * other recipients addressed indirectly through the reply command
1240 * shall not be substituted in this manner).
1241 * S-nail thus violates POSIX, as has been pointed out correctly by
1242 * Martin Neitzel, but logic and usability of POSIX standards is not seldom
1243 * disputable anyway. Go for user friendliness */
1245 /* Do alias expansion on Reply-To: members, too */
1246 /* TODO puthead() YET (!!! see ONCE note above) expands the value, but
1247 * TODO doesn't perform alias expansion; encapsulate in the ONCE-o */
1248 if (hp->h_replyto == NULL && (cp = ok_vlook(replyto)) != NULL)
1249 hp->h_replyto = checkaddrs(lextract(cp, GEXTRA | GFULL));
1250 if (hp->h_replyto != NULL)
1251 hp->h_replyto = elide(usermap(hp->h_replyto, TRU1));
1253 /* TODO what happens now is that all recipients are merged into
1254 * TODO a duplicated list with expanded aliases, then this list is
1255 * TODO splitted again into the three individual recipient lists (with
1256 * TODO duplicates removed).
1257 * TODO later on we use the merged list for outof() pipe/file saving,
1258 * TODO then we eliminate duplicates (again) and then we use that one
1259 * TODO for mightrecord() and transfer(), and again. ... Please ... */
1261 /* NOTE: Due to elide() in fixhead(), ENSURE to,cc,bcc order of to!,
1262 * because otherwise the recipients will be "degraded" if they occur
1263 * multiple times */
1264 to = usermap(cat(hp->h_to, cat(hp->h_cc, hp->h_bcc)), FAL0);
1265 if (to == NULL) {
1266 fprintf(stderr, tr(186, "No recipients specified\n"));
1267 _sendout_error = TRU1;
1269 to = fixhead(hp, to);
1271 /* 'Bit ugly kind of control flow until we find a charset that does it */
1272 for (charset_iter_reset(hp->h_charset);;) {
1273 if (charset_iter_next() == NULL)
1275 else if ((nmtf = infix(hp, mtf)) != NULL)
1276 break;
1277 else if ((err = errno) == EILSEQ || err == EINVAL) {
1278 rewind(mtf);
1279 continue;
1282 perror("");
1283 #ifdef HAVE_SSL
1284 jfail_dead:
1285 #endif
1286 _sendout_error = TRU1;
1287 savedeadletter(mtf, 1);
1288 fputs(tr(187, ". . . message not sent.\n"), stderr);
1289 goto jleave;
1292 mtf = nmtf;
1293 #ifdef HAVE_SSL
1294 if (dosign) {
1295 if ((nmtf = smime_sign(mtf, hp)) == NULL)
1296 goto jfail_dead;
1297 Fclose(mtf);
1298 mtf = nmtf;
1300 #endif
1302 /* TODO truly - i still don't get what follows: (1) we deliver file
1303 * TODO and pipe addressees, (2) we mightrecord() and (3) we transfer
1304 * TODO even if (1) savedeadletter() etc. To me this doesn't make sense? */
1306 /* Deliver pipe and file addressees */
1307 to = outof(to, mtf, hp, &_sendout_error);
1308 if (_sendout_error)
1309 savedeadletter(mtf, 0);
1311 to = elide(to); /* XXX needed only to drop GDELs due to outof()! */
1312 dosign = (int)count(to); /* XXX ui32; reuse as "do-transfer" */
1313 if ((!recipient_record || dosign) &&
1314 mightrecord(mtf, (recipient_record ? to : NULL)) != OKAY)
1315 goto jleave;
1316 if (dosign)
1317 rv = transfer(to, mtf, hp);
1318 else if (!_sendout_error)
1319 rv = OKAY;
1321 jleave:
1322 Fclose(mtf);
1323 j_leave:
1324 if (_sendout_error)
1325 exit_status |= EXIT_SEND_ERROR;
1326 NYD_LEAVE;
1327 return rv;
1330 FL int
1331 mkdate(FILE *fo, char const *field)
1333 struct tm *tmptr;
1334 int tzdiff, tzdiff_hour, tzdiff_min, rv;
1335 NYD_ENTER;
1337 tzdiff = time_current.tc_time - mktime(&time_current.tc_gm);
1338 tzdiff_hour = (int)(tzdiff / 60);
1339 tzdiff_min = tzdiff_hour % 60;
1340 tzdiff_hour /= 60;
1341 tmptr = &time_current.tc_local;
1342 if (tmptr->tm_isdst > 0)
1343 ++tzdiff_hour;
1344 rv = fprintf(fo, "%s: %s, %02d %s %04d %02d:%02d:%02d %+05d\n",
1345 field,
1346 weekday_names[tmptr->tm_wday],
1347 tmptr->tm_mday, month_names[tmptr->tm_mon],
1348 tmptr->tm_year + 1900, tmptr->tm_hour,
1349 tmptr->tm_min, tmptr->tm_sec,
1350 tzdiff_hour * 100 + tzdiff_min);
1351 NYD_LEAVE;
1352 return rv;
1355 FL int
1356 puthead(struct header *hp, FILE *fo, enum gfield w, enum sendaction action,
1357 enum conversion convert, char const *contenttype, char const *charset)
1359 #define FMT_CC_AND_BCC() \
1360 do {\
1361 if (hp->h_cc != NULL && w & GCC) {\
1362 if (fmt("Cc:", hp->h_cc, fo, (w & (GCOMMA | GFILES)), 0,\
1363 (action != SEND_TODISP)))\
1364 goto jleave;\
1365 ++gotcha;\
1367 if (hp->h_bcc != NULL && w & GBCC) {\
1368 if (fmt("Bcc:", hp->h_bcc, fo, (w & (GCOMMA | GFILES)), 0,\
1369 (action != SEND_TODISP)))\
1370 goto jleave;\
1371 ++gotcha;\
1373 } while (0)
1375 char const *addr;
1376 size_t gotcha, l;
1377 struct name *np, *fromfield = NULL, *senderfield = NULL;
1378 int stealthmua, rv = 1;
1379 bool_t nodisp;
1380 NYD_ENTER;
1382 if ((addr = ok_vlook(stealthmua)) != NULL)
1383 stealthmua = !strcmp(addr, "noagent") ? -1 : 1;
1384 else
1385 stealthmua = 0;
1386 gotcha = 0;
1387 nodisp = (action != SEND_TODISP);
1389 if (w & GDATE)
1390 mkdate(fo, "Date"), ++gotcha;
1391 if (w & GIDENT) {
1392 if (hp->h_from != NULL) {
1393 if (fmt("From:", hp->h_from, fo, (w & (GCOMMA | GFILES)), 0, nodisp))
1394 goto jleave;
1395 ++gotcha;
1396 fromfield = hp->h_from;
1397 } else if ((addr = myaddrs(hp)) != NULL) {
1398 if (_putname(addr, w, action, &gotcha, "From:", fo, &fromfield))
1399 goto jleave;
1400 hp->h_from = fromfield;
1403 if (((addr = hp->h_organization) != NULL ||
1404 (addr = ok_vlook(ORGANIZATION)) != NULL) &&
1405 (l = strlen(addr)) > 0) {
1406 fwrite("Organization: ", sizeof(char), 14, fo);
1407 if (xmime_write(addr, l, fo, (!nodisp ? CONV_NONE : CONV_TOHDR),
1408 (!nodisp ? TD_ISPR | TD_ICONV : TD_ICONV), NULL) < 0)
1409 goto jleave;
1410 ++gotcha;
1411 putc('\n', fo);
1414 /* TODO see the ONCE TODO note somewhere around this file;
1415 * TODO but anyway, do NOT perform alias expansion UNLESS
1416 * TODO we are actually sending out! */
1417 if (hp->h_replyto != NULL) {
1418 if (fmt("Reply-To:", hp->h_replyto, fo, w & GCOMMA, 0, nodisp))
1419 goto jleave;
1420 ++gotcha;
1421 } else if ((addr = ok_vlook(replyto)) != NULL)
1422 if (_putname(addr, w, action, &gotcha, "Reply-To:", fo, NULL))
1423 goto jleave;
1425 if (hp->h_sender != NULL) {
1426 if (fmt("Sender:", hp->h_sender, fo, w & GCOMMA, 0, nodisp))
1427 goto jleave;
1428 ++gotcha;
1429 senderfield = hp->h_sender;
1430 } else if ((addr = ok_vlook(sender)) != NULL)
1431 if (_putname(addr, w, action, &gotcha, "Sender:", fo, &senderfield))
1432 goto jleave;
1434 if (check_from_and_sender(fromfield, senderfield))
1435 goto jleave;
1438 if (hp->h_to != NULL && w & GTO) {
1439 if (fmt("To:", hp->h_to, fo, (w & (GCOMMA | GFILES)), 0, nodisp))
1440 goto jleave;
1441 ++gotcha;
1444 if (!ok_blook(bsdcompat) && !ok_blook(bsdorder))
1445 FMT_CC_AND_BCC();
1447 if (hp->h_subject != NULL && w & GSUBJECT) {
1448 fwrite("Subject: ", sizeof (char), 9, fo);
1449 if (!ascncasecmp(hp->h_subject, "re: ", 4)) {
1450 fwrite("Re: ", sizeof(char), 4, fo);
1451 if (strlen(hp->h_subject + 4) > 0 &&
1452 xmime_write(hp->h_subject + 4, strlen(hp->h_subject + 4), fo,
1453 (!nodisp ? CONV_NONE : CONV_TOHDR),
1454 (!nodisp ? TD_ISPR | TD_ICONV : TD_ICONV), NULL) < 0)
1455 goto jleave;
1456 } else if (*hp->h_subject != '\0') {
1457 if (xmime_write(hp->h_subject, strlen(hp->h_subject), fo,
1458 (!nodisp ? CONV_NONE : CONV_TOHDR),
1459 (!nodisp ? TD_ISPR | TD_ICONV : TD_ICONV), NULL) < 0)
1460 goto jleave;
1462 ++gotcha;
1463 fwrite("\n", sizeof (char), 1, fo);
1466 if (ok_blook(bsdcompat) || ok_blook(bsdorder))
1467 FMT_CC_AND_BCC();
1469 if (w & GMSGID && stealthmua <= 0)
1470 message_id(fo, hp), ++gotcha;
1472 if ((np = hp->h_ref) != NULL && w & GREF) {
1473 fmt("References:", np, fo, 0, 1, 0);
1474 if (np->n_name) {
1475 while (np->n_flink)
1476 np = np->n_flink;
1477 if (!is_addr_invalid(np, 0)) {
1478 fprintf(fo, "In-Reply-To: %s\n", np->n_name);
1479 ++gotcha;
1484 if (w & GUA && stealthmua == 0)
1485 fprintf(fo, "User-Agent: %s %s\n", uagent, version), ++gotcha;
1487 if (w & GMIME) {
1488 fputs("MIME-Version: 1.0\n", fo), ++gotcha;
1489 if (hp->h_attach != NULL) {
1490 _sendout_boundary = mime_create_boundary();/*TODO carrier*/
1491 fprintf(fo, "Content-Type: multipart/mixed;\n boundary=\"%s\"\n",
1492 _sendout_boundary);
1493 } else {
1494 fprintf(fo, "Content-Type: %s", contenttype);
1495 if (charset != NULL)
1496 fprintf(fo, "; charset=%s", charset);
1497 fprintf(fo, "\nContent-Transfer-Encoding: %s\n",
1498 _get_encoding(convert));
1502 if (gotcha && w & GNL)
1503 putc('\n', fo);
1504 rv = 0;
1505 jleave:
1506 NYD_LEAVE;
1507 return rv;
1508 #undef FMT_CC_AND_BCC
1511 FL enum okay
1512 resend_msg(struct message *mp, struct name *to, int add_resent) /* TODO check */
1514 FILE *ibuf, *nfo, *nfi;
1515 char *tempMail;
1516 struct header head;
1517 enum okay rv = STOP;
1518 NYD_ENTER;
1520 _sendout_error = FAL0;
1522 /* Update some globals we likely need first */
1523 time_current_update(&time_current, TRU1);
1525 memset(&head, 0, sizeof head);
1527 if ((to = checkaddrs(to)) == NULL) {
1528 _sendout_error = TRU1;
1529 goto jleave;
1532 if ((nfo = Ftmp(&tempMail, "resend", OF_WRONLY | OF_HOLDSIGS | OF_REGISTER,
1533 0600)) == NULL) {
1534 _sendout_error = TRU1;
1535 perror(tr(189, "temporary mail file"));
1536 goto jleave;
1538 if ((nfi = Fopen(tempMail, "r")) == NULL) {
1539 _sendout_error = TRU1;
1540 perror(tempMail);
1542 Ftmp_release(&tempMail);
1543 if (nfi == NULL)
1544 goto jerr_o;
1546 if ((ibuf = setinput(&mb, mp, NEED_BODY)) == NULL)
1547 goto jerr_all;
1549 head.h_to = to;
1550 to = fixhead(&head, to);
1552 if (infix_resend(ibuf, nfo, mp, head.h_to, add_resent) != 0) {
1553 savedeadletter(nfi, 1);
1554 fputs(tr(190, ". . . message not sent.\n"), stderr);
1555 jerr_all:
1556 Fclose(nfi);
1557 jerr_o:
1558 Fclose(nfo);
1559 _sendout_error = TRU1;
1560 goto jleave;
1562 Fclose(nfo);
1563 rewind(nfi);
1565 to = outof(to, nfi, &head, &_sendout_error);
1566 if (_sendout_error)
1567 savedeadletter(nfi, 0);
1569 to = elide(to); /* TODO should have been done in fixhead()? */
1571 if (count(to) != 0) {
1572 if (!ok_blook(record_resent) || mightrecord(nfi, to) == OKAY)
1573 rv = transfer(to, nfi, NULL);
1574 } else if (!_sendout_error)
1575 rv = OKAY;
1577 Fclose(nfi);
1578 jleave:
1579 if (_sendout_error)
1580 exit_status |= EXIT_SEND_ERROR;
1581 NYD_LEAVE;
1582 return rv;
1585 /* vim:set fenc=utf-8:s-it-mode */