Blame src/wireless-security/ws-wep-key.c

Packit Service d328f3
// SPDX-License-Identifier: GPL-2.0+
Packit Service d328f3
/* NetworkManager Applet -- allow user control over networking
Packit Service d328f3
 *
Packit Service d328f3
 * Dan Williams <dcbw@redhat.com>
Packit Service d328f3
 *
Packit Service d328f3
 * Copyright 2007 - 2014 Red Hat, Inc.
Packit Service d328f3
 */
Packit Service d328f3
Packit Service d328f3
#include "nm-default.h"
Packit Service d328f3
Packit Service d328f3
#include <string.h>
Packit Service d328f3
Packit Service d328f3
#include "wireless-security.h"
Packit Service d328f3
#include "utils.h"
Packit Service d328f3
#include "helpers.h"
Packit Service d328f3
#include "nma-ui-utils.h"
Packit Service d328f3
Packit Service d328f3
struct _WirelessSecurityWEPKey {
Packit Service d328f3
	WirelessSecurity parent;
Packit Service d328f3
Packit Service d328f3
	gboolean editing_connection;
Packit Service d328f3
	const char *password_flags_name;
Packit Service d328f3
Packit Service d328f3
	NMWepKeyType type;
Packit Service d328f3
	char keys[4][65];
Packit Service d328f3
	guint8 cur_index;
Packit Service d328f3
};
Packit Service d328f3
Packit Service d328f3
static void
Packit Service d328f3
show_toggled_cb (GtkCheckButton *button, WirelessSecurity *sec)
Packit Service d328f3
{
Packit Service d328f3
	GtkWidget *widget;
Packit Service d328f3
	gboolean visible;
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (sec->builder, "wep_key_entry"));
Packit Service d328f3
	g_assert (widget);
Packit Service d328f3
Packit Service d328f3
	visible = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (button));
Packit Service d328f3
	gtk_entry_set_visibility (GTK_ENTRY (widget), visible);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static void
