1 /*@ S-nail - a mail user agent derived from Berkeley Mail.
2 *@ Handling of attachments.
4 * Copyright (c) 2000-2004 Gunnar Ritter, Freiburg i. Br., Germany.
5 * Copyright (c) 2012 - 2013 Steffen "Daode" Nurpmeso <sdaoden@users.sf.net>.
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
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
40 #ifndef HAVE_AMALGAMATION
44 /* We use calloc() for struct attachment */
47 /* Fill in some attachment fields; don't be interactive if number==0n */
48 static struct attachment
* _fill_in(struct attachment
*ap
,
49 char const *file
, ui_it number
);
51 /* Ask the user to edit file names and other data for the given attachment */
52 static struct attachment
* _read_attachment_data(struct attachment
*ap
,
55 /* Try to create temporary charset converted version */
57 static int _attach_iconv(struct attachment
*ap
);
60 static struct attachment
*
61 _fill_in(struct attachment
*ap
, char const *file
, ui_it number
)
64 * XXX The "attachment-ask-content-*" variables are left undocumented
65 * since they are for RFC connoisseurs only.
70 if ((file
= strrchr(file
, '/')) != NULL
)
75 ap
->a_content_type
= mime_classify_content_type_by_fileext(file
);
76 if (number
> 0 && value("attachment-ask-content-type")) {
77 snprintf(prefix
, sizeof prefix
, "#%u\tContent-Type: ", number
);
78 ap
->a_content_type
= readstr_input(prefix
, ap
->a_content_type
);
81 if (number
> 0 && value("attachment-ask-content-disposition")) {
82 snprintf(prefix
, sizeof prefix
,
83 "#%u\tContent-Disposition: ", number
);
84 ap
->a_content_disposition
= readstr_input(prefix
,
85 ap
->a_content_disposition
);
87 if (ap
->a_content_disposition
== NULL
)
88 ap
->a_content_disposition
= "attachment";
90 if (number
> 0 && value("attachment-ask-content-id")) {
91 snprintf(prefix
, sizeof prefix
, "#%u\tContent-ID: ", number
);
92 ap
->a_content_id
= readstr_input(prefix
, ap
->a_content_id
);
95 if (number
> 0 && value("attachment-ask-content-description")) {
96 snprintf(prefix
, sizeof prefix
,
97 "#%u\tContent-Description: ", number
);
98 ap
->a_content_description
= readstr_input(prefix
,
99 ap
->a_content_description
);
104 static struct attachment
*
105 _read_attachment_data(struct attachment
*ap
, ui_it number
)
108 char const *cslc
= NULL
/*cc uninit?*/, *cp
, *defcs
;
111 ap
= csalloc(1, sizeof *ap
);
112 else if (ap
->a_msgno
) {
113 char *ecp
= salloc(16);
114 snprintf(ecp
, 16, "#%u", (ui_it
)ap
->a_msgno
);
117 } else if (ap
->a_conv
== AC_TMPFILE
) {
119 ap
->a_conv
= AC_DEFAULT
;
122 snprintf(prefix
, sizeof prefix
, tr(50, "#%u\tfilename: "), number
);
124 if ((ap
->a_name
= readstr_input(prefix
, ap
->a_name
)) == NULL
) {
128 /* May be a message number */
129 if (ap
->a_name
[0] == '#') {
131 int msgno
= (int)strtol(ap
->a_name
+ 1, &ecp
, 10);
132 if (msgno
> 0 && msgno
<= msgCount
&& *ecp
== '\0') {
134 ap
->a_content_description
=
135 tr(513, "Attached message content");
136 if (options
& OPT_INTERACTIVE
)
137 printf(tr(2, "~@: added message #%u\n"),
142 if ((cp
= file_expand(ap
->a_name
)) != NULL
&&
143 access(cp
, R_OK
) == 0) {
150 ap
= _fill_in(ap
, cp
, number
);
153 * Character set of attachments: enum attach_conv
155 cslc
= charset_get_lc();
157 if (! (options
& OPT_INTERACTIVE
))
159 if ((cp
= ap
->a_content_type
) != NULL
&&
160 ascncasecmp(cp
, "text/", 5) != 0 &&
161 ! yorn(tr(162, "Filename doesn't indicate text "
162 "content - want to edit charsets? "))) {
163 ap
->a_conv
= AC_DEFAULT
;
167 charset_iter_reset(NULL
);
170 snprintf(prefix
, sizeof prefix
, tr(160, "#%u\tinput charset: "),
172 if ((defcs
= ap
->a_input_charset
) == NULL
)
174 cp
= ap
->a_input_charset
= readstr_input(prefix
, defcs
);
176 if (! (options
& OPT_INTERACTIVE
)) {
178 ap
->a_conv
= (cp
!= NULL
) ? AC_FIX_INCS
: AC_DEFAULT
;
183 snprintf(prefix
, sizeof prefix
, tr(161, "#%u\toutput (send) charset: "),
185 if ((defcs
= ap
->a_charset
) == NULL
)
186 defcs
= charset_iter_next();
187 defcs
= ap
->a_charset
= readstr_input(prefix
, defcs
);
189 if (cp
!= NULL
&& defcs
== NULL
) {
190 ap
->a_conv
= AC_FIX_INCS
;
193 if (cp
== NULL
&& defcs
== NULL
) {
194 ap
->a_conv
= AC_DEFAULT
;
195 ap
->a_input_charset
= cslc
;
196 ap
->a_charset
= charset_iter_current();
197 } else if (cp
== NULL
&& defcs
!= NULL
) {
198 ap
->a_conv
= AC_FIX_OUTCS
;
199 ap
->a_input_charset
= cslc
;
201 ap
->a_conv
= AC_TMPFILE
;
203 printf(tr(197, "Trying conversion from %s to %s\n"),
204 ap
->a_input_charset
, ap
->a_charset
);
205 if (_attach_iconv(ap
))
206 ap
->a_conv
= AC_TMPFILE
;
208 ap
->a_input_charset
= cp
;
209 ap
->a_charset
= defcs
;
214 if (options
& OPT_INTERACTIVE
)
215 printf(tr(19, "~@: added attachment \"%s\"\n"), ap
->a_name
);
222 _attach_iconv(struct attachment
*ap
)
224 struct str oul
= {NULL
, 0}, inl
= {NULL
, 0};
225 FILE *fo
= NULL
, *fi
= NULL
;
229 if ((icp
= n_iconv_open(ap
->a_charset
, ap
->a_input_charset
))
234 perror("iconv_open");
238 if ((fi
= Fopen(ap
->a_name
, "r")) == NULL
) {
245 if ((fo
= Ftemp(&inl
.s
, "aiconv", "w+", 0600, 1)) == NULL
) {
246 perror(tr(51, "temporary mail file"));
253 for (inl
.s
= NULL
, lbsize
= 0;;) {
254 if (fgetline(&inl
.s
, &lbsize
, &cnt
, &inl
.l
, fi
, 0) == NULL
) {
257 perror(tr(195, "I/O read error occurred"));
261 if (n_iconv_str(icp
, &oul
, &inl
, NULL
, FAL0
) != 0)
263 if ((inl
.l
=fwrite(oul
.s
, sizeof *oul
.s
, oul
.l
, fo
)) != oul
.l
) {
264 perror(tr(196, "I/O write error occurred"));
278 if (icp
!= (iconv_t
)-1)
282 fprintf(stderr
, tr(179, "Cannot convert from %s to %s\n"),
283 ap
->a_input_charset
, ap
->a_charset
);
290 #endif /* HAVE_ICONV */
292 FL
struct attachment
*
293 add_attachment(struct attachment
*aphead
, char *file
, struct attachment
**newap
)
295 struct attachment
*nap
= NULL
, *ap
;
297 if ((file
= file_expand(file
)) == NULL
)
299 if (access(file
, R_OK
) != 0)
302 nap
= _fill_in(csalloc(1, sizeof *nap
), file
, 0);
305 if (aphead
!= NULL
) {
306 for (ap
= aphead
; ap
->a_flink
!= NULL
; ap
= ap
->a_flink
)
319 FL
struct attachment
*
320 append_attachments(struct attachment
*aphead
, char *names
)
323 struct attachment
*xaph
, *nap
;
325 while ((cp
= strcomma(&names
, 1)) != NULL
) {
326 if ((xaph
= add_attachment(aphead
, cp
, &nap
)) != NULL
) {
328 if (options
& OPT_INTERACTIVE
)
329 printf(tr(19, "~@: added attachment \"%s\"\n"),
337 FL
struct attachment
*
338 edit_attachments(struct attachment
*aphead
)
340 struct attachment
*ap
, *nap
;
343 /* Modify already present ones? */
344 for (ap
= aphead
; ap
!= NULL
; ap
= ap
->a_flink
) {
345 if (_read_attachment_data(ap
, attno
) != NULL
) {
350 if (ap
->a_blink
!= NULL
)
351 ap
->a_blink
->a_flink
= nap
;
355 nap
->a_blink
= ap
->a_blink
;
361 while ((nap
= _read_attachment_data(NULL
, attno
)) != NULL
) {
362 if ((ap
= aphead
) != NULL
) {
363 while (ap
->a_flink
!= NULL
)