logo

utils-std

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

rmdir.c (2822B)


  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 "../config.h" // HAS_*
  6. #include "../lib/getopt_nolong.h"
  7. #include <errno.h>
  8. #include <stdbool.h>
  9. #include <stdio.h> // fprintf
  10. #include <stdlib.h> // abort
  11. #include <string.h> // strerror, strrchr
  12. #include <unistd.h> // getopt, rmdir
  13. #ifdef HAS_GETOPT_LONG
  14. #include <getopt.h>
  15. #endif
  16. const char *argv0 = "rmdir";
  17. static void
  18. usage(void)
  19. {
  20. #ifdef HAS_GETOPT_LONG
  21. fprintf(stderr, "Usage: rmdir [-pv] [--ignore-fail-on-non-empty] directory...\n");
  22. #else
  23. fprintf(stderr, "Usage: rmdir [-pv] directory...\n");
  24. #endif
  25. }
  26. int
  27. main(int argc, char *argv[])
  28. {
  29. bool parents = false, verbose = false, ign_enotempty = false;
  30. #ifdef HAS_GETOPT_LONG
  31. // clang-format off
  32. enum long_opt_vals {
  33. IGN_ENOTEMPTY = 1
  34. };
  35. static struct option opts[] = {
  36. {"ignore-fail-on-non-empty", no_argument, 0, IGN_ENOTEMPTY},
  37. {"parents", no_argument, 0, 'p'},
  38. {"verbose", no_argument, 0, 'v'},
  39. {0, 0, 0, 0},
  40. };
  41. // clang-format on
  42. // Need + as first character to get POSIX-style option parsing
  43. for(int c = -1; (c = getopt_long(argc, argv, "+:pv", opts, NULL)) != -1;)
  44. #else
  45. for(int c = -1; (c = getopt_nolong(argc, argv, ":pv")) != -1;)
  46. #endif
  47. {
  48. switch(c)
  49. {
  50. #ifdef HAS_GETOPT_LONG
  51. case IGN_ENOTEMPTY:
  52. ign_enotempty = true;
  53. break;
  54. #endif
  55. case 'p':
  56. parents = true;
  57. break;
  58. case 'v':
  59. verbose = true;
  60. break;
  61. case ':':
  62. fprintf(stderr, "rmdir: error: Missing operand for option: '-%c'\n", optopt);
  63. usage();
  64. return 1;
  65. case '?':
  66. if(!got_long_opt) fprintf(stderr, "rmdir: error: Unrecognised option: '-%c'\n", optopt);
  67. usage();
  68. return 1;
  69. default:
  70. abort();
  71. }
  72. }
  73. argc -= optind;
  74. argv += optind;
  75. if(argc == 0)
  76. {
  77. fprintf(stderr, "rmdir: error: missing operand\n");
  78. usage();
  79. return 1;
  80. }
  81. int err = 0;
  82. for(int i = 0; i < argc; i++)
  83. {
  84. errno = 0;
  85. char *path = argv[i];
  86. if(rmdir(path) < 0)
  87. {
  88. if(ign_enotempty && (errno == ENOTEMPTY || errno == EEXIST)) continue;
  89. fprintf(stderr, "rmdir: error: Failed removing '%s': %s\n", path, strerror(errno));
  90. err = 1;
  91. continue;
  92. }
  93. if(verbose) fprintf(stderr, "rmdir: Removed '%s'\n", path);
  94. if(!parents) continue;
  95. while(true)
  96. {
  97. char *sep = strrchr(path, '/');
  98. if(sep == NULL) break;
  99. *sep = 0;
  100. if(*path == 0) break;
  101. errno = 0;
  102. if(rmdir(path) < 0)
  103. {
  104. if(errno == ENOTDIR) break;
  105. if(ign_enotempty && (errno == ENOTEMPTY || errno == EEXIST)) break;
  106. fprintf(stderr, "rmdir: error: Failed removing '%s': %s\n", path, strerror(errno));
  107. err = 1;
  108. break;
  109. }
  110. if(verbose) fprintf(stderr, "rmdir: Removed '%s'\n", path);
  111. }
  112. }
  113. return err;
  114. }