logo

utils-std

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

realpath.c (4717B)


  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. #define _POSIX_C_SOURCE 200809L
  5. #define _XOPEN_SOURCE 700 // realpath is in XSI
  6. #include "../lib/fs.h"
  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(), puts()
  12. #include <stdlib.h> // realpath()
  13. #include <string.h> // strncmp(), strnlen, strerror
  14. #include <unistd.h> // getopt
  15. static bool must_exists = false;
  16. static bool offline = false;
  17. const char *argv0 = NULL;
  18. static char sep = '\n';
  19. static int
  20. print_realpath(char *path)
  21. {
  22. char *file = NULL;
  23. if(offline)
  24. file = offline_realpath(path, NULL);
  25. else
  26. file = realpath(path, NULL);
  27. if(file)
  28. {
  29. if(printf("%s", file) < 0)
  30. {
  31. fprintf(stderr, "%s: error: Failed writing to stdout: %s\n", argv0, strerror(errno));
  32. free(file);
  33. return 1;
  34. }
  35. free(file);
  36. return 0;
  37. }
  38. if(must_exists || errno != ENOENT)
  39. {
  40. fprintf(stderr, "%s: error: Failed canonilizing \"%s\": %s\n", argv0, path, strerror(errno));
  41. return 1;
  42. }
  43. char *child = path_split_static(path, true);
  44. if(child == NULL)
  45. {
  46. // Return as if realpath just failed
  47. fprintf(stderr, "%s: error: Failed canonilizing \"%s\": %s\n", argv0, path, strerror(errno));
  48. return 1;
  49. }
  50. errno = 0;
  51. char *parent = realpath(path, NULL);
  52. if(!parent)
  53. {
  54. fprintf(stderr,
  55. "%s: error: Failed canonilizing parent of full path \"%s/%s\": %s\n",
  56. argv0,
  57. path,
  58. child,
  59. strerror(errno));
  60. return 1;
  61. }
  62. if(printf("%s/%s", parent, child) < 0)
  63. {
  64. fprintf(stderr, "%s: error: Failed writing to stdout: %s\n", argv0, strerror(errno));
  65. free(parent);
  66. return 1;
  67. }
  68. free(parent);
  69. return 0;
  70. }
  71. static void
  72. usage_realpath(void)
  73. {
  74. fprintf(stderr, "Usage: realpath [-E|-e] [-n|-z] path...\n");
  75. }
  76. static int
  77. main_realpath(int argc, char *argv[])
  78. {
  79. must_exists = false;
  80. int offset_sep = 0;
  81. for(int c = -1; (c = getopt_nolong(argc, argv, ":Eemnsz")) != -1;)
  82. {
  83. switch(c)
  84. {
  85. case 'E':
  86. must_exists = false;
  87. break;
  88. case 'e':
  89. must_exists = true;
  90. break;
  91. case 'n':
  92. offset_sep = 1;
  93. break;
  94. case 's':
  95. offline = true;
  96. break;
  97. case 'z':
  98. sep = '\0';
  99. break;
  100. case ':':
  101. fprintf(stderr, "%s: error: Missing operand for option: '-%c'\n", argv0, optopt);
  102. usage_realpath();
  103. return 1;
  104. case '?':
  105. if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option: '-%c'\n", argv0, optopt);
  106. usage_realpath();
  107. return 1;
  108. }
  109. }
  110. argv += optind;
  111. argc -= optind;
  112. if(argc == 0)
  113. {
  114. fprintf(stderr, "%s: error: Expected one file as argument, got 0\n", argv0);
  115. usage_realpath();
  116. return 1;
  117. }
  118. for(int i = 0; i < argc; i++)
  119. {
  120. if(print_realpath(argv[i]) != 0) return 1;
  121. if(i != argc - offset_sep) printf("%c", sep);
  122. }
  123. return 0;
  124. }
  125. static void
  126. usage_readlink(void)
  127. {
  128. fprintf(stderr, "Usage: readlink [-f|-e] [-n|-z] file...\n");
  129. }
  130. static int
  131. main_readlink(int argc, char *argv[])
  132. {
  133. must_exists = true;
  134. bool canonicalize = false;
  135. int offset_sep = 0;
  136. for(int c = -1; (c = getopt_nolong(argc, argv, ":femnz")) != -1;)
  137. {
  138. switch(c)
  139. {
  140. case 'f':
  141. canonicalize = true;
  142. must_exists = false;
  143. break;
  144. case 'e':
  145. must_exists = true;
  146. break;
  147. case 'n':
  148. offset_sep = 1;
  149. break;
  150. case 'z':
  151. sep = '\0';
  152. break;
  153. case ':':
  154. fprintf(stderr, "%s: error: Missing operand for option: '-%c'\n", argv0, optopt);
  155. usage_readlink();
  156. return 1;
  157. case '?':
  158. if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option: '-%c'\n", argv0, optopt);
  159. usage_readlink();
  160. return 1;
  161. }
  162. }
  163. argv += optind;
  164. argc -= optind;
  165. if(argc == 0)
  166. {
  167. fprintf(stderr, "%s: error: Expected one file as argument, got 0\n", argv0);
  168. usage_readlink();
  169. return 1;
  170. }
  171. for(int i = 0; i < argc; i++)
  172. {
  173. char *path = argv[i];
  174. if(canonicalize)
  175. {
  176. if(print_realpath(path) != 0) return 1;
  177. if(i != argc) printf("%c", sep);
  178. continue;
  179. }
  180. char buf[PATH_MAX] = "";
  181. if(readlink(path, buf, sizeof(buf) - 1) < 0)
  182. {
  183. fprintf(stderr,
  184. "%s: error: Failed reading symbolic link of '%s': %s\n",
  185. argv0,
  186. path,
  187. strerror(errno));
  188. return 1;
  189. }
  190. printf("%s", buf);
  191. if(i != argc - offset_sep) printf("%c", sep);
  192. }
  193. return 0;
  194. }
  195. int
  196. main(int argc, char *argv[])
  197. {
  198. argv0 = static_basename(argv[0]);
  199. if(strcmp(argv0, "realpath") == 0) return main_realpath(argc, argv);
  200. if(strcmp(argv0, "readlink") == 0) return main_readlink(argc, argv);
  201. fprintf(stderr, "%s: error: Unknown utility '%s', expected realpath or readlink\n", argv0, argv0);
  202. return 1;
  203. }