Blame clients/cli/common.c

Packit Service b23acc
// SPDX-License-Identifier: GPL-2.0+
Packit Service b23acc
/*
Packit Service b23acc
 * Copyright (C) 2012 - 2018 Red Hat, Inc.
Packit Service b23acc
 */
Packit Service b23acc
Packit Service b23acc
#include "nm-default.h"
Packit Service b23acc
Packit Service b23acc
#include "common.h"
Packit Service b23acc
Packit Service b23acc
#include <stdio.h>
Packit Service b23acc
#include <stdlib.h>
Packit Service b23acc
#include <sys/ioctl.h>
Packit Service b23acc
#include <readline/readline.h>
Packit Service b23acc
#include <readline/history.h>
Packit Service b23acc
Packit Service b23acc
#include "nm-libnm-aux/nm-libnm-aux.h"
Packit Service b23acc
Packit Service b23acc
#include "nm-vpn-helpers.h"
Packit Service b23acc
#include "nm-client-utils.h"
Packit Service b23acc
#include "nm-glib-aux/nm-secret-utils.h"
Packit Service b23acc
Packit Service b23acc
#include "utils.h"
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static char **
Packit Service b23acc
_ip_config_get_routes (NMIPConfig *cfg)
Packit Service b23acc
{
Packit Service b23acc
	gs_unref_hashtable GHashTable *hash = NULL;
Packit Service b23acc
	GPtrArray *ptr_array;
Packit Service b23acc
	char **arr;
Packit Service b23acc
	guint i;
Packit Service b23acc
Packit Service b23acc
	ptr_array = nm_ip_config_get_routes (cfg);
Packit Service b23acc
	if (!ptr_array)
Packit Service b23acc
		return NULL;
Packit Service b23acc
Packit Service b23acc
	if (ptr_array->len == 0)
Packit Service b23acc
		return NULL;
Packit Service b23acc
Packit Service b23acc
	arr = g_new (char *, ptr_array->len + 1);
Packit Service b23acc
	for (i = 0; i < ptr_array->len; i++) {
Packit Service b23acc
		NMIPRoute *route = g_ptr_array_index (ptr_array, i);
Packit Service b23acc
		gs_strfreev char **names = NULL;
Packit Service b23acc
		gsize j;
Packit Service b23acc
		GString *str;
Packit Service b23acc
		guint64 metric;
Packit Service b23acc
		gs_free char *attributes = NULL;
Packit Service b23acc
Packit Service b23acc
		str = g_string_new (NULL);
Packit Service b23acc
		g_string_append_printf (str,
Packit Service b23acc
		                        "dst = %s/%u, nh = %s",
Packit Service b23acc
		                        nm_ip_route_get_dest (route),
Packit Service b23acc
		                        nm_ip_route_get_prefix (route),
Packit Service b23acc
		                        nm_ip_route_get_next_hop (route)
Packit Service b23acc
		                          ?: (nm_ip_route_get_family (route) == AF_INET ? "0.0.0.0" : "::"));
Packit Service b23acc
Packit Service b23acc
		metric = nm_ip_route_get_metric (route);
Packit Service b23acc
		if (metric != -1) {
Packit Service b23acc
			g_string_append_printf (str,
Packit Service b23acc
			                        ", mt = %u",
Packit Service b23acc
			                        (guint) metric);
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		names = nm_ip_route_get_attribute_names (route);
Packit Service b23acc
		if (names[0]) {
Packit Service b23acc
			if (!hash)
Packit Service b23acc
				hash = g_hash_table_new (nm_str_hash, g_str_equal);
Packit Service b23acc
			else
Packit Service b23acc
				g_hash_table_remove_all (hash);
Packit Service b23acc
Packit Service b23acc
			for (j = 0; names[j]; j++)
Packit Service b23acc
				g_hash_table_insert (hash, names[j], nm_ip_route_get_attribute (route, names[j]));
Packit Service b23acc
Packit Service b23acc
			attributes = nm_utils_format_variant_attributes (hash, ',', '=');
Packit Service b23acc
			if (attributes) {
Packit Service b23acc
				g_string_append (str, ", ");
Packit Service b23acc
				g_string_append (str, attributes);
Packit Service b23acc
			}
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		arr[i] = g_string_free (str, FALSE);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	nm_assert (i == ptr_array->len);
Packit Service b23acc
	arr[i] = NULL;
Packit Service b23acc
Packit Service b23acc
	return arr;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static gconstpointer
Packit Service b23acc
_metagen_ip4_config_get_fcn (NMC_META_GENERIC_INFO_GET_FCN_ARGS)
Packit Service b23acc
{
Packit Service b23acc
	NMIPConfig *cfg4 = target;
Packit Service b23acc
	GPtrArray *ptr_array;
Packit Service b23acc
	char **arr;
Packit Service b23acc
	const char *const*arrc;
Packit Service b23acc
	guint i = 0;
Packit Service b23acc
	const char *str;
Packit Service b23acc
Packit Service b23acc
	nm_assert (info->info_type < _NMC_GENERIC_INFO_TYPE_IP4_CONFIG_NUM);
Packit Service b23acc
Packit Service b23acc
	NMC_HANDLE_COLOR (NM_META_COLOR_NONE);
Packit Service b23acc
	NM_SET_OUT (out_is_default, TRUE);
Packit Service b23acc
Packit Service b23acc
	switch (info->info_type) {
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP4_CONFIG_ADDRESS:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		ptr_array = nm_ip_config_get_addresses (cfg4);
Packit Service b23acc
		if (ptr_array) {
Packit Service b23acc
			arr = g_new (char *, ptr_array->len + 1);
Packit Service b23acc
			for (i = 0; i < ptr_array->len; i++) {
Packit Service b23acc
				NMIPAddress *addr = g_ptr_array_index (ptr_array, i);
Packit Service b23acc
Packit Service b23acc
				arr[i] = g_strdup_printf ("%s/%u",
Packit Service b23acc
				                          nm_ip_address_get_address (addr),
Packit Service b23acc
				                          nm_ip_address_get_prefix (addr));
Packit Service b23acc
			}
Packit Service b23acc
			arr[i] = NULL;
Packit Service b23acc
		} else
Packit Service b23acc
			arr = NULL;
Packit Service b23acc
		goto arr_out;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP4_CONFIG_GATEWAY:
Packit Service b23acc
		str = nm_ip_config_get_gateway (cfg4);
Packit Service b23acc
		NM_SET_OUT (out_is_default, !str);
Packit Service b23acc
		return str;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP4_CONFIG_ROUTE:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		arr = _ip_config_get_routes (cfg4);
Packit Service b23acc
		goto arr_out;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP4_CONFIG_DNS:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		arrc = nm_ip_config_get_nameservers (cfg4);
Packit Service b23acc
		goto arrc_out;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP4_CONFIG_DOMAIN:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		arrc = nm_ip_config_get_domains (cfg4);
Packit Service b23acc
		goto arrc_out;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP4_CONFIG_WINS:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		arrc = nm_ip_config_get_wins_servers (cfg4);
Packit Service b23acc
		goto arrc_out;
Packit Service b23acc
	default:
Packit Service b23acc
		break;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_reached (NULL);
Packit Service b23acc
Packit Service b23acc
arrc_out:
Packit Service b23acc
	NM_SET_OUT (out_is_default, !arrc || !arrc[0]);
Packit Service b23acc
	*out_flags |= NM_META_ACCESSOR_GET_OUT_FLAGS_STRV;
Packit Service b23acc
	return arrc;
Packit Service b23acc
Packit Service b23acc
arr_out:
Packit Service b23acc
	NM_SET_OUT (out_is_default, !arr || !arr[0]);
Packit Service b23acc
	*out_flags |= NM_META_ACCESSOR_GET_OUT_FLAGS_STRV;
Packit Service b23acc
	*out_to_free = arr;
Packit Service b23acc
	return arr;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
const NmcMetaGenericInfo *const metagen_ip4_config[_NMC_GENERIC_INFO_TYPE_IP4_CONFIG_NUM + 1] = {
Packit Service b23acc
#define _METAGEN_IP4_CONFIG(type, name) \
Packit Service b23acc
	[type] = NMC_META_GENERIC(name, .info_type = type, .get_fcn = _metagen_ip4_config_get_fcn)
Packit Service b23acc
	_METAGEN_IP4_CONFIG (NMC_GENERIC_INFO_TYPE_IP4_CONFIG_ADDRESS, "ADDRESS"),
Packit Service b23acc
	_METAGEN_IP4_CONFIG (NMC_GENERIC_INFO_TYPE_IP4_CONFIG_GATEWAY, "GATEWAY"),
Packit Service b23acc
	_METAGEN_IP4_CONFIG (NMC_GENERIC_INFO_TYPE_IP4_CONFIG_ROUTE,   "ROUTE"),
Packit Service b23acc
	_METAGEN_IP4_CONFIG (NMC_GENERIC_INFO_TYPE_IP4_CONFIG_DNS,     "DNS"),
Packit Service b23acc
	_METAGEN_IP4_CONFIG (NMC_GENERIC_INFO_TYPE_IP4_CONFIG_DOMAIN,  "DOMAIN"),
Packit Service b23acc
	_METAGEN_IP4_CONFIG (NMC_GENERIC_INFO_TYPE_IP4_CONFIG_WINS,    "WINS"),
Packit Service b23acc
};
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static gconstpointer
Packit Service b23acc
_metagen_ip6_config_get_fcn (NMC_META_GENERIC_INFO_GET_FCN_ARGS)
Packit Service b23acc
{
Packit Service b23acc
	NMIPConfig *cfg6 = target;
Packit Service b23acc
	GPtrArray *ptr_array;
Packit Service b23acc
	char **arr;
Packit Service b23acc
	const char *const*arrc;
Packit Service b23acc
	guint i = 0;
Packit Service b23acc
	const char *str;
Packit Service b23acc
Packit Service b23acc
	nm_assert (info->info_type < _NMC_GENERIC_INFO_TYPE_IP6_CONFIG_NUM);
Packit Service b23acc
Packit Service b23acc
	NMC_HANDLE_COLOR (NM_META_COLOR_NONE);
Packit Service b23acc
	NM_SET_OUT (out_is_default, TRUE);
Packit Service b23acc
Packit Service b23acc
	switch (info->info_type) {
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP6_CONFIG_ADDRESS:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		ptr_array = nm_ip_config_get_addresses (cfg6);
Packit Service b23acc
		if (ptr_array) {
Packit Service b23acc
			arr = g_new (char *, ptr_array->len + 1);
Packit Service b23acc
			for (i = 0; i < ptr_array->len; i++) {
Packit Service b23acc
				NMIPAddress *addr = g_ptr_array_index (ptr_array, i);
Packit Service b23acc
Packit Service b23acc
				arr[i] = g_strdup_printf ("%s/%u",
Packit Service b23acc
				                          nm_ip_address_get_address (addr),
Packit Service b23acc
				                          nm_ip_address_get_prefix (addr));
Packit Service b23acc
			}
Packit Service b23acc
			arr[i] = NULL;
Packit Service b23acc
		} else
Packit Service b23acc
			arr = NULL;
Packit Service b23acc
		goto arr_out;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP6_CONFIG_GATEWAY:
Packit Service b23acc
		str = nm_ip_config_get_gateway (cfg6);
Packit Service b23acc
		NM_SET_OUT (out_is_default, !str);
Packit Service b23acc
		return str;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP6_CONFIG_ROUTE:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		arr = _ip_config_get_routes (cfg6);
Packit Service b23acc
		goto arr_out;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP6_CONFIG_DNS:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		arrc = nm_ip_config_get_nameservers (cfg6);
Packit Service b23acc
		goto arrc_out;
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_IP6_CONFIG_DOMAIN:
Packit Service b23acc
		if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
			return NULL;
Packit Service b23acc
		arrc = nm_ip_config_get_domains (cfg6);
Packit Service b23acc
		goto arrc_out;
Packit Service b23acc
	default:
Packit Service b23acc
		break;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_reached (NULL);
Packit Service b23acc
Packit Service b23acc
arrc_out:
Packit Service b23acc
	NM_SET_OUT (out_is_default, !arrc || !arrc[0]);
Packit Service b23acc
	*out_flags |= NM_META_ACCESSOR_GET_OUT_FLAGS_STRV;
Packit Service b23acc
	return arrc;
Packit Service b23acc
Packit Service b23acc
arr_out:
Packit Service b23acc
	NM_SET_OUT (out_is_default, !arr || !arr[0]);
Packit Service b23acc
	*out_flags |= NM_META_ACCESSOR_GET_OUT_FLAGS_STRV;
Packit Service b23acc
	*out_to_free = arr;
Packit Service b23acc
	return arr;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
const NmcMetaGenericInfo *const metagen_ip6_config[_NMC_GENERIC_INFO_TYPE_IP6_CONFIG_NUM + 1] = {
Packit Service b23acc
#define _METAGEN_IP6_CONFIG(type, name) \
Packit Service b23acc
	[type] = NMC_META_GENERIC(name, .info_type = type, .get_fcn = _metagen_ip6_config_get_fcn)
Packit Service b23acc
	_METAGEN_IP6_CONFIG (NMC_GENERIC_INFO_TYPE_IP6_CONFIG_ADDRESS, "ADDRESS"),
Packit Service b23acc
	_METAGEN_IP6_CONFIG (NMC_GENERIC_INFO_TYPE_IP6_CONFIG_GATEWAY, "GATEWAY"),
Packit Service b23acc
	_METAGEN_IP6_CONFIG (NMC_GENERIC_INFO_TYPE_IP6_CONFIG_ROUTE,   "ROUTE"),
Packit Service b23acc
	_METAGEN_IP6_CONFIG (NMC_GENERIC_INFO_TYPE_IP6_CONFIG_DNS,     "DNS"),
Packit Service b23acc
	_METAGEN_IP6_CONFIG (NMC_GENERIC_INFO_TYPE_IP6_CONFIG_DOMAIN,  "DOMAIN"),
Packit Service b23acc
};
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static gconstpointer
Packit Service b23acc
_metagen_dhcp_config_get_fcn (NMC_META_GENERIC_INFO_GET_FCN_ARGS)
Packit Service b23acc
{
Packit Service b23acc
	NMDhcpConfig *dhcp = target;
Packit Service b23acc
	guint i;
Packit Service b23acc
	char **arr = NULL;
Packit Service b23acc
Packit Service b23acc
	NMC_HANDLE_COLOR (NM_META_COLOR_NONE);
Packit Service b23acc
Packit Service b23acc
	switch (info->info_type) {
Packit Service b23acc
	case NMC_GENERIC_INFO_TYPE_DHCP_CONFIG_OPTION:
Packit Service b23acc
		{
Packit Service b23acc
			GHashTable *table;
Packit Service b23acc
			gs_free char **arr2 = NULL;
Packit Service b23acc
			guint n;
Packit Service b23acc
Packit Service b23acc
			if (!NM_FLAGS_HAS (get_flags, NM_META_ACCESSOR_GET_FLAGS_ACCEPT_STRV))
Packit Service b23acc
				return NULL;
Packit Service b23acc
Packit Service b23acc
			table = nm_dhcp_config_get_options (dhcp);
Packit Service b23acc
			if (!table)
Packit Service b23acc
				goto arr_out;
Packit Service b23acc
Packit Service b23acc
			arr2 = (char **) nm_utils_strdict_get_keys (table, TRUE, &n);
Packit Service b23acc
			if (!n)
Packit Service b23acc
				goto arr_out;
Packit Service b23acc
Packit Service b23acc
			nm_assert (arr2 && !arr2[n] && n == NM_PTRARRAY_LEN (arr2));
Packit Service b23acc
			for (i = 0; i < n; i++) {
Packit Service b23acc
				const char *k = arr2[i];
Packit Service b23acc
				const char *v;
Packit Service b23acc
Packit Service b23acc
				nm_assert (k);
Packit Service b23acc
				v = g_hash_table_lookup (table, k);
Packit Service b23acc
				arr2[i] = g_strdup_printf ("%s = %s", k, v);
Packit Service b23acc
			}
Packit Service b23acc
Packit Service b23acc
			arr = g_steal_pointer (&arr2);
Packit Service b23acc
			goto arr_out;
Packit Service b23acc
		}
Packit Service b23acc
	default:
Packit Service b23acc
		break;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_reached (NULL);
Packit Service b23acc
Packit Service b23acc
arr_out:
Packit Service b23acc
	NM_SET_OUT (out_is_default, !arr || !arr[0]);
Packit Service b23acc
	*out_flags |= NM_META_ACCESSOR_GET_OUT_FLAGS_STRV;
Packit Service b23acc
	*out_to_free = arr;
Packit Service b23acc
	return arr;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
const NmcMetaGenericInfo *const metagen_dhcp_config[_NMC_GENERIC_INFO_TYPE_DHCP_CONFIG_NUM + 1] = {
Packit Service b23acc
#define _METAGEN_DHCP_CONFIG(type, name) \
Packit Service b23acc
	[type] = NMC_META_GENERIC(name, .info_type = type, .get_fcn = _metagen_dhcp_config_get_fcn)
Packit Service b23acc
	_METAGEN_DHCP_CONFIG (NMC_GENERIC_INFO_TYPE_DHCP_CONFIG_OPTION, "OPTION"),
Packit Service b23acc
};
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
gboolean
Packit Service b23acc
print_ip_config (NMIPConfig *cfg,
Packit Service b23acc
                 int addr_family,
Packit Service b23acc
                 const NmcConfig *nmc_config,
Packit Service b23acc
                 const char *one_field)
Packit Service b23acc
{
Packit Service b23acc
	gs_free_error GError *error = NULL;
Packit Service b23acc
	gs_free char *field_str = NULL;
Packit Service b23acc
Packit Service b23acc
	if (!cfg)
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	if (one_field) {
Packit Service b23acc
		field_str = g_strdup_printf ("IP%c.%s",
Packit Service b23acc
		                             nm_utils_addr_family_to_char (addr_family),
Packit Service b23acc
		                             one_field);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (!nmc_print (nmc_config,
Packit Service b23acc
	                (gpointer[]) { cfg, NULL },
Packit Service b23acc
	                NULL,
Packit Service b23acc
	                NULL,
Packit Service b23acc
	                addr_family == AF_INET
Packit Service b23acc
	                  ? NMC_META_GENERIC_GROUP ("IP4", metagen_ip4_config, N_("GROUP"))
Packit Service b23acc
	                  : NMC_META_GENERIC_GROUP ("IP6", metagen_ip6_config, N_("GROUP")),
Packit Service b23acc
	                field_str,
Packit Service b23acc
	                &error)) {
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	}
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean
Packit Service b23acc
print_dhcp_config (NMDhcpConfig *dhcp,
Packit Service b23acc
                   int addr_family,
Packit Service b23acc
                   const NmcConfig *nmc_config,
Packit Service b23acc
                   const char *one_field)
Packit Service b23acc
{
Packit Service b23acc
	gs_free_error GError *error = NULL;
Packit Service b23acc
	gs_free char *field_str = NULL;
Packit Service b23acc
Packit Service b23acc
	if (!dhcp)
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	if (one_field) {
Packit Service b23acc
		field_str = g_strdup_printf ("DHCP%c.%s",
Packit Service b23acc
		                             nm_utils_addr_family_to_char (addr_family),
Packit Service b23acc
		                             one_field);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (!nmc_print (nmc_config,
Packit Service b23acc
	                (gpointer[]) { dhcp, NULL },
Packit Service b23acc
	                NULL,
Packit Service b23acc
	                NULL,
Packit Service b23acc
	                addr_family == AF_INET
Packit Service b23acc
	                  ? NMC_META_GENERIC_GROUP ("DHCP4", metagen_dhcp_config, N_("GROUP"))
Packit Service b23acc
	                  : NMC_META_GENERIC_GROUP ("DHCP6", metagen_dhcp_config, N_("GROUP")),
Packit Service b23acc
	                field_str,
Packit Service b23acc
	                &error)) {
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	}
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*
Packit Service b23acc
 * nmc_find_connection:
Packit Service b23acc
 * @connections: array of NMConnections to search in
Packit Service b23acc
 * @filter_type: "id", "uuid", "path", "filename", or %NULL
Packit Service b23acc
 * @filter_val: connection to find (connection name, UUID or path)
Packit Service b23acc
 * @out_result: if not NULL, attach all matching connection to this
Packit Service b23acc
 *   list. If necessary, a new array will be allocated. If the array
Packit Service b23acc
 *   already contains a connection, it will not be added a second time.
Packit Service b23acc
 *   All object are referenced by the array. If the function allocates
Packit Service b23acc
 *   a new array, it will set the free function to g_object_unref.
Packit Service b23acc
 * @complete: print possible completions
Packit Service b23acc
 *
Packit Service b23acc
 * Find a connection in @list according to @filter_val. @filter_type determines
Packit Service b23acc
 * what property is used for comparison. When @filter_type is NULL, compare
Packit Service b23acc
 * @filter_val against all types. Otherwise, only compare against the specified
Packit Service b23acc
 * type. If 'path' filter type is specified, comparison against numeric index
Packit Service b23acc
 * (in addition to the whole path) is allowed.
Packit Service b23acc
 *
Packit Service b23acc
 * Returns: found connection, or %NULL
Packit Service b23acc
 */
Packit Service b23acc
NMConnection *
Packit Service b23acc
nmc_find_connection (const GPtrArray *connections,
Packit Service b23acc
                     const char *filter_type,
Packit Service b23acc
                     const char *filter_val,
Packit Service b23acc
                     GPtrArray **out_result,
Packit Service b23acc
                     gboolean complete)
Packit Service b23acc
{
Packit Service b23acc
	NMConnection *best_candidate_uuid = NULL;
Packit Service b23acc
	NMConnection *best_candidate = NULL;
Packit Service b23acc
	gs_unref_ptrarray GPtrArray *result_allocated = NULL;
Packit Service b23acc
	GPtrArray *result = out_result ? *out_result : NULL;
Packit Service b23acc
	const guint result_inital_len = result ? result->len : 0u;
Packit Service b23acc
	guint i, j;
Packit Service b23acc
Packit Service b23acc
	nm_assert (connections);
Packit Service b23acc
	nm_assert (filter_val);
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < connections->len; i++) {
Packit Service b23acc
		gboolean match_by_uuid = FALSE;
Packit Service b23acc
		NMConnection *connection;
Packit Service b23acc
		const char *v;
Packit Service b23acc
		const char *v_num;
Packit Service b23acc
Packit Service b23acc
		connection = NM_CONNECTION (connections->pdata[i]);
Packit Service b23acc
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "uuid")) {
Packit Service b23acc
			v = nm_connection_get_uuid (connection);
Packit Service b23acc
			if (complete && (filter_type || *filter_val))
Packit Service b23acc
				nmc_complete_strings (filter_val, v);
Packit Service b23acc
			if (nm_streq0 (filter_val, v)) {
Packit Service b23acc
				match_by_uuid = TRUE;
Packit Service b23acc
				goto found;
Packit Service b23acc
			}
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "id")) {
Packit Service b23acc
			v = nm_connection_get_id (connection);
Packit Service b23acc
			if (complete)
Packit Service b23acc
				nmc_complete_strings (filter_val, v);
Packit Service b23acc
			if (nm_streq0 (filter_val, v))
Packit Service b23acc
				goto found;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "path")) {
Packit Service b23acc
			v = nm_connection_get_path (connection);
Packit Service b23acc
			v_num = nm_utils_dbus_path_get_last_component (v);
Packit Service b23acc
			if (complete && (filter_type || *filter_val))
Packit Service b23acc
				nmc_complete_strings (filter_val, v, (*filter_val ? v_num : NULL));
Packit Service b23acc
			if (   nm_streq0 (filter_val, v)
Packit Service b23acc
			    || (filter_type && nm_streq0 (filter_val, v_num)))
Packit Service b23acc
				goto found;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "filename")) {
Packit Service b23acc
			v = nm_remote_connection_get_filename (NM_REMOTE_CONNECTION (connections->pdata[i]));
Packit Service b23acc
			if (complete && (filter_type || *filter_val))
Packit Service b23acc
				nmc_complete_strings (filter_val, v);
Packit Service b23acc
			if (nm_streq0 (filter_val, v))
Packit Service b23acc
				goto found;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		continue;
Packit Service b23acc
Packit Service b23acc
found:
Packit Service b23acc
		if (match_by_uuid) {
Packit Service b23acc
			if (   !complete
Packit Service b23acc
			    && !out_result)
Packit Service b23acc
				return connection;
Packit Service b23acc
			best_candidate_uuid = connection;
Packit Service b23acc
		} else {
Packit Service b23acc
			if (!best_candidate)
Packit Service b23acc
				best_candidate = connection;
Packit Service b23acc
		}
Packit Service b23acc
		if (out_result) {
Packit Service b23acc
			gboolean already_tracked = FALSE;
Packit Service b23acc
Packit Service b23acc
			if (!result) {
Packit Service b23acc
				result_allocated = g_ptr_array_new_with_free_func (g_object_unref);
Packit Service b23acc
				result = result_allocated;
Packit Service b23acc
			} else {
Packit Service b23acc
				for (j = 0; j < result->len; j++) {
Packit Service b23acc
					if (connection == result->pdata[j]) {
Packit Service b23acc
						already_tracked = TRUE;
Packit Service b23acc
						break;
Packit Service b23acc
					}
Packit Service b23acc
				}
Packit Service b23acc
			}
Packit Service b23acc
			if (!already_tracked) {
Packit Service b23acc
				if (match_by_uuid) {
Packit Service b23acc
					/* the profile is matched exactly (by UUID). We prepend it
Packit Service b23acc
					 * to the list of all found profiles. */
Packit Service b23acc
					g_ptr_array_insert (result, result_inital_len, g_object_ref (connection));
Packit Service b23acc
				} else
Packit Service b23acc
					g_ptr_array_add (result, g_object_ref (connection));
Packit Service b23acc
			}
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (result_allocated)
Packit Service b23acc
		*out_result = g_steal_pointer (&result_allocated);
Packit Service b23acc
	return best_candidate_uuid ?: best_candidate;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
NMActiveConnection *
Packit Service b23acc
nmc_find_active_connection (const GPtrArray *active_cons,
Packit Service b23acc
                            const char *filter_type,
Packit Service b23acc
                            const char *filter_val,
Packit Service b23acc
                            GPtrArray **out_result,
Packit Service b23acc
                            gboolean complete)
Packit Service b23acc
{
Packit Service b23acc
	guint i, j;
Packit Service b23acc
	NMActiveConnection *best_candidate = NULL;
Packit Service b23acc
	GPtrArray *result = out_result ? *out_result : NULL;
Packit Service b23acc
Packit Service b23acc
	nm_assert (filter_val);
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < active_cons->len; i++) {
Packit Service b23acc
		NMRemoteConnection *con;
Packit Service b23acc
		NMActiveConnection *candidate = g_ptr_array_index (active_cons, i);
Packit Service b23acc
		const char *v, *v_num;
Packit Service b23acc
Packit Service b23acc
		con = nm_active_connection_get_connection (candidate);
Packit Service b23acc
Packit Service b23acc
		/* When filter_type is NULL, compare connection ID (filter_val)
Packit Service b23acc
		 * against all types. Otherwise, only compare against the specific
Packit Service b23acc
		 * type. If 'path' or 'apath' filter types are specified, comparison
Packit Service b23acc
		 * against numeric index (in addition to the whole path) is allowed.
Packit Service b23acc
		 */
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "id")) {
Packit Service b23acc
			v = nm_active_connection_get_id (candidate);
Packit Service b23acc
			if (complete)
Packit Service b23acc
				nmc_complete_strings (filter_val, v);
Packit Service b23acc
			if (nm_streq0 (filter_val, v))
Packit Service b23acc
				goto found;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "uuid")) {
Packit Service b23acc
			v = nm_active_connection_get_uuid (candidate);
Packit Service b23acc
			if (complete && (filter_type || *filter_val))
Packit Service b23acc
				nmc_complete_strings (filter_val, v);
Packit Service b23acc
			if (nm_streq0 (filter_val, v))
Packit Service b23acc
				goto found;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "path")) {
Packit Service b23acc
			v = con ? nm_connection_get_path (NM_CONNECTION (con)) : NULL;
Packit Service b23acc
			v_num = nm_utils_dbus_path_get_last_component (v);
Packit Service b23acc
			if (complete && (filter_type || *filter_val))
Packit Service b23acc
				nmc_complete_strings (filter_val, v, filter_type ? v_num : NULL);
Packit Service b23acc
			if (   nm_streq0 (filter_val, v)
Packit Service b23acc
			    || (filter_type && nm_streq0 (filter_val, v_num)))
Packit Service b23acc
				goto found;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "filename")) {
Packit Service b23acc
			v = nm_remote_connection_get_filename (con);
Packit Service b23acc
			if (complete && (filter_type || *filter_val))
Packit Service b23acc
				nmc_complete_strings (filter_val, v);
Packit Service b23acc
			if (nm_streq0 (filter_val, v))
Packit Service b23acc
				goto found;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		if (NM_IN_STRSET (filter_type, NULL, "apath")) {
Packit Service b23acc
			v = nm_object_get_path (NM_OBJECT (candidate));
Packit Service b23acc
			v_num = nm_utils_dbus_path_get_last_component (v);
Packit Service b23acc
			if (complete && (filter_type || *filter_val))
Packit Service b23acc
				nmc_complete_strings (filter_val, v, filter_type ? v_num : NULL);
Packit Service b23acc
			if (   nm_streq0 (filter_val, v)
Packit Service b23acc
			    || (filter_type && nm_streq0 (filter_val, v_num)))
Packit Service b23acc
				goto found;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		continue;
Packit Service b23acc
Packit Service b23acc
found:
Packit Service b23acc
		if (!out_result)
Packit Service b23acc
			return candidate;
Packit Service b23acc
		if (!best_candidate)
Packit Service b23acc
			best_candidate = candidate;
Packit Service b23acc
		if (!result)
Packit Service b23acc
			result = g_ptr_array_new_with_free_func (g_object_unref);
Packit Service b23acc
		for (j = 0; j < result->len; j++) {
Packit Service b23acc
			if (candidate == result->pdata[j])
Packit Service b23acc
				break;
Packit Service b23acc
		}
Packit Service b23acc
		if (j == result->len)
Packit Service b23acc
			g_ptr_array_add (result, g_object_ref (candidate));
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	NM_SET_OUT (out_result, result);
Packit Service b23acc
	return best_candidate;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
vpn_openconnect_get_secrets (NMConnection *connection, GPtrArray *secrets)
Packit Service b23acc
{
Packit Service b23acc
	GError *error = NULL;
Packit Service b23acc
	NMSettingVpn *s_vpn;
Packit Service b23acc
	const char *gw, *port;
Packit Service b23acc
	gs_free char *cookie = NULL;
Packit Service b23acc
	gs_free char *gateway = NULL;
Packit Service b23acc
	gs_free char *gwcert = NULL;
Packit Service b23acc
	int status = 0;
Packit Service b23acc
	int i;
Packit Service b23acc
	gboolean ret;
Packit Service b23acc
Packit Service b23acc
	if (!connection)
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	if (!nm_connection_is_type (connection, NM_SETTING_VPN_SETTING_NAME))
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	s_vpn = nm_connection_get_setting_vpn (connection);
Packit Service b23acc
	if (!nm_streq0 (nm_setting_vpn_get_service_type (s_vpn), NM_SECRET_AGENT_VPN_TYPE_OPENCONNECT))
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	/* Get gateway and port */
Packit Service b23acc
	gw = nm_setting_vpn_get_data_item (s_vpn, "gateway");
Packit Service b23acc
	port = gw ? strrchr (gw, ':') : NULL;
Packit Service b23acc
Packit Service b23acc
	/* Interactively authenticate to OpenConnect server and get secrets */
Packit Service b23acc
	ret = nm_vpn_openconnect_authenticate_helper (gw, &cookie, &gateway, &gwcert, &status, &error);
Packit Service b23acc
	if (!ret) {
Packit Service b23acc
		g_printerr (_("Error: openconnect failed: %s\n"), error->message);
Packit Service b23acc
		g_clear_error (&error);
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (WIFEXITED (status)) {
Packit Service b23acc
		if (WEXITSTATUS (status) != 0)
Packit Service b23acc
			g_printerr (_("Error: openconnect failed with status %d\n"), WEXITSTATUS (status));
Packit Service b23acc
	} else if (WIFSIGNALED (status))
Packit Service b23acc
		g_printerr (_("Error: openconnect failed with signal %d\n"), WTERMSIG (status));
Packit Service b23acc
Packit Service b23acc
	/* Append port to the host value */
Packit Service b23acc
	if (gateway && port) {
Packit Service b23acc
		gs_free char *tmp = gateway;
Packit Service b23acc
Packit Service b23acc
		gateway = g_strdup_printf ("%s%s", tmp, port);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Fill secrets to the array */
Packit Service b23acc
	for (i = 0; i < secrets->len; i++) {
Packit Service b23acc
		NMSecretAgentSimpleSecret *secret = secrets->pdata[i];
Packit Service b23acc
Packit Service b23acc
		if (secret->secret_type != NM_SECRET_AGENT_SECRET_TYPE_VPN_SECRET)
Packit Service b23acc
			continue;
Packit Service b23acc
		if (!nm_streq0 (secret->vpn_type, NM_SECRET_AGENT_VPN_TYPE_OPENCONNECT))
Packit Service b23acc
			continue;
Packit Service b23acc
Packit Service b23acc
		if (nm_streq0 (secret->entry_id, NM_SECRET_AGENT_ENTRY_ID_PREFX_VPN_SECRETS "cookie")) {
Packit Service b23acc
			g_free (secret->value);
Packit Service b23acc
			secret->value = g_steal_pointer (&cookie);
Packit Service b23acc
		} else if (nm_streq0 (secret->entry_id, NM_SECRET_AGENT_ENTRY_ID_PREFX_VPN_SECRETS "gateway")) {
Packit Service b23acc
			g_free (secret->value);
Packit Service b23acc
			secret->value = g_steal_pointer (&gateway);
Packit Service b23acc
		} else if (nm_streq0 (secret->entry_id, NM_SECRET_AGENT_ENTRY_ID_PREFX_VPN_SECRETS "gwcert")) {
Packit Service b23acc
			g_free (secret->value);
Packit Service b23acc
			secret->value = g_steal_pointer (&gwcert);
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
get_secrets_from_user (const NmcConfig *nmc_config,
Packit Service b23acc
                       const char *request_id,
Packit Service b23acc
                       const char *title,
Packit Service b23acc
                       const char *msg,
Packit Service b23acc
                       NMConnection *connection,
Packit Service b23acc
                       gboolean ask,
Packit Service b23acc
                       GHashTable *pwds_hash,
Packit Service b23acc
                       GPtrArray *secrets)
Packit Service b23acc
{
Packit Service b23acc
	int i;
Packit Service b23acc
Packit Service b23acc
	/* Check if there is a VPN OpenConnect secret to ask for */
Packit Service b23acc
	if (ask)
Packit Service b23acc
		vpn_openconnect_get_secrets (connection, secrets);
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < secrets->len; i++) {
Packit Service b23acc
		NMSecretAgentSimpleSecret *secret = secrets->pdata[i];
Packit Service b23acc
		char *pwd = NULL;
Packit Service b23acc
Packit Service b23acc
		/* First try to find the password in provided passwords file,
Packit Service b23acc
		 * then ask user. */
Packit Service b23acc
		if (pwds_hash && (pwd = g_hash_table_lookup (pwds_hash, secret->entry_id))) {
Packit Service b23acc
			pwd = g_strdup (pwd);
Packit Service b23acc
		} else {
Packit Service b23acc
			if (ask) {
Packit Service b23acc
				gboolean echo_on;
Packit Service b23acc
Packit Service b23acc
				if (secret->value) {
Packit Service b23acc
					if (!g_strcmp0 (secret->vpn_type, NM_DBUS_INTERFACE ".openconnect")) {
Packit Service b23acc
						/* Do not present and ask user for openconnect secrets, we already have them */
Packit Service b23acc
						continue;
Packit Service b23acc
					} else {
Packit Service b23acc
						/* Prefill the password if we have it. */
Packit Service b23acc
						rl_startup_hook = nmc_rl_set_deftext;
Packit Service b23acc
						nmc_rl_pre_input_deftext = g_strdup (secret->value);
Packit Service b23acc
					}
Packit Service b23acc
				}
Packit Service b23acc
				if (msg)
Packit Service b23acc
					g_print ("%s\n", msg);
Packit Service b23acc
Packit Service b23acc
				echo_on = secret->is_secret
Packit Service b23acc
				          ? nmc_config->show_secrets
Packit Service b23acc
				          : TRUE;
Packit Service b23acc
Packit Service b23acc
				if (secret->no_prompt_entry_id)
Packit Service b23acc
					pwd = nmc_readline_echo (nmc_config, echo_on, "%s: ", secret->pretty_name);
Packit Service b23acc
				else
Packit Service b23acc
					pwd = nmc_readline_echo (nmc_config, echo_on, "%s (%s): ", secret->pretty_name, secret->entry_id);
Packit Service b23acc
Packit Service b23acc
				if (!pwd)
Packit Service b23acc
					pwd = g_strdup ("");
Packit Service b23acc
			} else {
Packit Service b23acc
				if (msg)
Packit Service b23acc
					g_print ("%s\n", msg);
Packit Service b23acc
				g_printerr (_("Warning: password for '%s' not given in 'passwd-file' "
Packit Service b23acc
				              "and nmcli cannot ask without '--ask' option.\n"),
Packit Service b23acc
				            secret->entry_id);
Packit Service b23acc
			}
Packit Service b23acc
		}
Packit Service b23acc
		/* No password provided, cancel the secrets. */
Packit Service b23acc
		if (!pwd)
Packit Service b23acc
			return FALSE;
Packit Service b23acc
		nm_free_secret (secret->value);
Packit Service b23acc
		secret->value = pwd;
Packit Service b23acc
	}
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_secrets_requested:
Packit Service b23acc
 * @agent: the #NMSecretAgentSimple
Packit Service b23acc
 * @request_id: request ID, to eventually pass to
Packit Service b23acc
 *   nm_secret_agent_simple_response()
Packit Service b23acc
 * @title: a title for the password request
Packit Service b23acc
 * @msg: a prompt message for the password request
Packit Service b23acc
 * @secrets: (element-type #NMSecretAgentSimpleSecret): array of secrets
Packit Service b23acc
 *   being requested.
Packit Service b23acc
 * @user_data: user data passed to the function
Packit Service b23acc
 *
Packit Service b23acc
 * This function is used as a callback for "request-secrets" signal of
Packit Service b23acc
 * NMSecretAgentSimpleSecret.
Packit Service b23acc
*/
Packit Service b23acc
void
Packit Service b23acc
nmc_secrets_requested (NMSecretAgentSimple *agent,
Packit Service b23acc
                       const char          *request_id,
Packit Service b23acc
                       const char          *title,
Packit Service b23acc
                       const char          *msg,
Packit Service b23acc
                       GPtrArray           *secrets,
Packit Service b23acc
                       gpointer             user_data)
Packit Service b23acc
{
Packit Service b23acc
	NmCli *nmc = (NmCli *) user_data;
Packit Service b23acc
	NMConnection *connection = NULL;
Packit Service b23acc
	char *path, *p;
Packit Service b23acc
	gboolean success = FALSE;
Packit Service b23acc
	const GPtrArray *connections;
Packit Service b23acc
Packit Service b23acc
	if (nmc->nmc_config.print_output == NMC_PRINT_PRETTY)
Packit Service b23acc
		nmc_terminal_erase_line ();
Packit Service b23acc
Packit Service b23acc
	/* Find the connection for the request */
Packit Service b23acc
	path = g_strdup (request_id);
Packit Service b23acc
	if (path) {
Packit Service b23acc
		p = strrchr (path, '/');
Packit Service b23acc
		if (p)
Packit Service b23acc
			*p = '\0';
Packit Service b23acc
		connections = nm_client_get_connections (nmc->client);
Packit Service b23acc
		connection = nmc_find_connection (connections, "path", path, NULL, FALSE);
Packit Service b23acc
		g_free (path);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	success = get_secrets_from_user (&nmc->nmc_config,
Packit Service b23acc
	                                 request_id,
Packit Service b23acc
	                                 title,
Packit Service b23acc
	                                 msg,
Packit Service b23acc
	                                 connection,
Packit Service b23acc
	                                 nmc->nmc_config.in_editor || nmc->ask,
Packit Service b23acc
	                                 nmc->pwds_hash,
Packit Service b23acc
	                                 secrets);
Packit Service b23acc
	if (success)
Packit Service b23acc
		nm_secret_agent_simple_response (agent, request_id, secrets);
Packit Service b23acc
	else {
Packit Service b23acc
		/* Unregister our secret agent on failure, so that another agent
Packit Service b23acc
		 * may be tried */
Packit Service b23acc
		if (nmc->secret_agent) {
Packit Service b23acc
			nm_secret_agent_old_unregister (NM_SECRET_AGENT_OLD (nmc->secret_agent), NULL, NULL);
Packit Service b23acc
			g_clear_object (&nmc->secret_agent);
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
char *
Packit Service b23acc
nmc_unique_connection_name (const GPtrArray *connections, const char *try_name)
Packit Service b23acc
{
Packit Service b23acc
	NMConnection *connection;
Packit Service b23acc
	const char *name;
Packit Service b23acc
	char *new_name;
Packit Service b23acc
	unsigned num = 1;
Packit Service b23acc
	int i = 0;
Packit Service b23acc
Packit Service b23acc
	new_name = g_strdup (try_name);
Packit Service b23acc
	while (i < connections->len) {
Packit Service b23acc
		connection = NM_CONNECTION (connections->pdata[i]);
Packit Service b23acc
Packit Service b23acc
		name = nm_connection_get_id (connection);
Packit Service b23acc
		if (g_strcmp0 (new_name, name) == 0) {
Packit Service b23acc
			g_free (new_name);
Packit Service b23acc
			new_name = g_strdup_printf ("%s-%d", try_name, num++);
Packit Service b23acc
			i = 0;
Packit Service b23acc
		} else
Packit Service b23acc
			i++;
Packit Service b23acc
	}
Packit Service b23acc
	return new_name;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/* readline state variables */
Packit Service b23acc
static gboolean nmcli_in_readline = FALSE;
Packit Service b23acc
static gboolean rl_got_line;
Packit Service b23acc
static char *rl_string;
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_cleanup_readline:
Packit Service b23acc
 *
Packit Service b23acc
 * Cleanup readline when nmcli is terminated with a signal.
Packit Service b23acc
 * It makes sure the terminal is not garbled.
Packit Service b23acc
 */
Packit Service b23acc
void
Packit Service b23acc
nmc_cleanup_readline (void)
Packit Service b23acc
{
Packit Service b23acc
	rl_free_line_state ();
Packit Service b23acc
	rl_cleanup_after_signal ();
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean
Packit Service b23acc
nmc_get_in_readline (void)
Packit Service b23acc
{
Packit Service b23acc
	return nmcli_in_readline;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
void
Packit Service b23acc
nmc_set_in_readline (gboolean in_readline)
Packit Service b23acc
{
Packit Service b23acc
	nmcli_in_readline = in_readline;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
readline_cb (char *line)
Packit Service b23acc
{
Packit Service b23acc
	rl_got_line = TRUE;
Packit Service b23acc
	rl_string = line;
Packit Service b23acc
	rl_callback_handler_remove ();
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
stdin_ready_cb (int fd,
Packit Service b23acc
                GIOCondition condition,
Packit Service b23acc
                gpointer data)
Packit Service b23acc
{
Packit Service b23acc
	rl_callback_read_char ();
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static char *
Packit Service b23acc
nmc_readline_helper (const NmcConfig *nmc_config,
Packit Service b23acc
                     const char *prompt)
Packit Service b23acc
{
Packit Service b23acc
	GSource *io_source;
Packit Service b23acc
Packit Service b23acc
	nmc_set_in_readline (TRUE);
Packit Service b23acc
Packit Service b23acc
	io_source = nm_g_unix_fd_source_new (STDIN_FILENO,
Packit Service b23acc
	                                     G_IO_IN,
Packit Service b23acc
	                                     G_PRIORITY_DEFAULT,
Packit Service b23acc
	                                     stdin_ready_cb,
Packit Service b23acc
	                                     NULL,
Packit Service b23acc
	                                     NULL);
Packit Service b23acc
	g_source_attach (io_source, NULL);
Packit Service b23acc
Packit Service b23acc
read_again:
Packit Service b23acc
	rl_string = NULL;
Packit Service b23acc
	rl_got_line = FALSE;
Packit Service b23acc
	rl_callback_handler_install (prompt, readline_cb);
Packit Service b23acc
Packit Service b23acc
	while (   !rl_got_line
Packit Service b23acc
	       && g_main_loop_is_running (loop)
Packit Service b23acc
	       && !nmc_seen_sigint ())
Packit Service b23acc
		g_main_context_iteration (NULL, TRUE);
Packit Service b23acc
Packit Service b23acc
	/* If Ctrl-C was detected, complete the line */
Packit Service b23acc
	if (nmc_seen_sigint ()) {
Packit Service b23acc
		rl_echo_signal_char (SIGINT);
Packit Service b23acc
		if (!rl_got_line) {
Packit Service b23acc
			rl_stuff_char ('\n');
Packit Service b23acc
			rl_callback_read_char ();
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Add string to the history */
Packit Service b23acc
	if (rl_string && *rl_string)
Packit Service b23acc
		add_history (rl_string);
Packit Service b23acc
Packit Service b23acc
	if (nmc_seen_sigint ()) {
Packit Service b23acc
		/* Ctrl-C */
Packit Service b23acc
		nmc_clear_sigint ();
Packit Service b23acc
		if (   nmc_config->in_editor
Packit Service b23acc
		    || (rl_string  && *rl_string)) {
Packit Service b23acc
			/* In editor, or the line is not empty */
Packit Service b23acc
			/* Call readline again to get new prompt (repeat) */
Packit Service b23acc
			g_free (rl_string);
Packit Service b23acc
			goto read_again;
Packit Service b23acc
		} else {
Packit Service b23acc
			/* Not in editor and line is empty, exit */
Packit Service b23acc
			nmc_exit ();
Packit Service b23acc
		}
Packit Service b23acc
	} else if (!rl_string) {
Packit Service b23acc
		/* Ctrl-D, exit */
Packit Service b23acc
		nmc_exit ();
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Return NULL, not empty string */
Packit Service b23acc
	if (rl_string && *rl_string == '\0') {
Packit Service b23acc
		g_free (rl_string);
Packit Service b23acc
		rl_string = NULL;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source_inst (&io_source);
Packit Service b23acc
Packit Service b23acc
	nmc_set_in_readline (FALSE);
Packit Service b23acc
Packit Service b23acc
	return rl_string;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_readline:
Packit Service b23acc
 * @prompt_fmt: prompt to print (telling user what to enter). It is standard
Packit Service b23acc
 *   printf() format string
Packit Service b23acc
 * @...: a list of arguments according to the @prompt_fmt format string
Packit Service b23acc
 *
Packit Service b23acc
 * Wrapper around libreadline's readline() function.
Packit Service b23acc
 * If user pressed Ctrl-C, readline() is called again (if not in editor and
Packit Service b23acc
 * line is empty, nmcli will quit).
Packit Service b23acc
 * If user pressed Ctrl-D on empty line, nmcli will quit.
Packit Service b23acc
 *
Packit Service b23acc
 * Returns: the user provided string. In case the user entered empty string,
Packit Service b23acc
 * this function returns NULL.
Packit Service b23acc
 */
Packit Service b23acc
char *
Packit Service b23acc
nmc_readline (const NmcConfig *nmc_config,
Packit Service b23acc
              const char *prompt_fmt,
Packit Service b23acc
              ...)
Packit Service b23acc
{
Packit Service b23acc
	va_list args;
Packit Service b23acc
	gs_free char *prompt = NULL;
Packit Service b23acc
Packit Service b23acc
	rl_initialize ();
Packit Service b23acc
Packit Service b23acc
	va_start (args, prompt_fmt);
Packit Service b23acc
	prompt = g_strdup_vprintf (prompt_fmt, args);
Packit Service b23acc
	va_end (args);
Packit Service b23acc
	return nmc_readline_helper (nmc_config, prompt);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
nmc_secret_redisplay (void)
Packit Service b23acc
{
Packit Service b23acc
	int save_point = rl_point;
Packit Service b23acc
	int save_end = rl_end;
Packit Service b23acc
	char *save_line_buffer = rl_line_buffer;
Packit Service b23acc
	const char *subst = nmc_password_subst_char ();
Packit Service b23acc
	int subst_len = strlen (subst);
Packit Service b23acc
	int i;
Packit Service b23acc
Packit Service b23acc
	rl_point = g_utf8_strlen (save_line_buffer, save_point) * subst_len;
Packit Service b23acc
	rl_end = g_utf8_strlen (rl_line_buffer, -1) * subst_len;
Packit Service b23acc
	rl_line_buffer = g_slice_alloc (rl_end + 1);
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i + subst_len <= rl_end; i += subst_len)
Packit Service b23acc
		memcpy (&rl_line_buffer[i], subst, subst_len);
Packit Service b23acc
	rl_line_buffer[i] = '\0';
Packit Service b23acc
Packit Service b23acc
	rl_redisplay ();
Packit Service b23acc
	g_slice_free1 (rl_end + 1, rl_line_buffer);
Packit Service b23acc
	rl_line_buffer = save_line_buffer;
Packit Service b23acc
	rl_end = save_end;
Packit Service b23acc
	rl_point = save_point;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_readline_echo:
Packit Service b23acc
 *
Packit Service b23acc
 * The same as nmc_readline() except it can disable echoing of input characters if @echo_on is %FALSE.
Packit Service b23acc
 * nmc_readline(TRUE, ...) == nmc_readline(...)
Packit Service b23acc
 */
Packit Service b23acc
char *
Packit Service b23acc
nmc_readline_echo (const NmcConfig *nmc_config,
Packit Service b23acc
                   gboolean echo_on,
Packit Service b23acc
                   const char *prompt_fmt,
Packit Service b23acc
                   ...)
Packit Service b23acc
{
Packit Service b23acc
	va_list args;
Packit Service b23acc
	gs_free char *prompt = NULL;
Packit Service b23acc
	char *str;
Packit Service b23acc
	nm_auto_free HISTORY_STATE *saved_history = NULL;
Packit Service b23acc
	HISTORY_STATE passwd_history = { 0, };
Packit Service b23acc
Packit Service b23acc
	va_start (args, prompt_fmt);
Packit Service b23acc
	prompt = g_strdup_vprintf (prompt_fmt, args);
Packit Service b23acc
	va_end (args);
Packit Service b23acc
Packit Service b23acc
	rl_initialize ();
Packit Service b23acc
Packit Service b23acc
	/* Hide the actual password */
Packit Service b23acc
	if (!echo_on) {
Packit Service b23acc
		saved_history = history_get_history_state ();
Packit Service b23acc
		history_set_history_state (&passwd_history);
Packit Service b23acc
		/* stifling history is important as it tells readline to
Packit Service b23acc
		 * not store anything, otherwise sensitive data could be
Packit Service b23acc
		 * leaked */
Packit Service b23acc
		stifle_history (0);
Packit Service b23acc
		rl_redisplay_function = nmc_secret_redisplay;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	str = nmc_readline_helper (nmc_config, prompt);
Packit Service b23acc
Packit Service b23acc
	/* Restore the non-hiding behavior */
Packit Service b23acc
	if (!echo_on) {
Packit Service b23acc
		rl_redisplay_function = rl_redisplay;
Packit Service b23acc
		history_set_history_state (saved_history);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return str;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_rl_gen_func_basic:
Packit Service b23acc
 * @text: text to complete
Packit Service b23acc
 * @state: readline state; says whether start from scratch (state == 0)
Packit Service b23acc
 * @words: strings for completion
Packit Service b23acc
 *
Packit Service b23acc
 * Basic function generating list of completion strings for readline.
Packit Service b23acc
 * See e.g. http://cnswww.cns.cwru.edu/php/chet/readline/readline.html#SEC49
Packit Service b23acc
 */
Packit Service b23acc
char *
Packit Service b23acc
nmc_rl_gen_func_basic (const char *text, int state, const char *const*words)
Packit Service b23acc
{
Packit Service b23acc
	static int list_idx, len;
Packit Service b23acc
	const char *name;
Packit Service b23acc
Packit Service b23acc
	if (!state) {
Packit Service b23acc
		list_idx = 0;
Packit Service b23acc
		len = strlen (text);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Return the next name which partially matches one from the 'words' list. */
Packit Service b23acc
	while ((name = words[list_idx])) {
Packit Service b23acc
		list_idx++;
Packit Service b23acc
Packit Service b23acc
		if (strncmp (name, text, len) == 0)
Packit Service b23acc
			return g_strdup (name);
Packit Service b23acc
	}
Packit Service b23acc
	return NULL;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static struct {
Packit Service b23acc
	bool initialized;
Packit Service b23acc
	guint idx;
Packit Service b23acc
	char **values;
Packit Service b23acc
} _rl_compentry_func_wrap = { 0 };
Packit Service b23acc
Packit Service b23acc
static char *
Packit Service b23acc
_rl_compentry_func_wrap_fcn (const char *text, int state)
Packit Service b23acc
{
Packit Service b23acc
	g_return_val_if_fail (_rl_compentry_func_wrap.initialized, NULL);
Packit Service b23acc
Packit Service b23acc
	while (   _rl_compentry_func_wrap.values
Packit Service b23acc
	       && _rl_compentry_func_wrap.values[_rl_compentry_func_wrap.idx]
Packit Service b23acc
	       && !g_str_has_prefix (_rl_compentry_func_wrap.values[_rl_compentry_func_wrap.idx], text))
Packit Service b23acc
		_rl_compentry_func_wrap.idx++;
Packit Service b23acc
Packit Service b23acc
	if (   !_rl_compentry_func_wrap.values
Packit Service b23acc
	    || !_rl_compentry_func_wrap.values[_rl_compentry_func_wrap.idx]) {
Packit Service b23acc
		g_strfreev (_rl_compentry_func_wrap.values);
Packit Service b23acc
		_rl_compentry_func_wrap.values = NULL;
Packit Service b23acc
		_rl_compentry_func_wrap.initialized = FALSE;
Packit Service b23acc
		return NULL;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return g_strdup (_rl_compentry_func_wrap.values[_rl_compentry_func_wrap.idx++]);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
NmcCompEntryFunc
Packit Service b23acc
nmc_rl_compentry_func_wrap (const char *const*values)
Packit Service b23acc
{
Packit Service b23acc
	g_strfreev (_rl_compentry_func_wrap.values);
Packit Service b23acc
	_rl_compentry_func_wrap.values = g_strdupv ((char **) values);
Packit Service b23acc
	_rl_compentry_func_wrap.idx = 0;
Packit Service b23acc
	_rl_compentry_func_wrap.initialized = TRUE;
Packit Service b23acc
	return _rl_compentry_func_wrap_fcn;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
char *
Packit Service b23acc
nmc_rl_gen_func_ifnames (const char *text, int state)
Packit Service b23acc
{
Packit Service b23acc
	int i;
Packit Service b23acc
	const GPtrArray *devices;
Packit Service b23acc
	const char **ifnames;
Packit Service b23acc
	char *ret;
Packit Service b23acc
Packit Service b23acc
	devices = nm_client_get_devices (nm_cli_global_readline->client);
Packit Service b23acc
	if (devices->len == 0)
Packit Service b23acc
		return NULL;
Packit Service b23acc
Packit Service b23acc
	ifnames = g_new (const char *, devices->len + 1);
Packit Service b23acc
	for (i = 0; i < devices->len; i++) {
Packit Service b23acc
		NMDevice *dev = g_ptr_array_index (devices, i);
Packit Service b23acc
		const char *ifname = nm_device_get_iface (dev);
Packit Service b23acc
		ifnames[i] = ifname;
Packit Service b23acc
	}
Packit Service b23acc
	ifnames[i] = NULL;
Packit Service b23acc
Packit Service b23acc
	ret = nmc_rl_gen_func_basic (text, state, ifnames);
Packit Service b23acc
Packit Service b23acc
	g_free (ifnames);
Packit Service b23acc
	return ret;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/* for pre-filling a string to readline prompt */
Packit Service b23acc
char *nmc_rl_pre_input_deftext;
Packit Service b23acc
Packit Service b23acc
int
Packit Service b23acc
nmc_rl_set_deftext (void)
Packit Service b23acc
{
Packit Service b23acc
	if (nmc_rl_pre_input_deftext && rl_startup_hook) {
Packit Service b23acc
		rl_insert_text (nmc_rl_pre_input_deftext);
Packit Service b23acc
		g_free (nmc_rl_pre_input_deftext);
Packit Service b23acc
		nmc_rl_pre_input_deftext = NULL;
Packit Service b23acc
		rl_startup_hook = NULL;
Packit Service b23acc
	}
Packit Service b23acc
	return 0;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_parse_lldp_capabilities:
Packit Service b23acc
 * @value: the capabilities value
Packit Service b23acc
 *
Packit Service b23acc
 * Parses LLDP capabilities flags
Packit Service b23acc
 *
Packit Service b23acc
 * Returns: a newly allocated string containing capabilities names separated by commas.
Packit Service b23acc
 */
Packit Service b23acc
char *
Packit Service b23acc
nmc_parse_lldp_capabilities (guint value)
Packit Service b23acc
{
Packit Service b23acc
	/* IEEE Std 802.1AB-2009 - Table 8.4 */
Packit Service b23acc
	const char *names[] = { "other", "repeater", "mac-bridge", "wlan-access-point",
Packit Service b23acc
	                        "router", "telephone", "docsis-cable-device", "station-only",
Packit Service b23acc
	                        "c-vlan-component", "s-vlan-component", "tpmr" };
Packit Service b23acc
	gboolean first = TRUE;
Packit Service b23acc
	GString *str;
Packit Service b23acc
	int i;
Packit Service b23acc
Packit Service b23acc
	if (!value)
Packit Service b23acc
		return g_strdup ("none");
Packit Service b23acc
Packit Service b23acc
	str = g_string_new ("");
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < G_N_ELEMENTS (names); i++) {
Packit Service b23acc
		if (value & (1 << i)) {
Packit Service b23acc
			if (!first)
Packit Service b23acc
				g_string_append_c (str, ',');
Packit Service b23acc
Packit Service b23acc
			first = FALSE;
Packit Service b23acc
			value &= ~(1 << i);
Packit Service b23acc
			g_string_append (str, names[i]);
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (value) {
Packit Service b23acc
		if (!first)
Packit Service b23acc
			g_string_append_c (str, ',');
Packit Service b23acc
		g_string_append (str, "reserved");
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return g_string_free (str, FALSE);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
command_done (GObject *object, GAsyncResult *res, gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	GTask *task = G_TASK (res);
Packit Service b23acc
	NmCli *nmc = user_data;
Packit Service b23acc
	gs_free_error GError *error = NULL;
Packit Service b23acc
Packit Service b23acc
	if (!g_task_propagate_boolean (task, &error)) {
Packit Service b23acc
		nmc->return_value = error->code;
Packit Service b23acc
		g_string_assign (nmc->return_text, error->message);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (!nmc->should_wait)
Packit Service b23acc
		g_main_loop_quit (loop);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
typedef struct {
Packit Service b23acc
	const NMCCommand *cmd;
Packit Service b23acc
	int argc;
Packit Service b23acc
	char **argv;
Packit Service b23acc
	GTask *task;
Packit Service b23acc
} CmdCall;
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
call_cmd (NmCli *nmc, GTask *task, const NMCCommand *cmd, int argc, const char *const*argv);
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
got_client (GObject *source_object, GAsyncResult *res, gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	gs_unref_object GTask *task = NULL;
Packit Service b23acc
	gs_free_error GError *error = NULL;
Packit Service b23acc
	CmdCall *call = user_data;
Packit Service b23acc
	NmCli *nmc;
Packit Service b23acc
Packit Service b23acc
	nm_assert (NM_IS_CLIENT (source_object));
Packit Service b23acc
Packit Service b23acc
	task = g_steal_pointer (&call->task);
Packit Service b23acc
	nmc = g_task_get_task_data (task);
Packit Service b23acc
Packit Service b23acc
	nmc->should_wait--;
Packit Service b23acc
Packit Service b23acc
	if (!g_async_initable_init_finish (G_ASYNC_INITABLE (source_object),
Packit Service b23acc
	                                   res,
Packit Service b23acc
	                                   &error)) {
Packit Service b23acc
		g_object_unref (source_object);
Packit Service b23acc
		g_task_return_new_error (task, NMCLI_ERROR, NMC_RESULT_ERROR_UNKNOWN,
Packit Service b23acc
		                         _("Error: Could not create NMClient object: %s."),
Packit Service b23acc
		                         error->message);
Packit Service b23acc
	} else {
Packit Service b23acc
		nmc->client = NM_CLIENT (source_object);
Packit Service b23acc
		call_cmd (nmc, g_steal_pointer (&task), call->cmd, call->argc, (const char *const*) call->argv);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	g_strfreev (call->argv);
Packit Service b23acc
	nm_g_slice_free (call);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
call_cmd (NmCli *nmc, GTask *task, const NMCCommand *cmd, int argc, const char *const*argv)
Packit Service b23acc
{
Packit Service b23acc
	CmdCall *call;
Packit Service b23acc
Packit Service b23acc
	if (nmc->client || !cmd->needs_client) {
Packit Service b23acc
Packit Service b23acc
		/* Check whether NetworkManager is running */
Packit Service b23acc
		if (cmd->needs_nm_running && !nm_client_get_nm_running (nmc->client)) {
Packit Service b23acc
			g_task_return_new_error (task, NMCLI_ERROR, NMC_RESULT_ERROR_NM_NOT_RUNNING,
Packit Service b23acc
			                         _("Error: NetworkManager is not running."));
Packit Service b23acc
		} else {
Packit Service b23acc
			cmd->func (cmd, nmc, argc, argv);
Packit Service b23acc
			g_task_return_boolean (task, TRUE);
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		g_object_unref (task);
Packit Service b23acc
	} else {
Packit Service b23acc
		nm_assert (nmc->client == NULL);
Packit Service b23acc
Packit Service b23acc
		nmc->should_wait++;
Packit Service b23acc
		call = g_slice_new (CmdCall);
Packit Service b23acc
		*call = (CmdCall) {
Packit Service b23acc
			.cmd  = cmd,
Packit Service b23acc
			.argc = argc,
Packit Service b23acc
			.argv = nm_utils_strv_dup ((char **) argv, argc, TRUE),
Packit Service b23acc
			.task = task,
Packit Service b23acc
		};
Packit Service b23acc
		nmc_client_new_async (NULL,
Packit Service b23acc
		                      got_client,
Packit Service b23acc
		                      call,
Packit Service b23acc
		                      NM_CLIENT_INSTANCE_FLAGS, (guint) NM_CLIENT_INSTANCE_FLAGS_NO_AUTO_FETCH_PERMISSIONS,
Packit Service b23acc
		                      NULL);
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
nmc_complete_help (const char *prefix)
Packit Service b23acc
{
Packit Service b23acc
	nmc_complete_strings (prefix, "help");
Packit Service b23acc
	if (*prefix == '-')
Packit Service b23acc
		nmc_complete_strings (prefix, "-help", "--help");
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_do_cmd:
Packit Service b23acc
 * @nmc: Client instance
Packit Service b23acc
 * @cmds: Command table
Packit Service b23acc
 * @cmd: Command
Packit Service b23acc
 * @argc: Argument count
Packit Service b23acc
 * @argv: Arguments vector. Must be a global variable.
Packit Service b23acc
 *
Packit Service b23acc
 * Picks the right callback to handle command from the command table.
Packit Service b23acc
 * If --help argument follows and the usage callback is specified for the command
Packit Service b23acc
 * it calls the usage callback.
Packit Service b23acc
 *
Packit Service b23acc
 * The command table is terminated with a %NULL command. The terminating
Packit Service b23acc
 * entry's handlers are called if the command is empty.
Packit Service b23acc
 *
Packit Service b23acc
 * The argument vector needs to be a pointer to the global arguments vector that is
Packit Service b23acc
 * never freed, since the command handler will be called asynchronously and there's
Packit Service b23acc
 * no callback to free the memory in (for simplicity).
Packit Service b23acc
 */
Packit Service b23acc
void
Packit Service b23acc
nmc_do_cmd (NmCli *nmc, const NMCCommand cmds[], const char *cmd, int argc, const char *const*argv)
Packit Service b23acc
{
Packit Service b23acc
	const NMCCommand *c;
Packit Service b23acc
	gs_unref_object GTask *task = NULL;
Packit Service b23acc
Packit Service b23acc
	task = nm_g_task_new (NULL, NULL, nmc_do_cmd, command_done, nmc);
Packit Service b23acc
	g_task_set_task_data (task, nmc, NULL);
Packit Service b23acc
Packit Service b23acc
	if (argc == 0 && nmc->complete) {
Packit Service b23acc
		g_task_return_boolean (task, TRUE);
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (argc == 1 && nmc->complete) {
Packit Service b23acc
		for (c = cmds; c->cmd; ++c) {
Packit Service b23acc
			if (!*cmd || matches (cmd, c->cmd))
Packit Service b23acc
				g_print ("%s\n", c->cmd);
Packit Service b23acc
		}
Packit Service b23acc
		nmc_complete_help (cmd);
Packit Service b23acc
		g_task_return_boolean (task, TRUE);
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	for (c = cmds; c->cmd; ++c) {
Packit Service b23acc
		if (cmd && matches (cmd, c->cmd))
Packit Service b23acc
			break;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (c->cmd) {
Packit Service b23acc
		/* A valid command was specified. */
Packit Service b23acc
		if (c->usage && argc == 2 && nmc->complete)
Packit Service b23acc
			nmc_complete_help (*(argv+1));
Packit Service b23acc
		if (!nmc->complete && c->usage && nmc_arg_is_help (*(argv+1))) {
Packit Service b23acc
			c->usage ();
Packit Service b23acc
			g_task_return_boolean (task, TRUE);
Packit Service b23acc
		} else {
Packit Service b23acc
			call_cmd (nmc, g_steal_pointer (&task), c, argc, (const char *const*) argv);
Packit Service b23acc
		}
Packit Service b23acc
	} else if (cmd) {
Packit Service b23acc
		/* Not a known command. */
Packit Service b23acc
		if (nmc_arg_is_help (cmd) && c->usage) {
Packit Service b23acc
			c->usage ();
Packit Service b23acc
			g_task_return_boolean (task, TRUE);
Packit Service b23acc
		} else {
Packit Service b23acc
			g_task_return_new_error (task, NMCLI_ERROR, NMC_RESULT_ERROR_USER_INPUT,
Packit Service b23acc
			                         _("Error: argument '%s' not understood. Try passing --help instead."), cmd);
Packit Service b23acc
		}
Packit Service b23acc
	} else if (c->func) {
Packit Service b23acc
		/* No command, run the default handler. */
Packit Service b23acc
		call_cmd (nmc, g_steal_pointer (&task), c, argc, (const char *const*) argv);
Packit Service b23acc
	} else {
Packit Service b23acc
		/* No command and no default handler. */
Packit Service b23acc
		g_task_return_new_error (task, NMCLI_ERROR, NMC_RESULT_ERROR_USER_INPUT,
Packit Service b23acc
		                         _("Error: missing argument. Try passing --help."));
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_complete_strings:
Packit Service b23acc
 * @prefix: a string to match
Packit Service b23acc
 * @nargs: the number of elements in @args. Or -1 if @args is a NULL terminated
Packit Service b23acc
 *   strv array.
Packit Service b23acc
 * @args: the argument list. If @nargs is not -1, then some elements may
Packit Service b23acc
 *   be %NULL to indicate to silently skip the values.
Packit Service b23acc
 *
Packit Service b23acc
 * Prints all the matching candidates for completion. Useful when there's
Packit Service b23acc
 * no better way to suggest completion other than a hardcoded string list.
Packit Service b23acc
 */
Packit Service b23acc
void
Packit Service b23acc
nmc_complete_strv (const char *prefix, gssize nargs, const char *const*args)
Packit Service b23acc
{
Packit Service b23acc
	gsize i, n;
Packit Service b23acc
Packit Service b23acc
	if (prefix && !prefix[0])
Packit Service b23acc
		prefix = NULL;
Packit Service b23acc
Packit Service b23acc
	if (nargs < 0) {
Packit Service b23acc
		nm_assert (nargs == -1);
Packit Service b23acc
		n = NM_PTRARRAY_LEN (args);
Packit Service b23acc
	} else
Packit Service b23acc
		n = (gsize) nargs;
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < n; i++) {
Packit Service b23acc
		const char *candidate = args[i];
Packit Service b23acc
Packit Service b23acc
		if (!candidate)
Packit Service b23acc
			continue;
Packit Service b23acc
		if (   prefix
Packit Service b23acc
		    && !matches (prefix, candidate))
Packit Service b23acc
			continue;
Packit Service b23acc
Packit Service b23acc
		g_print ("%s\n", candidate);
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_complete_bool:
Packit Service b23acc
 * @prefix: a string to match
Packit Service b23acc
 * @...: a %NULL-terminated list of candidate strings
Packit Service b23acc
 *
Packit Service b23acc
 * Prints all the matching possible boolean values for completion.
Packit Service b23acc
 */
Packit Service b23acc
void
Packit Service b23acc
nmc_complete_bool (const char *prefix)
Packit Service b23acc
{
Packit Service b23acc
	nmc_complete_strings (prefix, "true", "yes", "on",
Packit Service b23acc
	                              "false", "no", "off");
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/**
Packit Service b23acc
 * nmc_error_get_simple_message:
Packit Service b23acc
 * @error: a GError
Packit Service b23acc
 *
Packit Service b23acc
 * Returns a simplified message for some errors hard to understand.
Packit Service b23acc
 */
Packit Service b23acc
const char *
Packit Service b23acc
nmc_error_get_simple_message (GError *error)
Packit Service b23acc
{
Packit Service b23acc
	/* Return a clear message instead of the obscure D-Bus policy error */
Packit Service b23acc
	if (g_error_matches (error, G_DBUS_ERROR, G_DBUS_ERROR_ACCESS_DENIED))
Packit Service b23acc
		return _("access denied");
Packit Service b23acc
	if (g_error_matches (error, G_DBUS_ERROR, G_DBUS_ERROR_SERVICE_UNKNOWN))
Packit Service b23acc
		return _("NetworkManager is not running");
Packit Service b23acc
	else
Packit Service b23acc
		return error->message;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
GVariant *
Packit Service b23acc
nmc_dbus_call_sync (NmCli *nmc,
Packit Service b23acc
                    const char *object_path,
Packit Service b23acc
                    const char *interface_name,
Packit Service b23acc
                    const char *method_name,
Packit Service b23acc
                    GVariant *parameters,
Packit Service b23acc
                    const GVariantType *reply_type,
Packit Service b23acc
                    GError **error)
Packit Service b23acc
{
Packit Service b23acc
	gs_unref_object GDBusConnection *connection = NULL;
Packit Service b23acc
	gs_free_error GError *local = NULL;
Packit Service b23acc
	GVariant *result;
Packit Service b23acc
Packit Service b23acc
	if (nmc->timeout == -1)
Packit Service b23acc
		nmc->timeout = 90;
Packit Service b23acc
Packit Service b23acc
	connection = g_bus_get_sync (G_BUS_TYPE_SYSTEM, NULL, &local);
Packit Service b23acc
	if (!connection) {
Packit Service b23acc
		g_set_error (error,
Packit Service b23acc
		             NMCLI_ERROR,
Packit Service b23acc
		             NMC_RESULT_ERROR_UNKNOWN,
Packit Service b23acc
		             _("Error: error connecting to system bus: %s"),
Packit Service b23acc
		             local->message);
Packit Service b23acc
		return NULL;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	result = g_dbus_connection_call_sync (connection,
Packit Service b23acc
	                                      "org.freedesktop.NetworkManager",
Packit Service b23acc
	                                      object_path,
Packit Service b23acc
	                                      interface_name,
Packit Service b23acc
	                                      method_name,
Packit Service b23acc
	                                      parameters,
Packit Service b23acc
	                                      reply_type,
Packit Service b23acc
	                                      G_DBUS_CALL_FLAGS_NONE,
Packit Service b23acc
	                                      nmc->timeout * 1000,
Packit Service b23acc
	                                      NULL,
Packit Service b23acc
	                                      error);
Packit Service b23acc
Packit Service b23acc
	if (error && *error)
Packit Service b23acc
		g_dbus_error_strip_remote_error (*error);
Packit Service b23acc
Packit Service b23acc
	return result;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
NM_UTILS_LOOKUP_STR_DEFINE (nm_connectivity_to_string, NMConnectivityState,
Packit Service b23acc
	NM_UTILS_LOOKUP_DEFAULT (N_("unknown")),
Packit Service b23acc
	NM_UTILS_LOOKUP_ITEM (NM_CONNECTIVITY_NONE,    N_("none")),
Packit Service b23acc
	NM_UTILS_LOOKUP_ITEM (NM_CONNECTIVITY_PORTAL,  N_("portal")),
Packit Service b23acc
	NM_UTILS_LOOKUP_ITEM (NM_CONNECTIVITY_LIMITED, N_("limited")),
Packit Service b23acc
	NM_UTILS_LOOKUP_ITEM (NM_CONNECTIVITY_FULL,    N_("full")),
Packit Service b23acc
	NM_UTILS_LOOKUP_ITEM_IGNORE (NM_CONNECTIVITY_UNKNOWN),
Packit Service b23acc
);