Blame dispatcher/nm-dispatcher.c

Packit Service 87a54e
/* SPDX-License-Identifier: GPL-2.0-or-later */
Packit 5756e2
/*
Packit 5756e2
 * Copyright (C) 2008 - 2012 Red Hat, Inc.
Packit 5756e2
 */
Packit 5756e2
Packit Service 2bceb2
#include "libnm/nm-default-client.h"
Packit 5756e2
Packit 5756e2
#include <syslog.h>
Packit 5756e2
#include <stdio.h>
Packit 5756e2
#include <unistd.h>
Packit 5756e2
#include <stdlib.h>
Packit 5756e2
#include <sys/types.h>
Packit 5756e2
#include <signal.h>
Packit 5756e2
#include <sys/stat.h>
Packit 5756e2
#include <sys/wait.h>
Packit 5756e2
#include <arpa/inet.h>
Packit 5756e2
#include <glib-unix.h>
Packit 5756e2
Packit 5756e2
#include "nm-libnm-core-aux/nm-dispatcher-api.h"
Packit 5756e2
#include "nm-dispatcher-utils.h"
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
typedef struct Request Request;
Packit 5756e2
Packit 5756e2
static struct {
Packit Service a1bd4f
    GDBusConnection *dbus_connection;
Packit Service a1bd4f
    GMainLoop *      loop;
Packit Service a1bd4f
    gboolean         debug;
Packit Service a1bd4f
    gboolean         persist;
Packit Service a1bd4f
    guint            quit_id;
Packit Service a1bd4f
    guint            request_id_counter;
Packit Service a1bd4f
    gboolean         ever_acquired_name;
Packit Service a1bd4f
    bool             exit_with_failure;
Packit Service a1bd4f
Packit Service a1bd4f
    Request *current_request;
Packit Service a1bd4f
    GQueue * requests_waiting;
Packit Service a1bd4f
    int      num_requests_pending;
Packit 5756e2
} gl;
Packit 5756e2
Packit 5756e2
typedef struct {
Packit Service a1bd4f
    Request *request;
Packit Service a1bd4f
Packit Service a1bd4f
    char *         script;
Packit Service a1bd4f
    GPid           pid;
Packit Service a1bd4f
    DispatchResult result;
Packit Service a1bd4f
    char *         error;
Packit Service a1bd4f
    gboolean       wait;
Packit Service a1bd4f
    gboolean       dispatched;
Packit Service a1bd4f
    guint          watch_id;
Packit Service a1bd4f
    guint          timeout_id;
Packit 5756e2
} ScriptInfo;
Packit 5756e2
Packit 5756e2
struct Request {
Packit Service a1bd4f
    guint request_id;
Packit Service a1bd4f
Packit Service a1bd4f
    GDBusMethodInvocation *context;
Packit Service a1bd4f
    char *                 action;
Packit Service a1bd4f
    char *                 iface;
Packit Service a1bd4f
    char **                envp;
Packit Service a1bd4f
    gboolean               debug;
Packit Service a1bd4f
Packit Service a1bd4f
    GPtrArray *scripts; /* list of ScriptInfo */
Packit Service a1bd4f
    guint      idx;
Packit Service a1bd4f
    int        num_scripts_done;
Packit Service a1bd4f
    int        num_scripts_nowait;
Packit 5756e2
};
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit Service a1bd4f
#define __LOG_print(print_cmd, ...)                                                                    \
Packit Service a1bd4f
    G_STMT_START                                                                                       \
Packit Service a1bd4f
    {                                                                                                  \
Packit Service a1bd4f
        if (FALSE) {                                                                                   \
Packit Service a1bd4f
            /* g_message() alone does not warn about invalid format. Add a dummy printf() statement to
Packit Service a1bd4f
             * get a compiler warning about wrong format. */ \
Packit Service a1bd4f
            printf(__VA_ARGS__);                                                                       \
Packit Service a1bd4f
        }                                                                                              \
Packit Service a1bd4f
        print_cmd(__VA_ARGS__);                                                                        \
Packit Service a1bd4f
    }                                                                                                  \
Packit Service a1bd4f
    G_STMT_END
Packit Service a1bd4f
Packit Service a1bd4f
#define __LOG_print_R(print_cmd, _request, ...)                                      \
Packit Service a1bd4f
    G_STMT_START                                                                     \
Packit Service a1bd4f
    {                                                                                \
Packit Service a1bd4f
        __LOG_print(print_cmd,                                                       \
Packit Service a1bd4f
                    "req:%u '%s'%s%s%s" _NM_UTILS_MACRO_FIRST(__VA_ARGS__),          \
Packit Service a1bd4f
                    (_request)->request_id,                                          \
Packit Service a1bd4f
                    (_request)->action,                                              \
Packit Service a1bd4f
                    (_request)->iface ? " [" : "",                                   \
Packit Service a1bd4f
                    (_request)->iface ?: "",                                         \
Packit Service a1bd4f
                    (_request)->iface ? "]" : "" _NM_UTILS_MACRO_REST(__VA_ARGS__)); \
Packit Service a1bd4f
    }                                                                                \
Packit Service a1bd4f
    G_STMT_END
Packit Service a1bd4f
Packit Service a1bd4f
#define __LOG_print_S(print_cmd, _request, _script, ...)                        \
Packit Service a1bd4f
    G_STMT_START                                                                \
Packit Service a1bd4f
    {                                                                           \
Packit Service a1bd4f
        __LOG_print_R(print_cmd,                                                \
Packit Service a1bd4f
                      (_request),                                               \
Packit Service a1bd4f
                      "%s%s%s" _NM_UTILS_MACRO_FIRST(__VA_ARGS__),              \
Packit Service a1bd4f
                      (_script) ? ", \"" : "",                                  \
Packit Service a1bd4f
                      (_script) ? (_script)->script : "",                       \
Packit Service a1bd4f
                      (_script) ? "\"" : "" _NM_UTILS_MACRO_REST(__VA_ARGS__)); \
Packit Service a1bd4f
    }                                                                           \
Packit Service a1bd4f
    G_STMT_END
Packit Service a1bd4f
Packit Service a1bd4f
#define _LOG_X_(enabled_cmd, print_cmd, ...)     \
Packit Service a1bd4f
    G_STMT_START                                 \
Packit Service a1bd4f
    {                                            \
Packit Service a1bd4f
        if (enabled_cmd)                         \
Packit Service a1bd4f
            __LOG_print(print_cmd, __VA_ARGS__); \
Packit Service a1bd4f
    }                                            \
Packit Service a1bd4f
    G_STMT_END
Packit Service a1bd4f
Packit Service a1bd4f
#define _LOG_R_(enabled_cmd, x_request, print_cmd, ...)          \
Packit Service a1bd4f
    G_STMT_START                                                 \
Packit Service a1bd4f
    {                                                            \
Packit Service a1bd4f
        const Request *const _request = (x_request);             \
Packit Service a1bd4f
                                                                 \
Packit Service a1bd4f
        nm_assert(_request);                                     \
Packit Service a1bd4f
        if (enabled_cmd)                                         \
Packit Service a1bd4f
            __LOG_print_R(print_cmd, _request, ": "__VA_ARGS__); \
Packit Service a1bd4f
    }                                                            \
Packit Service a1bd4f
    G_STMT_END
Packit Service a1bd4f
Packit Service a1bd4f
#define _LOG_S_(enabled_cmd, x_script, print_cmd, ...)                        \
Packit Service a1bd4f
    G_STMT_START                                                              \