Packit Service d328f3
key_index_combo_changed_cb (GtkWidget *combo, WirelessSecurity *parent)
Packit Service d328f3
{
Packit Service d328f3
	WirelessSecurityWEPKey *sec = (WirelessSecurityWEPKey *) parent;
Packit Service d328f3
	GtkWidget *entry;
Packit Service d328f3
	const char *key;
Packit Service d328f3
	int key_index;
Packit Service d328f3
Packit Service d328f3
	/* Save WEP key for old key index */
Packit Service d328f3
	entry = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wep_key_entry"));
Packit Service d328f3
	key = gtk_entry_get_text (GTK_ENTRY (entry));
Packit Service d328f3
	if (key)
Packit Service d328f3
		g_strlcpy (sec->keys[sec->cur_index], key, sizeof (sec->keys[sec->cur_index]));
Packit Service d328f3
	else
Packit Service d328f3
		memset (sec->keys[sec->cur_index], 0, sizeof (sec->keys[sec->cur_index]));
Packit Service d328f3
Packit Service d328f3
	key_index = gtk_combo_box_get_active (GTK_COMBO_BOX (combo));
Packit Service d328f3
	g_return_if_fail (key_index <= 3);
Packit Service d328f3
	g_return_if_fail (key_index >= 0);
Packit Service d328f3
Packit Service d328f3
	/* Populate entry with key from new index */
Packit Service d328f3
	gtk_entry_set_text (GTK_ENTRY (entry), sec->keys[key_index]);
Packit Service d328f3
	sec->cur_index = key_index;
Packit Service d328f3
Packit Service d328f3
	wireless_security_changed_cb (combo, parent);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static void
Packit Service d328f3
destroy (WirelessSecurity *parent)
Packit Service d328f3
{
Packit Service d328f3
	WirelessSecurityWEPKey *sec = (WirelessSecurityWEPKey *) parent;
Packit Service d328f3
	int i;
Packit Service d328f3
Packit Service d328f3
	for (i = 0; i < 4; i++)
Packit Service d328f3
		memset (sec->keys[i], 0, sizeof (sec->keys[i]));
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static gboolean
Packit Service d328f3
validate (WirelessSecurity *parent, GError **error)
Packit Service d328f3
{
Packit Service d328f3
	WirelessSecurityWEPKey *sec = (WirelessSecurityWEPKey *) parent;
Packit Service d328f3
	NMSettingSecretFlags secret_flags;
Packit Service d328f3
	GtkWidget *entry;
Packit Service d328f3
	const char *key;
Packit Service d328f3
	int i;
Packit Service d328f3
Packit Service d328f3
	entry = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wep_key_entry"));
Packit Service d328f3
	g_assert (entry);
Packit Service d328f3
Packit Service d328f3
	secret_flags = nma_utils_menu_to_secret_flags (entry);
Packit Service d328f3
	key = gtk_entry_get_text (GTK_ENTRY (entry));
Packit Service d328f3
Packit Service d328f3
        if (   secret_flags & NM_SETTING_SECRET_FLAG_NOT_SAVED
Packit Service d328f3
            || secret_flags & NM_SETTING_SECRET_FLAG_NOT_REQUIRED) {
Packit Service d328f3
		/* All good. */
Packit Service d328f3
	} else if (!key) {
Packit Service d328f3
		widget_set_error (entry);
Packit Service d328f3
		g_set_error_literal (error, NMA_ERROR, NMA_ERROR_GENERIC, _("missing wep-key"));
Packit Service d328f3
		return FALSE;
Packit Service d328f3
	} else if (sec->type == NM_WEP_KEY_TYPE_KEY) {
Packit Service d328f3
		if ((strlen (key) == 10) || (strlen (key) == 26)) {
Packit Service d328f3
			for (i = 0; i < strlen (key); i++) {
Packit Service d328f3
				if (!g_ascii_isxdigit (key[i])) {
Packit Service d328f3
					widget_set_error (entry);
Packit Service d328f3
					g_set_error (error, NMA_ERROR, NMA_ERROR_GENERIC, _("invalid wep-key: key with a length of %zu must contain only hex-digits"), strlen (key));
Packit Service d328f3
					return FALSE;
Packit Service d328f3
				}
Packit Service d328f3
			}
Packit Service d328f3
		} else if ((strlen (key) == 5) || (strlen (key) == 13)) {
Packit Service d328f3
			for (i = 0; i < strlen (key); i++) {
Packit Service d328f3
				if (!utils_char_is_ascii_print (key[i])) {
Packit Service d328f3
					widget_set_error (entry);
Packit Service d328f3
					g_set_error (error, NMA_ERROR, NMA_ERROR_GENERIC, _("invalid wep-key: key with a length of %zu must contain only ascii characters"), strlen (key));
Packit Service d328f3
					return FALSE;
Packit Service d328f3
				}
Packit Service d328f3
			}
Packit Service d328f3
		} else {
Packit Service d328f3
			widget_set_error (entry);
Packit Service d328f3
			g_set_error (error, NMA_ERROR, NMA_ERROR_GENERIC, _("invalid wep-key: wrong key length %zu. A key must be either of length 5/13 (ascii) or 10/26 (hex)"), strlen (key));
Packit Service d328f3
			return FALSE;
Packit Service d328f3
		}
Packit Service d328f3
	} else if (sec->type == NM_WEP_KEY_TYPE_PASSPHRASE) {
Packit Service d328f3
		if (!*key || (strlen (key) > 64)) {
Packit Service d328f3
			widget_set_error (entry);
Packit Service d328f3
			if (!*key)
Packit Service d328f3
				g_set_error_literal (error, NMA_ERROR, NMA_ERROR_GENERIC, _("invalid wep-key: passphrase must be non-empty"));
Packit Service d328f3
			else
Packit Service d328f3
				g_set_error_literal (error, NMA_ERROR, NMA_ERROR_GENERIC, _("invalid wep-key: passphrase must be shorter than 64 characters"));
Packit Service d328f3
			return FALSE;
Packit Service d328f3
		}
Packit Service d328f3
	}
Packit Service d328f3
	widget_unset_error (entry);
Packit Service d328f3
Packit Service d328f3
	return TRUE;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static void
Packit Service d328f3
add_to_size_group (WirelessSecurity *parent, GtkSizeGroup *group)
Packit Service d328f3
{
Packit Service d328f3
	GtkWidget *widget;
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "auth_method_label"));
Packit Service d328f3
	gtk_size_group_add_widget (group, widget);
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wep_key_label"));
Packit Service d328f3
	gtk_size_group_add_widget (group, widget);
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "key_index_label"));
Packit Service d328f3
	gtk_size_group_add_widget (group, widget);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static void
