Blame SOURCES/0048-TOOLS-Add-a-new-sssctl-command-access-report.patch

ced1f5
From 73a04a5c53c0e7701aa7753fd459ffbea52e28b8 Mon Sep 17 00:00:00 2001
ced1f5
From: Jakub Hrozek <jhrozek@redhat.com>
ced1f5
Date: Mon, 23 Oct 2017 18:08:12 +0200
ced1f5
Subject: [PATCH 48/57] TOOLS: Add a new sssctl command access-report
ced1f5
MIME-Version: 1.0
ced1f5
Content-Type: text/plain; charset=UTF-8
ced1f5
Content-Transfer-Encoding: 8bit
ced1f5
ced1f5
Resolves:
ced1f5
https://pagure.io/SSSD/sssd/issue/2840
ced1f5
ced1f5
Reviewed-by: Pavel Březina <pbrezina@redhat.com>
ced1f5
Reviewed-by: Fabiano Fidêncio <fidencio@redhat.com>
ced1f5
(cherry picked from commit 3ee8659bc6a77a78bc6c61b9650a36bd18ea95c8)
ced1f5
---
ced1f5
 Makefile.am                             |   1 +
ced1f5
 src/tools/sssctl/sssctl.c               |   1 +
ced1f5
 src/tools/sssctl/sssctl.h               |   5 +
ced1f5
 src/tools/sssctl/sssctl_access_report.c | 435 ++++++++++++++++++++++++++++++++
ced1f5
 4 files changed, 442 insertions(+)
ced1f5
 create mode 100644 src/tools/sssctl/sssctl_access_report.c
ced1f5
ced1f5
diff --git a/Makefile.am b/Makefile.am
ced1f5
index 16bcb4efc028b05c1196249245f4f3091b9366af..5917bd904054055a259eb69217282e4fb914c700 100644
ced1f5
--- a/Makefile.am
ced1f5
+++ b/Makefile.am
ced1f5
@@ -1754,6 +1754,7 @@ sssctl_SOURCES = \
ced1f5
     src/tools/sssctl/sssctl_sifp.c \
ced1f5
     src/tools/sssctl/sssctl_config.c \
ced1f5
     src/tools/sssctl/sssctl_user_checks.c \
ced1f5
+    src/tools/sssctl/sssctl_access_report.c \
ced1f5
     $(SSSD_TOOLS_OBJ) \
ced1f5
     $(NULL)
ced1f5
 sssctl_LDADD = \
ced1f5
diff --git a/src/tools/sssctl/sssctl.c b/src/tools/sssctl/sssctl.c
ced1f5
index 1e061c00d2238bf34adff4183e560dc127dd62c7..eee2d613966a5dda81627d2e225bfdc9bade4041 100644
ced1f5
--- a/src/tools/sssctl/sssctl.c
ced1f5
+++ b/src/tools/sssctl/sssctl.c
ced1f5
@@ -264,6 +264,7 @@ int main(int argc, const char **argv)
ced1f5
         SSS_TOOL_COMMAND("domain-list", "List available domains", 0, sssctl_domain_list),
ced1f5
         SSS_TOOL_COMMAND("domain-status", "Print information about domain", 0, sssctl_domain_status),
ced1f5
         SSS_TOOL_COMMAND("user-checks", "Print information about a user and check authentication", 0, sssctl_user_checks),
ced1f5
+        SSS_TOOL_COMMAND("access-report", "Generate access report for a domain", 0, sssctl_access_report),
ced1f5
         SSS_TOOL_DELIMITER("Information about cached content:"),
ced1f5
         SSS_TOOL_COMMAND("user-show", "Information about cached user", 0, sssctl_user_show),
ced1f5
         SSS_TOOL_COMMAND("group-show", "Information about cached group", 0, sssctl_group_show),
