123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342 |
- /* $Id: tmpfileplus.c $ */
- /*
- * $Date: 2016-06-01 03:31Z $
- * $Revision: 2.0.0 $
- * $Author: dai $
- */
- /*
- * This Source Code Form is subject to the terms of the Mozilla Public
- * License, v. 2.0. If a copy of the MPL was not distributed with this
- * file, You can obtain one at http://mozilla.org/MPL/2.0/.
- *
- * Copyright (c) 2012-16 David Ireland, DI Management Services Pty Ltd
- * <http://www.di-mgt.com.au/contact/>.
- */
- /*
- * NAME
- * tmpfileplus - create a unique temporary file
- *
- * SYNOPSIS
- * FILE *tmpfileplus(const char *dir, const char *prefix, char **pathname, int keep)
- *
- * DESCRIPTION
- * The tmpfileplus() function opens a unique temporary file in binary
- * read/write (w+b) mode. The file is opened with the O_EXCL flag,
- * guaranteeing that the caller is the only user. The filename will consist
- * of the string given by `prefix` followed by 10 random characters. If
- * `prefix` is NULL, then the string "tmp." will be used instead. The file
- * will be created in an appropriate directory chosen by the first
- * successful attempt in the following sequence:
- *
- * a) The directory given by the `dir` argument (so the caller can specify
- * a secure directory to take precedence).
- *
- * b) The directory name in the environment variables:
- *
- * (i) "TMP" [Windows only]
- * (ii) "TEMP" [Windows only]
- * (iii) "TMPDIR" [Unix only]
- *
- * c) `P_tmpdir` as defined in <stdio.h> [Unix only] (in Windows, this is
- * usually "\", which is no good).
- *
- * d) The current working directory.
- *
- * If a file cannot be created in any of the above directories, then the
- * function fails and NULL is returned.
- *
- * If the argument `pathname` is not a null pointer, then it will point to
- * the full pathname of the file. The pathname is allocated using `malloc`
- * and therefore should be freed by `free`.
- *
- * If `keep` is nonzero and `pathname` is not a null pointer, then the file
- * will be kept after it is closed. Otherwise the file will be
- * automatically deleted when it is closed or the program terminates.
- *
- *
- * RETURN VALUE
- * The tmpfileplus() function returns a pointer to the open file stream,
- * or NULL if a unique file cannot be opened.
- *
- *
- * ERRORS
- * ENOMEM Not enough memory to allocate filename.
- *
- */
- /* ADDED IN v2.0 */
- /*
- * NAME
- * tmpfileplus_f - create a unique temporary file with filename stored in a fixed-length buffer
- *
- * SYNOPSIS
- * FILE *tmpfileplus_f(const char *dir, const char *prefix, char *pathnamebuf, size_t pathsize, int keep);
- *
- * DESCRIPTION
- * Same as tmpfileplus() except receives filename in a fixed-length buffer. No allocated memory to free.
- * ERRORS
- * E2BIG Resulting filename is too big for the buffer `pathnamebuf`.
- */
- #include "tmpfileplus.h"
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <time.h>
- #include <errno.h>
- /* Non-ANSI include files that seem to work in both MSVC and Linux */
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <fcntl.h>
- #ifdef _WIN32
- #include <io.h>
- #else
- #include <unistd.h>
- #endif
- #ifdef _WIN32
- /* MSVC nags to enforce ISO C++ conformant function names with leading "_",
- * so we define our own function names to avoid whingeing compilers...
- */
- #define OPEN_ _open
- #define FDOPEN_ _fdopen
- #else
- #define OPEN_ open
- #define FDOPEN_ fdopen
- #endif
- /* DEBUGGING STUFF */
- #if defined(_DEBUG) && defined(SHOW_DPRINTF)
- #define DPRINTF1(s, a1) printf(s, a1)
- #else
- #define DPRINTF1(s, a1)
- #endif
- #ifdef _WIN32
- #define FILE_SEPARATOR "\\"
- #else
- #define FILE_SEPARATOR "/"
- #endif
- #define RANDCHARS "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"
- #define NRANDCHARS (sizeof(RANDCHARS) - 1)
- /** Replace each byte in string s with a random character from TEMPCHARS */
- static char *set_randpart(char *s)
- {
- size_t i;
- unsigned int r;
- static unsigned int seed; /* NB static */
- if (seed == 0)
- { /* First time set our seed using current time and clock */
- seed = ((unsigned)time(NULL)<<8) ^ (unsigned)clock();
- }
- srand(seed++);
- for (i = 0; i < strlen(s); i++)
- {
- r = rand() % NRANDCHARS;
- s[i] = (RANDCHARS)[r];
- }
- return s;
- }
- /** Return 1 if path is a valid directory otherwise 0 */
- static int is_valid_dir(const char *path)
- {
- struct stat st;
- if ((stat(path, &st) == 0) && (st.st_mode & S_IFDIR))
- return 1;
- return 0;
- }
- /** Call getenv and save a copy in buf */
- static char *getenv_save(const char *varname, char *buf, size_t bufsize)
- {
- char *ptr = getenv(varname);
- buf[0] = '\0';
- if (ptr)
- {
- strncpy(buf, ptr, bufsize);
- buf[bufsize-1] = '\0';
- return buf;
- }
- return NULL;
- }
- /**
- * Try and create a randomly-named file in directory `tmpdir`.
- * If successful, allocate memory and set `tmpname_ptr` to full filepath, and return file pointer;
- * otherwise return NULL.
- * If `keep` is zero then create the file as temporary and it should not exist once closed.
- */
- static FILE *mktempfile_internal(const char *tmpdir, const char *pfx, char **tmpname_ptr, int keep)
- /* PRE:
- * pfx is not NULL and points to a valid null-terminated string
- * tmpname_ptr is not NULL.
- */
- {
- FILE *fp;
- int fd;
- char randpart[] = "1234567890";
- size_t lentempname;
- int i;
- char *tmpname = NULL;
- int oflag, pmode;
- /* In Windows, we use the _O_TEMPORARY flag with `open` to ensure the file is deleted when closed.
- * In Unix, we use the unlink function after opening the file. (This does not work in Windows,
- * which does not allow an open file to be unlinked.)
- */
- #ifdef _WIN32
- /* MSVC flags */
- oflag = _O_BINARY|_O_CREAT|_O_EXCL|_O_RDWR;
- if (!keep)
- oflag |= _O_TEMPORARY;
- pmode = _S_IREAD | _S_IWRITE;
- #else
- /* Standard POSIX flags */
- oflag = O_CREAT|O_EXCL|O_RDWR;
- pmode = S_IRUSR|S_IWUSR;
- #endif
- if (!tmpdir || !is_valid_dir(tmpdir)) {
- errno = ENOENT;
- return NULL;
- }
- lentempname = strlen(tmpdir) + strlen(FILE_SEPARATOR) + strlen(pfx) + strlen(randpart);
- DPRINTF1("lentempname=%d\n", lentempname);
- tmpname = malloc(lentempname + 1);
- if (!tmpname)
- {
- errno = ENOMEM;
- return NULL;
- }
- /* If we don't manage to create a file after 10 goes, there is something wrong... */
- for (i = 0; i < 10; i++)
- {
- sprintf(tmpname, "%s%s%s%s", tmpdir, FILE_SEPARATOR, pfx, set_randpart(randpart));
- DPRINTF1("[%s]\n", tmpname);
- fd = OPEN_(tmpname, oflag, pmode);
- if (fd != -1) break;
- }
- DPRINTF1("strlen(tmpname)=%d\n", strlen(tmpname));
- if (fd != -1)
- { /* Success, so return user a proper ANSI C file pointer */
- fp = FDOPEN_(fd, "w+b");
- errno = 0;
- #ifndef _WIN32
- /* [Unix only] And make sure the file will be deleted once closed */
- if (!keep) unlink(tmpname);
- #endif
- }
- else
- { /* We failed */
- fp = NULL;
- }
- if (!fp)
- {
- free(tmpname);
- tmpname = NULL;
- }
- *tmpname_ptr = tmpname;
- return fp;
- }
- /**********************/
- /* EXPORTED FUNCTIONS */
- /**********************/
- FILE *tmpfileplus(const char *dir, const char *prefix, char **pathname, int keep)
- {
- FILE *fp = NULL;
- char *tmpname = NULL;
- char *tmpdir = NULL;
- const char *pfx = (prefix ? prefix : "tmp.");
- char *tempdirs[12] = { 0 };
- #ifdef _WIN32
- char env1[FILENAME_MAX+1] = { 0 };
- char env2[FILENAME_MAX+1] = { 0 };
- #else
- char env3[FILENAME_MAX+1] = { 0 };
- #endif
- int ntempdirs = 0;
- int i;
- /* Set up a list of temp directories we will try in order */
- i = 0;
- tempdirs[i++] = (char *)dir;
- #ifdef _WIN32
- tempdirs[i++] = getenv_save("TMP", env1, sizeof(env1));
- tempdirs[i++] = getenv_save("TEMP", env2, sizeof(env2));
- #else
- tempdirs[i++] = getenv_save("TMPDIR", env3, sizeof(env3));
- tempdirs[i++] = P_tmpdir;
- #endif
- tempdirs[i++] = ".";
- ntempdirs = i;
- errno = 0;
- /* Work through list we set up before, and break once we are successful */
- for (i = 0; i < ntempdirs; i++)
- {
- tmpdir = tempdirs[i];
- DPRINTF1("Trying tmpdir=[%s]\n", tmpdir);
- fp = mktempfile_internal(tmpdir, pfx, &tmpname, keep);
- if (fp) break;
- }
- /* If we succeeded and the user passed a pointer, set it to the alloc'd pathname: the user must free this */
- if (fp && pathname)
- *pathname = tmpname;
- else /* Otherwise, free the alloc'd memory */
- free(tmpname);
- return fp;
- }
- /* Same as tmpfileplus() but with fixed length buffer for output filename and no memory allocation */
- FILE *tmpfileplus_f(const char *dir, const char *prefix, char *pathnamebuf, size_t pathsize, int keep)
- {
- char *tmpbuf = NULL;
- FILE *fp;
- /* If no buffer provided, do the normal way */
- if (!pathnamebuf || (int)pathsize <= 0) {
- return tmpfileplus(dir, prefix, NULL, keep);
- }
- /* Call with a temporary buffer */
- fp = tmpfileplus(dir, prefix, &tmpbuf, keep);
- if (fp && strlen(tmpbuf) > pathsize - 1) {
- /* Succeeded but not enough room in output buffer, so clean up and return an error */
- pathnamebuf[0] = 0;
- fclose(fp);
- if (keep) remove(tmpbuf);
- free(tmpbuf);
- errno = E2BIG;
- return NULL;
- }
- /* Copy name into buffer */
- strcpy(pathnamebuf, tmpbuf);
- free(tmpbuf);
- return fp;
- }
|