logo

utils-std

Collection of commonly available Unix tools git clone https://anongit.hacktivis.me/git/utils-std.git/

mktemp.c (4370B)


  1. // utils-std: Collection of commonly available Unix tools
  2. // SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me>
  3. // SPDX-License-Identifier: MPL-2.0
  4. // getentropy got in POSIX.1-2024 but defining _POSIX_C_SOURCE causes too much side-effects
  5. #define _DEFAULT_SOURCE // getentropy
  6. #include "../config.h" // HAS_*
  7. #include "../lib/getopt_nolong.h"
  8. #include <errno.h>
  9. #include <limits.h> // PATH_MAX
  10. #include <stdbool.h>
  11. #include <stdio.h> // fprintf
  12. #include <stdlib.h> // getenv
  13. #include <string.h> // strerror
  14. #include <unistd.h> // getopt, getentropy
  15. #ifdef HAS_GETOPT_LONG
  16. #include <getopt.h>
  17. #endif
  18. const char *argv0 = "mktemp";
  19. static int
  20. unsafe_mktemp(char *template)
  21. {
  22. size_t l = strlen(template);
  23. size_t len = 0;
  24. size_t off = l - 1;
  25. while(off > 0 && template[off] == 'X')
  26. off--, len++;
  27. if(len < 6)
  28. {
  29. fprintf(stderr,
  30. "%s: error: Template '%s' is invalid (need >= 6 final X, got %zd)\n",
  31. argv0,
  32. template,
  33. len);
  34. return 1;
  35. }
  36. for(size_t i = 0; i < l; i++)
  37. {
  38. if(template[i] == '\n')
  39. {
  40. fprintf(stderr, "%s: error: Invalid character (newline) in template\n", argv0);
  41. return 1;
  42. }
  43. }
  44. for(int retries = 100; 0 < retries; retries--)
  45. {
  46. unsigned long r = 0;
  47. if(getentropy(&r, sizeof(r)) != 0)
  48. {
  49. fprintf(stderr,
  50. "%s: error: Failed to get entropy for random filename: %s\n",
  51. argv0,
  52. strerror(errno));
  53. return 1;
  54. }
  55. for(int i = len; i > 0; i--, r >>= 5)
  56. template[off + i] = 'A' + (r & 15) + (r & 16) * 2;
  57. if(access(template, F_OK) != 0 && errno == ENOENT)
  58. {
  59. puts(template);
  60. return 0;
  61. }
  62. }
  63. fprintf(stderr, "%s: error: Exhausted all 100 attempts at generating a filename\n", argv0);
  64. return 1;
  65. }
  66. int
  67. main(int argc, char *argv[])
  68. {
  69. bool o_create_dir = false, o_quiet = false, o_unsafe = false;
  70. static char template[PATH_MAX] = "tmp.XXXXXXXXXX";
  71. const char *tmpdir = NULL;
  72. #ifdef HAS_GETOPT_LONG
  73. // Strictly for GNUisms compatibility so no long-only options
  74. // clang-format off
  75. static struct option opts[] = {
  76. {"directory", no_argument, 0, 'd'},
  77. {"dry-run", required_argument, 0, 'u'},
  78. {"quiet", no_argument, 0, 'q'},
  79. {"tmpdir", required_argument, 0, 'p'},
  80. {0, 0, 0, 0},
  81. };
  82. // clang-format on
  83. // Need + as first character to get POSIX-style option parsing
  84. for(int c = -1; (c = getopt_long(argc, argv, "+:dqp:tu", opts, NULL)) != -1;)
  85. #else
  86. for(int c = -1; (c = getopt_nolong(argc, argv, ":dqp:tu")) != -1;)
  87. #endif
  88. {
  89. switch(c)
  90. {
  91. case 'd':
  92. o_create_dir = true;
  93. break;
  94. case 'q':
  95. o_quiet = true;
  96. break;
  97. case 'p':
  98. tmpdir = optarg;
  99. break;
  100. case 't':
  101. if(tmpdir == NULL) tmpdir = getenv("TMPDIR");
  102. if(tmpdir == NULL) tmpdir = "/tmp";
  103. break;
  104. case 'u':
  105. o_unsafe = true;
  106. break;
  107. case '?':
  108. if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option: '-%c'\n", argv0, optopt);
  109. return 1;
  110. default:
  111. fprintf(stderr, "%s: error: Unhandled getopt case '%c'\n", argv0, c);
  112. abort();
  113. }
  114. }
  115. argc -= optind;
  116. argv += optind;
  117. if(argc == 1)
  118. {
  119. strncpy(template, *argv, PATH_MAX);
  120. }
  121. else if(argc > 1)
  122. {
  123. fprintf(stderr, "%s: error: Only one template argument is supported, got %d\n", argv0, argc);
  124. return 1;
  125. }
  126. if(tmpdir)
  127. {
  128. if(chdir(tmpdir) < 0)
  129. {
  130. fprintf(stderr,
  131. "%s: error: Failed changing directory into tmpdir '%s': %s\n",
  132. argv0,
  133. tmpdir,
  134. strerror(errno));
  135. return 1;
  136. }
  137. printf("%s/", tmpdir);
  138. }
  139. static char template_copy[PATH_MAX] = "";
  140. memcpy(template_copy, template, PATH_MAX);
  141. if(o_unsafe) return unsafe_mktemp(template);
  142. if(o_create_dir)
  143. {
  144. char *dir = mkdtemp(template);
  145. if(dir == NULL)
  146. {
  147. if(!o_quiet)
  148. fprintf(stderr,
  149. "%s: error: Failed creating random directory with template '%s': %s\n",
  150. argv0,
  151. template_copy,
  152. strerror(errno));
  153. return 1;
  154. }
  155. puts(dir);
  156. return 0;
  157. }
  158. int fd = mkstemp(template);
  159. if(fd < 0)
  160. {
  161. if(!o_quiet)
  162. fprintf(stderr,
  163. "%s: error: Failed creating random file with template '%s': %s\n",
  164. argv0,
  165. template_copy,
  166. strerror(errno));
  167. return 1;
  168. }
  169. puts(template);
  170. if(close(fd) < 0)
  171. {
  172. if(!o_quiet) fprintf(stderr, "%s: error: Failed closing file: %s\n", argv0, strerror(errno));
  173. return 1;
  174. }
  175. return 0;
  176. }