2007-04-05 11:18:42 +00:00
|
|
|
/*
|
2009-04-23 21:31:22 +00:00
|
|
|
rc-update
|
|
|
|
Manage init scripts and runlevels
|
|
|
|
*/
|
2007-04-05 11:18:42 +00:00
|
|
|
|
2008-01-14 05:05:22 +00:00
|
|
|
/*
|
2009-01-12 23:53:13 +00:00
|
|
|
* Copyright 2007-2009 Roy Marples <roy@marples.name>
|
2007-11-14 15:22:04 +00:00
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``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 OR CONTRIBUTORS 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.
|
|
|
|
*/
|
|
|
|
|
2007-04-05 11:18:42 +00:00
|
|
|
#include <errno.h>
|
2007-08-09 09:53:47 +00:00
|
|
|
#include <getopt.h>
|
2007-04-05 11:18:42 +00:00
|
|
|
#include <limits.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2007-08-04 15:05:12 +00:00
|
|
|
#include "builtins.h"
|
2008-01-07 14:20:13 +00:00
|
|
|
#include "einfo.h"
|
2007-04-05 11:18:42 +00:00
|
|
|
#include "rc.h"
|
2008-01-07 14:20:13 +00:00
|
|
|
#include "rc-misc.h"
|
2007-04-05 11:18:42 +00:00
|
|
|
|
2008-02-11 20:14:09 +00:00
|
|
|
extern const char *applet;
|
2007-04-05 11:18:42 +00:00
|
|
|
|
2007-09-25 17:03:19 +00:00
|
|
|
/* Return the number of changes made:
|
|
|
|
* -1 = no changes (error)
|
|
|
|
* 0 = no changes (nothing to do)
|
|
|
|
* 1+ = number of runlevels updated
|
|
|
|
*/
|
2009-01-10 12:18:00 +00:00
|
|
|
static int
|
|
|
|
add(const char *runlevel, const char *service)
|
2007-04-05 11:18:42 +00:00
|
|
|
{
|
2007-12-14 20:09:53 +00:00
|
|
|
int retval = -1;
|
2007-04-11 12:44:47 +00:00
|
|
|
|
2009-02-28 14:12:19 +00:00
|
|
|
if (!rc_service_exists(service)) {
|
|
|
|
if (errno == ENOEXEC)
|
2009-04-23 21:31:22 +00:00
|
|
|
eerror("%s: service `%s' is not executeable",
|
|
|
|
applet, service);
|
2009-02-28 14:12:19 +00:00
|
|
|
else
|
2009-04-23 21:31:22 +00:00
|
|
|
eerror("%s: service `%s' does not exist",
|
|
|
|
applet, service);
|
2009-02-28 14:12:19 +00:00
|
|
|
} else if (rc_service_in_runlevel(service, runlevel)) {
|
2007-08-09 09:53:47 +00:00
|
|
|
ewarn ("%s: %s already installed in runlevel `%s'; skipping",
|
2009-04-23 21:31:22 +00:00
|
|
|
applet, service, runlevel);
|
2007-09-25 17:03:19 +00:00
|
|
|
retval = 0;
|
2009-02-28 14:12:19 +00:00
|
|
|
} else if (rc_service_add(runlevel, service)) {
|
2007-04-11 12:44:47 +00:00
|
|
|
einfo ("%s added to runlevel %s", service, runlevel);
|
2007-09-25 17:03:19 +00:00
|
|
|
retval = 1;
|
2007-08-08 22:35:32 +00:00
|
|
|
} else
|
2009-02-28 14:12:19 +00:00
|
|
|
eerror("%s: failed to add service `%s' to runlevel `%s': %s",
|
2009-04-23 21:31:22 +00:00
|
|
|
applet, service, runlevel, strerror (errno));
|
2007-04-11 12:44:47 +00:00
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
return retval;
|
2007-04-05 11:18:42 +00:00
|
|
|
}
|
|
|
|
|
2009-01-10 12:18:00 +00:00
|
|
|
static int
|
|
|
|
delete(const char *runlevel, const char *service)
|
2007-08-08 22:35:32 +00:00
|
|
|
{
|
2007-12-14 20:09:53 +00:00
|
|
|
int retval = -1;
|
2007-08-08 22:35:32 +00:00
|
|
|
|
2007-12-14 12:24:16 +00:00
|
|
|
errno = 0;
|
2009-02-28 14:12:19 +00:00
|
|
|
if (rc_service_delete(runlevel, service)) {
|
|
|
|
einfo("%s removed from runlevel %s", service, runlevel);
|
2007-12-14 12:24:16 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (errno == ENOENT)
|
|
|
|
eerror ("%s: service `%s' is not in the runlevel `%s'",
|
2009-04-23 21:31:22 +00:00
|
|
|
applet, service, runlevel);
|
2008-01-14 05:05:22 +00:00
|
|
|
else
|
2007-12-14 12:24:16 +00:00
|
|
|
eerror ("%s: failed to remove service `%s' from runlevel `%s': %s",
|
2009-04-23 21:31:22 +00:00
|
|
|
applet, service, runlevel, strerror (errno));
|
2007-09-25 17:03:19 +00:00
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
return retval;
|
2007-08-08 22:35:32 +00:00
|
|
|
}
|
|
|
|
|
2009-01-10 12:18:00 +00:00
|
|
|
static void
|
|
|
|
show(RC_STRINGLIST *runlevels, bool verbose)
|
2007-04-05 11:18:42 +00:00
|
|
|
{
|
2008-03-16 17:00:56 +00:00
|
|
|
RC_STRINGLIST *services = rc_services_in_runlevel(NULL);
|
|
|
|
RC_STRING *service;
|
|
|
|
RC_STRING *runlevel;
|
|
|
|
RC_STRINGLIST *in;
|
|
|
|
bool inone;
|
|
|
|
char buffer[PATH_MAX];
|
|
|
|
size_t l;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(service, services, entries) {
|
|
|
|
in = rc_stringlist_new();
|
|
|
|
inone = false;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(runlevel, runlevels, entries) {
|
|
|
|
if (rc_service_in_runlevel(service->value,
|
2009-04-23 21:31:22 +00:00
|
|
|
runlevel->value))
|
2008-03-16 17:00:56 +00:00
|
|
|
{
|
|
|
|
rc_stringlist_add(in, runlevel->value);
|
2007-08-09 09:53:47 +00:00
|
|
|
inone = true;
|
|
|
|
} else {
|
2008-03-16 17:00:56 +00:00
|
|
|
l = strlen(runlevel->value);
|
|
|
|
memset (buffer, ' ', l);
|
|
|
|
buffer[l] = 0;
|
|
|
|
rc_stringlist_add (in, buffer);
|
2007-08-09 09:53:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
if (inone || verbose) {
|
|
|
|
printf(" %20s |", service->value);
|
|
|
|
TAILQ_FOREACH(runlevel, in, entries)
|
2009-04-23 21:31:22 +00:00
|
|
|
printf (" %s", runlevel->value);
|
2008-03-16 17:00:56 +00:00
|
|
|
printf ("\n");
|
|
|
|
}
|
|
|
|
rc_stringlist_free(in);
|
2007-08-09 09:53:47 +00:00
|
|
|
}
|
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
rc_stringlist_free (services);
|
2007-08-09 09:53:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#include "_usage.h"
|
2009-04-23 21:31:22 +00:00
|
|
|
#define usagestring "" \
|
|
|
|
"Usage: rc-update [options] add service <runlevel>\n" \
|
|
|
|
" rc-update [options] del service <runlevel>\n" \
|
2008-01-15 09:50:14 +00:00
|
|
|
" rc-update [options] show"
|
2009-01-12 23:53:13 +00:00
|
|
|
#define getoptstring "u" getoptstring_COMMON
|
2008-02-01 19:54:46 +00:00
|
|
|
static const struct option longopts[] = {
|
2009-01-12 23:53:13 +00:00
|
|
|
{ "update", 0, NULL, 'u' },
|
2007-08-09 09:53:47 +00:00
|
|
|
longopts_COMMON
|
|
|
|
};
|
2007-09-25 16:21:38 +00:00
|
|
|
static const char * const longopts_help[] = {
|
2009-01-12 23:53:13 +00:00
|
|
|
"Force an update of the dependency tree",
|
2007-09-25 16:21:38 +00:00
|
|
|
longopts_help_COMMON
|
|
|
|
};
|
2007-08-09 09:53:47 +00:00
|
|
|
#include "_usage.c"
|
|
|
|
|
2007-12-18 18:01:05 +00:00
|
|
|
#define DOADD (1 << 1)
|
|
|
|
#define DODELETE (1 << 2)
|
|
|
|
#define DOSHOW (1 << 3)
|
2007-08-09 09:53:47 +00:00
|
|
|
|
2009-01-10 12:18:00 +00:00
|
|
|
int
|
|
|
|
rc_update(int argc, char **argv)
|
2007-08-09 09:53:47 +00:00
|
|
|
{
|
2008-03-16 17:00:56 +00:00
|
|
|
RC_STRINGLIST *runlevels;
|
|
|
|
RC_STRING *runlevel;
|
2007-08-09 09:53:47 +00:00
|
|
|
char *service = NULL;
|
2008-03-16 17:00:56 +00:00
|
|
|
char *p;
|
2007-08-09 09:53:47 +00:00
|
|
|
int action = 0;
|
|
|
|
bool verbose = false;
|
|
|
|
int opt;
|
|
|
|
int retval = EXIT_FAILURE;
|
2008-03-16 17:00:56 +00:00
|
|
|
int num_updated = 0;
|
|
|
|
int (*actfunc)(const char *, const char *);
|
|
|
|
int ret;
|
2007-04-11 12:44:47 +00:00
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
while ((opt = getopt_long(argc, argv, getoptstring,
|
2009-04-23 21:31:22 +00:00
|
|
|
longopts, (int *)0)) != -1)
|
2007-08-09 09:53:47 +00:00
|
|
|
switch (opt) {
|
2009-01-12 23:53:13 +00:00
|
|
|
case 'u':
|
|
|
|
_rc_deptree_load(-1, &ret);
|
|
|
|
return ret;
|
2009-04-23 21:31:22 +00:00
|
|
|
case_RC_COMMON_GETOPT
|
|
|
|
}
|
2007-08-09 09:53:47 +00:00
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
verbose = rc_yesno(getenv ("EINFO_VERBOSE"));
|
2007-09-25 23:17:25 +00:00
|
|
|
|
2007-08-09 09:53:47 +00:00
|
|
|
if ((action & DOSHOW && action != DOSHOW) ||
|
2008-01-11 15:51:40 +00:00
|
|
|
(action & DOADD && action != DOADD) ||
|
|
|
|
(action & DODELETE && action != DODELETE))
|
2008-03-16 17:00:56 +00:00
|
|
|
eerrorx("%s: cannot mix commands", applet);
|
2007-08-09 09:53:47 +00:00
|
|
|
|
|
|
|
/* We need to be backwards compatible */
|
2008-01-15 09:50:14 +00:00
|
|
|
if (optind < argc) {
|
2008-03-16 17:00:56 +00:00
|
|
|
if (strcmp(argv[optind], "add") == 0)
|
2008-01-15 09:50:14 +00:00
|
|
|
action = DOADD;
|
2008-03-16 17:00:56 +00:00
|
|
|
else if (strcmp(argv[optind], "delete") == 0 ||
|
2009-04-23 21:31:22 +00:00
|
|
|
strcmp(argv[optind], "del") == 0)
|
2008-01-15 09:50:14 +00:00
|
|
|
action = DODELETE;
|
2008-03-16 17:00:56 +00:00
|
|
|
else if (strcmp(argv[optind], "show") == 0)
|
2008-01-15 09:50:14 +00:00
|
|
|
action = DOSHOW;
|
|
|
|
if (action)
|
|
|
|
optind++;
|
|
|
|
else
|
2009-04-23 21:31:22 +00:00
|
|
|
eerrorx("%s: invalid command `%s'",
|
|
|
|
applet, argv[optind]);
|
2007-08-09 09:53:47 +00:00
|
|
|
}
|
2009-01-10 12:18:00 +00:00
|
|
|
if (!action)
|
2008-01-15 09:50:14 +00:00
|
|
|
action = DOSHOW;
|
2007-08-09 09:53:47 +00:00
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
runlevels = rc_stringlist_new();
|
|
|
|
|
2007-08-09 09:53:47 +00:00
|
|
|
if (optind >= argc) {
|
2009-04-30 16:45:18 +01:00
|
|
|
if (!(action & DOSHOW))
|
2008-03-16 17:00:56 +00:00
|
|
|
eerrorx("%s: no service specified", applet);
|
2007-08-09 09:53:47 +00:00
|
|
|
} else {
|
|
|
|
service = argv[optind];
|
|
|
|
optind++;
|
|
|
|
|
|
|
|
while (optind < argc)
|
2008-03-16 17:00:56 +00:00
|
|
|
if (rc_runlevel_exists(argv[optind]))
|
|
|
|
rc_stringlist_add(runlevels, argv[optind++]);
|
2007-08-09 09:53:47 +00:00
|
|
|
else {
|
2008-03-16 17:00:56 +00:00
|
|
|
rc_stringlist_free(runlevels);
|
|
|
|
eerrorx ("%s: `%s' is not a valid runlevel",
|
2009-04-23 21:31:22 +00:00
|
|
|
applet, argv[optind]);
|
2007-08-09 09:53:47 +00:00
|
|
|
}
|
|
|
|
}
|
2007-04-11 12:44:47 +00:00
|
|
|
|
2007-09-25 17:03:19 +00:00
|
|
|
retval = EXIT_SUCCESS;
|
2007-08-09 09:53:47 +00:00
|
|
|
if (action & DOSHOW) {
|
|
|
|
if (service)
|
2008-03-16 17:00:56 +00:00
|
|
|
rc_stringlist_add(runlevels, service);
|
2009-01-10 12:18:00 +00:00
|
|
|
if (!TAILQ_FIRST(runlevels)) {
|
2008-03-16 17:00:56 +00:00
|
|
|
free(runlevels);
|
|
|
|
runlevels = rc_runlevel_list();
|
|
|
|
}
|
2007-04-11 12:44:47 +00:00
|
|
|
|
2007-08-09 09:53:47 +00:00
|
|
|
show (runlevels, verbose);
|
|
|
|
} else {
|
2009-01-10 12:18:00 +00:00
|
|
|
if (!service)
|
2007-08-09 09:53:47 +00:00
|
|
|
eerror ("%s: no service specified", applet);
|
|
|
|
else {
|
2007-12-08 12:10:45 +00:00
|
|
|
if (action & DOADD) {
|
2007-09-25 17:03:19 +00:00
|
|
|
actfunc = add;
|
2007-12-08 12:10:45 +00:00
|
|
|
} else if (action & DODELETE) {
|
2007-09-25 17:03:19 +00:00
|
|
|
actfunc = delete;
|
2008-03-16 17:00:56 +00:00
|
|
|
} else {
|
|
|
|
rc_stringlist_free(runlevels);
|
2009-01-10 12:18:00 +00:00
|
|
|
eerrorx("%s: invalid action", applet);
|
2008-03-16 17:00:56 +00:00
|
|
|
}
|
2007-12-08 12:10:45 +00:00
|
|
|
|
2009-01-10 12:18:00 +00:00
|
|
|
if (!TAILQ_FIRST(runlevels)) {
|
2008-03-16 17:00:56 +00:00
|
|
|
p = rc_runlevel_get();
|
|
|
|
rc_stringlist_add(runlevels, p);
|
|
|
|
free(p);
|
|
|
|
}
|
2007-12-14 23:00:29 +00:00
|
|
|
|
2009-01-10 12:18:00 +00:00
|
|
|
if (!TAILQ_FIRST(runlevels)) {
|
2008-03-16 17:00:56 +00:00
|
|
|
free(runlevels);
|
2009-04-23 21:31:22 +00:00
|
|
|
eerrorx("%s: no runlevels found", applet);
|
2008-03-16 17:00:56 +00:00
|
|
|
}
|
2007-12-08 12:10:45 +00:00
|
|
|
|
2009-01-10 12:18:00 +00:00
|
|
|
TAILQ_FOREACH(runlevel, runlevels, entries) {
|
|
|
|
if (!rc_runlevel_exists(runlevel->value)) {
|
2008-03-16 17:00:56 +00:00
|
|
|
eerror ("%s: runlevel `%s' does not exist",
|
2009-04-23 21:31:22 +00:00
|
|
|
applet, runlevel->value);
|
2007-12-14 12:24:16 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
ret = actfunc(runlevel->value, service);
|
2007-09-25 17:03:19 +00:00
|
|
|
if (ret < 0)
|
|
|
|
retval = EXIT_FAILURE;
|
|
|
|
num_updated += ret;
|
2007-04-11 12:44:47 +00:00
|
|
|
}
|
2007-12-08 12:10:45 +00:00
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
if (retval == EXIT_SUCCESS &&
|
|
|
|
num_updated == 0 && action & DODELETE)
|
|
|
|
ewarnx("%s: service `%s' not found in any"
|
2009-04-23 21:31:22 +00:00
|
|
|
" of the specified runlevels",
|
|
|
|
applet, service);
|
2007-04-11 12:44:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-16 17:00:56 +00:00
|
|
|
rc_stringlist_free(runlevels);
|
|
|
|
return retval;
|
2007-04-05 11:18:42 +00:00
|
|
|
}
|