1*34f9b3eeSRoland Mainz /*********************************************************************** 2*34f9b3eeSRoland Mainz * * 3*34f9b3eeSRoland Mainz * This software is part of the ast package * 4*34f9b3eeSRoland Mainz * Copyright (c) 1992-2009 AT&T Intellectual Property * 5*34f9b3eeSRoland Mainz * and is licensed under the * 6*34f9b3eeSRoland Mainz * Common Public License, Version 1.0 * 7*34f9b3eeSRoland Mainz * by AT&T Intellectual Property * 8*34f9b3eeSRoland Mainz * * 9*34f9b3eeSRoland Mainz * A copy of the License is available at * 10*34f9b3eeSRoland Mainz * http://www.opensource.org/licenses/cpl1.0.txt * 11*34f9b3eeSRoland Mainz * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) * 12*34f9b3eeSRoland Mainz * * 13*34f9b3eeSRoland Mainz * Information and Software Systems Research * 14*34f9b3eeSRoland Mainz * AT&T Research * 15*34f9b3eeSRoland Mainz * Florham Park NJ * 16*34f9b3eeSRoland Mainz * * 17*34f9b3eeSRoland Mainz * Glenn Fowler <gsf@research.att.com> * 18*34f9b3eeSRoland Mainz * David Korn <dgk@research.att.com> * 19*34f9b3eeSRoland Mainz * * 20*34f9b3eeSRoland Mainz ***********************************************************************/ 21*34f9b3eeSRoland Mainz #pragma prototyped 22*34f9b3eeSRoland Mainz 23*34f9b3eeSRoland Mainz static const char usage[] = 24*34f9b3eeSRoland Mainz "[-?\n@(#)$Id: mktemp (AT&T Research) 2009-06-19 $\n]" 25*34f9b3eeSRoland Mainz USAGE_LICENSE 26*34f9b3eeSRoland Mainz "[+NAME?mktemp - make temporary file or directory]" 27*34f9b3eeSRoland Mainz "[+DESCRIPTION?\bmktemp\b creates a temporary file with optional base " 28*34f9b3eeSRoland Mainz "name prefix \aprefix\a. If \aprefix\a is omitted then \btmp_\b is used " 29*34f9b3eeSRoland Mainz "and \b--tmp\b is implied. If \aprefix\a contains a directory prefix " 30*34f9b3eeSRoland Mainz "then that directory overrides any of the directories described below. A " 31*34f9b3eeSRoland Mainz "temporary file will have mode \brw-------\b and a temporary directory " 32*34f9b3eeSRoland Mainz "will have mode \brwx------\b, subject to \bumask\b(1). Generated paths " 33*34f9b3eeSRoland Mainz "have these attributes:]" 34*34f9b3eeSRoland Mainz "{" 35*34f9b3eeSRoland Mainz "[+*?Lower case to avoid clashes on case ignorant filesystems.]" 36*34f9b3eeSRoland Mainz "[+*?Pseudo-random part to deter denial of service attacks.]" 37*34f9b3eeSRoland Mainz "[+*?Pseudo-random part is \a3-chars\a.\a3-chars\a to accomodate " 38*34f9b3eeSRoland Mainz "8.3 filesystems.]" 39*34f9b3eeSRoland Mainz "}" 40*34f9b3eeSRoland Mainz "[+?A consecutive sequence of \bX\b's in \aprefix\a is replaced by the " 41*34f9b3eeSRoland Mainz "pseudo-random part. If there are no \bX\b's then the pseudo-random part " 42*34f9b3eeSRoland Mainz "is appended to the prefix.]" 43*34f9b3eeSRoland Mainz "[d:directory?Create a directory instead of a regular file.]" 44*34f9b3eeSRoland Mainz "[m:mode]:[mode?Set the mode of the created temporary to \amode\a. " 45*34f9b3eeSRoland Mainz "\amode\a is symbolic or octal mode as in \bchmod\b(1). Relative modes " 46*34f9b3eeSRoland Mainz "assume an initial mode of \bu=rwx\b.]" 47*34f9b3eeSRoland Mainz "[p:default?Use \adirectory\a if the \bTMPDIR\b environment variable is " 48*34f9b3eeSRoland Mainz "not defined. Implies \b--tmp\b.]:[directory]" 49*34f9b3eeSRoland Mainz "[q:quiet?Suppress file and directory error diagnostics.]" 50*34f9b3eeSRoland Mainz "[t:tmp|temporary-directory?Create a path rooted in a temporary " 51*34f9b3eeSRoland Mainz "directory.]" 52*34f9b3eeSRoland Mainz "[u:unsafe|dry-run?Check for file/directory existence but do not create. " 53*34f9b3eeSRoland Mainz "Who would want to do that.]" 54*34f9b3eeSRoland Mainz "\n" 55*34f9b3eeSRoland Mainz "\n[ prefix ]\n" 56*34f9b3eeSRoland Mainz "\n" 57*34f9b3eeSRoland Mainz "[+SEE ALSO?\bmkdir\b(1), \bpathtemp\b(3), \bmktemp\b(3)]" 58*34f9b3eeSRoland Mainz ; 59*34f9b3eeSRoland Mainz 60*34f9b3eeSRoland Mainz #include <cmd.h> 61*34f9b3eeSRoland Mainz #include <ls.h> 62*34f9b3eeSRoland Mainz 63*34f9b3eeSRoland Mainz int 64*34f9b3eeSRoland Mainz b_mktemp(int argc, char** argv, void* context) 65*34f9b3eeSRoland Mainz { 66*34f9b3eeSRoland Mainz mode_t mode = 0; 67*34f9b3eeSRoland Mainz mode_t mask; 68*34f9b3eeSRoland Mainz int fd; 69*34f9b3eeSRoland Mainz int i; 70*34f9b3eeSRoland Mainz int quiet = 0; 71*34f9b3eeSRoland Mainz int unsafe = 0; 72*34f9b3eeSRoland Mainz int* fdp = &fd; 73*34f9b3eeSRoland Mainz char* dir = ""; 74*34f9b3eeSRoland Mainz char* pfx; 75*34f9b3eeSRoland Mainz char* t; 76*34f9b3eeSRoland Mainz char path[PATH_MAX]; 77*34f9b3eeSRoland Mainz 78*34f9b3eeSRoland Mainz cmdinit(argc, argv, context, ERROR_CATALOG, ERROR_NOTIFY); 79*34f9b3eeSRoland Mainz for (;;) 80*34f9b3eeSRoland Mainz { 81*34f9b3eeSRoland Mainz switch (optget(argv, usage)) 82*34f9b3eeSRoland Mainz { 83*34f9b3eeSRoland Mainz case 0: 84*34f9b3eeSRoland Mainz break; 85*34f9b3eeSRoland Mainz case 'd': 86*34f9b3eeSRoland Mainz fdp = 0; 87*34f9b3eeSRoland Mainz continue; 88*34f9b3eeSRoland Mainz case 'm': 89*34f9b3eeSRoland Mainz mode = strperm(pfx = opt_info.arg, &opt_info.arg, S_IRWXU); 90*34f9b3eeSRoland Mainz if (*opt_info.arg) 91*34f9b3eeSRoland Mainz error(ERROR_exit(0), "%s: invalid mode", pfx); 92*34f9b3eeSRoland Mainz continue; 93*34f9b3eeSRoland Mainz case 'p': 94*34f9b3eeSRoland Mainz if ((t = getenv("TMPDIR")) && *t) 95*34f9b3eeSRoland Mainz dir = 0; 96*34f9b3eeSRoland Mainz else 97*34f9b3eeSRoland Mainz dir = opt_info.arg; 98*34f9b3eeSRoland Mainz continue; 99*34f9b3eeSRoland Mainz case 'q': 100*34f9b3eeSRoland Mainz quiet = 1; 101*34f9b3eeSRoland Mainz continue; 102*34f9b3eeSRoland Mainz case 't': 103*34f9b3eeSRoland Mainz dir = 0; 104*34f9b3eeSRoland Mainz continue; 105*34f9b3eeSRoland Mainz case 'u': 106*34f9b3eeSRoland Mainz unsafe = 1; 107*34f9b3eeSRoland Mainz fdp = 0; 108*34f9b3eeSRoland Mainz continue; 109*34f9b3eeSRoland Mainz case ':': 110*34f9b3eeSRoland Mainz error(2, "%s", opt_info.arg); 111*34f9b3eeSRoland Mainz continue; 112*34f9b3eeSRoland Mainz case '?': 113*34f9b3eeSRoland Mainz error(ERROR_usage(2), "%s", opt_info.arg); 114*34f9b3eeSRoland Mainz continue; 115*34f9b3eeSRoland Mainz } 116*34f9b3eeSRoland Mainz break; 117*34f9b3eeSRoland Mainz } 118*34f9b3eeSRoland Mainz argv += opt_info.index; 119*34f9b3eeSRoland Mainz if (error_info.errors || (pfx = *argv++) && *argv) 120*34f9b3eeSRoland Mainz error(ERROR_usage(2), "%s", optusage(NiL)); 121*34f9b3eeSRoland Mainz mask = umask(0); 122*34f9b3eeSRoland Mainz if (!mode) 123*34f9b3eeSRoland Mainz mode = (fdp ? (S_IRUSR|S_IWUSR) : S_IRWXU) & ~mask; 124*34f9b3eeSRoland Mainz umask(~mode & (S_IRWXU|S_IRWXG|S_IRWXO)); 125*34f9b3eeSRoland Mainz if (!pfx) 126*34f9b3eeSRoland Mainz { 127*34f9b3eeSRoland Mainz pfx = "tmp_"; 128*34f9b3eeSRoland Mainz if (dir && !*dir) 129*34f9b3eeSRoland Mainz dir = 0; 130*34f9b3eeSRoland Mainz } 131*34f9b3eeSRoland Mainz if (t = strrchr(pfx, '/')) 132*34f9b3eeSRoland Mainz { 133*34f9b3eeSRoland Mainz i = ++t - pfx; 134*34f9b3eeSRoland Mainz dir = fmtbuf(i); 135*34f9b3eeSRoland Mainz memcpy(dir, pfx, i); 136*34f9b3eeSRoland Mainz dir[i] = 0; 137*34f9b3eeSRoland Mainz pfx = t; 138*34f9b3eeSRoland Mainz } 139*34f9b3eeSRoland Mainz for (;;) 140*34f9b3eeSRoland Mainz { 141*34f9b3eeSRoland Mainz if (!pathtemp(path, sizeof(path), dir, pfx, fdp)) 142*34f9b3eeSRoland Mainz { 143*34f9b3eeSRoland Mainz if (quiet) 144*34f9b3eeSRoland Mainz error_info.errors++; 145*34f9b3eeSRoland Mainz else 146*34f9b3eeSRoland Mainz error(ERROR_SYSTEM|2, "cannot create temporary path"); 147*34f9b3eeSRoland Mainz break; 148*34f9b3eeSRoland Mainz } 149*34f9b3eeSRoland Mainz if (fdp || unsafe || !mkdir(path, mode)) 150*34f9b3eeSRoland Mainz { 151*34f9b3eeSRoland Mainz if (fdp) 152*34f9b3eeSRoland Mainz close(*fdp); 153*34f9b3eeSRoland Mainz sfputr(sfstdout, path, '\n'); 154*34f9b3eeSRoland Mainz break; 155*34f9b3eeSRoland Mainz } 156*34f9b3eeSRoland Mainz if (sh_checksig(context)) 157*34f9b3eeSRoland Mainz { 158*34f9b3eeSRoland Mainz error_info.errors++; 159*34f9b3eeSRoland Mainz break; 160*34f9b3eeSRoland Mainz } 161*34f9b3eeSRoland Mainz } 162*34f9b3eeSRoland Mainz umask(mask); 163*34f9b3eeSRoland Mainz return error_info.errors != 0; 164*34f9b3eeSRoland Mainz } 165