2008-03-21 09:10:59 +00:00
|
|
|
/*
|
2009-04-23 21:31:22 +00:00
|
|
|
rc-service.c
|
|
|
|
Finds all OpenRC services
|
|
|
|
*/
|
2008-03-21 09:10:59 +00:00
|
|
|
|
|
|
|
/*
|
2009-05-01 15:11:40 +01:00
|
|
|
* Copyright (c) 2008 Roy Marples <roy@marples.name>
|
2011-06-29 19:46:31 -04:00
|
|
|
*
|
2008-03-21 09:10:59 +00:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <getopt.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "builtins.h"
|
|
|
|
#include "einfo.h"
|
|
|
|
#include "rc.h"
|
|
|
|
#include "rc-misc.h"
|
|
|
|
|
|
|
|
extern char *applet;
|
|
|
|
|
|
|
|
#include "_usage.h"
|
2009-05-01 08:38:57 +01:00
|
|
|
#define getoptstring "e:ilr:" getoptstring_COMMON
|
2008-03-21 09:10:59 +00:00
|
|
|
static const struct option longopts[] = {
|
2009-05-01 08:38:57 +01:00
|
|
|
{ "exists", 1, NULL, 'e' },
|
|
|
|
{ "ifexists", 0, NULL, 'i' },
|
|
|
|
{ "list", 0, NULL, 'l' },
|
|
|
|
{ "resolve", 1, NULL, 'r' },
|
2008-03-21 09:10:59 +00:00
|
|
|
longopts_COMMON
|
|
|
|
};
|
|
|
|
static const char * const longopts_help[] = {
|
2008-04-07 11:48:32 +00:00
|
|
|
"tests if the service exists or not",
|
2009-05-01 08:38:57 +01:00
|
|
|
"if the service exsits then run the command",
|
2008-03-21 09:10:59 +00:00
|
|
|
"list all available services",
|
2008-04-15 16:16:59 +00:00
|
|
|
"resolve the service name to an init script",
|
2008-03-21 09:10:59 +00:00
|
|
|
longopts_help_COMMON
|
|
|
|
};
|
|
|
|
#include "_usage.c"
|
|
|
|
|
2008-10-10 08:37:21 +00:00
|
|
|
int
|
|
|
|
rc_service(int argc, char **argv)
|
2008-03-21 09:10:59 +00:00
|
|
|
{
|
|
|
|
int opt;
|
|
|
|
char *service;
|
|
|
|
RC_STRINGLIST *list;
|
|
|
|
RC_STRING *s;
|
2009-05-01 08:38:57 +01:00
|
|
|
bool if_exists = false;
|
2008-03-21 09:10:59 +00:00
|
|
|
|
|
|
|
/* Ensure that we are only quiet when explicitly told to be */
|
|
|
|
unsetenv("EINFO_QUIET");
|
|
|
|
|
|
|
|
while ((opt = getopt_long(argc, argv, getoptstring,
|
2009-04-23 21:31:22 +00:00
|
|
|
longopts, (int *) 0)) != -1)
|
2008-03-21 09:10:59 +00:00
|
|
|
{
|
|
|
|
switch (opt) {
|
2008-03-26 08:08:47 +00:00
|
|
|
case 'e':
|
|
|
|
service = rc_service_resolve(optarg);
|
|
|
|
opt = service ? EXIT_SUCCESS : EXIT_FAILURE;
|
2008-10-10 08:37:21 +00:00
|
|
|
#ifdef DEBUG_MEMORY
|
2008-03-26 08:08:47 +00:00
|
|
|
free(service);
|
2008-10-10 08:37:21 +00:00
|
|
|
#endif
|
2008-03-26 08:08:47 +00:00
|
|
|
return opt;
|
|
|
|
/* NOTREACHED */
|
2009-05-01 08:38:57 +01:00
|
|
|
case 'i':
|
|
|
|
if_exists = true;
|
|
|
|
break;
|
2008-03-21 09:10:59 +00:00
|
|
|
case 'l':
|
|
|
|
list = rc_services_in_runlevel(NULL);
|
2009-05-01 08:38:57 +01:00
|
|
|
if (TAILQ_FIRST(list) == NULL)
|
2008-03-21 09:10:59 +00:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
rc_stringlist_sort(&list);
|
|
|
|
TAILQ_FOREACH(s, list, entries)
|
2009-04-23 21:31:22 +00:00
|
|
|
printf("%s\n", s->value);
|
2008-10-10 08:37:21 +00:00
|
|
|
#ifdef DEBUG_MEMORY
|
2008-03-21 09:10:59 +00:00
|
|
|
rc_stringlist_free(list);
|
2008-10-10 08:37:21 +00:00
|
|
|
#endif
|
2008-03-21 09:10:59 +00:00
|
|
|
return EXIT_SUCCESS;
|
|
|
|
/* NOTREACHED */
|
2008-03-26 08:08:47 +00:00
|
|
|
case 'r':
|
|
|
|
service = rc_service_resolve(optarg);
|
2009-05-01 08:38:57 +01:00
|
|
|
if (service == NULL)
|
2008-03-26 08:08:47 +00:00
|
|
|
return EXIT_FAILURE;
|
|
|
|
printf("%s\n", service);
|
2008-10-10 08:37:21 +00:00
|
|
|
#ifdef DEBUG_MEMORY
|
2008-03-26 08:08:47 +00:00
|
|
|
free(service);
|
2008-10-10 08:37:21 +00:00
|
|
|
#endif
|
2008-03-26 08:08:47 +00:00
|
|
|
return EXIT_SUCCESS;
|
|
|
|
/* NOTREACHED */
|
2008-03-21 09:10:59 +00:00
|
|
|
|
2009-04-23 21:31:22 +00:00
|
|
|
case_RC_COMMON_GETOPT
|
|
|
|
}
|
2008-03-21 09:10:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
argc -= optind;
|
|
|
|
argv += optind;
|
2009-05-01 08:38:57 +01:00
|
|
|
if (*argv == NULL)
|
2008-03-21 09:10:59 +00:00
|
|
|
eerrorx("%s: you need to specify a service", applet);
|
2009-05-01 08:38:57 +01:00
|
|
|
if ((service = rc_service_resolve(*argv)) == NULL) {
|
|
|
|
if (if_exists)
|
|
|
|
return 0;
|
2008-03-21 09:10:59 +00:00
|
|
|
eerrorx("%s: service `%s' does not exist", applet, *argv);
|
2009-05-01 08:38:57 +01:00
|
|
|
}
|
2008-03-21 09:10:59 +00:00
|
|
|
*argv = service;
|
|
|
|
execv(*argv, argv);
|
|
|
|
eerrorx("%s: %s", applet, strerror(errno));
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|