blob: cca14a396055bb78bebea98b130a411d5e7b1ab4 [file] [log] [blame]
/* libnih
*
* option.c - command-line argument and option parsing
*
* Copyright © 2009 Scott James Remnant <scott@netsplit.com>.
* Copyright © 2009 Canonical Ltd.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2, as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif /* HAVE_CONFIG_H */
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <nih/macros.h>
#include <nih/alloc.h>
#include <nih/string.h>
#include <nih/main.h>
#include <nih/option.h>
#include <nih/logging.h>
/**
* NihOptionCtx:
* @parent: parent object for returned array,
* @argc: number of arguments,
* @argv: arguments array,
* @options: list of options,
* @nargs: number of non-option arguments,
* @args: non-option arguments to return,
* @arg: current argument index,
* @nonopt: index of first non-option argument,
* @optend: index of end of options.
*
* This structure is passed between the various parsing functions to avoid
* having to pass a dozen different variables around and to keep us
* re-entrant.
**/
typedef struct nih_option_ctx {
const void *parent;
int argc;
char **argv;
NihOption *options;
size_t nargs;
char **args;
int arg;
int nonopt;
int optend;
} NihOptionCtx;
/* Prototypes for static functions */
static NihOption * nih_option_get_short (NihOptionCtx *ctx, int option);
static int nih_option_short (NihOptionCtx *ctx);
static NihOption * nih_option_get_long (NihOptionCtx *ctx,
const char *option, size_t len);
static int nih_option_long (NihOptionCtx *ctx);
static int nih_option_handle (NihOptionCtx *ctx, NihOption *opt);
static int nih_option_handle_arg (NihOptionCtx *ctx, NihOption *opt,
const char *arg);
static const char *nih_option_next_nonopt (NihOptionCtx *ctx);
static void nih_option_help (NihOption *options);
static void nih_option_group_help (NihOptionGroup *group,
NihOption *options,
NihOptionGroup **groups);
/**
* default_options:
*
* These default options are appended to those defined by the user
* so they can be overriden.
**/
static const NihOption default_options[] = {
{ 'q', "quiet",
N_("reduce output to errors only"),
NULL, NULL, NULL, nih_option_quiet },
{ 'v', "verbose",
N_("increase output to include informational messages"),
NULL, NULL, NULL, nih_option_verbose },
/* Deliberately hidden, you get told about this when you file
* a bug ;-)
*/
{ 0, "debug", NULL, NULL, NULL, NULL, nih_option_debug },
{ 0, "help",
N_("display this help and exit"),
NULL, NULL, NULL, NULL },
{ 0, "version",
N_("output version information and exit"),
NULL, NULL, NULL, NULL },
NIH_OPTION_LAST
};
/**
* usage_stem:
*
* This string is prepended to the program's usage line if given.
**/
static const char *usage_stem = NULL;
/**
* usage_string:
*
* This string is appended to the program's usage line if given.
**/
static const char *usage_string = NULL;
/**
* synopsis_string:
*
* This string is output after the program's usage string if given.
**/
static const char *synopsis_string = NULL;
/**
* help_string:
*
* This string is output after the options if given.
**/
static const char *help_string = NULL;
/**
* footer_string:
*
* This string is output after the options and help if given.
**/
static const char *footer_string = NULL;
/**
* nih_option_parser:
* @parent: parent object for returned array,
* @argc: number of arguments,
* @argv: command-line arguments,
* @options: options to look for,
* @break_nonopt: stop processing options at first non-option argument.
*
* Parses the command-line arguments given in @argv looking for options
* described in @options, or those built-in. Options are handled according
* to common UNIX semantics so that short options may be grouped together
* and arguments need not immediately follow the option that requires it.
*
* Remaining non-option arguments are placed into an array for processing
* by the caller. If @break_nonopt is FALSE then the first non-option
* argument concludes option processing and all subsequent options are
* considered to be ordinary arguments; this is most useful when the
* first argument should be a command.
*
* Both the array itself, and the array items are allocated with nih_alloc();
* the items are children of the array, so it is only necessary to call
* nih_free() on the array.
*
* If @parent is not NULL, it should be a pointer to another object which
* will be used as a parent for the returned array. When all parents
* of the returned array are freed, the returned array will also be
* freed.
*
* Errors are handled by printing a message to standard error.
*
* Returns: non-option arguments array or NULL on error.
**/
char **
nih_option_parser (const void *parent,
int argc,
char *argv[],
NihOption *options,
int break_nonopt)
{
NihOptionCtx ctx;
nih_assert (argc > 0);
nih_assert (argv != NULL);
nih_assert (options != NULL);
ctx.parent = parent;
ctx.argc = argc;
ctx.argv = argv;
ctx.options = nih_option_join (NULL, options, default_options);
ctx.nargs = 0;
ctx.args = NIH_MUST (nih_str_array_new (parent));
ctx.nonopt = 0;
ctx.optend = 0;
/* Iterate the arguments looking for options */
for (ctx.arg = 1; ctx.arg < argc; ctx.arg++) {
char *arg;
arg = ctx.argv[ctx.arg];
if ((arg[0] != '-') || (arg[1] == '\0')
|| (ctx.optend && ctx.arg > ctx.optend)) {
/* Not an option */
if (ctx.arg > ctx.nonopt) {
NIH_MUST (nih_str_array_add
(&ctx.args, parent, &ctx.nargs,
ctx.argv[ctx.arg]));
if (break_nonopt)
ctx.optend = ctx.arg;
}
} else if (arg[1] != '-') {
/* Short option */
if (nih_option_short (&ctx) < 0)
goto error;
} else if (arg[2] != '\0') {
/* Long option */
if (nih_option_long (&ctx) < 0)
goto error;
} else {
/* End of options */
ctx.optend = ctx.arg;
}
}
nih_free (ctx.options);
return ctx.args;
error:
nih_free (ctx.options);
nih_free (ctx.args);
return NULL;
}
/**
* nih_option_get_short:
* @ctx: parsing context,
* @option: option to find.
*
* Find the option structure with the given short @option. If an option
* exists with the short option '-' this is used instead if no specific
* option is found.
*
* Returns; pointer to option, or NULL if not found.
**/
static NihOption *
nih_option_get_short (NihOptionCtx *ctx,
int option)
{
NihOption *opt, *catch = NULL;
for (opt = ctx->options; (opt->option || opt->long_option); opt++) {
if (opt->option == '-')
catch = opt;
if (opt->option == option)
return opt;
}
return catch;
}
/**
* nih_option_short:
* @ctx: parsing context.
*
* Process the current argument as a list of short options, handling
* each one individually.
*
* If the first option in the list expects a value, then the rest of
* the argument is taken to be the option argument rather than further
* options.
*
* Returns: zero on success, negative value on invalid option.
**/
static int
nih_option_short (NihOptionCtx *ctx)
{
char *ptr;
nih_assert (ctx != NULL);
nih_assert (program_name != NULL);
for (ptr = ctx->argv[ctx->arg] + 1; *ptr != '\0'; ptr++) {
NihOption *opt;
opt = nih_option_get_short (ctx, *ptr);
if (! opt) {
fprintf (stderr, _("%s: invalid option: -%c\n"),
program_name, *ptr);
nih_main_suggest_help ();
return -1;
}
/* If the option takes an argument, this is the first
* option in the list and there are further characters;
* treat the rest as the argument
*/
if (opt->arg_name && (ptr[-1] == '-') && (ptr[1] != '\0')) {
if (nih_option_handle_arg (ctx, opt, ptr + 1) < 0)
return -1;
break;
}
/* Otherwise it's an ordinary option */
if (nih_option_handle (ctx, opt) < 0)
return -1;
}
return 0;
}
/**
* nih_option_get_long:
* @ctx: parsing context,
* @option: option to find,
* @len: length of option.
*
* Find the option structure with the given long @option, of which only
* the first @len characters will be read. If an option named "--" exists
* then it is used if no other option could be found.
*
* Returns; pointer to option, or NULL if not found.
**/
static NihOption *
nih_option_get_long (NihOptionCtx *ctx,
const char *option,
size_t len)
{
NihOption *opt, *catch = NULL;
for (opt = ctx->options; (opt->option || opt->long_option); opt++) {
if (! opt->long_option)
continue;
if (! strcmp (opt->long_option, "--"))
catch = opt;
if (strlen (opt->long_option) > len)
continue;
if (! strncmp (option, opt->long_option, len))
return opt;
}
return catch;
}
/**
* nih_option_long:
* @ctx: parsing context.
*
* Process the current argument as a long option to be handled.
*
* If the option expects a value then it may be separated from the option
* name by an '=' sign.
*
* Returns: zero on success, negative value on invalid option.
**/
static int
nih_option_long (NihOptionCtx *ctx)
{
NihOption *opt;
char *arg, *ptr;
size_t len;
nih_assert (ctx != NULL);
nih_assert (program_name != NULL);
/* Check for an equals sign that separates the option name from
* an argument.
*/
arg = ctx->argv[ctx->arg] + 2;
ptr = strchr (arg, '=');
len = (ptr ? (size_t)(ptr - arg) : strlen (arg));
/* Find the option */
opt = nih_option_get_long (ctx, arg, len);
if (! opt) {
fprintf (stderr, _("%s: invalid option: --%s\n"),
program_name, arg);
nih_main_suggest_help ();
return -1;
}
/* Handle the case where there's an argument; either we need
* to process it, or error
*/
if (ptr != NULL) {
if (opt->arg_name) {
if (nih_option_handle_arg (ctx, opt, ptr + 1) < 0)
return -1;
return 0;
} else {
fprintf (stderr, _("%s: unexpected argument: --%s\n"),
program_name, arg);
nih_main_suggest_help ();
return -1;
}
}
/* Otherwise it's an ordinary option */
if (nih_option_handle (ctx, opt) < 0)
return -1;
return 0;
}
/**
* nih_option_handle:
* @ctx: parsing context,
* @opt: option to handle.
*
* Handle an option which either does not take an argument, or should
* take the next non-option argument from the command-line. For options
* with arguments, this calls nih_option_handle_arg(); for those without,
* this calls the setter function or treats the value member as a pointer
* to an int to store TRUE in.
*
* Returns: zero on success, negative value on invalid option.
**/
static int
nih_option_handle (NihOptionCtx *ctx,
NihOption *opt)
{
nih_assert (ctx != NULL);
nih_assert (opt != NULL);
/* Handle the special cased options first */
if (opt->long_option && (! strcmp (opt->long_option, "help"))) {
/* --help */
nih_option_help (ctx->options);
nih_free (ctx->options);
nih_free (ctx->args);
exit (0);
} else if (opt->long_option
&& (! strcmp (opt->long_option, "version"))) {
/* --version */
nih_main_version ();
nih_free (ctx->options);
nih_free (ctx->args);
exit (0);
}
if (opt->arg_name) {
const char *arg;
arg = nih_option_next_nonopt (ctx);
if (! arg) {
fprintf (stderr, _("%s: missing argument: %s\n"),
program_name, ctx->argv[ctx->arg]);
nih_main_suggest_help ();
return -1;
}
return nih_option_handle_arg (ctx, opt, arg);
} else if (opt->setter) {
return opt->setter (opt, NULL);
} else if (opt->value) {
int *value = (int *)opt->value;
*value = TRUE;
}
return 0;
}
/**
* nih_option_handle_arg:
* @ctx: parsing context,
* @opt: option to handle,
* @arg: argument.
*
* Handle an option which has the argument specified, either calling
* the setter function or treating the value member (if present) as the
* address of a char * to store the duplicated argument value in.
*
* Returns: zero on success, negative value on invalid option.
**/
static int
nih_option_handle_arg (NihOptionCtx *ctx,
NihOption *opt,
const char *arg)
{
nih_assert (ctx != NULL);
nih_assert (opt != NULL);
nih_assert (opt->arg_name != NULL);
nih_assert (arg != NULL);
if (opt->setter) {
return opt->setter (opt, arg);
} else if (opt->value) {
char **value = (char **)opt->value;
if (*value)
nih_free (*value);
*value = NIH_MUST (nih_strdup (ctx->parent, arg));
}
return 0;
}
/**
* nih_option_next_nonopt:
* @ctx: parsing context.
*
* Iterates the command-line arguments looking for the next argument
* that is not an option. Updates the nonopt member of @ctx to point
* at the option used.
*
* Returns: next non-option argument or NULL if none remain.
**/
static const char *
nih_option_next_nonopt (NihOptionCtx *ctx)
{
nih_assert (ctx != NULL);
if (ctx->nonopt < ctx->arg)
ctx->nonopt = ctx->arg;
while (++ctx->nonopt < ctx->argc) {
char *arg;
arg = ctx->argv[ctx->nonopt];
if ((arg[0] != '-')
|| (ctx->optend && ctx->nonopt > ctx->optend)) {
return arg;
} else if ((arg[1] == '-') && (arg[2] == '\0')) {
/* End of options */
ctx->optend = ctx->nonopt;
}
}
return NULL;
}
/**
* nih_option_join:
* @parent: parent object for new array,
* @a: first option array,
* @b: second option array.
*
* Joins the two option arrays together to produce a combined array containing
* the options from @a followed by the options from @b.
*
* The new list is allocated with nih_alloc(), but the members are just
* copied in from @a and @b including any pointers therein. Freeing the
* new array with nih_free() is entirely safe.
*
* If @parent is not NULL, it should be a pointer to another object which
* will be used as a parent for the returned array. When all parents
* of the returned array are freed, the returned array will also be
* freed.
*
* Returns: combined option array.
**/
NihOption *
nih_option_join (const void *parent,
const NihOption *a,
const NihOption *b)
{
const NihOption *opt;
NihOption *opts;
size_t alen = 0, blen = 0;
nih_assert (a != NULL);
nih_assert (b != NULL);
/* Count options in first list */
for (opt = a; (opt->option || opt->long_option); opt++)
alen++;
/* Count options in second list */
for (opt = b; (opt->option || opt->long_option); opt++)
blen++;
/* Allocate combined list */
opts = NIH_MUST (nih_alloc (parent,
sizeof (NihOption) * (alen + blen + 1)));
/* Copy options, making sure to copy the last option from b */
memcpy (opts, a, sizeof (NihOption) * alen);
memcpy (opts + alen, b, sizeof (NihOption) * (blen + 1));
return opts;
}
/**
* nih_option_count:
* @option: NihOption invoked,
* @arg: argument to parse.
*
* This option setter may be used to create arguments that count the number
* of times they are placed on the command line.
*
* The value member of @option must be a pointer to an integer variable,
* the arg_name member must be NULL.
*
* Returns: always returns zero.
**/
int
nih_option_count (NihOption *option,
const char *arg)
{
int *value;
nih_assert (option != NULL);
nih_assert (option->value != NULL);
nih_assert (arg == NULL);
value = (int *)option->value;
(*value)++;
return 0;
}
/**
* nih_option_int:
* @option: NihOption invoked,
* @arg: argument to parse.
*
* This option setter may be used to parse an integer from the command line
* and store it in the value member of @option, which must be a pointer to
* an integer variable.
*
* The arg_name member of @option must not be NULL.
*
* Returns: zero on success, non-zero on error.
**/
int
nih_option_int (NihOption *option,
const char *arg)
{
char *endptr;
int *value;
nih_assert (option != NULL);
nih_assert (option->value != NULL);
nih_assert (arg != NULL);
value = (int *)option->value;
*value = strtol (arg, &endptr, 10);
if (*endptr) {
fprintf (stderr, _("%s: illegal argument: %s\n"),
program_name, arg);
nih_main_suggest_help ();
return -1;
}
return 0;
}
/**
* nih_option_quiet:
* @option: NihOption invoked,
* @arg: argument to parse.
*
* This option setter is used by the built-in -q/--quiet option to set the
* default logging level to ERROR.
*
* Returns: always returns zero.
**/
int
nih_option_quiet (NihOption *option,
const char *arg)
{
nih_assert (option != NULL);
nih_assert (arg == NULL);
nih_log_set_priority (NIH_LOG_ERROR);
return 0;
}
/**
* nih_option_verbose:
* @option: NihOption invoked,
* @arg: argument to parse.
*
* This option setter is used by the built-in -v/--verbose option to set the
* default logging level to INFO.
*
* Returns: always returns zero.
**/
int
nih_option_verbose (NihOption *option,
const char *arg)
{
nih_assert (option != NULL);
nih_assert (arg == NULL);
nih_log_set_priority (NIH_LOG_INFO);
return 0;
}
/**
* nih_option_debug:
* @option: NihOption invoked,
* @arg: argument to parse.
*
* This option setter is used by the built-in --debug option to set the
* default logging level to DEBUG.
*
* Returns: always returns zero.
**/
int
nih_option_debug (NihOption *option,
const char *arg)
{
nih_assert (option != NULL);
nih_assert (arg == NULL);
nih_log_set_priority (NIH_LOG_DEBUG);
return 0;
}
/**
* nih_option_set_usage_stem:
* @usage: usage stem.
*
* Set the usage stem prepended to the program usage line in the help output,
* this should be a static translated string.
*
* The string should not be terminated with a newline.
**/
void
nih_option_set_usage_stem (const char *usage)
{
usage_stem = usage;
}
/**
* nih_option_set_usage:
* @usage: usage string.
*
* Set the usage string appended to the program usage line in the help output,
* this should be a static translated string.
*
* The string should not be terminated with a newline.
**/
void
nih_option_set_usage (const char *usage)
{
usage_string = usage;
}
/**
* nih_option_set_synopsis:
* @synopsis: synopsis string.
*
* Set the synopsis string, shown after the program usage line in the help
* output. This should be a static translated string. It will be
* automatically wrapped to the screen width.
*
* The string should not be terminated with a newline.
**/
void
nih_option_set_synopsis (const char *synopsis)
{
synopsis_string = synopsis;
}
/**
* nih_option_set_help:
* @help: help string.
*
* Set the help string, this is displayed after the options in the help
* output. This should be a static translated string. It will be
* automatically wrapped to the screen width.
*
* The string should not be terminated with a newline.
**/
void
nih_option_set_help (const char *help)
{
help_string = help;
}
/**
* nih_option_set_footer:
* @footer: footer string.
*
* Set the footer string, this is displayed after the options and help
* text in the output. This should be a static translated string.
*
* The string should not be terminated with a newline.
**/
void
nih_option_set_footer (const char *footer)
{
footer_string = footer;
}
/**
* nih_option_help:
* @options: program options list.
*
* Output a description of the program's options to standard output
* grouped by the group member of the option.
**/
static void
nih_option_help (NihOption *options)
{
NihOption *opt;
nih_local NihOptionGroup **groups = NULL;
size_t group, ngroups;
int other = FALSE;
nih_assert (program_name != NULL);
/* Count the number of option groups */
ngroups = 0;
for (opt = options; (opt->option || opt->long_option); opt++) {
if (! opt->group) {
other = TRUE;
continue;
}
for (group = 0; group < ngroups; group++) {
if (groups[group] == opt->group)
break;
}
if (group < ngroups)
continue;
groups = NIH_MUST (nih_realloc (groups, NULL,
(sizeof (NihOptionGroup *)
* (ngroups + 1))));
groups[ngroups++] = opt->group;
}
printf ("%s: %s", _("Usage"), program_name);
if (usage_stem) {
printf (" %s", usage_stem);
} else {
printf (" %s", _("[OPTION]..."));
}
if (usage_string)
printf (" %s", usage_string);
printf ("\n");
/* Wrap the synopsis to the screen width */
if (synopsis_string) {
nih_local char *str;
str = NIH_MUST (nih_str_screen_wrap (NULL, synopsis_string,
0, 0));
printf ("%s\n", str);
}
printf ("\n");
/* Iterate the option groups we found in order, and display
* only their options
*/
for (group = 0; group < ngroups; group++)
nih_option_group_help (groups[group], options, groups);
/* Display the other group */
if (other)
nih_option_group_help (NULL, options, groups);
/* Wrap the help to the screen width */
if (help_string) {
nih_local char *str;
str = NIH_MUST (nih_str_screen_wrap (NULL, help_string, 0, 0));
printf ("%s\n", str);
if (package_bugreport || footer_string)
printf ("\n");
}
/* Append the footer */
if (footer_string) {
printf ("%s\n", footer_string);
if (package_bugreport)
printf ("\n");
}
/* Append the bug report address */
if (package_bugreport) {
if (strchr (package_bugreport, '@')) {
printf (_("Report bugs to <%s>\n"), package_bugreport);
} else {
printf (_("Report bugs at <%s>\n"), package_bugreport);
}
}
}
/**
* nih_option_group_help:
* @group: group to display,
* @options: program options list,
* @groups: all groups.
*
* Output a list of the program's options in the given option group to
* standard output.
**/
static void
nih_option_group_help (NihOptionGroup *group,
NihOption *options,
NihOptionGroup **groups)
{
NihOption *opt;
size_t width;
nih_assert (options != NULL);
if (group) {
printf (_("%s options:\n"), _(group->title));
} else if (groups) {
printf (_("Other options:\n"));
} else {
printf (_("Options:\n"));
}
width = nih_max (nih_str_screen_width (), 50U) - 31;
for (opt = options; (opt->option || opt->long_option); opt++) {
nih_local char *str = NULL;
char *ptr;
size_t len = 0;
if (opt->group != group)
continue;
if (! opt->help)
continue;
/* Indent by two spaces */
printf (" ");
len += 2;
/* Display the short option */
if (opt->option) {
printf ("-%c", opt->option);
len += 2;
/* Seperate short and long option, or
* give the argument name
*/
if (opt->long_option) {
printf (", ");
len += 2;
} else if (opt->arg_name) {
printf (" %s", opt->arg_name);
len += strlen (opt->arg_name) + 1;
}
} else {
/* Make all long options the same indent
* whether or not there's a short one
*/
printf (" ");
len += 4;
}
/* Display the long option */
if (opt->long_option) {
printf ("--%s", opt->long_option);
len += strlen (opt->long_option) + 2;
/* With the argument name */
if (opt->arg_name) {
printf ("=%s", opt->arg_name);
len += strlen (opt->arg_name) + 1;
}
}
/* Format the help string to fit in the latter
* half of the screen
*/
str = NIH_MUST (nih_str_wrap (NULL, opt->help, width, 0, 2));
/* Write the description to the screen */
ptr = str;
while (ptr && *ptr) {
size_t linelen;
/* Not enough room on this line */
if (len > 28) {
printf ("\n");
len = 0;
}
/* Pad line up to the right column */
while (len < 30) {
printf (" ");
len++;
}
/* Output the line up until the next line */
linelen = strcspn (ptr, "\n");
printf ("%.*s\n", (int)linelen, ptr);
len = 0;
/* Skip to the next line */
ptr += linelen;
if (*ptr == '\n')
ptr++;
}
}
printf ("\n");
}