| 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;
 
- }
 
 
  |