summaryrefslogtreecommitdiff
path: root/src/responder
diff options
context:
space:
mode:
Diffstat (limited to 'src/responder')
-rw-r--r--src/responder/sudo/sudosrv_get_sudorules.c446
1 files changed, 444 insertions, 2 deletions
diff --git a/src/responder/sudo/sudosrv_get_sudorules.c b/src/responder/sudo/sudosrv_get_sudorules.c
index aa466c8b..5d54f95a 100644
--- a/src/responder/sudo/sudosrv_get_sudorules.c
+++ b/src/responder/sudo/sudosrv_get_sudorules.c
@@ -27,11 +27,453 @@
#include "db/sysdb_sudo.h"
#include "responder/sudo/sudosrv_private.h"
+static errno_t sudosrv_get_user(struct sudo_dom_ctx *dctx);
+static errno_t sudosrv_get_rules(struct sudo_dom_ctx *dctx);
+
errno_t sudosrv_get_sudorules(struct sudo_dom_ctx *dctx)
{
- dctx->res = NULL;
- dctx->res_count = 0;
+ errno_t ret;
+
+ dctx->check_provider = true;
+ ret = sudosrv_get_user(dctx);
+ if (ret == EAGAIN) {
+ DEBUG(SSSDBG_TRACE_INTERNAL,
+ ("Looking up the user info from Data Provider\n"));
+ return EAGAIN;
+ } else if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE,
+ ("Error looking up user information [%d]: %s\n", ret, strerror(ret)));
+ return ret;
+ }
+
+ /* OK, got the user from cache. Try to get the rules. */
+ ret = sudosrv_get_rules(dctx);
+ if (ret == EAGAIN) {
+ DEBUG(SSSDBG_TRACE_INTERNAL,
+ ("Looking up the sudo rules from Data Provider\n"));
+ return EAGAIN;
+ } else if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE,
+ ("Error looking up sudo rules [%d]: %s\n", ret, strerror(ret)));
+ return ret;
+ }
+
+ return EOK;
+}
+
+static void sudosrv_dp_send_acct_req_done(struct tevent_req *req);
+static void sudosrv_check_user_dp_callback(uint16_t err_maj, uint32_t err_min,
+ const char *err_msg, void *ptr);
+
+static errno_t sudosrv_get_user(struct sudo_dom_ctx *dctx)
+{
+ struct sss_domain_info *dom = dctx->domain;
+ struct sudo_cmd_ctx *cmd_ctx = dctx->cmd_ctx;
+ struct cli_ctx *cli_ctx = dctx->cmd_ctx->cli_ctx;
+ struct sysdb_ctx *sysdb;
+ time_t cache_expire = 0;
+ struct tevent_req *dpreq;
+ struct dp_callback_ctx *cb_ctx;
+ errno_t ret;
+
+ while (dom) {
+ /* if it is a domainless search, skip domains that require fully
+ * qualified names instead */
+ while (dom && cmd_ctx->check_next && dom->fqnames) {
+ dom = dom->next;
+ }
+
+ if (!dom) break;
+
+ /* make sure to update the dctx if we changed domain */
+ dctx->domain = dom;
+
+ DEBUG(SSSDBG_FUNC_DATA, ("Requesting info about [%s@%s]\n",
+ cmd_ctx->username, dom->name));
+
+ ret = sysdb_get_ctx_from_list(cli_ctx->rctx->db_list,
+ dctx->domain, &sysdb);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("sysdb context not found for this domain!\n"));
+ return EIO;
+ }
+
+ ret = sysdb_getpwnam(dctx, sysdb, cmd_ctx->username, &dctx->user);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE,
+ ("Failed to make request to our cache!\n"));
+ return EIO;
+ }
+
+ if (dctx->user->count > 1) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("getpwnam call returned more than one result !?!\n"));
+ return EIO;
+ }
+
+ if (dctx->user->count == 0 && !dctx->check_provider) {
+ /* if a multidomain search, try with next */
+ if (cmd_ctx->check_next) {
+ dctx->check_provider = true;
+ dom = dom->next;
+ if (dom) continue;
+ }
+
+ DEBUG(SSSDBG_MINOR_FAILURE, ("No results for getpwnam call\n"));
+ return ENOENT;
+ }
+
+ /* One result found, check cache expiry */
+ if (dctx->user->count == 1) {
+ cache_expire = ldb_msg_find_attr_as_uint64(dctx->user->msgs[0],
+ SYSDB_CACHE_EXPIRE, 0);
+ }
+
+ /* If cache miss and we haven't checked DP yet OR the entry is
+ * outdated, go to DP */
+ if ((dctx->user->count == 0 && dctx->check_provider) ||
+ cache_expire < time(NULL)) {
+ dpreq = sss_dp_get_account_send(cli_ctx, cli_ctx->rctx,
+ dom, false, SSS_DP_INITGROUPS,
+ cmd_ctx->username, 0);
+ if (!dpreq) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("Out of memory sending data provider request\n"));
+ return ENOMEM;
+ }
+
+ cb_ctx = talloc_zero(cli_ctx, struct dp_callback_ctx);
+ if(!cb_ctx) {
+ talloc_zfree(dpreq);
+ return ENOMEM;
+ }
+
+ cb_ctx->callback = sudosrv_check_user_dp_callback;
+ cb_ctx->ptr = dctx;
+ cb_ctx->cctx = cli_ctx;
+ cb_ctx->mem_ctx = cli_ctx;
+
+ tevent_req_set_callback(dpreq, sudosrv_dp_send_acct_req_done, cb_ctx);
+
+ /* tell caller we are in an async call */
+ return EAGAIN;
+ }
+
+ DEBUG(SSSDBG_TRACE_FUNC, ("Returning info for user [%s@%s]\n",
+ cmd_ctx->username, dctx->domain->name));
+ return EOK;
+ }
+
+ return ENOENT;
+}
+
+static void sudosrv_dp_send_acct_req_done(struct tevent_req *req)
+{
+ struct dp_callback_ctx *cb_ctx =
+ tevent_req_callback_data(req, struct dp_callback_ctx);
+
+ errno_t ret;
+ dbus_uint16_t err_maj;
+ dbus_uint32_t err_min;
+ char *err_msg;
+
+ ret = sss_dp_get_account_recv(cb_ctx->mem_ctx, req,
+ &err_maj, &err_min,
+ &err_msg);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("Fatal error, killing connection!\n"));
+ talloc_free(cb_ctx->cctx);
+ return;
+ }
+
+ cb_ctx->callback(err_maj, err_min, err_msg, cb_ctx->ptr);
+}
+
+static void sudosrv_check_user_dp_callback(uint16_t err_maj, uint32_t err_min,
+ const char *err_msg, void *ptr)
+{
+ errno_t ret;
+ struct sudo_dom_ctx *dctx = talloc_get_type(ptr, struct sudo_dom_ctx);
+
+ if (err_maj) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("Unable to get information from Data Provider\n"
+ "Error: %u, %u, %s\n",
+ (unsigned int)err_maj, (unsigned int)err_min, err_msg));
+ }
+
+ DEBUG(SSSDBG_TRACE_INTERNAL, ("Data Provider returned, check the cache again\n"));
+ dctx->check_provider = false;
+ ret = sudosrv_get_user(dctx);
+ /* FIXME - set entry into cache so that we don't perform initgroups too often */
+ if (ret == EAGAIN) {
+ goto done;
+ } else if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE,
+ ("Could not look up the user [%d]: %s\n",
+ ret, strerror(ret)));
+ sudosrv_cmd_done(dctx, EIO);
+ return;
+ }
+
+ DEBUG(SSSDBG_TRACE_INTERNAL, ("Looking up sudo rules..\n"));
+ ret = sudosrv_get_rules(dctx);
+ if (ret == EAGAIN) {
+ goto done;
+ } else if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE,
+ ("Error getting sudo rules [%d]: %s\n",
+ ret, strerror(ret)));
+ sudosrv_cmd_done(dctx, EIO);
+ return;
+ }
+
+done:
+ sudosrv_cmd_done(dctx, ret);
+}
+
+static errno_t sudosrv_get_sudorules_from_cache(struct sudo_dom_ctx *dctx);
+static void sudosrv_get_sudorules_dp_callback(struct tevent_req *req);
+
+static errno_t sudosrv_get_rules(struct sudo_dom_ctx *dctx)
+{
+ struct tevent_req *dpreq;
+ struct sudo_cmd_ctx *cmd_ctx = dctx->cmd_ctx;
+
+ /* FIXME - cache logic will be here. For now, just refresh
+ * the cache unconditionally */
+ dpreq = sudosrv_dp_refresh_send(cmd_ctx->cli_ctx->rctx,
+ dctx->domain, cmd_ctx->username);
+ if (dpreq == NULL) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("Fatal: Sysdb CTX not found for this domain!\n"));
+ return EIO;
+ }
+ tevent_req_set_callback(dpreq, sudosrv_get_sudorules_dp_callback, dctx);
+ return EAGAIN;
+}
+
+static void sudosrv_get_sudorules_dp_callback(struct tevent_req *req)
+{
+ struct sudo_dom_ctx *dctx;
+ errno_t ret;
+ dbus_uint16_t err_maj;
+ dbus_uint32_t err_min;
+
+ dctx = tevent_req_callback_data(req, struct sudo_dom_ctx);
+
+ ret = sudosrv_dp_refresh_recv(req, &err_maj, &err_min);
+ talloc_zfree(req);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE,
+ ("Data provider returned an error [%d]: %s "
+ "DBus error min: %d maj %d\n",
+ ret, strerror(ret), err_maj, err_min));
+ sudosrv_cmd_done(dctx, EIO);
+ return;
+ }
+
+ DEBUG(SSSDBG_TRACE_INTERNAL, ("About to get sudo rules from cache\n"));
+ ret = sudosrv_get_sudorules_from_cache(dctx);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE,
+ ("Failed to make a request to our cache [%d]: %s\n",
+ ret, strerror(ret)));
+ sudosrv_cmd_done(dctx, EIO);
+ return;
+ }
+
+ sudosrv_cmd_done(dctx, ret);
+}
+
+static errno_t sudosrv_get_sudorules_query_cache(TALLOC_CTX *mem_ctx,
+ struct sysdb_ctx *sysdb,
+ const char *username,
+ uid_t uid,
+ char **groupnames,
+ struct sysdb_attrs ***_rules,
+ size_t *_count);
+
+static errno_t sudosrv_get_sudorules_from_cache(struct sudo_dom_ctx *dctx)
+{
+ TALLOC_CTX *tmp_ctx;
+ errno_t ret;
+ struct sysdb_ctx *sysdb;
+ struct cli_ctx *cli_ctx = dctx->cmd_ctx->cli_ctx;
+ uid_t uid;
+ char **groupnames;
+
+ tmp_ctx = talloc_new(NULL);
+ if (tmp_ctx == NULL) return ENOMEM;
+
+ ret = sysdb_get_ctx_from_list(cli_ctx->rctx->db_list,
+ dctx->domain, &sysdb);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("sysdb context not found for this domain!\n"));
+ ret = EIO;
+ goto done;
+ }
+
+ ret = sysdb_get_sudo_user_info(tmp_ctx, dctx->cmd_ctx->username,
+ sysdb, &uid, &groupnames);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("Unable to retrieve user info [%d]: %s\n", strerror(ret)));
+ goto done;
+ }
+
+ ret = sudosrv_get_sudorules_query_cache(dctx, sysdb,
+ dctx->cmd_ctx->username,
+ uid, groupnames,
+ &dctx->res, &dctx->res_count);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("Unable to retrieve sudo rules [%d]: %s\n", strerror(ret)));
+ goto done;
+ }
+
+ DEBUG(SSSDBG_TRACE_FUNC, ("Returning rules for [%s@%s]\n",
+ dctx->cmd_ctx->username, dctx->domain->name));
+
+ ret = EOK;
+done:
+ talloc_free(tmp_ctx);
+ return ret;
+}
+
+static errno_t
+sort_sudo_rules(struct sysdb_attrs **rules, size_t count);
+
+static errno_t sudosrv_get_sudorules_query_cache(TALLOC_CTX *mem_ctx,
+ struct sysdb_ctx *sysdb,
+ const char *username,
+ uid_t uid,
+ char **groupnames,
+ struct sysdb_attrs ***_rules,
+ size_t *_count)
+{
+ TALLOC_CTX *tmp_ctx;
+ char *filter;
+ errno_t ret;
+ size_t count;
+ struct sysdb_attrs **rules;
+ struct ldb_message **msgs;
+ const char *attrs[] = { SYSDB_OBJECTCLASS
+ SYSDB_SUDO_CACHE_AT_OC,
+ SYSDB_SUDO_CACHE_AT_CN,
+ SYSDB_SUDO_CACHE_AT_USER,
+ SYSDB_SUDO_CACHE_AT_HOST,
+ SYSDB_SUDO_CACHE_AT_COMMAND,
+ SYSDB_SUDO_CACHE_AT_OPTION,
+ SYSDB_SUDO_CACHE_AT_RUNASUSER,
+ SYSDB_SUDO_CACHE_AT_RUNASGROUP,
+ SYSDB_SUDO_CACHE_AT_NOTBEFORE,
+ SYSDB_SUDO_CACHE_AT_NOTAFTER,
+ SYSDB_SUDO_CACHE_AT_ORDER,
+ NULL };
+
+ tmp_ctx = talloc_new(NULL);
+ if (tmp_ctx == NULL) return ENOMEM;
+ ret = sysdb_get_sudo_filter(tmp_ctx, username, uid, groupnames,
+ (SYSDB_SUDO_FILTER_NGRS | SYSDB_SUDO_FILTER_INCLUDE_ALL |
+ SYSDB_SUDO_FILTER_INCLUDE_DFL), &filter);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("Could not construct the search filter [%d]: %s\n",
+ ret, strerror(ret)));
+ goto done;
+ }
+
+ DEBUG(SSSDBG_FUNC_DATA, ("Searching sysdb with [%s]\n", filter));
+
+ ret = sysdb_search_custom(tmp_ctx, sysdb, filter,
+ SUDORULE_SUBDIR, attrs,
+ &count, &msgs);
+ if (ret != EOK && ret != ENOENT) {
+ DEBUG(SSSDBG_CRIT_FAILURE, ("Error looking up SUDO rules"));
+ goto done;
+ } if (ret == ENOENT) {
+ *_rules = NULL;
+ *_count = 0;
+ ret = EOK;
+ goto done;
+ }
+
+ ret = sysdb_msg2attrs(tmp_ctx, count, msgs, &rules);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE,
+ ("Could not convert ldb message to sysdb_attrs\n"));
+ goto done;
+ }
+
+ ret = sort_sudo_rules(rules, count);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE,
+ ("Could not sort rules by sudoOrder\n"));
+ goto done;
+ }
+
+ *_rules = talloc_steal(mem_ctx, rules);
+ *_count = count;
+
+ ret = EOK;
+done:
+ talloc_free(tmp_ctx);
+ return ret;
+}
+
+static int
+sudo_order_cmp_fn(const void *a, const void *b)
+{
+ struct sysdb_attrs *r1, *r2;
+ uint32_t o1, o2;
+ int ret;
+
+ r1 = * (struct sysdb_attrs * const *) a;
+ r2 = * (struct sysdb_attrs * const *) b;
+ if (!r1 || !r2) {
+ DEBUG(SSSDBG_CRIT_FAILURE, ("BUG: Wrong data?\n"));
+ return 0;
+ }
+
+ ret = sysdb_attrs_get_uint32_t(r1, SYSDB_SUDO_CACHE_AT_ORDER, &o1);
+ if (ret == ENOENT) {
+ /* man sudoers-ldap: If the sudoOrder attribute is not present,
+ * a value of 0 is assumed */
+ o1 = 0;
+ } else if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE, ("Cannot get sudoOrder value\n"));
+ return 0;
+ }
+
+ ret = sysdb_attrs_get_uint32_t(r2, SYSDB_SUDO_CACHE_AT_ORDER, &o2);
+ if (ret == ENOENT) {
+ /* man sudoers-ldap: If the sudoOrder attribute is not present,
+ * a value of 0 is assumed */
+ o2 = 0;
+ } else if (ret != EOK) {
+ DEBUG(SSSDBG_OP_FAILURE, ("Cannot get sudoOrder value\n"));
+ return 0;
+ }
+
+ if (o1 > o2) {
+ return 1;
+ } else if (o1 < o2) {
+ return -1;
+ }
+
+ return 0;
+}
+
+static errno_t
+sort_sudo_rules(struct sysdb_attrs **rules, size_t count)
+{
+ qsort(rules, count, sizeof(struct sysdb_attrs *),
+ sudo_order_cmp_fn);
return EOK;
}