ced1f5
diff --git a/src/tools/sssctl/sssctl.h b/src/tools/sssctl/sssctl.h
ced1f5
index 22ca5d41e2c084e64b58bc5aa066414b002e7e8b..70fc19eff07317c264978a1ecb9159ae3acdfced 100644
ced1f5
--- a/src/tools/sssctl/sssctl.h
ced1f5
+++ b/src/tools/sssctl/sssctl.h
ced1f5
@@ -133,4 +133,9 @@ errno_t sssctl_config_check(struct sss_cmdline *cmdline,
ced1f5
 errno_t sssctl_user_checks(struct sss_cmdline *cmdline,
ced1f5
                            struct sss_tool_ctx *tool_ctx,
ced1f5
                            void *pvt);
ced1f5
+
ced1f5
+errno_t sssctl_access_report(struct sss_cmdline *cmdline,
ced1f5
+                             struct sss_tool_ctx *tool_ctx,
ced1f5
+                             void *pvt);
ced1f5
+
ced1f5
 #endif /* _SSSCTL_H_ */
ced1f5
diff --git a/src/tools/sssctl/sssctl_access_report.c b/src/tools/sssctl/sssctl_access_report.c
ced1f5
new file mode 100644
ced1f5
index 0000000000000000000000000000000000000000..11172329817b4dedaca480ab8a4537149853c330
ced1f5
--- /dev/null
ced1f5
+++ b/src/tools/sssctl/sssctl_access_report.c
ced1f5
@@ -0,0 +1,435 @@
ced1f5
+/*
ced1f5
+    Copyright (C) 2017 Red Hat
ced1f5
+
ced1f5
+    This program is free software; you can redistribute it and/or modify
ced1f5
+    it under the terms of the GNU Lesser General Public License as published by
ced1f5
+    the Free Software Foundation; either version 3 of the License, or
ced1f5
+    (at your option) any later version.
ced1f5
+
ced1f5
+    This program is distributed in the hope that it will be useful,
ced1f5
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
ced1f5
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
ced1f5
+    GNU Lesser General Public License for more details.
ced1f5
+
ced1f5
+    You should have received a copy of the GNU Lesser General Public License
ced1f5
+    along with this program.  If not, see <http://www.gnu.org/licenses/>.
ced1f5
+*/
ced1f5
+
ced1f5
+#include <security/pam_appl.h>
ced1f5
+
ced1f5
+#include "util/util.h"
ced1f5
+#include "tools/common/sss_tools.h"
ced1f5
+#include "tools/sssctl/sssctl.h"
ced1f5
+
ced1f5
+/*
ced1f5
+ * We're searching the cache directly..
ced1f5
+ */
ced1f5
+#include "providers/ipa/ipa_hbac_private.h"
ced1f5
+#include "providers/ipa/ipa_rules_common.h"
ced1f5
+
ced1f5
+#ifdef HAVE_SECURITY_PAM_MISC_H
ced1f5
+# include <security/pam_misc.h>
ced1f5
+#elif defined(HAVE_SECURITY_OPENPAM_H)
ced1f5
+# include <security/openpam.h>
ced1f5
+#endif
ced1f5
+
ced1f5
+#ifdef HAVE_SECURITY_PAM_MISC_H
ced1f5
+static struct pam_conv conv = {
ced1f5
+    misc_conv,
ced1f5
+    NULL
ced1f5
+};
ced1f5
+#elif defined(HAVE_SECURITY_OPENPAM_H)
ced1f5
+static struct pam_conv conv = {
ced1f5
+    openpam_ttyconv,
ced1f5
+    NULL
ced1f5
+};
ced1f5
+#else
ced1f5
+# error "Missing text based pam conversation function"
ced1f5
+#endif
ced1f5
+
ced1f5
+#ifndef DEFAULT_SERVICE
ced1f5
+#define DEFAULT_SERVICE "system-auth"
ced1f5
+#endif /* DEFAULT_SERVICE */
ced1f5
+
ced1f5
+#ifndef DEFAULT_USER
ced1f5
+#define DEFAULT_USER "admin"
ced1f5
+#endif /* DEFAULT_USER */
ced1f5
+
ced1f5
+typedef errno_t (*sssctl_dom_access_reporter_fn)(struct sss_tool_ctx *tool_ctx,
ced1f5
+                                                 const char *user,
ced1f5
+                                                 const char *service,
ced1f5
+                                                 struct sss_domain_info *domain);
ced1f5
+
ced1f5
+static errno_t run_pam_acct(struct sss_tool_ctx *tool_ctx,
ced1f5
+                            const char *user,
ced1f5
+                            const char *service,
ced1f5
+                            struct sss_domain_info *domain)
ced1f5
+{
ced1f5
+    errno_t ret;
ced1f5
+    pam_handle_t *pamh;
ced1f5
+
ced1f5
+    ret = pam_start(service, user, &conv, &pamh);
ced1f5
+    if (ret != PAM_SUCCESS) {
ced1f5
+        ERROR("pam_start failed: %s\n", pam_strerror(pamh, ret));
ced1f5
+        return EIO;
ced1f5
+    }
ced1f5
+
ced1f5
+    ret = pam_acct_mgmt(pamh, 0);
ced1f5
+    pam_end(pamh, ret);
ced1f5
+    return ret;
ced1f5
+}
ced1f5
+
ced1f5
+static errno_t get_rdn_value(TALLOC_CTX *mem_ctx,
ced1f5
+                             struct sss_domain_info *dom,
ced1f5
+                             const char *dn_attr,
ced1f5
+                             const char **_rdn_value)
ced1f5
+{
ced1f5
+    errno_t ret;
ced1f5
+    TALLOC_CTX *tmp_ctx;
ced1f5
+    struct ldb_dn *dn = NULL;
ced1f5
+    const struct ldb_val *rdn_val;
ced1f5
+    const char *rdn_str;
ced1f5
+
ced1f5
+    tmp_ctx = talloc_new(NULL);
ced1f5
+    if (tmp_ctx == NULL) {
ced1f5
+        return ENOMEM;
ced1f5
+    }
ced1f5
+
ced1f5
+    dn = ldb_dn_new(tmp_ctx, sysdb_ctx_get_ldb(dom->sysdb), dn_attr);
ced1f5
+    if (dn == NULL) {
ced1f5
+        ret = ENOMEM;
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    rdn_val = ldb_dn_get_rdn_val(dn);
ced1f5
+    if (rdn_val == NULL) {
ced1f5
+        DEBUG(SSSDBG_OP_FAILURE, "No RDN value?\n");
ced1f5
+        ret = ENOMEM;
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    rdn_str = talloc_strndup(tmp_ctx,
ced1f5
+                               (const char *)rdn_val->data,
ced1f5
+                               rdn_val->length);
ced1f5
+    if (rdn_str == NULL) {
ced1f5
+        ret = ENOMEM;
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    ret = EOK;
ced1f5
+    *_rdn_value = talloc_steal(mem_ctx, rdn_str);
ced1f5
+done:
ced1f5
+    talloc_zfree(tmp_ctx);
ced1f5
+    return ret;
ced1f5
+}
ced1f5
+
ced1f5
+static errno_t is_member_group(struct sss_domain_info *dom,
ced1f5
+                               const char *dn_attr,
ced1f5
+                               const char *group_rdn,
ced1f5
+                               bool *_is_group)
ced1f5
+{
ced1f5
+    const char *comp_name;
ced1f5
+    const struct ldb_val *comp_val;
ced1f5
+    TALLOC_CTX *tmp_ctx;
ced1f5
+    bool is_group = false;
ced1f5
+    errno_t ret;
ced1f5
+    struct ldb_dn *dn = NULL;
ced1f5
+
ced1f5
+    tmp_ctx = talloc_new(NULL);
ced1f5
+    if (tmp_ctx == NULL) {
ced1f5
+        return ENOMEM;
ced1f5
+    }
ced1f5
+
ced1f5
+    dn = ldb_dn_new(tmp_ctx, sysdb_ctx_get_ldb(dom->sysdb), dn_attr);
ced1f5
+    if (dn == NULL) {
ced1f5
+        ret = ENOMEM;
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    comp_name = ldb_dn_get_component_name(dn, 1);
ced1f5
+    comp_val = ldb_dn_get_component_val(dn, 1);
ced1f5
+    if (strcasecmp("cn", comp_name) == 0
ced1f5
+            && strncasecmp(group_rdn,
ced1f5
+                           (const char *) comp_val->data,
ced1f5
+                           comp_val->length) == 0) {
ced1f5
+        is_group = true;
ced1f5
+    }
ced1f5
+
ced1f5
+    ret = EOK;
ced1f5
+done:
ced1f5
+    *_is_group = is_group;
ced1f5
+    talloc_zfree(tmp_ctx);
ced1f5
+    return ret;
ced1f5
+}
ced1f5
+
ced1f5
+static void print_category(struct sss_domain_info *domain,
ced1f5
+                           struct ldb_message *rule_msg,
ced1f5
+                           const char *category_attr_name,
ced1f5
+                           const char *category_label)
ced1f5
+{
ced1f5
+    struct ldb_message_element *category_attr;
ced1f5
+
ced1f5
+    category_attr = ldb_msg_find_element(rule_msg, category_attr_name);
ced1f5
+    if (category_attr == NULL) {
ced1f5
+        DEBUG(SSSDBG_MINOR_FAILURE, "Cannot find %s\n", category_attr_name);
ced1f5
+        return;
ced1f5
+    }
ced1f5
+
ced1f5
+    if (category_attr->num_values > 0) {
ced1f5
+        PRINT("\t%s: ", category_label);
ced1f5
+        for (unsigned i = 0; i < category_attr->num_values; i++) {
ced1f5
+            PRINT("%s%s",
ced1f5
+                  i > 0 ? ", " : "",
ced1f5
+                  (const char *) category_attr->values[i].data);
ced1f5
+        }
ced1f5
+        PRINT("\n");
ced1f5
+    }
ced1f5
+}
ced1f5
+
ced1f5
+static void print_member_attr(struct sss_domain_info *domain,
ced1f5
+                              struct ldb_message *rule_msg,
ced1f5
+                              const char *member_attr_name,
ced1f5
+                              const char *group_rdn,
ced1f5
+                              const char *object_label,
ced1f5
+                              const char *group_label)
ced1f5
+{
ced1f5
+    errno_t ret;
ced1f5
+    TALLOC_CTX *tmp_ctx = NULL;
ced1f5
+    const char **member_names = NULL;
ced1f5
+    size_t name_count = 0;
ced1f5
+    const char **member_group_names = NULL;
ced1f5
+    size_t group_count = 0;
ced1f5
+    struct ldb_message_element *member_attr = NULL;
ced1f5
+
ced1f5
+    tmp_ctx = talloc_new(NULL);
ced1f5
+    if (tmp_ctx == NULL) {
ced1f5
+        return;
ced1f5
+    }
ced1f5
+
ced1f5
+    member_attr = ldb_msg_find_element(rule_msg, member_attr_name);
ced1f5
+    if (member_attr == NULL) {
ced1f5
+        DEBUG(SSSDBG_MINOR_FAILURE, "Cannot find %s\n", member_attr_name);
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    member_names = talloc_zero_array(tmp_ctx,
ced1f5
+                                      const char *,
ced1f5
+                                      member_attr->num_values + 1);
ced1f5
+    member_group_names = talloc_zero_array(tmp_ctx,
ced1f5
+                                           const char *,
ced1f5
+                                           member_attr->num_values + 1);
ced1f5
+    if (member_names == NULL || member_group_names == NULL) {
ced1f5
+        DEBUG(SSSDBG_CRIT_FAILURE, "OOM?\n");
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    for (size_t i = 0; i < member_attr->num_values; i++) {
ced1f5
+        bool is_group;
ced1f5
+        const char *rdn_string;
ced1f5
+        const char *dn_attr;
ced1f5
+
ced1f5
+        dn_attr = (const char *) member_attr->values[i].data;
ced1f5
+
ced1f5
+        ret = is_member_group(domain, dn_attr, group_rdn, &is_group);
ced1f5
+        if (ret != EOK) {
ced1f5
+            continue;
ced1f5
+        }
ced1f5
+
ced1f5
+        ret = get_rdn_value(tmp_ctx, domain, dn_attr, &rdn_string);
ced1f5
+        if (ret != EOK) {
ced1f5
+            continue;
ced1f5
+        }
ced1f5
+
ced1f5
+        if (is_group == false) {
ced1f5
+            member_names[name_count] = talloc_steal(member_names,
ced1f5
+                                                    rdn_string);
ced1f5
+            if (member_names[name_count] == NULL) {
ced1f5
+                goto done;
ced1f5
+            }
ced1f5
+            name_count++;
ced1f5
+        } else {
ced1f5
+            member_group_names[group_count] = talloc_strdup(member_group_names,
ced1f5
+                                                            rdn_string);
ced1f5
+            if (member_group_names[group_count] == NULL) {
ced1f5
+                goto done;
ced1f5
+            }
ced1f5
+            group_count++;
ced1f5
+        }
ced1f5
+    }
ced1f5
+
ced1f5
+    if (member_names[0] != NULL) {
ced1f5
+        PRINT("\t%s: ", object_label);
ced1f5
+        for (int i = 0; member_names[i]; i++) {
ced1f5
+            PRINT("%s%s", i > 0 ? ", " : "", member_names[i]);
ced1f5
+        }
ced1f5
+        PRINT("\n");
ced1f5
+    }
ced1f5
+
ced1f5
+    if (member_group_names[0] != NULL) {
ced1f5
+        PRINT("\t%s: ", group_label);
ced1f5
+        for (int i = 0; member_group_names[i]; i++) {
ced1f5
+            PRINT("%s%s", i > 0 ? ", " : "", member_group_names[i]);
ced1f5
+        }
ced1f5
+        PRINT("\n");
ced1f5
+    }
ced1f5
+
ced1f5
+done:
ced1f5
+    talloc_free(tmp_ctx);
ced1f5
+}
ced1f5
+
ced1f5
+static void print_ipa_hbac_rule(struct sss_domain_info *domain,
ced1f5
+                                struct ldb_message *rule_msg)
ced1f5
+{
ced1f5
+    struct ldb_message_element *el;
ced1f5
+
ced1f5
+    el = ldb_msg_find_element(rule_msg, IPA_CN);
ced1f5
+    if (el == NULL || el->num_values < 1) {
ced1f5
+        DEBUG(SSSDBG_MINOR_FAILURE, "A rule with no name\n");
ced1f5
+        return;
ced1f5
+    }
ced1f5
+
ced1f5
+    PRINT("Rule name: %1$s\n", el->values[0].data);
ced1f5
+
ced1f5
+    print_member_attr(domain,
ced1f5
+                      rule_msg,
ced1f5
+                      IPA_MEMBER_USER,
ced1f5
+                      "groups",
ced1f5
+                      _("Member users"),
ced1f5
+                      _("Member groups"));
ced1f5
+    print_category(domain,
ced1f5
+                   rule_msg,
ced1f5
+                   IPA_USER_CATEGORY,
ced1f5
+                   _("User category"));
ced1f5
+
ced1f5
+    print_member_attr(domain,
ced1f5
+                      rule_msg,
ced1f5
+                      IPA_MEMBER_SERVICE,
ced1f5
+                      "hbacservicegroups",
ced1f5
+                      _("Member services"),
ced1f5
+                      _("Member service groups"));
ced1f5
+    print_category(domain,
ced1f5
+                   rule_msg,
ced1f5
+                   IPA_SERVICE_CATEGORY,
ced1f5
+                   _("Service category"));
ced1f5
+
ced1f5
+    PRINT("\n");
ced1f5
+}
ced1f5
+
ced1f5
+static errno_t sssctl_ipa_access_report(struct sss_tool_ctx *tool_ctx,
ced1f5
+                                        const char *user,
ced1f5
+                                        const char *service,
ced1f5
+                                        struct sss_domain_info *domain)
ced1f5
+{
ced1f5
+    TALLOC_CTX *tmp_ctx = NULL;
ced1f5
+    const char *filter = NULL;
ced1f5
+    errno_t ret;
ced1f5
+    const char *attrs[] = {
ced1f5
+        OBJECTCLASS,
ced1f5
+        IPA_CN,
ced1f5
+        IPA_MEMBER_USER,
ced1f5
+        IPA_USER_CATEGORY,
ced1f5
+        IPA_MEMBER_SERVICE,
ced1f5
+        IPA_SERVICE_CATEGORY,
ced1f5
+        IPA_MEMBER_HOST,
ced1f5
+        IPA_HOST_CATEGORY,
ced1f5
+        NULL,
ced1f5
+    };
ced1f5
+    size_t rule_count;
ced1f5
+    struct ldb_message **msgs = NULL;
ced1f5
+
ced1f5
+    /* Run the pam account phase to make sure the rules are fetched by SSSD */
ced1f5
+    ret = run_pam_acct(tool_ctx, user, service, domain);
ced1f5
+    if (ret != PAM_SUCCESS && ret != PAM_PERM_DENIED) {
ced1f5
+        ERROR("Cannot run the PAM account phase, reporting stale rules\n");
ced1f5
+        /* Non-fatal */
ced1f5
+    }
ced1f5
+
ced1f5
+    tmp_ctx = talloc_new(tool_ctx);
ced1f5
+    if (tmp_ctx == NULL) {
ced1f5
+        return ENOMEM;
ced1f5
+    }
ced1f5
+
ced1f5
+    filter = talloc_asprintf(tmp_ctx, "(objectClass=%s)", IPA_HBAC_RULE);
ced1f5
+    if (filter == NULL) {
ced1f5
+        ret = ENOMEM;
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    ret = sysdb_search_custom(tmp_ctx, domain, filter,
ced1f5
+                              HBAC_RULES_SUBDIR, attrs,
ced1f5
+                              &rule_count, &msgs);
ced1f5
+    if (ret != EOK && ret != ENOENT) {
ced1f5
+        DEBUG(SSSDBG_CRIT_FAILURE, "Error looking up HBAC rules\n");
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    if (ret == ENOENT) {
ced1f5
+        PRINT("No cached rules. All users will be denied access\n");
ced1f5
+        ret = EOK;
ced1f5
+        goto done;
ced1f5
+    }
ced1f5
+
ced1f5
+    PRINT("%1$zu rules cached\n\n", rule_count);
ced1f5
+
ced1f5
+    for (size_t i = 0; i < rule_count; i++) {
ced1f5
+        print_ipa_hbac_rule(domain, msgs[i]);
ced1f5
+    }
ced1f5
+
ced1f5
+    ret = EOK;
ced1f5
+done:
ced1f5
+    talloc_zfree(tmp_ctx);
ced1f5
+    return ret;
ced1f5
+}
ced1f5
+
ced1f5
+sssctl_dom_access_reporter_fn get_report_fn(const char *provider)
ced1f5
+{
ced1f5
+    if (strcmp(provider, "ipa") == 0) {
ced1f5
+        return sssctl_ipa_access_report;
ced1f5
+    }
ced1f5
+
ced1f5
+    return NULL;
ced1f5
+}
ced1f5
+
ced1f5
+errno_t sssctl_access_report(struct sss_cmdline *cmdline,
ced1f5
+                             struct sss_tool_ctx *tool_ctx,
ced1f5
+                             void *pvt)
ced1f5
+{
ced1f5
+    errno_t ret;
ced1f5
+    const char *domname = NULL;
ced1f5
+    sssctl_dom_access_reporter_fn reporter;
ced1f5
+    struct sss_domain_info *dom;
ced1f5
+    const char *user = DEFAULT_USER;
ced1f5
+    const char *service = DEFAULT_SERVICE;
ced1f5
+
ced1f5
+    /* Parse command line. */
ced1f5
+    struct poptOption options[] = {
ced1f5
+        { "user", 'u', POPT_ARG_STRING, &user, 0,
ced1f5
+          _("PAM user, default: " DEFAULT_USER), NULL },
ced1f5
+        { "service", 's', POPT_ARG_STRING, &service, 0,
ced1f5
+          _("PAM service, default: " DEFAULT_SERVICE), NULL },
ced1f5
+        POPT_TABLEEND
ced1f5
+    };
ced1f5
+
ced1f5
+    ret = sss_tool_popt_ex(cmdline, options, SSS_TOOL_OPT_OPTIONAL,
ced1f5
+                           NULL, NULL, "DOMAIN", _("Specify domain name."),
ced1f5
+                           &domname, NULL);
ced1f5
+    if (ret != EOK) {
ced1f5
+        DEBUG(SSSDBG_CRIT_FAILURE, "Unable to parse command arguments\n");
ced1f5
+        return ret;
ced1f5
+    }
ced1f5
+
ced1f5
+    dom = find_domain_by_name(tool_ctx->domains, domname, true);
ced1f5
+    if (dom == NULL) {
ced1f5
+        ERROR("Cannot find domain %1$s\n", domname);
ced1f5
+        return ERR_DOMAIN_NOT_FOUND;
ced1f5
+    }
ced1f5
+
ced1f5
+    reporter = get_report_fn(dom->provider);
ced1f5
+    if (reporter == NULL) {
ced1f5
+        ERROR("Access report not implemented for domains of type %1$s\n",
ced1f5
+              dom->provider);
ced1f5
+        return ret;
ced1f5
+    }
ced1f5
+
ced1f5
+    return reporter(tool_ctx, user, service, dom);
ced1f5
+}
ced1f5
-- 
ced1f5
2.14.3
ced1f5