mirror of
https://gitlab.freedesktop.org/NetworkManager/NetworkManager.git
synced 2025-12-28 00:30:09 +01:00
This is not merely cosmetic. I will need the obj_properties array to lookup GParamSpec by their PROP_* enum value. The alternative would be lookup by name, which is more expensive.
479 lines
15 KiB
C
479 lines
15 KiB
C
// SPDX-License-Identifier: LGPL-2.1+
|
|
/*
|
|
* Copyright (C) 2008 Novell, Inc.
|
|
* Copyright (C) 2009 - 2012 Red Hat, Inc.
|
|
*/
|
|
|
|
#include "nm-default.h"
|
|
|
|
#include "nm-remote-settings.h"
|
|
|
|
#include "c-list/src/c-list.h"
|
|
#include "nm-dbus-interface.h"
|
|
#include "nm-connection.h"
|
|
#include "nm-client.h"
|
|
#include "nm-remote-connection.h"
|
|
#include "nm-remote-connection-private.h"
|
|
#include "nm-object-private.h"
|
|
#include "nm-dbus-helpers.h"
|
|
#include "nm-core-internal.h"
|
|
|
|
#include "introspection/org.freedesktop.NetworkManager.Settings.h"
|
|
|
|
G_DEFINE_TYPE (NMRemoteSettings, nm_remote_settings, NM_TYPE_OBJECT)
|
|
|
|
#define NM_REMOTE_SETTINGS_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), NM_TYPE_REMOTE_SETTINGS, NMRemoteSettingsPrivate))
|
|
|
|
typedef struct {
|
|
NMDBusSettings *proxy;
|
|
GPtrArray *all_connections;
|
|
GPtrArray *visible_connections;
|
|
|
|
/* AddConnectionInfo objects that are waiting for the connection to become initialized */
|
|
CList add_lst_head;
|
|
|
|
char *hostname;
|
|
gboolean can_modify;
|
|
} NMRemoteSettingsPrivate;
|
|
|
|
NM_GOBJECT_PROPERTIES_DEFINE_BASE (
|
|
PROP_CONNECTIONS,
|
|
PROP_HOSTNAME,
|
|
PROP_CAN_MODIFY,
|
|
);
|
|
|
|
/* Signals */
|
|
enum {
|
|
CONNECTION_ADDED,
|
|
CONNECTION_REMOVED,
|
|
|
|
LAST_SIGNAL
|
|
};
|
|
static guint signals[LAST_SIGNAL] = { 0 };
|
|
|
|
/*****************************************************************************/
|
|
|
|
typedef struct {
|
|
CList add_lst;
|
|
NMRemoteSettings *self;
|
|
GTask *task;
|
|
char *connection_path;
|
|
GVariant *extra_results;
|
|
gulong cancellable_id;
|
|
} AddConnectionInfo;
|
|
|
|
static AddConnectionInfo *
|
|
_add_connection_info_find (NMRemoteSettings *self, const char *connection_path)
|
|
{
|
|
NMRemoteSettingsPrivate *priv = NM_REMOTE_SETTINGS_GET_PRIVATE (self);
|
|
AddConnectionInfo *info;
|
|
|
|
c_list_for_each_entry (info, &priv->add_lst_head, add_lst) {
|
|
if (nm_streq (info->connection_path, connection_path))
|
|
return info;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
_add_connection_info_complete (AddConnectionInfo *info,
|
|
NMRemoteConnection *connection,
|
|
GError *error_take)
|
|
{
|
|
nm_assert (info);
|
|
|
|
c_list_unlink_stale (&info->add_lst);
|
|
|
|
nm_clear_g_signal_handler (g_task_get_cancellable (info->task), &info->cancellable_id);
|
|
|
|
if (error_take)
|
|
g_task_return_error (info->task, error_take);
|
|
else {
|
|
NMAddConnectionResultData *result_info;
|
|
|
|
result_info = g_slice_new (NMAddConnectionResultData);
|
|
*result_info = (NMAddConnectionResultData) {
|
|
.connection = g_object_ref (connection),
|
|
.extra_results = g_steal_pointer (&info->extra_results),
|
|
};
|
|
g_task_return_pointer (info->task, result_info, (GDestroyNotify) nm_add_connection_result_data_free);
|
|
}
|
|
|
|
g_object_unref (info->task);
|
|
g_object_unref (info->self);
|
|
g_free (info->connection_path);
|
|
nm_g_variant_unref (info->extra_results);
|
|
nm_g_slice_free (info);
|
|
}
|
|
|
|
static void
|
|
_wait_for_connection_cancelled_cb (GCancellable *cancellable,
|
|
AddConnectionInfo *info)
|
|
{
|
|
_add_connection_info_complete (info,
|
|
NULL,
|
|
g_error_new_literal (G_IO_ERROR,
|
|
G_IO_ERROR_CANCELLED,
|
|
"Operation was cancelled"));
|
|
}
|
|
|
|
typedef const char * (*ConnectionStringGetter) (NMConnection *);
|
|
|
|
static NMRemoteConnection *
|
|
get_connection_by_string (NMRemoteSettings *settings,
|
|
const char *string,
|
|
ConnectionStringGetter get_comparison_string)
|
|
{
|
|
NMRemoteSettingsPrivate *priv;
|
|
NMConnection *candidate;
|
|
int i;
|
|
|
|
priv = NM_REMOTE_SETTINGS_GET_PRIVATE (settings);
|
|
|
|
for (i = 0; i < priv->visible_connections->len; i++) {
|
|
candidate = priv->visible_connections->pdata[i];
|
|
if (!g_strcmp0 (string, get_comparison_string (candidate)))
|
|
return NM_REMOTE_CONNECTION (candidate);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
NMRemoteConnection *
|
|
nm_remote_settings_get_connection_by_id (NMRemoteSettings *settings, const char *id)
|
|
{
|
|
g_return_val_if_fail (NM_IS_REMOTE_SETTINGS (settings), NULL);
|
|
g_return_val_if_fail (id != NULL, NULL);
|
|
|
|
return get_connection_by_string (settings, id, nm_connection_get_id);
|
|
}
|
|
|
|
NMRemoteConnection *
|
|
nm_remote_settings_get_connection_by_path (NMRemoteSettings *settings, const char *path)
|
|
{
|
|
g_return_val_if_fail (NM_IS_REMOTE_SETTINGS (settings), NULL);
|
|
g_return_val_if_fail (path != NULL, NULL);
|
|
|
|
return get_connection_by_string (settings, path, nm_connection_get_path);
|
|
}
|
|
|
|
NMRemoteConnection *
|
|
nm_remote_settings_get_connection_by_uuid (NMRemoteSettings *settings, const char *uuid)
|
|
{
|
|
g_return_val_if_fail (NM_IS_REMOTE_SETTINGS (settings), NULL);
|
|
g_return_val_if_fail (uuid != NULL, NULL);
|
|
|
|
return get_connection_by_string (settings, uuid, nm_connection_get_uuid);
|
|
}
|
|
|
|
static void
|
|
connection_visible_changed (GObject *object,
|
|
GParamSpec *pspec,
|
|
gpointer user_data)
|
|
{
|
|
NMRemoteConnection *connection = NM_REMOTE_CONNECTION (object);
|
|
NMRemoteSettings *self = NM_REMOTE_SETTINGS (user_data);
|
|
|
|
if (nm_remote_connection_get_visible (connection))
|
|
g_signal_emit (self, signals[CONNECTION_ADDED], 0, connection);
|
|
else
|
|
g_signal_emit (self, signals[CONNECTION_REMOVED], 0, connection);
|
|
}
|
|
|
|
static void
|
|
cleanup_connection (NMRemoteSettings *self,
|
|
NMRemoteConnection *remote)
|
|
{
|
|
g_signal_handlers_disconnect_by_func (remote, G_CALLBACK (connection_visible_changed), self);
|
|
}
|
|
|
|
static void
|
|
connection_removed (NMRemoteSettings *self,
|
|
NMRemoteConnection *remote)
|
|
{
|
|
NMRemoteSettingsPrivate *priv = NM_REMOTE_SETTINGS_GET_PRIVATE (self);
|
|
gboolean still_exists = FALSE;
|
|
int i;
|
|
|
|
/* Check if the connection was actually removed or if it just turned invisible. */
|
|
for (i = 0; i < priv->all_connections->len; i++) {
|
|
if (remote == priv->all_connections->pdata[i]) {
|
|
still_exists = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!still_exists)
|
|
cleanup_connection (self, remote);
|
|
|
|
/* Allow the signal to propagate if and only if @remote was in visible_connections */
|
|
if (!g_ptr_array_remove (priv->visible_connections, remote))
|
|
g_signal_stop_emission (self, signals[CONNECTION_REMOVED], 0);
|
|
}
|
|
|
|
static void
|
|
connection_added (NMRemoteSettings *self,
|
|
NMRemoteConnection *remote)
|
|
{
|
|
NMRemoteSettingsPrivate *priv = NM_REMOTE_SETTINGS_GET_PRIVATE (self);
|
|
AddConnectionInfo *info;
|
|
const char *path;
|
|
|
|
if (!g_signal_handler_find (remote, G_SIGNAL_MATCH_FUNC | G_SIGNAL_MATCH_DATA, 0, 0, NULL,
|
|
G_CALLBACK (connection_visible_changed), self)) {
|
|
g_signal_connect (remote,
|
|
"notify::" NM_REMOTE_CONNECTION_VISIBLE,
|
|
G_CALLBACK (connection_visible_changed),
|
|
self);
|
|
}
|
|
|
|
if (nm_remote_connection_get_visible (remote))
|
|
g_ptr_array_add (priv->visible_connections, remote);
|
|
else
|
|
g_signal_stop_emission (self, signals[CONNECTION_ADDED], 0);
|
|
|
|
/* FIXME: this doesn't look right. Why does it not care about whether the
|
|
* connection is visible? Anyway, this will be reworked. */
|
|
path = nm_connection_get_path (NM_CONNECTION (remote));
|
|
info = path
|
|
? _add_connection_info_find (self, path)
|
|
: NULL;
|
|
if (info)
|
|
_add_connection_info_complete (info, remote, NULL);
|
|
}
|
|
|
|
static void
|
|
object_creation_failed (NMObject *object,
|
|
const char *failed_path)
|
|
{
|
|
NMRemoteSettings *self = NM_REMOTE_SETTINGS (object);
|
|
AddConnectionInfo *info;
|
|
|
|
info = _add_connection_info_find (self, failed_path);
|
|
if (!info)
|
|
return;
|
|
|
|
_add_connection_info_complete (info,
|
|
NULL,
|
|
g_error_new_literal (NM_CLIENT_ERROR,
|
|
NM_CLIENT_ERROR_OBJECT_CREATION_FAILED,
|
|
_("Connection removed before it was initialized")));
|
|
}
|
|
|
|
const GPtrArray *
|
|
nm_remote_settings_get_connections (NMRemoteSettings *settings)
|
|
{
|
|
g_return_val_if_fail (NM_IS_REMOTE_SETTINGS (settings), NULL);
|
|
|
|
return NM_REMOTE_SETTINGS_GET_PRIVATE (settings)->visible_connections;
|
|
}
|
|
|
|
void
|
|
nm_remote_settings_wait_for_connection (NMRemoteSettings *self,
|
|
const char *connection_path,
|
|
GVariant *extra_results_take,
|
|
GTask *task_take)
|
|
{
|
|
NMRemoteSettingsPrivate *priv;
|
|
gs_unref_object GTask *task = task_take;
|
|
gs_unref_variant GVariant *extra_results = extra_results_take;
|
|
GCancellable *cancellable;
|
|
AddConnectionInfo *info;
|
|
|
|
priv = NM_REMOTE_SETTINGS_GET_PRIVATE (self);
|
|
|
|
/* FIXME: there is no timeout for how long we wait. But this entire
|
|
* code will be reworked, also that we have a suitable GMainContext
|
|
* where we can schedule the timeout (we shouldn't use g_main_context_default()). */
|
|
|
|
info = g_slice_new (AddConnectionInfo);
|
|
*info = (AddConnectionInfo) {
|
|
.self = g_object_ref (self),
|
|
.connection_path = g_strdup (connection_path),
|
|
.task = g_steal_pointer (&task),
|
|
.extra_results = g_steal_pointer (&extra_results),
|
|
};
|
|
c_list_link_tail (&priv->add_lst_head, &info->add_lst);
|
|
|
|
cancellable = g_task_get_cancellable (info->task);
|
|
/* On success, we still have to wait until the connection is fully
|
|
* initialized before calling the callback.
|
|
*/
|
|
if (cancellable) {
|
|
gulong id;
|
|
|
|
id = g_cancellable_connect (cancellable,
|
|
G_CALLBACK (_wait_for_connection_cancelled_cb),
|
|
info,
|
|
NULL);
|
|
if (id == 0) {
|
|
/* the callback was invoked synchronously, which destroyed @info.
|
|
* We must not touch @info anymore. */
|
|
} else
|
|
info->cancellable_id = id;
|
|
}
|
|
|
|
/* FIXME: OK, we just assume the the connection is here, and that we are bound
|
|
* to get the suitable signal when the connection is fully initalized (or failed).
|
|
* Obviously, that needs reworking. */
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
static void
|
|
nm_remote_settings_init (NMRemoteSettings *self)
|
|
{
|
|
NMRemoteSettingsPrivate *priv = NM_REMOTE_SETTINGS_GET_PRIVATE (self);
|
|
|
|
c_list_init (&priv->add_lst_head);
|
|
priv->all_connections = g_ptr_array_new ();
|
|
priv->visible_connections = g_ptr_array_new ();
|
|
}
|
|
|
|
static void
|
|
init_dbus (NMObject *object)
|
|
{
|
|
NMRemoteSettingsPrivate *priv = NM_REMOTE_SETTINGS_GET_PRIVATE (object);
|
|
const NMPropertiesInfo property_info[] = {
|
|
{ NM_REMOTE_SETTINGS_CONNECTIONS, &priv->all_connections, NULL, NM_TYPE_REMOTE_CONNECTION, "connection" },
|
|
{ NM_REMOTE_SETTINGS_HOSTNAME, &priv->hostname },
|
|
{ NM_REMOTE_SETTINGS_CAN_MODIFY, &priv->can_modify },
|
|
{ NULL },
|
|
};
|
|
|
|
NM_OBJECT_CLASS (nm_remote_settings_parent_class)->init_dbus (object);
|
|
|
|
priv->proxy = NMDBUS_SETTINGS (_nm_object_get_proxy (object, NM_DBUS_INTERFACE_SETTINGS));
|
|
_nm_object_register_properties (object,
|
|
NM_DBUS_INTERFACE_SETTINGS,
|
|
property_info);
|
|
}
|
|
|
|
static GObject *
|
|
constructor (GType type,
|
|
guint n_construct_params,
|
|
GObjectConstructParam *construct_params)
|
|
{
|
|
guint i;
|
|
const char *dbus_path;
|
|
|
|
/* Fill in the right D-Bus path if none was specified */
|
|
for (i = 0; i < n_construct_params; i++) {
|
|
if (strcmp (construct_params[i].pspec->name, NM_OBJECT_PATH) == 0) {
|
|
dbus_path = g_value_get_string (construct_params[i].value);
|
|
if (dbus_path == NULL) {
|
|
g_value_set_static_string (construct_params[i].value, NM_DBUS_PATH_SETTINGS);
|
|
} else {
|
|
if (!g_variant_is_object_path (dbus_path)) {
|
|
g_warning ("Passed D-Bus object path '%s' is invalid; using default '%s' instead",
|
|
dbus_path, NM_DBUS_PATH);
|
|
g_value_set_static_string (construct_params[i].value, NM_DBUS_PATH_SETTINGS);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
return G_OBJECT_CLASS (nm_remote_settings_parent_class)->constructor (type,
|
|
n_construct_params,
|
|
construct_params);
|
|
}
|
|
|
|
static void
|
|
dispose (GObject *object)
|
|
{
|
|
NMRemoteSettings *self = NM_REMOTE_SETTINGS (object);
|
|
NMRemoteSettingsPrivate *priv = NM_REMOTE_SETTINGS_GET_PRIVATE (self);
|
|
guint i;
|
|
|
|
if (priv->all_connections) {
|
|
for (i = 0; i < priv->all_connections->len; i++)
|
|
cleanup_connection (self, priv->all_connections->pdata[i]);
|
|
g_clear_pointer (&priv->all_connections, g_ptr_array_unref);
|
|
}
|
|
|
|
g_clear_pointer (&priv->visible_connections, g_ptr_array_unref);
|
|
g_clear_pointer (&priv->hostname, g_free);
|
|
g_clear_object (&priv->proxy);
|
|
|
|
G_OBJECT_CLASS (nm_remote_settings_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
get_property (GObject *object, guint prop_id,
|
|
GValue *value, GParamSpec *pspec)
|
|
{
|
|
NMRemoteSettingsPrivate *priv = NM_REMOTE_SETTINGS_GET_PRIVATE (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_CONNECTIONS:
|
|
g_value_take_boxed (value, _nm_utils_copy_object_array (priv->visible_connections));
|
|
break;
|
|
case PROP_HOSTNAME:
|
|
g_value_set_string (value, priv->hostname);
|
|
break;
|
|
case PROP_CAN_MODIFY:
|
|
g_value_set_boolean (value, priv->can_modify);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
nm_remote_settings_class_init (NMRemoteSettingsClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
NMObjectClass *nm_object_class = NM_OBJECT_CLASS (class);
|
|
|
|
g_type_class_add_private (class, sizeof (NMRemoteSettingsPrivate));
|
|
|
|
object_class->get_property = get_property;
|
|
object_class->constructor = constructor;
|
|
object_class->dispose = dispose;
|
|
|
|
nm_object_class->init_dbus = init_dbus;
|
|
nm_object_class->object_creation_failed = object_creation_failed;
|
|
|
|
class->connection_added = connection_added;
|
|
class->connection_removed = connection_removed;
|
|
|
|
obj_properties[PROP_CONNECTIONS] =
|
|
g_param_spec_boxed (NM_REMOTE_SETTINGS_CONNECTIONS, "", "",
|
|
G_TYPE_PTR_ARRAY,
|
|
G_PARAM_READABLE |
|
|
G_PARAM_STATIC_STRINGS);
|
|
|
|
obj_properties[PROP_HOSTNAME] =
|
|
g_param_spec_string (NM_REMOTE_SETTINGS_HOSTNAME, "", "",
|
|
NULL,
|
|
G_PARAM_READABLE |
|
|
G_PARAM_STATIC_STRINGS);
|
|
|
|
obj_properties[PROP_CAN_MODIFY] =
|
|
g_param_spec_boolean (NM_REMOTE_SETTINGS_CAN_MODIFY, "", "",
|
|
FALSE,
|
|
G_PARAM_READABLE |
|
|
G_PARAM_STATIC_STRINGS);
|
|
|
|
g_object_class_install_properties (object_class, _PROPERTY_ENUMS_LAST, obj_properties);
|
|
|
|
signals[CONNECTION_ADDED] =
|
|
g_signal_new (NM_REMOTE_SETTINGS_CONNECTION_ADDED,
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (NMRemoteSettingsClass, connection_added),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
NM_TYPE_REMOTE_CONNECTION);
|
|
|
|
signals[CONNECTION_REMOVED] =
|
|
g_signal_new (NM_REMOTE_SETTINGS_CONNECTION_REMOVED,
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (NMRemoteSettingsClass, connection_removed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 1,
|
|
NM_TYPE_REMOTE_CONNECTION);
|
|
}
|