/* SPDX-License-Identifier: GPL-2.0-or-later */
/*
* Copyright (C) 2012, 2013 Red Hat, Inc.
*/
/**
* SECTION:nm-editor-utils
* @short_description: Miscellaneous connection editor utilities
*
* nm-editor-utils contains helper functions for connection editors.
* The goal is that this should eventually be shared between nmtui,
* nm-connection-editor, and gnome-control-center.
*/
#include "libnm-client-aux-extern/nm-default-client.h"
#include "nm-editor-utils.h"
#if 0
#include "libnmc-base/nm-vpn-helpers.h"
static GSList *vpn_plugins;
static int
sort_vpn_plugins (gconstpointer a, gconstpointer b)
{
NMVpnEditorPlugin *aa = NM_VPN_EDITOR_PLUGIN (a);
NMVpnEditorPlugin *bb = NM_VPN_EDITOR_PLUGIN (b);
char *aa_desc = NULL, *bb_desc = NULL;
int ret;
g_object_get (aa, NM_VPN_EDITOR_PLUGIN_NAME, &aa_desc, NULL);
g_object_get (bb, NM_VPN_EDITOR_PLUGIN_NAME, &bb_desc, NULL);
ret = g_strcmp0 (aa_desc, bb_desc);
g_free (aa_desc);
g_free (bb_desc);
return ret;
}
#endif
static void
wifi_connection_setup_func(NMConnection *connection, NMSettingConnection *s_con, NMSetting *s_hw)
{
g_object_set(G_OBJECT(s_hw), NM_SETTING_WIRELESS_MODE, NM_SETTING_WIRELESS_MODE_INFRA, NULL);
}
static void
bond_connection_setup_func(NMConnection *connection, NMSettingConnection *s_con, NMSetting *s_hw)
{
NMSettingBond * s_bond = NM_SETTING_BOND(s_hw);
guint i;
const char * value;
static const char *const options[] = {
NM_SETTING_BOND_OPTION_MODE,
NM_SETTING_BOND_OPTION_MIIMON,
NM_SETTING_BOND_OPTION_DOWNDELAY,
NM_SETTING_BOND_OPTION_UPDELAY,
};
for (i = 0; i < G_N_ELEMENTS(options); i++) {
value = nm_setting_bond_get_option_default(s_bond, options[i]);
if (value)
nm_setting_bond_add_option(s_bond, options[i], value);
}
}
typedef void (*NMEditorNewConnectionSetupFunc)(NMConnection * connection,
NMSettingConnection *s_con,
NMSetting * s_hw);
typedef struct {
NMEditorConnectionTypeData data;
const char * id_format;
NMEditorNewConnectionSetupFunc connection_setup_func;
gboolean no_autoconnect;
} NMEditorConnectionTypeDataReal;
static int
sort_types(gconstpointer a, gconstpointer b)
{
NMEditorConnectionTypeData *typea = *(NMEditorConnectionTypeData **) a;
NMEditorConnectionTypeData *typeb = *(NMEditorConnectionTypeData **) b;
if (typea->virtual && !typeb->virtual)
return 1;
else if (typeb->virtual && !typea->virtual)
return -1;
if (typea->setting_type == NM_TYPE_SETTING_VPN && typeb->setting_type != NM_TYPE_SETTING_VPN)
return 1;
else if (typeb->setting_type == NM_TYPE_SETTING_VPN
&& typea->setting_type != NM_TYPE_SETTING_VPN)
return -1;
return g_utf8_collate(typea->name, typeb->name);
}
/**
* nm_editor_utils_get_connection_type_list:
*
* Gets an array of information about supported connection types. The
* array is sorted in a standard presentation order (hardware types
* first, alphabetized, then virtual types, alphabetized, then VPN
* types, alphabetized).
*
* Returns: the array of connection type information
*/
NMEditorConnectionTypeData **
nm_editor_utils_get_connection_type_list(void)
{
GPtrArray * array;
NMEditorConnectionTypeDataReal * item;
static NMEditorConnectionTypeData **list;
#if 0
GHashTable *vpn_plugins_hash;
gboolean have_vpn_plugins;
#endif
if (list)
return list;
array = g_ptr_array_new();
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("Ethernet");
item->data.setting_type = NM_TYPE_SETTING_WIRED;
item->data.device_type = NM_TYPE_DEVICE_ETHERNET;
item->data.virtual = FALSE;
item->id_format = _("Ethernet connection %d");
g_ptr_array_add(array, item);
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("Veth");
item->data.setting_type = NM_TYPE_SETTING_VETH;
item->data.device_type = NM_TYPE_DEVICE_VETH;
item->data.virtual = TRUE;
item->id_format = _("Veth connection %d");
g_ptr_array_add(array, item);
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("Wi-Fi");
item->data.setting_type = NM_TYPE_SETTING_WIRELESS;
item->data.device_type = NM_TYPE_DEVICE_WIFI;
item->data.virtual = FALSE;
item->id_format = _("Wi-Fi connection %d");
item->connection_setup_func = wifi_connection_setup_func;
g_ptr_array_add(array, item);
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("InfiniBand");
item->data.setting_type = NM_TYPE_SETTING_INFINIBAND;
item->data.device_type = NM_TYPE_DEVICE_INFINIBAND;
item->data.virtual = FALSE;
item->id_format = _("InfiniBand connection %d");
g_ptr_array_add(array, item);
#if 0
item = g_new0 (NMEditorConnectionTypeDataReal, 1);
item->data.name = _("Mobile Broadband");
item->data.setting_type = NM_TYPE_SETTING_GSM;
item->data.virtual = FALSE;
item->id_format = _("Mobile broadband connection %d");
item->no_autoconnect = TRUE;
g_ptr_array_add (array, item);
#endif
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("DSL");
item->data.setting_type = NM_TYPE_SETTING_PPPOE;
item->data.device_type = NM_TYPE_DEVICE_ETHERNET;
item->data.virtual = FALSE;
item->id_format = _("DSL connection %d");
item->no_autoconnect = TRUE;
g_ptr_array_add(array, item);
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("Bond");
item->data.setting_type = NM_TYPE_SETTING_BOND;
item->data.device_type = NM_TYPE_DEVICE_BOND;
item->data.virtual = TRUE;
item->id_format = _("Bond connection %d");
item->connection_setup_func = bond_connection_setup_func;
g_ptr_array_add(array, item);
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("Bridge");
item->data.setting_type = NM_TYPE_SETTING_BRIDGE;
item->data.slave_setting_type = NM_TYPE_SETTING_BRIDGE_PORT;
item->data.device_type = NM_TYPE_DEVICE_BRIDGE;
item->data.virtual = TRUE;
item->id_format = _("Bridge connection %d");
g_ptr_array_add(array, item);
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("Team");
item->data.setting_type = NM_TYPE_SETTING_TEAM;
item->data.slave_setting_type = NM_TYPE_SETTING_TEAM_PORT;
item->data.device_type = NM_TYPE_DEVICE_TEAM;
item->data.virtual = TRUE;
item->id_format = _("Team connection %d");
g_ptr_array_add(array, item);
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("VLAN");
item->data.setting_type = NM_TYPE_SETTING_VLAN;
item->data.device_type = NM_TYPE_DEVICE_VLAN;
item->data.virtual = TRUE;
item->id_format = _("VLAN connection %d");
g_ptr_array_add(array, item);
item = g_new0(NMEditorConnectionTypeDataReal, 1);
item->data.name = _("IP tunnel");
item->data.setting_type = NM_TYPE_SETTING_IP_TUNNEL;
item->data.device_type = NM_TYPE_DEVICE_IP_TUNNEL;
item->data.virtual = TRUE;
item->id_format = _("IP tunnel connection %d");
g_ptr_array_add(array, item);
#if 0
/* Add "VPN" only if there are plugins */
vpn_plugins_hash = nm_vpn_get_plugin_infos ();
have_vpn_plugins = vpn_plugins_hash && g_hash_table_size (vpn_plugins_hash);
if (have_vpn_plugins) {
GHashTableIter iter;
gpointer name, plugin;
item = g_new0 (NMEditorConnectionTypeDataReal, 1);
item->data.name = _("VPN");
item->data.setting_type = NM_TYPE_SETTING_VPN;
item->data.virtual = TRUE;
item->id_format = _("VPN connection %d");
item->no_autoconnect = TRUE;
g_ptr_array_add (array, item);
vpn_plugins = NULL;
g_hash_table_iter_init (&iter, vpn_plugins_hash);
while (g_hash_table_iter_next (&iter, &name, &plugin))
vpn_plugins = g_slist_prepend (vpn_plugins, plugin);
vpn_plugins = g_slist_sort (vpn_plugins, sort_vpn_plugins);
}
#endif
g_ptr_array_sort(array, sort_types);
g_ptr_array_add(array, NULL);
list = (NMEditorConnectionTypeData **) g_ptr_array_free(array, FALSE);
return list;
}
static gboolean
_assert_format_int(const char *format)
{
g_assert(format);
format = strchr(format, '%');
g_assert(format);
format++;
g_assert(!strchr(format, '%'));
g_assert(format[0] == 'd');
return TRUE;
}
static char *
get_available_connection_name(const char *format, NMClient *client)
{
const GPtrArray *conns;
GSList * names = NULL, *iter;
char * cname = NULL;
int i = 0;
nm_assert(_assert_format_int(format));
conns = nm_client_get_connections(client);
for (i = 0; i < conns->len; i++) {
const char *id;
id = nm_connection_get_id(NM_CONNECTION(conns->pdata[i]));
g_assert(id);
names = g_slist_append(names, (gpointer) id);
}
/* Find the next available unique connection name */
for (i = 1; !cname && i < 10000; i++) {
char * temp;
gboolean found = FALSE;
NM_PRAGMA_WARNING_DISABLE("-Wformat-nonliteral")
temp = g_strdup_printf(format, i);
NM_PRAGMA_WARNING_REENABLE
for (iter = names; iter; iter = g_slist_next(iter)) {
if (!strcmp(iter->data, temp)) {
found = TRUE;
break;
}
}
if (!found)
cname = temp;
else
g_free(temp);
}
g_slist_free(names);
return cname;
}
static char *
get_available_iface_name(const char *try_name, NMClient *client)
{
const GPtrArray *connections;
NMConnection * connection;
char * new_name;
unsigned num = 1;
int i = 0;
const char * ifname = NULL;
connections = nm_client_get_connections(client);
new_name = g_strdup(try_name);
while (i < connections->len) {
connection = NM_CONNECTION(connections->pdata[i]);
ifname = nm_connection_get_interface_name(connection);
if (g_strcmp0(new_name, ifname) == 0) {
g_free(new_name);
new_name = g_strdup_printf("%s%d", try_name, num++);
i = 0;
} else
i++;
}
return new_name;
}
/**
* nm_editor_utils_create_connection:
* @type: the type of the connection's primary #NMSetting
* @master: (allow-none): the connection's master, if any
* @client: an #NMClient
*
* Creates a new #NMConnection of the given type, automatically
* creating a UUID and an appropriate not-currently-in-use connection
* name, setting #NMSettingConnection:autoconnect appropriately for
* the connection type, filling in slave-related information if
* @master is not %NULL, and initializing any other mandatory-to-set
* properties to reasonable initial values.
*
* Returns: a new #NMConnection
*/
NMConnection *
nm_editor_utils_create_connection(GType type, NMConnection *master, NMClient *client)
{
NMEditorConnectionTypeData ** types;
NMEditorConnectionTypeDataReal *type_data = NULL;
const char * master_setting_type = NULL, *master_uuid = NULL;
GType master_type = G_TYPE_INVALID, slave_setting_type = G_TYPE_INVALID;
NMConnection * connection;
NMSettingConnection *s_con;
NMSetting * s_hw, *s_slave;
char * uuid, *id, *ifname;
int i;
if (master) {
NMSettingConnection *master_s_con;
master_s_con = nm_connection_get_setting_connection(master);
master_setting_type = nm_setting_connection_get_connection_type(master_s_con);
master_uuid = nm_setting_connection_get_uuid(master_s_con);
master_type = nm_setting_lookup_type(master_setting_type);
}
types = nm_editor_utils_get_connection_type_list();
for (i = 0; types[i]; i++) {
if (types[i]->setting_type == type)
type_data = (NMEditorConnectionTypeDataReal *) types[i];
if (types[i]->setting_type == master_type)
slave_setting_type = types[i]->slave_setting_type;
}
if (!type_data) {
g_return_val_if_reached(NULL);
return NULL;
}
connection = nm_simple_connection_new();
s_con = NM_SETTING_CONNECTION(nm_setting_connection_new());
nm_connection_add_setting(connection, NM_SETTING(s_con));
s_hw = g_object_new(type, NULL);
nm_connection_add_setting(connection, s_hw);
if (type == NM_TYPE_SETTING_BOND)
ifname = get_available_iface_name("nm-bond", client);
else if (type == NM_TYPE_SETTING_TEAM)
ifname = get_available_iface_name("nm-team", client);
else if (type == NM_TYPE_SETTING_BRIDGE)
ifname = get_available_iface_name("nm-bridge", client);
else
ifname = NULL;
if (slave_setting_type != G_TYPE_INVALID) {
s_slave = g_object_new(slave_setting_type, NULL);
nm_connection_add_setting(connection, s_slave);
}
uuid = nm_utils_uuid_generate();
id = get_available_connection_name(type_data->id_format, client);
g_object_set(s_con,
NM_SETTING_CONNECTION_UUID,
uuid,
NM_SETTING_CONNECTION_ID,
id,
NM_SETTING_CONNECTION_TYPE,
nm_setting_get_name(s_hw),
NM_SETTING_CONNECTION_AUTOCONNECT,
!type_data->no_autoconnect,
NM_SETTING_CONNECTION_MASTER,
master_uuid,
NM_SETTING_CONNECTION_SLAVE_TYPE,
master_setting_type,
NM_SETTING_CONNECTION_INTERFACE_NAME,
ifname,
NULL);
g_free(uuid);
g_free(id);
g_free(ifname);
if (type_data->connection_setup_func)
type_data->connection_setup_func(connection, s_con, s_hw);
return connection;
}
/**
* nm_editor_utils_get_connection_type_data:
* @conn: an #NMConnection
*
* Gets the #NMEditorConnectionTypeData corresponding to
* @conn's connection type.
*
* Returns: the #NMEditorConnectionTypeData
*/
NMEditorConnectionTypeData *
nm_editor_utils_get_connection_type_data(NMConnection *conn)
{
NMSettingConnection * s_con;
const char * conn_type;
GType conn_gtype;
NMEditorConnectionTypeData **types;
int i;
s_con = nm_connection_get_setting_connection(conn);
g_return_val_if_fail(s_con != NULL, NULL);
conn_type = nm_setting_connection_get_connection_type(s_con);
conn_gtype = nm_setting_lookup_type(conn_type);
g_return_val_if_fail(conn_gtype != G_TYPE_INVALID, NULL);
types = nm_editor_utils_get_connection_type_list();
for (i = 0; types[i]; i++) {
if (types[i]->setting_type == conn_gtype)
return types[i];
}
return NULL;
}