Packit Service d328f3
fill_connection (WirelessSecurity *parent, NMConnection *connection)
Packit Service d328f3
{
Packit Service d328f3
	WirelessSecurityWEPKey *sec = (WirelessSecurityWEPKey *) parent;
Packit Service d328f3
	NMSettingWirelessSecurity *s_wsec;
Packit Service d328f3
	NMSettingSecretFlags secret_flags;
Packit Service d328f3
	GtkWidget *widget, *passwd_entry;
Packit Service d328f3
	gint auth_alg;
Packit Service d328f3
	const char *key;
Packit Service d328f3
	int i;
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "auth_method_combo"));
Packit Service d328f3
	auth_alg = gtk_combo_box_get_active (GTK_COMBO_BOX (widget));
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wep_key_entry"));
Packit Service d328f3
	passwd_entry = widget;
Packit Service d328f3
	key = gtk_entry_get_text (GTK_ENTRY (widget));
Packit Service d328f3
	g_strlcpy (sec->keys[sec->cur_index], key, sizeof (sec->keys[sec->cur_index]));
Packit Service d328f3
Packit Service d328f3
	/* Blow away the old security setting by adding a clear one */
Packit Service d328f3
	s_wsec = (NMSettingWirelessSecurity *) nm_setting_wireless_security_new ();
Packit Service d328f3
	nm_connection_add_setting (connection, (NMSetting *) s_wsec);
Packit Service d328f3
Packit Service d328f3
	g_object_set (s_wsec,
Packit Service d328f3
	              NM_SETTING_WIRELESS_SECURITY_KEY_MGMT, "none",
Packit Service d328f3
	              NM_SETTING_WIRELESS_SECURITY_WEP_TX_KEYIDX, sec->cur_index,
Packit Service d328f3
	              NM_SETTING_WIRELESS_SECURITY_AUTH_ALG, (auth_alg == 1) ? "shared" : "open",
Packit Service d328f3
	              NM_SETTING_WIRELESS_SECURITY_WEP_KEY_TYPE, sec->type,
Packit Service d328f3
	              NULL);
