summaryrefslogtreecommitdiffstats
path: root/plugins/sudoers/testsudoers.c
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/sudoers/testsudoers.c')
-rw-r--r--plugins/sudoers/testsudoers.c782
1 files changed, 782 insertions, 0 deletions
diff --git a/plugins/sudoers/testsudoers.c b/plugins/sudoers/testsudoers.c
new file mode 100644
index 0000000..2ebd7b4
--- /dev/null
+++ b/plugins/sudoers/testsudoers.c
@@ -0,0 +1,782 @@
+/*
+ * SPDX-License-Identifier: ISC
+ *
+ * Copyright (c) 1996, 1998-2005, 2007-2023
+ * Todd C. Miller <Todd.Miller@sudo.ws>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ *
+ * Sponsored in part by the Defense Advanced Research Projects
+ * Agency (DARPA) and Air Force Research Laboratory, Air Force
+ * Materiel Command, USAF, under agreement number F39502-99-1-0512.
+ */
+
+/*
+ * This is an open source non-commercial project. Dear PVS-Studio, please check it.
+ * PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
+ */
+
+#include <config.h>
+
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+#include <stdio.h>
+#include <stdlib.h>
+#ifdef HAVE_STDBOOL_H
+# include <stdbool.h>
+#else
+# include <compat/stdbool.h>
+#endif /* HAVE_STDBOOL_H */
+#include <string.h>
+#ifdef HAVE_STRINGS_H
+# include <strings.h>
+#endif /* HAVE_STRINGS_H */
+#include <unistd.h>
+#include <errno.h>
+
+#include <testsudoers_pwutil.h>
+#include <toke.h>
+#include <tsgetgrpw.h>
+#include <sudoers.h>
+#include <interfaces.h>
+#include <sudo_conf.h>
+#include <sudo_lbuf.h>
+#include <gram.h>
+
+#ifndef YYDEBUG
+# define YYDEBUG 0
+#endif
+
+enum sudoers_formats {
+ format_ldif,
+ format_sudoers
+};
+
+/*
+ * Function Prototypes
+ */
+static void dump_sudoers(void);
+static void set_runaspw(struct sudoers_context *ctx, const char *);
+static void set_runasgr(struct sudoers_context *ctx, const char *);
+static int testsudoers_error(const char * restrict buf);
+static int testsudoers_output(const char * restrict buf);
+sudo_noreturn static void usage(void);
+static void cb_lookup(const struct sudoers_parse_tree *parse_tree, const struct userspec *us, int user_match, const struct privilege *priv, int host_match, const struct cmndspec *cs, int date_match, int runas_match, int cmnd_match, void *closure);
+static int testsudoers_query(struct sudoers_context *ctx, const struct sudo_nss *nss, struct passwd *pw);
+
+/*
+ * Globals
+ */
+static const char *orig_cmnd;
+static char *runas_group, *runas_user;
+
+#if defined(SUDO_DEVEL) && defined(__OpenBSD__)
+extern char *malloc_options;
+#endif
+
+sudo_dso_public int main(int argc, char *argv[]);
+
+int
+main(int argc, char *argv[])
+{
+ struct sudoers_context test_ctx = SUDOERS_CONTEXT_INITIALIZER;
+ struct sudo_nss_list snl = TAILQ_HEAD_INITIALIZER(snl);
+ enum sudoers_formats input_format = format_sudoers;
+ struct sudo_nss testsudoers_nss;
+ char *p, *grfile, *pwfile, *shells;
+ const char *host = NULL;
+ const char *errstr;
+ int ch, dflag, exitcode = EXIT_FAILURE;
+ unsigned int validated;
+ int status = FOUND;
+ int pwflag = 0;
+ char cwdbuf[PATH_MAX];
+ time_t now;
+ id_t id;
+ debug_decl(main, SUDOERS_DEBUG_MAIN);
+
+#if defined(SUDO_DEVEL) && defined(__OpenBSD__)
+ malloc_options = "S";
+#endif
+#if YYDEBUG
+ sudoersdebug = 1;
+#endif
+
+ initprogname(argc > 0 ? argv[0] : "testsudoers");
+
+ if (!sudoers_initlocale(setlocale(LC_ALL, ""), def_sudoers_locale))
+ sudo_fatalx(U_("%s: %s"), __func__, U_("unable to allocate memory"));
+ sudo_warn_set_locale_func(sudoers_warn_setlocale);
+ bindtextdomain("sudoers", LOCALEDIR); /* XXX - should have own domain */
+ textdomain("sudoers");
+ time(&now);
+
+ /* Initialize the debug subsystem. */
+ if (sudo_conf_read(NULL, SUDO_CONF_DEBUG) == -1)
+ goto done;
+ if (!sudoers_debug_register(getprogname(), sudo_conf_debug_files(getprogname())))
+ goto done;
+
+ dflag = 0;
+ grfile = pwfile = shells = NULL;
+ test_ctx.mode = MODE_RUN;
+ while ((ch = getopt(argc, argv, "+D:dg:G:h:i:L:lP:p:R:S:T:tu:U:v")) != -1) {
+ switch (ch) {
+ case 'D':
+ test_ctx.runas.cwd = optarg;
+ break;
+ case 'd':
+ dflag = 1;
+ break;
+ case 'G':
+ id = sudo_strtoid(optarg, &errstr);
+ if (errstr != NULL)
+ sudo_fatalx("group-ID %s: %s", optarg, errstr);
+ test_ctx.parser_conf.sudoers_gid = (gid_t)id;
+ break;
+ case 'g':
+ runas_group = optarg;
+ SET(test_ctx.settings.flags, RUNAS_GROUP_SPECIFIED);
+ break;
+ case 'h':
+ host = optarg;
+ break;
+ case 'i':
+ if (strcasecmp(optarg, "ldif") == 0) {
+ input_format = format_ldif;
+ } else if (strcasecmp(optarg, "sudoers") == 0) {
+ input_format = format_sudoers;
+ } else {
+ sudo_warnx(U_("unsupported input format %s"), optarg);
+ usage();
+ }
+ break;
+ case 'L':
+ test_ctx.runas.list_pw = sudo_getpwnam(optarg);
+ if (test_ctx.runas.list_pw == NULL) {
+ sudo_warnx(U_("unknown user %s"), optarg);
+ usage();
+ }
+ FALLTHROUGH;
+ case 'l':
+ if (test_ctx.mode != MODE_RUN) {
+ sudo_warnx(
+ "only one of the -l or -v flags may be specified");
+ usage();
+ }
+ test_ctx.mode = MODE_LIST;
+ pwflag = I_LISTPW;
+ orig_cmnd = "list";
+ break;
+ case 'p':
+ pwfile = optarg;
+ break;
+ case 'P':
+ grfile = optarg;
+ break;
+ case 'S':
+ shells = optarg;
+ break;
+ case 'T':
+ now = parse_gentime(optarg);
+ if (now == -1)
+ sudo_fatalx("invalid time: %s", optarg);
+ break;
+ case 'R':
+ test_ctx.runas.chroot = optarg;
+ break;
+ case 't':
+ trace_print = testsudoers_error;
+ break;
+ case 'U':
+ id = sudo_strtoid(optarg, &errstr);
+ if (errstr != NULL)
+ sudo_fatalx("user-ID %s: %s", optarg, errstr);
+ test_ctx.parser_conf.sudoers_uid = (uid_t)id;
+ break;
+ case 'u':
+ runas_user = optarg;
+ SET(test_ctx.settings.flags, RUNAS_USER_SPECIFIED);
+ break;
+ case 'v':
+ if (test_ctx.mode != MODE_RUN) {
+ sudo_warnx(
+ "only one of the -l or -v flags may be specified");
+ usage();
+ }
+ test_ctx.mode = MODE_VALIDATE;
+ pwflag = I_VERIFYPW;
+ orig_cmnd = "validate";
+ break;
+ default:
+ usage();
+ /* NOTREACHED */
+ }
+ }
+ argc -= optind;
+ argv += optind;
+
+ if (grfile != NULL || pwfile != NULL || shells != NULL) {
+ /* Set group/passwd/shells file and init the cache. */
+ if (grfile)
+ testsudoers_setgrfile(grfile);
+ if (pwfile)
+ testsudoers_setpwfile(pwfile);
+ if (shells)
+ testsudoers_setshellfile(shells);
+
+ /* Use custom passwd/group backend. */
+ sudo_pwutil_set_backend(testsudoers_make_pwitem,
+ testsudoers_make_gritem, testsudoers_make_gidlist_item,
+ testsudoers_make_grlist_item, testsudoers_valid_shell);
+ }
+
+ if (argc < 2) {
+ /* No command or user specified. */
+ if (dflag) {
+ orig_cmnd = "true";
+ } else if (pwflag == 0) {
+ usage();
+ }
+ test_ctx.user.name = strdup(argc ? *argv++ : "root");
+ if (test_ctx.user.name == NULL) {
+ sudo_fatalx(U_("%s: %s"), __func__,
+ U_("unable to allocate memory"));
+ }
+ argc = 0;
+ } else {
+ if (argc > 2 && test_ctx.mode == MODE_LIST)
+ test_ctx.mode = MODE_CHECK;
+ test_ctx.user.name = strdup(*argv++);
+ if (test_ctx.user.name == NULL) {
+ sudo_fatalx(U_("%s: %s"), __func__,
+ U_("unable to allocate memory"));
+ }
+ argc--;
+ if (orig_cmnd == NULL) {
+ orig_cmnd = *argv++;
+ argc--;
+ }
+ }
+ test_ctx.user.cmnd = strdup(orig_cmnd);
+ if (test_ctx.user.cmnd == NULL)
+ sudo_fatalx(U_("%s: %s"), __func__, U_("unable to allocate memory"));
+ test_ctx.user.cmnd_base = sudo_basename(test_ctx.user.cmnd);
+
+ if (getcwd(cwdbuf, sizeof(cwdbuf)) == NULL)
+ strlcpy(cwdbuf, "/", sizeof(cwdbuf));
+ test_ctx.user.cwd = strdup(cwdbuf);
+ if (test_ctx.user.cwd == NULL)
+ sudo_fatalx(U_("%s: %s"), __func__, U_("unable to allocate memory"));
+
+ if ((test_ctx.user.pw = sudo_getpwnam(test_ctx.user.name)) == NULL)
+ sudo_fatalx(U_("unknown user %s"), test_ctx.user.name);
+ test_ctx.user.uid = test_ctx.user.pw->pw_uid;
+ test_ctx.user.gid = test_ctx.user.pw->pw_gid;
+
+ if (!sudoers_sethost(&test_ctx, host, NULL))
+ goto done;
+
+ /* Fill in test_ctx.user.cmnd_args from argv. */
+ if (argc > 0) {
+ size_t n, size = 0;
+ char *cp;
+ int i;
+
+ for (i = 0; i < argc; i++)
+ size += strlen(argv[i]) + 1;
+
+ if ((test_ctx.user.cmnd_args = malloc(size)) == NULL)
+ sudo_fatalx(U_("%s: %s"), __func__, U_("unable to allocate memory"));
+ for (cp = test_ctx.user.cmnd_args, i = 0; i < argc; i++) {
+ n = strlcpy(cp, argv[i], size - (size_t)(cp - test_ctx.user.cmnd_args));
+ if (n >= size - (size_t)(cp - test_ctx.user.cmnd_args))
+ sudo_fatalx(U_("internal error, %s overflow"), getprogname());
+ cp += n;
+ *cp++ = ' ';
+ }
+ *--cp = '\0';
+ }
+
+ /* Initialize default values. */
+ if (!init_defaults())
+ sudo_fatalx("%s", U_("unable to initialize sudoers default values"));
+
+ /* Set group_plugin callback. */
+ sudo_defs_table[I_GROUP_PLUGIN].callback = cb_group_plugin;
+
+ /* Set runas callback. */
+ sudo_defs_table[I_RUNAS_DEFAULT].callback = cb_runas_default;
+
+ /* Set locale callback. */
+ sudo_defs_table[I_SUDOERS_LOCALE].callback = sudoers_locale_callback;
+
+ /* Load ip addr/mask for each interface. */
+ if (get_net_ifs(&p) > 0) {
+ if (!set_interfaces(p))
+ sudo_fatal("%s", U_("unable to parse network address list"));
+ free(p);
+ }
+
+ /* Initialize the parser and set sudoers filename to "sudoers". */
+ test_ctx.parser_conf.strict = true;
+ test_ctx.parser_conf.verbose = 2;
+ init_parser(&test_ctx, "sudoers");
+
+ /*
+ * Set runas passwd/group entries based on command line or sudoers.
+ * Note that if runas_group was specified without runas_user we
+ * run the command as the invoking user.
+ */
+ if (runas_group != NULL) {
+ set_runasgr(&test_ctx, runas_group);
+ set_runaspw(&test_ctx, runas_user ? runas_user : test_ctx.user.name);
+ } else
+ set_runaspw(&test_ctx, runas_user ? runas_user : def_runas_default);
+
+ /* Parse the policy file. */
+ sudoers_setlocale(SUDOERS_LOCALE_SUDOERS, NULL);
+ switch (input_format) {
+ case format_ldif:
+ if (!sudoers_parse_ldif(&parsed_policy, stdin, NULL, true)) {
+ (void) puts("Parse error in LDIF");
+ parse_error = true;
+ }
+ break;
+ case format_sudoers:
+ if (sudoersparse() != 0)
+ parse_error = true;
+ break;
+ default:
+ sudo_fatalx("error: unhandled input %d", input_format);
+ }
+ if (!update_defaults(&test_ctx, &parsed_policy, NULL, SETDEF_ALL, false))
+ parse_error = true;
+
+ if (!parse_error)
+ (void) puts("Parses OK");
+
+ if (dflag) {
+ (void) putchar('\n');
+ dump_sudoers();
+ if (argc < 2) {
+ exitcode = parse_error ? 1 : 0;
+ goto done;
+ }
+ }
+
+ /* Fake up a minimal sudo nss list with the parsed policy. */
+ TAILQ_INSERT_TAIL(&snl, &testsudoers_nss, entries);
+ testsudoers_nss.query = testsudoers_query;
+ testsudoers_nss.parse_tree = &parsed_policy;
+
+ printf("\nEntries for user %s:\n", test_ctx.user.name);
+ validated = sudoers_lookup(&snl, &test_ctx, now, cb_lookup, NULL,
+ &status, pwflag);
+
+ /* Validate user-specified chroot or cwd (if any) and runas user shell. */
+ if (ISSET(validated, VALIDATE_SUCCESS)) {
+ if (!user_shell_valid(test_ctx.runas.pw)) {
+ printf(U_("\nInvalid shell for user %s: %s\n"),
+ test_ctx.runas.pw->pw_name, test_ctx.runas.pw->pw_shell);
+ CLR(validated, VALIDATE_SUCCESS);
+ SET(validated, VALIDATE_FAILURE);
+ }
+ if (check_user_runchroot(test_ctx.runas.chroot) != true) {
+ printf("\nUser %s is not allowed to change root directory to %s\n",
+ test_ctx.user.name, test_ctx.runas.chroot);
+ CLR(validated, VALIDATE_SUCCESS);
+ SET(validated, VALIDATE_FAILURE);
+ }
+ if (check_user_runcwd(test_ctx.runas.cwd) != true) {
+ printf("\nUser %s is not allowed to change directory to %s\n",
+ test_ctx.user.name, test_ctx.runas.cwd);
+ CLR(validated, VALIDATE_SUCCESS);
+ SET(validated, VALIDATE_FAILURE);
+ }
+ }
+ if (def_authenticate) {
+ puts(U_("\nPassword required"));
+ }
+
+ /*
+ * Exit codes:
+ * 0 - parsed OK and command matched.
+ * 1 - parse error
+ * 2 - command not matched
+ * 3 - command denied
+ */
+ if (parse_error || ISSET(validated, VALIDATE_ERROR)) {
+ puts(U_("\nParse error"));
+ exitcode = 1;
+ } else if (ISSET(validated, VALIDATE_SUCCESS)) {
+ puts(U_("\nCommand allowed"));
+ exitcode = 0;
+ } else if (ISSET(validated, VALIDATE_FAILURE)) {
+ puts(U_("\nCommand denied"));
+ exitcode = 3;
+ } else {
+ puts(U_("\nCommand unmatched"));
+ exitcode = 2;
+ }
+
+done:
+ sudoers_ctx_free(&test_ctx);
+ sudo_freepwcache();
+ sudo_freegrcache();
+ sudo_debug_exit_int(__func__, __FILE__, __LINE__, sudo_debug_subsys, exitcode);
+ return exitcode;
+}
+
+static void
+set_runaspw(struct sudoers_context *ctx, const char *user)
+{
+ struct passwd *pw = NULL;
+ debug_decl(set_runaspw, SUDOERS_DEBUG_UTIL);
+
+ if (*user == '#') {
+ const char *errstr;
+ uid_t uid = sudo_strtoid(user + 1, &errstr);
+ if (errstr == NULL) {
+ if ((pw = sudo_getpwuid(uid)) == NULL)
+ pw = sudo_fakepwnam(user, ctx->user.gid);
+ }
+ }
+ if (pw == NULL) {
+ if ((pw = sudo_getpwnam(user)) == NULL)
+ sudo_fatalx(U_("unknown user %s"), user);
+ }
+ if (ctx->runas.pw != NULL)
+ sudo_pw_delref(ctx->runas.pw);
+ ctx->runas.pw = pw;
+ debug_return;
+}
+
+static void
+set_runasgr(struct sudoers_context *ctx, const char *group)
+{
+ struct group *gr = NULL;
+ debug_decl(set_runasgr, SUDOERS_DEBUG_UTIL);
+
+ if (*group == '#') {
+ const char *errstr;
+ gid_t gid = sudo_strtoid(group + 1, &errstr);
+ if (errstr == NULL) {
+ if ((gr = sudo_getgrgid(gid)) == NULL)
+ gr = sudo_fakegrnam(group);
+ }
+ }
+ if (gr == NULL) {
+ if ((gr = sudo_getgrnam(group)) == NULL)
+ sudo_fatalx(U_("unknown group %s"), group);
+ }
+ if (ctx->runas.gr != NULL)
+ sudo_gr_delref(ctx->runas.gr);
+ ctx->runas.gr = gr;
+ debug_return;
+}
+
+bool
+cb_log_input(struct sudoers_context *ctx, const char *file,
+ int line, int column, const union sudo_defs_val *sd_un, int op)
+{
+ return true;
+}
+
+bool
+cb_log_output(struct sudoers_context *ctx, const char *file,
+ int line, int column, const union sudo_defs_val *sd_un, int op)
+{
+ return true;
+}
+
+/*
+ * Callback for runas_default sudoers setting.
+ */
+bool
+cb_runas_default(struct sudoers_context *ctx, const char *file,
+ int line, int column, const union sudo_defs_val *sd_un, int op)
+{
+ /* Only reset runaspw if user didn't specify one. */
+ if (!runas_user && !runas_group)
+ set_runaspw(ctx, sd_un->str);
+ return true;
+}
+
+bool
+sudo_nss_can_continue(const struct sudo_nss *nss, int match)
+{
+ return true;
+}
+
+void
+sudo_setspent(void)
+{
+ return;
+}
+
+void
+sudo_endspent(void)
+{
+ return;
+}
+
+FILE *
+open_sudoers(const char *file, char **outfile, bool doedit, bool *keepopen)
+{
+ struct stat sb;
+ FILE *fp = NULL;
+ const char *base;
+ int error, fd;
+ debug_decl(open_sudoers, SUDOERS_DEBUG_UTIL);
+
+ /* Report errors using the basename for consistent test output. */
+ base = sudo_basename(file);
+ fd = sudo_secure_open_file(file, sudoers_file_uid(), sudoers_file_gid(),
+ &sb, &error);
+ if (fd != -1) {
+ if ((fp = fdopen(fd, "r")) == NULL) {
+ sudo_warn("unable to open %s", base);
+ close(fd);
+ }
+ } else {
+ switch (error) {
+ case SUDO_PATH_MISSING:
+ sudo_warn("unable to open %s", base);
+ break;
+ case SUDO_PATH_BAD_TYPE:
+ sudo_warnx("%s is not a regular file", base);
+ break;
+ case SUDO_PATH_WRONG_OWNER:
+ sudo_warnx("%s should be owned by uid %u",
+ base, (unsigned int) sudoers_file_uid());
+ break;
+ case SUDO_PATH_WORLD_WRITABLE:
+ sudo_warnx("%s is world writable", base);
+ break;
+ case SUDO_PATH_GROUP_WRITABLE:
+ sudo_warnx("%s should be owned by gid %u",
+ base, (unsigned int) sudoers_file_gid());
+ break;
+ default:
+ sudo_warnx("%s: internal error, unexpected error %d",
+ __func__, error);
+ break;
+ }
+ }
+
+ debug_return_ptr(fp);
+}
+
+bool
+init_envtables(void)
+{
+ return(true);
+}
+
+bool
+set_perms(const struct sudoers_context *ctx, int perm)
+{
+ return true;
+}
+
+bool
+restore_perms(void)
+{
+ return true;
+}
+
+void
+init_eventlog_config(void)
+{
+ return;
+}
+
+bool
+pivot_root(const char *new_root, struct sudoers_pivot *state)
+{
+ return true;
+}
+
+bool
+unpivot_root(struct sudoers_pivot *state)
+{
+ return true;
+}
+
+int
+set_cmnd_path(struct sudoers_context *ctx, const char *runchroot)
+{
+ /* Reallocate test_ctx.user.cmnd to catch bugs in command_matches(). */
+ char *new_cmnd = strdup(orig_cmnd);
+ if (new_cmnd == NULL)
+ return NOT_FOUND_ERROR;
+ free(ctx->user.cmnd);
+ ctx->user.cmnd = new_cmnd;
+ return FOUND;
+}
+
+static void
+cb_lookup(const struct sudoers_parse_tree *parse_tree,
+ const struct userspec *us, int user_match, const struct privilege *priv,
+ int host_match, const struct cmndspec *cs, int date_match, int runas_match,
+ int cmnd_match, void *closure)
+{
+ static const struct privilege *prev_priv;
+ struct sudo_lbuf lbuf;
+
+ /* Only output info for the selected user. */
+ if (user_match != ALLOW) {
+ prev_priv = NULL;
+ return;
+ }
+
+ if (priv != prev_priv) {
+ /* No word wrap on output. */
+ sudo_lbuf_init(&lbuf, testsudoers_output, 0, NULL, 0);
+ sudo_lbuf_append(&lbuf, "\n");
+ sudoers_format_privilege(&lbuf, &parsed_policy, priv, false);
+ sudo_lbuf_print(&lbuf);
+ sudo_lbuf_destroy(&lbuf);
+
+ printf("\thost %s\n", host_match == ALLOW ? "allowed" :
+ host_match == DENY ? "denied" : "unmatched");
+ }
+
+ if (host_match == ALLOW) {
+ if (date_match != UNSPEC)
+ printf("\tdate %s\n", date_match == ALLOW ? "allowed" : "denied");
+ if (date_match != DENY) {
+ printf("\trunas %s\n", runas_match == ALLOW ? "allowed" :
+ runas_match == DENY ? "denied" : "unmatched");
+ if (runas_match == ALLOW) {
+ printf("\tcmnd %s\n", cmnd_match == ALLOW ? "allowed" :
+ cmnd_match == DENY ? "denied" : "unmatched");
+ }
+ }
+ }
+
+ prev_priv = priv;
+}
+
+static int
+testsudoers_query(struct sudoers_context *ctx, const struct sudo_nss *nss,
+ struct passwd *pw)
+{
+ /* Nothing to do. */
+ return 0;
+}
+
+static bool
+print_defaults(struct sudo_lbuf *lbuf)
+{
+ struct defaults *def, *next;
+ debug_decl(print_defaults, SUDOERS_DEBUG_UTIL);
+
+ TAILQ_FOREACH_SAFE(def, &parsed_policy.defaults, entries, next)
+ sudoers_format_default_line(lbuf, &parsed_policy, def, &next, false);
+
+ debug_return_bool(!sudo_lbuf_error(lbuf));
+}
+
+static int
+print_alias(struct sudoers_parse_tree *parse_tree, struct alias *a, void *v)
+{
+ struct sudo_lbuf *lbuf = v;
+ struct member *m;
+ debug_decl(print_alias, SUDOERS_DEBUG_UTIL);
+
+ sudo_lbuf_append(lbuf, "%s %s = ", alias_type_to_string(a->type),
+ a->name);
+ TAILQ_FOREACH(m, &a->members, entries) {
+ if (m != TAILQ_FIRST(&a->members))
+ sudo_lbuf_append(lbuf, ", ");
+ sudoers_format_member(lbuf, parse_tree, m, NULL, UNSPEC);
+ }
+ sudo_lbuf_append(lbuf, "\n");
+
+ debug_return_int(sudo_lbuf_error(lbuf) ? -1 : 0);
+}
+
+static bool
+print_aliases(struct sudo_lbuf *lbuf)
+{
+ debug_decl(print_aliases, SUDOERS_DEBUG_UTIL);
+
+ alias_apply(&parsed_policy, print_alias, lbuf);
+
+ debug_return_bool(!sudo_lbuf_error(lbuf));
+}
+
+static void
+dump_sudoers(void)
+{
+ struct sudo_lbuf lbuf;
+ debug_decl(dump_sudoers, SUDOERS_DEBUG_UTIL);
+
+ /* No word wrap on output. */
+ sudo_lbuf_init(&lbuf, testsudoers_output, 0, NULL, 0);
+
+ /* Print Defaults */
+ if (!print_defaults(&lbuf))
+ goto done;
+ if (lbuf.len > 0) {
+ sudo_lbuf_print(&lbuf);
+ sudo_lbuf_append(&lbuf, "\n");
+ }
+
+ /* Print Aliases */
+ if (!print_aliases(&lbuf))
+ goto done;
+ if (lbuf.len > 1) {
+ sudo_lbuf_print(&lbuf);
+ sudo_lbuf_append(&lbuf, "\n");
+ }
+
+ /* Print User_Specs */
+ if (!sudoers_format_userspecs(&lbuf, &parsed_policy, NULL, false, true))
+ goto done;
+ if (lbuf.len > 1) {
+ sudo_lbuf_print(&lbuf);
+ }
+
+done:
+ if (sudo_lbuf_error(&lbuf)) {
+ if (errno == ENOMEM)
+ sudo_fatalx(U_("%s: %s"), __func__, U_("unable to allocate memory"));
+ }
+ sudo_lbuf_destroy(&lbuf);
+
+ debug_return;
+}
+
+static int
+testsudoers_output(const char * restrict buf)
+{
+ return fputs(buf, stdout);
+}
+
+static int
+testsudoers_error(const char *restrict buf)
+{
+ return fputs(buf, stderr);
+}
+
+sudo_noreturn static void
+usage(void)
+{
+ (void) fprintf(stderr, "usage: %s [-dltv] [-G sudoers_gid] [-g group] [-h host] [-i input_format] [-L list_user] [-P grfile] [-p pwfile] [-S shells] [-U sudoers_uid] [-u user] <user> <command> [args]\n", getprogname());
+ exit(EXIT_FAILURE);
+}