logo

utils-std

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

install.c (7758B)


  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. // For copy_file_range
  5. #define _GNU_SOURCE // musl, glibc
  6. #define _DEFAULT_SOURCE // FreeBSD
  7. #include "../config.h" // HAS_*
  8. #include "../lib/fs.h"
  9. #include "../lib/getopt_nolong.h"
  10. #include "../lib/lib_mkdir.h"
  11. #include "../lib/mode.h"
  12. #include "../lib/user_group_parse.h"
  13. #include <assert.h>
  14. #include <errno.h>
  15. #include <fcntl.h> // linkat, AT_SYMLINK_FOLLOW
  16. #include <libgen.h> // dirname
  17. #include <limits.h> // PATH_MAX
  18. #include <stdbool.h>
  19. #include <stdio.h> // fprintf
  20. #include <string.h> // strerror
  21. #include <sys/stat.h>
  22. #include <unistd.h> // getopt
  23. #ifdef HAS_GETOPT_LONG
  24. #include <getopt.h>
  25. #endif
  26. bool preserve_times = false, create_directories = false;
  27. // See lib/lib_mkdir.c
  28. bool mkdir_parents_verbose = false;
  29. mode_t mkdir_parents_filemask;
  30. mode_t mode = 00755;
  31. uid_t user = (uid_t)-1;
  32. gid_t group = (gid_t)-1;
  33. const char *argv0 = "install";
  34. bool opt_v = false;
  35. static int
  36. do_install(char *src, char *dest, bool is_dir)
  37. {
  38. int src_fd = open(src, O_RDONLY);
  39. if(src_fd < 0)
  40. {
  41. fprintf(stderr,
  42. "%s: error: Failed opening file '%s' for reading: %s\n",
  43. argv0,
  44. src,
  45. strerror(errno));
  46. return -1;
  47. }
  48. struct stat src_stat;
  49. if(fstat(src_fd, &src_stat) < 0)
  50. {
  51. fprintf(stderr,
  52. "%s: error: Failed getting status for source '%s': %s\n",
  53. argv0,
  54. dest,
  55. strerror(errno));
  56. return -1;
  57. }
  58. if(!is_dir)
  59. {
  60. struct stat dest_stat;
  61. if(stat(dest, &dest_stat) < 0)
  62. {
  63. if(errno != ENOENT)
  64. {
  65. fprintf(stderr,
  66. "%s: error: Failed getting status for destination '%s': %s\n",
  67. argv0,
  68. dest,
  69. strerror(errno));
  70. return -1;
  71. }
  72. else
  73. errno = 0;
  74. }
  75. else
  76. {
  77. if(S_ISDIR(dest_stat.st_mode))
  78. {
  79. is_dir = true;
  80. }
  81. else
  82. {
  83. if(unlink(dest) < 0)
  84. {
  85. fprintf(stderr,
  86. "%s: error: Failed removing existing file at destination '%s': %s\n",
  87. argv0,
  88. dest,
  89. strerror(errno));
  90. return -1;
  91. }
  92. }
  93. }
  94. }
  95. char *dest_path = dest;
  96. if(is_dir)
  97. {
  98. static char target[PATH_MAX] = "";
  99. char *src_basename = static_basename(src);
  100. if(snprintf(target, PATH_MAX, "%s/%s", dest, src_basename) < 0)
  101. {
  102. fprintf(stderr,
  103. "%s: error: Failed joining destination '%s' and source '%s'\n",
  104. argv0,
  105. dest,
  106. src_basename);
  107. return -1;
  108. }
  109. dest_path = target;
  110. }
  111. int dest_fd = open(dest_path, O_CREAT | O_WRONLY | O_TRUNC, mode);
  112. if(dest_fd < 0)
  113. {
  114. fprintf(stderr,
  115. "%s: error: Failed create-opening file '%s' for writing: %s\n",
  116. argv0,
  117. dest_path,
  118. strerror(errno));
  119. return -1;
  120. }
  121. if(opt_v) fprintf(stderr, "%s: Made file: %s\n", argv0, dest);
  122. if(auto_file_copy(src_fd, dest_fd, src_stat.st_size, 0) < 0)
  123. {
  124. fprintf(stderr,
  125. "%s: error: Failed copying data from '%s' to '%s': %s\n",
  126. argv0,
  127. src,
  128. dest_path,
  129. strerror(errno));
  130. return -1;
  131. }
  132. if(user != (uid_t)-1 || group != (gid_t)-1)
  133. if(fchown(dest_fd, user, group) < 0)
  134. {
  135. fprintf(stderr,
  136. "%s: error: Failed changing ownership of '%s': %s\n",
  137. argv0,
  138. dest_path,
  139. strerror(errno));
  140. return -1;
  141. }
  142. if(preserve_times)
  143. {
  144. struct timespec src_times[2] = {src_stat.st_atim, src_stat.st_mtim};
  145. if(futimens(dest_fd, src_times) != 0)
  146. {
  147. fprintf(stderr,
  148. "%s: error: Failed setting access/modification times on '%s': %s\n",
  149. argv0,
  150. dest_path,
  151. strerror(errno));
  152. return -1;
  153. }
  154. }
  155. if(close(src_fd) < 0)
  156. {
  157. fprintf(stderr, "%s: error: Failed closing '%s'\n", argv0, src);
  158. return -1;
  159. }
  160. if(close(dest_fd) < 0)
  161. {
  162. fprintf(stderr, "%s: error: Failed closing '%s'\n", argv0, dest_path);
  163. return -1;
  164. }
  165. return 0;
  166. }
  167. static void
  168. usage(void)
  169. {
  170. fprintf(stderr, "\
  171. Usage: install [-CcDpTv] [-g group] [-m mode] [-o owner] source... destination\n\
  172. install [-CcDpTv] [-g group] [-m mode] [-o owner] -t destination source...\n\
  173. install -d [-cv] [-g group] [-m mode] [-o owner] directory...\n\
  174. ");
  175. }
  176. int
  177. main(int argc, char *argv[])
  178. {
  179. const char *errstr = NULL;
  180. bool create_parents = false;
  181. bool opt_T = false;
  182. char *dest = NULL;
  183. mkdir_parents_filemask = umask(0);
  184. umask(mkdir_parents_filemask);
  185. #ifdef HAS_GETOPT_LONG
  186. // Strictly for GNUisms compatibility so no long-only options
  187. // clang-format off
  188. static struct option opts[] = {
  189. {"compare", no_argument, 0, 'c'},
  190. {"directory", no_argument, 0, 'd'},
  191. {"group", required_argument, 0, 'g'},
  192. {"mode", required_argument, 0, 'm'},
  193. {"owner", required_argument, 0, 'o'},
  194. {"preserve-timestamps", no_argument, 0, 'p'},
  195. {"target-directory", required_argument, 0, 't'},
  196. {"no-target-directory", no_argument, 0, 'T'},
  197. {"verbose", no_argument, 0, 'v'},
  198. {0, 0, 0, 0},
  199. };
  200. // clang-format on
  201. // Need + as first character to get POSIX-style option parsing
  202. for(int c = -1; (c = getopt_long(argc, argv, "+:CcDdpTt:g:m:o:v", opts, NULL)) != -1;)
  203. #else
  204. for(int c = -1; (c = getopt_nolong(argc, argv, ":CcDdpTt:g:m:o:v")) != -1;)
  205. #endif
  206. {
  207. switch(c)
  208. {
  209. case 'C':
  210. // ignore, for compatibility
  211. break;
  212. case 'c':
  213. // ignore, modern default behavior
  214. break;
  215. case 'D':
  216. create_parents = true;
  217. break;
  218. case 'd':
  219. create_directories = true;
  220. break;
  221. case 'p':
  222. preserve_times = true;
  223. break;
  224. case 'g':
  225. if(parse_group(optarg, &group) != 0) return 1;
  226. break;
  227. case 'o':
  228. if(parse_group(optarg, &user) != 0) return 1;
  229. break;
  230. case 'T':
  231. opt_T = true;
  232. break;
  233. case 't':
  234. dest = optarg;
  235. break;
  236. case 'm':
  237. mode = new_mode(optarg, 0755, &errstr);
  238. if(errstr != NULL)
  239. {
  240. fprintf(stderr, "install: error: Failed parsing mode '%s': %s\n", optarg, errstr);
  241. return 1;
  242. }
  243. break;
  244. case 'v':
  245. opt_v = true;
  246. mkdir_parents_verbose = true;
  247. break;
  248. case ':':
  249. fprintf(stderr, "install: error: Missing operand for option: '-%c'\n", optopt);
  250. usage();
  251. return 1;
  252. case '?':
  253. if(!got_long_opt) fprintf(stderr, "install: error: Unknown option '-%c'\n", optopt);
  254. usage();
  255. return 1;
  256. }
  257. }
  258. argc -= optind;
  259. argv += optind;
  260. if(create_directories)
  261. {
  262. for(int i = 0; i < argc; i++)
  263. {
  264. char *dest = argv[i];
  265. if(mkdir_parents(dest, mode) != 0) return -1;
  266. if(opt_v) fprintf(stderr, "%s: Made directory: %s\n", argv0, dest);
  267. if(user != (uid_t)-1 || group != (gid_t)-1)
  268. {
  269. if(chown(dest, user, group) < 0)
  270. {
  271. fprintf(stderr,
  272. "%s: error: Failed changing ownership of '%s': %s\n",
  273. argv0,
  274. dest,
  275. strerror(errno));
  276. return -1;
  277. }
  278. }
  279. }
  280. return 0;
  281. }
  282. if(dest == NULL) dest = argv[--argc];
  283. bool multi_src = argc > 1;
  284. if(opt_T)
  285. {
  286. if(argc != 1)
  287. {
  288. fprintf(stderr,
  289. "%s: error: Option -T passed, expected exactly 1 source operand, got %d\n",
  290. argv0,
  291. argc);
  292. return 1;
  293. }
  294. assert(!multi_src);
  295. }
  296. else if(argc < 1)
  297. {
  298. fprintf(stderr, "%s: error: Expected at least 1 source operand\n", argv0);
  299. return 1;
  300. }
  301. if(create_parents)
  302. {
  303. char *destdir = dest;
  304. // Same as in mkdir_parents
  305. mode_t parent_mode = (S_IWUSR | S_IXUSR | ~mkdir_parents_filemask) & 0777;
  306. if(!multi_src)
  307. {
  308. char path_dup[PATH_MAX] = "";
  309. strncpy(path_dup, dest, PATH_MAX);
  310. destdir = dirname(path_dup);
  311. }
  312. if(mkdir_parents(destdir, parent_mode) != 0) return 1;
  313. }
  314. for(int i = 0; i < argc; i++)
  315. {
  316. char *src = argv[i];
  317. if(do_install(src, dest, multi_src) < 0) return 1;
  318. }
  319. return 0;
  320. }