/* SSSD SSSD - D-BUS interface Copyright (C) Stephen Gallagher 2008 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 . */ #ifndef _SSSD_DBUS_H_ #define _SSSD_DBUS_H_ struct sbus_connection; struct sbus_interface; struct sbus_request; #include #include #include "util/util.h" /* Older platforms (such as RHEL-6) might not have these error constants * defined */ #ifndef DBUS_ERROR_UNKNOWN_INTERFACE /** Interface you invoked a method on isn't known by the object. */ #define DBUS_ERROR_UNKNOWN_INTERFACE \ "org.freedesktop.DBus.Error.UnknownInterface" #endif /* DBUS_ERROR_UNKNOWN_INTERFACE */ #ifndef DBUS_ERROR_UNKNOWN_PROPERTY /** Property you tried to access isn't known by the object. */ #define DBUS_ERROR_UNKNOWN_PROPERTY \ "org.freedesktop.DBus.Error.UnknownProperty" #endif /* DBUS_ERROR_UNKNOWN_PROPERTY */ #ifndef DBUS_ERROR_PROPERTY_READ_ONLY /** Property you tried to set is read-only. */ #define DBUS_ERROR_PROPERTY_READ_ONLY \ "org.freedesktop.DBus.Error.PropertyReadOnly" #endif /* DBUS_ERROR_PROPERTY_READ_ONLY */ #ifndef DBUS_ERROR_INIT #define DBUS_ERROR_INIT { NULL, NULL, TRUE, 0, 0, 0, 0, NULL } #endif /* DBUS_ERROR_INIT */ /** * Note: internal functions do not rely on the value of this constant to * simplify implementation. If this constant change, some functions in * sssd_dbus_interface.c needs to be amended. */ #define SBUS_SUBTREE_SUFFIX "/*" /** * It is not possible to send NULL over D-Bus. We can only test if it * is empty or not. */ #define SBUS_IS_STRING_EMPTY(str) ((str) == NULL || (str)[0] == '\0') #define SBUS_SET_STRING(str) (SBUS_IS_STRING_EMPTY(str) ? NULL : (str)) typedef int (*sbus_msg_handler_fn)(struct sbus_request *dbus_req, void *handler_data); /* * sbus_conn_destructor_fn * Function to be called when a connection is finalized */ typedef int (*sbus_conn_destructor_fn)(void *); typedef void (*sbus_conn_reconn_callback_fn)(struct sbus_connection *, int, void *); /* * sbus_server_conn_init_fn * Set up function for connection-specific activities * This function should define the sbus_conn_destructor_fn * for this connection at a minimum */ typedef int (*sbus_server_conn_init_fn)(struct sbus_connection *, void *); typedef const char ** (* sbus_nodes_fn)(TALLOC_CTX *mem_ctx, const char *path, void *data); enum { SBUS_CONN_TYPE_PRIVATE = 1, SBUS_CONN_TYPE_SHARED, SBUS_CONN_TYPE_SYSBUS }; enum { SBUS_RECONNECT_SUCCESS = 1, SBUS_RECONNECT_EXCEEDED_RETRIES, SBUS_RECONNECT_ERROR }; /* * This represents vtable of interface handlers for methods and * properties and so on. The actual vtable structs derive from this struct * (ie: have this struct as their first member). * * The offsets for matching vtable function pointers are in sbus_method_meta * These are used to dynamically dispatch the method invocations. */ struct sbus_vtable { const struct sbus_interface_meta *meta; int flags; /* unused for now */ /* derived structs place function pointers here. */ }; /* Special interface and method for D-BUS introspection */ #define DBUS_INTROSPECT_INTERFACE "org.freedesktop.DBus.Introspectable" #define DBUS_INTROSPECT_METHOD "Introspect" /* Special interface and method for D-BUS properties */ #define DBUS_PROPERTIES_INTERFACE "org.freedesktop.DBus.Properties" struct sbus_interface { const char *path; struct sbus_vtable *vtable; void *handler_data; }; /* Server Functions */ int sbus_new_server(TALLOC_CTX *mem_ctx, struct tevent_context *ev, const char *address, uid_t uid, gid_t gid, bool use_symlink, struct sbus_connection **server, sbus_server_conn_init_fn init_fn, void *init_pvt_data); /* Connection Functions */ /* sbus_new_connection * Use this function when connecting a new process to * the standard SSSD interface. * This will connect to the address specified and then * call sbus_add_connection to integrate with the main * loop. */ int sbus_new_connection(TALLOC_CTX *ctx, struct tevent_context *ev, const char *address, struct sbus_connection **conn); /* sbus_add_connection * Integrates a D-BUS connection with the TEvent main * loop. Use this function when you already have a * DBusConnection object (for example from dbus_bus_get) * Connection type can be either: * SBUS_CONN_TYPE_PRIVATE: Used only from within a D-BUS * server such as the Monitor in the * new_connection_callback * SBUS_CONN_TYPE_SHARED: Used for all D-BUS client * connections, including those retrieved from * dbus_bus_get */ int sbus_init_connection(TALLOC_CTX *ctx, struct tevent_context *ev, DBusConnection *dbus_conn, int connection_type, struct sbus_connection **_conn); DBusConnection *sbus_get_connection(struct sbus_connection *conn); void sbus_disconnect(struct sbus_connection *conn); /* * Register a new interface to be available at given object path. * * The interface will be exported at @object_path. The method handlers are * represented by @iface_vtable. @pvt contains additional caller specific data * which is made available to handlers. */ int sbus_conn_register_iface(struct sbus_connection *conn, struct sbus_vtable *iface_vtable, const char *object_path, void *handler_data); struct sbus_iface_map { const char *path; struct sbus_vtable *vtable; }; errno_t sbus_conn_register_iface_map(struct sbus_connection *conn, struct sbus_iface_map *map, void *pvt); void sbus_conn_register_nodes(struct sbus_connection *conn, const char *path, sbus_nodes_fn nodes_fn, void *data); errno_t sbus_conn_reregister_paths(struct sbus_connection *conn); char * sbus_opath_escape_part(TALLOC_CTX *mem_ctx, const char *object_path_part); char * sbus_opath_unescape_part(TALLOC_CTX *mem_ctx, const char *object_path_part); char * _sbus_opath_compose(TALLOC_CTX *mem_ctx, const char *base, const char *part, ...); #define sbus_opath_compose(mem_ctx, base, ...) \ _sbus_opath_compose(mem_ctx, base, ##__VA_ARGS__, NULL) errno_t sbus_opath_decompose(TALLOC_CTX *mem_ctx, const char *object_path, const char *prefix, char ***_components, size_t *_len); errno_t sbus_opath_decompose_exact(TALLOC_CTX *mem_ctx, const char *object_path, const char *prefix, size_t expected, char ***_components); const char * sbus_opath_strip_prefix(const char *object_path, const char *prefix); char * sbus_opath_get_object_name(TALLOC_CTX *mem_ctx, const char *object_path, const char *base_path); bool sbus_conn_disconnecting(struct sbus_connection *conn); /* max_retries < 0: retry forever * max_retries = 0: never retry (why are you calling this function?) * max_retries > 0: obvious */ void sbus_reconnect_init(struct sbus_connection *conn, int max_retries, sbus_conn_reconn_callback_fn callback, void *pvt); /* * Send a message across the SBUS * If requested, the DBusPendingCall object will * be returned to the caller. * * This function will return EAGAIN in the event * that the connection is not open for * communication. */ int sbus_conn_send(struct sbus_connection *conn, DBusMessage *msg, int timeout_ms, DBusPendingCallNotifyFunction reply_handler, void *pvt, DBusPendingCall **pending); void sbus_conn_send_reply(struct sbus_connection *conn, DBusMessage *reply); /* Set up D-BUS access control. If there is a SSSD user, we must allow * him to connect. root is always allowed */ void sbus_allow_uid(struct sbus_connection *conn, uid_t *uid); /* * This structure is passed to all dbus method and property * handlers. It is a talloc context which will be valid until * the request is completed with either the sbus_request_complete() * or sbus_request_fail() functions. */ struct sbus_request { int64_t client; struct sbus_connection *conn; DBusMessage *message; struct sbus_interface *intf; const struct sbus_method_meta *method; const char *path; }; /* * Complete a DBus request, and free the @dbus_req context. The @dbus_req * and associated talloc context are no longer valid after this function * returns. * * If @reply is non-NULL then the reply is sent to the caller. Not sending * a reply when the caller is expecting one is fairly rude behavior. * * The return value is useful for logging, but not much else. In particular * even if this function return !EOK, @dbus_req is still unusable after this * function returns. */ int sbus_request_finish(struct sbus_request *dbus_req, DBusMessage *reply); /* * Return a reply for a DBus method call request. The variable * arguments are (unfortunately) formatted exactly the same as those of the * dbus_message_append_args() function. Documented here: * * http://dbus.freedesktop.org/doc/api/html/group__DBusMessage.html * * Important: don't pass int or bool or such types as * values to this function. That's not portable. Use actual dbus types. * You must also pass pointers as the values: * * dbus_bool_t val1 = TRUE; * dbus_int32_t val2 = 5; * ret = sbus_request_finish(dbus_req, * DBUS_TYPE_BOOLEAN, &val1, * DBUS_TYPE_INT32, &val2, * DBUS_TYPE_INVALID); * * To pass arrays to this function, use the following syntax. Never * pass actual C arrays with [] syntax to this function. The C standard is * rather vague with C arrays and varargs, and it just plain doesn't work. * * const char *array[] = { "one", "two", "three" }; * int count = 3; // yes, a plain int * const char **ptr = array; * ret = sbus_request_finish(dbus_req, * DBUS_TYPE_ARRAY, DBUS_TYPE_STRING, &ptr, 3, * DBUS_TYPE_INVALID); * * The @dbus_req and associated talloc context are no longer valid after this * function returns, even if this function returns an error code. */ int sbus_request_return_and_finish(struct sbus_request *dbus_req, int first_arg_type, ...); /* * Return an error for a DBus method call request. The @error is a normal * DBusError. * * The @dbus_req and associated talloc context are no longer valid after this * function returns, even if this function returns an error code. */ int sbus_request_fail_and_finish(struct sbus_request *dbus_req, const DBusError *error); /* * Construct a new DBusError instance which can be consumed by functions such * as @sbus_request_fail_and_finish(). * * The @error is a string constant representing a DBus error as documented at * http://dbus.freedesktop.org/doc/api/html/group__DBusProtocol.html. * The parameter @err_msg is a human-readable error representation (or * NULL for none). The returned DBusError is a talloc context and the err_msg * is duplicated using the returned DBusError instance as a talloc parent. */ DBusError *sbus_error_new(TALLOC_CTX *mem_ctx, const char *dbus_err_name, const char *fmt, ...) SSS_ATTRIBUTE_PRINTF(3,4); /* * Parse a DBus method call request. * * If parsing the method call message does not succeed, then an error is * sent to the DBus caller and the request is finished. If this function * returns false then @request is no longer valid. * * This also means if this method returns false within a handler, you should * return EOK from the handler. The message has been handled, appropriate * logs have been written, and everything should just move on. * * If the method call does not match the expected arguments, then a * org.freedesktop.DBus.Error.InvalidArgs is returned to the caller as * expected. * * The variable arguments are (unfortunately) formatted exactly the same * as those of the dbus_message_get_args() function. Documented here: * * http://dbus.freedesktop.org/doc/api/html/group__DBusMessage.html * * Exception: You don't need to free string arrays returned by this * function. They are automatically talloc parented to the request memory * context and can be used until the request has been finished. * * Important: don't pass int or bool or such types as values to this * function. That's not portable. Use actual dbus types. You must also pass * pointers as the values: * * dbus_bool_t val1; * dbus_int32_t val2; * ret = sbus_request_parse_or_finish(request, * DBUS_TYPE_BOOLEAN, &val1, * DBUS_TYPE_INT32, &val2, * DBUS_TYPE_INVALID); * * To pass arrays to this function, use the following syntax. Never * pass actual C arrays with [] syntax to this function. The C standard is * rather vague with C arrays and varargs, and it just plain doesn't work. * * int count; // yes, a plain int * const char **array; * ret = sbus_request_parse_or_finish(request, * DBUS_TYPE_ARRAY, DBUS_TYPE_STRING, &array, &count, * DBUS_TYPE_INVALID); */ bool sbus_request_parse_or_finish(struct sbus_request *request, int first_arg_type, ...); struct sbus_incoming_signal { struct sbus_connection *conn; DBusMessage *message; int64_t client; const char *interface; const char *signal; const char *path; }; typedef void (*sbus_incoming_signal_fn)(struct sbus_incoming_signal *sbus_signal, void *handler_data); errno_t sbus_signal_listen(struct sbus_connection *conn, const char *iface, const char *signal, sbus_incoming_signal_fn handler_fn, void *handler_data); #endif /* _SSSD_DBUS_H_*/