nsssrv_services.c revision 188f9e1e646b0bed530913ca76bbcdf0f342cc66
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher/*
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SSSD
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher Authors:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher Stephen Gallagher <sgallagh@redhat.com>
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher Copyright (C) 2012 Red Hat
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher This program is free software; you can redistribute it and/or modify
056302a92862fda16351d7192600746746f38e5dStephen Gallagher it under the terms of the GNU General Public License as published by
b355dcb54194f498921743ca33304eac35d89718Stephen Gallagher the Free Software Foundation; either version 3 of the License, or
b355dcb54194f498921743ca33304eac35d89718Stephen Gallagher (at your option) any later version.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher This program is distributed in the hope that it will be useful,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher but WITHOUT ANY WARRANTY; without even the implied warranty of
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher GNU General Public License for more details.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher You should have received a copy of the GNU General Public License
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher along with this program. If not, see <http://www.gnu.org/licenses/>.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher*/
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include <collection.h>
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include <arpa/inet.h>
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include "util/util.h"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include "responder/nss/nsssrv.h"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include "responder/nss/nsssrv_private.h"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include "responder/nss/nsssrv_services.h"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include "responder/common/negcache.h"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include "confdb/confdb.h"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include "db/sysdb.h"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#include "db/sysdb_services.h"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstruct getserv_ctx {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint16_t port;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_context *ev;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_dom_ctx *dctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_domain_info **domains;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher size_t dom_idx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *name;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *cased_name;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *proto;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *cased_proto;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct ldb_result *res;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher};
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic errno_t lookup_service_step(struct tevent_req *req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void lookup_service_done(struct tevent_req *req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#define SVC_NAME_CASED (dom->case_sensitive ? state->name \
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher : state->cased_name)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher#define SVC_PROTO_CASED (dom->case_sensitive ? state->proto \
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher : state->cased_proto)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher/* Provider Lookup Logic:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Iterate through the available caches. If the cached entry is
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * present and not expired, return it immediately(*). If it is
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * present and expired, add it to a list of domains eligible to
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * be checked. If it is in the negative cache, skip over it and
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * do not add it to the eligible domain list.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher *
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Once we have searched all of the caches, if the entry has not
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * been determined to be available, search all domains in order
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * to see if any of them contain the requested entry.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher *
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * (*) Optionally perform a midpoint cache refresh if appropriate.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic struct tevent_req *
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghergetserv_send(TALLOC_CTX *mem_ctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_context *ev,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint16_t port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher const char *service_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher const char *service_protocol,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_dom_ctx *dctx)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_req *req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_req *subreq;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct getserv_ctx *state;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_cmd_ctx *cmdctx = dctx->cmdctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct cli_ctx *cctx = cmdctx->cctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_domain_info *dom;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher size_t num_domains = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher size_t dom_idx = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_get_type(cctx->rctx->pvt_ctx, struct nss_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sysdb_ctx *sysdb;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher time_t now = time(NULL);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint64_t lastUpdate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint64_t cacheExpire;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint64_t midpoint_refresh;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher req = tevent_req_create(mem_ctx, &state, struct getserv_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!req) return NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->dctx = dctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher for (dom = cctx->rctx->domains; dom; dom = dom->next) num_domains++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Create an array of domains to check. To save resizes, we'll
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * assume that all will be checked
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->domains = talloc_zero_array(state,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_domain_info *,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num_domains + 1);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->domains) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->port = port;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Store both the case-sensitive and lowercased names
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * in the state object, to avoid recalculating the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * lowercase in multiple domains.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (service_protocol) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher state->proto = talloc_strdup(state, service_protocol);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->proto) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = ENOMEM;
3a8abe04137d028b8ebd1cb33152aefa55893efbStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->cased_proto = sss_get_cased_name(state, service_protocol,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher false);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->cased_proto) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->proto = NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->cased_proto = NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If we're looking up by name */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (service_name) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Store both the case-sensitive and lowercased names
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * in the state object, to avoid recalculating the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * lowercase in multiple domains.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->name = talloc_strdup(state, service_name);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->name) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->cased_name = sss_get_cased_name(state, service_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher false);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->cased_name) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = cctx->rctx->domains;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher while(dom) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* if it is a domainless search, skip domains that require fully
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * qualified names instead */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher while (dom && cmdctx->check_next && dom->fqnames) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = dom->next;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!dom) break;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sysdb = dom->sysdb;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (sysdb == NULL) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Critical: Sysdb CTX not found for [%s]!\n", dom->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EINVAL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If we're looking up by name */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (service_name) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Check the negative cache */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_ncache_check_service(nctx->ncache,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher nctx->neg_timeout,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If negatively cached, return we didn't find it */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EEXIST) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Service [%s:%s] does not exist in [%s]! "
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher "(negative cache)\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED ? SVC_PROTO_CASED : "<ANY>",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If this is a multi-domain search, try the next one */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (cmdctx->check_next) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = dom->next;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* This was a single-domain search.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * exit the loop. Since it was negatively-
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * cached, don't add it to the eligible
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * domains list.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher continue;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Check the cache */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Checking cache for [%s:%s@%s]\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED ? SVC_PROTO_CASED : "<ANY>",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sysdb_getservbyname(state, sysdb,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &state->res);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else { /* Looking up by port */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Check the negative cache */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_ncache_check_service_port(nctx->ncache,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher nctx->neg_timeout,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom, port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If negatively cached, return we didn't find it */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EEXIST) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Service [%lu:%s] does not exist in [%s]! "
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher "(negative cache)\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED ? SVC_PROTO_CASED : "<ANY>",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If this is a multi-domain search, try the next one */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (cmdctx->check_next) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = dom->next;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* This was a single-domain search.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * exit the loop. Since it was negatively-
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * cached, don't add it to the eligible
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * domains list.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher continue;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Check the cache */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Checking cache for [%lu:%s@%s]\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED ? SVC_PROTO_CASED : "<ANY>",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom->name));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sysdb_getservbyport(state, sysdb, port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &state->res);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK && ret != ENOENT) goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == ENOENT) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Not found in the cache. Add this domain to the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * list of eligible domains to check the provider.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (NEED_CHECK_PROVIDER(dom->provider)) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->domains[dom_idx] = dom;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom_idx++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* No provider to check. Set the negative cache here */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (state->name) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_ncache_set_service_name(nctx->ncache, false,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Failure to set the negative cache is non-fatal.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * We'll log an error and continue.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_MINOR_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Could not set negative cache for [%s][%s]\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED, SVC_PROTO_CASED));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_ncache_set_service_port(nctx->ncache, false,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher state->port,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher SVC_PROTO_CASED);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Failure to set the negative cache is non-fatal.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * We'll log an error and continue.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_MINOR_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Could not set negative cache for [%lu][%s]\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->port, SVC_PROTO_CASED));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If this is a multi-domain search, try the next one */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (cmdctx->check_next) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = dom->next;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* This was a single-domain search.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * exit the loop.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher continue;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Found a result. Check its validity */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (state->res->count > 1) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_OP_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("getservby* returned more than one result!\n"));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = ENOENT;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher lastUpdate = ldb_msg_find_attr_as_uint64(state->res->msgs[0],
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SYSDB_LAST_UPDATE, 0);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cacheExpire = ldb_msg_find_attr_as_uint64(state->res->msgs[0],
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SYSDB_CACHE_EXPIRE, 0);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher midpoint_refresh = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if(nctx->cache_refresh_percent) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher midpoint_refresh = lastUpdate +
056302a92862fda16351d7192600746746f38e5dStephen Gallagher (cacheExpire - lastUpdate)*nctx->cache_refresh_percent/100;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (midpoint_refresh - lastUpdate < 10) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If the percentage results in an expiration
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * less than ten seconds after the lastUpdate time,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * that's too often we will simply set it to 10s
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher midpoint_refresh = lastUpdate+10;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (cacheExpire > now) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* cache still valid */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (NEED_CHECK_PROVIDER(dom->provider)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher && midpoint_refresh
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher && midpoint_refresh < now) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* We're past the the cache refresh timeout
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * We'll return the value from the cache, but we'll also
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * queue the cache entry for update out-of-band.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Performing midpoint cache update\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Update the cache */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher subreq = sss_dp_get_account_send(cctx, cctx->rctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher dom, true,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SSS_DP_SERVICES,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher port, NULL);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!subreq) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Out of memory sending out-of-band data provider "
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher "request\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* This is non-fatal, so we'll continue here */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* We don't need to listen for a reply, so we will free the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * request here.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_zfree(subreq);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* The cache is valid. Return it */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EOK;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Cache is expired. Add this domain to the
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * list of eligible domains to check the provider.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (NEED_CHECK_PROVIDER(dom->provider)) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->domains[dom_idx] = dom;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom_idx++;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* If this is a multi-domain search, try the next one */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (cmdctx->check_next) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = dom->next;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* This was a single-domain search.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * exit the loop.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* No valid cached entries found and
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * not found in negative caches.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Iterate through the domains and try
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * to look the data up.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->dom_idx = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->domains[state->dom_idx]) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* No domains to search. Return ENOENT */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOENT;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = lookup_service_step(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherimmediate:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_done(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_error(req, ret);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_post(req, ev);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic errno_t lookup_service_step(struct tevent_req *req)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct getserv_ctx *state =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_data(req, struct getserv_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct tevent_req *subreq;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct cli_ctx *cctx = state->dctx->cmdctx->cctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_domain_info *dom =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->domains[state->dom_idx];
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Update the cache */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher subreq = sss_dp_get_account_send(req,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cctx->rctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher true,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SSS_DP_SERVICES,
b355dcb54194f498921743ca33304eac35d89718Stephen Gallagher SVC_NAME_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!subreq) return ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_set_callback(subreq, lookup_service_done, req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return EOK;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void lookup_service_done(struct tevent_req *subreq)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dbus_uint16_t err_maj;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dbus_uint32_t err_min;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *err_msg;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sysdb_ctx *sysdb;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct tevent_req *req =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_callback_data(subreq, struct tevent_req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct getserv_ctx *state =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_data(req, struct getserv_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct cli_ctx *cctx = state->dctx->cmdctx->cctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_get_type(cctx->rctx->pvt_ctx, struct nss_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_domain_info *dom = state->domains[state->dom_idx];
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_dp_get_account_recv(state, subreq,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &err_maj, &err_min,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &err_msg);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_zfree(subreq);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_OP_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Unable to get information from Data Provider\n"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher "dp_error: [%u], errno: [%u], error_msg: [%s]\n"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher "Will try to return what we have in cache\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher (unsigned int)err_maj, (unsigned int)err_min,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher err_msg ? err_msg : "none"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Recheck the cache after the lookup.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * We can ignore the expiration values here, because
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * either we have just updated it or the provider is
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * offline. Either way, whatever is in the cache should
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * be returned, if it exists. Otherwise, move to the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * next provider.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher sysdb = dom->sysdb;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (sysdb == NULL) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Critical: Sysdb CTX not found for [%s]!\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EINVAL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (state->name) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Re-checking cache for [%s:%s@%s]\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher SVC_PROTO_CASED ? SVC_PROTO_CASED : "<ANY>",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sysdb_getservbyname(state, sysdb,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher SVC_PROTO_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &state->res);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Re-checking cache for [%lu:%s@%s]\n",
056302a92862fda16351d7192600746746f38e5dStephen Gallagher state->port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED ? SVC_PROTO_CASED : "<ANY>",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sysdb_getservbyport(state, sysdb,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher state->port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &state->res);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret == ENOENT) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Nothing in the cache.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Set the negative cache
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (state->name) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = sss_ncache_set_service_name(nctx->ncache, false,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher dom,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_PROTO_CASED);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Failure to set the negative cache is non-fatal.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * We'll log an error and continue.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
b355dcb54194f498921743ca33304eac35d89718Stephen Gallagher DEBUG(SSSDBG_MINOR_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Could not set negative cache for [%s][%s]\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SVC_NAME_CASED, SVC_PROTO_CASED));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
b355dcb54194f498921743ca33304eac35d89718Stephen Gallagher ret = sss_ncache_set_service_port(nctx->ncache, false,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->port,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher SVC_PROTO_CASED);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret != EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Failure to set the negative cache is non-fatal.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * We'll log an error and continue.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_MINOR_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Could not set negative cache for [%lu][%s]\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->port, SVC_PROTO_CASED));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Need to check other domains */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->dom_idx++;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!state->domains[state->dom_idx]) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* No more domains to search. Return ENOENT */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOENT;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = lookup_service_step(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Set EAGAIN so we will re-enter the mainloop */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EAGAIN;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherdone:
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret == EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Cache contained results. Return them */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_done(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else if (ret != EAGAIN) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* An error occurred, fail the request */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_error(req, ret);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* ret == EAGAIN: Reenter mainloop */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic errno_t
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghergetserv_recv(TALLOC_CTX *mem_ctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_req *req,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct ldb_result **_res)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct getserv_ctx *state =
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_data(req, struct getserv_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher TEVENT_REQ_RETURN_ON_ERROR(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher *_res = talloc_steal(mem_ctx, state->res);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return EOK;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic errno_t
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherfill_service(struct sss_packet *packet,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_domain_info *dom,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct nss_ctx *nctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher const char *protocol,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct ldb_message **msgs,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher unsigned int *count)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher unsigned int msg_count = *count;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher size_t rzero, rsize, aptr;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher unsigned int num, i, j;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint32_t num_aliases, written_aliases;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct ldb_message *msg;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct ldb_message_element *el;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher TALLOC_CTX *tmp_ctx = NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher const char *orig_name;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher const char *orig_proto;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sized_string cased_name;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sized_string cased_proto;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint16_t port;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *tmpstr;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint8_t *body;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher size_t blen;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct sized_string alias;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* FIXME: Should we account for fully-qualified
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * service names?
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* first 2 fields (len and reserved), filled up later */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_packet_grow(packet, 2 * sizeof(uint32_t));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher rzero = 2 * sizeof(uint32_t);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher rsize = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher for (i = 0; i < msg_count; i++) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher talloc_zfree(tmp_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tmp_ctx = talloc_new(NULL);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!tmp_ctx) return ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher msg = msgs[i];
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* new service */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!ldb_msg_check_string_attribute(msg, "objectClass",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SYSDB_SVC_CLASS)) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(1, ("Wrong object (%s) found on stack!\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ldb_dn_get_linearized(msg->dn)));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher continue;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* new result starts at end of previous result */
b355dcb54194f498921743ca33304eac35d89718Stephen Gallagher rzero += rsize;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher rsize = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Get the service name */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher orig_name = ldb_msg_find_attr_as_string(msg, SYSDB_NAME, NULL);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tmpstr = sss_get_cased_name(tmp_ctx, orig_name, dom->case_sensitive);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (tmpstr == NULL) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Could not identify service name, skipping\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher continue;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher to_sized_string(&cased_name, tmpstr);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Get the port */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher port = (uint16_t) ldb_msg_find_attr_as_uint(msg, SYSDB_SVC_PORT, 0);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!port) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("No port for service [%s]. Skipping\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Get the service protocol.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Use the requested protocol if present,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * otherwise take the first protocol returned
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * by the sysdb.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * If more than one is available, select the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * first in the message.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (protocol) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher orig_proto = protocol;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher el = ldb_msg_find_element(msg, SYSDB_SVC_PROTO);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (el->num_values == 0) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EINVAL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher orig_proto = (const char *)el->values[0].data;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tmpstr = sss_get_cased_name(tmp_ctx, orig_proto, dom->case_sensitive);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (tmpstr == NULL) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("sss_get_cased_name failed, skipping\n"));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher continue;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher to_sized_string(&cased_proto, tmpstr);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_packet_grow(packet, 2 * sizeof(uint16_t)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher + sizeof(uint32_t)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher + cased_name.len
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher + cased_proto.len);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num = 0;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_packet_get_body(packet, &body, &blen);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Store the port number */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SAFEALIGN_SET_UINT32(&body[rzero + rsize], (uint32_t)htons(port), &rsize);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Get the aliases */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher el = ldb_msg_find_element(msg, SYSDB_NAME_ALIAS);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!el) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* No aliases for this user */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num_aliases = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher num_aliases = el->num_values;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* We'll store the alias count here */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher aptr = rzero+rsize;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher rsize += sizeof(uint32_t);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Store the primary name */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher safealign_memcpy(&body[rzero + rsize],
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cased_name.str,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cased_name.len,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &rsize);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Store the protocol */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher safealign_memcpy(&body[rzero + rsize],
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cased_proto.str,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cased_proto.len,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &rsize);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher written_aliases = 0;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher for (j = 0; j < num_aliases; j++) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (sss_string_equal(dom->case_sensitive,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher (const char *)el->values[j].data,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cased_name.str)) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher continue;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher to_sized_string(&alias, (const char *)el->values[j].data);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_packet_grow(packet, alias.len);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher num = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_packet_get_body(packet, &body, &blen);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Store the alias */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher safealign_memcpy(&body[rzero + rsize],
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher alias.str,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher alias.len,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &rsize);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher written_aliases++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_zfree(tmpstr);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher SAFEALIGN_SET_UINT32(&body[aptr], written_aliases, &rsize);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EOK;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherdone:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_free(tmp_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK ||num == 0) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* if num is 0 most probably something went wrong,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * reset packet and return ENOENT */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_packet_set_size(packet, 0);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return ENOENT;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ((uint32_t *)body)[0] = num; /* num results */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ((uint32_t *)body)[1] = 0; /* reserved */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher/*****************
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * getservbyname *
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher *****************/
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghererrno_t parse_getservbyname(TALLOC_CTX *mem_ctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher uint8_t *body, size_t blen,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_names_ctx *names,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char **domain_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char **service_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char **service_protocol);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghernss_cmd_getserv_done(struct tevent_req *req);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherint nss_cmd_getservbyname(struct cli_ctx *cctx)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_cmd_ctx *cmdctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_dom_ctx *dctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher char *domname;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *service_name;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *service_protocol;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint8_t *body;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher size_t blen;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_req *req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cmdctx = talloc_zero(cctx, struct nss_cmd_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!cmdctx) return ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx->cctx = cctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx = talloc_zero(cmdctx, struct nss_dom_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!dctx) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->cmdctx = cmdctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* get service name and protocol */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_packet_get_body(cctx->creq->in, &body, &blen);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* if not terminated fail */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (body[blen -1] != '\0') {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EINVAL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = parse_getservbyname(cmdctx, body, blen,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cctx->rctx->names,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &domname,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &service_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &service_protocol);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_OP_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Could not parse request\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->protocol = service_protocol;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Requesting info for service [%s:%s] from [%s]\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher service_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher service_protocol ? service_protocol : "<ANY>",
056302a92862fda16351d7192600746746f38e5dStephen Gallagher domname ? domname : "<ALL>"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (domname) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->domain = responder_get_domain(dctx, cctx->rctx, domname);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!dctx->domain) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOENT;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* this is a multidomain search */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->domain = cctx->rctx->domains;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx->check_next = true;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Identify if this backend requires a provider check */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->check_provider = NEED_CHECK_PROVIDER(dctx->domain->provider);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Ok, find it! */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher req = getserv_send(cmdctx, cctx->ev, 0,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher service_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher service_protocol,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!req) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_set_callback(req, nss_cmd_getserv_done, dctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherdone:
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return nss_cmd_done(cmdctx, ret);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghererrno_t parse_getservbyname(TALLOC_CTX *mem_ctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint8_t *body, size_t blen,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_names_ctx *names,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char **domain_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char **service_name,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char **service_protocol)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher size_t i, j, namelen;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *rawname;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *domname;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher char *svc_name;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *protocol;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher TALLOC_CTX *tmp_ctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tmp_ctx = talloc_new(NULL);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!tmp_ctx) return ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* The raw name is at most one character shorter
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * than the body length (if the protocol wasn't
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * specified). Since this is a common case, we'll
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * just assume the maximum memory size for the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * rawname.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher rawname = talloc_array(tmp_ctx, char, blen - 1);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!rawname) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher i = j = 0;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Copy in the service name */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher while (body[i] && i < (blen - 1)) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher rawname[j] = body[i];
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher i++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher j++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (body[i] != '\0') {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* blen - 1 was reached without hitting
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * a NULL-terminator. No protocol field
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * is possible.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EINVAL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher rawname[j] = '\0';
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher i++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher namelen = i;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher j = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Copy in the protocol */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (body[i] == '\0') {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Zero-length protocol
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Just set the protocol to NULL
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher protocol = NULL;
bdd205037059e56484de3174951b22ff8f0f79f8Stephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* The protocol must be no longer than the remaining
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * body space, after the name was copied.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher protocol = talloc_array(tmp_ctx, char, blen - i - 1);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!protocol) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher while (body[i] && i < blen) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher protocol[j] = body[i];
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher i++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher j++;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (body[i] != '\0') {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* blen was reached without hitting
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * a NULL-terminator.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EINVAL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher protocol[j] = '\0';
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (j != blen - namelen - 1) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_MINOR_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Body longer than the name and protocol\n"));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = EINVAL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = sss_parse_name(tmp_ctx, names, rawname,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &domname, &svc_name);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_MINOR_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Could not split name and domain of [%s]\n",
056302a92862fda16351d7192600746746f38e5dStephen Gallagher rawname));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher *domain_name = talloc_steal(mem_ctx, domname);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher *service_name = talloc_steal(mem_ctx, svc_name);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher *service_protocol = talloc_steal(mem_ctx, protocol);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EOK;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherdone:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_free(tmp_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghernss_cmd_getserv_done(struct tevent_req *req)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret, reqret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher unsigned int i;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_dom_ctx *dctx =
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_callback_data(req, struct nss_dom_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_cmd_ctx *cmdctx = dctx->cmdctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct cli_ctx *cctx = cmdctx->cctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_get_type(cctx->rctx->pvt_ctx, struct nss_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher reqret = getserv_recv(dctx, req, &dctx->res);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_zfree(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (reqret != EOK && reqret != ENOENT) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_OP_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("getservbyname failed\n"));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher nss_cmd_done(cmdctx, reqret);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Either we succeeded or no domains were eligible */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_packet_new(cmdctx->cctx->creq, 0,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_packet_get_cmd(cmdctx->cctx->creq->in),
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &cmdctx->cctx->creq->out);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (reqret == ENOENT) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Notify the caller that this entry wasn't found */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_cmd_empty_packet(cmdctx->cctx->creq->out);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher i = dctx->res->count;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = fill_service(cmdctx->cctx->creq->out,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->domain,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher nctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->protocol,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->res->msgs,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher &i);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_OP_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Could not create response packet: [%s]\n",
056302a92862fda16351d7192600746746f38e5dStephen Gallagher strerror(ret)));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_cmd_done(cmdctx->cctx, cmdctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_OP_FAILURE, ("Error creating packet\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghererrno_t parse_getservbyport(TALLOC_CTX *mem_ctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher uint8_t *body, size_t blen,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_names_ctx *names,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher uint16_t *service_port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char **service_protocol)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher size_t i, j;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher size_t port_and_padding_len;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher uint16_t c, port;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher char *protocol;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher TALLOC_CTX *tmp_ctx = talloc_new(NULL);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!tmp_ctx) return ENOMEM;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Copy in the port */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher SAFEALIGN_COPY_UINT16(&c, body, NULL);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher port = ntohs(c);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher port_and_padding_len = 2 * sizeof(uint16_t);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher i = port_and_padding_len;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher j = 0;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Copy in the protocol */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (body[i] == '\0') {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Zero-length protocol
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * Just set the protocol to NULL
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher protocol = NULL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher } else {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* The protocol must be no longer than the remaining
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * body space.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher protocol = talloc_array(tmp_ctx, char, blen - i);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!protocol) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = ENOMEM;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher while (body[i] && i < blen) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher protocol[j] = body[i];
056302a92862fda16351d7192600746746f38e5dStephen Gallagher i++;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher j++;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (body[i] != '\0') {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* blen was reached without hitting
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * a NULL-terminator.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = EINVAL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher protocol[j] = '\0';
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (j != blen - port_and_padding_len - 1) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_MINOR_FAILURE,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Body longer than the name and protocol\n"));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = EINVAL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher *service_port = port;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher *service_protocol = talloc_steal(mem_ctx, protocol);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = EOK;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherdone:
056302a92862fda16351d7192600746746f38e5dStephen Gallagher talloc_free(tmp_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return ret;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher/*****************
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * getservbyport *
056302a92862fda16351d7192600746746f38e5dStephen Gallagher *****************/
056302a92862fda16351d7192600746746f38e5dStephen Gallagherint nss_cmd_getservbyport(struct cli_ctx *cctx)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher errno_t ret;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct nss_cmd_ctx *cmdctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct nss_dom_ctx *dctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher uint16_t port;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher char *service_protocol;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher uint8_t *body;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher size_t blen;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct tevent_req *req;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx = talloc_zero(cctx, struct nss_cmd_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!cmdctx) return ENOMEM;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx->cctx = cctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher dctx = talloc_zero(cmdctx, struct nss_dom_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!dctx) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = ENOMEM;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher dctx->cmdctx = cmdctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* get service port and protocol */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher sss_packet_get_body(cctx->creq->in, &body, &blen);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* if not terminated fail */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (body[blen -1] != '\0') {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = EINVAL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = parse_getservbyport(cmdctx, body, blen,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cctx->rctx->names,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher &port,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher &service_protocol);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret != EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_OP_FAILURE,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Could not parse request\n"));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto done;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher dctx->protocol = service_protocol;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Requesting info for service on port [%lu/%s]\n",
056302a92862fda16351d7192600746746f38e5dStephen Gallagher port, service_protocol ? service_protocol : "<ANY>"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* All port lookups are multidomain searches */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->domain = cctx->rctx->domains;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cmdctx->check_next = true;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Identify if this backend requires a provider check */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dctx->check_provider = NEED_CHECK_PROVIDER(dctx->domain->provider);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Ok, find it! */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher req = getserv_send(cmdctx, cctx->ev, port,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher NULL, service_protocol, dctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!req) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_set_callback(req, nss_cmd_getserv_done, dctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherdone:
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return nss_cmd_done(cmdctx, ret);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstruct setservent_ctx {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct cli_ctx *cctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_dom_ctx *dctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct getent_ctx *getent_ctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher};
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic errno_t
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghersetservent_step(struct setent_step_ctx *step_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
056302a92862fda16351d7192600746746f38e5dStephen Gallaghersetservent_step_done(struct tevent_req *req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic struct tevent_req *
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherlookup_servent_send(TALLOC_CTX *mem_ctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct resp_ctx *rctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct sss_domain_info *dom);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic struct tevent_req *
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghersetservent_send(TALLOC_CTX *mem_ctx, struct cli_ctx *cctx)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher unsigned int num_domains;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_req *req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct setservent_ctx *state;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sss_domain_info *dom;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct setent_step_ctx *step_ctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_get_type(cctx->rctx->pvt_ctx, struct nss_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC, ("Received setservent request\n"));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Reset the read pointers */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cctx->svc_dom_idx = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cctx->svcent_cur = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher req = tevent_req_create(mem_ctx, &state, struct setservent_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!req) return NULL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->nctx = nctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->cctx = cctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher state->dctx = talloc_zero(state, struct nss_dom_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->dctx) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->dctx->domain = cctx->rctx->domains;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Is the result context already available */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (state->nctx->svcctx) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (state->nctx->svcctx->ready) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* All of the necessary data is in place
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * We can return now, getservent requests will work at this point
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EOK;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Object is still being constructed
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Register for notification when it's
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * ready.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = nss_setent_add_ref(state, state->nctx->svcctx, req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Create a new result context
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * We are creating it on the nss_ctx so that it doesn't
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * go away if the original request does. We will delete
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * it when the refcount goes to zero;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->nctx->svcctx = talloc_zero(nctx, struct getent_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->nctx->svcctx) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->getent_ctx = nctx->svcctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Assume that all domains will have results (to avoid having
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * to reallocate later
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num_domains = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = state->cctx->rctx->domains;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher while (dom) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num_domains++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dom = dom->next;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher state->nctx->svcctx->doms = talloc_zero_array(state->nctx->svcctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct dom_ctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher num_domains);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!state->nctx->svcctx->doms) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Add a callback reference for ourselves */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = nss_setent_add_ref(state, state->nctx->svcctx, req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* ok, start the searches */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher step_ctx = talloc_zero(state->getent_ctx, struct setent_step_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!step_ctx) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Steal the dom_ctx onto the step_ctx so it doesn't go out of scope if
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * this request is canceled while other requests are in-progress.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher step_ctx->dctx = talloc_steal(step_ctx, state->dctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher step_ctx->nctx = state->nctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher step_ctx->getent_ctx = state->getent_ctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher step_ctx->rctx = cctx->rctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher step_ctx->cctx = cctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher step_ctx->returned_to_mainloop = false;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher while (step_ctx->dctx->domain) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* There are more domains to check */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = setservent_step(step_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret == EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Re-enter the mainloop */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return req;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Error [%s] requesting info from domain [%s]. Skipping.\n",
056302a92862fda16351d7192600746746f38e5dStephen Gallagher strerror(ret), step_ctx->dctx->domain->name));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher step_ctx->dctx->domain = step_ctx->dctx->domain->next;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* All domains failed */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = EIO;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherimmediate:
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret == EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_done(req);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher } else {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_error(req, ret);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_post(req, cctx->rctx->ev);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return req;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic errno_t
056302a92862fda16351d7192600746746f38e5dStephen Gallaghersetservent_step(struct setent_step_ctx *step_ctx)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct tevent_req *req;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher req = lookup_servent_send(step_ctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher step_ctx->rctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher step_ctx->dctx->domain);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!req) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return ENOMEM;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_set_callback(req, setservent_step_done, step_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return EOK;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstruct lookup_servent_ctx {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct resp_ctx *rctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct sss_domain_info *dom;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct ldb_result *res;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher};
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic void
056302a92862fda16351d7192600746746f38e5dStephen Gallagherlookup_servent_done(struct tevent_req *subreq);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic void
056302a92862fda16351d7192600746746f38e5dStephen Gallaghersetservent_finalize(struct setent_step_ctx *step_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic struct tevent_req *
056302a92862fda16351d7192600746746f38e5dStephen Gallagherlookup_servent_send(TALLOC_CTX *mem_ctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct resp_ctx *rctx,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct sss_domain_info *dom)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher errno_t ret;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct tevent_req *req;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct tevent_req *subreq;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct lookup_servent_ctx *state;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct sysdb_ctx *sysdb;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher req = tevent_req_create(mem_ctx, &state, struct lookup_servent_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!req) return NULL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher state->rctx = rctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher state->dom = dom;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!dom->enumerate) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = ENOENT;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto immediate;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!(NEED_CHECK_PROVIDER(dom->name))) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* No provider check required. Just ask the
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * sysdb.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher sysdb = dom->sysdb;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (sysdb == NULL) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_FATAL_FAILURE,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Sysdb CTX not found for [%s]!\n", dom->name));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = EINVAL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher goto immediate;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = sysdb_enumservent(state, sysdb, &state->res);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Whatever the result, we're done, so report it */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* We need to ask the provider for an enumeration */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Update the cache */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher subreq = sss_dp_get_account_send(req, rctx, state->dom,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher true, SSS_DP_SERVICES,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher NULL, 0, NULL);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!subreq) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto immediate;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_set_callback(subreq, lookup_servent_done, req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherimmediate:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_done(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_error(req, ENOENT);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_post(req, rctx->ev);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherlookup_servent_done(struct tevent_req *subreq)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dbus_uint16_t dp_err;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher dbus_uint32_t dp_ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher char *err_msg;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct sysdb_ctx *sysdb;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct tevent_req *req =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_callback_data(subreq, struct tevent_req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct lookup_servent_ctx *state =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_data(req, struct lookup_servent_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = sss_dp_get_account_recv(state, subreq,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &dp_err, &dp_ret, &err_msg);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_zfree(subreq);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_OP_FAILURE,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Unable to get information from Data Provider\n"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher "dp_error: [%u], errno: [%u], error_msg: [%s]\n"
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher "Will try to return what we have in cache\n",
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher (unsigned int)dp_err, (unsigned int)dp_ret,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher err_msg ? err_msg : "none"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Check the cache now */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sysdb = state->dom->sysdb;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (sysdb == NULL) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_FATAL_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Sysdb CTX not found for [%s]!\n", state->dom->name));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = EINVAL;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sysdb_enumservent(state, sysdb, &state->res);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Whatever the result, we're done, so report it */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherdone:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_done(req);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_error(req, ret);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic errno_t
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherlookup_servent_recv(TALLOC_CTX *mem_ctx,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_req *req,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct ldb_result **res)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct lookup_servent_ctx *state =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_data(req, struct lookup_servent_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher TEVENT_REQ_RETURN_ON_ERROR(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher *res = talloc_steal(mem_ctx, state->res);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return EOK;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghersetservent_step_done(struct tevent_req *req)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct ldb_result *res;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct setent_step_ctx *step_ctx =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_callback_data(req, struct setent_step_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_dom_ctx *dctx = step_ctx->dctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct getent_ctx *svcctx = step_ctx->getent_ctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = lookup_servent_recv(step_ctx, req, &res);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_zfree(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == ENOENT) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Domain [%s] returned no results\n", dctx->domain->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Error [%s] while retrieving info from domain [%s]. "
056302a92862fda16351d7192600746746f38e5dStephen Gallagher "Skipping.\n", strerror(ret), dctx->domain->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Continue on */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher } else {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Got some results
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Add the retrieved results to the list
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher svcctx->doms[svcctx->num].domain = dctx->domain;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher svcctx->doms[svcctx->num].res = talloc_steal(svcctx->doms, res);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher svcctx->num++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher step_ctx->dctx->domain = step_ctx->dctx->domain->next;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher while (step_ctx->dctx->domain) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* There are more domains to check */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = setservent_step(step_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Re-enter the mainloop */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Error [%s] requesting info from domain [%s]. Skipping.\n",
056302a92862fda16351d7192600746746f38e5dStephen Gallagher strerror(ret), step_ctx->dctx->domain->name));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher step_ctx->dctx->domain = step_ctx->dctx->domain->next;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* All domains have been checked */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher setservent_finalize(step_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghersetservent_result_timeout(struct tevent_context *ev,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_timer *te,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct timeval current_time,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher void *pvt);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghersetservent_finalize(struct setent_step_ctx *step_ctx)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx = step_ctx->nctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct resp_ctx *rctx = step_ctx->rctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct timeval tv;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_timer *te;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* We've finished all our lookups
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * The result object is now safe to read.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher nctx->svcctx->ready = true;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Set up a lifetime timer for this result object
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * We don't want this result object to outlive the
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * enum cache refresh timeout
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tv = tevent_timeval_current_ofs(nctx->enum_cache_timeout, 0);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher te = tevent_add_timer(rctx->ev, nctx->svcctx, tv,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher setservent_result_timeout, nctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!te) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Could not set up life timer for setservent result object. "
056302a92862fda16351d7192600746746f38e5dStephen Gallagher "Entries may become stale.\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher nss_setent_notify_done(nctx->svcctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghersetservent_result_timeout(struct tevent_context *ev,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_timer *te,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct timeval current_time,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher void *pvt)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx = talloc_get_type(pvt, struct nss_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("setservent result object has expired. Cleaning up.\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Free the service enumeration context.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * If additional getservent requests come in, they will invoke
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * an implicit setservent and refresh the result object.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_zfree(nctx->svcctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic errno_t
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghersetservent_recv(struct tevent_req *req)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher TEVENT_REQ_RETURN_ON_ERROR(req);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return EOK;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghernss_cmd_setservent_done(struct tevent_req *req);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherint
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghernss_cmd_setservent(struct cli_ctx *cctx)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_cmd_ctx *cmdctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct tevent_req *req;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret = EOK;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cmdctx = talloc_zero(cctx, struct nss_cmd_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!cmdctx) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return ENOMEM;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cmdctx->cctx = cctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher req = setservent_send(cmdctx, cctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!req) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ("Fatal error calling nss_cmd_setservent_send\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = EIO;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_set_callback(req, nss_cmd_setservent_done, cmdctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherdone:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return nss_cmd_done(cmdctx, ret);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghernss_cmd_setservent_done(struct tevent_req *req)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher errno_t ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_cmd_ctx *cmdctx =
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher tevent_req_callback_data(req, struct nss_cmd_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = setservent_recv(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher talloc_zfree(req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EOK || ret == ENOENT) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Either we succeeded or no domains
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * were eligible.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Return an acknowledgment
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = sss_packet_new(cmdctx->cctx->creq, 0,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_packet_get_cmd(cmdctx->cctx->creq->in),
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &cmdctx->cctx->creq->out);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher sss_cmd_done(cmdctx->cctx, cmdctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Something bad happened.
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher * Return an error
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher nss_cmd_done(cmdctx, ret);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic void
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghernss_cmd_implicit_setservent_done(struct tevent_req *req);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic errno_t
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghernss_cmd_getservent_immediate(struct nss_cmd_ctx *cmdctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherstatic errno_t
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherretservent(struct cli_ctx *cctx, int num);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherint nss_cmd_getservent(struct cli_ctx *cctx)
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct nss_cmd_ctx *cmdctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct tevent_req *req;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Requesting info for all services\n"));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx = talloc_zero(cctx, struct nss_cmd_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!cmdctx) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return ENOMEM;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx->cctx = cctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Save the current index and cursor locations
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * If we end up calling setservent implicitly, because the response object
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * expired and has to be recreated, we want to resume from the same
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * location.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx->saved_dom_idx = cctx->svc_dom_idx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx->saved_cur = cctx->svcent_cur;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher nctx = talloc_get_type(cctx->rctx->pvt_ctx, struct nss_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if(!nctx->svcctx || !nctx->svcctx->ready) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Make sure we invoke setservent if it hasn't been run or is still
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * processing from another client
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher req = setservent_send(cmdctx, cctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (!req) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return EIO;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_set_callback(req,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher nss_cmd_implicit_setservent_done,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return EOK;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return nss_cmd_getservent_immediate(cmdctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic void
056302a92862fda16351d7192600746746f38e5dStephen Gallaghernss_cmd_implicit_setservent_done(struct tevent_req *req)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher errno_t ret;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct nss_cmd_ctx *cmdctx =
056302a92862fda16351d7192600746746f38e5dStephen Gallagher tevent_req_callback_data(req, struct nss_cmd_ctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = setservent_recv(req);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher talloc_zfree(req);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* ENOENT is acceptable, as it just means that there were no entries
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * to be returned. This will be handled gracefully in retservent
056302a92862fda16351d7192600746746f38e5dStephen Gallagher * later.
056302a92862fda16351d7192600746746f38e5dStephen Gallagher */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret != EOK && ret != ENOENT) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Implicit setservent failed with unexpected error [%d][%s]\n",
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret, strerror(ret)));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher NSS_CMD_FATAL_ERROR(cmdctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* Restore the saved index and cursor locations */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx->cctx->svc_dom_idx = cmdctx->saved_dom_idx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cmdctx->cctx->svcent_cur = cmdctx->saved_cur;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = nss_cmd_getservent_immediate(cmdctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret != EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher DEBUG(SSSDBG_CRIT_FAILURE,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Immediate retrieval failed with unexpected error "
056302a92862fda16351d7192600746746f38e5dStephen Gallagher "[%d][%s]\n", ret, strerror(ret)));
056302a92862fda16351d7192600746746f38e5dStephen Gallagher NSS_CMD_FATAL_ERROR(cmdctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic errno_t
056302a92862fda16351d7192600746746f38e5dStephen Gallaghernss_cmd_getservent_immediate(struct nss_cmd_ctx *cmdctx)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct cli_ctx *cctx = cmdctx->cctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher uint8_t *body;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher size_t blen;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher uint32_t num;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher int ret;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* get max num of entries to return in one call */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher sss_packet_get_body(cctx->creq->in, &body, &blen);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (blen != sizeof(uint32_t)) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return EINVAL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher num = *((uint32_t *)body);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher /* create response packet */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = sss_packet_new(cctx->creq, 0,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher sss_packet_get_cmd(cctx->creq->in),
056302a92862fda16351d7192600746746f38e5dStephen Gallagher &cctx->creq->out);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (ret != EOK) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return ret;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = retservent(cctx, num);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher sss_packet_set_error(cctx->creq->out, ret);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher sss_cmd_done(cctx, cmdctx);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher return EOK;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher}
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagherstatic errno_t
056302a92862fda16351d7192600746746f38e5dStephen Gallagherretservent(struct cli_ctx *cctx, int num)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct nss_ctx *nctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct getent_ctx *svcctx;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct ldb_message **msgs = NULL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher struct dom_ctx *pdom = NULL;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher unsigned int n = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher int ret = ENOENT;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher nctx = talloc_get_type(cctx->rctx->pvt_ctx, struct nss_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!nctx->svcctx) goto none;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher svcctx = nctx->svcctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher while (ret == ENOENT) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (cctx->svc_dom_idx >= svcctx->num) break;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher pdom = &svcctx->doms[cctx->svc_dom_idx];
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher n = pdom->res->count - cctx->svcent_cur;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher if (n <= 0 && (cctx->svc_dom_idx+1 < svcctx->num)) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cctx->svc_dom_idx++;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher pdom = &svcctx->doms[cctx->svc_dom_idx];
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher n = pdom->res->count;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cctx->svcent_cur = 0;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (!n) break;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (n > num) n = num;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
056302a92862fda16351d7192600746746f38e5dStephen Gallagher msgs = &(pdom->res->msgs[cctx->svcent_cur]);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher ret = fill_service(cctx->creq->out,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher pdom->domain,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher nctx, NULL, msgs,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &n);
056302a92862fda16351d7192600746746f38e5dStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cctx->svcent_cur += n;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallaghernone:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret == ENOENT) {
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = sss_cmd_empty_packet(cctx->creq->out);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherint nss_cmd_endservent(struct cli_ctx *cctx)
056302a92862fda16351d7192600746746f38e5dStephen Gallagher{
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher struct nss_ctx *nctx;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher int ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher DEBUG(SSSDBG_TRACE_FUNC,
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ("Terminating request info for all accounts\n"));
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher nctx = talloc_get_type(cctx->rctx->pvt_ctx, struct nss_ctx);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* create response packet */
056302a92862fda16351d7192600746746f38e5dStephen Gallagher ret = sss_packet_new(cctx->creq, 0,
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_packet_get_cmd(cctx->creq->in),
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher &cctx->creq->out);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (ret != EOK) {
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return ret;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher }
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher if (nctx->svcctx == NULL) goto done;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher /* Reset the indices so that subsequent requests start at zero */
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher cctx->svc_dom_idx = 0;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher cctx->svcent_cur = 0;
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagherdone:
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher sss_cmd_done(cctx, NULL);
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher return EOK;
056302a92862fda16351d7192600746746f38e5dStephen Gallagher}
52261fe16203dec6e6f69177c6d0a810b47d073fStephen Gallagher