run_parts.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. /* vi: set sw=4 ts=4: */
  2. /*
  3. * Mini run-parts implementation for busybox
  4. *
  5. * Copyright (C) 2007 Bernhard Reutner-Fischer
  6. *
  7. * Based on a older version that was in busybox which was 1k big.
  8. * Copyright (C) 2001 by Emanuele Aina <emanuele.aina@tiscali.it>
  9. *
  10. * Based on the Debian run-parts program, version 1.15
  11. * Copyright (C) 1996 Jeff Noxon <jeff@router.patch.net>,
  12. * Copyright (C) 1996-1999 Guy Maor <maor@debian.org>
  13. *
  14. * Licensed under GPLv2 or later, see file LICENSE in this source tree.
  15. */
  16. /* This is my first attempt to write a program in C (well, this is my first
  17. * attempt to write a program! :-) . */
  18. /* This piece of code is heavily based on the original version of run-parts,
  19. * taken from debian-utils. I've only removed the long options and the
  20. * report mode. As the original run-parts support only long options, I've
  21. * broken compatibility because the BusyBox policy doesn't allow them.
  22. */
  23. //config:config RUN_PARTS
  24. //config: bool "run-parts"
  25. //config: default y
  26. //config: help
  27. //config: run-parts is a utility designed to run all the scripts in a directory.
  28. //config:
  29. //config: It is useful to set up a directory like cron.daily, where you need to
  30. //config: execute all the scripts in that directory.
  31. //config:
  32. //config: In this implementation of run-parts some features (such as report
  33. //config: mode) are not implemented.
  34. //config:
  35. //config: Unless you know that run-parts is used in some of your scripts
  36. //config: you can safely say N here.
  37. //config:
  38. //config:config FEATURE_RUN_PARTS_LONG_OPTIONS
  39. //config: bool "Enable long options"
  40. //config: default y
  41. //config: depends on RUN_PARTS && LONG_OPTS
  42. //config: help
  43. //config: Support long options for the run-parts applet.
  44. //config:
  45. //config:config FEATURE_RUN_PARTS_FANCY
  46. //config: bool "Support additional arguments"
  47. //config: default y
  48. //config: depends on RUN_PARTS
  49. //config: help
  50. //config: Support additional options:
  51. //config: -l --list print the names of the all matching files (not
  52. //config: limited to executables), but don't actually run them.
  53. //applet:IF_RUN_PARTS(APPLET_ODDNAME(run-parts, run_parts, BB_DIR_BIN, BB_SUID_DROP, run_parts))
  54. //kbuild:lib-$(CONFIG_RUN_PARTS) += run_parts.o
  55. //usage:#define run_parts_trivial_usage
  56. //usage: "[-a ARG]... [-u UMASK] "
  57. //usage: IF_FEATURE_RUN_PARTS_LONG_OPTIONS("[--reverse] [--test] [--exit-on-error] "IF_FEATURE_RUN_PARTS_FANCY("[--list] "))
  58. //usage: "DIRECTORY"
  59. //usage:#define run_parts_full_usage "\n\n"
  60. //usage: "Run a bunch of scripts in DIRECTORY\n"
  61. //usage: "\n -a ARG Pass ARG as argument to scripts"
  62. //usage: "\n -u UMASK Set UMASK before running scripts"
  63. //usage: IF_FEATURE_RUN_PARTS_LONG_OPTIONS(
  64. //usage: "\n --reverse Reverse execution order"
  65. //usage: "\n --test Dry run"
  66. //usage: "\n --exit-on-error Exit if a script exits with non-zero"
  67. //usage: IF_FEATURE_RUN_PARTS_FANCY(
  68. //usage: "\n --list Print names of matching files even if they are not executable"
  69. //usage: )
  70. //usage: )
  71. //usage:
  72. //usage:#define run_parts_example_usage
  73. //usage: "$ run-parts -a start /etc/init.d\n"
  74. //usage: "$ run-parts -a stop=now /etc/init.d\n\n"
  75. //usage: "Let's assume you have a script foo/dosomething:\n"
  76. //usage: "#!/bin/sh\n"
  77. //usage: "for i in $*; do eval $i; done; unset i\n"
  78. //usage: "case \"$1\" in\n"
  79. //usage: "start*) echo starting something;;\n"
  80. //usage: "stop*) set -x; shutdown -h $stop;;\n"
  81. //usage: "esac\n\n"
  82. //usage: "Running this yields:\n"
  83. //usage: "$run-parts -a stop=+4m foo/\n"
  84. //usage: "+ shutdown -h +4m"
  85. #include "libbb.h"
  86. #include "common_bufsiz.h"
  87. struct globals {
  88. char **names;
  89. int cur;
  90. char *cmd[2 /* using 1 provokes compiler warning */];
  91. } FIX_ALIASING;
  92. #define G (*(struct globals*)bb_common_bufsiz1)
  93. #define names (G.names)
  94. #define cur (G.cur )
  95. #define cmd (G.cmd )
  96. #define INIT_G() do { setup_common_bufsiz(); } while (0)
  97. enum { NUM_CMD = (COMMON_BUFSIZE - sizeof(G)) / sizeof(cmd[0]) - 1 };
  98. enum {
  99. OPT_a = (1 << 0),
  100. OPT_u = (1 << 1),
  101. OPT_r = (1 << 2) * ENABLE_FEATURE_RUN_PARTS_LONG_OPTIONS,
  102. OPT_t = (1 << 3) * ENABLE_FEATURE_RUN_PARTS_LONG_OPTIONS,
  103. OPT_e = (1 << 4) * ENABLE_FEATURE_RUN_PARTS_LONG_OPTIONS,
  104. OPT_l = (1 << 5) * ENABLE_FEATURE_RUN_PARTS_LONG_OPTIONS
  105. * ENABLE_FEATURE_RUN_PARTS_FANCY,
  106. };
  107. /* Is this a valid filename (upper/lower alpha, digits,
  108. * underscores, and hyphens only?)
  109. */
  110. static bool invalid_name(const char *c)
  111. {
  112. c = bb_basename(c);
  113. while (*c && (isalnum(*c) || *c == '_' || *c == '-'))
  114. c++;
  115. return *c; /* TRUE (!0) if terminating NUL is not reached */
  116. }
  117. static int bb_alphasort(const void *p1, const void *p2)
  118. {
  119. int r = strcmp(*(char **) p1, *(char **) p2);
  120. return (option_mask32 & OPT_r) ? -r : r;
  121. }
  122. static int FAST_FUNC act(const char *file, struct stat *statbuf, void *args UNUSED_PARAM, int depth)
  123. {
  124. if (depth == 1)
  125. return TRUE;
  126. if (depth == 2
  127. && ( !(statbuf->st_mode & (S_IFREG | S_IFLNK))
  128. || invalid_name(file)
  129. || (!(option_mask32 & OPT_l) && access(file, X_OK) != 0))
  130. ) {
  131. return SKIP;
  132. }
  133. names = xrealloc_vector(names, 4, cur);
  134. names[cur++] = xstrdup(file);
  135. /*names[cur] = NULL; - xrealloc_vector did it */
  136. return TRUE;
  137. }
  138. #if ENABLE_FEATURE_RUN_PARTS_LONG_OPTIONS
  139. static const char runparts_longopts[] ALIGN1 =
  140. "arg\0" Required_argument "a"
  141. "umask\0" Required_argument "u"
  142. //TODO: "verbose\0" No_argument "v"
  143. "reverse\0" No_argument "\xf0"
  144. "test\0" No_argument "\xf1"
  145. "exit-on-error\0" No_argument "\xf2"
  146. #if ENABLE_FEATURE_RUN_PARTS_FANCY
  147. "list\0" No_argument "\xf3"
  148. #endif
  149. ;
  150. #endif
  151. int run_parts_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
  152. int run_parts_main(int argc UNUSED_PARAM, char **argv)
  153. {
  154. const char *umask_p = "22";
  155. llist_t *arg_list = NULL;
  156. unsigned n;
  157. int ret;
  158. INIT_G();
  159. #if ENABLE_FEATURE_RUN_PARTS_LONG_OPTIONS
  160. applet_long_options = runparts_longopts;
  161. #endif
  162. /* We require exactly one argument: the directory name */
  163. opt_complementary = "=1:a::";
  164. getopt32(argv, "a:u:", &arg_list, &umask_p);
  165. umask(xstrtou_range(umask_p, 8, 0, 07777));
  166. n = 1;
  167. while (arg_list && n < NUM_CMD) {
  168. cmd[n++] = llist_pop(&arg_list);
  169. }
  170. /* cmd[n] = NULL; - is already zeroed out */
  171. /* run-parts has to sort executables by name before running them */
  172. recursive_action(argv[optind],
  173. ACTION_RECURSE|ACTION_FOLLOWLINKS,
  174. act, /* file action */
  175. act, /* dir action */
  176. NULL, /* user data */
  177. 1 /* depth */
  178. );
  179. if (!names)
  180. return 0;
  181. qsort(names, cur, sizeof(char *), bb_alphasort);
  182. n = 0;
  183. while (1) {
  184. char *name = *names++;
  185. if (!name)
  186. break;
  187. if (option_mask32 & (OPT_t | OPT_l)) {
  188. puts(name);
  189. continue;
  190. }
  191. cmd[0] = name;
  192. ret = spawn_and_wait(cmd);
  193. if (ret == 0)
  194. continue;
  195. n = 1;
  196. if (ret < 0)
  197. bb_perror_msg("can't execute '%s'", name);
  198. else /* ret > 0 */
  199. bb_error_msg("%s: exit status %u", name, ret & 0xff);
  200. if (option_mask32 & OPT_e)
  201. xfunc_die();
  202. }
  203. return n;
  204. }