2013-08-22 13:27:19 +00:00
|
|
|
/*
|
|
|
|
* virpolkit.c: helpers for using polkit APIs
|
|
|
|
*
|
2016-01-14 19:34:28 +00:00
|
|
|
* Copyright (C) 2013, 2014, 2016 Red Hat, Inc.
|
2013-08-22 13:27:19 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library. If not, see
|
|
|
|
* <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
2020-02-16 21:59:28 +00:00
|
|
|
#include <unistd.h>
|
2013-08-22 13:27:19 +00:00
|
|
|
|
|
|
|
#include "virpolkit.h"
|
|
|
|
#include "virerror.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virstring.h"
|
|
|
|
#include "virprocess.h"
|
|
|
|
#include "viralloc.h"
|
2020-09-09 14:42:16 +00:00
|
|
|
#include "virgdbus.h"
|
2016-02-09 15:09:44 +00:00
|
|
|
#include "virfile.h"
|
2020-02-16 21:59:28 +00:00
|
|
|
#include "virutil.h"
|
2013-08-22 13:27:19 +00:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_POLKIT
|
|
|
|
|
|
|
|
VIR_LOG_INIT("util.polkit");
|
|
|
|
|
2018-03-07 09:14:23 +00:00
|
|
|
#if WITH_POLKIT
|
2020-01-17 11:19:13 +00:00
|
|
|
# include <poll.h>
|
2016-02-09 15:09:44 +00:00
|
|
|
|
|
|
|
struct _virPolkitAgent {
|
|
|
|
virCommandPtr cmd;
|
|
|
|
};
|
|
|
|
|
2013-08-22 13:27:19 +00:00
|
|
|
/*
|
|
|
|
* virPolkitCheckAuth:
|
|
|
|
* @actionid: permission to check
|
|
|
|
* @pid: client process ID
|
|
|
|
* @startTime: process start time, or 0
|
|
|
|
* @uid: client process user ID
|
|
|
|
* @details: NULL terminated (key, value) pair list
|
|
|
|
* @allowInteraction: true if auth prompts are allowed
|
|
|
|
*
|
|
|
|
* Check if a client is authenticated with polkit
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on failure, -2 on auth denied
|
|
|
|
*/
|
|
|
|
int virPolkitCheckAuth(const char *actionid,
|
|
|
|
pid_t pid,
|
|
|
|
unsigned long long startTime,
|
|
|
|
uid_t uid,
|
|
|
|
const char **details,
|
|
|
|
bool allowInteraction)
|
|
|
|
{
|
2020-09-09 14:42:16 +00:00
|
|
|
GDBusConnection *sysbus;
|
|
|
|
GVariantBuilder builder;
|
|
|
|
GVariant *gprocess = NULL;
|
|
|
|
GVariant *gdetails = NULL;
|
|
|
|
g_autoptr(GVariant) message = NULL;
|
|
|
|
g_autoptr(GVariant) reply = NULL;
|
|
|
|
g_autoptr(GVariantIter) iter = NULL;
|
|
|
|
char *retkey;
|
|
|
|
char *retval;
|
|
|
|
gboolean is_authorized;
|
|
|
|
gboolean is_challenge;
|
2014-09-10 13:52:48 +00:00
|
|
|
bool is_dismissed = false;
|
|
|
|
size_t i;
|
2013-08-22 13:27:19 +00:00
|
|
|
|
2020-09-09 14:42:16 +00:00
|
|
|
if (!(sysbus = virGDBusGetSystemBus()))
|
|
|
|
return -1;
|
2013-08-22 13:27:19 +00:00
|
|
|
|
2014-09-10 13:52:48 +00:00
|
|
|
VIR_INFO("Checking PID %lld running as %d",
|
|
|
|
(long long) pid, uid);
|
2013-08-22 13:27:19 +00:00
|
|
|
|
2020-09-09 14:42:16 +00:00
|
|
|
g_variant_builder_init(&builder, G_VARIANT_TYPE("a{sv}"));
|
|
|
|
g_variant_builder_add(&builder, "{sv}", "pid", g_variant_new_uint32(pid));
|
|
|
|
g_variant_builder_add(&builder, "{sv}", "start-time", g_variant_new_uint64(startTime));
|
|
|
|
g_variant_builder_add(&builder, "{sv}", "uid", g_variant_new_int32(uid));
|
|
|
|
gprocess = g_variant_builder_end(&builder);
|
|
|
|
|
|
|
|
g_variant_builder_init(&builder, G_VARIANT_TYPE("a{ss}"));
|
|
|
|
for (i = 0; i < virStringListLength(details); i += 2)
|
|
|
|
g_variant_builder_add(&builder, "{ss}", details[i], details[i + 1]);
|
|
|
|
gdetails = g_variant_builder_end(&builder);
|
|
|
|
|
|
|
|
message = g_variant_new("((s@a{sv})s@a{ss}us)",
|
|
|
|
"unix-process",
|
|
|
|
gprocess,
|
|
|
|
actionid,
|
|
|
|
gdetails,
|
|
|
|
allowInteraction,
|
|
|
|
"" /* cancellation ID */);
|
|
|
|
|
|
|
|
if (virGDBusCallMethod(sysbus,
|
|
|
|
&reply,
|
2020-09-21 13:42:00 +00:00
|
|
|
G_VARIANT_TYPE("((bba{ss}))"),
|
2020-09-09 14:42:16 +00:00
|
|
|
NULL,
|
|
|
|
"org.freedesktop.PolicyKit1",
|
|
|
|
"/org/freedesktop/PolicyKit1/Authority",
|
|
|
|
"org.freedesktop.PolicyKit1.Authority",
|
|
|
|
"CheckAuthorization",
|
|
|
|
message) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
g_variant_get(reply, "((bba{ss}))", &is_authorized, &is_challenge, &iter);
|
|
|
|
|
|
|
|
while (g_variant_iter_loop(iter, "{ss}", &retkey, &retval)) {
|
|
|
|
if (STREQ(retkey, "polkit.dismissed") && STREQ(retval, "true"))
|
2014-09-10 13:52:48 +00:00
|
|
|
is_dismissed = true;
|
|
|
|
}
|
2013-08-22 13:27:19 +00:00
|
|
|
|
2014-09-10 13:52:48 +00:00
|
|
|
VIR_DEBUG("is auth %d is challenge %d",
|
|
|
|
is_authorized, is_challenge);
|
2013-08-22 13:27:19 +00:00
|
|
|
|
2020-09-09 14:42:16 +00:00
|
|
|
if (is_authorized)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (is_dismissed) {
|
|
|
|
virReportError(VIR_ERR_AUTH_CANCELLED, "%s",
|
|
|
|
_("user cancelled authentication process"));
|
|
|
|
} else if (is_challenge) {
|
|
|
|
virReportError(VIR_ERR_AUTH_UNAVAILABLE,
|
|
|
|
_("no polkit agent available to authenticate action '%s'"),
|
|
|
|
actionid);
|
2014-09-10 13:52:48 +00:00
|
|
|
} else {
|
2020-09-09 14:42:16 +00:00
|
|
|
virReportError(VIR_ERR_AUTH_FAILED, "%s",
|
|
|
|
_("access denied by policy"));
|
2013-08-22 13:27:19 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 14:42:16 +00:00
|
|
|
return -2;
|
2013-08-22 13:27:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-09 15:09:44 +00:00
|
|
|
/* virPolkitAgentDestroy:
|
|
|
|
* @cmd: Pointer to the virCommandPtr created during virPolkitAgentCreate
|
|
|
|
*
|
|
|
|
* Destroy resources used by Polkit Agent
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
virPolkitAgentDestroy(virPolkitAgentPtr agent)
|
|
|
|
{
|
|
|
|
if (!agent)
|
|
|
|
return;
|
|
|
|
|
|
|
|
virCommandFree(agent->cmd);
|
|
|
|
VIR_FREE(agent);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* virPolkitAgentCreate:
|
|
|
|
*
|
|
|
|
* Allocate and setup a polkit agent
|
|
|
|
*
|
|
|
|
* Returns a virCommandPtr on success and NULL on failure
|
|
|
|
*/
|
|
|
|
virPolkitAgentPtr
|
|
|
|
virPolkitAgentCreate(void)
|
|
|
|
{
|
2016-03-02 00:36:37 +00:00
|
|
|
virPolkitAgentPtr agent = NULL;
|
2016-02-09 15:09:44 +00:00
|
|
|
int pipe_fd[2] = {-1, -1};
|
|
|
|
struct pollfd pollfd;
|
|
|
|
int outfd = STDOUT_FILENO;
|
|
|
|
int errfd = STDERR_FILENO;
|
|
|
|
|
|
|
|
if (!isatty(STDIN_FILENO))
|
|
|
|
goto error;
|
|
|
|
|
2020-01-24 15:22:12 +00:00
|
|
|
if (virPipe(pipe_fd) < 0)
|
2016-02-09 15:09:44 +00:00
|
|
|
goto error;
|
|
|
|
|
2020-10-05 17:13:12 +00:00
|
|
|
agent = g_new0(virPolkitAgent, 1);
|
2016-05-09 09:02:51 +00:00
|
|
|
|
|
|
|
agent->cmd = virCommandNewArgList(PKTTYAGENT, "--process", NULL);
|
|
|
|
|
|
|
|
virCommandAddArgFormat(agent->cmd, "%lld", (long long int) getpid());
|
|
|
|
virCommandAddArg(agent->cmd, "--notify-fd");
|
|
|
|
virCommandAddArgFormat(agent->cmd, "%d", pipe_fd[1]);
|
|
|
|
virCommandAddArg(agent->cmd, "--fallback");
|
|
|
|
virCommandSetInputFD(agent->cmd, STDIN_FILENO);
|
|
|
|
virCommandSetOutputFD(agent->cmd, &outfd);
|
|
|
|
virCommandSetErrorFD(agent->cmd, &errfd);
|
|
|
|
virCommandPassFD(agent->cmd, pipe_fd[1], VIR_COMMAND_PASS_FD_CLOSE_PARENT);
|
2019-04-30 09:17:22 +00:00
|
|
|
pipe_fd[1] = -1;
|
2016-05-09 09:02:51 +00:00
|
|
|
if (virCommandRunAsync(agent->cmd, NULL) < 0)
|
2016-02-09 15:09:44 +00:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
pollfd.fd = pipe_fd[0];
|
|
|
|
pollfd.events = POLLHUP;
|
|
|
|
|
|
|
|
if (poll(&pollfd, 1, -1) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
return agent;
|
|
|
|
|
|
|
|
error:
|
|
|
|
VIR_FORCE_CLOSE(pipe_fd[0]);
|
|
|
|
VIR_FORCE_CLOSE(pipe_fd[1]);
|
|
|
|
virPolkitAgentDestroy(agent);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-07 09:14:23 +00:00
|
|
|
#else /* ! WITH_POLKIT */
|
2013-08-22 13:27:19 +00:00
|
|
|
|
2019-10-14 12:45:33 +00:00
|
|
|
int virPolkitCheckAuth(const char *actionid G_GNUC_UNUSED,
|
|
|
|
pid_t pid G_GNUC_UNUSED,
|
|
|
|
unsigned long long startTime G_GNUC_UNUSED,
|
|
|
|
uid_t uid G_GNUC_UNUSED,
|
|
|
|
const char **details G_GNUC_UNUSED,
|
|
|
|
bool allowInteraction G_GNUC_UNUSED)
|
2013-08-22 13:27:19 +00:00
|
|
|
{
|
|
|
|
VIR_ERROR(_("Polkit auth attempted, even though polkit is not available"));
|
|
|
|
virReportError(VIR_ERR_AUTH_FAILED, "%s",
|
|
|
|
_("authentication failed"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-09 15:09:44 +00:00
|
|
|
void
|
2019-10-14 12:45:33 +00:00
|
|
|
virPolkitAgentDestroy(virPolkitAgentPtr agent G_GNUC_UNUSED)
|
2016-02-09 15:09:44 +00:00
|
|
|
{
|
|
|
|
return; /* do nothing */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-03-01 14:51:37 +00:00
|
|
|
virPolkitAgentPtr
|
2016-02-09 15:09:44 +00:00
|
|
|
virPolkitAgentCreate(void)
|
|
|
|
{
|
|
|
|
virReportError(VIR_ERR_AUTH_FAILED, "%s",
|
|
|
|
_("polkit text authentication agent unavailable"));
|
|
|
|
return NULL;
|
|
|
|
}
|
2018-03-07 09:14:23 +00:00
|
|
|
#endif /* WITH_POLKIT */
|