3 * mhstoresbr.c -- routines to save/store the contents of MIME messages
5 * This code is Copyright (c) 2002, by the authors of nmh. See the
6 * COPYRIGHT file in the root directory of the nmh distribution for
7 * complete copyright information.
16 #include <h/mhparse.h>
19 enum clobber_policy_t
{
20 NMH_CLOBBER_ALWAYS
= 0,
27 static enum clobber_policy_t
clobber_policy (const char *);
30 CT
*cts
; /* Top-level list of contents to store. */
31 char *cwd
; /* cached current directory */
32 int autosw
; /* -auto enabled */
33 int verbosw
; /* -verbose enabled */
34 int files_not_clobbered
; /* output flag indicating that store failed
35 in order to not clobber an existing file */
37 /* The following must never be touched by a caller: they are for
38 internal use by the mhstoresbr functions. */
39 char *dir
; /* directory in which to store contents */
40 enum clobber_policy_t clobber_policy
; /* -clobber selection */
43 static bool use_param_as_filename(const char *p
);
46 mhstoreinfo_create (CT
*ct
, char *pwd
, const char *csw
, int asw
, int vsw
) {
54 info
->files_not_clobbered
= 0;
56 info
->clobber_policy
= clobber_policy (csw
);
62 mhstoreinfo_free (mhstoreinfo_t info
) {
69 mhstoreinfo_files_not_clobbered (const mhstoreinfo_t info
) {
70 return info
->files_not_clobbered
;
75 * Type for a compare function for qsort. This keeps
78 typedef int (*qsort_comp
) (const void *, const void *);
83 int type_ok (CT
, int);
84 void flush_errors (void);
89 static void store_single_message (CT
, mhstoreinfo_t
);
90 static int store_switch (CT
, mhstoreinfo_t
);
91 static int store_generic (CT
, mhstoreinfo_t
);
92 static int store_application (CT
, mhstoreinfo_t
);
93 static int store_multi (CT
, mhstoreinfo_t
);
94 static int store_partial (CT
, mhstoreinfo_t
);
95 static int store_external (CT
, mhstoreinfo_t
);
96 static int ct_compar (CT
*, CT
*);
97 static int store_content (CT
, CT
, mhstoreinfo_t
);
98 static int output_content_file (CT
, int);
99 static int output_content_folder (char *, char *);
100 static int parse_format_string (CT
, char *, char *, int, char *);
101 static void get_storeproc (CT
);
102 static int copy_some_headers (FILE *, CT
);
103 static char *clobber_check (char *, mhstoreinfo_t
);
106 * Main entry point to store content
107 * from a collection of messages.
111 store_all_messages (mhstoreinfo_t info
)
117 * Check for the directory in which to
118 * store any contents.
120 if ((cp
= context_find (nmhstorage
)) && *cp
)
121 info
->dir
= mh_xstrdup(cp
);
123 info
->dir
= getcpy (info
->cwd
);
125 for (ctp
= info
->cts
; *ctp
; ctp
++) {
127 store_single_message (ct
, info
);
135 * Entry point to store the content
136 * in a (single) message
140 store_single_message (CT ct
, mhstoreinfo_t info
)
142 if (type_ok (ct
, 1)) {
144 store_switch (ct
, info
);
149 if (ct
->c_ceclosefnx
)
150 (*ct
->c_ceclosefnx
) (ct
);
156 * Switching routine to store different content types
160 store_switch (CT ct
, mhstoreinfo_t info
)
162 switch (ct
->c_type
) {
164 return store_multi (ct
, info
);
167 switch (ct
->c_subtype
) {
168 case MESSAGE_PARTIAL
:
169 return store_partial (ct
, info
);
171 case MESSAGE_EXTERNAL
:
172 return store_external (ct
, info
);
176 return store_generic (ct
, info
);
181 return store_application (ct
, info
);
187 return store_generic (ct
, info
);
190 return OK
; /* NOT REACHED */
195 * Generic routine to store a MIME content.
196 * (audio, video, image, text, message/rfc822)
200 store_generic (CT ct
, mhstoreinfo_t info
)
203 * Check if the content specifies a filename.
204 * Don't bother with this for type "message"
205 * (only "message/rfc822" will use store_generic).
207 if (info
->autosw
&& ct
->c_type
!= CT_MESSAGE
)
210 return store_content (ct
, NULL
, info
);
215 * Store content of type "application"
219 store_application (CT ct
, mhstoreinfo_t info
)
221 CI ci
= &ct
->c_ctinfo
;
223 /* Check if the content specifies a filename */
228 * If storeproc is not defined, and the content is type
229 * "application/octet-stream", we also check for various
230 * attribute/value pairs which specify if this a tar file.
232 if (!ct
->c_storeproc
&& ct
->c_subtype
== APPLICATION_OCTETS
) {
233 int tarP
= 0, zP
= 0, gzP
= 0;
236 if ((cp
= get_param(ci
->ci_first_pm
, "type", ' ', 1))) {
237 if (strcasecmp (cp
, "tar") == 0)
241 /* check for "conversions=compress" attribute */
242 if ((cp
= get_param(ci
->ci_first_pm
, "conversions", ' ', 1)) ||
243 (cp
= get_param(ci
->ci_first_pm
, "x-conversions", ' ', 1))) {
244 if (strcasecmp (cp
, "compress") == 0 ||
245 strcasecmp (cp
, "x-compress") == 0) {
248 if (strcasecmp (cp
, "gzip") == 0 ||
249 strcasecmp (cp
, "x-gzip") == 0) {
255 ct
->c_showproc
= add (zP
? "%euncompress | tar tvf -"
256 : (gzP
? "%egzip -dc | tar tvf -"
257 : "%etar tvf -"), NULL
);
258 if (!ct
->c_storeproc
) {
260 ct
->c_storeproc
= add (zP
? "| uncompress | tar xvpf -"
261 : (gzP
? "| gzip -dc | tar xvpf -"
262 : "| tar xvpf -"), NULL
);
265 ct
->c_storeproc
= add (zP
? "%m%P.tar.Z"
266 : (gzP
? "%m%P.tar.gz"
267 : "%m%P.tar"), NULL
);
273 return store_content (ct
, NULL
, info
);
278 * Store the content of a multipart message
282 store_multi (CT ct
, mhstoreinfo_t info
)
285 struct multipart
*m
= (struct multipart
*) ct
->c_ctparams
;
289 for (part
= m
->mp_parts
; part
; part
= part
->mp_next
) {
290 CT p
= part
->mp_part
;
292 if (part_ok (p
) && type_ok (p
, 1)) {
294 /* Support mhstore -outfile. The MIME parser doesn't
295 load c_storage, so we know that p->c_storage is
297 p
->c_storage
= mh_xstrdup(ct
->c_storage
);
299 result
= store_switch (p
, info
);
301 if (result
== OK
&& ct
->c_subtype
== MULTI_ALTERNATE
)
311 * Reassemble and store the contents of a collection
312 * of messages of type "message/partial".
316 store_partial (CT ct
, mhstoreinfo_t info
)
321 struct partial
*pm
, *qm
;
323 qm
= (struct partial
*) ct
->c_ctparams
;
328 for (ctp
= info
->cts
; *ctp
; ctp
++) {
330 if (p
->c_type
== CT_MESSAGE
&& p
->c_subtype
== ct
->c_subtype
) {
331 pm
= (struct partial
*) p
->c_ctparams
;
333 && strcmp (qm
->pm_partid
, pm
->pm_partid
) == 0) {
334 pm
->pm_marked
= pm
->pm_partno
;
346 advise (NULL
, "missing (at least) last part of multipart message");
350 base
= mh_xcalloc(i
+ 1, sizeof *base
);
352 for (ctp
= info
->cts
; *ctp
; ctp
++) {
354 if (p
->c_type
== CT_MESSAGE
&& p
->c_subtype
== ct
->c_subtype
) {
355 pm
= (struct partial
*) p
->c_ctparams
;
363 qsort ((char *) base
, i
, sizeof(*base
), (qsort_comp
) ct_compar
);
366 for (ctq
= base
; *ctq
; ctq
++) {
368 pm
= (struct partial
*) p
->c_ctparams
;
369 if (pm
->pm_marked
!= cur
) {
370 if (pm
->pm_marked
== cur
- 1) {
372 "duplicate part %d of %d part multipart message",
379 "missing %spart %d of %d part multipart message",
380 cur
!= hi
? "(at least) " : "", cur
, hi
);
392 * Now cycle through the sorted list of messages of type
393 * "message/partial" and save/append them to a file.
398 if (store_content (ct
, NULL
, info
) == NOTOK
) {
404 for (; *ctq
; ctq
++) {
406 if (store_content (p
, ct
, info
) == NOTOK
)
416 * Store content from a message of type "message/external".
420 store_external (CT ct
, mhstoreinfo_t info
)
423 struct exbody
*e
= (struct exbody
*) ct
->c_ctparams
;
424 CT p
= e
->eb_content
;
430 * Check if the parameters for the external body
431 * specified a filename.
436 if ((cp
= e
->eb_name
) && use_param_as_filename(cp
)) {
437 if (!ct
->c_storeproc
)
438 ct
->c_storeproc
= mh_xstrdup(cp
);
440 p
->c_storeproc
= mh_xstrdup(cp
);
445 * Since we will let the Content structure for the
446 * external body substitute for the current content,
447 * we temporarily change its partno (number inside
448 * multipart), so everything looks right.
450 p
->c_partno
= ct
->c_partno
;
452 /* we probably need to check if content is really there */
454 /* Support mhstore -outfile. The MIME parser doesn't load
455 c_storage, so we know that p->c_storage is NULL here. */
456 p
->c_storage
= mh_xstrdup(ct
->c_storage
);
458 result
= store_switch (p
, info
);
466 * Compare the numbering from two different
467 * message/partials (needed for sorting).
471 ct_compar (CT
*a
, CT
*b
)
473 struct partial
*am
= (struct partial
*) ((*a
)->c_ctparams
);
474 struct partial
*bm
= (struct partial
*) ((*b
)->c_ctparams
);
476 return (am
->pm_marked
- bm
->pm_marked
);
481 * Store contents of a message or message part to
482 * a folder, a file, the standard output, or pass
483 * the contents to a command.
485 * If the current content to be saved is a followup part
486 * to a collection of messages of type "message/partial",
487 * then field "p" is a pointer to the Content structure
488 * to the first message/partial in the group.
492 store_content (CT ct
, CT p
, mhstoreinfo_t info
)
494 int appending
= 0, msgnum
= 0;
495 int is_partial
= 0, first_partial
= 0;
496 int last_partial
= 0;
497 char *cp
, buffer
[BUFSIZ
];
500 * Do special processing for messages of
501 * type "message/partial".
503 * We first check if this content is of type
504 * "message/partial". If it is, then we need to check
505 * whether it is the first and/or last in the group.
507 * Then if "p" is a valid pointer, it points to the Content
508 * structure of the first partial in the group. So we copy
509 * the file name and/or folder name from that message. In
510 * this case, we also note that we will be appending.
512 if (ct
->c_type
== CT_MESSAGE
&& ct
->c_subtype
== MESSAGE_PARTIAL
) {
513 struct partial
*pm
= (struct partial
*) ct
->c_ctparams
;
515 /* Yep, it's a message/partial */
518 /* But is it the first and/or last in the collection? */
519 if (pm
->pm_partno
== 1)
521 if (pm
->pm_maxno
&& pm
->pm_partno
== pm
->pm_maxno
)
525 * If "p" is a valid pointer, then it points to the
526 * Content structure for the first message in the group.
527 * So we just copy the filename or foldername information
528 * from the previous iteration of this function.
532 if (! ct
->c_storage
) {
533 ct
->c_storage
= add (p
->c_storage
, NULL
);
535 /* record the folder name */
537 ct
->c_folder
= mh_xstrdup(p
->c_folder
);
545 * Get storage formatting string.
547 * 1) If we have storeproc defined, then use that
548 * 2) Else check for a mhn-store-<type>/<subtype> entry
549 * 3) Else check for a mhn-store-<type> entry
550 * 4) Else if content is "message", use "+" (current folder)
551 * 5) Else use string "%m%P.%s".
553 if ((cp
= ct
->c_storeproc
) == NULL
|| *cp
== '\0') {
554 CI ci
= &ct
->c_ctinfo
;
556 cp
= context_find_by_type ("store", ci
->ci_type
, ci
->ci_subtype
);
558 cp
= ct
->c_type
== CT_MESSAGE
? "+" : "%m%P.%s";
562 if (! ct
->c_storage
) {
564 * Check the beginning of storage formatting string
565 * to see if we are saving content to a folder.
567 if (*cp
== '+' || *cp
== '@') {
568 char *tmpfilenam
, *folder
;
570 /* Store content in temporary file for now */
571 if ((tmpfilenam
= m_mktemp(invo_name
, NULL
, NULL
)) == NULL
) {
572 adios(NULL
, "unable to create temporary file in %s",
575 ct
->c_storage
= mh_xstrdup(tmpfilenam
);
577 /* Get the folder name */
579 folder
= pluspath (cp
);
581 folder
= getfolder (1);
583 /* Check if folder exists */
584 create_folder(m_mailpath(folder
), 0, exit
);
586 /* Record the folder name */
587 ct
->c_folder
= add (folder
, NULL
);
596 * Parse and expand the storage formatting string
597 * in `cp' into `buffer'.
599 parse_format_string (ct
, cp
, buffer
, sizeof(buffer
), info
->dir
);
602 * If formatting begins with '|' or '!', then pass
603 * content to standard input of a command and return.
605 if (buffer
[0] == '|' || buffer
[0] == '!')
606 return show_content_aux (ct
, 0, buffer
+ 1, info
->dir
, NULL
);
608 /* record the filename */
609 if ((ct
->c_storage
= clobber_check (mh_xstrdup(buffer
), info
)) ==
614 /* The output filename was explicitly specified, so use it. */
615 if ((ct
->c_storage
= clobber_check (ct
->c_storage
, info
)) ==
622 /* flush the output stream */
625 /* Now save or append the content to a file */
626 if (output_content_file (ct
, appending
) == NOTOK
)
630 * If necessary, link the file into a folder and remove
631 * the temporary file. If this message is a partial,
632 * then only do this if it is the last one in the group.
634 if (ct
->c_folder
&& (!is_partial
|| last_partial
)) {
635 msgnum
= output_content_folder (ct
->c_folder
, ct
->c_storage
);
636 (void) m_unlink (ct
->c_storage
);
643 * Now print out the name/number of the message
644 * that we are storing.
648 fprintf (stderr
, "reassembling partials ");
650 fputs(ct
->c_file
, stderr
);
652 fprintf (stderr
, "%s,", ct
->c_file
);
654 fprintf (stderr
, "storing message %s", ct
->c_file
);
656 fprintf (stderr
, " part %s", ct
->c_partno
);
660 * Unless we are in the "middle" of group of message/partials,
661 * we now print the name of the file, folder, and/or message
662 * to which we are storing the content.
664 if (!is_partial
|| last_partial
) {
666 fprintf (stderr
, " to folder %s as message %d\n", ct
->c_folder
,
668 } else if (!strcmp(ct
->c_storage
, "-")) {
669 fprintf (stderr
, " to stdout\n");
671 int cwdlen
= strlen (info
->cwd
);
673 fprintf (stderr
, " as file %s\n",
674 !has_prefix(ct
->c_storage
, info
->cwd
)
675 || ct
->c_storage
[cwdlen
] != '/'
676 ? ct
->c_storage
: ct
->c_storage
+ cwdlen
+ 1);
686 * Output content to a file
690 output_content_file (CT ct
, int appending
)
693 char *file
, buffer
[BUFSIZ
];
698 * If the pathname is absolute, make sure
699 * all the relevant directories exist.
701 if (strchr(ct
->c_storage
, '/')
702 && make_intermediates (ct
->c_storage
) == NOTOK
)
705 if (ct
->c_encoding
!= CE_7BIT
) {
708 if (!ct
->c_ceopenfnx
) {
709 advise (NULL
, "don't know how to decode part %s of message %s",
710 ct
->c_partno
, ct
->c_file
);
714 file
= appending
|| !strcmp (ct
->c_storage
, "-") ? NULL
716 if ((fd
= (*ct
->c_ceopenfnx
) (ct
, &file
)) == NOTOK
)
718 if (!strcmp (file
, ct
->c_storage
)) {
719 (*ct
->c_ceclosefnx
) (ct
);
724 * Send to standard output
726 if (!strcmp (ct
->c_storage
, "-")) {
729 if ((gd
= dup (fileno (stdout
))) == NOTOK
) {
730 advise ("stdout", "unable to dup");
732 (*ct
->c_ceclosefnx
) (ct
);
735 if ((fp
= fdopen (gd
, appending
? "a" : "w")) == NULL
) {
736 advise ("stdout", "unable to fdopen (%d, \"%s\") from", gd
,
737 appending
? "a" : "w");
745 if ((fp
= fopen (ct
->c_storage
, appending
? "a" : "w")) == NULL
) {
746 advise (ct
->c_storage
, "unable to fopen for %s",
747 appending
? "appending" : "writing");
753 * Filter the header fields of the initial enclosing
754 * message/partial into the file.
756 if (ct
->c_type
== CT_MESSAGE
&& ct
->c_subtype
== MESSAGE_PARTIAL
) {
757 struct partial
*pm
= (struct partial
*) ct
->c_ctparams
;
759 if (pm
->pm_partno
== 1)
760 copy_some_headers (fp
, ct
);
764 switch (cc
= read (fd
, buffer
, sizeof(buffer
))) {
766 advise (file
, "error reading content from");
773 if ((int) fwrite (buffer
, sizeof(*buffer
), cc
, fp
) < cc
) {
774 advise ("output_content_file", "fwrite");
781 (*ct
->c_ceclosefnx
) (ct
);
783 if (cc
!= NOTOK
&& fflush (fp
))
784 advise (ct
->c_storage
, "error writing to");
788 return (cc
!= NOTOK
? OK
: NOTOK
);
791 if (!ct
->c_fp
&& (ct
->c_fp
= fopen (ct
->c_file
, "r")) == NULL
) {
792 advise (ct
->c_file
, "unable to open for reading");
798 fseek (ct
->c_fp
, pos
, SEEK_SET
);
800 if (!strcmp (ct
->c_storage
, "-")) {
803 if ((gd
= dup (fileno (stdout
))) == NOTOK
) {
804 advise ("stdout", "unable to dup");
807 if ((fp
= fdopen (gd
, appending
? "a" : "w")) == NULL
) {
808 advise ("stdout", "unable to fdopen (%d, \"%s\") from", gd
,
809 appending
? "a" : "w");
814 if ((fp
= fopen (ct
->c_storage
, appending
? "a" : "w")) == NULL
) {
815 advise (ct
->c_storage
, "unable to fopen for %s",
816 appending
? "appending" : "writing");
822 * Copy a few of the header fields of the initial
823 * enclosing message/partial into the file.
826 if (ct
->c_type
== CT_MESSAGE
&& ct
->c_subtype
== MESSAGE_PARTIAL
) {
827 struct partial
*pm
= (struct partial
*) ct
->c_ctparams
;
829 if (pm
->pm_partno
== 1) {
830 copy_some_headers (fp
, ct
);
835 while (fgets (buffer
, sizeof buffer
, ct
->c_fp
)) {
836 if ((pos
+= strlen (buffer
)) > last
) {
839 diff
= strlen (buffer
) - (pos
- last
);
844 * If this is the first content of a group of
845 * message/partial contents, then we only copy a few
846 * of the header fields of the enclosed message.
861 if (!uprf (buffer
, XXX_FIELD_PRF
)
862 && !uprf (buffer
, VRSN_FIELD
)
863 && !uprf (buffer
, "Subject:")
864 && !uprf (buffer
, "Encrypted:")
865 && !uprf (buffer
, "Message-ID:")) {
880 advise (ct
->c_storage
, "error writing to");
890 * Add a file to a folder.
892 * Return the new message number of the file
893 * when added to the folder. Return -1, if
898 output_content_folder (char *folder
, char *filename
)
903 /* Read the folder. */
904 if ((mp
= folder_read (folder
, 0))) {
905 /* Link file into folder */
906 msgnum
= folder_addmsg (&mp
, filename
, 0, 0, 0, 0, NULL
);
908 advise (NULL
, "unable to read folder %s", folder
);
912 /* free folder structure */
916 * Return msgnum. We are relying on the fact that
917 * msgnum will be -1, if folder_addmsg() had an error.
924 * Parse and expand the storage formatting string
925 * pointed to by "cp" into "buffer".
929 parse_format_string (CT ct
, char *cp
, char *buffer
, int buflen
, char *dir
)
933 CI ci
= &ct
->c_ctinfo
;
936 * If storage string is "-", just copy it, and
937 * return (send content to standard output).
939 if (cp
[0] == '-' && cp
[1] == '\0') {
940 strncpy (buffer
, cp
, buflen
);
948 * If formatting string is a pathname that doesn't
949 * begin with '/', then preface the path with the
950 * appropriate directory.
952 if (*cp
!= '/' && *cp
!= '|' && *cp
!= '!') {
953 snprintf (bp
, buflen
, "%s/", dir
[1] ? dir
: "");
961 /* We are processing a storage escape */
966 * Insert parameters from Content-Type.
967 * This is only valid for '|' commands.
969 if (buffer
[0] != '|' && buffer
[0] != '!') {
978 for (pm
= ci
->ci_first_pm
; pm
; pm
= pm
->pm_next
) {
979 snprintf (bp
, buflen
, "%s%s=\"%s\"", s
,
980 pm
->pm_name
, get_param_value(pm
, '?'));
990 /* insert message number */
991 snprintf (bp
, buflen
, "%s", r1bindex (ct
->c_file
, '/'));
995 /* insert part number with leading dot */
997 snprintf (bp
, buflen
, ".%s", ct
->c_partno
);
1001 /* insert part number withouth leading dot */
1003 strncpy (bp
, ct
->c_partno
, buflen
);
1007 /* insert content type */
1008 strncpy (bp
, ci
->ci_type
, buflen
);
1012 /* insert content subtype */
1013 strncpy (bp
, ci
->ci_subtype
, buflen
);
1017 /* insert the character % */
1027 /* Advance bp and decrement buflen */
1045 * Check if the content specifies a filename
1046 * in its MIME parameters.
1050 get_storeproc (CT ct
)
1056 * If the storeproc has already been defined,
1057 * we just return (for instance, if this content
1058 * is part of a "message/external".
1060 if (ct
->c_storeproc
)
1064 * If there's a Content-Disposition header and it has a filename,
1065 * use that (RFC-2183).
1068 if ((cp
= get_param(ct
->c_dispo_first
, "filename", '_', 0)) &&
1069 use_param_as_filename(cp
)) {
1070 ct
->c_storeproc
= mh_xstrdup(cp
);
1078 * Check the attribute/value pairs, for the attribute "name".
1079 * If found, do a few sanity checks and copy the value into
1083 if ((cp
= get_param(ci
->ci_first_pm
, "name", '_', 0)) &&
1084 use_param_as_filename(cp
)) {
1085 ct
->c_storeproc
= mh_xstrdup(cp
);
1093 * Copy some of the header fields of the initial message/partial
1094 * message into the header of the reassembled message.
1098 copy_some_headers (FILE *out
, CT ct
)
1102 hp
= ct
->c_first_hf
; /* start at first header field */
1106 * A few of the header fields of the enclosing
1107 * messages are not copied.
1109 if (!uprf (hp
->name
, XXX_FIELD_PRF
)
1110 && strcasecmp (hp
->name
, VRSN_FIELD
)
1111 && strcasecmp (hp
->name
, "Subject")
1112 && strcasecmp (hp
->name
, "Encrypted")
1113 && strcasecmp (hp
->name
, "Message-ID"))
1114 fprintf (out
, "%s:%s", hp
->name
, hp
->value
);
1115 hp
= hp
->next
; /* next header field */
1121 /******************************************************************************/
1122 /* -clobber support */
1125 enum clobber_policy_t
1126 clobber_policy (const char *value
) {
1127 if (value
== NULL
|| ! strcasecmp (value
, "always")) {
1128 return NMH_CLOBBER_ALWAYS
;
1130 if (! strcasecmp (value
, "auto")) {
1131 return NMH_CLOBBER_AUTO
;
1133 if (! strcasecmp (value
, "suffix")) {
1134 return NMH_CLOBBER_SUFFIX
;
1136 if (! strcasecmp (value
, "ask")) {
1137 return NMH_CLOBBER_ASK
;
1139 if (! strcasecmp (value
, "never")) {
1140 return NMH_CLOBBER_NEVER
;
1143 adios (NULL
, "invalid argument, %s, to clobber", value
);
1148 next_version (char *file
, enum clobber_policy_t clobber_policy
) {
1149 const size_t max_versions
= 1000000;
1150 /* 8 = log max_versions + one for - or . + one for null terminator */
1151 const size_t buflen
= strlen (file
) + 8;
1152 char *buffer
= mh_xmalloc (buflen
);
1155 char *extension
= NULL
;
1156 if (clobber_policy
== NMH_CLOBBER_AUTO
&&
1157 ((extension
= strrchr (file
, '.')) != NULL
)) {
1158 *extension
++ = '\0';
1161 for (version
= 1; version
< max_versions
; ++version
) {
1164 switch (clobber_policy
) {
1165 case NMH_CLOBBER_AUTO
: {
1166 snprintf (buffer
, buflen
, "%s-%ld%s%s", file
, (long) version
,
1167 extension
== NULL
? "" : ".",
1168 extension
== NULL
? "" : extension
);
1172 case NMH_CLOBBER_SUFFIX
:
1173 snprintf (buffer
, buflen
, "%s.%ld", file
, (long) version
);
1177 /* Should never get here. */
1178 advise (NULL
, "will not overwrite %s, invalid clobber policy", buffer
);
1183 /* Actually (try to) create the file here to avoid a race
1184 condition on file naming + creation. This won't solve the
1185 problem with old NFS that doesn't support O_EXCL, though.
1186 Let the umask strip off permissions from 0666 as desired.
1187 That's what fopen () would do if it was creating the file. */
1188 if ((fd
= open (buffer
, O_CREAT
| O_EXCL
,
1189 S_IRUSR
| S_IWUSR
| S_IRGRP
| S_IWGRP
|
1190 S_IROTH
| S_IWOTH
)) >= 0) {
1198 if (version
>= max_versions
) {
1199 advise (NULL
, "will not overwrite %s, too many versions", buffer
);
1209 clobber_check (char *original_file
, mhstoreinfo_t info
) {
1210 /* clobber policy return value
1211 * -------------- ------------
1212 * -always original_file
1213 * -auto original_file-<digits>.extension
1214 * -suffix original_file.<digits>
1215 * -ask original_file, 0, or another filename/path
1223 if (! strcmp (original_file
, "-")) {
1224 return original_file
;
1227 if (info
->clobber_policy
== NMH_CLOBBER_ASK
) {
1228 /* Save cwd for possible use in loop below. */
1231 cwd
= mh_xstrdup(original_file
);
1232 slash
= strrchr (cwd
, '/');
1237 /* original_file isn't a full path, which should only happen if
1247 file
= original_file
;
1250 switch (info
->clobber_policy
) {
1251 case NMH_CLOBBER_ALWAYS
:
1254 case NMH_CLOBBER_SUFFIX
:
1255 case NMH_CLOBBER_AUTO
:
1256 if (stat (file
, &st
) == OK
) {
1257 if ((file
= next_version (original_file
, info
->clobber_policy
)) ==
1259 ++info
->files_not_clobbered
;
1264 case NMH_CLOBBER_ASK
:
1265 if (stat (file
, &st
) == OK
) {
1266 enum answers
{ NMH_YES
, NMH_NO
, NMH_RENAME
};
1267 static struct swit answer
[4] = {
1268 { "yes", 0, NMH_YES
},
1269 { "no", 0, NMH_NO
},
1270 { "rename", 0, NMH_RENAME
},
1274 if (isatty (fileno (stdin
))) {
1276 concat ("Overwrite \"", file
, "\" [y/n/rename]? ", NULL
);
1277 ans
= read_switch_multiword (prompt
, answer
);
1280 /* Overwrite, that's what nmh used to do. And warn. */
1281 advise (NULL
, "-clobber ask but no tty, so overwrite %s", file
);
1285 switch ((enum answers
) smatch (*ans
, answer
)) {
1291 ++info
->files_not_clobbered
;
1295 printf ("Enter filename or full path of the new file: ");
1296 if (fgets (buf
, sizeof buf
, stdin
) == NULL
||
1299 ++info
->files_not_clobbered
;
1301 trim_suffix_c(buf
, '\n');
1306 if (buf
[0] == '/') {
1307 /* Full path, use it. */
1308 file
= mh_xstrdup(buf
);
1310 /* Relative path. */
1311 file
= cwd
? concat (cwd
, "/", buf
, NULL
) : mh_xstrdup(buf
);
1321 case NMH_CLOBBER_NEVER
:
1322 if (stat (file
, &st
) == OK
) {
1323 /* Keep count of files that would have been clobbered,
1324 and return that as process exit status. */
1325 advise (NULL
, "will not overwrite %s with -clobber never", file
);
1328 ++info
->files_not_clobbered
;
1333 original_file
= file
;
1334 } while (check_again
);
1341 static bool use_param_as_filename(const char *p
)
1343 /* Preserve result of original test that considered an empty string
1345 return !*p
|| (!strchr("/.|!", *p
) && !strchr(p
, '%'));
1348 /* -clobber support */
1349 /******************************************************************************/