busybox/runit/sv.c

794 lines
21 KiB
C
Raw Normal View History

/*
Copyright (c) 2001-2006, Gerrit Pape
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. The name of the author may not be used to endorse or promote products
derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ''AS IS'' AND ANY EXPRESS OR IMPLIED
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/* Taken from http://smarden.org/runit/sv.8.html:
sv - control and manage services monitored by runsv
sv [-v] [-w sec] command services
/etc/init.d/service [-w sec] command
The sv program reports the current status and controls the state of services
monitored by the runsv(8) supervisor.
services consists of one or more arguments, each argument naming a directory
service used by runsv(8). If service doesn't start with a dot or slash and
doesn't end with a slash, it is searched in the default services directory
/var/service/, otherwise relative to the current directory.
command is one of up, down, status, once, pause, cont, hup, alarm, interrupt,
1, 2, term, kill, or exit, or start, stop, reload, restart, shutdown,
force-stop, force-reload, force-restart, force-shutdown, try-restart.
status
Report the current status of the service, and the appendant log service
if available, to standard output.
up
If the service is not running, start it. If the service stops, restart it.
down
If the service is running, send it the TERM signal, and the CONT signal.
If ./run exits, start ./finish if it exists. After it stops, do not
restart service.
once
If the service is not running, start it. Do not restart it if it stops.
pause cont hup alarm interrupt quit 1 2 term kill
If the service is running, send it the STOP, CONT, HUP, ALRM, INT, QUIT,
USR1, USR2, TERM, or KILL signal respectively.
exit
If the service is running, send it the TERM signal, and the CONT signal.
Do not restart the service. If the service is down, and no log service
exists, runsv(8) exits. If the service is down and a log service exists,
runsv(8) closes the standard input of the log service and waits for it to
terminate. If the log service is down, runsv(8) exits. This command is
ignored if it is given to an appendant log service.
sv actually looks only at the first character of above commands.
Commands compatible to LSB init script actions:
status
Same as status.
start
Same as up, but wait up to 7 seconds for the command to take effect.
Then report the status or timeout. If the script ./check exists in
the service directory, sv runs this script to check whether the service
is up and available; it's considered to be available if ./check exits
with 0.
stop
Same as down, but wait up to 7 seconds for the service to become down.
Then report the status or timeout.
reload
Same as hup, and additionally report the status afterwards.
restart
Send the commands term, cont, and up to the service, and wait up to
7 seconds for the service to restart. Then report the status or timeout.
If the script ./check exists in the service directory, sv runs this script
to check whether the service is up and available again; it's considered
to be available if ./check exits with 0.
shutdown
Same as exit, but wait up to 7 seconds for the runsv(8) process
to terminate. Then report the status or timeout.
force-stop
Same as down, but wait up to 7 seconds for the service to become down.
Then report the status, and on timeout send the service the kill command.
force-reload
Send the service the term and cont commands, and wait up to
7 seconds for the service to restart. Then report the status,
and on timeout send the service the kill command.
force-restart
Send the service the term, cont and up commands, and wait up to
7 seconds for the service to restart. Then report the status, and
on timeout send the service the kill command. If the script ./check
exists in the service directory, sv runs this script to check whether
the service is up and available again; it's considered to be available
if ./check exits with 0.
force-shutdown
Same as exit, but wait up to 7 seconds for the runsv(8) process to
terminate. Then report the status, and on timeout send the service
the kill command.
try-restart
if the service is running, send it the term and cont commands, and wait up to
7 seconds for the service to restart. Then report the status or timeout.
Additional Commands
check
Check for the service to be in the state that's been requested. Wait up to
7 seconds for the service to reach the requested state, then report
the status or timeout. If the requested state of the service is up,
and the script ./check exists in the service directory, sv runs
this script to check whether the service is up and running;
it's considered to be up if ./check exits with 0.
Options
-v
If the command is up, down, term, once, cont, or exit, then wait up to 7
seconds for the command to take effect. Then report the status or timeout.
-w sec
Override the default timeout of 7 seconds with sec seconds. Implies -v.
Environment
SVDIR
The environment variable $SVDIR overrides the default services directory
/var/service.
SVWAIT
The environment variable $SVWAIT overrides the default 7 seconds to wait
for a command to take effect. It is overridden by the -w option.
Exit Codes
sv exits 0, if the command was successfully sent to all services, and,
if it was told to wait, the command has taken effect to all services.
For each service that caused an error (e.g. the directory is not
controlled by a runsv(8) process, or sv timed out while waiting),
sv increases the exit code by one and exits non zero. The maximum
is 99. sv exits 100 on error.
*/
/* Busyboxed by Denys Vlasenko <vda.linux@googlemail.com> */
//config:config SV
//config: bool "sv (8.5 kb)"
//config: default y
//config: help
//config: sv reports the current status and controls the state of services
//config: monitored by the runsv supervisor.
//config:
//config:config SV_DEFAULT_SERVICE_DIR
//config: string "Default directory for services"
//config: default "/var/service"
//config: depends on SV || SVC || SVOK
//config: help
//config: Default directory for services.
//config: Defaults to "/var/service"
//config:
//config:config SVC
//config: bool "svc (8.4 kb)"
//config: default y
//config: help
//config: svc controls the state of services monitored by the runsv supervisor.
//config: It is compatible with daemontools command with the same name.
//config:
//config:config SVOK
//config: bool "svok (1.5 kb)"
//config: default y
//config: help
//config: svok checks whether runsv supervisor is running.
//config: It is compatible with daemontools command with the same name.
//applet:IF_SV( APPLET_NOEXEC(sv, sv, BB_DIR_USR_BIN, BB_SUID_DROP, sv ))
//applet:IF_SVC( APPLET_NOEXEC(svc, svc, BB_DIR_USR_BIN, BB_SUID_DROP, svc ))
//applet:IF_SVOK(APPLET_NOEXEC(svok, svok, BB_DIR_USR_BIN, BB_SUID_DROP, svok))
//kbuild:lib-$(CONFIG_SV) += sv.o
//kbuild:lib-$(CONFIG_SVC) += sv.o
//kbuild:lib-$(CONFIG_SVOK) += sv.o
2006-11-17 18:58:49 +00:00
#include <sys/file.h>
#include "libbb.h"
#include "common_bufsiz.h"
2006-11-17 18:58:49 +00:00
#include "runit_lib.h"
struct globals {
const char *acts;
char **service;
unsigned rc;
/* "Bernstein" time format: unix + 0x400000000000000aULL */
uint64_t tstart, tnow;
svstatus_t svstatus;
smallint islog;
} FIX_ALIASING;
#define G (*(struct globals*)bb_common_bufsiz1)
#define acts (G.acts )
#define service (G.service )
#define rc (G.rc )
#define tstart (G.tstart )
#define tnow (G.tnow )
#define svstatus (G.svstatus )
#define islog (G.islog )
#define INIT_G() do { \
setup_common_bufsiz(); \
/* need to zero out, svc calls sv() repeatedly */ \
memset(&G, 0, sizeof(G)); \
} while (0)
2006-11-17 18:58:49 +00:00
#define str_equal(s,t) (strcmp((s), (t)) == 0)
#if ENABLE_SV || ENABLE_SVC
2008-07-05 09:18:54 +00:00
static void fatal_cannot(const char *m1) NORETURN;
static void fatal_cannot(const char *m1)
2006-11-17 18:58:49 +00:00
{
2008-07-21 13:46:54 +00:00
bb_perror_msg("fatal: can't %s", m1);
2006-11-17 18:58:49 +00:00
_exit(151);
}
static void out(const char *p, const char *m1)
2006-11-17 18:58:49 +00:00
{
printf("%s%s%s: %s", p, *service, islog ? "/log" : "", m1);
2006-11-17 18:58:49 +00:00
if (errno) {
printf(": "STRERROR_FMT STRERROR_ERRNO);
2006-11-17 18:58:49 +00:00
}
bb_putchar('\n'); /* will also flush the output */
2006-11-17 18:58:49 +00:00
}
#define WARN "warning: "
#define OK "ok: "
2007-04-16 22:32:04 +00:00
static void fail(const char *m1)
{
++rc;
out("fail: ", m1);
}
2007-04-16 22:32:04 +00:00
static void failx(const char *m1)
{
errno = 0;
fail(m1);
}
static void warn(const char *m1)
2007-04-16 22:32:04 +00:00
{
++rc;
/* "warning: <service>: <m1>\n" */
out("warning: ", m1);
}
2007-04-16 22:32:04 +00:00
static void ok(const char *m1)
{
errno = 0;
out(OK, m1);
}
2006-11-17 18:58:49 +00:00
static int svstatus_get(void)
{
int fd, r;
fd = open("supervise/ok", O_WRONLY|O_NDELAY);
if (fd == -1) {
2006-11-17 18:58:49 +00:00
if (errno == ENODEV) {
*acts == 'x' ? ok("runsv not running")
: failx("runsv not running");
return 0;
}
warn("can't open supervise/ok");
2006-11-17 18:58:49 +00:00
return -1;
}
close(fd);
fd = open("supervise/status", O_RDONLY|O_NDELAY);
if (fd == -1) {
warn("can't open supervise/status");
2006-11-17 18:58:49 +00:00
return -1;
}
r = read(fd, &svstatus, 20);
2006-11-17 18:58:49 +00:00
close(fd);
switch (r) {
case 20:
break;
case -1:
warn("can't read supervise/status");
return -1;
default:
errno = 0;
warn("can't read supervise/status: bad format");
return -1;
2006-11-17 18:58:49 +00:00
}
return 1;
}
static unsigned svstatus_print(const char *m)
2006-11-17 18:58:49 +00:00
{
int diff;
2006-11-17 18:58:49 +00:00
int pid;
int normallyup = 0;
struct stat s;
uint64_t timestamp;
2006-11-17 18:58:49 +00:00
if (stat("down", &s) == -1) {
if (errno != ENOENT) {
bb_perror_msg(WARN"can't stat %s/down", *service);
2006-11-17 18:58:49 +00:00
return 0;
}
normallyup = 1;
}
pid = SWAP_LE32(svstatus.pid_le32);
timestamp = SWAP_BE64(svstatus.time_be64);
switch (svstatus.run_or_finish) {
case 0: printf("down: "); break;
case 1: printf("run: "); break;
case 2: printf("finish: "); break;
2006-11-17 18:58:49 +00:00
}
printf("%s: ", m);
if (svstatus.run_or_finish)
printf("(pid %d) ", pid);
diff = tnow - timestamp;
printf("%us", (diff < 0 ? 0 : diff));
if (pid) {
if (!normallyup) printf(", normally down");
if (svstatus.paused) printf(", paused");
if (svstatus.want == 'd') printf(", want down");
if (svstatus.got_term) printf(", got TERM");
} else {
if (normallyup) printf(", normally up");
if (svstatus.want == 'u') printf(", want up");
2006-11-17 18:58:49 +00:00
}
return pid ? 1 : 2;
}
2008-07-05 09:18:54 +00:00
static int status(const char *unused UNUSED_PARAM)
2006-11-17 18:58:49 +00:00
{
int r;
if (svstatus_get() <= 0)
return 0;
2006-11-17 18:58:49 +00:00
r = svstatus_print(*service);
islog = 1;
2006-11-17 18:58:49 +00:00
if (chdir("log") == -1) {
if (errno != ENOENT) {
printf("; ");
warn("can't change directory");
} else
bb_putchar('\n');
} else {
2006-11-17 18:58:49 +00:00
printf("; ");
if (svstatus_get()) {
r = svstatus_print("log");
bb_putchar('\n');
}
2006-11-17 18:58:49 +00:00
}
islog = 0;
2006-11-17 18:58:49 +00:00
return r;
}
static int checkscript(void)
{
char *prog[2];
struct stat s;
int pid, w;
if (stat("check", &s) == -1) {
if (errno == ENOENT) return 1;
bb_perror_msg(WARN"can't stat %s/check", *service);
2006-11-17 18:58:49 +00:00
return 0;
}
/* if (!(s.st_mode & S_IXUSR)) return 1; */
prog[0] = (char*)"./check";
prog[1] = NULL;
pid = spawn(prog);
if (pid <= 0) {
bb_perror_msg(WARN"can't %s child %s/check", "run", *service);
2006-11-17 18:58:49 +00:00
return 0;
}
while (safe_waitpid(pid, &w, 0) == -1) {
bb_perror_msg(WARN"can't %s child %s/check", "wait for", *service);
2006-11-17 18:58:49 +00:00
return 0;
}
return WEXITSTATUS(w) == 0;
2006-11-17 18:58:49 +00:00
}
static int check(const char *a)
2006-11-17 18:58:49 +00:00
{
int r;
unsigned pid_le32;
uint64_t timestamp;
2006-11-17 18:58:49 +00:00
r = svstatus_get();
if (r == -1)
return -1;
while (*a) {
if (r == 0) {
if (*a == 'x')
return 1;
return -1;
}
pid_le32 = svstatus.pid_le32;
switch (*a) {
case 'x':
2006-11-17 18:58:49 +00:00
return 0;
case 'u':
if (!pid_le32 || svstatus.run_or_finish != 1)
return 0;
if (!checkscript())
return 0;
break;
case 'd':
if (pid_le32 || svstatus.run_or_finish != 0)
return 0;
break;
case 'C':
if (pid_le32 && !checkscript())
return 0;
break;
case 't':
case 'k':
if (!pid_le32 && svstatus.want == 'd')
break;
timestamp = SWAP_BE64(svstatus.time_be64);
if ((tstart > timestamp) || !pid_le32 || svstatus.got_term || !checkscript())
return 0;
break;
case 'o':
timestamp = SWAP_BE64(svstatus.time_be64);
if ((!pid_le32 && tstart > timestamp) || (pid_le32 && svstatus.want != 'd'))
return 0;
break;
case 'p':
if (pid_le32 && !svstatus.paused)
return 0;
break;
case 'c':
if (pid_le32 && svstatus.paused)
return 0;
break;
}
++a;
2006-11-17 18:58:49 +00:00
}
printf(OK);
svstatus_print(*service);
bb_putchar('\n'); /* will also flush the output */
2006-11-17 18:58:49 +00:00
return 1;
}
static int control(const char *a)
2006-11-17 18:58:49 +00:00
{
int fd, r, l;
if (svstatus_get() <= 0)
return -1;
if (svstatus.want == *a && (*a != 'd' || svstatus.got_term == 1))
return 0;
fd = open("supervise/control", O_WRONLY|O_NDELAY);
if (fd == -1) {
2006-11-17 18:58:49 +00:00
if (errno != ENODEV)
warn("can't open supervise/control");
2006-11-17 18:58:49 +00:00
else
*a == 'x' ? ok("runsv not running") : failx("runsv not running");
return -1;
}
l = strlen(a);
r = write(fd, a, l);
2006-11-17 18:58:49 +00:00
close(fd);
if (r != l) {
warn("can't write to supervise/control");
2006-11-17 18:58:49 +00:00
return -1;
}
return 1;
}
//usage:#define sv_trivial_usage
//usage: "[-v] [-w SEC] CMD SERVICE_DIR..."
//usage:#define sv_full_usage "\n\n"
//usage: "Control services monitored by runsv supervisor.\n"
//usage: "Commands (only first character is enough):\n"
//usage: "\n"
//usage: "status: query service status\n"
//usage: "up: if service isn't running, start it. If service stops, restart it\n"
//usage: "once: like 'up', but if service stops, don't restart it\n"
//usage: "down: send TERM and CONT signals. If ./run exits, start ./finish\n"
//usage: " if it exists. After it stops, don't restart service\n"
//usage: "exit: send TERM and CONT signals to service and log service. If they exit,\n"
//usage: " runsv exits too\n"
//usage: "pause, cont, hup, alarm, interrupt, quit, 1, 2, term, kill: send\n"
//usage: "STOP, CONT, HUP, ALRM, INT, QUIT, USR1, USR2, TERM, KILL signal to service"
static int sv(char **argv)
2006-11-17 18:58:49 +00:00
{
char *x;
char *action;
const char *varservice = CONFIG_SV_DEFAULT_SERVICE_DIR;
unsigned waitsec = 7;
smallint kll = 0;
int verbose = 0;
int (*act)(const char*);
int (*cbk)(const char*);
int curdir;
2006-11-17 18:58:49 +00:00
INIT_G();
xfunc_error_retval = 100;
x = getenv("SVDIR");
if (x) varservice = x;
x = getenv("SVWAIT");
if (x) waitsec = xatou(x);
getopt32: remove opt_complementary function old new delta vgetopt32 1318 1392 +74 runsvdir_main 703 713 +10 bb_make_directory 423 425 +2 collect_cpu 546 545 -1 opt_chars 3 - -3 opt_complementary 4 - -4 tftpd_main 567 562 -5 ntp_init 476 471 -5 zcip_main 1266 1256 -10 xxd_main 428 418 -10 whois_main 140 130 -10 who_main 463 453 -10 which_main 212 202 -10 wget_main 2535 2525 -10 watchdog_main 291 281 -10 watch_main 222 212 -10 vlock_main 399 389 -10 uuencode_main 332 322 -10 uudecode_main 316 306 -10 unlink_main 45 35 -10 udhcpd_main 1482 1472 -10 udhcpc_main 2762 2752 -10 tune2fs_main 290 280 -10 tunctl_main 366 356 -10 truncate_main 218 208 -10 tr_main 518 508 -10 time_main 1134 1124 -10 tftp_main 286 276 -10 telnetd_main 1873 1863 -10 tcpudpsvd_main 1785 1775 -10 taskset_main 521 511 -10 tar_main 1009 999 -10 tail_main 1644 1634 -10 syslogd_main 1967 1957 -10 switch_root_main 368 358 -10 svlogd_main 1454 1444 -10 sv 1296 1286 -10 stat_main 104 94 -10 start_stop_daemon_main 1028 1018 -10 split_main 542 532 -10 sort_main 796 786 -10 slattach_main 624 614 -10 shuf_main 504 494 -10 setsid_main 96 86 -10 setserial_main 1132 1122 -10 setfont_main 388 378 -10 setconsole_main 78 68 -10 sendmail_main 1209 1199 -10 sed_main 677 667 -10 script_main 1077 1067 -10 run_parts_main 325 315 -10 rtcwake_main 454 444 -10 rm_main 175 165 -10 reformime_main 119 109 -10 readlink_main 123 113 -10 rdate_main 246 236 -10 pwdx_main 189 179 -10 pstree_main 317 307 -10 pscan_main 663 653 -10 popmaildir_main 818 808 -10 pmap_main 80 70 -10 nc_main 1042 1032 -10 mv_main 558 548 -10 mountpoint_main 477 467 -10 mount_main 1264 1254 -10 modprobe_main 768 758 -10 modinfo_main 333 323 -10 mktemp_main 200 190 -10 mkswap_main 324 314 -10 mkfs_vfat_main 1489 1479 -10 microcom_main 715 705 -10 md5_sha1_sum_main 521 511 -10 man_main 867 857 -10 makedevs_main 1052 1042 -10 ls_main 563 553 -10 losetup_main 432 422 -10 loadfont_main 89 79 -10 ln_main 524 514 -10 link_main 75 65 -10 ipcalc_main 544 534 -10 iostat_main 2397 2387 -10 install_main 768 758 -10 id_main 480 470 -10 i2cset_main 1239 1229 -10 i2cget_main 380 370 -10 i2cdump_main 1482 1472 -10 i2cdetect_main 682 672 -10 hwclock_main 406 396 -10 httpd_main 741 731 -10 grep_main 837 827 -10 getty_main 1559 1549 -10 fuser_main 297 287 -10 ftpgetput_main 345 335 -10 ftpd_main 2232 2222 -10 fstrim_main 251 241 -10 fsfreeze_main 77 67 -10 fsck_minix_main 2921 2911 -10 flock_main 314 304 -10 flashcp_main 740 730 -10 flash_eraseall_main 833 823 -10 fdformat_main 532 522 -10 expand_main 680 670 -10 eject_main 335 325 -10 dumpleases_main 630 620 -10 du_main 314 304 -10 dos2unix_main 441 431 -10 diff_main 1350 1340 -10 df_main 1064 1054 -10 date_main 1095 1085 -10 cut_main 961 951 -10 cryptpw_main 228 218 -10 crontab_main 575 565 -10 crond_main 1149 1139 -10 cp_main 370 360 -10 common_traceroute_main 3834 3824 -10 common_ping_main 1767 1757 -10 comm_main 239 229 -10 cmp_main 655 645 -10 chrt_main 379 369 -10 chpst_main 704 694 -10 chpasswd_main 308 298 -10 chown_main 171 161 -10 chmod_main 158 148 -10 cat_main 428 418 -10 bzip2_main 120 110 -10 blkdiscard_main 264 254 -10 base64_main 221 211 -10 arping_main 1665 1655 -10 ar_main 556 546 -10 adjtimex_main 406 396 -10 adduser_main 882 872 -10 addgroup_main 411 401 -10 acpid_main 1198 1188 -10 optstring 11 - -11 opt_string 18 - -18 OPT_STR 25 - -25 ubi_tools_main 1288 1258 -30 ls_options 31 - -31 ------------------------------------------------------------------------------ (add/remove: 0/6 grow/shrink: 3/129 up/down: 86/-1383) Total: -1297 bytes text data bss dec hex filename 915428 485 6876 922789 e14a5 busybox_old 914629 485 6872 921986 e1182 busybox_unstripped Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-08-08 21:55:02 +02:00
getopt32(argv, "^" "w:+v" "\0" "vv" /* -w N, -v is a counter */,
&waitsec, &verbose
);
argv += optind;
action = *argv++;
if (!action || !*argv) bb_show_usage();
2006-11-17 18:58:49 +00:00
tnow = time(NULL) + 0x400000000000000aULL;
tstart = tnow;
curdir = open(".", O_RDONLY|O_NDELAY);
if (curdir == -1)
2006-11-17 18:58:49 +00:00
fatal_cannot("open current directory");
act = &control;
acts = "s";
cbk = &check;
2006-11-17 18:58:49 +00:00
switch (*action) {
case 'x':
case 'e':
acts = "x";
if (!verbose) cbk = NULL;
break;
case 'X':
case 'E':
acts = "x";
kll = 1;
break;
2006-11-17 18:58:49 +00:00
case 'D':
acts = "d";
kll = 1;
break;
2006-11-17 18:58:49 +00:00
case 'T':
acts = "tc";
kll = 1;
break;
case 't':
if (str_equal(action, "try-restart")) {
acts = "tc";
break;
}
2006-11-17 18:58:49 +00:00
case 'c':
if (str_equal(action, "check")) {
act = NULL;
acts = "C";
2006-11-17 18:58:49 +00:00
break;
}
case 'u': case 'd': case 'o': case 'p': case 'h':
2006-11-17 18:58:49 +00:00
case 'a': case 'i': case 'k': case 'q': case '1': case '2':
action[1] = '\0';
acts = action;
if (!verbose)
cbk = NULL;
break;
2006-11-17 18:58:49 +00:00
case 's':
if (str_equal(action, "shutdown")) {
2006-11-17 18:58:49 +00:00
acts = "x";
break;
}
if (str_equal(action, "start")) {
2006-11-17 18:58:49 +00:00
acts = "u";
break;
}
if (str_equal(action, "stop")) {
2006-11-17 18:58:49 +00:00
acts = "d";
break;
}
/* "status" */
act = &status;
cbk = NULL;
break;
2006-11-17 18:58:49 +00:00
case 'r':
if (str_equal(action, "restart")) {
2006-11-17 18:58:49 +00:00
acts = "tcu";
break;
}
if (str_equal(action, "reload")) {
acts = "h";
break;
}
bb_show_usage();
2006-11-17 18:58:49 +00:00
case 'f':
if (str_equal(action, "force-reload")) {
acts = "tc";
kll = 1;
break;
}
if (str_equal(action, "force-restart")) {
acts = "tcu";
kll = 1;
break;
}
if (str_equal(action, "force-shutdown")) {
acts = "x";
kll = 1;
break;
}
if (str_equal(action, "force-stop")) {
acts = "d";
kll = 1;
break;
}
2006-11-17 18:58:49 +00:00
default:
bb_show_usage();
2006-11-17 18:58:49 +00:00
}
service = argv;
while ((x = *service) != NULL) {
if (x[0] != '/' && x[0] != '.'
&& !last_char_is(x, '/')
) {
if (chdir(varservice) == -1)
goto chdir_failed_0;
}
if (chdir(x) == -1) {
chdir_failed_0:
fail("can't change to service directory");
goto nullify_service_0;
}
if (act && (act(acts) == -1)) {
nullify_service_0:
*service = (char*) -1L; /* "dead" */
2006-11-17 18:58:49 +00:00
}
if (fchdir(curdir) == -1)
fatal_cannot("change to original directory");
2006-11-17 18:58:49 +00:00
service++;
}
if (cbk) while (1) {
int want_exit;
int diff;
diff = tnow - tstart;
service = argv;
want_exit = 1;
while ((x = *service) != NULL) {
if (x == (char*) -1L) /* "dead" */
goto next;
if (x[0] != '/' && x[0] != '.') {
if (chdir(varservice) == -1)
goto chdir_failed;
}
if (chdir(x) == -1) {
2007-01-27 22:21:52 +00:00
chdir_failed:
fail("can't change to service directory");
goto nullify_service;
}
if (cbk(acts) != 0)
goto nullify_service;
want_exit = 0;
if (diff >= waitsec) {
printf(kll ? "kill: " : "timeout: ");
if (svstatus_get() > 0) {
svstatus_print(x);
++rc;
2006-11-17 18:58:49 +00:00
}
bb_putchar('\n'); /* will also flush the output */
if (kll)
control("k");
2007-01-27 22:21:52 +00:00
nullify_service:
*service = (char*) -1L; /* "dead" */
2006-11-17 18:58:49 +00:00
}
if (fchdir(curdir) == -1)
fatal_cannot("change to original directory");
next:
service++;
2006-11-17 18:58:49 +00:00
}
if (want_exit) break;
usleep(420000);
tnow = time(NULL) + 0x400000000000000aULL;
2007-01-27 22:21:52 +00:00
}
2006-11-17 18:58:49 +00:00
return rc > 99 ? 99 : rc;
}
#endif
#if ENABLE_SV
int sv_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
int sv_main(int argc UNUSED_PARAM, char **argv)
{
return sv(argv);
}
#endif
//usage:#define svc_trivial_usage
//usage: "[-udopchaitkx] SERVICE_DIR..."
//usage:#define svc_full_usage "\n\n"
//usage: "Control services monitored by runsv supervisor"
//usage: "\n"
//usage: "\n"" -u If service is not running, start it; restart if it stops"
//usage: "\n"" -d If service is running, send TERM+CONT signals; do not restart it"
//usage: "\n"" -o Once: if service is not running, start it; do not restart it"
//usage: "\n"" -pchaitk Send STOP, CONT, HUP, ALRM, INT, TERM, KILL signal to service"
//usage: "\n"" -x Exit: runsv will exit as soon as the service is down"
#if ENABLE_SVC
int svc_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
int svc_main(int argc UNUSED_PARAM, char **argv)
{
char command[2];
const char *optstring;
unsigned opts;
optstring = "udopchaitkx";
opts = getopt32(argv, optstring);
argv += optind;
if (!argv[0] || !opts)
bb_show_usage();
argv -= 2;
if (optind > 2) {
argv--;
argv[2] = (char*)"--";
}
argv[0] = (char*)"sv";
argv[1] = command;
command[1] = '\0';
do {
if (opts & 1) {
int r;
command[0] = *optstring;
/* getopt() was already called by getopt32():
* reset the libc getopt() function's internal state.
*/
GETOPT_RESET();
r = sv(argv);
if (r)
return 1;
}
optstring++;
opts >>= 1;
} while (opts);
return 0;
}
#endif
//usage:#define svok_trivial_usage
//usage: "SERVICE_DIR"
//usage:#define svok_full_usage "\n\n"
//usage: "Check whether runsv supervisor is running.\n"
//usage: "Exit code is 0 if it does, 100 if it does not,\n"
//usage: "111 (with error message) if SERVICE_DIR does not exist."
#if ENABLE_SVOK
int svok_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
int svok_main(int argc UNUSED_PARAM, char **argv)
{
const char *dir = argv[1];
if (!dir)
bb_show_usage();
xfunc_error_retval = 111;
/*
* daemontools has no concept of "default service dir", runit does.
* Let's act as runit.
*/
if (dir[0] != '/' && dir[0] != '.'
&& !last_char_is(dir, '/')
) {
xchdir(CONFIG_SV_DEFAULT_SERVICE_DIR);
}
xchdir(dir);
if (open("supervise/ok", O_WRONLY) < 0) {
if (errno == ENOENT || errno == ENXIO)
return 100;
bb_perror_msg_and_die("can't open '%s'", "supervise/ok");
}
return 0;
}
#endif