Packit Service d328f3
Packit Service d328f3
	for (i = 0; i < 4; i++) {
Packit Service d328f3
		if (strlen (sec->keys[i]))
Packit Service d328f3
			nm_setting_wireless_security_set_wep_key (s_wsec, i, sec->keys[i]);
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	/* Save WEP_KEY_FLAGS to the connection */
Packit Service d328f3
	secret_flags = nma_utils_menu_to_secret_flags (passwd_entry);
Packit Service d328f3
	g_object_set (s_wsec, NM_SETTING_WIRELESS_SECURITY_WEP_KEY_FLAGS, secret_flags, NULL);
Packit Service d328f3
Packit Service d328f3
	/* Update secret flags and popup when editing the connection */
Packit Service d328f3
	if (sec->editing_connection)
Packit Service d328f3
		nma_utils_update_password_storage (passwd_entry, secret_flags,
Packit Service d328f3
		                                   NM_SETTING (s_wsec), sec->password_flags_name);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static void
Packit Service d328f3
wep_entry_filter_cb (GtkEditable *editable,
Packit Service d328f3
                     gchar *text,
Packit Service d328f3
                     gint length,
Packit Service d328f3
                     gint *position,
Packit Service d328f3
                     gpointer data)
Packit Service d328f3
{
Packit Service d328f3
	WirelessSecurityWEPKey *sec = (WirelessSecurityWEPKey *) data;
Packit Service d328f3
Packit Service d328f3
	if (sec->type == NM_WEP_KEY_TYPE_KEY) {
Packit Service d328f3
		utils_filter_editable_on_insert_text (editable,
Packit Service d328f3
		                                      text, length, position, data,
Packit Service d328f3
		                                      utils_char_is_ascii_print,
Packit Service d328f3
		                                      wep_entry_filter_cb);
Packit Service d328f3
	}
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static void
Packit Service d328f3
update_secrets (WirelessSecurity *parent, NMConnection *connection)
Packit Service d328f3
{
Packit Service d328f3
	WirelessSecurityWEPKey *sec = (WirelessSecurityWEPKey *) parent;
Packit Service d328f3
	NMSettingWirelessSecurity *s_wsec;
Packit Service d328f3
	GtkWidget *widget;
Packit Service d328f3
	const char *tmp;
Packit Service d328f3
	int i;
Packit Service d328f3
Packit Service d328f3
	s_wsec = nm_connection_get_setting_wireless_security (connection);
Packit Service d328f3
	for (i = 0; s_wsec && i < 4; i++) {
Packit Service d328f3
		tmp = nm_setting_wireless_security_get_wep_key (s_wsec, i);
Packit Service d328f3
		if (tmp)
Packit Service d328f3
			g_strlcpy (sec->keys[i], tmp, sizeof (sec->keys[i]));
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wep_key_entry"));
Packit Service d328f3
	if (strlen (sec->keys[sec->cur_index]))
Packit Service d328f3
		gtk_entry_set_text (GTK_ENTRY (widget), sec->keys[sec->cur_index]);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
WirelessSecurityWEPKey *
Packit Service d328f3
ws_wep_key_new (NMConnection *connection,
Packit Service d328f3
                NMWepKeyType type,
Packit Service d328f3
                gboolean adhoc_create,
Packit Service d328f3
                gboolean secrets_only)
Packit Service d328f3
{
Packit Service d328f3
	WirelessSecurity *parent;
Packit Service d328f3
	WirelessSecurityWEPKey *sec;
Packit Service d328f3
	GtkWidget *widget;
Packit Service d328f3
	NMSettingWirelessSecurity *s_wsec = NULL;
Packit Service d328f3
	NMSetting *setting = NULL;
Packit Service d328f3
	guint8 default_key_idx = 0;
Packit Service d328f3
	gboolean is_adhoc = adhoc_create;
Packit Service d328f3
	gboolean is_shared_key = FALSE;
Packit Service d328f3
Packit Service d328f3
	parent = wireless_security_init (sizeof (WirelessSecurityWEPKey),
Packit Service d328f3
	                                 validate,
Packit Service d328f3
	                                 add_to_size_group,
Packit Service d328f3
	                                 fill_connection,
Packit Service d328f3
	                                 update_secrets,
Packit Service d328f3
	                                 destroy,
Packit Service d328f3
	                                 "/org/freedesktop/network-manager-applet/ws-wep-key.ui",
Packit Service d328f3
	                                 "wep_key_notebook",
Packit Service d328f3
	                                 "wep_key_entry");
Packit Service d328f3
	if (!parent)
Packit Service d328f3
		return NULL;
Packit Service d328f3
Packit Service d328f3
	sec = (WirelessSecurityWEPKey *) parent;
Packit Service d328f3
	sec->editing_connection = secrets_only ? FALSE : TRUE;
Packit Service d328f3
	sec->password_flags_name = NM_SETTING_WIRELESS_SECURITY_WEP_KEY0;
Packit Service d328f3
	sec->type = type;
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "wep_key_entry"));
Packit Service d328f3
	g_assert (widget);
Packit Service d328f3
	gtk_entry_set_width_chars (GTK_ENTRY (widget), 28);
Packit Service d328f3
Packit Service d328f3
	/* Create password-storage popup menu for password entry under entry's secondary icon */
Packit Service d328f3
	if (connection)
Packit Service d328f3
		setting = (NMSetting *) nm_connection_get_setting_wireless_security (connection);
Packit Service d328f3
	nma_utils_setup_password_storage (widget, 0, setting, sec->password_flags_name,
Packit Service d328f3
	                                  FALSE, secrets_only);
Packit Service d328f3
Packit Service d328f3
	if (connection) {
Packit Service d328f3
		NMSettingWireless *s_wireless;
Packit Service d328f3
		const char *mode, *auth_alg;
Packit Service d328f3
Packit Service d328f3
		s_wireless = nm_connection_get_setting_wireless (connection);
Packit Service d328f3
		mode = s_wireless ? nm_setting_wireless_get_mode (s_wireless) : NULL;
Packit Service d328f3
		if (mode && !strcmp (mode, "adhoc"))
Packit Service d328f3
			is_adhoc = TRUE;
Packit Service d328f3
Packit Service d328f3
		s_wsec = nm_connection_get_setting_wireless_security (connection);
Packit Service d328f3
		if (s_wsec) {
Packit Service d328f3
			auth_alg = nm_setting_wireless_security_get_auth_alg (s_wsec);
Packit Service d328f3
			if (auth_alg && !strcmp (auth_alg, "shared"))
Packit Service d328f3
				is_shared_key = TRUE;
Packit Service d328f3
		}
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	g_signal_connect (G_OBJECT (widget), "changed",
Packit Service d328f3
	                  (GCallback) wireless_security_changed_cb,
Packit Service d328f3
	                  sec);
Packit Service d328f3
	g_signal_connect (G_OBJECT (widget), "insert-text",
Packit Service d328f3
	                  (GCallback) wep_entry_filter_cb,
Packit Service d328f3
	                  sec);
Packit Service d328f3
	if (sec->type == NM_WEP_KEY_TYPE_KEY)
Packit Service d328f3
		gtk_entry_set_max_length (GTK_ENTRY (widget), 26);
Packit Service d328f3
	else if (sec->type == NM_WEP_KEY_TYPE_PASSPHRASE)
Packit Service d328f3
		gtk_entry_set_max_length (GTK_ENTRY (widget), 64);
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "key_index_combo"));
Packit Service d328f3
	if (connection && s_wsec)
Packit Service d328f3
		default_key_idx = nm_setting_wireless_security_get_wep_tx_keyidx (s_wsec);
Packit Service d328f3
Packit Service d328f3
	gtk_combo_box_set_active (GTK_COMBO_BOX (widget), default_key_idx);
Packit Service d328f3
	sec->cur_index = default_key_idx;
Packit Service d328f3
	g_signal_connect (G_OBJECT (widget), "changed",
Packit Service d328f3
	                  (GCallback) key_index_combo_changed_cb,
Packit Service d328f3
	                  sec);
Packit Service d328f3
Packit Service d328f3
	/* Key index is useless with adhoc networks */
Packit Service d328f3
	if (is_adhoc || secrets_only) {
Packit Service d328f3
		gtk_widget_hide (widget);
Packit Service d328f3
		widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "key_index_label"));
Packit Service d328f3
		gtk_widget_hide (widget);
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	/* Fill the key entry with the key for that index */
Packit Service d328f3
	if (connection)
Packit Service d328f3
		update_secrets (WIRELESS_SECURITY (sec), connection);
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "show_checkbutton_wep"));
Packit Service d328f3
	g_assert (widget);
