/* This file is part of GNUnet. Copyright (C) 2012 GNUnet e.V. GNUnet is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. GNUnet 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 Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . SPDX-License-Identifier: AGPL3.0-or-later */ /** * @file util/gnunet-config.c * @brief tool to access and manipulate GNUnet configuration files * @author Christian Grothoff */ #include "platform.h" #include "gnunet_util_lib.h" /** * Name of the section */ static char *section; /** * Name of the option */ static char *option; /** * Value to set */ static char *value; /** * Backend to check if the respective plugin is * loadable. NULL if no check is to be performed. * The value is the "basename" of the plugin to load. */ static char *backend_check; /** * Treat option as a filename. */ static int is_filename; /** * Whether to show the sections. */ static int list_sections; /** * Return value from 'main'. */ static int global_ret; /** * Should we generate a configuration file that is clean and * only contains the deltas to the defaults? */ static int rewrite; /** * Print each option in a given section. * * @param cls closure * @param section name of the section * @param option name of the option * @param value value of the option */ static void print_option (void *cls, const char *section, const char *option, const char *value) { const struct GNUNET_CONFIGURATION_Handle *cfg = cls; (void) section; if (is_filename) { char *value_fn; char *fn; GNUNET_assert (GNUNET_OK == GNUNET_CONFIGURATION_get_value_filename (cfg, section, option, &value_fn)); fn = GNUNET_STRINGS_filename_expand (value_fn); if (NULL == fn) fn = value_fn; else GNUNET_free (value_fn); fprintf (stdout, "%s = %s\n", option, fn); GNUNET_free (fn); } else { fprintf (stdout, "%s = %s\n", option, value); } } /** * Print out given section name. * * @param cls unused * @param section a section in the configuration file */ static void print_section_name (void *cls, const char *section) { (void) cls; fprintf (stdout, "%s\n", section); } /** * Main function that will be run by the scheduler. * * @param cls closure * @param args remaining command-line arguments * @param cfgfile name of the configuration file used (for saving, * can be NULL!) * @param cfg configuration */ static void run (void *cls, char *const *args, const char *cfgfile, const struct GNUNET_CONFIGURATION_Handle *cfg) { struct GNUNET_CONFIGURATION_Handle *out = NULL; struct GNUNET_CONFIGURATION_Handle *diff = NULL; char *cfg_fn; (void) cls; (void) args; if (NULL != backend_check) { char *name; GNUNET_asprintf (&name, "libgnunet_plugin_%s", backend_check); global_ret = (GNUNET_OK == GNUNET_PLUGIN_test (name)) ? 0 : 77; GNUNET_free (name); return; } if (rewrite) { struct GNUNET_CONFIGURATION_Handle *def; def = GNUNET_CONFIGURATION_create (); if (GNUNET_OK != GNUNET_CONFIGURATION_load (def, NULL)) { fprintf (stderr, _ ("failed to load configuration defaults")); global_ret = 1; return; } diff = GNUNET_CONFIGURATION_get_diff (def, cfg); cfg = diff; } if (((! rewrite) && (NULL == section)) || list_sections) { if (! list_sections) { fprintf (stderr, _ ("%s or %s argument is required\n"), "--section", "--list-sections"); global_ret = 1; } else { fprintf (stderr, _ ("The following sections are available:\n")); GNUNET_CONFIGURATION_iterate_sections (cfg, &print_section_name, NULL); } goto cleanup; } if ((NULL != section) && (NULL == value)) { if (NULL == option) { GNUNET_CONFIGURATION_iterate_section_values (cfg, section, &print_option, (void *) cfg); } else { if (is_filename) { if (GNUNET_OK != GNUNET_CONFIGURATION_get_value_filename (cfg, section, option, &value)) { GNUNET_log_config_missing (GNUNET_ERROR_TYPE_ERROR, section, option); global_ret = 3; goto cleanup; } } else { if (GNUNET_OK != GNUNET_CONFIGURATION_get_value_string (cfg, section, option, &value)) { GNUNET_log_config_missing (GNUNET_ERROR_TYPE_ERROR, section, option); global_ret = 3; goto cleanup; } } fprintf (stdout, "%s\n", value); } } else if (NULL != section) { if (NULL == option) { fprintf (stderr, _ ("--option argument required to set value\n")); global_ret = 1; goto cleanup; } out = GNUNET_CONFIGURATION_dup (cfg); GNUNET_CONFIGURATION_set_value_string (out, section, option, value); } cfg_fn = NULL; if (NULL == cfgfile) { const char *xdg = getenv ("XDG_CONFIG_HOME"); if (NULL != xdg) GNUNET_asprintf (&cfg_fn, "%s%s%s", xdg, DIR_SEPARATOR_STR, GNUNET_OS_project_data_get ()->config_file); else cfg_fn = GNUNET_strdup (GNUNET_OS_project_data_get ()->user_config_file); cfgfile = cfg_fn; } if ((NULL != diff) || (NULL != out)) { if (GNUNET_OK != GNUNET_CONFIGURATION_write ((NULL == out) ? diff : out, cfgfile)) global_ret = 2; } GNUNET_free_non_null (cfg_fn); if (NULL != out) GNUNET_CONFIGURATION_destroy (out); cleanup: if (NULL != diff) GNUNET_CONFIGURATION_destroy (diff); } /** * Program to manipulate configuration files. * * @param argc number of arguments from the command line * @param argv command line arguments * @return 0 ok, 1 on error */ int main (int argc, char *const *argv) { struct GNUNET_GETOPT_CommandLineOption options[] = { GNUNET_GETOPT_option_flag ( 'f', "filename", gettext_noop ("interpret option value as a filename (with $-expansion)"), &is_filename), GNUNET_GETOPT_option_exclusive (GNUNET_GETOPT_option_string ( 'b', "supported-backend", "BACKEND", gettext_noop ( "test if the current installation supports the specified BACKEND"), &backend_check)), GNUNET_GETOPT_option_string ('s', "section", "SECTION", gettext_noop ( "name of the section to access"), §ion), GNUNET_GETOPT_option_string ('o', "option", "OPTION", gettext_noop ("name of the option to access"), &option), GNUNET_GETOPT_option_string ('V', "value", "VALUE", gettext_noop ("value to set"), &value), GNUNET_GETOPT_option_flag ('S', "list-sections", gettext_noop ( "print available configuration sections"), &list_sections), GNUNET_GETOPT_option_flag ( 'w', "rewrite", gettext_noop ( "write configuration file that only contains delta to defaults"), &rewrite), GNUNET_GETOPT_OPTION_END }; int ret; if (GNUNET_OK != GNUNET_STRINGS_get_utf8_args (argc, argv, &argc, &argv)) return 2; ret = GNUNET_PROGRAM_run (argc, argv, "gnunet-config [OPTIONS]", gettext_noop ("Manipulate GNUnet configuration files"), options, &run, NULL); GNUNET_free ((void *) argv); if (GNUNET_OK == ret) return global_ret; return ret; } /* end of gnunet-config.c */