NetworkManager/clients/cli/nmcli.c

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

1052 lines
36 KiB
C
Raw Normal View History

/* SPDX-License-Identifier: GPL-2.0-or-later */
/*
2010-02-25 09:52:30 -08:00
* Jiri Klimes <jklimes@redhat.com>
* Copyright (C) 2010 - 2018 Red Hat, Inc.
2010-02-25 09:52:30 -08:00
*/
#include "libnm/nm-default-client.h"
2010-02-25 09:52:30 -08:00
#include "nmcli.h"
2010-02-25 09:52:30 -08:00
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
#include <termios.h>
#include <unistd.h>
2010-02-25 09:52:30 -08:00
#include <locale.h>
#include <glib-unix.h>
#include <readline/readline.h>
#include <readline/history.h>
2010-02-25 09:52:30 -08:00
#include "nm-client-utils.h"
#include "polkit-agent.h"
2010-02-25 09:52:30 -08:00
#include "utils.h"
#include "common.h"
2010-02-25 09:52:30 -08:00
#include "connections.h"
#include "devices.h"
2016-07-27 16:24:30 +02:00
#include "settings.h"
2010-02-25 09:52:30 -08:00
#if defined(NM_DIST_VERSION)
#define NMCLI_VERSION NM_DIST_VERSION
#else
#define NMCLI_VERSION VERSION
#endif
2010-02-25 09:52:30 -08:00
#define _NMC_COLOR_PALETTE_INIT() \
{ \
.ansi_seq = { \
[NM_META_COLOR_CONNECTION_ACTIVATED] = "32", \
[NM_META_COLOR_CONNECTION_ACTIVATING] = "33", \
[NM_META_COLOR_CONNECTION_DISCONNECTING] = "31", \
[NM_META_COLOR_CONNECTION_INVISIBLE] = "2", \
[NM_META_COLOR_CONNECTION_EXTERNAL] = "32;2", \
[NM_META_COLOR_CONNECTIVITY_FULL] = "32", \
[NM_META_COLOR_CONNECTIVITY_LIMITED] = "33", \
[NM_META_COLOR_CONNECTIVITY_NONE] = "31", \
[NM_META_COLOR_CONNECTIVITY_PORTAL] = "33", \
[NM_META_COLOR_DEVICE_ACTIVATED] = "32", \
[NM_META_COLOR_DEVICE_ACTIVATING] = "33", \
[NM_META_COLOR_DEVICE_DISCONNECTED] = "31", \
[NM_META_COLOR_DEVICE_FIRMWARE_MISSING] = "31", \
[NM_META_COLOR_DEVICE_PLUGIN_MISSING] = "31", \
[NM_META_COLOR_DEVICE_UNAVAILABLE] = "2", \
[NM_META_COLOR_DEVICE_DISABLED] = "31", \
[NM_META_COLOR_DEVICE_EXTERNAL] = "32;2", \
[NM_META_COLOR_MANAGER_RUNNING] = "32", \
[NM_META_COLOR_MANAGER_STARTING] = "33", \
[NM_META_COLOR_MANAGER_STOPPED] = "31", \
[NM_META_COLOR_PERMISSION_AUTH] = "33", \
[NM_META_COLOR_PERMISSION_NO] = "31", \
[NM_META_COLOR_PERMISSION_YES] = "32", \
[NM_META_COLOR_STATE_ASLEEP] = "31", \
[NM_META_COLOR_STATE_CONNECTED_GLOBAL] = "32", \
[NM_META_COLOR_STATE_CONNECTED_LOCAL] = "32", \
[NM_META_COLOR_STATE_CONNECTED_SITE] = "32", \
[NM_META_COLOR_STATE_CONNECTING] = "33", \
[NM_META_COLOR_STATE_DISCONNECTED] = "31", \
[NM_META_COLOR_STATE_DISCONNECTING] = "33", \
[NM_META_COLOR_WIFI_SIGNAL_EXCELLENT] = "32", \
[NM_META_COLOR_WIFI_SIGNAL_FAIR] = "35", \
[NM_META_COLOR_WIFI_SIGNAL_GOOD] = "33", \
[NM_META_COLOR_WIFI_SIGNAL_POOR] = "36", \
[NM_META_COLOR_WIFI_SIGNAL_UNKNOWN] = "2", \
[NM_META_COLOR_ENABLED] = "32", \
[NM_META_COLOR_DISABLED] = "31", \
}, \
}
static NmCli nm_cli = {
.client = NULL,
.return_value = NMC_RESULT_SUCCESS,
.timeout = -1,
.secret_agent = NULL,
.pwds_hash = NULL,
.pk_listener = NULL,
.should_wait = 0,
.nowait_flag = TRUE,
.nmc_config.print_output = NMC_PRINT_NORMAL,
.nmc_config.multiline_output = FALSE,
.mode_specified = FALSE,
.nmc_config.escape_values = TRUE,
.required_fields = NULL,
.ask = FALSE,
.complete = FALSE,
.nmc_config.show_secrets = FALSE,
.nmc_config.in_editor = FALSE,
.nmc_config.palette = _NMC_COLOR_PALETTE_INIT(),
.editor_status_line = FALSE,
.editor_save_confirmation = TRUE,
};
2010-02-25 09:52:30 -08:00
const NmCli *const nm_cli_global_readline = &nm_cli;
const NmCli *const nmc_meta_environment_arg = &nm_cli;
/*****************************************************************************/
2010-02-25 09:52:30 -08:00
typedef struct {
NmCli *nmc;
int argc;
char **argv;
} ArgsInfo;
/* --- Global variables --- */
GMainLoop * loop = NULL;
struct termios termios_orig;
2010-02-25 09:52:30 -08:00
NM_CACHED_QUARK_FCN("nmcli-error-quark", nmcli_error_quark);
2016-07-27 16:24:30 +02:00
static void
complete_field_setting(GHashTable *h, NMMetaSettingType setting_type)
{
const NMMetaSettingInfoEditor *setting_info = &nm_meta_setting_infos_editor[setting_type];
guint i;
for (i = 0; i < setting_info->properties_num; i++) {
g_hash_table_add(h,
g_strdup_printf("%s.%s",
setting_info->general->setting_name,
setting_info->properties[i]->property_name));
}
}
static void
complete_field(GHashTable *h, const NmcMetaGenericInfo *const *field)
2016-07-27 16:24:30 +02:00
{
int i;
for (i = 0; field[i]; i++)
g_hash_table_add(h, g_strdup(field[i]->name));
2016-07-27 16:24:30 +02:00
}
static void
complete_one(gpointer key, gpointer value, gpointer user_data)
{
const char **option_with_value = user_data;
const char * option = option_with_value[0];
const char * prefix = option_with_value[1];
2016-07-27 16:24:30 +02:00
const char * name = key;
const char * last;
2016-07-27 16:24:30 +02:00
last = strrchr(prefix, ',');
if (last)
last++;
else
last = prefix;
if ((!*last && !strchr(name, '.')) || matches(last, name)) {
if (option != prefix) {
/* value prefix was not a standalone argument,
* it was part of --option=<value> argument.
* Repeat the part leading to "=". */
g_print("%s=", option);
}
2016-07-27 16:24:30 +02:00
g_print("%.*s%s%s\n",
(int) (last - prefix),
prefix,
name,
strcmp(last, name) == 0 ? "," : "");
}
}
static void
complete_fields(const char *option, const char *prefix)
2016-07-27 16:24:30 +02:00
{
guint i;
2016-07-27 16:24:30 +02:00
GHashTable *h;
const char *option_with_value[2] = {option, prefix};
h = g_hash_table_new_full(nm_str_hash, g_str_equal, g_free, NULL);
complete_field(h, metagen_ip4_config);
2018-04-26 17:43:13 +02:00
complete_field(h, metagen_dhcp_config);
complete_field(h, metagen_ip6_config);
cli: rework printing of `nmcli connection` for multiple active connections The output of `nmcli connection show` contains also information about whether the profile is currently active, for example the device and the current (activation) state. Even when a profile can be activated only once (without supporting mutiple activations at the same time), there are moments when a connection is activating and still deactivating on another device. NetworkManager ensures in the case with single activations that a profile cannot be in state "activated" multiple times. But that doesn't mean, that one profile cannot have multiple active connection which reference it. That was already handled wrongly before, because `nmcli connection show` would only search the first matching active-connection. That is, it would arbitrarily pick an active connection in case there were multiple and only show activation state about one. Furthermore, we will soon also add the possibility, that a profile can be active multiple times (at the same time). Especially then, we need to extend the output format to show all the devices on which the profile is currently active. Rework printing the connection list to use nmc_print(), and fix various issues. - as discussed, a profile may have multiple active connections at each time. There are only two possibilities: if a profile is active multiple times, show a line for each activation, or otherwise, show the information about multiple activations combined in one line, e.g. by printing "DEVICE eth0,eth1". This patch, does the former. We will now print a line for each active connection, to show all the devices and activation states in multiple lines. Yes, this may result in the same profile being printed multiple times. That is a change in behavior, and inconvenient if you do something like for UUID in $(nmcli connection show | awk '{print$2}'); do ... However, above is anyway wrong because it assumes that there are no spaces in the connection name. The proper way to do this is like for UUID in $(nmcli -g UUID connection show); do ... In the latter case, whenever a user selects a subset of fields (--fields, --get) which don't print information about active connections, these multiple lines are combined. So, above still works as expected, never returning duplicate UUIDs. - if a user has no permissions to see a connection, we previously would print "<invisible> $NAME". No longer do this but just print the ID was it is reported by the active-connection. If the goal of this was to prevent users from accidentally access the non-existing connection by $NAME, then this was a bad solution, because a script would instead try to access "<invisible> $NAME". This is now solved better by hiding the active connection if the user selects "-g NAME". - the --order option now sorts according to how the fields are shown. For example, with --terse mode, it will evaluate type "802-11-wireless" but with pretty mode it will consider "wifi". This may change the ordering in which connections are shown. Also, for sorting the name, we use g_utf8_collate() because it's unicode.
2018-04-25 10:00:40 +02:00
complete_field(h, metagen_con_show);
complete_field(h, metagen_con_active_general);
complete_field(h, metagen_con_active_vpn);
complete_field(h, nmc_fields_con_active_details_groups);
2018-04-26 20:28:02 +02:00
complete_field(h, metagen_device_status);
complete_field(h, metagen_device_detail_general);
complete_field(h, metagen_device_detail_connections);
complete_field(h, metagen_device_detail_capabilities);
complete_field(h, metagen_device_detail_wired_properties);
complete_field(h, metagen_device_detail_wifi_properties);
complete_field(h, metagen_device_detail_wimax_properties);
complete_field(h, nmc_fields_dev_wifi_list);
complete_field(h, nmc_fields_dev_wimax_list);
complete_field(h, nmc_fields_dev_show_master_prop);
complete_field(h, nmc_fields_dev_show_team_prop);
complete_field(h, nmc_fields_dev_show_vlan_prop);
complete_field(h, nmc_fields_dev_show_bluetooth);
complete_field(h, nmc_fields_dev_show_sections);
complete_field(h, nmc_fields_dev_lldp_list);
for (i = 0; i < _NM_META_SETTING_TYPE_NUM; i++)
complete_field_setting(h, i);
g_hash_table_foreach(h, complete_one, (gpointer) &option_with_value[0]);
2016-07-27 16:24:30 +02:00
g_hash_table_destroy(h);
}
static void
complete_option_with_value(const char *option, const char *prefix, ...)
{
va_list args;
const char *candidate;
va_start(args, prefix);
while ((candidate = va_arg(args, const char *))) {
if (!*prefix || matches(prefix, candidate)) {
if (option != prefix) {
/* value prefix was not a standalone argument,
* it was part of --option=<value> argument.
* Repeat the part leading to "=". */
g_print("%s=", option);
}
g_print("%s\n", candidate);
}
}
va_end(args);
}
2010-02-25 09:52:30 -08:00
static void
usage(void)
2010-02-25 09:52:30 -08:00
{
g_printerr(_(
"Usage: nmcli [OPTIONS] OBJECT { COMMAND | help }\n"
"\n"
"OPTIONS\n"
" -a, --ask ask for missing parameters\n"
" -c, --colors auto|yes|no whether to use colors in output\n"
" -e, --escape yes|no escape columns separators in values\n"
" -f, --fields <field,...>|all|common specify fields to output\n"
" -g, --get-values <field,...>|all|common shortcut for -m tabular -t -f\n"
" -h, --help print this help\n"
" -m, --mode tabular|multiline output mode\n"
" -o, --overview overview mode\n"
" -p, --pretty pretty output\n"
" -s, --show-secrets allow displaying passwords\n"
" -t, --terse terse output\n"
" -v, --version show program version\n"
" -w, --wait <seconds> set timeout waiting for finishing operations\n"
"\n"
"OBJECT\n"
" g[eneral] NetworkManager's general status and operations\n"
" n[etworking] overall networking control\n"
" r[adio] NetworkManager radio switches\n"
" c[onnection] NetworkManager's connections\n"
" d[evice] devices managed by NetworkManager\n"
" a[gent] NetworkManager secret agent or polkit agent\n"
" m[onitor] monitor NetworkManager changes\n"
"\n"));
2010-02-25 09:52:30 -08:00
}
static gboolean
matches_arg(NmCli *nmc, int *argc, const char *const **argv, const char *pattern, char **arg)
{
gs_free char *opt_free = NULL;
const char * opt = (*argv)[0];
gs_free char *arg_tmp = NULL;
const char * s;
nm_assert(opt);
nm_assert(opt[0] == '-');
nm_assert(!arg || !*arg);
if (nmc->return_value != NMC_RESULT_SUCCESS) {
/* Don't process further matches if there has been an error. */
return FALSE;
}
if (opt[1] == '-') {
/* We know one '-' was already seen by the caller.
* Skip it if there's a second one*/
opt++;
}
if (arg) {
/* If there's a "=" separator, replace it with NUL so that matches()
* works and consider the part after it to be the argument's value. */
s = strchr(opt, '=');
if (s) {
opt = nm_strndup_a(300, opt, s - opt, &opt_free);
arg_tmp = g_strdup(&s[1]);
}
}
if (!matches(opt, pattern))
return FALSE;
if (arg) {
if (arg_tmp)
*arg = g_steal_pointer(&arg_tmp);
else {
/* We need a value, but the option didn't contain a "=<value>" part.
* Proceed to the next argument. */
if (*argc <= 1) {
g_string_printf(nmc->return_text,
_("Error: missing argument for '%s' option."),
opt);
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
}
(*argc)--;
(*argv)++;
*arg = g_strdup(*argv[0]);
}
}
return TRUE;
}
/*************************************************************************************/
typedef enum {
NMC_USE_COLOR_AUTO,
NMC_USE_COLOR_YES,
NMC_USE_COLOR_NO,
} NmcColorOption;
static char *
check_colors_construct_filename(const char *base_dir,
const char *name,
const char *term,
const char *type)
{
return g_strdup_printf("%s/terminal-colors.d/%s%s%s%s%s",
base_dir,
name ? name : "",
term ? "@" : "",
term ? term : "",
(name || term) ? "." : "",
type);
}
static NmcColorOption
check_colors_check_enabled_one_file(const char *base_dir, const char *name, const char *term)
{
gs_free char *filename_e = NULL;
gs_free char *filename_d = NULL;
filename_e = check_colors_construct_filename(base_dir, name, term, "enable");
if (g_file_test(filename_e, G_FILE_TEST_EXISTS))
return NMC_USE_COLOR_YES;
filename_d = check_colors_construct_filename(base_dir, name, term, "disable");
if (g_file_test(filename_d, G_FILE_TEST_EXISTS))
return NMC_USE_COLOR_NO;
return NMC_USE_COLOR_AUTO;
}
static char *
check_colors_check_palette_one_file(const char *base_dir, const char *name, const char *term)
{
static const char *const extensions[] = {
"scheme",
"schem",
};
guint i;
for (i = 0; i < G_N_ELEMENTS(extensions); i++) {
gs_free char *filename = NULL;
char * contents;
filename = check_colors_construct_filename(base_dir, name, term, extensions[i]);
if (g_file_get_contents(filename, &contents, NULL, NULL))
return contents;
}
return NULL;
}
static gboolean
check_colors_check_enabled(const char *base_dir_1,
const char *base_dir_2,
const char *name,
const char *term)
{
int i;
if (term && strchr(term, '/'))
term = NULL;
#define CHECK_AND_RETURN(cmd) \
G_STMT_START \
{ \
NmcColorOption _color_option; \
\
_color_option = (cmd); \
if (_color_option != NMC_USE_COLOR_AUTO) \
return _color_option == NMC_USE_COLOR_YES; \
} \
G_STMT_END
for (i = 0; i < 2; i++) {
const char *base_dir = (i == 0 ? base_dir_1 : base_dir_2);
if (!base_dir)
continue;
if (name && term)
CHECK_AND_RETURN(check_colors_check_enabled_one_file(base_dir, name, term));
if (name)
CHECK_AND_RETURN(check_colors_check_enabled_one_file(base_dir, name, NULL));
if (term)
CHECK_AND_RETURN(check_colors_check_enabled_one_file(base_dir, NULL, term));
if (TRUE)
CHECK_AND_RETURN(check_colors_check_enabled_one_file(base_dir, NULL, NULL));
}
#undef CHECK_AND_RETURN
return TRUE;
}
static char *
check_colors_check_palette(const char *base_dir_1,
const char *base_dir_2,
const char *name,
const char *term)
{
int i;
if (term && strchr(term, '/'))
term = NULL;
#define CHECK_AND_RETURN(cmd) \
G_STMT_START \
{ \
char *_palette; \
\
_palette = (cmd); \
if (_palette) \
return _palette; \
} \
G_STMT_END
for (i = 0; i < 2; i++) {
const char *base_dir = (i == 0 ? base_dir_1 : base_dir_2);
if (!base_dir)
continue;
if (name && term)
CHECK_AND_RETURN(check_colors_check_palette_one_file(base_dir, name, term));
if (name)
CHECK_AND_RETURN(check_colors_check_palette_one_file(base_dir, name, NULL));
if (term)
CHECK_AND_RETURN(check_colors_check_palette_one_file(base_dir, NULL, term));
if (TRUE)
CHECK_AND_RETURN(check_colors_check_palette_one_file(base_dir, NULL, NULL));
}
#undef CHECK_AND_RETURN
return NULL;
}
static gboolean
check_colors(NmcColorOption color_option, char **out_palette_str)
{
const char * base_dir_1, *base_dir_2;
const char *const NAME = "nmcli";
const char * term;
*out_palette_str = NULL;
if (!NM_IN_SET(color_option, NMC_USE_COLOR_AUTO, NMC_USE_COLOR_YES)) {
/* nothing to do. Colors are disabled. */
return FALSE;
}
if (color_option == NMC_USE_COLOR_AUTO && g_getenv("NO_COLOR")) {
/* https://no-color.org/ */
return FALSE;
}
term = g_getenv("TERM");
if (color_option == NMC_USE_COLOR_AUTO) {
if (nm_streq0(term, "dumb") || !isatty(STDOUT_FILENO))
return FALSE;
}
base_dir_1 = g_get_user_config_dir();
base_dir_2 = "" SYSCONFDIR;
if (base_dir_1) {
if (nm_streq(base_dir_1, base_dir_2) || !g_file_test(base_dir_1, G_FILE_TEST_EXISTS))
base_dir_1 = NULL;
}
if (!g_file_test(base_dir_2, G_FILE_TEST_EXISTS))
base_dir_2 = NULL;
if (color_option == NMC_USE_COLOR_AUTO
&& !check_colors_check_enabled(base_dir_1, base_dir_2, NAME, term))
return FALSE;
*out_palette_str = check_colors_check_palette(base_dir_1, base_dir_2, NAME, term);
return TRUE;
}
static NM_UTILS_STRING_TABLE_LOOKUP_DEFINE(
_resolve_color_alias,
const char *,
{ nm_assert(name); },
{ return NULL; },
{"black", "30"},
{"blink", "5"},
{"blue", "34"},
{"bold", "1"},
{"brown", "33"},
{"cyan", "36"},
{"darkgray", "90"},
{"gray", "37"},
{"green", "32"},
{"halfbright", "2"},
{"lightblue", "94"},
{"lightcyan", "96"},
{"lightgray", "97"},
{"lightgreen", "92"},
{"lightmagenta", "95"},
{"lightred", "91"},
{"magenta", "35"},
{"red", "31"},
{"reset", "0"},
{"reverse", "7"},
{"underscore", "4"},
{"white", "1;37"},
{"yellow", "33" /* well, yellow */}, );
static NM_UTILS_STRING_TABLE_LOOKUP_DEFINE(
_nm_meta_color_from_name,
NMMetaColor,
{ nm_assert(name); },
{ return NM_META_COLOR_NONE; },
{"connection-activated", NM_META_COLOR_CONNECTION_ACTIVATED},
{"connection-activating", NM_META_COLOR_CONNECTION_ACTIVATING},
{"connection-disconnecting", NM_META_COLOR_CONNECTION_DISCONNECTING},
{"connection-external", NM_META_COLOR_CONNECTION_EXTERNAL},
{"connection-invisible", NM_META_COLOR_CONNECTION_INVISIBLE},
{"connection-unknown", NM_META_COLOR_CONNECTION_UNKNOWN},
{"connectivity-full", NM_META_COLOR_CONNECTIVITY_FULL},
{"connectivity-limited", NM_META_COLOR_CONNECTIVITY_LIMITED},
{"connectivity-none", NM_META_COLOR_CONNECTIVITY_NONE},
{"connectivity-portal", NM_META_COLOR_CONNECTIVITY_PORTAL},
{"connectivity-unknown", NM_META_COLOR_CONNECTIVITY_UNKNOWN},
{"device-activated", NM_META_COLOR_DEVICE_ACTIVATED},
{"device-activating", NM_META_COLOR_DEVICE_ACTIVATING},
{"device-disabled", NM_META_COLOR_DEVICE_DISABLED},
{"device-disconnected", NM_META_COLOR_DEVICE_DISCONNECTED},
{"device-external", NM_META_COLOR_DEVICE_EXTERNAL},
{"device-firmware-missing", NM_META_COLOR_DEVICE_FIRMWARE_MISSING},
{"device-plugin-missing", NM_META_COLOR_DEVICE_PLUGIN_MISSING},
{"device-unavailable", NM_META_COLOR_DEVICE_UNAVAILABLE},
{"device-unknown", NM_META_COLOR_DEVICE_UNKNOWN},
{"disabled", NM_META_COLOR_DISABLED},
{"enabled", NM_META_COLOR_ENABLED},
{"manager-running", NM_META_COLOR_MANAGER_RUNNING},
{"manager-starting", NM_META_COLOR_MANAGER_STARTING},
{"manager-stopped", NM_META_COLOR_MANAGER_STOPPED},
{"permission-auth", NM_META_COLOR_PERMISSION_AUTH},
{"permission-no", NM_META_COLOR_PERMISSION_NO},
{"permission-unknown", NM_META_COLOR_PERMISSION_UNKNOWN},
{"permission-yes", NM_META_COLOR_PERMISSION_YES},
{"prompt", NM_META_COLOR_PROMPT},
{"state-asleep", NM_META_COLOR_STATE_ASLEEP},
{"state-connected-global", NM_META_COLOR_STATE_CONNECTED_GLOBAL},
{"state-connected-local", NM_META_COLOR_STATE_CONNECTED_LOCAL},
{"state-connected-site", NM_META_COLOR_STATE_CONNECTED_SITE},
{"state-connecting", NM_META_COLOR_STATE_CONNECTING},
{"state-disconnected", NM_META_COLOR_STATE_DISCONNECTED},
{"state-disconnecting", NM_META_COLOR_STATE_DISCONNECTING},
{"state-unknown", NM_META_COLOR_STATE_UNKNOWN},
{"wifi-signal-excellent", NM_META_COLOR_WIFI_SIGNAL_EXCELLENT},
{"wifi-signal-fair", NM_META_COLOR_WIFI_SIGNAL_FAIR},
{"wifi-signal-good", NM_META_COLOR_WIFI_SIGNAL_GOOD},
{"wifi-signal-poor", NM_META_COLOR_WIFI_SIGNAL_POOR},
{"wifi-signal-unknown", NM_META_COLOR_WIFI_SIGNAL_UNKNOWN}, );
static gboolean
parse_color_scheme(char *palette_buffer, NmcColorPalette *out_palette, GError **error)
{
char *p = palette_buffer;
nm_assert(out_palette);
*out_palette = (NmcColorPalette) _NMC_COLOR_PALETTE_INIT();
/* This reads through the raw color scheme file contents, identifying the
* color names and sequences, putting in terminating NULs in place, so that
* pointers into the buffer can readily be used as strings in the palette. */
while (1) {
NMMetaColor name_idx;
const char *name;
const char *color;
/* Leading whitespace. */
while (nm_utils_is_separator(*p) || *p == '\n')
p++;
if (*p == '\0')
break;
/* Comments. */
if (*p == '#') {
while (*p != '\n' && *p != '\0')
p++;
continue;
}
/* Color name. */
name = p;
while (g_ascii_isgraph(*p))
p++;
if (*p == '\0') {
g_set_error(error, NMCLI_ERROR, 0, _("Unexpected end of file following '%s'\n"), name);
return FALSE;
}
/* Separating whitespace. */
if (!nm_utils_is_separator(*p)) {
*p = '\0';
g_set_error(error, NMCLI_ERROR, 0, _("Expected whitespace following '%s'\n"), name);
return FALSE;
}
while (nm_utils_is_separator(*p)) {
*p = '\0';
p++;
}
/* Color sequence. */
color = p;
if (!g_ascii_isgraph(*p)) {
g_set_error(error, NMCLI_ERROR, 0, _("Expected a value for '%s'\n"), name);
return FALSE;
}
while (g_ascii_isgraph(*p))
p++;
/* Trailing whitespace. */
while (nm_utils_is_separator(*p)) {
*p = '\0';
p++;
}
if (*p != '\0') {
if (*p != '\n') {
g_set_error(error,
NMCLI_ERROR,
0,
_("Expected a line break following '%s'\n"),
color);
return FALSE;
}
*p = '\0';
p++;
}
name_idx = _nm_meta_color_from_name(name);
if (name_idx == NM_META_COLOR_NONE) {
g_debug("Ignoring an unrecognized color: '%s'\n", name);
continue;
}
out_palette->ansi_seq[name_idx] = _resolve_color_alias(color) ?: color;
}
return TRUE;
}
static void
set_colors(NmcColorOption color_option,
bool * out_use_colors,
char ** out_palette_buffer,
NmcColorPalette *out_palette)
{
gs_free char *palette_str = NULL;
gboolean use_colors;
gboolean palette_set = FALSE;
nm_assert(out_use_colors);
nm_assert(out_palette);
nm_assert(out_palette_buffer && !*out_palette_buffer);
use_colors = check_colors(color_option, &palette_str);
*out_use_colors = use_colors;
if (use_colors && palette_str) {
gs_free_error GError *error = NULL;
NmcColorPalette palette;
if (!parse_color_scheme(palette_str, &palette, &error))
g_debug("Error parsing color scheme: %s", error->message);
else {
*out_palette_buffer = g_steal_pointer(&palette_str);
*out_palette = palette;
palette_set = TRUE;
}
}
if (!palette_set)
*out_palette = (NmcColorPalette) _NMC_COLOR_PALETTE_INIT();
}
/*************************************************************************************/
static gboolean
process_command_line(NmCli *nmc, int argc, char **argv_orig)
2010-02-25 09:52:30 -08:00
{
static const NMCCommand nmcli_cmds[] = {
{"general", nmc_command_func_general, NULL, FALSE, FALSE},
{"monitor", nmc_command_func_monitor, NULL, TRUE, FALSE},
{"networking", nmc_command_func_networking, NULL, FALSE, FALSE},
{"radio", nmc_command_func_radio, NULL, FALSE, FALSE},
{"connection", nmc_command_func_connection, NULL, FALSE, FALSE},
{"device", nmc_command_func_device, NULL, FALSE, FALSE},
{"agent", nmc_command_func_agent, NULL, FALSE, FALSE},
{NULL, nmc_command_func_overview, usage, TRUE, TRUE},
};
NmcColorOption colors = NMC_USE_COLOR_AUTO;
const char * base;
const char *const *argv;
base = strrchr(argv_orig[0], '/');
2010-02-25 09:52:30 -08:00
if (base == NULL)
base = argv_orig[0];
2010-02-25 09:52:30 -08:00
else
base++;
if (argc > 1 && nm_streq(argv_orig[1], "--complete-args")) {
nmc->complete = TRUE;
argv_orig[1] = argv_orig[0];
argc--;
argv_orig++;
}
argv = (const char *const *) argv_orig;
next_arg(nmc, &argc, &argv, NULL);
2010-02-25 09:52:30 -08:00
/* parse options */
while (argc) {
gs_free char *value = NULL;
if (argv[0][0] != '-')
2010-02-25 09:52:30 -08:00
break;
2016-07-27 15:21:06 +02:00
if (argc == 1 && nmc->complete) {
nmc_complete_strings(argv[0],
"--terse",
"--pretty",
"--mode",
"--overview",
"--colors",
"--escape",
"--fields",
"--nocheck",
"--get-values",
"--wait",
"--version",
"--help");
2016-07-27 15:21:06 +02:00
}
if (argv[0][1] == '-' && argv[0][2] == '\0') {
/* '--' ends options */
next_arg(nmc, &argc, &argv, NULL);
break;
}
if (matches_arg(nmc, &argc, &argv, "-overview", NULL)) {
nmc->nmc_config_mutable.overview = TRUE;
} else if (matches_arg(nmc, &argc, &argv, "-terse", NULL)) {
if (nmc->nmc_config.print_output == NMC_PRINT_TERSE) {
g_string_printf(nmc->return_text,
_("Error: Option '--terse' is specified the second time."));
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
} else if (nmc->nmc_config.print_output == NMC_PRINT_PRETTY) {
g_string_printf(
nmc->return_text,
_("Error: Option '--terse' is mutually exclusive with '--pretty'."));
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
} else
nmc->nmc_config_mutable.print_output = NMC_PRINT_TERSE;
} else if (matches_arg(nmc, &argc, &argv, "-pretty", NULL)) {
if (nmc->nmc_config.print_output == NMC_PRINT_PRETTY) {
g_string_printf(nmc->return_text,
_("Error: Option '--pretty' is specified the second time."));
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
} else if (nmc->nmc_config.print_output == NMC_PRINT_TERSE) {
g_string_printf(
nmc->return_text,
_("Error: Option '--pretty' is mutually exclusive with '--terse'."));
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
} else
nmc->nmc_config_mutable.print_output = NMC_PRINT_PRETTY;
} else if (matches_arg(nmc, &argc, &argv, "-mode", &value)) {
nmc->mode_specified = TRUE;
2016-07-27 16:24:30 +02:00
if (argc == 1 && nmc->complete)
complete_option_with_value(argv[0], value, "tabular", "multiline", NULL);
if (matches(value, "tabular"))
nmc->nmc_config_mutable.multiline_output = FALSE;
else if (matches(value, "multiline"))
nmc->nmc_config_mutable.multiline_output = TRUE;
else {
g_string_printf(nmc->return_text,
_("Error: '%s' is not a valid argument for '%s' option."),
value,
argv[0]);
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
}
} else if (matches_arg(nmc, &argc, &argv, "-colors", &value)) {
2016-07-27 16:24:30 +02:00
if (argc == 1 && nmc->complete)
complete_option_with_value(argv[0], value, "yes", "no", "auto", NULL);
if (matches(value, "auto"))
colors = NMC_USE_COLOR_AUTO;
else if (matches(value, "yes"))
colors = NMC_USE_COLOR_YES;
else if (matches(value, "no"))
colors = NMC_USE_COLOR_NO;
else {
g_string_printf(nmc->return_text,
_("Error: '%s' is not valid argument for '%s' option."),
value,
argv[0]);
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
}
} else if (matches_arg(nmc, &argc, &argv, "-escape", &value)) {
2016-07-27 16:24:30 +02:00
if (argc == 1 && nmc->complete)
complete_option_with_value(argv[0], value, "yes", "no", NULL);
if (matches(value, "yes"))
nmc->nmc_config_mutable.escape_values = TRUE;
else if (matches(value, "no"))
nmc->nmc_config_mutable.escape_values = FALSE;
else {
g_string_printf(nmc->return_text,
_("Error: '%s' is not valid argument for '%s' option."),
value,
argv[0]);
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
}
} else if (matches_arg(nmc, &argc, &argv, "-fields", &value)) {
2016-07-27 16:24:30 +02:00
if (argc == 1 && nmc->complete)
complete_fields(argv[0], value);
nmc->required_fields = g_strdup(value);
} else if (matches_arg(nmc, &argc, &argv, "-get-values", &value)) {
if (argc == 1 && nmc->complete)
complete_fields(argv[0], value);
nmc->required_fields = g_strdup(value);
nmc->nmc_config_mutable.print_output = NMC_PRINT_TERSE;
/* We want fixed tabular mode here, but just set the mode specified and rely on defaults:
* in this way we allow use of "-m multiline" to swap the output mode also if placed
* before the "-g <field>" option (-g may be still more practical and easy to remember than -t -f).
*/
nmc->mode_specified = TRUE;
} else if (matches_arg(nmc, &argc, &argv, "-nocheck", NULL)) {
/* ignore for backward compatibility */
} else if (matches_arg(nmc, &argc, &argv, "-wait", &value)) {
unsigned long timeout;
if (!nmc_string_to_uint(value, TRUE, 0, G_MAXINT, &timeout)) {
g_string_printf(nmc->return_text, _("Error: '%s' is not a valid timeout."), value);
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
return FALSE;
}
nmc->timeout = (int) timeout;
} else if (matches_arg(nmc, &argc, &argv, "-version", NULL)) {
if (!nmc->complete)
g_print(_("nmcli tool, version %s\n"), NMCLI_VERSION);
2010-02-25 09:52:30 -08:00
return NMC_RESULT_SUCCESS;
} else if (matches_arg(nmc, &argc, &argv, "-help", NULL)) {
if (!nmc->complete)
usage();
2010-02-25 09:52:30 -08:00
return NMC_RESULT_SUCCESS;
} else {
if (nmc->return_value == NMC_RESULT_SUCCESS) {
g_string_printf(nmc->return_text,
_("Error: Option '%s' is unknown, try 'nmcli -help'."),
argv[0]);
nmc->return_value = NMC_RESULT_ERROR_USER_INPUT;
}
return FALSE;
2010-02-25 09:52:30 -08:00
}
next_arg(nmc, &argc, &argv, NULL);
2010-02-25 09:52:30 -08:00
}
/* Ignore --overview when fields are set explicitly */
if (nmc->required_fields)
nmc->nmc_config_mutable.overview = FALSE;
set_colors(colors,
&nmc->nmc_config_mutable.use_colors,
&nmc->palette_buffer,
&nmc->nmc_config_mutable.palette);
/* Now run the requested command */
nmc_do_cmd(nmc, nmcli_cmds, *argv, argc, argv);
return TRUE;
2010-02-25 09:52:30 -08:00
}
static gboolean nmcli_sigint = FALSE;
gboolean
nmc_seen_sigint(void)
{
return nmcli_sigint;
}
void
nmc_clear_sigint(void)
{
nmcli_sigint = FALSE;
}
void
nmc_exit(void)
{
tcsetattr(STDIN_FILENO, TCSADRAIN, &termios_orig);
nmc_cleanup_readline();
exit(1);
}
static gboolean
signal_handler(gpointer user_data)
2010-02-25 09:52:30 -08:00
{
int signo = GPOINTER_TO_INT(user_data);
switch (signo) {
case SIGINT:
if (nmc_get_in_readline()) {
nmcli_sigint = TRUE;
} else {
nm_cli.return_value = 0x80 + signo;
g_string_printf(nm_cli.return_text,
_("Error: nmcli terminated by signal %s (%d)"),
strsignal(signo),
signo);
g_main_loop_quit(loop);
}
break;
case SIGTERM:
nm_cli.return_value = 0x80 + signo;
g_string_printf(nm_cli.return_text,
_("Error: nmcli terminated by signal %s (%d)"),
strsignal(signo),
signo);
nmc_exit();
break;
}
return G_SOURCE_CONTINUE;
2010-02-25 09:52:30 -08:00
}
void
nm_cli_spawn_pager(const NmcConfig *nmc_config, NmcPagerData *pager_data)
{
if (pager_data->pid != 0)
return;
pager_data->pid = nmc_terminal_spawn_pager(nmc_config);
}
2010-02-25 09:52:30 -08:00
static void
nmc_cleanup(NmCli *nmc)
{
pid_t ret;
g_clear_object(&nmc->client);
2010-02-25 09:52:30 -08:00
if (nmc->return_text)
g_string_free(g_steal_pointer(&nmc->return_text), TRUE);
2010-02-25 09:52:30 -08:00
if (nmc->secret_agent) {
nm_secret_agent_old_unregister(NM_SECRET_AGENT_OLD(nmc->secret_agent), NULL, NULL);
g_clear_object(&nmc->secret_agent);
}
nm_clear_pointer(&nmc->pwds_hash, g_hash_table_destroy);
nm_clear_g_free(&nmc->required_fields);
if (nmc->pager_data.pid != 0) {
pid_t pid = nm_steal_int(&nmc->pager_data.pid);
fclose(stdout);
fclose(stderr);
do {
ret = waitpid(pid, NULL, 0);
} while (ret == -1 && errno == EINTR);
}
nm_clear_g_free(&nmc->palette_buffer);
nmc_polkit_agent_fini(nmc);
2010-02-25 09:52:30 -08:00
}
int
main(int argc, char *argv[])
{
/* Set locale to use environment variables */
setlocale(LC_ALL, "");
#ifdef GETTEXT_PACKAGE
/* Set i18n stuff */
bindtextdomain(GETTEXT_PACKAGE, NMLOCALEDIR);
2010-02-25 09:52:30 -08:00
bind_textdomain_codeset(GETTEXT_PACKAGE, "UTF-8");
textdomain(GETTEXT_PACKAGE);
#endif
/* Save terminal settings */
tcgetattr(STDIN_FILENO, &termios_orig);
2010-02-25 09:52:30 -08:00
nm_cli.return_text = g_string_new(_("Success"));
loop = g_main_loop_new(NULL, FALSE);
g_unix_signal_add(SIGTERM, signal_handler, GINT_TO_POINTER(SIGTERM));
g_unix_signal_add(SIGINT, signal_handler, GINT_TO_POINTER(SIGINT));
if (process_command_line(&nm_cli, argc, argv))
g_main_loop_run(loop);
2010-02-25 09:52:30 -08:00
if (nm_cli.complete) {
/* Remove error statuses from command completion runs. */
if (nm_cli.return_value < NMC_RESULT_COMPLETE_FILE)
nm_cli.return_value = NMC_RESULT_SUCCESS;
} else if (nm_cli.return_value != NMC_RESULT_SUCCESS) {
/* Print result descripting text */
g_printerr("%s\n", nm_cli.return_text->str);
2010-02-25 09:52:30 -08:00
}
nmc_cleanup(&nm_cli);
cli: unref main loop after destroying NMClient instance Callbacks might reference the main loop when destroying the NMClient instance. Unref the main loop later. # G_DEBUG=fatal-warnings valgrind --num-callers=100 nmcli device wifi connect home ^C Error: nmcli terminated by signal Interrupt (2) Error: Connection activation failed: (0) No reason given. ==11050== Invalid read of size 4 ==11050== at 0x4C90D3D: g_main_loop_quit (in /usr/lib64/libglib-2.0.so.0.6200.6) ==11050== by 0x431435: quit (devices.c:934) ==11050== by 0x43272C: connected_state_cb (devices.c:1919) ==11050== by 0x4BF6741: g_closure_invoke (in /usr/lib64/libgobject-2.0.so.0.6200.6) ==11050== by 0x4C0A603: ??? (in /usr/lib64/libgobject-2.0.so.0.6200.6) ==11050== by 0x4C133AD: g_signal_emit_valist (in /usr/lib64/libgobject-2.0.so.0.6200.6) ==11050== by 0x4C139D2: g_signal_emit (in /usr/lib64/libgobject-2.0.so.0.6200.6) ==11050== by 0x4BFB1C3: ??? (in /usr/lib64/libgobject-2.0.so.0.6200.6) ==11050== by 0x4BFAAEC: ??? (in /usr/lib64/libgobject-2.0.so.0.6200.6) ==11050== by 0x4BFD86A: g_object_thaw_notify (in /usr/lib64/libgobject-2.0.so.0.6200.6) ==11050== by 0x48BA040: _nm_client_notify_event_emit (nm-client.c:937) ==11050== by 0x48CA01F: _dbus_handle_changes_commit (nm-client.c:2850) ==11050== by 0x48CC221: _dbus_handle_changes (nm-client.c:2864) ==11050== by 0x48CC833: _init_release_all (nm-client.c:6969) ==11050== by 0x48D2818: dispose (nm-client.c:7826) ==11050== by 0x4BFBC27: g_object_unref (in /usr/lib64/libgobject-2.0.so.0.6200.6) ==11050== by 0x43FF93: nmc_cleanup (nmcli.c:941) ==11050== by 0x4410AD: main (nmcli.c:1005) ==11050== Address 0x54738fc is 12 bytes inside a block of size 16 free'd ==11050== at 0x4839A0C: free (vg_replace_malloc.c:540) ==11050== by 0x4C9649C: g_free (in /usr/lib64/libglib-2.0.so.0.6200.6) ==11050== by 0x4410A3: main (nmcli.c:1004) ==11050== Block was alloc'd at ==11050== at 0x483AB1A: calloc (vg_replace_malloc.c:762) ==11050== by 0x4C96400: g_malloc0 (in /usr/lib64/libglib-2.0.so.0.6200.6) ==11050== by 0x4C90A45: g_main_loop_new (in /usr/lib64/libglib-2.0.so.0.6200.6) ==11050== by 0x441020: main (nmcli.c:987) https://gitlab.freedesktop.org/NetworkManager/NetworkManager/-/merge_requests/501
2020-05-12 13:47:32 +02:00
g_main_loop_unref(loop);
2010-02-25 09:52:30 -08:00
return nm_cli.return_value;
2010-02-25 09:52:30 -08:00
}