/*
Authors:
Simo Sorce <ssorce@redhat.com>
Stephen Gallagher <sgallagh@redhat.com>
Copyright (C) 2009 Red Hat
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 <time.h>
#include "responder/common/responder_packet.h"
#include "responder/common/responder.h"
#include "providers/data_provider.h"
#include "providers/data_provider/dp_responder_iface.h"
#include "sbus/sbus_client.h"
struct sss_dp_req;
struct sss_dp_callback {
};
struct sss_dp_req {
char *err_msg;
};
{
return EOK;
}
{
int hret;
/* Cancel Dbus pending reply if still pending */
if (sdp_req->pending_reply) {
}
/* Do not call callbacks if the responder is shutting down, because
* the top level responder context (pam_ctx, sudo_ctx, ...) may be
* already semi-freed and we may end up accessing freed memory.
*/
return 0;
}
/* If there are callbacks that haven't been invoked, return
* an error now.
*/
/* tevent_req_done/error will free cb */
/* Freeing the cb removes it from the cb_list.
* Therefore, the cb_list should now be pointing
* at a new callback. If it's not, it means the
* callback handler didn't free cb and may leak
* memory. Be paranoid and protect against this
* situation.
*/
"BUG: a callback did not free its request. "
"May leak memory\n");
/* Skip to the next since a memory leak is non-fatal */
}
}
/* Destroy the hash entry */
if (hret != HASH_SUCCESS) {
/* This should never happen */
"BUG: Could not clear [%d:%lu:%s] from request queue: [%s]\n",
return -1;
}
return 0;
}
struct tevent_timer *te,
{
/* ptr is a pointer to sidereq */
/* Just free it to kill all waiting requests when the timeout fires */
}
{
int ret;
unsigned long count, i;
if (!rctx->dp_request_table) {
return;
}
if (ret != HASH_SUCCESS) {
"not all request might be handled after reconnect.\n");
return;
}
"Will handle %lu requests after reconnect\n", count);
for (i=0; i<count; i++) {
}
}
char **err_msg)
{
int type;
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.
*/
"Severe error. A reply callback was called but no reply "
"was received and no timeout occurred\n");
/* FIXME: Destroy this connection? */
goto done;
}
switch (type) {
if (!ret) {
/* FIXME: Destroy this connection? */
goto done;
}
"Got reply from Data Provider - "
"DP error code: %u errno: %u error message: %s\n",
break;
case DBUS_MESSAGE_TYPE_ERROR:
DBUS_ERROR_NO_REPLY) == 0) {
goto done;
}
SBUS_ERROR_DP_NOTSUP) == 0) {
"Data Provider does not support this operation.\n");
} else {
"The Data Provider returned an error [%s]\n",
}
/* Falling through to default intentionally*/
default:
/*
* Timeout or other error occurred or something
* unexpected happened.
* It doesn't matter which, because either way we
* know that this connection isn't trustworthy.
* We'll destroy it now.
*/
/* FIXME: Destroy this connection? */
}
done:
return err;
}
static struct tevent_req *
struct sss_domain_info *dom,
DBusMessage *msg);
static void
struct tevent_req *cb_req)
{
}
struct tevent_req *nreq)
{
int hret;
if (!tmp_ctx) {
return ENOMEM;
}
if (!key) {
goto fail;
}
goto fail;
}
return EOK;
}
/* Check the hash for existing references to this request */
switch (hret) {
case HASH_SUCCESS:
/* Request already in progress */
break;
case HASH_ERROR_KEY_NOT_FOUND:
/* No such request in progress
* Create a new request
*/
if (!msg) {
goto fail;
}
if (!sidereq) {
goto fail;
}
/* add timeout handling so we do not hang forever should something
* go wrong in the provider. Use 2 sec less than the idle timeout to
* give it a chance to reply to the client before closing the
* connection. */
if (!te) {
/* Nothing much we can do */
goto fail;
}
/* We should now be able to find the sdp_req in the hash table */
if (hret != HASH_SUCCESS) {
/* Something must have gone wrong with creating the request */
goto fail;
}
break;
default:
"Could not query request list (%s)\n",
goto fail;
}
/* Register this request for results */
if (!sdp_req) {
goto fail;
}
if (!cb) {
goto fail;
}
/* Add it to the list of requests to call */
struct sss_dp_callback *);
fail:
return ret;
}
static void
{
/* Nothing to do here. The callbacks have already been invoked */
}
struct tevent_req *sidereq,
char **err_msg)
{
if (TRROEstate == TEVENT_REQ_USER_ERROR) {
if (TRROEerr == 0) {
return ERR_INTERNAL;
}
*dp_err = DP_ERR_FATAL;
} else {
return EIO;
}
}
return EOK;
}
/* Send a request to the data provider
* Once this function is called, the communication
* with the data provider will always run to
* completion. Freeing the returned tevent_req will
* cancel the notification of completion, but not
* the data provider action.
*/
enum sss_dp_acct_type type_in,
const char *opt_name_in,
enum sss_dp_acct_type *_type_out,
const char **_opt_name_out);
struct sss_dp_account_info {
bool fast_reply;
const char *opt_name;
const char *extra;
};
struct tevent_req *
struct sss_domain_info *dom,
bool fast_reply,
enum sss_dp_acct_type type,
const char *opt_name,
const char *extra)
{
char *key;
if (!req) {
return NULL;
}
/* either, or, not both */
goto error;
}
if (!dom) {
goto error;
}
/* This is a special case. If the files provider is just being updated,
* we issue an enumeration request. We always use the same request type
* (user enumeration) to make sure concurrent requests are just chained
* in the Data Provider
*/
goto error;
"Failed to set files provider update: %d: %s\n",
goto error;
}
/* EAGAIN, fall through to issuing the request */
} else {
goto error;
}
}
goto error;
}
if (opt_name) {
if (extra) {
} else {
}
} else if (opt_id) {
if (extra) {
} else {
}
} else {
}
if (!key) {
goto error;
}
"Could not issue DP request [%d]: %s\n",
goto error;
}
return req;
} else {
}
return req;
}
enum sss_dp_acct_type type_in,
const char *opt_name_in,
enum sss_dp_acct_type *_type_out,
const char **_opt_name_out)
{
"The entries in the files domain are up-to-date\n");
return EOK;
}
"Domain files is not consistent, issuing update\n");
switch(type_in) {
case SSS_DP_USER:
case SSS_DP_GROUP:
*_opt_name_out = NULL;
return EAGAIN;
case SSS_DP_INITGROUPS:
/* There is no initgroups enumeration so let's use a dummy
* name to let the DP chain the requests
*/
return EAGAIN;
/* These are not handled by the files provider, just fall back */
case SSS_DP_NETGR:
case SSS_DP_SERVICES:
case SSS_DP_SECID:
case SSS_DP_USER_AND_GROUP:
case SSS_DP_CERT:
case SSS_DP_WILDCARD_USER:
case SSS_DP_WILDCARD_GROUP:
return EOK;
}
return EINVAL;
}
static DBusMessage *
{
char *filter;
case SSS_DP_USER:
case SSS_DP_WILDCARD_USER:
break;
case SSS_DP_GROUP:
case SSS_DP_WILDCARD_GROUP:
break;
case SSS_DP_INITGROUPS:
break;
case SSS_DP_NETGR:
break;
case SSS_DP_SERVICES:
break;
case SSS_DP_SECID:
break;
case SSS_DP_USER_AND_GROUP:
break;
case SSS_DP_CERT:
break;
}
} else {
}
} else {
}
if (!filter) {
return NULL;
}
return NULL;
}
/* create the message */
"Creating request for [%s][%#x][%s][%s:%s]\n",
/* D-Bus can't deal with NULL. */
}
if (!dbret) {
return NULL;
}
return msg;
}
struct tevent_req *req,
char **err_msg)
{
}
struct dp_internal_get_state {
};
static struct tevent_req *
struct sss_domain_info *dom,
{
int hret;
/* Internal requests need to be allocated on the responder context
* so that they don't go away if a client disconnects. The worst-
* case scenario here is that the cache is updated without any
* client expecting a response.
*/
&state,
struct dp_internal_get_state);
goto error;
}
/* Copy the key to use when calling the destructor
* It needs to be a copy because the original request
* might be freed if it no longer cares about the reply.
*/
/* double check dp_ctx has actually been initialized.
* in some pathological cases it may happen that nss starts up before
* dp connection code is actually able to establish a connection.
*/
"BUG: The Data Provider connection for %s is not available!\n",
goto error;
}
req,
/*
* Critical Failure
* We can't communicate on this connection
*/
"D-BUS send failed.\n");
goto error;
}
/* Add this sdp_req to the hash table */
if (hret != HASH_SUCCESS) {
"Could not store request query (%s)\n",
goto error;
}
return req;
return req;
}
{
int ret;
/* prevent trying to cancel a reply that we already received */
}
else {
"Failed to get reply from Data Provider");
}
}
/* Check whether we need to issue any callbacks */
/* Don't bother checking for NULL. If it fails due to ENOMEM,
* we can't really handle it anyway.
*/
/* tevent_req_done/error will free cb */
} else {
}
/* Freeing the cb removes it from the cb_list.
* Therefore, the cb_list should now be pointing
* at a new callback. If it's not, it means the
* callback handler didn't free cb and may leak
* memory. Be paranoid and protect against this
* situation.
*/
"BUG: a callback did not free its request. "
"May leak memory\n");
/* Skip to the next since a memory leak is non-fatal */
}
}
/* We're done with this request. Free the sdp_req
* This will clean up the hash table entry as well
*/
/* Free the sidereq to free the rest of the memory allocated with the
* internal dp request. */
} else {
}
}