diff --git a/changelogs/unreleased/config-help-env-list.md b/changelogs/unreleased/config-help-env-list.md new file mode 100644 index 0000000000000000000000000000000000000000..b9f39d3292fdabf3ef80fe35458c2005dd27bf27 --- /dev/null +++ b/changelogs/unreleased/config-help-env-list.md @@ -0,0 +1,3 @@ +## feature/config + +* Added `--help-env-list` CLI option (gh-8862). diff --git a/src/box/lua/config/init.lua b/src/box/lua/config/init.lua index c83ca07d0068f20ac5f66164a771188109044fba..d239dbaa5172be6f334fac86409d17ac3d6246ee 100644 --- a/src/box/lua/config/init.lua +++ b/src/box/lua/config/init.lua @@ -356,6 +356,11 @@ function methods._startup(self, instance_name, config_file) self:_set_status_based_on_alerts() end +function methods._print_env_list(self) + local env_source = require('internal.config.source.env').new() + io.stdout:write(env_source:_env_list()) +end + function methods.get(self, path) selfcheck(self, 'get') if self._status == 'uninitialized' then diff --git a/src/box/lua/config/source/env.lua b/src/box/lua/config/source/env.lua index 735fb17aa25c156727e8a7e45c0d792ef0b5db42..cc297283ae9e6f7a582116ab828667ff71a1fa65 100644 --- a/src/box/lua/config/source/env.lua +++ b/src/box/lua/config/source/env.lua @@ -1,4 +1,5 @@ local schema = require('internal.config.utils.schema') +local tabulate = require('internal.config.utils.tabulate') local instance_config = require('internal.config.instance_config') local methods = {} @@ -14,6 +15,62 @@ function methods._env_var_name(self, path_in_schema) return env_var_name end +function methods._env_list(self) + local rows = {} + + local ce = 'Community Edition' + local ee = 'Enterprise Edition' + + -- A header of the table. + table.insert(rows, { + 'ENVIRONMENT VARIABLE', + 'TYPE', + 'DEFAULT', + 'AVAILABILITY', + }) + table.insert(rows, tabulate.SPACER) + + -- Environment variables that duplicate CLI options. + table.insert(rows, { + 'TT_INSTANCE_NAME', + 'string', + 'N/A', + ce, + }) + table.insert(rows, { + 'TT_CONFIG', + 'string', + 'nil', + ce, + }) + table.insert(rows, tabulate.SPACER) + + -- Collect the options from the schema and sort them + -- lexicographically. + local options = instance_config:pairs():totable() + table.sort(options, function(a, b) + return table.concat(a.path, '_') < table.concat(b.path, '_') + end) + + -- Transform the schema nodes (options) into the environment + -- variables description. + for _, w in ipairs(options) do + local default = w.schema.default + if default == nil and type(default) == 'cdata' then + default = 'box.NULL' + end + + table.insert(rows, { + self._env_var_name(self, w.path), + w.schema.type, + tostring(default), + w.schema.computed.annotations.enterprise_edition and ee or ce, + }) + end + + return tabulate.encode(rows) +end + -- Gather most actual config values. function methods.sync(self, _config_module, _iconfig) local values = {} diff --git a/src/lua/init.c b/src/lua/init.c index 1a0927779e0dded3646a266acdcf58a7bf724d2d..94de2dac91c90bc98a7e14abae1d7f1e7a38d573 100644 --- a/src/lua/init.c +++ b/src/lua/init.c @@ -1042,6 +1042,7 @@ run_script_f(va_list ap) bool interactive = opt_mask & O_INTERACTIVE; bool bytecode = opt_mask & O_BYTECODE; bool debugging = opt_mask & O_DEBUGGING; + bool help_env_list = opt_mask & O_HELP_ENV_LIST; /* * An error is returned via an external diag. A caller * can't use fiber_join(), because the script can call @@ -1108,6 +1109,23 @@ run_script_f(va_list ap) } } + /* + * Show a list of environment variables that are + * considered by tarantool and exit. + */ + if (help_env_list) { + /* require('config'):_print_env_list() */ + if (lua_require_lib(L, "config") != 0) + goto error; + lua_pushstring(L, "_print_env_list"); + lua_gettable(L, -2); + lua_pushvalue(L, -2); + if (luaT_call(L, 1, 0) != 0) + goto error; + lua_settop(L, 0); + goto end; + } + /* * Start an instance using an externally provided * configuration if the --name option is passed. diff --git a/src/lua/init.h b/src/lua/init.h index 6eaf28b2ad4c1755b4ddafba769381e927ac944e..28553c732f2c35f3ecaaae66e025fb0e98550ff5 100644 --- a/src/lua/init.h +++ b/src/lua/init.h @@ -55,6 +55,7 @@ struct instance_state { #define O_BYTECODE 0x2 #define O_DEBUGGING 0x4 #define O_EXECUTE 0x8 +#define O_HELP_ENV_LIST 0x10 /** * Create tarantool_L and initialize built-in Lua modules. diff --git a/src/main.cc b/src/main.cc index 5c75871359155f7206c17e42a2426a69667c5059..3ff9600f1aa9b8af2fa0cb584de4eac946a5d100 100644 --- a/src/main.cc +++ b/src/main.cc @@ -632,6 +632,7 @@ print_help(FILE *stream, const char *program) " %s [OPTIONS] [SCRIPT [ARGS]]\n\n" "Options:\n\n" " -h, --help display this help and exit\n" + " --help-env-list display env variables taken into account\n" " -v, --version print program version and exit\n" " -c, --config PATH set a path to yaml config file as 'PATH'\n" " -n, --name INSTANCE set an instance name as 'INSTANCE'\n" @@ -677,6 +678,18 @@ main(int argc, char **argv) {"version", no_argument, 0, 'v'}, {"config", required_argument, 0, 'c'}, {"name", required_argument, 0, 'n'}, + /* + * Use 'E' character as an indicator of the + * --help-env-list option. + * + * Note: there is no -E short option, see the + * `opts` variable below. + * + * An arbitrary `int` value that is not used for + * another option may be used here. Feel free to + * change it if -E short option should be added. + */ + {"help-env-list", no_argument, 0, 'E'}, {NULL, 0, 0, 0}, }; static const char *opts = "+hVvb::ij:e:l:dc:n:"; @@ -706,6 +719,9 @@ main(int argc, char **argv) case 'h': print_help(stdout, basename(argv[0])); return 0; + case 'E': + opt_mask |= O_HELP_ENV_LIST; + break; case 'i': /* Force interactive mode */ opt_mask |= O_INTERACTIVE; @@ -805,7 +821,23 @@ main(int argc, char **argv) if (strlen(tarantool_path) < strlen(tarantool_bin)) panic("executable path is trimmed"); - if (script == NULL && (opt_mask & (O_INTERACTIVE | O_EXECUTE)) == 0 && + /* + * The idea of the check below is that we can't run + * tarantool without any action: there should be at least + * one. + * + * There are the following actions. + * + * * Print a help message or a version (--help, --version; + * these actions are handled above). + * * Print environment variables list (--help-env-list). + * * Start an instance (with a name and a config). + * * Run a script pointed by a positional argument or + * written using -e option. + * * Start interactive REPL (-i). + */ + uint32_t action_opt_mask = O_INTERACTIVE | O_EXECUTE | O_HELP_ENV_LIST; + if (script == NULL && (opt_mask & action_opt_mask) == 0 && instance.name == NULL) { static const char misuse_msg[] = "Invalid usage: " "please either provide a Lua script name\n" diff --git a/test/box-py/args.result b/test/box-py/args.result index 9baa9cd4d2277e854c25a730f25613dc2d7325a3..d6c7ae1b997aeb3ff8fda0857528edf0e6e98bc1 100644 --- a/test/box-py/args.result +++ b/test/box-py/args.result @@ -24,6 +24,7 @@ Usage: Options: -h, --help display this help and exit + --help-env-list display env variables taken into account -v, --version print program version and exit -c, --config PATH set a path to yaml config file as 'PATH' -n, --name INSTANCE set an instance name as 'INSTANCE' @@ -65,6 +66,7 @@ Usage: Options: -h, --help display this help and exit + --help-env-list display env variables taken into account -v, --version print program version and exit -c, --config PATH set a path to yaml config file as 'PATH' -n, --name INSTANCE set an instance name as 'INSTANCE' diff --git a/test/config-luatest/cli_test.lua b/test/config-luatest/cli_test.lua new file mode 100644 index 0000000000000000000000000000000000000000..ffd00ae96268c66375268ea5ee0ca538139fc54b --- /dev/null +++ b/test/config-luatest/cli_test.lua @@ -0,0 +1,72 @@ +local t = require('luatest') +local justrun = require('test.justrun') + +local g = t.group() + +g.test_help_env_list = function() + local res = justrun.tarantool('.', {}, {'--help-env-list'}, {nojson = true}) + t.assert_equals(res.exit_code, 0) + + -- Pick up several well-known env variables to verify the idea + -- of the listing, but at the same time, don't list all the + -- options in the test case. + local cases = { + -- Verify that the env variables that duplicates CLI + -- options are present. + { + name = 'TT_INSTANCE_NAME', + type = 'string', + default = 'N/A', + availability = 'Community Edition', + }, + { + name = 'TT_CONFIG', + type = 'string', + default = 'nil', + availability = 'Community Edition', + }, + -- An env variable that accepts a numeric value and has + -- explicit default value. + { + name = 'TT_MEMTX_MEMORY', + type = 'integer', + default = '268435456', + availability = 'Community Edition', + }, + -- An env variable that has box.NULL default. + { + name = 'TT_DATABASE_MODE', + type = 'string', + default = 'box.NULL', + availability = 'Community Edition', + }, + -- An option with a template default. + { + name = 'TT_CONSOLE_SOCKET', + type = 'string', + default = '{{ instance_name }}.control', + availability = 'Community Edition', + }, + -- An Enterprise Edition option and, at the same time, + -- an option with non-string type. + { + name = 'TT_CONFIG_ETCD_ENDPOINTS', + type = 'array', + default = 'nil', + availability = 'Enterprise Edition', + }, + -- Yet another Enterprise Edition option. + { + name = 'TT_WAL_EXT_NEW', + type = 'boolean', + default = 'nil', + availability = 'Enterprise Edition', + }, + } + + for _, case in ipairs(cases) do + local needle = ('%s.*%s.*%s.*%s'):format(case.name, case.type, + case.default, case.availability) + t.assert(res.stdout:find(needle), case.name) + end +end