diff options
Diffstat (limited to 'src/providers')
-rw-r--r-- | src/providers/ldap/sdap_access.c | 132 |
1 files changed, 73 insertions, 59 deletions
diff --git a/src/providers/ldap/sdap_access.c b/src/providers/ldap/sdap_access.c index 8a156bc3..7d7832ad 100644 --- a/src/providers/ldap/sdap_access.c +++ b/src/providers/ldap/sdap_access.c @@ -90,6 +90,7 @@ struct sdap_access_req_ctx { struct tevent_context *ev; struct sdap_access_ctx *access_ctx; struct sdap_id_ctx *sdap_ctx; + struct sdap_id_op *sdap_op; struct sysdb_handle *handle; struct be_ctx *be_ctx; const char **attrs; @@ -98,6 +99,8 @@ struct sdap_access_req_ctx { char *basedn; }; +static int sdap_access_decide_offline(struct tevent_req *req); +static int sdap_access_retry(struct tevent_req *req); static void sdap_access_connect_done(struct tevent_req *subreq); static void sdap_access_get_access_done(struct tevent_req *req); static struct tevent_req *sdap_access_send(TALLOC_CTX *mem_ctx, @@ -109,7 +112,6 @@ static struct tevent_req *sdap_access_send(TALLOC_CTX *mem_ctx, errno_t ret; struct sdap_access_req_ctx *state; struct tevent_req *req; - struct tevent_req *subreq; struct ldb_result *res; const char *basedn; @@ -180,17 +182,7 @@ static struct tevent_req *sdap_access_send(TALLOC_CTX *mem_ctx, /* Ok, we have one result, check if we are online or offline */ if (be_is_offline(state->be_ctx)) { /* Ok, we're offline. Return from the cache */ - if (state->cached_access) { - DEBUG(6, ("Access granted by cached credentials\n")); - ret = EOK; - state->pam_status = PAM_SUCCESS; - goto finished; - } - - /* Access denied */ - DEBUG(6, ("Access denied by cached credentials\n")); - ret = EOK; - state->pam_status = PAM_PERM_DENIED; + ret = sdap_access_decide_offline(req); goto finished; } @@ -226,37 +218,17 @@ static struct tevent_req *sdap_access_send(TALLOC_CTX *mem_ctx, DEBUG(6, ("Checking filter against LDAP\n")); - /* Check whether we have an active LDAP connection */ - if (state->sdap_ctx->gsh == NULL || ! state->sdap_ctx->gsh->connected) { - subreq = sdap_cli_connect_send(state, state->ev, - state->sdap_ctx->opts, - state->sdap_ctx->be, - state->sdap_ctx->service, - NULL); - if (!subreq) { - DEBUG(1, ("sdap_cli_connect_send failed.\n")); - goto failed; - } - - tevent_req_set_callback(subreq, sdap_access_connect_done, req); - return req; + state->sdap_op = sdap_id_op_create(state, state->sdap_ctx->conn_cache); + if (!state->sdap_op) { + DEBUG(2, ("sdap_id_op_create failed\n")); + goto failed; } - /* Make the LDAP request */ - subreq = sdap_get_generic_send(state, - state->ev, - state->sdap_ctx->opts, - state->sdap_ctx->gsh, - state->basedn, LDAP_SCOPE_BASE, - state->filter, NULL, - NULL, 0); - if (subreq == NULL) { - DEBUG(1, ("Could not start LDAP communication\n")); + ret = sdap_access_retry(req); + if (ret != EOK) { goto failed; } - tevent_req_set_callback(subreq, sdap_access_get_access_done, req); - return req; failed: @@ -269,33 +241,61 @@ finished: return req; } +static int sdap_access_decide_offline(struct tevent_req *req) +{ + struct sdap_access_req_ctx *state = + tevent_req_data(req, struct sdap_access_req_ctx); + + if (state->cached_access) { + DEBUG(6, ("Access granted by cached credentials\n")); + state->pam_status = PAM_SUCCESS; + } else { + DEBUG(6, ("Access denied by cached credentials\n")); + state->pam_status = PAM_PERM_DENIED; + } + + return EOK; +} + +static int sdap_access_retry(struct tevent_req *req) +{ + struct sdap_access_req_ctx *state = + tevent_req_data(req, struct sdap_access_req_ctx); + struct tevent_req *subreq; + int ret; + + subreq = sdap_id_op_connect_send(state->sdap_op, state, &ret); + if (!subreq) { + DEBUG(2, ("sdap_id_op_connect_send failed: %d (%s)\n", ret, strerror(ret))); + return ret; + } + + tevent_req_set_callback(subreq, sdap_access_connect_done, req); + return EOK; +} + static void sdap_access_connect_done(struct tevent_req *subreq) { - errno_t ret; struct tevent_req *req = tevent_req_callback_data(subreq, struct tevent_req); struct sdap_access_req_ctx *state = tevent_req_data(req, struct sdap_access_req_ctx); + int ret, dp_error; - ret = sdap_cli_connect_recv(subreq, state->sdap_ctx, - &state->sdap_ctx->gsh, NULL); + ret = sdap_id_op_connect_recv(subreq, &dp_error); talloc_zfree(subreq); + if (ret != EOK) { - /* Could not connect to LDAP. Mark as offline and return - * from cache. - */ - be_mark_offline(state->be_ctx); - if (state->cached_access) { - DEBUG(6, ("Access granted by cached credentials\n")); - state->pam_status = PAM_SUCCESS; - tevent_req_done(req); - return; + if (dp_error == DP_ERR_OFFLINE) { + ret = sdap_access_decide_offline(req); + if (ret == EOK) { + tevent_req_done(req); + return; + } } - /* Access denied */ - DEBUG(6, ("Access denied by cached credentials\n")); - state->pam_status = PAM_PERM_DENIED; - tevent_req_done(req); + tevent_req_error(req, ret); + return; } /* Connection to LDAP succeeded @@ -304,7 +304,7 @@ static void sdap_access_connect_done(struct tevent_req *subreq) subreq = sdap_get_generic_send(state, state->ev, state->sdap_ctx->opts, - state->sdap_ctx->gsh, + sdap_id_op_handle(state->sdap_op), state->basedn, LDAP_SCOPE_BASE, state->filter, NULL, @@ -321,7 +321,7 @@ static void sdap_access_connect_done(struct tevent_req *subreq) static void sdap_access_get_access_done(struct tevent_req *subreq) { - errno_t ret; + int ret, dp_error; size_t num_results; bool found = false; struct sysdb_attrs *attrs; @@ -333,10 +333,24 @@ static void sdap_access_get_access_done(struct tevent_req *subreq) ret = sdap_get_generic_recv(subreq, state, &num_results, &results); talloc_zfree(subreq); + + ret = sdap_id_op_done(state->sdap_op, ret, &dp_error); if (ret != EOK) { - DEBUG(1, ("sdap_get_generic_send() returned error [%d][%s]", - ret, strerror(ret))); - state->pam_status = PAM_SYSTEM_ERR; + if (dp_error == DP_ERR_OK) { + /* retry */ + ret = sdap_access_retry(req); + if (ret == EOK) { + return; + } + state->pam_status = PAM_SYSTEM_ERR; + } else if (dp_error == DP_ERR_OFFLINE) { + ret = sdap_access_decide_offline(req); + } else { + DEBUG(1, ("sdap_get_generic_send() returned error [%d][%s]\n", + ret, strerror(ret))); + state->pam_status = PAM_SYSTEM_ERR; + } + goto done; } |