mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-08 12:44:23 +01:00
d8e6d1e9ed
* common/argparse.c (gnupg_argparse): Set attribute flags -- GnuPG-bug-id: 5799 This is a backport from the fix in libgpg-error in case gnupg 2.2 is build against an older version of libgpg-error.
2810 lines
85 KiB
C
2810 lines
85 KiB
C
/* [argparse.c wk 17.06.97] Argument Parser for option handling
|
|
* Copyright (C) 1998-2001, 2006-2008, 2012 Free Software Foundation, Inc.
|
|
* Copyright (C) 1997-2001, 2006-2008, 2013-2017 Werner Koch
|
|
*
|
|
* This file is part of GnuPG.
|
|
*
|
|
* GnuPG is free software; you can redistribute and/or modify this
|
|
* part of GnuPG under the terms of either
|
|
*
|
|
* - the GNU Lesser General Public License as published by the Free
|
|
* Software Foundation; either version 3 of the License, or (at
|
|
* your option) any later version.
|
|
*
|
|
* or
|
|
*
|
|
* - the GNU General Public License as published by the Free
|
|
* Software Foundation; either version 2 of the License, or (at
|
|
* your option) any later version.
|
|
*
|
|
* or both in parallel, as here.
|
|
*
|
|
* GnuPG 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 copies of the GNU General Public License
|
|
* and the GNU Lesser General Public License along with this program;
|
|
* if not, see <https://gnu.org/licenses/>.
|
|
*/
|
|
|
|
/* This is a modified version of gpgrt/libgpg-error src/argparse.c.
|
|
* We use this to require a dependency on a newer gpgrt version.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
#include <string.h>
|
|
#include <stdarg.h>
|
|
#include <limits.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <time.h>
|
|
|
|
#include "util.h"
|
|
#include "common-defs.h"
|
|
#include "i18n.h"
|
|
#include "mischelp.h"
|
|
#include "stringhelp.h"
|
|
#include "logging.h"
|
|
#include "utf8conv.h"
|
|
#include "sysutils.h"
|
|
#include "argparse.h"
|
|
|
|
|
|
/* Optional handler to write strings. See gnupg_set_usage_outfnc. */
|
|
static int (*custom_outfnc) (int, const char *);
|
|
|
|
|
|
#if USE_INTERNAL_ARGPARSE
|
|
|
|
/* The almost always needed user handler for strusage. */
|
|
static const char *(*strusage_handler)( int ) = NULL;
|
|
/* Optional handler to map strings. See gnupg_set_fixed_string_mapper. */
|
|
static const char *(*fixed_string_mapper)(const char*);
|
|
|
|
|
|
/* Hidden argparse flag used to mark the object as initialized. */
|
|
#define ARGPARSE_FLAG__INITIALIZED (1u << ((8*4)-1))
|
|
|
|
/* Special short options which are auto-inserterd. Must fit into an
|
|
* unsigned short. */
|
|
#define ARGPARSE_SHORTOPT_HELP 32768
|
|
#define ARGPARSE_SHORTOPT_VERSION 32769
|
|
#define ARGPARSE_SHORTOPT_WARRANTY 32770
|
|
#define ARGPARSE_SHORTOPT_DUMP_OPTIONS 32771
|
|
#define ARGPARSE_SHORTOPT_DUMP_OPTTBL 32772
|
|
|
|
|
|
/* The malloced configuration directories or NULL. */
|
|
static struct
|
|
{
|
|
char *user;
|
|
char *sys;
|
|
} confdir;
|
|
|
|
|
|
/* The states for the gnupg_argparser machinery. */
|
|
enum argparser_states
|
|
{
|
|
STATE_init = 0,
|
|
STATE_open_sys,
|
|
STATE_open_user,
|
|
STATE_open_cmdline,
|
|
STATE_read_sys,
|
|
STATE_read_user,
|
|
STATE_read_cmdline,
|
|
STATE_finished
|
|
};
|
|
|
|
|
|
/* An internal object used to store the user provided option table and
|
|
* some meta information. */
|
|
typedef struct
|
|
{
|
|
unsigned short short_opt;
|
|
unsigned short ordinal; /* (for --help) */
|
|
unsigned int flags;
|
|
const char *long_opt; /* Points into the user provided table. */
|
|
const char *description; /* Points into the user provided table. */
|
|
unsigned int forced:1; /* Forced to use the sysconf value. */
|
|
unsigned int ignore:1; /* Ignore this option everywhere but in
|
|
* the sysconf file. */
|
|
unsigned int explicit_ignore:1; /* Ignore was explicitly set. */
|
|
} opttable_t;
|
|
|
|
|
|
/* Internal object of the public gnupg_argparse_t object. */
|
|
struct _argparse_internal_s
|
|
{
|
|
int idx; /* Note that this is saved and restored in gnupg_argparser. */
|
|
int inarg; /* (index into args) */
|
|
unsigned int verbose:1; /* Print diagnostics. */
|
|
unsigned int stopped:1; /* Option processing has stopped. */
|
|
unsigned int in_sysconf:1; /* Processing global config file. */
|
|
unsigned int mark_forced:1; /* Mark options as forced. */
|
|
unsigned int mark_ignore:1; /* Mark options as to be ignored. */
|
|
unsigned int explicit_ignore:1; /* Option has explicitly been set
|
|
* to ignore or unignore. */
|
|
unsigned int ignore_all_seen:1; /* [ignore-all] has been seen. */
|
|
unsigned int user_seen:1; /* A [user] has been seen. */
|
|
unsigned int user_wildcard:1; /* A [user *] has been seen. */
|
|
unsigned int user_any_active:1; /* Any user section was active. */
|
|
unsigned int user_active:1; /* User section active. */
|
|
unsigned int explicit_confopt:1; /* A conffile option has been given. */
|
|
char *explicit_conffile; /* Malloced name of an explicit
|
|
* conffile. */
|
|
char *username; /* Malloced current user name. */
|
|
unsigned int opt_flags; /* Current option flags. */
|
|
enum argparser_states state; /* State of the gnupg_argparser. */
|
|
const char *last;
|
|
void *aliases;
|
|
const void *cur_alias;
|
|
void *iio_list;
|
|
estream_t conffp;
|
|
char *confname;
|
|
opttable_t *opts; /* Malloced option table. */
|
|
unsigned int nopts; /* Number of items in OPTS. */
|
|
};
|
|
|
|
|
|
typedef struct alias_def_s *ALIAS_DEF;
|
|
struct alias_def_s {
|
|
ALIAS_DEF next;
|
|
char *name; /* malloced buffer with name, \0, value */
|
|
const char *value; /* ptr into name */
|
|
};
|
|
|
|
|
|
/* Object to store the names for the --ignore-invalid-option option.
|
|
This is a simple linked list. */
|
|
typedef struct iio_item_def_s *IIO_ITEM_DEF;
|
|
struct iio_item_def_s
|
|
{
|
|
IIO_ITEM_DEF next;
|
|
char name[1]; /* String with the long option name. */
|
|
};
|
|
|
|
|
|
static int set_opt_arg (gnupg_argparse_t *arg, unsigned int flags, char *s);
|
|
static void show_help (opttable_t *opts, unsigned int nopts,unsigned int flags);
|
|
static void show_version (void);
|
|
static void dump_option_table (gnupg_argparse_t *arg);
|
|
static int writestrings (int is_error, const char *string,
|
|
...) GPGRT_ATTR_SENTINEL(0);
|
|
|
|
static int arg_parse (gnupg_argparse_t *arg, gnupg_opt_t *opts, int no_init);
|
|
|
|
|
|
|
|
/* Set a function to write strings which is then used instead of
|
|
* estream. The first arg of that function is MODE and the second the
|
|
* STRING to write. A mode of 1 is used for writing to stdout and a
|
|
* mode of 2 to write to stderr. Other modes are reserved and should
|
|
* not output anything. A NULL for STRING requests a flush. */
|
|
void
|
|
gnupg_set_usage_outfnc (int (*f)(int, const char *))
|
|
{
|
|
custom_outfnc = f;
|
|
}
|
|
|
|
|
|
/* Register function F as a string mapper which takes a string as
|
|
* argument, replaces known "@FOO@" style macros and returns a new
|
|
* fixed string. Warning: The input STRING must have been allocated
|
|
* statically. */
|
|
void
|
|
gnupg_set_fixed_string_mapper (const char *(*f)(const char*))
|
|
{
|
|
fixed_string_mapper = f;
|
|
}
|
|
|
|
|
|
/* Register a configuration directory for use by the argparse
|
|
* functions. The defined values for WHAT are:
|
|
*
|
|
* GNUPG_CONFDIR_SYS The systems's configuration dir.
|
|
* The default is /etc
|
|
*
|
|
* GNUPG_CONFDIR_USER The user's configuration directory.
|
|
* The default is $HOME.
|
|
*
|
|
* A trailing slash is ignored; to have the function lookup
|
|
* configuration files in the current directory, use ".". There is no
|
|
* error return; more configuraion values may be added in future
|
|
* revisions of this library.
|
|
*/
|
|
void
|
|
gnupg_set_confdir (int what, const char *name)
|
|
{
|
|
char *buf, *p;
|
|
|
|
if (what == GNUPG_CONFDIR_SYS)
|
|
{
|
|
xfree (confdir.sys);
|
|
buf = confdir.sys = xtrystrdup (name);
|
|
}
|
|
else if (what == GNUPG_CONFDIR_USER)
|
|
{
|
|
xfree (confdir.user);
|
|
buf = confdir.user = xtrystrdup (name);
|
|
}
|
|
else
|
|
return;
|
|
|
|
if (!buf)
|
|
log_fatal ("out of core in %s\n", __func__);
|
|
#ifdef HAVE_W32_SYSTEM
|
|
for (p=buf; *p; p++)
|
|
if (*p == '\\')
|
|
*p = '/';
|
|
#endif
|
|
/* Strip trailing slashes unless buf is "/" or any other single char
|
|
* string. */
|
|
if (*buf)
|
|
{
|
|
for (p=buf + strlen (buf)-1; p > buf; p--)
|
|
if (*p == '/')
|
|
*p = 0;
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
static const char *
|
|
map_fixed_string (const char *string)
|
|
{
|
|
return fixed_string_mapper? fixed_string_mapper (string) : string;
|
|
}
|
|
|
|
#endif /* USE_INTERNAL_ARGPARSE */
|
|
|
|
|
|
/* Write STRING and all following const char * arguments either to
|
|
stdout or, if IS_ERROR is set, to stderr. The list of strings must
|
|
be terminated by a NULL. */
|
|
static int
|
|
writestrings (int is_error, const char *string, ...)
|
|
{
|
|
va_list arg_ptr;
|
|
const char *s;
|
|
int count = 0;
|
|
|
|
if (string)
|
|
{
|
|
s = string;
|
|
va_start (arg_ptr, string);
|
|
do
|
|
{ /* Fixme: Swicth to estream? */
|
|
if (custom_outfnc)
|
|
custom_outfnc (is_error? 2:1, s);
|
|
else
|
|
fputs (s, is_error? stderr : stdout);
|
|
count += strlen (s);
|
|
}
|
|
while ((s = va_arg (arg_ptr, const char *)));
|
|
va_end (arg_ptr);
|
|
}
|
|
return count;
|
|
}
|
|
|
|
|
|
static void
|
|
flushstrings (int is_error)
|
|
{
|
|
if (custom_outfnc)
|
|
custom_outfnc (is_error? 2:1, NULL);
|
|
else
|
|
fflush (is_error? stderr : stdout);
|
|
}
|
|
|
|
|
|
#if USE_INTERNAL_ARGPARSE
|
|
|
|
static void
|
|
deinitialize (gnupg_argparse_t *arg)
|
|
{
|
|
if (arg->internal)
|
|
{
|
|
xfree (arg->internal->username);
|
|
xfree (arg->internal->explicit_conffile);
|
|
xfree (arg->internal->opts);
|
|
xfree (arg->internal);
|
|
arg->internal = NULL;
|
|
}
|
|
|
|
arg->flags &= ARGPARSE_FLAG__INITIALIZED;
|
|
arg->lineno = 0;
|
|
arg->err = 0;
|
|
}
|
|
|
|
/* Our own exit handler to clean up used memory. */
|
|
static void
|
|
my_exit (gnupg_argparse_t *arg, int code)
|
|
{
|
|
deinitialize (arg);
|
|
exit (code);
|
|
}
|
|
|
|
|
|
static gpg_err_code_t
|
|
initialize (gnupg_argparse_t *arg, gnupg_opt_t *opts, estream_t fp)
|
|
{
|
|
/* We use a dedicated flag to detect whether *ARG has been
|
|
* initialized. This is because the old version of that struct, as
|
|
* used in GnuPG, had no requirement to zero out all fields of the
|
|
* object and existing code still sets only argc,argv and flags. */
|
|
if (!(arg->flags & ARGPARSE_FLAG__INITIALIZED)
|
|
|| (arg->flags & ARGPARSE_FLAG_RESET)
|
|
|| !arg->internal)
|
|
{
|
|
/* Allocate internal data. */
|
|
if (!(arg->flags & ARGPARSE_FLAG__INITIALIZED) || !arg->internal)
|
|
{
|
|
arg->internal = xtrymalloc (sizeof *arg->internal);
|
|
if (!arg->internal)
|
|
return gpg_err_code_from_syserror ();
|
|
arg->flags |= ARGPARSE_FLAG__INITIALIZED; /* Mark as initialized. */
|
|
}
|
|
else if (arg->internal->opts)
|
|
xfree (arg->internal->opts);
|
|
arg->internal->opts = NULL;
|
|
arg->internal->nopts = 0;
|
|
|
|
/* Initialize this instance. */
|
|
arg->internal->idx = 0;
|
|
arg->internal->last = NULL;
|
|
arg->internal->inarg = 0;
|
|
arg->internal->stopped = 0;
|
|
arg->internal->in_sysconf = 0;
|
|
arg->internal->user_seen = 0;
|
|
arg->internal->user_wildcard = 0;
|
|
arg->internal->user_any_active = 0;
|
|
arg->internal->user_active = 0;
|
|
arg->internal->username = NULL;
|
|
arg->internal->mark_forced = 0;
|
|
arg->internal->mark_ignore = 0;
|
|
arg->internal->explicit_ignore = 0;
|
|
arg->internal->ignore_all_seen = 0;
|
|
arg->internal->explicit_confopt = 0;
|
|
arg->internal->explicit_conffile = NULL;
|
|
arg->internal->opt_flags = 0;
|
|
arg->internal->state = STATE_init;
|
|
arg->internal->aliases = NULL;
|
|
arg->internal->cur_alias = NULL;
|
|
arg->internal->iio_list = NULL;
|
|
arg->internal->conffp = NULL;
|
|
arg->internal->confname = NULL;
|
|
|
|
/* Clear the copy of the option list. */
|
|
/* Clear the error indicator. */
|
|
arg->err = 0;
|
|
|
|
/* Usually an option file will be parsed from the start.
|
|
* However, we do not open the stream and thus we have no way to
|
|
* know the current lineno. Using this flag we can allow the
|
|
* user to provide a lineno which we don't reset. */
|
|
if (fp || arg->internal->conffp || !(arg->flags & ARGPARSE_FLAG_NOLINENO))
|
|
arg->lineno = 0;
|
|
|
|
/* Need to clear the reset request. */
|
|
arg->flags &= ~ARGPARSE_FLAG_RESET;
|
|
|
|
/* Check initial args. */
|
|
if ( *arg->argc < 0 )
|
|
log_bug ("invalid argument passed to gnupg_argparse\n");
|
|
|
|
}
|
|
|
|
/* Create an array with pointers to the provided list of options.
|
|
* Keeping a copy is useful to sort that array and thus do a binary
|
|
* search and to allow for extra space at the end to insert the
|
|
* hidden options. An ARGPARSE_FLAG_RESET can be used to reinit
|
|
* this array. */
|
|
if (!arg->internal->opts)
|
|
{
|
|
int seen_help = 0;
|
|
int seen_version = 0;
|
|
int seen_warranty = 0;
|
|
int seen_dump_options = 0;
|
|
int seen_dump_option_table = 0;
|
|
int i;
|
|
|
|
for (i=0; opts[i].short_opt; i++)
|
|
{
|
|
if (opts[i].long_opt)
|
|
{
|
|
if (!strcmp(opts[i].long_opt, "help"))
|
|
seen_help = 1;
|
|
else if (!strcmp(opts[i].long_opt, "version"))
|
|
seen_version = 1;
|
|
else if (!strcmp(opts[i].long_opt, "warranty"))
|
|
seen_warranty = 1;
|
|
else if (!strcmp(opts[i].long_opt, "dump-options"))
|
|
seen_dump_options = 1;
|
|
else if (!strcmp(opts[i].long_opt, "dump-option-table"))
|
|
seen_dump_option_table = 1;
|
|
}
|
|
}
|
|
i += 5; /* The number of the above internal options. */
|
|
i++; /* End of list marker. */
|
|
arg->internal->opts = xtrycalloc (i, sizeof *arg->internal->opts);
|
|
if (!arg->internal->opts)
|
|
return gpg_err_code_from_syserror ();
|
|
for(i=0; opts[i].short_opt; i++)
|
|
{
|
|
arg->internal->opts[i].short_opt = opts[i].short_opt;
|
|
arg->internal->opts[i].flags = opts[i].flags;
|
|
arg->internal->opts[i].long_opt = opts[i].long_opt;
|
|
arg->internal->opts[i].description = opts[i].description;
|
|
arg->internal->opts[i].ordinal = i;
|
|
}
|
|
|
|
if (!seen_help)
|
|
{
|
|
arg->internal->opts[i].short_opt = ARGPARSE_SHORTOPT_HELP;
|
|
arg->internal->opts[i].flags = ARGPARSE_TYPE_NONE;
|
|
arg->internal->opts[i].long_opt = "help";
|
|
arg->internal->opts[i].description = "@";
|
|
arg->internal->opts[i].ordinal = i;
|
|
i++;
|
|
}
|
|
if (!seen_version)
|
|
{
|
|
arg->internal->opts[i].short_opt = ARGPARSE_SHORTOPT_VERSION;
|
|
arg->internal->opts[i].flags = ARGPARSE_TYPE_NONE;
|
|
arg->internal->opts[i].long_opt = "version";
|
|
arg->internal->opts[i].description = "@";
|
|
arg->internal->opts[i].ordinal = i;
|
|
i++;
|
|
}
|
|
|
|
if (!seen_warranty)
|
|
{
|
|
arg->internal->opts[i].short_opt = ARGPARSE_SHORTOPT_WARRANTY;
|
|
arg->internal->opts[i].flags = ARGPARSE_TYPE_NONE;
|
|
arg->internal->opts[i].long_opt = "warranty";
|
|
arg->internal->opts[i].description = "@";
|
|
arg->internal->opts[i].ordinal = i;
|
|
i++;
|
|
}
|
|
|
|
if (!seen_dump_option_table)
|
|
{
|
|
arg->internal->opts[i].short_opt = ARGPARSE_SHORTOPT_DUMP_OPTTBL;
|
|
arg->internal->opts[i].flags = ARGPARSE_TYPE_NONE;
|
|
arg->internal->opts[i].long_opt = "dump-option-table";
|
|
arg->internal->opts[i].description = "@";
|
|
arg->internal->opts[i].ordinal = i;
|
|
i++;
|
|
}
|
|
|
|
if (!seen_dump_options)
|
|
{
|
|
arg->internal->opts[i].short_opt = ARGPARSE_SHORTOPT_DUMP_OPTIONS;
|
|
arg->internal->opts[i].flags = ARGPARSE_TYPE_NONE;
|
|
arg->internal->opts[i].long_opt = "dump-options";
|
|
arg->internal->opts[i].description = "@";
|
|
arg->internal->opts[i].ordinal = i;
|
|
i++;
|
|
}
|
|
/* Take care: When adding new options remember to increase the
|
|
* size of the array. */
|
|
|
|
arg->internal->opts[i].short_opt = 0;
|
|
|
|
/* Note that we do not count the end marker but keep it in the
|
|
* table anyway as an extra item. */
|
|
arg->internal->nopts = i;
|
|
}
|
|
|
|
if (arg->err)
|
|
{
|
|
/* Last option was erroneous. */
|
|
const char *s;
|
|
|
|
if (!fp && arg->internal->conffp)
|
|
fp = arg->internal->conffp;
|
|
|
|
if (fp)
|
|
{
|
|
if ( arg->r_opt == ARGPARSE_UNEXPECTED_ARG )
|
|
s = _("argument not expected");
|
|
else if ( arg->r_opt == ARGPARSE_READ_ERROR )
|
|
s = _("read error");
|
|
else if ( arg->r_opt == ARGPARSE_KEYWORD_TOO_LONG )
|
|
s = _("keyword too long");
|
|
else if ( arg->r_opt == ARGPARSE_MISSING_ARG )
|
|
s = _("missing argument");
|
|
else if ( arg->r_opt == ARGPARSE_INVALID_ARG )
|
|
s = _("invalid argument");
|
|
else if ( arg->r_opt == ARGPARSE_INVALID_COMMAND )
|
|
s = _("invalid command");
|
|
else if ( arg->r_opt == ARGPARSE_INVALID_ALIAS )
|
|
s = _("invalid alias definition");
|
|
else if ( arg->r_opt == ARGPARSE_PERMISSION_ERROR )
|
|
s = _("permission error");
|
|
else if ( arg->r_opt == ARGPARSE_OUT_OF_CORE )
|
|
s = _("out of core");
|
|
else if ( arg->r_opt == ARGPARSE_NO_CONFFILE )
|
|
s = NULL; /* Error has already been printed. */
|
|
else if ( arg->r_opt == ARGPARSE_INVALID_META )
|
|
s = _("invalid meta command");
|
|
else if ( arg->r_opt == ARGPARSE_UNKNOWN_META )
|
|
s = _("unknown meta command");
|
|
else if ( arg->r_opt == ARGPARSE_UNEXPECTED_META )
|
|
s = _("unexpected meta command");
|
|
else
|
|
s = _("invalid option");
|
|
if (s)
|
|
log_error ("%s:%u: %s\n",
|
|
gpgrt_fname_get (fp), arg->lineno, s);
|
|
}
|
|
else
|
|
{
|
|
s = arg->internal->last? arg->internal->last:"[??]";
|
|
|
|
if ( arg->r_opt == ARGPARSE_MISSING_ARG )
|
|
log_error (_("missing argument for option \"%.50s\"\n"), s);
|
|
else if ( arg->r_opt == ARGPARSE_INVALID_ARG )
|
|
log_error (_("invalid argument for option \"%.50s\"\n"), s);
|
|
else if ( arg->r_opt == ARGPARSE_UNEXPECTED_ARG )
|
|
log_error (_("option \"%.50s\" does not expect "
|
|
"an argument\n"), s);
|
|
else if ( arg->r_opt == ARGPARSE_INVALID_COMMAND )
|
|
log_error (_("invalid command \"%.50s\"\n"), s);
|
|
else if ( arg->r_opt == ARGPARSE_AMBIGUOUS_OPTION )
|
|
log_error (_("option \"%.50s\" is ambiguous\n"), s);
|
|
else if ( arg->r_opt == ARGPARSE_AMBIGUOUS_COMMAND )
|
|
log_error (_("command \"%.50s\" is ambiguous\n"),s );
|
|
else if ( arg->r_opt == ARGPARSE_OUT_OF_CORE )
|
|
log_error ("%s\n", _("out of core"));
|
|
else if ( arg->r_opt == ARGPARSE_PERMISSION_ERROR )
|
|
log_error ("%s\n", _("permission error"));
|
|
else if ( arg->r_opt == ARGPARSE_NO_CONFFILE)
|
|
; /* Error has already been printed. */
|
|
else if ( arg->r_opt == ARGPARSE_INVALID_META )
|
|
log_error ("%s\n", _("invalid meta command"));
|
|
else if ( arg->r_opt == ARGPARSE_UNKNOWN_META )
|
|
log_error ("%s\n", _("unknown meta command"));
|
|
else if ( arg->r_opt == ARGPARSE_UNEXPECTED_META )
|
|
log_error ("%s\n",_("unexpected meta command"));
|
|
else
|
|
log_error (_("invalid option \"%.50s\"\n"), s);
|
|
}
|
|
if (arg->err != ARGPARSE_PRINT_WARNING)
|
|
my_exit (arg, 2);
|
|
arg->err = 0;
|
|
}
|
|
|
|
/* Zero out the return value union. */
|
|
arg->r.ret_str = NULL;
|
|
arg->r.ret_long = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void
|
|
store_alias( ARGPARSE_ARGS *arg, char *name, char *value )
|
|
{
|
|
/* TODO: replace this dummy function with a rea one
|
|
* and fix the probelms IRIX has with (ALIAS_DEV)arg..
|
|
* used as lvalue
|
|
*/
|
|
(void)arg;
|
|
(void)name;
|
|
(void)value;
|
|
#if 0
|
|
ALIAS_DEF a = xmalloc( sizeof *a );
|
|
a->name = name;
|
|
a->value = value;
|
|
a->next = (ALIAS_DEF)arg->internal.aliases;
|
|
(ALIAS_DEF)arg->internal.aliases = a;
|
|
#endif
|
|
}
|
|
|
|
|
|
/* Return true if KEYWORD is in the ignore-invalid-option list. */
|
|
static int
|
|
ignore_invalid_option_p (ARGPARSE_ARGS *arg, const char *keyword)
|
|
{
|
|
IIO_ITEM_DEF item = arg->internal->iio_list;
|
|
|
|
for (; item; item = item->next)
|
|
if (!strcmp (item->name, keyword))
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Add the keywords up to the next LF to the list of to be ignored
|
|
options. After returning FP will either be at EOF or the next
|
|
character read wll be the first of a new line. The function
|
|
returns 0 on success or true on malloc failure. */
|
|
static int
|
|
ignore_invalid_option_add (ARGPARSE_ARGS *arg, estream_t fp)
|
|
{
|
|
IIO_ITEM_DEF item;
|
|
int c;
|
|
char name[100];
|
|
int namelen = 0;
|
|
int ready = 0;
|
|
enum { skipWS, collectNAME, skipNAME, addNAME} state = skipWS;
|
|
|
|
while (!ready)
|
|
{
|
|
c = gpgrt_getc (fp);
|
|
if (c == '\n')
|
|
ready = 1;
|
|
else if (c == EOF)
|
|
{
|
|
c = '\n';
|
|
ready = 1;
|
|
}
|
|
again:
|
|
switch (state)
|
|
{
|
|
case skipWS:
|
|
if (!isascii (c) || !isspace(c))
|
|
{
|
|
namelen = 0;
|
|
state = collectNAME;
|
|
goto again;
|
|
}
|
|
break;
|
|
|
|
case collectNAME:
|
|
if (isspace (c))
|
|
{
|
|
state = addNAME;
|
|
goto again;
|
|
}
|
|
else if (namelen < DIM(name)-1)
|
|
name[namelen++] = c;
|
|
else /* Too long. */
|
|
state = skipNAME;
|
|
break;
|
|
|
|
case skipNAME:
|
|
if (isspace (c))
|
|
{
|
|
state = skipWS;
|
|
goto again;
|
|
}
|
|
break;
|
|
|
|
case addNAME:
|
|
name[namelen] = 0;
|
|
if (!ignore_invalid_option_p (arg, name))
|
|
{
|
|
item = xtrymalloc (sizeof *item + namelen);
|
|
if (!item)
|
|
return 1;
|
|
strcpy (item->name, name);
|
|
item->next = (IIO_ITEM_DEF)arg->internal->iio_list;
|
|
arg->internal->iio_list = item;
|
|
}
|
|
state = skipWS;
|
|
goto again;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Clear the entire ignore-invalid-option list. */
|
|
static void
|
|
ignore_invalid_option_clear (ARGPARSE_ARGS *arg)
|
|
{
|
|
IIO_ITEM_DEF item, tmpitem;
|
|
|
|
for (item = arg->internal->iio_list; item; item = tmpitem)
|
|
{
|
|
tmpitem = item->next;
|
|
xfree (item);
|
|
}
|
|
arg->internal->iio_list = NULL;
|
|
}
|
|
|
|
|
|
/* Make sure the username field is filled. Return 0 on success. */
|
|
static int
|
|
assure_username (gnupg_argparse_t *arg)
|
|
{
|
|
if (!arg->internal->username)
|
|
{
|
|
arg->internal->username = gnupg_getusername ();
|
|
if (!arg->internal->username)
|
|
{
|
|
log_error ("%s:%u: error getting current user's name: %s\n",
|
|
arg->internal->confname, arg->lineno,
|
|
gpg_strerror (gpg_error_from_syserror ()));
|
|
/* Not necessary the correct error code but given that we
|
|
* either have a malloc error or some internal system error,
|
|
* it is the best we can do. */
|
|
return ARGPARSE_PERMISSION_ERROR;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Implementation of the "user" command. ARG is the context. ARGS is
|
|
* a non-empty string which this function is allowed to modify. */
|
|
static int
|
|
handle_meta_user (gnupg_argparse_t *arg, unsigned int alternate, char *args)
|
|
{
|
|
int rc;
|
|
|
|
(void)alternate;
|
|
|
|
rc = assure_username (arg);
|
|
if (rc)
|
|
return rc;
|
|
|
|
arg->internal->user_seen = 1;
|
|
if (*args == '*' && !args[1])
|
|
{
|
|
arg->internal->user_wildcard = 1;
|
|
arg->internal->user_active = !arg->internal->user_any_active;
|
|
}
|
|
else if (arg->internal->user_wildcard)
|
|
{
|
|
/* All other user statements are ignored after a wildcard. */
|
|
arg->internal->user_active = 0;
|
|
}
|
|
else if (!strcasecmp (args, arg->internal->username))
|
|
{
|
|
arg->internal->user_any_active = 1;
|
|
arg->internal->user_active = 1;
|
|
}
|
|
else
|
|
{
|
|
arg->internal->user_active = 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Implementation of the "force" command. ARG is the context. A
|
|
* value of 0 for ALTERNATE is "force", a value of 1 requests an
|
|
* unforce". ARGS is the empty string and not used. */
|
|
static int
|
|
handle_meta_force (gnupg_argparse_t *arg, unsigned int alternate, char *args)
|
|
{
|
|
(void)args;
|
|
|
|
arg->internal->mark_forced = alternate? 0 : 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Implementation of the "ignore" command. ARG is the context. A
|
|
* value of 0 for ALTERNATE is a plain "ignore", a value of 1 request
|
|
* an "unignore, a value of 2 requests an "ignore-all". ARGS is the
|
|
* empty string and not used. */
|
|
static int
|
|
handle_meta_ignore (gnupg_argparse_t *arg, unsigned int alternate, char *args)
|
|
{
|
|
(void)args;
|
|
|
|
if (!alternate)
|
|
{
|
|
arg->internal->mark_ignore = 1;
|
|
arg->internal->explicit_ignore = 1;
|
|
}
|
|
else if (alternate == 1)
|
|
{
|
|
arg->internal->mark_ignore = 0;
|
|
arg->internal->explicit_ignore = 1;
|
|
}
|
|
else
|
|
arg->internal->ignore_all_seen = 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Implementation of the "echo" command. ARG is the context. If
|
|
* ALTERNATE is true the filename is not printed. ARGS is the string
|
|
* to log. */
|
|
static int
|
|
handle_meta_echo (gnupg_argparse_t *arg, unsigned int alternate, char *args)
|
|
{
|
|
int rc = 0;
|
|
char *p, *pend;
|
|
|
|
if (alternate)
|
|
log_info ("%s", "");
|
|
else
|
|
log_info ("%s:%u: ", arg->internal->confname, arg->lineno);
|
|
|
|
while (*args)
|
|
{
|
|
p = strchr (args, '$');
|
|
if (!p)
|
|
{
|
|
log_printf ("%s", args);
|
|
break;
|
|
}
|
|
*p = 0;
|
|
log_printf ("%s", args);
|
|
if (p[1] == '$')
|
|
{
|
|
log_printf ("$");
|
|
args = p+2;
|
|
continue;
|
|
}
|
|
if (p[1] != '{')
|
|
{
|
|
log_printf ("$");
|
|
args = p+1;
|
|
continue;
|
|
}
|
|
pend = strchr (p+2, '}');
|
|
if (!pend) /* No closing brace. */
|
|
{
|
|
log_printf ("$");
|
|
args = p+1;
|
|
continue;
|
|
}
|
|
p += 2;
|
|
*pend = 0;
|
|
args = pend+1;
|
|
if (!strcmp (p, "user"))
|
|
{
|
|
rc = assure_username (arg);
|
|
if (rc)
|
|
goto leave;
|
|
log_printf ("%s", arg->internal->username);
|
|
}
|
|
else if (!strcmp (p, "file"))
|
|
log_printf ("%s", arg->internal->confname);
|
|
else if (!strcmp (p, "line"))
|
|
log_printf ("%u", arg->lineno);
|
|
else if (!strcmp (p, "epoch"))
|
|
log_printf ("%lu", (unsigned long)time (NULL));
|
|
}
|
|
|
|
leave:
|
|
log_printf ("\n");
|
|
return rc;
|
|
}
|
|
|
|
|
|
/* Implementation of the "verbose" command. ARG is the context. If
|
|
* ALTERNATE is true the verbosity is disabled. ARGS is not used. */
|
|
static int
|
|
handle_meta_verbose (gnupg_argparse_t *arg, unsigned int alternate, char *args)
|
|
{
|
|
(void)args;
|
|
|
|
if (alternate)
|
|
arg->internal->verbose = 0;
|
|
else
|
|
arg->internal->verbose = 1;
|
|
return 0;
|
|
}
|
|
|
|
/* Handle a meta command. KEYWORD has the content inside the brackets
|
|
* with leading and trailing spaces removed. The function may modify
|
|
* KEYWORD. On success 0 is returned, on error an ARGPARSE_ error
|
|
* code is returned. */
|
|
static int
|
|
handle_metacmd (gnupg_argparse_t *arg, char *keyword)
|
|
{
|
|
static struct {
|
|
const char *name; /* Name of the command. */
|
|
unsigned short alternate; /* Use alternate version of the command. */
|
|
unsigned short needarg:1; /* Command requires an argument. */
|
|
unsigned short always:1; /* Command allowed in all conf files. */
|
|
unsigned short noskip:1; /* Even done in non-active [user] mode. */
|
|
int (*func)(gnupg_argparse_t *arg,
|
|
unsigned int alternate, char *args); /*handler*/
|
|
} cmds[] =
|
|
{{ "user", 0, 1, 0, 1, handle_meta_user },
|
|
{ "force", 0, 0, 0, 0, handle_meta_force },
|
|
{ "+force", 0, 0, 0, 0, handle_meta_force },
|
|
{ "-force", 1, 0, 0, 0, handle_meta_force },
|
|
{ "ignore", 0, 0, 0, 0, handle_meta_ignore },
|
|
{ "+ignore", 0, 0, 0, 0, handle_meta_ignore },
|
|
{ "-ignore", 1, 0, 0, 0, handle_meta_ignore },
|
|
{ "ignore-all", 2, 0, 0, 0, handle_meta_ignore },
|
|
{ "+ignore-all", 2, 0, 0, 0, handle_meta_ignore },
|
|
{ "verbose", 0, 0, 1, 1, handle_meta_verbose },
|
|
{ "+verbose", 0, 0, 1, 1, handle_meta_verbose },
|
|
{ "-verbose", 1, 0, 1, 1, handle_meta_verbose },
|
|
{ "echo", 0, 1, 1, 1, handle_meta_echo },
|
|
{ "-echo", 1, 1, 1, 1, handle_meta_echo },
|
|
{ "info", 0, 1, 1, 0, handle_meta_echo },
|
|
{ "-info", 1, 1, 1, 0, handle_meta_echo }
|
|
};
|
|
char *rest;
|
|
int i;
|
|
|
|
for (rest = keyword; *rest && !(isascii (*rest) && isspace (*rest)); rest++)
|
|
;
|
|
if (*rest)
|
|
{
|
|
*rest++ = 0;
|
|
trim_spaces (rest);
|
|
}
|
|
|
|
for (i=0; i < DIM (cmds); i++)
|
|
if (!strcmp (cmds[i].name, keyword))
|
|
break;
|
|
if (!(i < DIM (cmds)))
|
|
return ARGPARSE_UNKNOWN_META;
|
|
if (cmds[i].needarg && !*rest)
|
|
return ARGPARSE_MISSING_ARG;
|
|
if (!cmds[i].needarg && *rest)
|
|
return ARGPARSE_UNEXPECTED_ARG;
|
|
if (!arg->internal->in_sysconf && !cmds[i].always)
|
|
return ARGPARSE_UNEXPECTED_META;
|
|
|
|
if (!cmds[i].noskip
|
|
&& arg->internal->in_sysconf
|
|
&& arg->internal->user_seen
|
|
&& !arg->internal->user_active)
|
|
return 0; /* Skip this meta command. */
|
|
|
|
return cmds[i].func (arg, cmds[i].alternate, rest);
|
|
}
|
|
|
|
|
|
/* Helper for gnupg_argparse. */
|
|
static void
|
|
prepare_arg_return (gnupg_argparse_t *arg, opttable_t *opts,
|
|
int idx, int in_alias, int set_ignore)
|
|
{
|
|
/* No argument found at the end of the line. */
|
|
if (in_alias)
|
|
arg->r_opt = ARGPARSE_MISSING_ARG;
|
|
else if (!(opts[idx].flags & ARGPARSE_TYPE_MASK))
|
|
arg->r_type = ARGPARSE_TYPE_NONE; /* Does not take an arg. */
|
|
else if ((opts[idx].flags & ARGPARSE_OPT_OPTIONAL))
|
|
arg->r_type = ARGPARSE_TYPE_NONE; /* No optional argument. */
|
|
else if (!(opts[idx].ignore && !opts[idx].forced) && !set_ignore)
|
|
arg->r_opt = ARGPARSE_MISSING_ARG;
|
|
|
|
/* If the caller wants us to return the attributes or
|
|
* ignored options, or these flags in. */
|
|
if ((arg->flags & ARGPARSE_FLAG_WITHATTR))
|
|
{
|
|
if (opts[idx].ignore)
|
|
arg->r_type |= ARGPARSE_ATTR_IGNORE;
|
|
if (opts[idx].forced)
|
|
arg->r_type |= ARGPARSE_ATTR_FORCE;
|
|
if (set_ignore)
|
|
arg->r_type |= ARGPARSE_OPT_IGNORE;
|
|
}
|
|
}
|
|
|
|
/****************
|
|
* Get options from a file.
|
|
* Lines starting with '#' are comment lines.
|
|
* Syntax is simply a keyword and the argument.
|
|
* Valid keywords are all keywords from the long_opt list without
|
|
* the leading dashes. The special keywords "help", "warranty" and "version"
|
|
* are not valid here.
|
|
* The special keyword "alias" may be used to store alias definitions,
|
|
* which are later expanded like long options.
|
|
* The option
|
|
* ignore-invalid-option OPTIONNAMEs
|
|
* is recognized and updates a list of option which should be ignored if they
|
|
* are not defined.
|
|
* Caller must free returned strings.
|
|
* If called with FP set to NULL command line args are parse instead.
|
|
*
|
|
* Q: Should we allow the syntax
|
|
* keyword = value
|
|
* and accept for boolean options a value of 1/0, yes/no or true/false?
|
|
* Note: Abbreviation of options is here not allowed.
|
|
*/
|
|
int
|
|
gnupg_argparse (estream_t fp, gnupg_argparse_t *arg, gnupg_opt_t *opts_orig)
|
|
{
|
|
enum { Ainit,
|
|
Acomment, /* In a comment line. */
|
|
Acopykeyword, /* Collecting a keyword. */
|
|
Awaitarg, /* Wait for an argument. */
|
|
Acopyarg, /* Copy the argument. */
|
|
Akeyword_eol, /* Got keyword at end of line. */
|
|
Akeyword_spc, /* Got keyword at space. */
|
|
Acopymetacmd, /* Copy a meta command. */
|
|
Askipmetacmd, /* Skip spaces after metacmd. */
|
|
Askipmetacmd2,/* Skip comment after metacmd. */
|
|
Ametacmd, /* Process the metacmd. */
|
|
Askipandleave /* Skip the rest of the line and then leave. */
|
|
} state;
|
|
opttable_t *opts;
|
|
unsigned int nopts;
|
|
int i, c;
|
|
int idx = 0;
|
|
char keyword[100];
|
|
char *buffer = NULL;
|
|
size_t buflen = 0;
|
|
int in_alias=0;
|
|
int set_ignore = 0;
|
|
int unread_buf[3]; /* We use an int so that we can store EOF. */
|
|
int unread_buf_count = 0;
|
|
|
|
if (arg && !opts_orig)
|
|
{
|
|
deinitialize (arg);
|
|
return 0;
|
|
}
|
|
|
|
if (!fp) /* Divert to arg_parse() in this case. */
|
|
return arg_parse (arg, opts_orig, 0);
|
|
|
|
if (initialize (arg, opts_orig, fp))
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
|
|
opts = arg->internal->opts;
|
|
nopts = arg->internal->nopts;
|
|
|
|
/* If the LINENO is zero we assume that we are at the start of a
|
|
* file and we skip over a possible Byte Order Mark. */
|
|
if (!arg->lineno)
|
|
{
|
|
unread_buf[0] = gpgrt_fgetc (fp);
|
|
unread_buf[1] = gpgrt_fgetc (fp);
|
|
unread_buf[2] = gpgrt_fgetc (fp);
|
|
if (unread_buf[0] != 0xef
|
|
|| unread_buf[1] != 0xbb
|
|
|| unread_buf[2] != 0xbf)
|
|
unread_buf_count = 3;
|
|
}
|
|
|
|
arg->internal->opt_flags = 0;
|
|
|
|
/* Find the next keyword. */
|
|
state = Ainit;
|
|
i = 0;
|
|
for (;;)
|
|
{
|
|
nextstate:
|
|
/* Before scanning the next char handle the keyword seen states. */
|
|
if (state == Akeyword_eol || state == Akeyword_spc)
|
|
{
|
|
/* We are either at the end of a line or right after a
|
|
* keyword. In the latter case we need to find the keyword
|
|
* so that we can decide whether an argument is required. */
|
|
|
|
/* Check the keyword. */
|
|
for (idx=0; idx < nopts; idx++ )
|
|
{
|
|
if (opts[idx].long_opt && !strcmp (opts[idx].long_opt, keyword))
|
|
break;
|
|
}
|
|
arg->r_opt = opts[idx].short_opt;
|
|
if (!(idx < nopts))
|
|
{
|
|
/* The option (keyword) is not known - check for
|
|
* internal keywords before returning an error. */
|
|
if (state == Akeyword_spc && !strcmp (keyword, "alias"))
|
|
{
|
|
in_alias = 1;
|
|
state = Awaitarg;
|
|
}
|
|
else if (!strcmp (keyword, "ignore-invalid-option"))
|
|
{
|
|
/* We might have keywords as argument - add them to
|
|
* the list of ignored keywords. Note that we
|
|
* ignore empty argument lists and thus do not to
|
|
* call the function in the Akeyword_eol state. */
|
|
if (state == Akeyword_spc)
|
|
{
|
|
if (ignore_invalid_option_add (arg, fp))
|
|
{
|
|
arg->r_opt = ARGPARSE_OUT_OF_CORE;
|
|
goto leave;
|
|
}
|
|
arg->lineno++;
|
|
}
|
|
state = Ainit;
|
|
i = 0;
|
|
}
|
|
else if (ignore_invalid_option_p (arg, keyword))
|
|
{
|
|
/* This invalid option is already in the iio list. */
|
|
state = state == Akeyword_eol? Ainit : Acomment;
|
|
i = 0;
|
|
}
|
|
else
|
|
{
|
|
arg->r_opt = ((opts[idx].flags & ARGPARSE_OPT_COMMAND)
|
|
? ARGPARSE_INVALID_COMMAND
|
|
: ARGPARSE_INVALID_OPTION);
|
|
if (state == Akeyword_spc)
|
|
state = Askipandleave;
|
|
else
|
|
goto leave;
|
|
}
|
|
}
|
|
else if (state != Akeyword_spc
|
|
&& arg->internal->in_sysconf
|
|
&& arg->internal->user_seen
|
|
&& !arg->internal->user_active)
|
|
{
|
|
/* We are in a [user] meta command and it is not active.
|
|
* Skip the command. */
|
|
state = state == Akeyword_eol? Ainit : Acomment;
|
|
i = 0;
|
|
}
|
|
else if (state != Akeyword_spc
|
|
&& (opts[idx].flags & ARGPARSE_OPT_IGNORE))
|
|
{
|
|
/* Known option is configured to be ignored. Start from
|
|
* scratch (new line) or process like a comment. */
|
|
state = state == Akeyword_eol? Ainit : Acomment;
|
|
i = 0;
|
|
}
|
|
else /* Known option */
|
|
{
|
|
set_ignore = 0;
|
|
|
|
if (arg->internal->in_sysconf)
|
|
{
|
|
/* Set the current forced and ignored attributes. */
|
|
if (arg->internal->mark_forced)
|
|
opts[idx].forced = 1;
|
|
if (arg->internal->mark_ignore)
|
|
opts[idx].ignore = 1;
|
|
if (arg->internal->explicit_ignore)
|
|
opts[idx].explicit_ignore = 1;
|
|
|
|
if (opts[idx].ignore && !opts[idx].forced)
|
|
{
|
|
if (arg->internal->verbose)
|
|
log_info ("%s:%u: ignoring option \"--%s\"\n",
|
|
arg->internal->confname,
|
|
arg->lineno,
|
|
opts[idx].long_opt);
|
|
if ((arg->flags & ARGPARSE_FLAG_WITHATTR))
|
|
set_ignore = 1;
|
|
else
|
|
{
|
|
state = state == Akeyword_eol? Ainit : Acomment;
|
|
i = 0;
|
|
goto nextstate; /* Ignore this one. */
|
|
}
|
|
}
|
|
}
|
|
else /* Non-sysconf file */
|
|
{ /* Act upon the forced and ignored attributes. */
|
|
if (opts[idx].ignore || opts[idx].forced)
|
|
{
|
|
if (arg->internal->verbose)
|
|
log_info ("%s:%u: ignoring option \"--%s\""
|
|
" due to attributes:%s%s\n",
|
|
arg->internal->confname,
|
|
arg->lineno,
|
|
opts[idx].long_opt,
|
|
opts[idx].forced? " forced":"",
|
|
opts[idx].ignore? " ignore":"");
|
|
if ((arg->flags & ARGPARSE_FLAG_WITHATTR))
|
|
set_ignore = 1;
|
|
else
|
|
{
|
|
state = state == Akeyword_eol? Ainit : Acomment;
|
|
i = 0;
|
|
goto nextstate; /* Ignore this one. */
|
|
}
|
|
}
|
|
}
|
|
|
|
if (state == Akeyword_spc)
|
|
{
|
|
/* If we shall ignore but not set the option we skip
|
|
* the argument. Otherwise we would need to use a
|
|
* made-up but not used args in the conf file. */
|
|
if (set_ignore || (opts[idx].ignore && !opts[idx].forced))
|
|
{
|
|
prepare_arg_return (arg, opts, idx, 0, set_ignore);
|
|
set_ignore = 0;
|
|
state = Askipandleave;
|
|
}
|
|
else
|
|
state = Awaitarg;
|
|
}
|
|
else
|
|
{
|
|
prepare_arg_return (arg, opts, idx, 0, set_ignore);
|
|
set_ignore = 0;
|
|
goto leave;
|
|
}
|
|
|
|
}
|
|
} /* (end state Akeyword_eol/Akeyword_spc) */
|
|
else if (state == Ametacmd)
|
|
{
|
|
/* We are at the end of a line. */
|
|
log_assert (*keyword == '[');
|
|
trim_spaces (keyword+1);
|
|
if (!keyword[1])
|
|
{
|
|
arg->r_opt = ARGPARSE_INVALID_META; /* Empty. */
|
|
goto leave;
|
|
}
|
|
c = handle_metacmd (arg, keyword+1);
|
|
if (c)
|
|
{
|
|
arg->r_opt = c; /* Return error. */
|
|
goto leave;
|
|
}
|
|
state = Ainit;
|
|
i = 0;
|
|
}
|
|
|
|
/* Get the next character from the line. */
|
|
if (unread_buf_count)
|
|
c = unread_buf[3 - unread_buf_count--];
|
|
else
|
|
c = gpgrt_fgetc (fp);
|
|
|
|
if (c == '\n' || c== EOF )
|
|
{ /* Handle end of line. */
|
|
if ( c != EOF )
|
|
arg->lineno++;
|
|
if (state == Askipandleave)
|
|
goto leave;
|
|
else if (state == Acopykeyword)
|
|
{
|
|
keyword[i] = 0;
|
|
state = Akeyword_eol;
|
|
goto nextstate;
|
|
}
|
|
else if (state == Acopymetacmd)
|
|
{
|
|
arg->r_opt = ARGPARSE_INVALID_META; /* "]" missing */
|
|
goto leave;
|
|
}
|
|
else if (state == Askipmetacmd || state == Askipmetacmd2)
|
|
{
|
|
state = Ametacmd;
|
|
goto nextstate;
|
|
}
|
|
else if (state == Awaitarg)
|
|
{
|
|
/* No argument found at the end of the line. */
|
|
prepare_arg_return (arg, opts, idx, in_alias, set_ignore);
|
|
set_ignore = 0;
|
|
goto leave;
|
|
}
|
|
else if (state == Acopyarg)
|
|
{
|
|
/* Has an argument at the end of a line. */
|
|
if (in_alias)
|
|
{
|
|
if (!buffer)
|
|
arg->r_opt = ARGPARSE_UNEXPECTED_ARG;
|
|
else
|
|
{
|
|
char *p;
|
|
|
|
buffer[i] = 0;
|
|
p = strpbrk (buffer, " \t");
|
|
if (p)
|
|
{
|
|
*p++ = 0;
|
|
trim_spaces (p);
|
|
}
|
|
if (!p || !*p)
|
|
{
|
|
xfree (buffer);
|
|
arg->r_opt = ARGPARSE_INVALID_ALIAS;
|
|
}
|
|
else
|
|
{
|
|
store_alias (arg, buffer, p);
|
|
}
|
|
}
|
|
}
|
|
else if (!(opts[idx].flags & ARGPARSE_TYPE_MASK))
|
|
arg->r_opt = ARGPARSE_UNEXPECTED_ARG;
|
|
else
|
|
{
|
|
char *p;
|
|
|
|
if (!buffer)
|
|
{
|
|
keyword[i] = 0;
|
|
buffer = xtrystrdup (keyword);
|
|
if (!buffer)
|
|
arg->r_opt = ARGPARSE_OUT_OF_CORE;
|
|
}
|
|
else
|
|
buffer[i] = 0;
|
|
|
|
if (buffer)
|
|
{
|
|
trim_spaces (buffer);
|
|
p = buffer;
|
|
if (*p == '"')
|
|
{
|
|
/* Remove quotes. */
|
|
p++;
|
|
if (*p && p[strlen(p)-1] == '\"' )
|
|
p[strlen(p)-1] = 0;
|
|
}
|
|
if (!set_opt_arg (arg, opts[idx].flags, p))
|
|
xfree (buffer);
|
|
else
|
|
gpgrt_annotate_leaked_object (buffer);
|
|
/* If the caller wants us to return the attributes or
|
|
* ignored options, or these flags in. */
|
|
if ((arg->flags & ARGPARSE_FLAG_WITHATTR))
|
|
{
|
|
if (opts[idx].ignore)
|
|
arg->r_type |= ARGPARSE_ATTR_IGNORE;
|
|
if (opts[idx].forced)
|
|
arg->r_type |= ARGPARSE_ATTR_FORCE;
|
|
if (set_ignore)
|
|
arg->r_type |= ARGPARSE_OPT_IGNORE;
|
|
}
|
|
}
|
|
}
|
|
goto leave;
|
|
}
|
|
else if (c == EOF)
|
|
{
|
|
ignore_invalid_option_clear (arg);
|
|
if (gpgrt_ferror (fp))
|
|
arg->r_opt = ARGPARSE_READ_ERROR;
|
|
else
|
|
arg->r_opt = 0; /* EOF. */
|
|
goto leave;
|
|
}
|
|
state = Ainit;
|
|
i = 0;
|
|
} /* (end handle end of line) */
|
|
else if (state == Askipandleave)
|
|
; /* Skip. */
|
|
else if (state == Ainit && isascii (c) && isspace(c))
|
|
; /* Skip leading white space. */
|
|
else if (state == Ainit && c == '#' )
|
|
state = Acomment; /* Start of a comment. */
|
|
else if (state == Acomment || state == Askipmetacmd2)
|
|
; /* Skip comments. */
|
|
else if (state == Askipmetacmd)
|
|
{
|
|
if (c == '#')
|
|
state = Askipmetacmd2;
|
|
else if (!(isascii (c) && isspace(c)))
|
|
{
|
|
arg->r_opt = ARGPARSE_INVALID_META;
|
|
state = Askipandleave;
|
|
}
|
|
}
|
|
else if (state == Acopykeyword && isascii (c) && isspace(c))
|
|
{
|
|
keyword[i] = 0;
|
|
state = Akeyword_spc;
|
|
goto nextstate;
|
|
}
|
|
else if (state == Acopymetacmd && c == ']')
|
|
{
|
|
keyword[i] = 0;
|
|
state = Askipmetacmd;
|
|
goto nextstate;
|
|
}
|
|
else if (state == Awaitarg)
|
|
{
|
|
/* Skip leading spaces of the argument. */
|
|
if (!isascii (c) || !isspace(c))
|
|
{
|
|
i = 0;
|
|
keyword[i++] = c;
|
|
state = Acopyarg;
|
|
}
|
|
}
|
|
else if (state == Acopyarg)
|
|
{
|
|
/* Collect the argument. */
|
|
if (buffer)
|
|
{
|
|
if (i < buflen-1)
|
|
buffer[i++] = c;
|
|
else
|
|
{
|
|
char *tmp;
|
|
size_t tmplen = buflen + 50;
|
|
|
|
tmp = xtryrealloc (buffer, tmplen);
|
|
if (tmp)
|
|
{
|
|
buflen = tmplen;
|
|
buffer = tmp;
|
|
buffer[i++] = c;
|
|
}
|
|
else
|
|
{
|
|
xfree (buffer);
|
|
arg->r_opt = ARGPARSE_OUT_OF_CORE;
|
|
goto leave;
|
|
}
|
|
}
|
|
}
|
|
else if (i < DIM(keyword)-1)
|
|
keyword[i++] = c;
|
|
else
|
|
{
|
|
size_t tmplen = DIM(keyword) + 50;
|
|
buffer = xtrymalloc (tmplen);
|
|
if (buffer)
|
|
{
|
|
buflen = tmplen;
|
|
memcpy(buffer, keyword, i);
|
|
buffer[i++] = c;
|
|
}
|
|
else
|
|
{
|
|
arg->r_opt = ARGPARSE_OUT_OF_CORE;
|
|
goto leave;
|
|
}
|
|
}
|
|
}
|
|
else if (i >= DIM(keyword)-1)
|
|
{
|
|
arg->r_opt = ARGPARSE_KEYWORD_TOO_LONG;
|
|
state = Askipandleave; /* Skip rest of line and leave. */
|
|
}
|
|
else if (!i)
|
|
{
|
|
state = c == '[' ? Acopymetacmd : Acopykeyword;
|
|
keyword[i++] = c;
|
|
}
|
|
else
|
|
{
|
|
keyword[i++] = c;
|
|
}
|
|
}
|
|
|
|
leave:
|
|
return arg->r_opt;
|
|
}
|
|
|
|
|
|
/* Return true if the list of options OPTS has any option marked with
|
|
* ARGPARSE_OPT_CONFFILE. */
|
|
static int
|
|
any_opt_conffile (opttable_t *opts, unsigned int nopts)
|
|
{
|
|
int i;
|
|
|
|
for (i=0; i < nopts; i++ )
|
|
if ((opts[i].flags & ARGPARSE_OPT_CONFFILE))
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Return true if FNAME is an absolute filename. */
|
|
static int
|
|
is_absfname (const char *fname)
|
|
{
|
|
const char *s;
|
|
|
|
#ifdef HAVE_W32_SYSTEM
|
|
s = strchr (fname, ':');
|
|
if (s)
|
|
s++;
|
|
else
|
|
s = fname;
|
|
#else
|
|
s = fname;
|
|
#endif
|
|
|
|
return (*s == '/'
|
|
#ifdef HAVE_W32_SYSTEM
|
|
|| *s == DIRSEP_C
|
|
#endif
|
|
);
|
|
}
|
|
|
|
|
|
/* If FNAME specifies two files of the form
|
|
* NAME1:/NAME2 (Unix)
|
|
* or
|
|
* NAME1;[x:]/NAME2 (Windows)
|
|
* return a pointer to the delimiter or NULL if there is none.
|
|
*/
|
|
static const char *
|
|
is_twopartfname (const char *fname)
|
|
{
|
|
const char *s;
|
|
|
|
if ((s = strchr (fname, PATHSEP_C)) && is_absfname (s+1) && s != fname)
|
|
return s;
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* Try to use a version-ed config file name. A version-ed config file
|
|
* name is one which has the packages version number appended. For
|
|
* example if the standard config file name is "foo.conf" and the
|
|
* version of the foo program is 1.2.3-beta1 the following config
|
|
* files are tried in order until one is readable:
|
|
*
|
|
* foo.conf-1.2.3-beta1
|
|
* foo.conf-1.2.3
|
|
* foo.conf-1.2
|
|
* foo.conf-1
|
|
* foo.conf
|
|
*
|
|
* The argument CONFIGNAME should already be expanded. On success a
|
|
* newly allocated file name is returned. On error NULL is returned.
|
|
*/
|
|
static char *
|
|
try_versioned_conffile (const char *configname)
|
|
{
|
|
const char *version = strusage (13);
|
|
char *name;
|
|
char *dash, *endp;
|
|
|
|
if (!version || !*version)
|
|
return NULL; /* No program version known. */
|
|
|
|
name = strconcat (configname, "-", version, NULL);
|
|
if (!name)
|
|
return NULL; /* Oops: Out of core - ignore. */
|
|
dash = name + strlen (configname);
|
|
|
|
endp = dash + strlen (dash) - 1;
|
|
while (endp > dash)
|
|
{
|
|
if (!gnupg_access (name, R_OK))
|
|
{
|
|
return name;
|
|
}
|
|
for (; endp > dash; endp--)
|
|
{
|
|
if (*endp == '-' || *endp == '.')
|
|
{
|
|
*endp = 0;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
xfree (name);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* This function is called after a sysconf file has been read. */
|
|
static void
|
|
finish_read_sys (gnupg_argparse_t *arg)
|
|
{
|
|
opttable_t *opts = arg->internal->opts;
|
|
unsigned int nopts = arg->internal->nopts;
|
|
int i;
|
|
|
|
if (arg->internal->ignore_all_seen)
|
|
{
|
|
/* [ignore-all] was used: Set all options which have not
|
|
* explictly been set as ignore or not ignore to ignore. */
|
|
for (i = 0; i < nopts; i++)
|
|
{
|
|
if (!opts[i].explicit_ignore)
|
|
opts[i].ignore = 1;
|
|
}
|
|
}
|
|
|
|
/* Reset all flags which pertain only to sysconf files. */
|
|
arg->internal->in_sysconf = 0;
|
|
arg->internal->user_active = 0;
|
|
arg->internal->mark_forced = 0;
|
|
arg->internal->mark_ignore = 0;
|
|
arg->internal->explicit_ignore = 0;
|
|
arg->internal->ignore_all_seen = 0;
|
|
}
|
|
|
|
/* The full arg parser which handles option files and command line
|
|
* arguments. The behaviour depends on the combinations of CONFNAME
|
|
* and the ARGPARSE_FLAG_xxx values:
|
|
*
|
|
* | CONFNAME | SYS | USER | Action |
|
|
* |----------+-----+------+--------------------|
|
|
* | NULL | - | - | cmdline |
|
|
* | string | 0 | 1 | user, cmdline |
|
|
* | string | 1 | 0 | sys, cmdline |
|
|
* | string | 1 | 1 | sys, user, cmdline |
|
|
*
|
|
* Note that if an option has been flagged with ARGPARSE_OPT_CONFFILE
|
|
* and a type of ARGPARSE_TYPE_STRING that option is not returned but
|
|
* the specified configuration file is processed directly; if
|
|
* ARGPARSE_TYPE_NONE is used no user configuration files are
|
|
* processed and from the system configuration files only those which
|
|
* are immutable are processed. The string values for CONFNAME shall
|
|
* not include a directory part because that is taken from the values
|
|
* set by gnupg_set_confdir. However, if CONFNAME is a twopart
|
|
* filename delimited by a colon (semicolon on Windows) with the
|
|
* second part being an absolute filename, the first part is used for
|
|
* the SYS file and the the entire second part for the USER file.
|
|
*/
|
|
int
|
|
gnupg_argparser (gnupg_argparse_t *arg, gnupg_opt_t *opts,
|
|
const char *confname)
|
|
{
|
|
/* First check whether releasing the resources has been requested. */
|
|
if (arg && !opts)
|
|
{
|
|
deinitialize (arg);
|
|
return 0;
|
|
}
|
|
|
|
/* Make sure that the internal data object is ready and also print
|
|
* warnings or errors from the last iteration. */
|
|
if (initialize (arg, opts, NULL))
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
|
|
next_state:
|
|
switch (arg->internal->state)
|
|
{
|
|
case STATE_init:
|
|
if (arg->argc && arg->argv && *arg->argc
|
|
&& any_opt_conffile (arg->internal->opts, arg->internal->nopts))
|
|
{
|
|
/* The list of option allow for conf files
|
|
* (e.g. gpg's "--option FILE" and "--no-options")
|
|
* Now check whether one was really given on the command
|
|
* line. Note that we don't need to run this code if no
|
|
* argument array was provided. */
|
|
int save_argc = *arg->argc;
|
|
char **save_argv = *arg->argv;
|
|
unsigned int save_flags = arg->flags;
|
|
int save_idx = arg->internal->idx;
|
|
int any_no_conffile = 0;
|
|
|
|
arg->flags = (ARGPARSE_FLAG_KEEP | ARGPARSE_FLAG_NOVERSION
|
|
| ARGPARSE_FLAG__INITIALIZED);
|
|
while (arg_parse (arg, opts, 1))
|
|
{
|
|
if ((arg->internal->opt_flags & ARGPARSE_OPT_CONFFILE))
|
|
{
|
|
arg->internal->explicit_confopt = 1;
|
|
if ((arg->r_type & ARGPARSE_TYPE_MASK) == ARGPARSE_TYPE_STRING
|
|
&& !arg->internal->explicit_conffile)
|
|
{
|
|
/* Store the first conffile name. All further
|
|
* conf file options are not handled. */
|
|
arg->internal->explicit_conffile
|
|
= xtrystrdup (arg->r.ret_str);
|
|
if (!arg->internal->explicit_conffile)
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
|
|
}
|
|
else if ((arg->r_type & ARGPARSE_TYPE_MASK)
|
|
== ARGPARSE_TYPE_NONE)
|
|
any_no_conffile = 1;
|
|
}
|
|
}
|
|
if (any_no_conffile)
|
|
{
|
|
/* A NoConffile option overrides any other conf file option. */
|
|
xfree (arg->internal->explicit_conffile);
|
|
arg->internal->explicit_conffile = NULL;
|
|
}
|
|
/* Restore parser. */
|
|
*arg->argc = save_argc;
|
|
*arg->argv = save_argv;
|
|
arg->flags = save_flags;
|
|
arg->internal->idx = save_idx;
|
|
}
|
|
|
|
if (confname && *confname)
|
|
{
|
|
if ((arg->flags & ARGPARSE_FLAG_SYS))
|
|
arg->internal->state = STATE_open_sys;
|
|
else if ((arg->flags & ARGPARSE_FLAG_USER))
|
|
arg->internal->state = STATE_open_user;
|
|
else
|
|
return (arg->r_opt = ARGPARSE_INVALID_ARG);
|
|
}
|
|
else
|
|
arg->internal->state = STATE_open_cmdline;
|
|
goto next_state;
|
|
|
|
case STATE_open_sys:
|
|
{
|
|
/* If it is a two part name take the first part. */
|
|
const char *s;
|
|
char *tmpname = NULL;
|
|
|
|
if ((s = is_twopartfname (confname)))
|
|
{
|
|
tmpname = xtrymalloc (s - confname + 1);
|
|
if (!tmpname)
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
memcpy (tmpname, confname, s-confname);
|
|
tmpname[s-confname] = 0;
|
|
s = tmpname;
|
|
}
|
|
else
|
|
s = confname;
|
|
xfree (arg->internal->confname);
|
|
arg->internal->confname = make_filename_try
|
|
(confdir.sys? confdir.sys : "/etc", s, NULL);
|
|
xfree (tmpname);
|
|
if (!arg->internal->confname)
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
}
|
|
arg->lineno = 0;
|
|
arg->internal->idx = 0;
|
|
arg->internal->verbose = 0;
|
|
arg->internal->stopped = 0;
|
|
arg->internal->inarg = 0;
|
|
gpgrt_fclose (arg->internal->conffp);
|
|
arg->internal->conffp = gpgrt_fopen (arg->internal->confname, "r");
|
|
if (!arg->internal->conffp)
|
|
{
|
|
if ((arg->flags & ARGPARSE_FLAG_VERBOSE) || arg->internal->verbose)
|
|
log_info (_("Note: no default option file '%s'\n"),
|
|
arg->internal->confname);
|
|
if ((arg->flags & ARGPARSE_FLAG_USER))
|
|
arg->internal->state = STATE_open_user;
|
|
else
|
|
arg->internal->state = STATE_open_cmdline;
|
|
goto next_state;
|
|
}
|
|
|
|
if ((arg->flags & ARGPARSE_FLAG_VERBOSE) || arg->internal->verbose)
|
|
log_info (_("reading options from '%s'\n"),
|
|
arg->internal->confname);
|
|
arg->internal->state = STATE_read_sys;
|
|
arg->internal->in_sysconf = 1;
|
|
arg->r.ret_str = xtrystrdup (arg->internal->confname);
|
|
if (!arg->r.ret_str)
|
|
arg->r_opt = ARGPARSE_OUT_OF_CORE;
|
|
else
|
|
{
|
|
gpgrt_annotate_leaked_object (arg->r.ret_str);
|
|
arg->r_opt = ARGPARSE_CONFFILE;
|
|
arg->r_type = ARGPARSE_TYPE_STRING;
|
|
}
|
|
break;
|
|
|
|
case STATE_open_user:
|
|
if (arg->internal->explicit_confopt
|
|
&& arg->internal->explicit_conffile)
|
|
{
|
|
/* An explict option to use a specific configuration file
|
|
* has been given - use that one. */
|
|
xfree (arg->internal->confname);
|
|
arg->internal->confname
|
|
= xtrystrdup (arg->internal->explicit_conffile);
|
|
if (!arg->internal->confname)
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
}
|
|
else if (arg->internal->explicit_confopt)
|
|
{
|
|
/* An explict option not to use a configuration file has
|
|
* been given - leap direct to command line reading. */
|
|
arg->internal->state = STATE_open_cmdline;
|
|
goto next_state;
|
|
}
|
|
else
|
|
{
|
|
/* Use the standard configure file. If it is a two part
|
|
* name take the second part. If it is the standard name
|
|
* and ARGPARSE_FLAG_USERVERS is set try versioned config
|
|
* files. */
|
|
const char *s;
|
|
char *nconf;
|
|
|
|
xfree (arg->internal->confname);
|
|
if ((s = is_twopartfname (confname)))
|
|
{
|
|
arg->internal->confname = make_filename_try (s + 1, NULL);
|
|
if (!arg->internal->confname)
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
}
|
|
else
|
|
{
|
|
arg->internal->confname = make_filename_try
|
|
(confdir.user? confdir.user : "~/.config", confname, NULL);
|
|
if (!arg->internal->confname)
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
if ((arg->flags & ARGPARSE_FLAG_USERVERS)
|
|
&& (nconf = try_versioned_conffile (arg->internal->confname)))
|
|
{
|
|
xfree (arg->internal->confname);
|
|
arg->internal->confname = nconf;
|
|
}
|
|
}
|
|
}
|
|
arg->lineno = 0;
|
|
arg->internal->idx = 0;
|
|
arg->internal->verbose = 0;
|
|
arg->internal->stopped = 0;
|
|
arg->internal->inarg = 0;
|
|
arg->internal->in_sysconf = 0;
|
|
gpgrt_fclose (arg->internal->conffp);
|
|
arg->internal->conffp = gpgrt_fopen (arg->internal->confname, "r");
|
|
if (!arg->internal->conffp)
|
|
{
|
|
arg->internal->state = STATE_open_cmdline;
|
|
if (arg->internal->explicit_confopt)
|
|
{
|
|
log_error (_("option file '%s': %s\n"),
|
|
arg->internal->confname, strerror (errno));
|
|
return (arg->r_opt = ARGPARSE_NO_CONFFILE);
|
|
}
|
|
else
|
|
{
|
|
if ((arg->flags & ARGPARSE_FLAG_VERBOSE)
|
|
|| arg->internal->verbose)
|
|
log_info (_("Note: no default option file '%s'\n"),
|
|
arg->internal->confname);
|
|
goto next_state;
|
|
}
|
|
}
|
|
|
|
if ((arg->flags & ARGPARSE_FLAG_VERBOSE) || arg->internal->verbose)
|
|
log_info (_("reading options from '%s'\n"),
|
|
arg->internal->confname);
|
|
arg->internal->state = STATE_read_user;
|
|
arg->r.ret_str = xtrystrdup (arg->internal->confname);
|
|
if (!arg->r.ret_str)
|
|
arg->r_opt = ARGPARSE_OUT_OF_CORE;
|
|
else
|
|
{
|
|
gpgrt_annotate_leaked_object (arg->r.ret_str);
|
|
arg->r_opt = ARGPARSE_CONFFILE;
|
|
arg->r_type = ARGPARSE_TYPE_STRING;
|
|
}
|
|
break;
|
|
|
|
case STATE_open_cmdline:
|
|
gpgrt_fclose (arg->internal->conffp);
|
|
arg->internal->conffp = NULL;
|
|
xfree (arg->internal->confname);
|
|
arg->internal->confname = NULL;
|
|
arg->internal->idx = 0;
|
|
arg->internal->verbose = 0;
|
|
arg->internal->stopped = 0;
|
|
arg->internal->inarg = 0;
|
|
arg->internal->in_sysconf = 0;
|
|
if (!arg->argc || !arg->argv || !*arg->argv)
|
|
{
|
|
/* No or empty argument vector - don't bother to parse things. */
|
|
arg->internal->state = STATE_finished;
|
|
goto next_state;
|
|
}
|
|
arg->r_opt = ARGPARSE_CONFFILE;
|
|
arg->r_type = ARGPARSE_TYPE_NONE;
|
|
arg->r.ret_str = NULL;
|
|
arg->internal->state = STATE_read_cmdline;
|
|
break;
|
|
|
|
case STATE_read_sys:
|
|
arg->r_opt = gnupg_argparse (arg->internal->conffp, arg, opts);
|
|
if (!arg->r_opt)
|
|
{
|
|
finish_read_sys (arg);
|
|
arg->internal->state = STATE_open_user;
|
|
goto next_state;
|
|
}
|
|
if ((arg->internal->opt_flags & ARGPARSE_OPT_CONFFILE))
|
|
goto next_state; /* Already handled - again. */
|
|
break;
|
|
|
|
case STATE_read_user:
|
|
arg->r_opt = gnupg_argparse (arg->internal->conffp, arg, opts);
|
|
if (!arg->r_opt)
|
|
{
|
|
arg->internal->state = STATE_open_cmdline;
|
|
goto next_state;
|
|
}
|
|
if ((arg->internal->opt_flags & ARGPARSE_OPT_CONFFILE))
|
|
goto next_state; /* Already handled - again. */
|
|
break;
|
|
|
|
case STATE_read_cmdline:
|
|
arg->r_opt = arg_parse (arg, opts, 1);
|
|
if (!arg->r_opt)
|
|
{
|
|
arg->internal->state = STATE_finished;
|
|
goto next_state;
|
|
}
|
|
if ((arg->internal->opt_flags & ARGPARSE_OPT_CONFFILE))
|
|
goto next_state; /* Already handled - again. */
|
|
break;
|
|
|
|
case STATE_finished:
|
|
arg->r_opt = 0;
|
|
break;
|
|
}
|
|
|
|
return arg->r_opt;
|
|
}
|
|
|
|
|
|
|
|
/* Given the list of options in ARG and a keyword, return the index of
|
|
* the long option matching KEYWORD. On error -1 is returned for not
|
|
* found or -2 for ambigious keyword. */
|
|
static int
|
|
find_long_option (gnupg_argparse_t *arg, const char *keyword)
|
|
{
|
|
int i;
|
|
size_t n;
|
|
opttable_t *opts = arg->internal->opts;
|
|
unsigned int nopts = arg->internal->nopts;
|
|
|
|
/* Would be better if we can do a binary search, but it is not
|
|
* possible to reorder our option table because we would mess up our
|
|
* help strings. What we can do is: Build an option lookup table
|
|
* when this function is first invoked. The latter has already been
|
|
* done. */
|
|
if (!*keyword)
|
|
return -1;
|
|
for (i=0; i < nopts; i++ )
|
|
if (opts[i].long_opt && !strcmp (opts[i].long_opt, keyword))
|
|
return i;
|
|
/* Not found. See whether it is an abbreviation. Aliases may not
|
|
* be abbreviated, though. */
|
|
n = strlen (keyword);
|
|
for (i=0; i < nopts; i++)
|
|
{
|
|
if (opts[i].long_opt && !strncmp (opts[i].long_opt, keyword, n))
|
|
{
|
|
int j;
|
|
for (j=i+1; j < nopts; j++)
|
|
{
|
|
if (opts[j].long_opt
|
|
&& !strncmp (opts[j].long_opt, keyword, n)
|
|
&& !(opts[j].short_opt == opts[i].short_opt
|
|
&& opts[j].flags == opts[i].flags ) )
|
|
return -2; /* Abbreviation is ambiguous. */
|
|
}
|
|
return i;
|
|
}
|
|
}
|
|
return -1; /* Not found. */
|
|
}
|
|
|
|
|
|
/* The option parser for command line options. */
|
|
static int
|
|
arg_parse (gnupg_argparse_t *arg, gnupg_opt_t *opts_orig, int no_init)
|
|
{
|
|
int idx;
|
|
opttable_t *opts;
|
|
unsigned int nopts;
|
|
int argc;
|
|
char **argv;
|
|
char *s, *s2;
|
|
int i;
|
|
|
|
if (no_init)
|
|
;
|
|
else if (initialize (arg, opts_orig, NULL))
|
|
return (arg->r_opt = ARGPARSE_OUT_OF_CORE);
|
|
|
|
opts = arg->internal->opts;
|
|
nopts = arg->internal->nopts;
|
|
argc = *arg->argc;
|
|
argv = *arg->argv;
|
|
idx = arg->internal->idx;
|
|
|
|
if (!idx && argc && !(arg->flags & ARGPARSE_FLAG_ARG0))
|
|
{
|
|
/* Skip the first argument. */
|
|
argc--; argv++; idx++;
|
|
}
|
|
|
|
next_one:
|
|
if (!argc || (s = *argv) == NULL)
|
|
{
|
|
/* No more args. */
|
|
arg->r_opt = 0;
|
|
goto leave; /* Ready. */
|
|
}
|
|
|
|
arg->internal->last = s;
|
|
arg->internal->opt_flags = 0;
|
|
|
|
if (arg->internal->stopped && (arg->flags & ARGPARSE_FLAG_ALL))
|
|
{
|
|
arg->r_opt = ARGPARSE_IS_ARG; /* Not an option but an argument. */
|
|
arg->r_type = ARGPARSE_TYPE_STRING;
|
|
arg->r.ret_str = s;
|
|
argc--; argv++; idx++; /* set to next one */
|
|
}
|
|
else if (arg->internal->stopped)
|
|
{
|
|
arg->r_opt = 0;
|
|
goto leave; /* Ready. */
|
|
}
|
|
else if ( *s == '-' && s[1] == '-' )
|
|
{
|
|
/* Long option. */
|
|
char *argpos;
|
|
|
|
arg->internal->inarg = 0;
|
|
if (!s[2] && !(arg->flags & ARGPARSE_FLAG_NOSTOP))
|
|
{
|
|
/* Stop option processing. */
|
|
arg->internal->stopped = 1;
|
|
arg->flags |= ARGPARSE_FLAG_STOP_SEEN;
|
|
argc--; argv++; idx++;
|
|
goto next_one;
|
|
}
|
|
|
|
argpos = strchr( s+2, '=' );
|
|
if ( argpos )
|
|
*argpos = 0;
|
|
i = find_long_option (arg, s+2);
|
|
if ( argpos )
|
|
*argpos = '=';
|
|
|
|
if (i > 0 && opts[i].short_opt == ARGPARSE_SHORTOPT_HELP)
|
|
{
|
|
show_help (opts, nopts, arg->flags);
|
|
my_exit (arg, 0);
|
|
}
|
|
else if (i > 0 && opts[i].short_opt == ARGPARSE_SHORTOPT_VERSION)
|
|
{
|
|
if (!(arg->flags & ARGPARSE_FLAG_NOVERSION))
|
|
{
|
|
show_version ();
|
|
my_exit (arg, 0);
|
|
}
|
|
}
|
|
else if (i > 0 && opts[i].short_opt == ARGPARSE_SHORTOPT_WARRANTY)
|
|
{
|
|
writestrings (0, strusage (16), "\n", NULL);
|
|
my_exit (arg, 0);
|
|
}
|
|
else if (i > 0 && opts[i].short_opt == ARGPARSE_SHORTOPT_DUMP_OPTTBL)
|
|
dump_option_table (arg);
|
|
else if (i > 0 && opts[i].short_opt == ARGPARSE_SHORTOPT_DUMP_OPTIONS)
|
|
{
|
|
for (i=0; i < nopts; i++ )
|
|
{
|
|
if (opts[i].long_opt && !(opts[i].flags & ARGPARSE_OPT_IGNORE))
|
|
writestrings (0, "--", opts[i].long_opt, "\n", NULL);
|
|
}
|
|
my_exit (arg, 0);
|
|
}
|
|
|
|
if ( i == -2 )
|
|
arg->r_opt = ARGPARSE_AMBIGUOUS_OPTION;
|
|
else if ( i == -1 )
|
|
{
|
|
arg->r_opt = ARGPARSE_INVALID_OPTION;
|
|
arg->r.ret_str = s+2;
|
|
}
|
|
else
|
|
arg->r_opt = opts[i].short_opt;
|
|
|
|
if ( i < 0 )
|
|
;
|
|
else if ( (opts[i].flags & ARGPARSE_TYPE_MASK) )
|
|
{
|
|
if ( argpos )
|
|
{
|
|
s2 = argpos+1;
|
|
if ( !*s2 )
|
|
s2 = NULL;
|
|
}
|
|
else
|
|
s2 = argv[1];
|
|
|
|
if ( !s2 && (opts[i].flags & ARGPARSE_OPT_OPTIONAL) )
|
|
{
|
|
arg->r_type = ARGPARSE_TYPE_NONE; /* Argument is optional. */
|
|
}
|
|
else if ( !s2 )
|
|
{
|
|
arg->r_opt = ARGPARSE_MISSING_ARG;
|
|
}
|
|
else if ( !argpos && *s2 == '-'
|
|
&& (opts[i].flags & ARGPARSE_OPT_OPTIONAL) )
|
|
{
|
|
/* The argument is optional and the next seems to be an
|
|
option. We do not check this possible option but
|
|
assume no argument */
|
|
arg->r_type = ARGPARSE_TYPE_NONE;
|
|
}
|
|
else
|
|
{
|
|
set_opt_arg (arg, opts[i].flags, s2);
|
|
if ( !argpos )
|
|
{
|
|
argc--; argv++; idx++; /* Skip one. */
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Does not take an argument. */
|
|
if ( argpos )
|
|
arg->r_type = ARGPARSE_UNEXPECTED_ARG;
|
|
else
|
|
{
|
|
arg->internal->opt_flags = opts[i].flags;
|
|
arg->r_type = ARGPARSE_TYPE_NONE;
|
|
}
|
|
}
|
|
argc--; argv++; idx++; /* Set to next one. */
|
|
}
|
|
else if ( (*s == '-' && s[1]) || arg->internal->inarg )
|
|
{
|
|
/* Short option. */
|
|
int dash_kludge = 0;
|
|
|
|
i = 0;
|
|
if ( !arg->internal->inarg )
|
|
{
|
|
arg->internal->inarg++;
|
|
if ( (arg->flags & ARGPARSE_FLAG_ONEDASH) )
|
|
{
|
|
for (i=0; i < nopts; i++ )
|
|
if ( opts[i].long_opt && !strcmp (opts[i].long_opt, s+1))
|
|
{
|
|
dash_kludge = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
s += arg->internal->inarg;
|
|
|
|
if (!dash_kludge )
|
|
{
|
|
for (i=0; i < nopts; i++ )
|
|
if ( opts[i].short_opt == *s )
|
|
break;
|
|
}
|
|
|
|
if ( !opts[i].short_opt && ( *s == 'h' || *s == '?' ) )
|
|
{
|
|
show_help (opts, nopts, arg->flags);
|
|
my_exit (arg, 0);
|
|
}
|
|
|
|
arg->r_opt = opts[i].short_opt;
|
|
if (!opts[i].short_opt )
|
|
{
|
|
arg->r_opt = (opts[i].flags & ARGPARSE_OPT_COMMAND)?
|
|
ARGPARSE_INVALID_COMMAND:ARGPARSE_INVALID_OPTION;
|
|
arg->internal->inarg++; /* Point to the next arg. */
|
|
arg->r.ret_str = s;
|
|
}
|
|
else if ( (opts[i].flags & ARGPARSE_TYPE_MASK) )
|
|
{
|
|
if ( s[1] && !dash_kludge )
|
|
{
|
|
s2 = s+1;
|
|
set_opt_arg (arg, opts[i].flags, s2);
|
|
}
|
|
else
|
|
{
|
|
s2 = argv[1];
|
|
if ( !s2 && (opts[i].flags & ARGPARSE_OPT_OPTIONAL) )
|
|
{
|
|
arg->r_type = ARGPARSE_TYPE_NONE;
|
|
arg->internal->opt_flags = opts[i].flags;
|
|
}
|
|
else if ( !s2 )
|
|
{
|
|
arg->r_opt = ARGPARSE_MISSING_ARG;
|
|
}
|
|
else if ( *s2 == '-' && s2[1]
|
|
&& (opts[i].flags & ARGPARSE_OPT_OPTIONAL) )
|
|
{
|
|
/* The argument is optional and the next seems to
|
|
be an option. We do not check this possible
|
|
option but assume no argument. */
|
|
arg->r_type = ARGPARSE_TYPE_NONE;
|
|
arg->internal->opt_flags = opts[i].flags;
|
|
}
|
|
else
|
|
{
|
|
set_opt_arg (arg, opts[i].flags, s2);
|
|
argc--; argv++; idx++; /* Skip one. */
|
|
}
|
|
}
|
|
s = "x"; /* This is so that !s[1] yields false. */
|
|
}
|
|
else
|
|
{
|
|
/* Does not take an argument. */
|
|
arg->r_type = ARGPARSE_TYPE_NONE;
|
|
arg->internal->opt_flags = opts[i].flags;
|
|
arg->internal->inarg++; /* Point to the next arg. */
|
|
}
|
|
if ( !s[1] || dash_kludge )
|
|
{
|
|
/* No more concatenated short options. */
|
|
arg->internal->inarg = 0;
|
|
argc--; argv++; idx++;
|
|
}
|
|
}
|
|
else if ( arg->flags & ARGPARSE_FLAG_MIXED )
|
|
{
|
|
arg->r_opt = ARGPARSE_IS_ARG;
|
|
arg->r_type = ARGPARSE_TYPE_STRING;
|
|
arg->r.ret_str = s;
|
|
argc--; argv++; idx++; /* Set to next one. */
|
|
}
|
|
else
|
|
{
|
|
arg->internal->stopped = 1; /* Stop option processing. */
|
|
goto next_one;
|
|
}
|
|
|
|
if (arg->r_opt > 0 && i >= 0 && i < nopts
|
|
&& ((opts[i].ignore && opts[i].explicit_ignore) || opts[i].forced))
|
|
{
|
|
|
|
if ((arg->flags & ARGPARSE_FLAG_WITHATTR))
|
|
{
|
|
if (opts[i].ignore)
|
|
arg->r_type |= ARGPARSE_ATTR_IGNORE;
|
|
if (opts[i].forced)
|
|
arg->r_type |= ARGPARSE_ATTR_FORCE;
|
|
arg->r_type |= ARGPARSE_OPT_IGNORE;
|
|
}
|
|
else
|
|
{
|
|
log_info (_("Note: ignoring option \"--%s\""
|
|
" due to global config\n"),
|
|
opts[i].long_opt);
|
|
goto next_one; /* Skip ignored/forced option. */
|
|
}
|
|
}
|
|
|
|
leave:
|
|
*arg->argc = argc;
|
|
*arg->argv = argv;
|
|
arg->internal->idx = idx;
|
|
return arg->r_opt;
|
|
}
|
|
|
|
|
|
|
|
/* Returns: -1 on error, 0 for an integer type and 1 for a non integer
|
|
type argument. */
|
|
static int
|
|
set_opt_arg (gnupg_argparse_t *arg, unsigned flags, char *s)
|
|
{
|
|
int base = (flags & ARGPARSE_OPT_PREFIX)? 0 : 10;
|
|
long l;
|
|
|
|
arg->internal->opt_flags = flags;
|
|
switch ( (arg->r_type = (flags & ARGPARSE_TYPE_MASK)) )
|
|
{
|
|
case ARGPARSE_TYPE_LONG:
|
|
case ARGPARSE_TYPE_INT:
|
|
errno = 0;
|
|
l = strtol (s, NULL, base);
|
|
if ((l == LONG_MIN || l == LONG_MAX) && errno == ERANGE)
|
|
{
|
|
arg->r_opt = ARGPARSE_INVALID_ARG;
|
|
return -1;
|
|
}
|
|
if (arg->r_type == ARGPARSE_TYPE_LONG)
|
|
arg->r.ret_long = l;
|
|
else if ( (l < 0 && l < INT_MIN) || l > INT_MAX )
|
|
{
|
|
arg->r_opt = ARGPARSE_INVALID_ARG;
|
|
return -1;
|
|
}
|
|
else
|
|
arg->r.ret_int = (int)l;
|
|
return 0;
|
|
|
|
case ARGPARSE_TYPE_ULONG:
|
|
while (isascii (*s) && isspace(*s))
|
|
s++;
|
|
if (*s == '-')
|
|
{
|
|
arg->r.ret_ulong = 0;
|
|
arg->r_opt = ARGPARSE_INVALID_ARG;
|
|
return -1;
|
|
}
|
|
errno = 0;
|
|
arg->r.ret_ulong = strtoul (s, NULL, base);
|
|
if (arg->r.ret_ulong == ULONG_MAX && errno == ERANGE)
|
|
{
|
|
arg->r_opt = ARGPARSE_INVALID_ARG;
|
|
return -1;
|
|
}
|
|
return 0;
|
|
|
|
case ARGPARSE_TYPE_STRING:
|
|
default:
|
|
arg->r.ret_str = s;
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
|
|
/* Return the length of the option O. This needs to consider the
|
|
* description as well as the option name. */
|
|
static size_t
|
|
long_opt_strlen (opttable_t *o)
|
|
{
|
|
size_t n = strlen (o->long_opt);
|
|
|
|
if ( o->description && *o->description == '|' )
|
|
{
|
|
const char *s;
|
|
int is_utf8 = is_native_utf8 ();
|
|
|
|
s=o->description+1;
|
|
if ( *s != '=' )
|
|
n++;
|
|
/* For a (mostly) correct length calculation we exclude
|
|
* continuation bytes (10xxxxxx) if we are on a native utf8
|
|
* terminal. */
|
|
for (; *s && *s != '|'; s++ )
|
|
if ( is_utf8 && (*s&0xc0) != 0x80 )
|
|
n++;
|
|
}
|
|
return n;
|
|
}
|
|
|
|
|
|
/* Qsort compare for show_help. */
|
|
static int
|
|
cmp_ordtbl (const void *a_v, const void *b_v)
|
|
{
|
|
const unsigned short *a = a_v;
|
|
const unsigned short *b = b_v;
|
|
|
|
return *a - *b;
|
|
}
|
|
|
|
|
|
/****************
|
|
* Print formatted help. The description string has some special
|
|
* meanings:
|
|
* - A description string which is "@" suppresses help output for
|
|
* this option
|
|
* - a description which starts with a '@' and is followed by
|
|
* any other characters is printed as is; this may be used for examples
|
|
* and such. This is a legacy methiod, moder codes uses the flags
|
|
* ARGPARSE_OPT_VERBATIM or ARGPARSE_OPT_HEADER.
|
|
* - A description which starts with a '|' outputs the string between this
|
|
* bar and the next one as arguments of the long option.
|
|
*/
|
|
static void
|
|
show_help (opttable_t *opts, unsigned int nopts, unsigned int flags)
|
|
{
|
|
const char *s;
|
|
char tmp[2];
|
|
unsigned int *ordtbl = NULL;
|
|
|
|
show_version ();
|
|
writestrings (0, "\n", NULL);
|
|
s = strusage (42);
|
|
if (s && *s == '1')
|
|
{
|
|
s = strusage (40);
|
|
writestrings (1, s, NULL);
|
|
if (*s && s[strlen(s)] != '\n')
|
|
writestrings (1, "\n", NULL);
|
|
}
|
|
s = strusage(41);
|
|
writestrings (0, s, "\n", NULL);
|
|
if ( nopts )
|
|
{
|
|
/* Auto format the option description. */
|
|
int i,j,indent;
|
|
const char *last_header = NULL;
|
|
|
|
ordtbl = xtrycalloc (nopts, sizeof *ordtbl);
|
|
if (!ordtbl)
|
|
{
|
|
writestrings (1, "\nOoops: Out of memory whilst printing the help.\n",
|
|
NULL);
|
|
goto leave;
|
|
}
|
|
|
|
/* Get max. length of long options. */
|
|
for (i=indent=0; i < nopts; i++ )
|
|
{
|
|
if ( opts[i].long_opt )
|
|
if ( !opts[i].description || *opts[i].description != '@' )
|
|
if ( (j=long_opt_strlen(opts+i)) > indent && j < 35 )
|
|
indent = j;
|
|
ordtbl[i] = opts[i].ordinal;
|
|
}
|
|
|
|
qsort (ordtbl, nopts, sizeof *ordtbl, cmp_ordtbl);
|
|
|
|
/* The first option needs to have a description; if not do not
|
|
* print the help at all. */
|
|
if (!opts[ordtbl[0]].description)
|
|
goto leave;
|
|
|
|
/* Example: " -v, --verbose Viele Sachen ausgeben" */
|
|
indent += 10;
|
|
if ( *opts[ordtbl[0]].description != '@'
|
|
&& !(opts[ordtbl[0]].flags
|
|
& (ARGPARSE_OPT_VERBATIM|ARGPARSE_OPT_HEADER)))
|
|
writestrings (0, "Options:", "\n", NULL);
|
|
for (i=0; i < nopts; i++ )
|
|
{
|
|
s = map_fixed_string (_( opts[ordtbl[i]].description ));
|
|
if ( s && *s== '@' && !s[1] ) /* Hide this line. */
|
|
continue;
|
|
if ( s && (opts[ordtbl[i]].flags & ARGPARSE_OPT_HEADER))
|
|
{
|
|
/* We delay printing until we have found one real output
|
|
* line. This avoids having a header above an empty
|
|
* section. */
|
|
last_header = s;
|
|
continue;
|
|
}
|
|
if (last_header)
|
|
{
|
|
if (*last_header)
|
|
writestrings (0, "\n", last_header, ":\n", NULL);
|
|
last_header = NULL;
|
|
}
|
|
if ( s && (opts[ordtbl[i]].flags & ARGPARSE_OPT_VERBATIM))
|
|
{
|
|
writestrings (0, s, NULL);
|
|
continue;
|
|
}
|
|
if ( s && *s == '@' ) /* Unindented legacy comment only line. */
|
|
{
|
|
for (s++; *s; s++ )
|
|
{
|
|
if ( *s == '\n' )
|
|
{
|
|
if( s[1] )
|
|
writestrings (0, "\n", NULL);
|
|
}
|
|
else
|
|
{
|
|
tmp[0] = *s;
|
|
tmp[1] = 0;
|
|
writestrings (0, tmp, NULL);
|
|
}
|
|
}
|
|
writestrings (0, "\n", NULL);
|
|
continue;
|
|
}
|
|
|
|
j = 3;
|
|
if ( opts[ordtbl[i]].short_opt < 256 )
|
|
{
|
|
tmp[0] = opts[ordtbl[i]].short_opt;
|
|
tmp[1] = 0;
|
|
writestrings (0, " -", tmp, NULL );
|
|
if ( !opts[ordtbl[i]].long_opt )
|
|
{
|
|
if (s && *s == '|' )
|
|
{
|
|
writestrings (0, " ", NULL); j++;
|
|
for (s++ ; *s && *s != '|'; s++, j++ )
|
|
{
|
|
tmp[0] = *s;
|
|
tmp[1] = 0;
|
|
writestrings (0, tmp, NULL);
|
|
}
|
|
if ( *s )
|
|
s++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
writestrings (0, " ", NULL);
|
|
if ( opts[ordtbl[i]].long_opt )
|
|
{
|
|
tmp[0] = opts[ordtbl[i]].short_opt < 256?',':' ';
|
|
tmp[1] = 0;
|
|
j += writestrings (0, tmp, " --", opts[ordtbl[i]].long_opt, NULL);
|
|
if (s && *s == '|' )
|
|
{
|
|
if ( *++s != '=' )
|
|
{
|
|
writestrings (0, " ", NULL);
|
|
j++;
|
|
}
|
|
for ( ; *s && *s != '|'; s++, j++ )
|
|
{
|
|
tmp[0] = *s;
|
|
tmp[1] = 0;
|
|
writestrings (0, tmp, NULL);
|
|
}
|
|
if ( *s )
|
|
s++;
|
|
}
|
|
writestrings (0, " ", NULL);
|
|
j += 3;
|
|
}
|
|
for (;j < indent; j++ )
|
|
writestrings (0, " ", NULL);
|
|
if ( s )
|
|
{
|
|
if ( *s && j > indent )
|
|
{
|
|
writestrings (0, "\n", NULL);
|
|
for (j=0;j < indent; j++ )
|
|
writestrings (0, " ", NULL);
|
|
}
|
|
for (; *s; s++ )
|
|
{
|
|
if ( *s == '\n' )
|
|
{
|
|
if ( s[1] )
|
|
{
|
|
writestrings (0, "\n", NULL);
|
|
for (j=0; j < indent; j++ )
|
|
writestrings (0, " ", NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
tmp[0] = *s;
|
|
tmp[1] = 0;
|
|
writestrings (0, tmp, NULL);
|
|
}
|
|
}
|
|
}
|
|
writestrings (0, "\n", NULL);
|
|
}
|
|
if ( (flags & ARGPARSE_FLAG_ONEDASH) )
|
|
writestrings (0, "\n(A single dash may be used "
|
|
"instead of the double ones)\n", NULL);
|
|
}
|
|
if ( (s=strusage(19)) )
|
|
{
|
|
writestrings (0, "\n", NULL);
|
|
writestrings (0, s, NULL);
|
|
}
|
|
|
|
leave:
|
|
flushstrings (0);
|
|
xfree (ordtbl);
|
|
}
|
|
|
|
|
|
static void
|
|
show_version ()
|
|
{
|
|
const char *s;
|
|
int i;
|
|
|
|
/* Version line. */
|
|
writestrings (0, strusage (11), NULL);
|
|
if ((s=strusage (12)))
|
|
writestrings (0, " (", s, ")", NULL);
|
|
writestrings (0, " ", strusage (13), "\n", NULL);
|
|
/* Additional version lines. */
|
|
for (i=20; i < 30; i++)
|
|
if ((s=strusage (i)))
|
|
writestrings (0, s, "\n", NULL);
|
|
/* Copyright string. */
|
|
if ((s=strusage (14)))
|
|
writestrings (0, s, "\n", NULL);
|
|
/* Licence string. */
|
|
if( (s=strusage (10)) )
|
|
writestrings (0, s, "\n", NULL);
|
|
/* Copying conditions. */
|
|
if ( (s=strusage(15)) )
|
|
writestrings (0, s, NULL);
|
|
/* Thanks. */
|
|
if ((s=strusage(18)))
|
|
writestrings (0, s, NULL);
|
|
/* Additional program info. */
|
|
for (i=30; i < 40; i++ )
|
|
if ( (s=strusage (i)) )
|
|
writestrings (0, s, NULL);
|
|
flushstrings (0);
|
|
}
|
|
|
|
|
|
/* Print the table of options with flags etc. */
|
|
static void
|
|
dump_option_table (gnupg_argparse_t *arg)
|
|
{
|
|
opttable_t *opts;
|
|
unsigned int nopts;
|
|
const char *s;
|
|
char tmp[50];
|
|
unsigned int *ordtbl = NULL;
|
|
int i;
|
|
|
|
opts = arg->internal->opts;
|
|
nopts = arg->internal->nopts;
|
|
if (!nopts)
|
|
return;
|
|
|
|
ordtbl = xtrycalloc (nopts, sizeof *ordtbl);
|
|
if (!ordtbl)
|
|
{
|
|
writestrings (1, "\nOoops: Out of memory whilst dumping the table.\n",
|
|
NULL);
|
|
flushstrings (1);
|
|
my_exit (arg, 2);
|
|
}
|
|
for (i=0; i < nopts; i++ )
|
|
ordtbl[i] = opts[i].ordinal;
|
|
qsort (ordtbl, nopts, sizeof *ordtbl, cmp_ordtbl);
|
|
for (i=0; i < nopts; i++ )
|
|
{
|
|
if (!opts[ordtbl[i]].long_opt)
|
|
continue;
|
|
writestrings (0, opts[ordtbl[i]].long_opt, ":", NULL);
|
|
snprintf (tmp, sizeof tmp, "%u:%u:",
|
|
opts[ordtbl[i]].short_opt,
|
|
opts[ordtbl[i]].flags);
|
|
writestrings (0, tmp, NULL);
|
|
s = opts[ordtbl[i]].description;
|
|
if (s)
|
|
{
|
|
for (; *s; s++)
|
|
{
|
|
if (*s == '%' || *s == ':' || *s == '\n')
|
|
snprintf (tmp, sizeof tmp, "%%%02X", *s);
|
|
else
|
|
{
|
|
tmp[0] = *s;
|
|
tmp[1] = 0;
|
|
}
|
|
writestrings (0, tmp, NULL);
|
|
}
|
|
}
|
|
writestrings (0, ":\n", NULL);
|
|
}
|
|
|
|
flushstrings (0);
|
|
xfree (ordtbl);
|
|
my_exit (arg, 0);
|
|
}
|
|
|
|
|
|
|
|
/* Level
|
|
* 0: Print copyright string to stderr
|
|
* 1: Print a short usage hint to stderr and terminate
|
|
* 2: Print a long usage hint to stdout and terminate
|
|
* 8: Return NULL for UTF-8 or string with the native charset.
|
|
* 9: Return the SPDX License tag.
|
|
* 10: Return license info string
|
|
* 11: Return the name of the program
|
|
* 12: Return optional name of package which includes this program.
|
|
* 13: version string
|
|
* 14: copyright string
|
|
* 15: Short copying conditions (with LFs)
|
|
* 16: Long copying conditions (with LFs)
|
|
* 17: Optional printable OS name
|
|
* 18: Optional thanks list (with LFs)
|
|
* 19: Bug report info
|
|
*20..29: Additional lib version strings.
|
|
*30..39: Additional program info (with LFs)
|
|
* 40: short usage note (with LF)
|
|
* 41: long usage note (with LF)
|
|
* 42: Flag string:
|
|
* First char is '1':
|
|
* The short usage notes needs to be printed
|
|
* before the long usage note.
|
|
*/
|
|
const char *
|
|
strusage( int level )
|
|
{
|
|
const char *p = strusage_handler? strusage_handler(level) : NULL;
|
|
const char *tmp;
|
|
|
|
if ( p )
|
|
return map_static_macro_string (p);
|
|
|
|
switch ( level )
|
|
{
|
|
|
|
case 8: break; /* Default to utf-8. */
|
|
case 9: p = "GPL-3.0-or-later"; break;
|
|
case 10:
|
|
tmp = strusage (9);
|
|
if (tmp && !strcmp (tmp, "LGPL-2.1-or-later"))
|
|
p = ("License GNU LGPL-2.1-or-later <https://gnu.org/licenses/>");
|
|
else /* Default to GPLv3+. */
|
|
p =("License GNU GPL-3.0-or-later <https://gnu.org/licenses/gpl.html>");
|
|
break;
|
|
case 11: p = "foo"; break;
|
|
case 13: p = "0.0"; break;
|
|
case 14: p = GNUPG_DEF_COPYRIGHT_LINE; break;
|
|
case 15: p =
|
|
"This is free software: you are free to change and redistribute it.\n"
|
|
"There is NO WARRANTY, to the extent permitted by law.\n";
|
|
break;
|
|
case 16:
|
|
tmp = strusage (9);
|
|
if (tmp && !strcmp (tmp, "LGPL-2.1-or-later"))
|
|
p =
|
|
"This is free software; you can redistribute it and/or modify\n"
|
|
"it under the terms of the GNU Lesser General Public License as\n"
|
|
"published by the Free Software Foundation; either version 2.1 of\n"
|
|
"the License, or (at your option) any later version.\n\n"
|
|
"It is distributed in the hope that it will be useful,\n"
|
|
"but WITHOUT ANY WARRANTY; without even the implied warranty of\n"
|
|
"MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n"
|
|
"GNU Lesser General Public License for more details.\n\n"
|
|
"You should have received a copy of the GNU Lesser General Public License\n"
|
|
"along with this software. If not, see <https://gnu.org/licenses/>.\n";
|
|
else /* Default */
|
|
p =
|
|
"This is free software; you can redistribute it and/or modify\n"
|
|
"it under the terms of the GNU General Public License as published by\n"
|
|
"the Free Software Foundation; either version 3 of the License, or\n"
|
|
"(at your option) any later version.\n\n"
|
|
"It is distributed in the hope that it will be useful,\n"
|
|
"but WITHOUT ANY WARRANTY; without even the implied warranty of\n"
|
|
"MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n"
|
|
"GNU General Public License for more details.\n\n"
|
|
"You should have received a copy of the GNU General Public License\n"
|
|
"along with this software. If not, see <https://gnu.org/licenses/>.\n";
|
|
break;
|
|
case 40: /* short and long usage */
|
|
case 41: p = ""; break;
|
|
}
|
|
|
|
return p;
|
|
}
|
|
|
|
|
|
/* Set the usage handler. This function is basically a constructor. */
|
|
void
|
|
set_strusage ( const char *(*f)( int ) )
|
|
{
|
|
strusage_handler = f;
|
|
}
|
|
|
|
#endif /* USE_INTERNAL_ARGPARSE */
|
|
|
|
|
|
void
|
|
usage (int level)
|
|
{
|
|
const char *p;
|
|
|
|
if (!level)
|
|
{
|
|
writestrings (1, strusage(11), " ", strusage(13), "; ",
|
|
strusage (14), "\n", NULL);
|
|
flushstrings (1);
|
|
}
|
|
else if (level == 1)
|
|
{
|
|
p = strusage (40);
|
|
writestrings (1, p, NULL);
|
|
if (*p && p[strlen(p)] != '\n')
|
|
writestrings (1, "\n", NULL);
|
|
exit (2);
|
|
}
|
|
else if (level == 2)
|
|
{
|
|
p = strusage (42);
|
|
if (p && *p == '1')
|
|
{
|
|
p = strusage (40);
|
|
writestrings (1, p, NULL);
|
|
if (*p && p[strlen(p)] != '\n')
|
|
writestrings (1, "\n", NULL);
|
|
}
|
|
writestrings (0, strusage(41), "\n", NULL);
|
|
exit (0);
|
|
}
|
|
}
|