Packit Service d328f3
	g_signal_connect (G_OBJECT (widget), "toggled",
Packit Service d328f3
	                  (GCallback) show_toggled_cb,
Packit Service d328f3
	                  sec);
Packit Service d328f3
Packit Service d328f3
	widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "auth_method_combo"));
Packit Service d328f3
	gtk_combo_box_set_active (GTK_COMBO_BOX (widget), is_shared_key ? 1 : 0);
Packit Service d328f3
Packit Service d328f3
	g_signal_connect (G_OBJECT (widget), "changed",
Packit Service d328f3
	                  (GCallback) wireless_security_changed_cb,
Packit Service d328f3
	                  sec);
Packit Service d328f3
Packit Service d328f3
	/* Don't show auth method for adhoc (which always uses open-system) or
Packit Service d328f3
	 * when in "simple" mode.
Packit Service d328f3
	 */
Packit Service d328f3
	if (is_adhoc || secrets_only) {
Packit Service d328f3
		/* Ad-Hoc connections can't use Shared Key auth */
Packit Service d328f3
		if (is_adhoc)
Packit Service d328f3
			gtk_combo_box_set_active (GTK_COMBO_BOX (widget), 0);
Packit Service d328f3
		gtk_widget_hide (widget);
Packit Service d328f3
		widget = GTK_WIDGET (gtk_builder_get_object (parent->builder, "auth_method_label"));
Packit Service d328f3
		gtk_widget_hide (widget);
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	return sec;
Packit Service d328f3
}
Packit Service d328f3