/*
Authors:
Stef Walter <stefw@redhat.com>
Copyright (C) 2014 Red Hat
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "util/util.h"
#include "util/sss_utf8.h"
#include "sbus/sssd_dbus.h"
#include "sbus/sssd_dbus_private.h"
#include <sys/time.h>
#include <dbus/dbus.h>
#define INTERNAL_ERROR "Internal Error"
static int sbus_request_destructor(struct sbus_request *dbus_req)
{
dbus_message_unref(dbus_req->message);
return 0;
}
struct sbus_request *
sbus_new_request(struct sbus_connection *conn,
struct sbus_interface *intf,
DBusMessage *message)
{
struct sbus_request *dbus_req;
dbus_req = talloc_zero(conn, struct sbus_request);
if (!dbus_req) {
DEBUG(SSSDBG_CRIT_FAILURE, "Out of memory allocating DBus request\n");
return NULL;
}
dbus_req->intf = intf;
dbus_req->conn = conn;
dbus_req->message = dbus_message_ref(message);
dbus_req->path = dbus_message_get_path(message);
talloc_set_destructor(dbus_req, sbus_request_destructor);
return dbus_req;
}
void
sbus_request_invoke_or_finish(struct sbus_request *dbus_req,
sbus_msg_handler_fn handler_fn,
void *handler_data,
sbus_method_invoker_fn invoker_fn)
{
DBusError error;
int ret;
if (invoker_fn != NULL) {
ret = invoker_fn(dbus_req, handler_fn);
} else if (handler_fn != NULL) {
ret = handler_fn(dbus_req, handler_data);
} else {
ret = EINVAL;
}
switch(ret) {
case ERR_SBUS_REQUEST_HANDLED:
case EOK:
return;
case ENOMEM:
DEBUG(SSSDBG_CRIT_FAILURE, "Out of memory handling DBus message\n");
sbus_request_finish(dbus_req, NULL);
break;
default:
DEBUG(SSSDBG_CRIT_FAILURE, "Handler failed [%d]: %s\n",
ret, sss_strerror(ret));
dbus_error_init(&error);
dbus_set_error_const(&error, DBUS_ERROR_FAILED, INTERNAL_ERROR);
sbus_request_fail_and_finish(dbus_req, &error);
break;
}
}
int sbus_request_finish(struct sbus_request *dbus_req,
DBusMessage *reply)
{
if (reply) {
sbus_conn_send_reply(dbus_req->conn, reply);
}
return talloc_free(dbus_req);
}
static int sbus_request_valist_check(va_list va, int first_arg_type)
{
int ret = EOK;
#ifdef HAVE_DBUSBASICVALUE
int type;
va_list va_check;
const DBusBasicValue *value;
bool ok;
va_copy(va_check, va);
type = first_arg_type;
while (type != DBUS_TYPE_INVALID) {
value = va_arg(va_check, const DBusBasicValue*);
if (type == DBUS_TYPE_STRING) {
ok = sss_utf8_check((const uint8_t *) value->str,
strlen(value->str));
if (!ok) {
DEBUG(SSSDBG_MINOR_FAILURE,
"sbus message argument [%s] contains invalid "
"non-UTF8 characters\n", value->str);
ret = EINVAL;
break;
}
}
type = va_arg(va_check, int);
}
va_end(va_check);
#endif /* HAVE_DBUSBASICVALUE */
return ret;
}
int sbus_request_return_and_finish(struct sbus_request *dbus_req,
int first_arg_type,
...)
{
DBusMessage *reply;
DBusError error = DBUS_ERROR_INIT;
dbus_bool_t dbret;
va_list va;
int ret;
va_start(va, first_arg_type);
ret = sbus_request_valist_check(va, first_arg_type);
if (ret != EOK) {
va_end(va);
dbus_set_error_const(&error, DBUS_ERROR_INVALID_ARGS, INTERNAL_ERROR);
return sbus_request_fail_and_finish(dbus_req, &error);
}
reply = dbus_message_new_method_return(dbus_req->message);
if (!reply) {
va_end(va);
DEBUG(SSSDBG_CRIT_FAILURE, "Out of memory allocating DBus message\n");
sbus_request_finish(dbus_req, NULL);
return ENOMEM;
}
dbret = dbus_message_append_args_valist(reply, first_arg_type, va);
va_end(va);
if (dbret) {
ret = sbus_request_finish(dbus_req, reply);
} else {
DEBUG(SSSDBG_CRIT_FAILURE, "Couldn't build DBus message\n");
sbus_request_finish(dbus_req, NULL);
ret = EINVAL;
}
dbus_message_unref(reply);
return ret;
}
int sbus_request_fail_and_finish(struct sbus_request *dbus_req,
const DBusError *error)
{
DBusMessage *reply;
int ret;
if (error == NULL) {
sbus_request_finish(dbus_req, NULL);
return ENOMEM;
}
reply = dbus_message_new_error(dbus_req->message, error->name, error->message);
if (!reply) {
DEBUG(SSSDBG_CRIT_FAILURE, "Out of memory allocating DBus message\n");
sbus_request_finish(dbus_req, NULL);
return ENOMEM;
}
ret = sbus_request_finish(dbus_req, reply);
dbus_message_unref(reply);
return ret;
}
static DBusError *sbus_error_new_va(TALLOC_CTX *mem_ctx,
const char *error_name,
const char *fmt,
va_list ap)
{
DBusError *error;
const char *error_msg;
error = talloc_zero(mem_ctx, DBusError);
if (error == NULL) {
return NULL;
}
if (fmt != NULL) {
error_msg = talloc_vasprintf(error, fmt, ap);
if (error_msg == NULL) {
talloc_free(error);
return NULL;
}
} else {
error_msg = NULL;
}
dbus_error_init(error);
dbus_set_error_const(error, error_name, error_msg);
return error;
}
DBusError *sbus_error_new(TALLOC_CTX *mem_ctx,
const char *dbus_error_name,
const char *fmt,
...)
{
DBusError *error;
va_list ap;
va_start(ap, fmt);
error = sbus_error_new_va(mem_ctx, dbus_error_name, fmt, ap);
va_end(ap);
return error;
}
void sbus_request_reply_error(struct sbus_request *sbus_req,
const char *error_name,
const char *fmt,
...)
{
DBusError *error;
va_list ap;
va_start(ap, fmt);
error = sbus_error_new_va(sbus_req, error_name, fmt, ap);
va_end(ap);
if (error == NULL) {
DEBUG(SSSDBG_CRIT_FAILURE,
"Unable to create D-Bus error, killing request!\n");
talloc_free(sbus_req);
return;
}
sbus_request_fail_and_finish(sbus_req, error);
}
struct array_arg {
char **dbus_array;
};
static int array_arg_destructor(struct array_arg *arg)
{
dbus_free_string_array(arg->dbus_array);
return 0;
}
static bool
parent_dbus_string_arrays(struct sbus_request *request, int first_arg_type,
va_list va)
{
struct array_arg *array_arg;
int arg_type;
void **arg_ptr;
/*
* Here we iterate through the entire thing again and look for
* things we need to fix allocation for. Normally certain types
* returned from dbus_message_get_args() and friends require
* later freeing. We tie those to the talloc context here.
*
* The list of argument has already been validated by the previous
* dbus_message_get_args() call, so we can be cheap.
*/
arg_type = first_arg_type;
while (arg_type != DBUS_TYPE_INVALID) {
if (arg_type == DBUS_TYPE_ARRAY) {
arg_type = va_arg(va, int); /* the array element type */
arg_ptr = va_arg(va, void **); /* the array elements */
va_arg(va, int *); /* the array length */
/* Arrays of these things need to be freed */
if (arg_type == DBUS_TYPE_STRING ||
arg_type == DBUS_TYPE_OBJECT_PATH ||
arg_type == DBUS_TYPE_SIGNATURE) {
array_arg = talloc_zero(request, struct array_arg);
if (array_arg == NULL) {
/* no kidding ... */
DEBUG(SSSDBG_CRIT_FAILURE, "Out of memory while trying not to leak memory\n");
return false;
}
array_arg->dbus_array = *arg_ptr;
talloc_set_destructor(array_arg, array_arg_destructor);
}
/* A non array argument */
} else {
arg_ptr = va_arg(va, void**);
}
/* The next type */
arg_type = va_arg(va, int);
}
return true;
}
bool
sbus_request_parse_or_finish(struct sbus_request *request,
int first_arg_type,
...)
{
DBusError error = DBUS_ERROR_INIT;
bool ret = true;
va_list va2;
va_list va;
va_start(va, first_arg_type);
va_copy(va2, va);
if (dbus_message_get_args_valist(request->message, &error,
first_arg_type, va)) {
ret = parent_dbus_string_arrays(request, first_arg_type, va2);
} else {
/* Trying to send the error back to the caller in this case is a joke */
if (!dbus_error_is_set(&error) &&
dbus_error_has_name(&error, DBUS_ERROR_NO_MEMORY)) {
DEBUG(SSSDBG_CRIT_FAILURE, "Out of memory parsing DBus message\n");
sbus_request_finish(request, NULL);
/* Log other errors and send them back, this include o.f.d.InvalidArgs */
} else {
DEBUG(SSSDBG_OP_FAILURE, "Couldn't parse DBus message %s.%s: %s\n",
dbus_message_get_interface(request->message),
dbus_message_get_member(request->message),
error.message);
sbus_request_fail_and_finish(request, &error);
}
dbus_error_free(&error);
ret = false;
}
va_end(va2);
va_end(va);
return ret;
}
struct sbus_get_sender_id_state {
struct sbus_connection *conn;
DBusConnection *sysbus_conn;
char *sender;
int64_t uid;
};
static void sbus_get_sender_id_done(DBusPendingCall *pending, void *ptr);
struct tevent_req *sbus_get_sender_id_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct sbus_connection *conn,
const char *sender)
{
struct tevent_req *req;
struct sbus_get_sender_id_state *state;
DBusError dbus_error;
DBusMessage *msg = NULL;
dbus_bool_t dbret;
errno_t ret;
hash_key_t key;
hash_value_t value;
req = tevent_req_create(mem_ctx, &state, struct sbus_get_sender_id_state);
if (req == NULL) {
return NULL;
}
state->conn = conn;
state->uid = -1;
if (conn->connection_type != SBUS_CONN_TYPE_SYSBUS) {
DEBUG(SSSDBG_TRACE_INTERNAL, "Not a sysbus message, quit\n");
ret = EOK;
goto immediate;
}
if (sender == NULL) {
ret = ERR_SBUS_NO_SENDER;
goto immediate;
}
if (strcmp(sender, "org.freedesktop.DBus") == 0) {
ret = ERR_SBUS_SENDER_BUS;
goto immediate;
}
state->sender = talloc_strdup(state, sender);
if (state->sender == NULL) {
ret = ENOMEM;
goto immediate;
}
DEBUG(SSSDBG_TRACE_INTERNAL,
"Looking for identity of sender [%s]\n", sender);
key.type = HASH_KEY_STRING;
key.str = discard_const(sender);
ret = hash_lookup(conn->clients, &key, &value);
if (ret == HASH_SUCCESS) {
DEBUG(SSSDBG_TRACE_INTERNAL,
"%s already present in the clients table\n", sender);
state->uid = (int64_t) value.ul;
ret = EOK;
goto immediate;
} else if (ret != HASH_ERROR_KEY_NOT_FOUND) {
DEBUG(SSSDBG_CRIT_FAILURE,
"Failed to look up %s in the clients table\n", sender);
ret = ERR_SBUS_GET_SENDER_ERROR;
goto immediate;
}
/* We don't know this sender yet, let's ask the system bus */
/* Connect to the well-known system bus */
dbus_error_init(&dbus_error);
state->sysbus_conn = dbus_bus_get(DBUS_BUS_SYSTEM, &dbus_error);
if (state->sysbus_conn == NULL) {
DEBUG(SSSDBG_CRIT_FAILURE,
"Failed to connect to D-BUS system bus.\n");
ret = ERR_SBUS_GET_SENDER_ERROR;
goto immediate;
}
dbus_connection_set_exit_on_disconnect(state->sysbus_conn, FALSE);
/* If we ever need to get the SELinux context or the PID here, we need
* to call GetConnectionCredentials instead
*/
msg = dbus_message_new_method_call("org.freedesktop.DBus", /* bus name */
"/org/freedesktop/DBus", /* path */
"org.freedesktop.DBus", /* interface */
"GetConnectionUnixUser");
if (msg == NULL) {
DEBUG(SSSDBG_CRIT_FAILURE, "Out of memory?!\n");
ret = ENOMEM;
goto immediate;
}
dbret = dbus_message_append_args(msg,
DBUS_TYPE_STRING, &sender,
DBUS_TYPE_INVALID);
if (!dbret) {
ret = ERR_INTERNAL;
goto immediate;
}
ret = sss_dbus_conn_send(state->sysbus_conn, msg, 3000,
sbus_get_sender_id_done,
req, NULL);
dbus_message_unref(msg);
msg = NULL;
if (ret != EOK) {
goto immediate;
}
return req;
immediate:
if (ret == EOK) {
tevent_req_done(req);
} else {
if (msg != NULL) {
dbus_message_unref(msg);
}
if (state->sysbus_conn != NULL) {
dbus_connection_unref(state->sysbus_conn);
}
tevent_req_error(req, ret);
}
tevent_req_post(req, ev);
return req;
}
static void sbus_get_sender_id_done(DBusPendingCall *pending, void *ptr)
{
struct tevent_req *req;
struct sbus_get_sender_id_state *state;
DBusMessage *reply;
DBusError dbus_error;
hash_key_t key;
hash_value_t value;
dbus_bool_t dbret;
int ret;
uid_t uid;
dbus_error_init(&dbus_error);
req = talloc_get_type(ptr, struct tevent_req);
state = tevent_req_data(req, struct sbus_get_sender_id_state);
reply = dbus_pending_call_steal_reply(pending);
if (!reply) {
/* reply should never be null. This function shouldn't be called
* until reply is valid or timeout has occurred. If reply is NULL
* here, something is seriously wrong and we should bail out.
*/
DEBUG(SSSDBG_CRIT_FAILURE,
"Severe error. A reply callback was called but no reply "
"was received and no timeout occurred\n");
ret = EIO;
goto done;
}
dbret = dbus_message_get_args(reply,
&dbus_error,
DBUS_TYPE_UINT32, &uid,
DBUS_TYPE_INVALID);
if (!dbret) {
DEBUG(SSSDBG_CRIT_FAILURE, "Could not parse reply!\n");
ret = EIO;
goto done;
}
state->uid = uid;
key.type = HASH_KEY_STRING;
key.str = talloc_steal(state->conn->clients, state->sender);
value.type = HASH_VALUE_UINT;
value.ul = state->uid;
ret = hash_enter(state->conn->clients, &key, &value);
if (ret != HASH_SUCCESS) {
DEBUG(SSSDBG_CRIT_FAILURE, "Could not add key to hash table!\n");
ret = EIO;
goto done;
}
ret = EOK;
done:
dbus_pending_call_unref(pending);
dbus_message_unref(reply);
dbus_connection_unref(state->sysbus_conn);
if (ret != EOK) {
tevent_req_error(req, ret);
} else {
tevent_req_done(req);
}
}
int sbus_get_sender_id_recv(struct tevent_req *req, int64_t *_uid)
{
struct sbus_get_sender_id_state *state = \
tevent_req_data(req, struct sbus_get_sender_id_state);
TEVENT_REQ_RETURN_ON_ERROR(req);
if (_uid) {
*_uid = state->uid;
}
return EOK;
}