Blame clients/cloud-setup/nmcs-provider.c

Packit Service 87a54e
/* SPDX-License-Identifier: LGPL-2.1-or-later */
Packit 5756e2
Packit Service 2bceb2
#include "libnm/nm-default-client.h"
Packit 5756e2
Packit 5756e2
#include "nmcs-provider.h"
Packit 5756e2
Packit 5756e2
#include "nm-cloud-setup-utils.h"
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit Service a1bd4f
NM_GOBJECT_PROPERTIES_DEFINE_BASE(PROP_HTTP_CLIENT, );
Packit 5756e2
Packit 5756e2
typedef struct _NMCSProviderPrivate {
Packit Service a1bd4f
    NMHttpClient *http_client;
Packit 5756e2
} NMCSProviderPrivate;
Packit 5756e2
Packit Service a1bd4f
G_DEFINE_TYPE(NMCSProvider, nmcs_provider, G_TYPE_OBJECT);
Packit 5756e2
Packit 5756e2
#define NMCS_PROVIDER_GET_PRIVATE(self) _NM_GET_PRIVATE_PTR(self, NMCSProvider, NMCS_IS_PROVIDER)
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
const char *
Packit Service a1bd4f
nmcs_provider_get_name(NMCSProvider *self)
Packit 5756e2
{
Packit Service a1bd4f
    NMCSProviderClass *klass;
Packit 5756e2
Packit Service a1bd4f
    g_return_val_if_fail(NMCS_IS_PROVIDER(self), NULL);
Packit 5756e2
Packit Service a1bd4f
    klass = NMCS_PROVIDER_GET_CLASS(self);
Packit Service a1bd4f
    nm_assert(klass->_name);
Packit Service a1bd4f
    return klass->_name;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
NMHttpClient *
Packit Service a1bd4f
nmcs_provider_get_http_client(NMCSProvider *self)
Packit 5756e2
{
Packit Service a1bd4f
    g_return_val_if_fail(NMCS_IS_PROVIDER(self), NULL);
Packit 5756e2
Packit Service a1bd4f
    return NMCS_PROVIDER_GET_PRIVATE(self)->http_client;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
GMainContext *
Packit Service a1bd4f
nmcs_provider_get_main_context(NMCSProvider *self)
Packit 5756e2
{
Packit Service a1bd4f
    g_return_val_if_fail(NMCS_IS_PROVIDER(self), NULL);
Packit 5756e2
Packit Service a1bd4f
    return nm_http_client_get_main_context(NMCS_PROVIDER_GET_PRIVATE(self)->http_client);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
void
Packit Service a1bd4f
nmcs_provider_detect(NMCSProvider *      self,
Packit Service a1bd4f
                     GCancellable *      cancellable,
Packit Service a1bd4f
                     GAsyncReadyCallback callback,
Packit Service a1bd4f
                     gpointer            user_data)
Packit 5756e2
{
Packit Service a1bd4f
    gs_unref_object GTask *task = NULL;
Packit Service a1bd4f
    const char *           env;
Packit 5756e2
Packit Service a1bd4f
    g_return_if_fail(NMCS_IS_PROVIDER(self));
Packit Service a1bd4f
    g_return_if_fail(!cancellable || G_IS_CANCELLABLE(cancellable));
Packit 5756e2
Packit Service a1bd4f
    task = nm_g_task_new(self, cancellable, nmcs_provider_detect, callback, user_data);
Packit 5756e2
Packit Service a1bd4f
    nmcs_wait_for_objects_register(task);
Packit 5756e2
Packit Service a1bd4f
    env = g_getenv(NMCS_PROVIDER_GET_CLASS(self)->_env_provider_enabled);
Packit Service a1bd4f
    if (!_nm_utils_ascii_str_to_bool(env, FALSE)) {
Packit Service a1bd4f
        g_task_return_error(task,
Packit Service a1bd4f
                            nm_utils_error_new(NM_UTILS_ERROR_UNKNOWN, "provider is disabled"));
Packit Service a1bd4f
        return;
Packit Service a1bd4f
    }
Packit 5756e2
Packit Service a1bd4f
    NMCS_PROVIDER_GET_CLASS(self)->detect(self, g_steal_pointer(&task));
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
gboolean
Packit Service a1bd4f
nmcs_provider_detect_finish(NMCSProvider *self, GAsyncResult *result, GError **error)
Packit 5756e2
{
Packit Service a1bd4f
    g_return_val_if_fail(NMCS_IS_PROVIDER(self), FALSE);
Packit Service a1bd4f
    g_return_val_if_fail(nm_g_task_is_valid(result, self, nmcs_provider_detect), FALSE);
Packit 5756e2
Packit Service a1bd4f
    return g_task_propagate_boolean(G_TASK(result), error);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
NMCSProviderGetConfigIfaceData *
Packit Service a1bd4f
nmcs_provider_get_config_iface_data_new(gboolean was_requested)
Packit 5756e2
{
Packit Service a1bd4f
    NMCSProviderGetConfigIfaceData *iface_data;
Packit Service a1bd4f
Packit Service a1bd4f
    iface_data  = g_slice_new(NMCSProviderGetConfigIfaceData);
Packit Service a1bd4f
    *iface_data = (NMCSProviderGetConfigIfaceData){
Packit Service a1bd4f
        .iface_idx     = -1,
Packit Service a1bd4f
        .was_requested = was_requested,
Packit Service a1bd4f
    };
Packit Service a1bd4f
    return iface_data;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
_iface_data_free(gpointer data)
Packit 5756e2
{
Packit Service a1bd4f
    NMCSProviderGetConfigIfaceData *iface_data = data;
Packit 5756e2
Packit Service a1bd4f
    g_free(iface_data->ipv4s_arr);
Packit Service a1bd4f
    g_free(iface_data->iproutes_arr);
Packit 5756e2
Packit Service a1bd4f
    nm_g_slice_free(iface_data);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service 87a54e
_get_config_task_maybe_return(NMCSProviderGetConfigTaskData *get_config_data, GError *error_take)
Packit 5756e2
{
Packit Service 87a54e
    gs_free_error GError *error = error_take;
Packit 5756e2
Packit Service 87a54e
    nm_assert(get_config_data);
Packit Service 87a54e
    nm_assert(G_IS_TASK(get_config_data->task));
Packit Service 87a54e
Packit Service 87a54e
    if (!error) {
Packit Service 87a54e
        if (get_config_data->n_pending > 0)
Packit Service 87a54e
            return;
Packit Service 87a54e
    }
Packit Service 87a54e
Packit Service 87a54e
    g_cancellable_cancel(get_config_data->intern_cancellable);
Packit Service 87a54e
Packit Service 87a54e
    if (error) {
Packit Service 87a54e
        if (nm_utils_error_is_cancelled(error))
Packit Service 87a54e
            _LOGD("get-config: cancelled");
Packit Service 87a54e
        else
Packit Service 87a54e
            _LOGD("get-config: failed: %s", error->message);
Packit Service 87a54e
        g_task_return_error(get_config_data->task, g_steal_pointer(&error));
Packit Service 87a54e
    } else {
Packit Service 87a54e
        _LOGD("get-config: success");
Packit Service 87a54e
        g_task_return_pointer(get_config_data->task,
Packit Service 87a54e
                              g_hash_table_ref(get_config_data->result_dict),
Packit Service 87a54e
                              (GDestroyNotify) g_hash_table_unref);
Packit Service 87a54e
    }
Packit Service 87a54e
Packit Service 87a54e
    nm_clear_g_signal_handler(g_task_get_cancellable(get_config_data->task),
Packit Service 87a54e
                              &get_config_data->extern_cancelled_id);
Packit Service 87a54e
Packit Service 87a54e
    if (get_config_data->extra_data_destroy)
Packit Service 87a54e
        get_config_data->extra_data_destroy(get_config_data->extra_data);
Packit 5756e2
Packit Service a1bd4f
    nm_clear_pointer(&get_config_data->result_dict, g_hash_table_unref);
Packit 5756e2
Packit Service 87a54e
    nm_g_object_unref(get_config_data->intern_cancellable);
Packit Service 87a54e
    g_object_unref(get_config_data->task);
Packit Service a1bd4f
    nm_g_slice_free(get_config_data);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
void
Packit Service 87a54e
_nmcs_provider_get_config_task_maybe_return(NMCSProviderGetConfigTaskData *get_config_data,
Packit Service 87a54e
                                            GError *                       error_take)
Packit Service 87a54e
{
Packit Service 87a54e
    nm_assert(!error_take || !nm_utils_error_is_cancelled(error_take));
Packit Service 87a54e
    _get_config_task_maybe_return(get_config_data, error_take);
Packit Service 87a54e
}
Packit Service 87a54e
Packit Service 87a54e
static void
Packit Service 87a54e
_get_config_cancelled_cb(GObject *object, gpointer user_data)
Packit Service 87a54e
{
Packit Service 87a54e
    _get_config_task_maybe_return(user_data, nm_utils_error_new_cancelled(FALSE, NULL));
Packit Service 87a54e
}
Packit Service 87a54e
Packit Service 87a54e
void
Packit Service a1bd4f
nmcs_provider_get_config(NMCSProvider *      self,
Packit Service a1bd4f
                         gboolean            any,
Packit Service a1bd4f
                         const char *const * hwaddrs,
Packit Service a1bd4f
                         GCancellable *      cancellable,
Packit Service a1bd4f
                         GAsyncReadyCallback callback,
Packit Service a1bd4f
                         gpointer            user_data)
Packit 5756e2
{
Packit Service a1bd4f
    NMCSProviderGetConfigTaskData *get_config_data;
Packit 5756e2
Packit Service a1bd4f
    g_return_if_fail(NMCS_IS_PROVIDER(self));
Packit Service a1bd4f
    g_return_if_fail(!cancellable || G_IS_CANCELLABLE(cancellable));
Packit 5756e2
Packit Service 87a54e
    _LOGD("get-config: starting");
Packit Service 87a54e
Packit Service a1bd4f
    get_config_data  = g_slice_new(NMCSProviderGetConfigTaskData);
Packit Service a1bd4f
    *get_config_data = (NMCSProviderGetConfigTaskData){
Packit Service a1bd4f
        .task = nm_g_task_new(self, cancellable, nmcs_provider_get_config, callback, user_data),
Packit Service a1bd4f
        .any  = any,
Packit Service a1bd4f
        .result_dict = g_hash_table_new_full(nm_str_hash, g_str_equal, g_free, _iface_data_free),
Packit Service a1bd4f
    };
Packit 5756e2
Packit Service a1bd4f
    nmcs_wait_for_objects_register(get_config_data->task);
Packit 5756e2
Packit Service a1bd4f
    for (; hwaddrs && hwaddrs[0]; hwaddrs++) {
Packit Service a1bd4f
        g_hash_table_insert(get_config_data->result_dict,
Packit Service a1bd4f
                            g_strdup(hwaddrs[0]),
Packit Service a1bd4f
                            nmcs_provider_get_config_iface_data_new(TRUE));
Packit Service a1bd4f
    }
Packit 5756e2
Packit Service 87a54e
    if (cancellable) {
Packit Service 87a54e
        gulong cancelled_id;
Packit Service 87a54e
Packit Service 87a54e
        cancelled_id = g_cancellable_connect(cancellable,
Packit Service 87a54e
                                             G_CALLBACK(_get_config_cancelled_cb),
Packit Service 87a54e
                                             get_config_data,
Packit Service 87a54e
                                             NULL);
Packit Service 87a54e
        if (cancelled_id == 0) {
Packit Service 87a54e
            /* the callback was already invoked synchronously and the task already returned. */
Packit Service 87a54e
            return;
Packit Service 87a54e
        }
Packit Service 87a54e
Packit Service 87a54e
        get_config_data->extern_cancelled_id = cancelled_id;
Packit Service 87a54e
        get_config_data->intern_cancellable  = g_cancellable_new();
Packit Service 87a54e
    }
Packit 5756e2
Packit Service a1bd4f
    NMCS_PROVIDER_GET_CLASS(self)->get_config(self, get_config_data);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
GHashTable *
Packit Service a1bd4f
nmcs_provider_get_config_finish(NMCSProvider *self, GAsyncResult *result, GError **error)
Packit 5756e2
{
Packit Service a1bd4f
    g_return_val_if_fail(NMCS_IS_PROVIDER(self), FALSE);
Packit Service a1bd4f
    g_return_val_if_fail(nm_g_task_is_valid(result, self, nmcs_provider_get_config), FALSE);
Packit 5756e2
Packit Service a1bd4f
    return g_task_propagate_pointer(G_TASK(result), error);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
Packit 5756e2
{
Packit Service a1bd4f
    NMCSProviderPrivate *priv = NMCS_PROVIDER_GET_PRIVATE(object);
Packit Service a1bd4f
Packit Service a1bd4f
    switch (prop_id) {
Packit Service a1bd4f
    case PROP_HTTP_CLIENT:
Packit Service a1bd4f
        priv->http_client = g_value_dup_object(value);
Packit Service a1bd4f
        g_return_if_fail(NM_IS_HTTP_CLIENT(priv->http_client));
Packit Service a1bd4f
        break;
Packit Service a1bd4f
    default:
Packit Service a1bd4f
        G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
Packit Service a1bd4f
        break;
Packit Service a1bd4f
    }
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
nmcs_provider_init(NMCSProvider *self)
Packit 5756e2
{
Packit Service a1bd4f
    NMCSProviderPrivate *priv;
Packit 5756e2
Packit Service a1bd4f
    priv = G_TYPE_INSTANCE_GET_PRIVATE(self, NMCS_TYPE_PROVIDER, NMCSProviderPrivate);
Packit 5756e2
Packit Service a1bd4f
    self->_priv = priv;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
dispose(GObject *object)
Packit 5756e2
{
Packit Service a1bd4f
    NMCSProvider *       self = NMCS_PROVIDER(object);
Packit Service a1bd4f
    NMCSProviderPrivate *priv = NMCS_PROVIDER_GET_PRIVATE(self);
Packit 5756e2
Packit Service a1bd4f
    g_clear_object(&priv->http_client);
Packit 5756e2
Packit Service a1bd4f
    G_OBJECT_CLASS(nmcs_provider_parent_class)->dispose(object);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
nmcs_provider_class_init(NMCSProviderClass *klass)
Packit 5756e2
{
Packit Service a1bd4f
    GObjectClass *object_class = G_OBJECT_CLASS(klass);
Packit 5756e2
Packit Service a1bd4f
    g_type_class_add_private(object_class, sizeof(NMCSProviderPrivate));
Packit 5756e2
Packit Service a1bd4f
    object_class->set_property = set_property;
Packit Service a1bd4f
    object_class->dispose      = dispose;
Packit 5756e2
Packit Service a1bd4f
    obj_properties[PROP_HTTP_CLIENT] =
Packit Service a1bd4f
        g_param_spec_object(NMCS_PROVIDER_HTTP_CLIENT,
Packit Service a1bd4f
                            "",
Packit Service a1bd4f
                            "",
Packit Service a1bd4f
                            NM_TYPE_HTTP_CLIENT,
Packit Service a1bd4f
                            G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
Packit 5756e2
Packit Service a1bd4f
    g_object_class_install_properties(object_class, _PROPERTY_ENUMS_LAST, obj_properties);
Packit 5756e2
}