logo

utils-std

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

head.c (5382B)


  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. #include "../lib/fs.h" // auto_fd_copy
  6. #include "../lib/getopt_nolong.h"
  7. #include "../lib/truncation.h" // apply_size_suffix
  8. #include <ctype.h> // isdigit
  9. #include <errno.h>
  10. #include <fcntl.h> // open
  11. #include <stdio.h> // fprintf, fopen, fread
  12. #include <stdlib.h> // strtoul
  13. #include <string.h> // strerror
  14. #include <unistd.h> // read
  15. static const char *header_fmt = "==> %s <==\n";
  16. const char *argv0 = "head";
  17. size_t lines = 10;
  18. size_t bytes = 0;
  19. char *buf = NULL;
  20. size_t buflen = 0;
  21. int delim = '\n';
  22. static int
  23. copy_bytes(const char *filename)
  24. {
  25. int fd = -1;
  26. if(filename[0] == '-' && filename[1] == 0)
  27. {
  28. fd = STDIN_FILENO;
  29. }
  30. else
  31. {
  32. fd = open(filename, O_RDONLY | O_CLOEXEC | O_NOCTTY);
  33. if(fd < 0)
  34. {
  35. fprintf(
  36. stderr, "%s: error: Failed opening file '%s': %s\n", argv0, filename, strerror(errno));
  37. return 1;
  38. }
  39. }
  40. int err = 0;
  41. if(auto_fd_copy(fd, STDOUT_FILENO, bytes) < 0)
  42. {
  43. fprintf(stderr,
  44. "%s: error: Failed copying data from '%s' to <stdout>: %s\n",
  45. argv0,
  46. filename,
  47. strerror(errno));
  48. err = 1;
  49. }
  50. if(fd != STDIN_FILENO)
  51. if(close(fd) != 0)
  52. {
  53. fprintf(
  54. stderr, "%s: error: Failed closing file '%s': %s\n", argv0, filename, strerror(errno));
  55. return 1;
  56. }
  57. return err;
  58. }
  59. static int
  60. copy_lines(const char *filename)
  61. {
  62. int err = 0;
  63. FILE *in = NULL;
  64. if(filename[0] == '-' && filename[1] == 0)
  65. {
  66. in = stdin;
  67. }
  68. else
  69. {
  70. in = fopen(filename, "r");
  71. if(in == NULL)
  72. {
  73. fprintf(
  74. stderr, "%s: error: Failed opening file '%s': %s\n", argv0, filename, strerror(errno));
  75. return 1;
  76. }
  77. }
  78. for(size_t i = 0; i < lines; i++)
  79. {
  80. ssize_t nread = getdelim(&buf, &buflen, delim, in);
  81. if(nread < 0)
  82. {
  83. if(errno == 0) break;
  84. fprintf(stderr,
  85. "%s: error: Failed reading line from '%s': %s\n",
  86. argv0,
  87. filename,
  88. strerror(errno));
  89. err = 1;
  90. break;
  91. }
  92. if(write(STDOUT_FILENO, buf, nread) < 0)
  93. {
  94. fprintf(stderr,
  95. "%s: error: Failed writing line from '%s' to stdout: %s\n",
  96. argv0,
  97. filename,
  98. strerror(errno));
  99. err = 1;
  100. break;
  101. }
  102. }
  103. if(in != stdin)
  104. if(fclose(in) != 0)
  105. {
  106. fprintf(
  107. stderr, "%s: error: Failed closing file '%s': %s\n", argv0, filename, strerror(errno));
  108. return 1;
  109. }
  110. return err;
  111. }
  112. static void
  113. usage(void)
  114. {
  115. fprintf(stderr, "Usage: head [-qvz] [-c size | -n num | -num] [file...]\n");
  116. }
  117. int
  118. main(int argc, char *argv[])
  119. {
  120. int (*copy_action)(const char *filename) = copy_lines;
  121. int print_header = 0;
  122. while(optind < argc)
  123. {
  124. /* Skip getopt(3) for parsing -num option due to how it works:
  125. * - optstring = "12"; args="head -12" considered the same as -1 -2
  126. * - optstring = "1:"; args="head -12" considered the same as -1 2
  127. *
  128. * And we need args="head -12" as "12"
  129. */
  130. if(argv[optind] && argv[optind][0] == '-' && isdigit(argv[optind][1]))
  131. {
  132. char *arg = argv[optind] + 1;
  133. char *endptr = NULL;
  134. lines = strtoul(arg, &endptr, 0);
  135. if(!(errno == 0 && endptr != NULL && *endptr == '\0'))
  136. {
  137. fprintf(stderr,
  138. "%s: error: Failed parsing number for `-%s`: %s\n",
  139. argv0,
  140. arg,
  141. strerror(errno));
  142. return 1;
  143. }
  144. optind++;
  145. continue;
  146. }
  147. int c = getopt_nolong(argc, argv, ":qvc:n:z");
  148. if(c == -1) break;
  149. switch(c)
  150. {
  151. case 'q':
  152. print_header = -1;
  153. break;
  154. case 'v':
  155. print_header = 1;
  156. break;
  157. case 'c':
  158. {
  159. char *endptr = NULL;
  160. unsigned long size = strtoul(optarg, &endptr, 0);
  161. if(errno != 0)
  162. {
  163. fprintf(stderr,
  164. "%s: error: Failed parsing number for `-n %s`: %s\n",
  165. argv0,
  166. optarg,
  167. strerror(errno));
  168. return 1;
  169. }
  170. if(endptr != NULL && *endptr != 0)
  171. if(apply_size_suffix(&size, endptr) != 0) return 1;
  172. bytes = size;
  173. copy_action = &copy_bytes;
  174. break;
  175. }
  176. case 'n':
  177. {
  178. char *endptr = NULL;
  179. lines = strtoul(optarg, &endptr, 0);
  180. if(!(errno == 0 && endptr != NULL && *endptr == '\0'))
  181. {
  182. fprintf(stderr,
  183. "%s: error: Failed parsing number for `-n %s`: %s\n",
  184. argv0,
  185. optarg,
  186. strerror(errno));
  187. return 1;
  188. }
  189. break;
  190. }
  191. case 'z':
  192. delim = '\0';
  193. break;
  194. case ':':
  195. fprintf(stderr, "%s: error: Missing operand for option: '-%c'\n", argv0, optopt);
  196. usage();
  197. return 1;
  198. case '?':
  199. if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option: '-%c'\n", argv0, optopt);
  200. usage();
  201. return 1;
  202. default:
  203. abort();
  204. }
  205. }
  206. argc -= optind;
  207. argv += optind;
  208. if(argc <= 0)
  209. {
  210. const char *filename = "-";
  211. if(print_header == 1)
  212. {
  213. printf(header_fmt, filename);
  214. fflush(stdout);
  215. }
  216. int err = copy_action(filename);
  217. if(buflen != 0) free(buf);
  218. return err;
  219. }
  220. if(print_header == 0) print_header = argc > 1;
  221. int err = 0;
  222. for(int i = 0; i < argc; i++)
  223. {
  224. char *filename = argv[i];
  225. if(print_header > 0)
  226. {
  227. printf(header_fmt, filename);
  228. header_fmt = "\n==> %s <==\n";
  229. fflush(stdout);
  230. }
  231. if(copy_action(filename) != 0)
  232. {
  233. err = 1;
  234. break;
  235. }
  236. }
  237. if(buflen != 0) free(buf);
  238. return err;
  239. }