]>
diplodocus.org Git - nmh/blob - sbr/m_mktemp.c
2 * m_mktemp.c -- Construct a temporary file.
4 * This code is Copyright (c) 2010, by the authors of nmh. See the
5 * COPYRIGHT file in the root directory of the nmh distribution for
6 * complete copyright information.
11 #include <h/signals.h>
13 static void register_for_removal(const char *);
16 /* Create a temporary file. If pfx_in is null, the temporary file
17 * will be created in the temporary directory (more on that later).
18 * If pfx_in is not null, then the temporary file location will be
19 * defined by the value pfx_in.
21 * The file created will be at the pathname specified appended with
22 * 6 random (we hope :) characters.
24 * The return value will be the pathname to the file created.
26 * CAUTION: The return pointer references static data. If
27 * you need to modify, or save, the return string, make a copy of it
30 * When pfx_in is null, the temporary directory is determined as
35 * User's mail directory.
37 * NOTE: One will probably use m_mktemp2() instead of this function.
38 * For example, if you want to create a temp file in the defined
39 * temporary directory, but with a custom basename prefix, do
40 * something like the following:
42 * char *tmp_pathname = m_mktemp2(NULL, "mypre", ...);
46 const char *pfx_in
, /* Pathname prefix for temporary file. */
47 int *fd_ret
, /* (return,optional) File descriptor to temp file. */
48 FILE **fp_ret
/* (return,optional) FILE pointer to temp file. */
51 static char tmpfil
[BUFSIZ
];
54 mode_t oldmode
= umask(077);
57 snprintf(tmpfil
, sizeof(tmpfil
), "%s/nmhXXXXXX", get_temp_dir());
59 snprintf(tmpfil
, sizeof(tmpfil
), "%sXXXXXX", pfx_in
);
69 register_for_removal(tmpfil
);
76 FILE *fp
= fdopen(fd
, "w+");
79 (void) m_unlink(tmpfil
);
95 /* This version allows one to specify the directory the temp file should
96 * by created based on a given pathname. Although m_mktemp() technically
97 * supports this, this version is when the directory is defined by
98 * a separate variable from the prefix, eliminating the caller from having
99 * to do string manipulation to generate the desired pathname prefix.
101 * The pfx_in parameter specifies a basename prefix for the file. If dir_in
102 * is NULL, then the defined temporary directory (see comments to m_mktemp()
103 * above) is used to create the temp file.
107 const char *dir_in
, /* Directory to place temp file. */
108 const char *pfx_in
, /* Basename prefix for temp file. */
109 int *fd_ret
, /* (return,optional) File descriptor to temp file. */
110 FILE **fp_ret
/* (return,optional) FILE pointer to temp file. */
113 static char buffer
[BUFSIZ
];
117 if (dir_in
== NULL
) {
118 if (pfx_in
== NULL
) {
119 return m_mktemp(NULL
, fd_ret
, fp_ret
);
121 snprintf(buffer
, sizeof(buffer
), "%s/%s", get_temp_dir(), pfx_in
);
122 return m_mktemp(buffer
, fd_ret
, fp_ret
);
125 if ((cp
= r1bindex ((char *)dir_in
, '/')) == dir_in
) {
126 /* No directory component */
127 return m_mktemp(pfx_in
, fd_ret
, fp_ret
);
129 n
= (int)(cp
-dir_in
); /* Length of dir component */
130 snprintf(buffer
, sizeof(buffer
), "%.*s%s", n
, dir_in
, pfx_in
);
131 return m_mktemp(buffer
, fd_ret
, fp_ret
);
136 * This version supports a suffix for the temp filename.
137 * It has two other differences from m_mktemp() and m_mktemp2():
138 * 1) It does not support specification of a directory for the temp
139 * file. Instead it always uses get_temp_dir().
140 * 2) It returns a dynamically allocated string. The caller is
141 * responsible for freeing the allocated memory.
145 const char *pfx_in
, /* Basename prefix for temp file. */
146 const char *suffix
, /* Suffix, including any '.', for temp file. */
147 int *fd_ret
, /* (return,optional) File descriptor to temp file. */
148 FILE **fp_ret
/* (return,optional) FILE pointer to temp file. */
154 mode_t oldmode
= umask(077);
156 if (suffix
== NULL
) {
157 if ((tmpfil
= m_mktemp2(NULL
, pfx_in
, fd_ret
, fp_ret
))) {
158 return mh_xstrdup(tmpfil
);
164 if (pfx_in
== NULL
) {
165 tmpfil
= concat(get_temp_dir(), "/nmhXXXXXX", suffix
, NULL
);
167 tmpfil
= concat(get_temp_dir(), "/", pfx_in
, "XXXXXX", suffix
, NULL
);
170 fd
= mkstemps(tmpfil
, (int) strlen(suffix
));
171 #else /* ! HAVE_MKSTEMPS */
172 /* NetBSD, Solaris 10 */
174 if (pfx_in
== NULL
) {
175 tmpfil
= concat(get_temp_dir(), "/nmhXXXXXX", NULL
);
177 tmpfil
= concat(get_temp_dir(), "/", pfx_in
, "XXXXXX", NULL
);
180 fd
= mkstemp(tmpfil
);
182 char *oldfilename
= tmpfil
;
183 tmpfil
= concat(oldfilename
, suffix
, NULL
);
185 if (rename(oldfilename
, tmpfil
) != 0) {
186 (void) unlink(oldfilename
);
194 #endif /* ! HAVE_MKSTEMPS */
202 register_for_removal(tmpfil
);
204 if (fd_ret
!= NULL
) {
208 if (fp_ret
!= NULL
) {
209 FILE *fp
= fdopen(fd
, "w+");
212 (void) m_unlink(tmpfil
);
234 /* Ignore envvars if we are setuid */
235 if ((getuid()==geteuid()) && (getgid()==getegid())) {
237 tmpdir
= getenv("MHTMPDIR");
238 if (tmpdir
!= NULL
&& *tmpdir
!= '\0') return tmpdir
;
240 tmpdir
= getenv("TMPDIR");
241 if (tmpdir
!= NULL
&& *tmpdir
!= '\0') return tmpdir
;
243 return m_maildir("");
248 * Array of files (full pathnames) to remove on process exit.
249 * Instead of removing slots from the array, just set to NULL
250 * to indicate that the file should no longer be removed.
252 static svector_t exit_filelist
= NULL
;
255 * Register a file for removal at program termination.
258 register_for_removal(const char *pathname
) {
259 if (exit_filelist
== NULL
) exit_filelist
= svector_create(20);
260 (void) svector_push_back(exit_filelist
, add(pathname
, NULL
));
264 * Unregister all files that were registered to be removed at program
265 * termination. When called with remove_files of 0, this function is
266 * intended for use by one of the programs after a fork(3) without a
267 * subsequent call to one of the exec(3) functions or _exit(3). When
268 * called with non-0 remove_files argument, it is intended for use by
269 * an atexit() function.
271 * Right after a fork() and before calling exec() or _exit(), if the
272 * child catches one of the appropriate signals, it will remove
273 * all the registered temporary files. Some of those may be needed by
274 * the parent. Some nmh programs ignore or block some of the signals
275 * in the child right after fork(). For the other programs, rather
276 * than complicate things by preventing that, just take the chance
277 * that it might happen. It is harmless to attempt to unlink a
278 * temporary file twice, assuming another one wasn't created too
279 * quickly created with the same name.
282 unregister_for_removal(int remove_files
) {
286 for (i
= 0; i
< svector_size(exit_filelist
); ++i
) {
287 char *filename
= svector_at(exit_filelist
, i
);
290 if (remove_files
) (void) unlink(filename
);
295 svector_free(exit_filelist
);
296 exit_filelist
= NULL
;
301 * If the file was registered for removal, deregister it. In
302 * any case, unlink it.
305 m_unlink(const char *pathname
) {
307 char **slot
= svector_find(exit_filelist
, pathname
);
315 return unlink(pathname
);
316 /* errno should be set to ENOENT if file was not found */
320 * Remove all registered temporary files.
323 remove_registered_files_atexit() {
324 unregister_for_removal(1);
328 * Remove all registered temporary files. Suitable for use as a
329 * signal handler. If the signal is one of the usual process
330 * termination signals, calls exit(). Otherwise, disables itself
331 * by restoring the default action and then re-raises the signal,
332 * in case the use was expecting a core dump.
335 remove_registered_files(int sig
) {
336 struct sigaction act
;
339 * Ignore this signal for the duration. And we either exit() or
340 * disable this signal handler below, so don't restore this handler.
342 act
.sa_handler
= SIG_IGN
;
343 (void) sigemptyset(&act
.sa_mask
);
345 (void) sigaction(sig
, &act
, 0);
347 if (sig
== SIGHUP
|| sig
== SIGINT
|| sig
== SIGQUIT
|| sig
== SIGTERM
) {
349 * We don't need to call remove_registered_files_atexit() if
350 * it was registered with atexit(), but let's not assume that.
351 * It's harmless to call it twice.
353 remove_registered_files_atexit();
357 remove_registered_files_atexit();
359 act
.sa_handler
= SIG_DFL
;
360 (void) sigemptyset(&act
.sa_mask
);
362 (void) sigaction(sig
, &act
, 0);