Blob Blame History Raw
// SPDX-License-Identifier: GPL-2.0+
/*
 * Copyright (C) 2014 Red Hat, Inc.
 */

#include "nm-default.h"

#include "polkit-agent.h"

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

#include "nm-polkit-listener.h"
#include "common.h"

static char *
polkit_read_passwd (gpointer instance,
                    const char *action_id,
                    const char *message,
                    const char *user,
                    gpointer user_data)
{
	NmCli *nmc = user_data;

	g_print ("%s\n", message);
	g_print ("(action_id: %s)\n", action_id);

	/* Ask user for polkit authorization password */
	if (user) {
		return nmc_readline_echo (&nmc->nmc_config, FALSE, "password (%s): ", user);
	}
	return nmc_readline_echo (&nmc->nmc_config, FALSE, "password: ");
}

static void
polkit_error (gpointer instance,
              const char *error,
              gpointer user_data)
{
	g_printerr (_("Error: polkit agent failed: %s\n"), error);
}

gboolean
nmc_polkit_agent_init (NmCli* nmc, gboolean for_session, GError **error)
{
	NMPolkitListener *listener;
	GDBusConnection *dbus_connection = NULL;

	g_return_val_if_fail (error == NULL || *error == NULL, FALSE);

	if (nmc->client && nm_client_get_dbus_connection (nmc->client)) {
		dbus_connection = nm_client_get_dbus_connection (nmc->client);
		listener = nm_polkit_listener_new (dbus_connection, for_session);
	} else {
		dbus_connection = g_bus_get_sync (G_BUS_TYPE_SYSTEM,
		                                  NULL,
		                                  error);

		if (!dbus_connection) {
			return FALSE;
		}

		listener = nm_polkit_listener_new (dbus_connection, for_session);
		g_object_unref (dbus_connection);
	}

	g_signal_connect (listener,
	                  NM_POLKIT_LISTENER_SIGNAL_REQUEST_SYNC,
	                  G_CALLBACK (polkit_read_passwd),
	                  nmc);
	g_signal_connect (listener,
	                  NM_POLKIT_LISTENER_SIGNAL_ERROR,
	                  G_CALLBACK (polkit_error),
	                  NULL);

	nmc->pk_listener = listener;
	return TRUE;
}

void
nmc_polkit_agent_fini (NmCli* nmc)
{
	if (nmc->pk_listener) {
		g_clear_object (&nmc->pk_listener);
	}
}

gboolean
nmc_start_polkit_agent_start_try (NmCli *nmc)
{
	gs_free_error GError *error = NULL;

	/* We don't register polkit agent at all when running non-interactively */
	if (!nmc->ask)
		return TRUE;

	if (!nmc_polkit_agent_init (nmc, FALSE, &error)) {
		g_printerr (_("Warning: polkit agent initialization failed: %s\n"),
		            error->message);
		return FALSE;
	}
	return TRUE;
}