Packit Service a1bd4f
    {                                                                         \
Packit Service a1bd4f
        const ScriptInfo *const _script  = (x_script);                        \
Packit Service a1bd4f
        const Request *const    _request = _script ? _script->request : NULL; \
Packit Service a1bd4f
                                                                              \
Packit Service a1bd4f
        nm_assert(_script &&_request);                                        \
Packit Service a1bd4f
        if (enabled_cmd)                                                      \
Packit Service a1bd4f
            __LOG_print_S(print_cmd, _request, _script, ": "__VA_ARGS__);     \
Packit Service a1bd4f
    }                                                                         \
Packit Service a1bd4f
    G_STMT_END
Packit 5756e2
Packit 5756e2
#define _LOG_X_D_enabled() (gl.debug)
Packit Service a1bd4f
#define _LOG_X_T_enabled() _LOG_X_D_enabled()
Packit 5756e2
Packit Service a1bd4f
#define _LOG_R_D_enabled(request) (_NM_ENSURE_TYPE_CONST(Request *, request)->debug)
Packit Service a1bd4f
#define _LOG_R_T_enabled(request) _LOG_R_D_enabled(request)
Packit 5756e2
Packit Service a1bd4f
#define _LOG_X_T(...) _LOG_X_(_LOG_X_T_enabled(), g_debug, __VA_ARGS__)
Packit Service a1bd4f
#define _LOG_X_D(...) _LOG_X_(_LOG_X_D_enabled(), g_info, __VA_ARGS__)
Packit Service a1bd4f
#define _LOG_X_I(...) _LOG_X_(TRUE, g_message, __VA_ARGS__)
Packit Service a1bd4f
#define _LOG_X_W(...) _LOG_X_(TRUE, g_warning, __VA_ARGS__)
Packit 5756e2
Packit Service a1bd4f
#define _LOG_R_T(request, ...) _LOG_R_(_LOG_R_T_enabled(_request), request, g_debug, __VA_ARGS__)
Packit Service a1bd4f
#define _LOG_R_D(request, ...) _LOG_R_(_LOG_R_D_enabled(_request), request, g_info, __VA_ARGS__)
Packit Service a1bd4f
#define _LOG_R_W(request, ...) _LOG_R_(TRUE, request, g_warning, __VA_ARGS__)
Packit 5756e2
Packit Service a1bd4f
#define _LOG_S_T(script, ...) _LOG_S_(_LOG_R_T_enabled(_request), script, g_debug, __VA_ARGS__)
Packit Service a1bd4f
#define _LOG_S_D(script, ...) _LOG_S_(_LOG_R_D_enabled(_request), script, g_info, __VA_ARGS__)
Packit Service a1bd4f
#define _LOG_S_W(script, ...) _LOG_S_(TRUE, script, g_warning, __VA_ARGS__)
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit Service a1bd4f
static gboolean dispatch_one_script(Request *request);
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
script_info_free(gpointer ptr)
Packit 5756e2
{
Packit Service a1bd4f
    ScriptInfo *info = ptr;
Packit 5756e2
Packit Service a1bd4f
    g_free(info->script);
Packit Service a1bd4f
    g_free(info->error);
Packit Service a1bd4f
    g_slice_free(ScriptInfo, info);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
request_free(Request *request)
Packit 5756e2
{
Packit Service a1bd4f
    g_assert_cmpuint(request->num_scripts_done, ==, request->scripts->len);
Packit Service a1bd4f
    g_assert_cmpuint(request->num_scripts_nowait, ==, 0);
Packit 5756e2
Packit Service a1bd4f
    g_free(request->action);
Packit Service a1bd4f
    g_free(request->iface);
Packit Service a1bd4f
    g_strfreev(request->envp);
Packit Service a1bd4f
    g_ptr_array_free(request->scripts, TRUE);
Packit 5756e2
Packit Service a1bd4f
    g_slice_free(Request, request);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
quit_timeout_cb(gpointer user_data)
Packit 5756e2
{
Packit Service a1bd4f
    gl.quit_id = 0;
Packit Service a1bd4f
    g_main_loop_quit(gl.loop);
Packit Service a1bd4f
    return G_SOURCE_REMOVE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
quit_timeout_reschedule(void)
Packit 5756e2
{
Packit Service a1bd4f
    if (!gl.persist) {
Packit Service a1bd4f
        nm_clear_g_source(&gl.quit_id);
Packit Service a1bd4f
        gl.quit_id = g_timeout_add_seconds(10, quit_timeout_cb, NULL);
Packit Service a1bd4f
    }
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
/**
Packit 5756e2
 * next_request:
Packit 5756e2
 *
Packit 5756e2
 * @request: (allow-none): the request to set as next. If %NULL, dequeue the next
Packit 5756e2
 * waiting request. Otherwise, try to set the given request.
Packit 5756e2
 *
Packit 5756e2
 * Sets the currently active request (@current_request). The current request
Packit 5756e2
 * is a request that has at least on "wait" script, because requests that only
Packit 5756e2
 * consist of "no-wait" scripts are handled right away and not enqueued to
Packit 5756e2
 * @requests_waiting nor set as @current_request.
Packit 5756e2
 *
Packit 5756e2
 * Returns: %TRUE, if there was currently not request in process and it set
Packit 5756e2
 * a new request as current.
Packit 5756e2
 */
Packit 5756e2
static gboolean
Packit Service a1bd4f
next_request(Request *request)
Packit 5756e2
{
Packit Service a1bd4f
    if (request) {
Packit Service a1bd4f
        if (gl.current_request) {
Packit Service a1bd4f
            g_queue_push_tail(gl.requests_waiting, request);
Packit Service a1bd4f
            return FALSE;
Packit Service a1bd4f
        }
Packit Service a1bd4f
    } else {
Packit Service a1bd4f
        /* when calling next_request() without explicit @request, we always
Packit Service a1bd4f
         * forcefully clear @current_request. That one is certainly
Packit Service a1bd4f
         * handled already. */
Packit Service a1bd4f
        gl.current_request = NULL;
Packit Service a1bd4f
Packit Service a1bd4f
        request = g_queue_pop_head(gl.requests_waiting);
Packit Service a1bd4f
        if (!request)
Packit Service a1bd4f
            return FALSE;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    _LOG_R_D(request, "start running ordered scripts...");
Packit Service a1bd4f
Packit Service a1bd4f
    gl.current_request = request;
Packit Service a1bd4f
Packit Service a1bd4f
    return TRUE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
/**
Packit 5756e2
 * complete_request:
Packit 5756e2
 * @request: the request
Packit 5756e2
 *
Packit 5756e2
 * Checks if all the scripts for the request have terminated and in such case
Packit 5756e2
 * it sends the D-Bus response and releases the request resources.
Packit 5756e2
 *
Packit 5756e2
 * It also decreases @num_requests_pending and possibly does quit_timeout_reschedule().
Packit 5756e2
 */
Packit 5756e2
static void
Packit Service a1bd4f
complete_request(Request *request)
Packit 5756e2
{
Packit Service a1bd4f
    GVariantBuilder results;
Packit Service a1bd4f
    GVariant *      ret;
Packit Service a1bd4f
    guint           i;
Packit 5756e2
Packit Service a1bd4f
    nm_assert(request);
Packit 5756e2
Packit Service a1bd4f
    /* Are there still pending scripts? Then do nothing (for now). */
Packit Service a1bd4f
    if (request->num_scripts_done < request->scripts->len)
Packit Service a1bd4f
        return;
Packit 5756e2
Packit Service a1bd4f
    g_variant_builder_init(&results, G_VARIANT_TYPE("a(sus)"));
Packit Service a1bd4f
    for (i = 0; i < request->scripts->len; i++) {
Packit Service a1bd4f
        ScriptInfo *script = g_ptr_array_index(request->scripts, i);
Packit 5756e2
Packit Service a1bd4f
        g_variant_builder_add(&results,
Packit Service a1bd4f
                              "(sus)",
Packit Service a1bd4f
                              script->script,
Packit Service a1bd4f
                              script->result,
Packit Service a1bd4f
                              script->error ?: "");
Packit Service a1bd4f
    }
Packit 5756e2
Packit Service a1bd4f
    ret = g_variant_new("(a(sus))", &results);
Packit Service a1bd4f
    g_dbus_method_invocation_return_value(request->context, ret);
Packit 5756e2
Packit Service a1bd4f
    _LOG_R_T(request, "completed (%u scripts)", request->scripts->len);
Packit 5756e2
Packit Service a1bd4f
    if (gl.current_request == request)
Packit Service a1bd4f
        gl.current_request = NULL;
Packit 5756e2
Packit Service a1bd4f
    request_free(request);
Packit 5756e2
Packit Service a1bd4f
    g_assert_cmpuint(gl.num_requests_pending, >, 0);
Packit Service a1bd4f
    if (--gl.num_requests_pending <= 0) {
Packit Service a1bd4f
        nm_assert(!gl.current_request && !g_queue_peek_head(gl.requests_waiting));
Packit Service a1bd4f
        quit_timeout_reschedule();
Packit Service a1bd4f
    }
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
complete_script(ScriptInfo *script)
Packit 5756e2
{
Packit Service a1bd4f
    Request *request;
Packit Service a1bd4f
    gboolean wait = script->wait;
Packit Service a1bd4f
Packit Service a1bd4f
    request = script->request;
Packit Service a1bd4f
Packit Service a1bd4f
    if (wait) {
Packit Service a1bd4f
        /* for "wait" scripts, try to schedule the next blocking script.
Packit Service a1bd4f
         * If that is successful, return (as we must wait for its completion). */
Packit Service a1bd4f
        if (dispatch_one_script(request))
Packit Service a1bd4f
            return;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    nm_assert(!wait || gl.current_request == request);
Packit Service a1bd4f
Packit Service a1bd4f
    /* Try to complete the request. @request will be possibly free'd,
Packit Service a1bd4f
     * making @script and @request a dangling pointer. */
Packit Service a1bd4f
    complete_request(request);
Packit Service a1bd4f
Packit Service a1bd4f
    if (!wait) {
Packit Service a1bd4f
        /* this was a "no-wait" script. We either completed the request,
Packit Service a1bd4f
         * or there is nothing to do. Especially, there is no need to
Packit Service a1bd4f
         * queue the next_request() -- because no-wait scripts don't block
Packit Service a1bd4f
         * requests. However, if this was the last "no-wait" script and
Packit Service a1bd4f
         * there are "wait" scripts ready to run, launch them.
Packit Service a1bd4f
         */
Packit Service a1bd4f
        if (gl.current_request == request && gl.current_request->num_scripts_nowait == 0) {
Packit Service a1bd4f
            if (dispatch_one_script(gl.current_request))
Packit Service a1bd4f
                return;
Packit Service a1bd4f
Packit Service a1bd4f
            complete_request(gl.current_request);
Packit Service a1bd4f
        } else
Packit Service a1bd4f
            return;
Packit Service a1bd4f
    } else {
Packit Service a1bd4f
        /* if the script is a "wait" script, we already tried above to
Packit Service a1bd4f
         * dispatch the next script. As we didn't do that, it means we
Packit Service a1bd4f
         * just completed the last script of @request and we can continue
Packit Service a1bd4f
         * with the next request...
Packit Service a1bd4f
         *
Packit Service a1bd4f
         * Also, it cannot be that there is another request currently being
Packit Service a1bd4f
         * processed because only requests with "wait" scripts can become
Packit Service a1bd4f
         * @current_request. As there can only be one "wait" script running
Packit Service a1bd4f
         * at any time, it means complete_request() above completed @request. */
Packit Service a1bd4f
        nm_assert(!gl.current_request);
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    while (next_request(NULL)) {
Packit Service a1bd4f
        request = gl.current_request;
Packit Service a1bd4f
Packit Service a1bd4f
        if (dispatch_one_script(request))
Packit Service a1bd4f
            return;
Packit Service a1bd4f
Packit Service a1bd4f
        /* Try to complete the request. It will be either completed
Packit Service a1bd4f
         * now, or when all pending "no-wait" scripts return. */
Packit Service a1bd4f
        complete_request(request);
Packit Service a1bd4f
Packit Service a1bd4f
        /* We can immediately start next_request(), because our current
Packit Service a1bd4f
         * @request has obviously no more "wait" scripts either.
Packit Service a1bd4f
         * Repeat... */
Packit Service a1bd4f
    }
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
script_watch_cb(GPid pid, int status, gpointer user_data)
Packit 5756e2
{
Packit Service a1bd4f
    ScriptInfo *script = user_data;
Packit Service a1bd4f
    guint       err;
Packit Service a1bd4f
Packit Service a1bd4f
    g_assert(pid == script->pid);
Packit Service a1bd4f
Packit Service a1bd4f
    script->watch_id = 0;
Packit Service a1bd4f
    nm_clear_g_source(&script->timeout_id);
Packit Service a1bd4f
    script->request->num_scripts_done++;
Packit Service a1bd4f
    if (!script->wait)
Packit Service a1bd4f
        script->request->num_scripts_nowait--;
Packit Service a1bd4f
Packit Service a1bd4f
    if (WIFEXITED(status)) {
Packit Service a1bd4f
        err = WEXITSTATUS(status);
Packit Service a1bd4f
        if (err == 0)
Packit Service a1bd4f
            script->result = DISPATCH_RESULT_SUCCESS;
Packit Service a1bd4f
        else {
Packit Service a1bd4f
            script->error =
Packit Service a1bd4f
                g_strdup_printf("Script '%s' exited with error status %d.", script->script, err);
Packit Service a1bd4f
        }
Packit Service a1bd4f
    } else if (WIFSTOPPED(status)) {
Packit Service a1bd4f
        script->error = g_strdup_printf("Script '%s' stopped unexpectedly with signal %d.",
Packit Service a1bd4f
                                        script->script,
Packit Service a1bd4f
                                        WSTOPSIG(status));
Packit Service a1bd4f
    } else if (WIFSIGNALED(status)) {
Packit Service a1bd4f
        script->error =
Packit Service a1bd4f
            g_strdup_printf("Script '%s' died with signal %d", script->script, WTERMSIG(status));
Packit Service a1bd4f
    } else {
Packit Service a1bd4f
        script->error = g_strdup_printf("Script '%s' died from an unknown cause", script->script);
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    if (script->result == DISPATCH_RESULT_SUCCESS) {
Packit Service a1bd4f
        _LOG_S_T(script, "complete");
Packit Service a1bd4f
    } else {
Packit Service a1bd4f
        script->result = DISPATCH_RESULT_FAILED;
Packit Service a1bd4f
        _LOG_S_W(script, "complete: failed with %s", script->error);
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    g_spawn_close_pid(script->pid);
Packit Service a1bd4f
Packit Service a1bd4f
    complete_script(script);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
script_timeout_cb(gpointer user_data)
Packit 5756e2
{
Packit Service a1bd4f
    ScriptInfo *script = user_data;
Packit 5756e2
Packit Service a1bd4f
    script->timeout_id = 0;
Packit Service a1bd4f
    nm_clear_g_source(&script->watch_id);
Packit Service a1bd4f
    script->request->num_scripts_done++;
Packit Service a1bd4f
    if (!script->wait)
Packit Service a1bd4f
        script->request->num_scripts_nowait--;
Packit 5756e2
Packit Service a1bd4f
    _LOG_S_W(script, "complete: timeout (kill script)");
Packit 5756e2
Packit Service a1bd4f
    kill(script->pid, SIGKILL);
Packit 5756e2
again:
Packit Service a1bd4f
    if (waitpid(script->pid, NULL, 0) == -1) {
Packit Service a1bd4f
        if (errno == EINTR)
Packit Service a1bd4f
            goto again;
Packit Service a1bd4f
    }
Packit 5756e2
Packit Service a1bd4f
    script->error  = g_strdup_printf("Script '%s' timed out.", script->script);
Packit Service a1bd4f
    script->result = DISPATCH_RESULT_TIMEOUT;
Packit 5756e2
Packit Service a1bd4f
    g_spawn_close_pid(script->pid);
Packit 5756e2
Packit Service a1bd4f
    complete_script(script);
Packit 5756e2
Packit Service a1bd4f
    return FALSE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
check_permissions(struct stat *s, const char **out_error_msg)
Packit 5756e2
{
Packit Service a1bd4f
    g_return_val_if_fail(s != NULL, FALSE);
Packit Service a1bd4f
    g_return_val_if_fail(out_error_msg != NULL, FALSE);
Packit Service a1bd4f
    g_return_val_if_fail(*out_error_msg == NULL, FALSE);
Packit Service a1bd4f
Packit Service a1bd4f
    /* Only accept files owned by root */
Packit Service a1bd4f
    if (s->st_uid != 0) {
Packit Service a1bd4f
        *out_error_msg = "not owned by root.";
Packit Service a1bd4f
        return FALSE;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    /* Only accept files not writable by group or other, and not SUID */
Packit Service a1bd4f
    if (s->st_mode & (S_IWGRP | S_IWOTH | S_ISUID)) {
Packit Service a1bd4f
        *out_error_msg = "writable by group or other, or set-UID.";
Packit Service a1bd4f
        return FALSE;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    /* Only accept files executable by the owner */
Packit Service a1bd4f
    if (!(s->st_mode & S_IXUSR)) {
Packit Service a1bd4f
        *out_error_msg = "not executable by owner.";
Packit Service a1bd4f
        return FALSE;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    return TRUE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
check_filename(const char *file_name)
Packit 5756e2
{
Packit Service a1bd4f
    static const char *bad_suffixes[] = {
Packit Service a1bd4f
        "~",
Packit Service a1bd4f
        ".rpmsave",
Packit Service a1bd4f
        ".rpmorig",
Packit Service a1bd4f
        ".rpmnew",
Packit Service a1bd4f
        ".swp",
Packit Service a1bd4f
    };
Packit Service a1bd4f
    char *tmp;
Packit Service a1bd4f
    guint i;
Packit Service a1bd4f
Packit Service a1bd4f
    /* File must not be a backup file, package management file, or start with '.' */
Packit Service a1bd4f
Packit Service a1bd4f
    if (file_name[0] == '.')
Packit Service a1bd4f
        return FALSE;
Packit Service a1bd4f
    for (i = 0; i < G_N_ELEMENTS(bad_suffixes); i++) {
Packit Service a1bd4f
        if (g_str_has_suffix(file_name, bad_suffixes[i]))
Packit Service a1bd4f
            return FALSE;
Packit Service a1bd4f
    }
Packit Service a1bd4f
    tmp = g_strrstr(file_name, ".dpkg-");
Packit Service a1bd4f
    if (tmp && !strchr(&tmp[1], '.'))
Packit Service a1bd4f
        return FALSE;
Packit Service a1bd4f
    return TRUE;
Packit 5756e2
}
Packit 5756e2
Packit Service a1bd4f
#define SCRIPT_TIMEOUT 600 /* 10 minutes */
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
script_dispatch(ScriptInfo *script)
Packit 5756e2
{
Packit Service a1bd4f
    gs_free_error GError *error = NULL;
Packit Service a1bd4f
    char *                argv[4];
Packit Service a1bd4f
    Request *             request = script->request;
Packit Service a1bd4f
Packit Service a1bd4f
    if (script->dispatched)
Packit Service a1bd4f
        return FALSE;
Packit Service a1bd4f
Packit Service a1bd4f
    script->dispatched = TRUE;
Packit Service a1bd4f
Packit Service a1bd4f
    /* Only for "hostname" action we coerce the interface name to "none". We don't
Packit Service a1bd4f
     * do so for "connectivity-check" action. */
Packit Service a1bd4f
Packit Service a1bd4f
    argv[0] = script->script;
Packit Service a1bd4f
    argv[1] = request->iface ?: (nm_streq(request->action, NMD_ACTION_HOSTNAME) ? "none" : "");
Packit Service a1bd4f
    argv[2] = request->action;
Packit Service a1bd4f
    argv[3] = NULL;
Packit Service a1bd4f
Packit Service a1bd4f
    _LOG_S_T(script, "run script%s", script->wait ? "" : " (no-wait)");
Packit Service a1bd4f
Packit Service a1bd4f
    if (!g_spawn_async("/",
Packit Service a1bd4f
                       argv,
Packit Service a1bd4f
                       request->envp,
Packit Service a1bd4f
                       G_SPAWN_DO_NOT_REAP_CHILD,
Packit Service a1bd4f
                       NULL,
Packit Service a1bd4f
                       NULL,
Packit Service a1bd4f
                       &script->pid,
Packit Service a1bd4f
                       &error)) {
Packit Service a1bd4f
        _LOG_S_W(script, "complete: failed to execute script: %s", error->message);
Packit Service a1bd4f
        script->result = DISPATCH_RESULT_EXEC_FAILED;
Packit Service a1bd4f
        script->error  = g_strdup(error->message);
Packit Service a1bd4f
        request->num_scripts_done++;
Packit Service a1bd4f
        return FALSE;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    script->watch_id   = g_child_watch_add(script->pid, (GChildWatchFunc) script_watch_cb, script);
Packit Service a1bd4f
    script->timeout_id = g_timeout_add_seconds(SCRIPT_TIMEOUT, script_timeout_cb, script);
Packit Service a1bd4f
    if (!script->wait)
Packit Service a1bd4f
        request->num_scripts_nowait++;
Packit Service a1bd4f
    return TRUE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
dispatch_one_script(Request *request)
Packit 5756e2
{
Packit Service a1bd4f
    if (request->num_scripts_nowait > 0)
Packit Service a1bd4f
        return TRUE;
Packit 5756e2
Packit Service a1bd4f
    while (request->idx < request->scripts->len) {
Packit Service a1bd4f
        ScriptInfo *script;
Packit 5756e2
Packit Service a1bd4f
        script = g_ptr_array_index(request->scripts, request->idx++);
Packit Service a1bd4f
        if (script_dispatch(script))
Packit Service a1bd4f
            return TRUE;
Packit Service a1bd4f
    }
Packit Service a1bd4f
    return FALSE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static int
Packit Service a1bd4f
_compare_basenames(gconstpointer a, gconstpointer b)
Packit 5756e2
{
Packit Service a1bd4f
    const char *basename_a = strrchr(a, '/');
Packit Service a1bd4f
    const char *basename_b = strrchr(b, '/');
Packit Service a1bd4f
    int         ret;
Packit 5756e2
Packit Service a1bd4f
    nm_assert(basename_a);
Packit Service a1bd4f
    nm_assert(basename_b);
Packit 5756e2
Packit Service a1bd4f
    ret = strcmp(++basename_a, ++basename_b);
Packit Service a1bd4f
    if (ret)
Packit Service a1bd4f
        return ret;
Packit 5756e2
Packit Service a1bd4f
    nm_assert_not_reached();
Packit Service a1bd4f
    return 0;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
_find_scripts(Request *request, GHashTable *scripts, const char *base, const char *subdir)
Packit 5756e2
{
Packit Service a1bd4f
    const char *  filename;
Packit Service a1bd4f
    gs_free char *dirname = NULL;
Packit Service a1bd4f
    GError *      error   = NULL;
Packit Service a1bd4f
    GDir *        dir;
Packit Service a1bd4f
Packit Service a1bd4f
    dirname = g_build_filename(base, "dispatcher.d", subdir, NULL);
Packit Service a1bd4f
Packit Service a1bd4f
    if (!(dir = g_dir_open(dirname, 0, &error))) {
Packit Service a1bd4f
        if (!g_error_matches(error, G_FILE_ERROR, G_FILE_ERROR_NOENT)) {
Packit Service a1bd4f
            _LOG_R_W(request,
Packit Service a1bd4f
                     "find-scripts: Failed to open dispatcher directory '%s': %s",
Packit Service a1bd4f
                     dirname,
Packit Service a1bd4f
                     error->message);
Packit Service a1bd4f
        }
Packit Service a1bd4f
        g_error_free(error);
Packit Service a1bd4f
        return;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    while ((filename = g_dir_read_name(dir))) {
Packit Service a1bd4f
        if (!check_filename(filename))
Packit Service a1bd4f
            continue;
Packit Service a1bd4f
Packit Service a1bd4f
        g_hash_table_insert(scripts, g_strdup(filename), g_build_filename(dirname, filename, NULL));
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    g_dir_close(dir);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static GSList *
Packit Service a1bd4f
find_scripts(Request *request)
Packit 5756e2
{
Packit Service a1bd4f
    gs_unref_hashtable GHashTable *scripts     = NULL;
Packit Service a1bd4f
    GSList *                       script_list = NULL;
Packit Service a1bd4f
    GHashTableIter                 iter;
Packit Service a1bd4f
    const char *                   subdir;
Packit Service a1bd4f
    char *                         path;
Packit Service a1bd4f
    char *                         filename;
Packit Service a1bd4f
Packit Service a1bd4f
    if (NM_IN_STRSET(request->action, NMD_ACTION_PRE_UP, NMD_ACTION_VPN_PRE_UP))
Packit Service a1bd4f
        subdir = "pre-up.d";
Packit Service a1bd4f
    else if (NM_IN_STRSET(request->action, NMD_ACTION_PRE_DOWN, NMD_ACTION_VPN_PRE_DOWN))
Packit Service a1bd4f
        subdir = "pre-down.d";
Packit Service a1bd4f
    else
Packit Service a1bd4f
        subdir = NULL;
Packit Service a1bd4f
Packit Service a1bd4f
    scripts = g_hash_table_new_full(nm_str_hash, g_str_equal, g_free, g_free);
Packit Service a1bd4f
Packit Service a1bd4f
    _find_scripts(request, scripts, NMLIBDIR, subdir);
Packit Service a1bd4f
    _find_scripts(request, scripts, NMCONFDIR, subdir);
Packit Service a1bd4f
Packit Service a1bd4f
    g_hash_table_iter_init(&iter, scripts);
Packit Service a1bd4f
    while (g_hash_table_iter_next(&iter, (gpointer *) &filename, (gpointer *) &path)) {
Packit Service a1bd4f
        gs_free char *link_target = NULL;
Packit Service a1bd4f
        const char *  err_msg     = NULL;
Packit Service a1bd4f
        struct stat   st;
Packit Service a1bd4f
        int           err;
Packit Service a1bd4f
Packit Service a1bd4f
        link_target = g_file_read_link(path, NULL);
Packit Service a1bd4f
        if (nm_streq0(link_target, "/dev/null"))
Packit Service a1bd4f
            continue;
Packit Service a1bd4f
Packit Service a1bd4f
        err = stat(path, &st);
Packit Service a1bd4f
        if (err)
Packit Service a1bd4f
            _LOG_R_W(request, "find-scripts: Failed to stat '%s': %d", path, err);
Packit Service a1bd4f
        else if (!S_ISREG(st.st_mode) || st.st_size == 0) {
Packit Service a1bd4f
            /* silently skip. */
Packit Service a1bd4f
        } else if (!check_permissions(&st, &err_msg))
Packit Service a1bd4f
            _LOG_R_W(request, "find-scripts: Cannot execute '%s': %s", path, err_msg);
Packit Service a1bd4f
        else {
Packit Service a1bd4f
            /* success */
Packit Service a1bd4f
            script_list = g_slist_prepend(script_list, g_strdup(path));
Packit Service a1bd4f
            continue;
Packit Service a1bd4f
        }
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    return g_slist_sort(script_list, _compare_basenames);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
script_must_wait(const char *path)
Packit 5756e2
{
Packit Service a1bd4f
    gs_free char *link = NULL;
Packit Service a1bd4f
Packit Service a1bd4f
    link = g_file_read_link(path, NULL);
Packit Service a1bd4f
    if (link) {
Packit Service a1bd4f
        gs_free char *     dir  = NULL;
Packit Service a1bd4f
        nm_auto_free char *real = NULL;
Packit Service a1bd4f
Packit Service a1bd4f
        if (!g_path_is_absolute(link)) {
Packit Service a1bd4f
            char *tmp;
Packit Service a1bd4f
Packit Service a1bd4f
            dir = g_path_get_dirname(path);
Packit Service a1bd4f
            tmp = g_build_path("/", dir, link, NULL);
Packit Service a1bd4f
            g_free(link);
Packit Service a1bd4f
            g_free(dir);
Packit Service a1bd4f
            link = tmp;
Packit Service a1bd4f
        }
Packit Service a1bd4f
Packit Service a1bd4f
        dir  = g_path_get_dirname(link);
Packit Service a1bd4f
        real = realpath(dir, NULL);
Packit Service a1bd4f
        if (NM_STR_HAS_SUFFIX(real, "/no-wait.d"))
Packit Service a1bd4f
            return FALSE;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    return TRUE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
_method_call_action(GDBusMethodInvocation *invocation, GVariant *parameters)
Packit 5756e2
{
Packit Service a1bd4f
    const char *     action;
Packit Service a1bd4f
    gs_unref_variant GVariant *connection              = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *connection_properties   = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *device_properties       = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *device_proxy_properties = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *device_ip4_config       = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *device_ip6_config       = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *device_dhcp4_config     = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *device_dhcp6_config     = NULL;
Packit Service a1bd4f
    const char *               connectivity_state;
Packit Service a1bd4f
    const char *               vpn_ip_iface;
Packit Service a1bd4f
    gs_unref_variant GVariant *vpn_proxy_properties = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *vpn_ip4_config       = NULL;
Packit Service a1bd4f
    gs_unref_variant GVariant *vpn_ip6_config       = NULL;
Packit Service a1bd4f
    gboolean                   debug;
Packit Service a1bd4f
    GSList *                   sorted_scripts = NULL;
Packit Service a1bd4f
    GSList *                   iter;
Packit Service a1bd4f
    Request *                  request;
Packit Service a1bd4f
    char **                    p;
Packit Service a1bd4f
    guint                      i, num_nowait = 0;
Packit Service a1bd4f
    const char *               error_message = NULL;
Packit Service a1bd4f
Packit Service a1bd4f
    g_variant_get(parameters,
Packit Service a1bd4f
                  "("
Packit Service a1bd4f
                  "&s"         /* action */
Packit Service a1bd4f
                  "@a{sa{sv}}" /* connection */
Packit Service a1bd4f
                  "@a{sv}"     /* connection_properties */
Packit Service a1bd4f
                  "@a{sv}"     /* device_properties */
Packit Service a1bd4f
                  "@a{sv}"     /* device_proxy_properties */
Packit Service a1bd4f
                  "@a{sv}"     /* device_ip4_config */
Packit Service a1bd4f
                  "@a{sv}"     /* device_ip6_config */
Packit Service a1bd4f
                  "@a{sv}"     /* device_dhcp4_config */
Packit Service a1bd4f
                  "@a{sv}"     /* device_dhcp6_config */
Packit Service a1bd4f
                  "&s"         /* connectivity_state */
Packit Service a1bd4f
                  "&s"         /* vpn_ip_iface */
Packit Service a1bd4f
                  "@a{sv}"     /* vpn_proxy_properties */
Packit Service a1bd4f
                  "@a{sv}"     /* vpn_ip4_config */
Packit Service a1bd4f
                  "@a{sv}"     /* vpn_ip6_config */
Packit Service a1bd4f
                  "b"          /* debug */
Packit Service a1bd4f
                  ")",
Packit Service a1bd4f
                  &action,
Packit Service a1bd4f
                  &connection,
Packit Service a1bd4f
                  &connection_properties,
Packit Service a1bd4f
                  &device_properties,
Packit Service a1bd4f
                  &device_proxy_properties,
Packit Service a1bd4f
                  &device_ip4_config,
Packit Service a1bd4f
                  &device_ip6_config,
Packit Service a1bd4f
                  &device_dhcp4_config,
Packit Service a1bd4f
                  &device_dhcp6_config,
Packit Service a1bd4f
                  &connectivity_state,
Packit Service a1bd4f
                  &vpn_ip_iface,
Packit Service a1bd4f
                  &vpn_proxy_properties,
Packit Service a1bd4f
                  &vpn_ip4_config,
Packit Service a1bd4f
                  &vpn_ip6_config,
Packit Service a1bd4f
                  &debug);
Packit Service a1bd4f
Packit Service a1bd4f
    request             = g_slice_new0(Request);
Packit Service a1bd4f
    request->request_id = ++gl.request_id_counter;
Packit Service a1bd4f
    request->debug      = debug || gl.debug;
Packit Service a1bd4f
    request->context    = invocation;
Packit Service a1bd4f
    request->action     = g_strdup(action);
Packit Service a1bd4f
Packit Service a1bd4f
    request->envp = nm_dispatcher_utils_construct_envp(action,
Packit Service a1bd4f
                                                       connection,
Packit Service a1bd4f
                                                       connection_properties,
Packit Service a1bd4f
                                                       device_properties,
Packit Service a1bd4f
                                                       device_proxy_properties,
Packit Service a1bd4f
                                                       device_ip4_config,
Packit Service a1bd4f
                                                       device_ip6_config,
Packit Service a1bd4f
                                                       device_dhcp4_config,
Packit Service a1bd4f
                                                       device_dhcp6_config,
Packit Service a1bd4f
                                                       connectivity_state,
Packit Service a1bd4f
                                                       vpn_ip_iface,
Packit Service a1bd4f
                                                       vpn_proxy_properties,
Packit Service a1bd4f
                                                       vpn_ip4_config,
Packit Service a1bd4f
                                                       vpn_ip6_config,
Packit Service a1bd4f
                                                       &request->iface,
Packit Service a1bd4f
                                                       &error_message);
Packit Service a1bd4f
Packit Service a1bd4f
    request->scripts = g_ptr_array_new_full(5, script_info_free);
Packit Service a1bd4f
Packit Service a1bd4f
    sorted_scripts = find_scripts(request);
Packit Service a1bd4f
    for (iter = sorted_scripts; iter; iter = g_slist_next(iter)) {
Packit Service a1bd4f
        ScriptInfo *s;
Packit Service a1bd4f
Packit Service a1bd4f
        s          = g_slice_new0(ScriptInfo);
Packit Service a1bd4f
        s->request = request;
Packit Service a1bd4f
        s->script  = iter->data;
Packit Service a1bd4f
        s->wait    = script_must_wait(s->script);
Packit Service a1bd4f
        g_ptr_array_add(request->scripts, s);
Packit Service a1bd4f
    }
Packit Service a1bd4f
    g_slist_free(sorted_scripts);
Packit Service a1bd4f
Packit Service a1bd4f
    _LOG_R_D(request, "new request (%u scripts)", request->scripts->len);
Packit Service a1bd4f
    if (_LOG_R_T_enabled(request) && request->envp) {
Packit Service a1bd4f
        for (p = request->envp; *p; p++)
Packit Service a1bd4f
            _LOG_R_T(request, "environment: %s", *p);
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    if (error_message || request->scripts->len == 0) {
Packit Service a1bd4f
        GVariant *results;
Packit Service a1bd4f
Packit Service a1bd4f
        if (error_message)
Packit Service a1bd4f
            _LOG_R_W(request, "completed: invalid request: %s", error_message);
Packit Service a1bd4f
        else
Packit Service a1bd4f
            _LOG_R_D(request, "completed: no scripts");
Packit Service a1bd4f
Packit Service a1bd4f
        results = g_variant_new_array(G_VARIANT_TYPE("(sus)"), NULL, 0);
Packit Service a1bd4f
        g_dbus_method_invocation_return_value(invocation, g_variant_new("(@a(sus))", results));
Packit Service a1bd4f
        request->num_scripts_done = request->scripts->len;
Packit Service a1bd4f
        request_free(request);
Packit Service a1bd4f
        return;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    nm_clear_g_source(&gl.quit_id);
Packit Service a1bd4f
Packit Service a1bd4f
    gl.num_requests_pending++;
Packit Service a1bd4f
Packit Service a1bd4f
    for (i = 0; i < request->scripts->len; i++) {
Packit Service a1bd4f
        ScriptInfo *s = g_ptr_array_index(request->scripts, i);
Packit Service a1bd4f
Packit Service a1bd4f
        if (!s->wait) {
Packit Service a1bd4f
            script_dispatch(s);
Packit Service a1bd4f
            num_nowait++;
Packit Service a1bd4f
        }
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    if (num_nowait < request->scripts->len) {
Packit Service a1bd4f
        /* The request has at least one wait script.
Packit Service a1bd4f
         * Try next_request() to schedule the request for
Packit Service a1bd4f
         * execution. This either enqueues the request or
Packit Service a1bd4f
         * sets it as gl.current_request. */
Packit Service a1bd4f
        if (next_request(request)) {
Packit Service a1bd4f
            /* @request is now @current_request. Go ahead and
Packit Service a1bd4f
             * schedule the first wait script. */
Packit Service a1bd4f
            if (!dispatch_one_script(request)) {
Packit Service a1bd4f
                /* If that fails, we might be already finished with the
Packit Service a1bd4f
                 * request. Try complete_request(). */
Packit Service a1bd4f
                complete_request(request);
Packit Service a1bd4f
Packit Service a1bd4f
                if (next_request(NULL)) {
Packit Service a1bd4f
                    /* As @request was successfully scheduled as next_request(), there is no
Packit Service a1bd4f
                     * other request in queue that can be scheduled afterwards. Assert against
Packit Service a1bd4f
                     * that, but call next_request() to clear current_request. */
Packit Service a1bd4f
                    g_assert_not_reached();
Packit Service a1bd4f
                }
Packit Service a1bd4f
            }
Packit Service a1bd4f
        }
Packit Service a1bd4f
    } else {
Packit Service a1bd4f
        /* The request contains only no-wait scripts. Try to complete
Packit Service a1bd4f
         * the request right away (we might have failed to schedule any
Packit Service a1bd4f
         * of the scripts). It will be either completed now, or later
Packit Service a1bd4f
         * when the pending scripts return.
Packit Service a1bd4f
         * We don't enqueue it to gl.requests_waiting.
Packit Service a1bd4f
         * There is no need to handle next_request(), because @request is
Packit Service a1bd4f
         * not the current request anyway and does not interfere with requests
Packit Service a1bd4f
         * that have any "wait" scripts. */
Packit Service a1bd4f
        complete_request(request);
Packit Service a1bd4f
    }
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
on_name_acquired(GDBusConnection *connection, const char *name, gpointer user_data)
Packit 5756e2
{
Packit Service a1bd4f
    gl.ever_acquired_name = TRUE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
on_name_lost(GDBusConnection *connection, const char *name, gpointer user_data)
Packit 5756e2
{
Packit Service a1bd4f
    if (!connection) {
Packit Service a1bd4f
        if (!gl.ever_acquired_name) {
Packit Service a1bd4f
            _LOG_X_W("Could not get the system bus.  Make sure the message bus daemon is running!");
Packit Service a1bd4f
            gl.exit_with_failure = TRUE;
Packit Service a1bd4f
        } else {
Packit Service a1bd4f
            _LOG_X_I("System bus stopped. Exiting");
Packit Service a1bd4f
        }
Packit Service a1bd4f
    } else if (!gl.ever_acquired_name) {
Packit Service a1bd4f
        _LOG_X_W("Could not acquire the " NM_DISPATCHER_DBUS_SERVICE " service.");
Packit Service a1bd4f
        gl.exit_with_failure = TRUE;
Packit Service a1bd4f
    } else
Packit Service a1bd4f
        _LOG_X_I("Lost the " NM_DISPATCHER_DBUS_SERVICE " name. Exiting");
Packit Service a1bd4f
Packit Service a1bd4f
    g_main_loop_quit(gl.loop);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
_method_call(GDBusConnection *      connection,
Packit Service a1bd4f
             const char *           sender,
Packit Service a1bd4f
             const char *           object_path,
Packit Service a1bd4f
             const char *           interface_name,
Packit Service a1bd4f
             const char *           method_name,
Packit Service a1bd4f
             GVariant *             parameters,
Packit Service a1bd4f
             GDBusMethodInvocation *invocation,
Packit Service a1bd4f
             gpointer               user_data)
Packit 5756e2
{
Packit Service a1bd4f
    if (nm_streq(interface_name, NM_DISPATCHER_DBUS_INTERFACE)) {
Packit Service a1bd4f
        if (nm_streq(method_name, "Action")) {
Packit Service a1bd4f
            _method_call_action(invocation, parameters);
Packit Service a1bd4f
            return;
Packit Service a1bd4f
        }
Packit Service a1bd4f
    }
Packit Service a1bd4f
    g_dbus_method_invocation_return_error(invocation,
Packit Service a1bd4f
                                          G_DBUS_ERROR,
Packit Service a1bd4f
                                          G_DBUS_ERROR_UNKNOWN_METHOD,
Packit Service a1bd4f
                                          "Unknown method %s",
Packit Service a1bd4f
                                          method_name);
Packit 5756e2
}
Packit 5756e2
Packit Service a1bd4f
static GDBusInterfaceInfo *const interface_info = NM_DEFINE_GDBUS_INTERFACE_INFO(
Packit Service a1bd4f
    NM_DISPATCHER_DBUS_INTERFACE,
Packit Service a1bd4f
    .methods = NM_DEFINE_GDBUS_METHOD_INFOS(
Packit Service a1bd4f
        NM_DEFINE_GDBUS_METHOD_INFO(
Packit Service a1bd4f
            "Action",
Packit Service a1bd4f
            .in_args = NM_DEFINE_GDBUS_ARG_INFOS(
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("action", "s"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("connection", "a{sa{sv}}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("connection_properties", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("device_properties", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("device_proxy_properties", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("device_ip4_config", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("device_ip6_config", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("device_dhcp4_config", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("device_dhcp6_config", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("connectivity_state", "s"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("vpn_ip_iface", "s"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("vpn_proxy_properties", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("vpn_ip4_config", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("vpn_ip6_config", "a{sv}"),
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFO("debug", "b"), ),
Packit Service a1bd4f
            .out_args =
Packit Service a1bd4f
                NM_DEFINE_GDBUS_ARG_INFOS(NM_DEFINE_GDBUS_ARG_INFO("results", "a(sus)"), ), ), ), );
Packit 5756e2
Packit 5756e2
static const GDBusInterfaceVTable interface_vtable = {
Packit Service a1bd4f
    .method_call = _method_call,
Packit 5756e2
};
Packit 5756e2
Packit 5756e2
/*****************************************************************************/
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
log_handler(const char *log_domain, GLogLevelFlags log_level, const char *message, gpointer ignored)
Packit 5756e2
{
Packit Service a1bd4f
    int syslog_priority;
Packit Service a1bd4f
Packit Service a1bd4f
    switch (log_level) {
Packit Service a1bd4f
    case G_LOG_LEVEL_ERROR:
Packit Service a1bd4f
        syslog_priority = LOG_CRIT;
Packit Service a1bd4f
        break;
Packit Service a1bd4f
    case G_LOG_LEVEL_CRITICAL:
Packit Service a1bd4f
        syslog_priority = LOG_ERR;
Packit Service a1bd4f
        break;
Packit Service a1bd4f
    case G_LOG_LEVEL_WARNING:
Packit Service a1bd4f
        syslog_priority = LOG_WARNING;
Packit Service a1bd4f
        break;
Packit Service a1bd4f
    case G_LOG_LEVEL_MESSAGE:
Packit Service a1bd4f
        syslog_priority = LOG_NOTICE;
Packit Service a1bd4f
        break;
Packit Service a1bd4f
    case G_LOG_LEVEL_DEBUG:
Packit Service a1bd4f
        syslog_priority = LOG_DEBUG;
Packit Service a1bd4f
        break;
Packit Service a1bd4f
    case G_LOG_LEVEL_INFO:
Packit Service a1bd4f
    default:
Packit Service a1bd4f
        syslog_priority = LOG_INFO;
Packit Service a1bd4f
        break;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    syslog(syslog_priority, "%s", message);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
logging_setup(void)
Packit 5756e2
{
Packit Service a1bd4f
    openlog(G_LOG_DOMAIN, LOG_CONS, LOG_DAEMON);
Packit Service a1bd4f
    g_log_set_handler(G_LOG_DOMAIN,
Packit Service a1bd4f
                      G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
Packit Service a1bd4f
                      log_handler,
Packit Service a1bd4f
                      NULL);
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static void
Packit Service a1bd4f
logging_shutdown(void)
Packit 5756e2
{
Packit Service a1bd4f
    closelog();
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
signal_handler(gpointer user_data)
Packit 5756e2
{
Packit Service a1bd4f
    int signo = GPOINTER_TO_INT(user_data);
Packit 5756e2
Packit Service a1bd4f
    _LOG_X_I("Caught signal %d, shutting down...", signo);
Packit Service a1bd4f
    g_main_loop_quit(gl.loop);
Packit 5756e2
Packit Service a1bd4f
    return G_SOURCE_CONTINUE;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
static gboolean
Packit Service a1bd4f
parse_command_line(int *p_argc, char ***p_argv, GError **error)
Packit 5756e2
{
Packit Service a1bd4f
    GOptionContext *opt_ctx;
Packit Service a1bd4f
    GOptionEntry    entries[] = {
Packit Service a1bd4f
        {"debug", 0, 0, G_OPTION_ARG_NONE, &gl.debug, "Output to console rather than syslog", NULL},
Packit Service a1bd4f
        {"persist", 0, 0, G_OPTION_ARG_NONE, &gl.persist, "Don't quit after a short timeout", NULL},
Packit Service a1bd4f
        {NULL}};
Packit Service a1bd4f
    gboolean success;
Packit 5756e2
Packit Service a1bd4f
    opt_ctx = g_option_context_new(NULL);
Packit Service a1bd4f
    g_option_context_set_summary(opt_ctx, "Executes scripts upon actions by NetworkManager.");
Packit Service a1bd4f
    g_option_context_add_main_entries(opt_ctx, entries, NULL);
Packit 5756e2
Packit Service a1bd4f
    success = g_option_context_parse(opt_ctx, p_argc, p_argv, error);
Packit 5756e2
Packit Service a1bd4f
    g_option_context_free(opt_ctx);
Packit 5756e2
Packit Service a1bd4f
    return success;
Packit 5756e2
}
Packit 5756e2
Packit 5756e2
int
Packit Service a1bd4f
main(int argc, char **argv)
Packit 5756e2
{
Packit Service a1bd4f
    gs_free_error GError *error            = NULL;
Packit Service a1bd4f
    guint                 signal_id_term   = 0;
Packit Service a1bd4f
    guint                 signal_id_int    = 0;
Packit Service a1bd4f
    guint                 dbus_regist_id   = 0;
Packit Service a1bd4f
    guint                 dbus_own_name_id = 0;
Packit Service a1bd4f
Packit Service a1bd4f
    if (!parse_command_line(&argc, &argv, &error)) {
Packit Service a1bd4f
        _LOG_X_W("Error parsing command line arguments: %s", error->message);
Packit Service a1bd4f
        gl.exit_with_failure = TRUE;
Packit Service a1bd4f
        goto done;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    signal_id_term = g_unix_signal_add(SIGTERM, signal_handler, GINT_TO_POINTER(SIGTERM));
Packit Service a1bd4f
    signal_id_int  = g_unix_signal_add(SIGINT, signal_handler, GINT_TO_POINTER(SIGINT));
Packit Service a1bd4f
Packit Service a1bd4f
    if (gl.debug) {
Packit Service a1bd4f
        if (!g_getenv("G_MESSAGES_DEBUG")) {
Packit Service a1bd4f
            /* we log our regular messages using g_debug() and g_info().
Packit Service a1bd4f
             * When we redirect glib logging to syslog, there is no problem.
Packit Service a1bd4f
             * But in "debug" mode, glib will no print these messages unless
Packit Service a1bd4f
             * we set G_MESSAGES_DEBUG. */
Packit Service a1bd4f
            g_setenv("G_MESSAGES_DEBUG", "all", TRUE);
Packit Service a1bd4f
        }
Packit Service a1bd4f
    } else
Packit Service a1bd4f
        logging_setup();
Packit Service a1bd4f
Packit Service a1bd4f
    gl.loop = g_main_loop_new(NULL, FALSE);
Packit Service a1bd4f
Packit Service a1bd4f
    gl.dbus_connection = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
Packit Service a1bd4f
    if (!gl.dbus_connection) {
Packit Service a1bd4f
        _LOG_X_W("Could not get the system bus (%s).  Make sure the message bus daemon is running!",
Packit Service a1bd4f
                 error->message);
Packit Service a1bd4f
        gl.exit_with_failure = TRUE;
Packit Service a1bd4f
        goto done;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    gl.requests_waiting = g_queue_new();
Packit Service a1bd4f
Packit Service a1bd4f
    dbus_regist_id =
Packit Service a1bd4f
        g_dbus_connection_register_object(gl.dbus_connection,
Packit Service a1bd4f
                                          NM_DISPATCHER_DBUS_PATH,
Packit Service a1bd4f
                                          interface_info,
Packit Service a1bd4f
                                          NM_UNCONST_PTR(GDBusInterfaceVTable, &interface_vtable),
Packit Service a1bd4f
                                          NULL,
Packit Service a1bd4f
                                          NULL,
Packit Service a1bd4f
                                          &error);
Packit Service a1bd4f
    if (dbus_regist_id == 0) {
Packit Service a1bd4f
        _LOG_X_W("Could not export Dispatcher D-Bus interface: %s", error->message);
Packit Service a1bd4f
        gl.exit_with_failure = 1;
Packit Service a1bd4f
        goto done;
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    dbus_own_name_id = g_bus_own_name_on_connection(gl.dbus_connection,
Packit Service a1bd4f
                                                    NM_DISPATCHER_DBUS_SERVICE,
Packit Service a1bd4f
                                                    G_BUS_NAME_OWNER_FLAGS_NONE,
Packit Service a1bd4f
                                                    on_name_acquired,
Packit Service a1bd4f
                                                    on_name_lost,
Packit Service a1bd4f
                                                    NULL,
Packit Service a1bd4f
                                                    NULL);
Packit Service a1bd4f
Packit Service a1bd4f
    quit_timeout_reschedule();
Packit Service a1bd4f
Packit Service a1bd4f
    g_main_loop_run(gl.loop);
Packit 5756e2
Packit 5756e2
done:
Packit 5756e2
Packit Service a1bd4f
    if (gl.num_requests_pending > 0) {
Packit Service a1bd4f
        /* this only happens when we quit due to SIGTERM (not due to the idle timer).
Packit Service a1bd4f
         *
Packit Service a1bd4f
         * Log a warning about pending scripts.
Packit Service a1bd4f
         *
Packit Service a1bd4f
         * Maybe we should notify NetworkManager that these scripts are left in an unknown state.
Packit Service a1bd4f
         * But this is either a bug of a dispatcher script (not terminating in time).
Packit Service a1bd4f
         *
Packit Service a1bd4f
         * FIXME(shutdown): Also, currently NetworkManager behaves wrongly on shutdown.
Packit Service a1bd4f
         * Note that systemd would not terminate NetworkManager-dispatcher before NetworkManager.
Packit Service a1bd4f
         * It's NetworkManager's responsibility to keep running long enough so that all requests
Packit Service a1bd4f
         * can complete (with a watchdog timer, and a warning that user provided scripts hang). */
Packit Service a1bd4f
        _LOG_X_W("exiting but there are still %u requests pending", gl.num_requests_pending);
Packit Service a1bd4f
    }
Packit Service a1bd4f
Packit Service a1bd4f
    if (dbus_own_name_id != 0)
Packit Service a1bd4f
        g_bus_unown_name(nm_steal_int(&dbus_own_name_id));
Packit Service a1bd4f
Packit Service a1bd4f
    if (dbus_regist_id != 0)
Packit Service a1bd4f
        g_dbus_connection_unregister_object(gl.dbus_connection, nm_steal_int(&dbus_regist_id));
Packit Service a1bd4f
Packit Service a1bd4f
    nm_clear_pointer(&gl.requests_waiting, g_queue_free);
Packit Service a1bd4f
Packit Service a1bd4f
    nm_clear_g_source(&signal_id_term);
Packit Service a1bd4f
    nm_clear_g_source(&signal_id_int);
Packit Service a1bd4f
    nm_clear_g_source(&gl.quit_id);
Packit Service a1bd4f
    nm_clear_pointer(&gl.loop, g_main_loop_unref);
Packit Service a1bd4f
    g_clear_object(&gl.dbus_connection);
Packit Service a1bd4f
Packit Service a1bd4f
    if (!gl.debug)
Packit Service a1bd4f
        logging_shutdown();
Packit Service a1bd4f
Packit Service a1bd4f
    return gl.exit_with_failure ? 1 : 0;
Packit 5756e2
}