Blame SOURCES/autofs-5.0.9-amd-lookup-update-lookup-yp-to-handle-amd-keys.patch

4d476f
autofs-5.0.9 - amd lookup update lookup yp to handle amd keys
4d476f
4d476f
From: Ian Kent <raven@themaw.net>
4d476f
4d476f
4d476f
---
4d476f
 modules/lookup_yp.c |  248 ++++++++++++++++++++++++++++++++++++++++++++-------
4d476f
 1 file changed, 216 insertions(+), 32 deletions(-)
4d476f
4d476f
diff --git a/modules/lookup_yp.c b/modules/lookup_yp.c
4d476f
index ba97ccc..146e39e 100644
4d476f
--- a/modules/lookup_yp.c
4d476f
+++ b/modules/lookup_yp.c
4d476f
@@ -36,9 +36,10 @@
4d476f
 #define MODPREFIX "lookup(yp): "
4d476f
 
4d476f
 struct lookup_context {
4d476f
-	const char *domainname;
4d476f
+	char *domainname;
4d476f
 	const char *mapname;
4d476f
 	unsigned long order;
4d476f
+	unsigned int check_defaults;
4d476f
 	struct parse_mod *parse;
4d476f
 };
4d476f
 
4d476f
@@ -113,7 +114,7 @@ int lookup_init(const char *mapfmt, int argc, const char *const *argv, void **co
4d476f
 	ctxt = malloc(sizeof(struct lookup_context));
4d476f
 	if (!ctxt) {
4d476f
 		char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
4d476f
-		logerr(MODPREFIX "%s", estr);
4d476f
+		logerr(MODPREFIX "malloc: %s", estr);
4d476f
 		return 1;
4d476f
 	}
4d476f
 	memset(ctxt, 0, sizeof(struct lookup_context));
4d476f
@@ -124,17 +125,28 @@ int lookup_init(const char *mapfmt, int argc, const char *const *argv, void **co
4d476f
 		return 1;
4d476f
 	}
4d476f
 	ctxt->mapname = argv[0];
4d476f
-
4d476f
-	/* This should, but doesn't, take a const char ** */
4d476f
-	err = yp_get_default_domain((char **) &ctxt->domainname);
4d476f
-	if (err) {
4d476f
-		size_t len = strlen(ctxt->mapname);
4d476f
-		char *name = alloca(len + 1);
4d476f
-		memcpy(name, ctxt->mapname, len);
4d476f
-		name[len] = '\0';
4d476f
-		free(ctxt);
4d476f
-		logerr(MODPREFIX "map %s: %s", name, yperr_string(err));
4d476f
-		return 1;
4d476f
+	ctxt->check_defaults = 1;
4d476f
+
4d476f
+	if (mapfmt && !strcmp(mapfmt, "amd"))
4d476f
+		ctxt->domainname = conf_amd_get_nis_domain();
4d476f
+
4d476f
+	if (!ctxt->domainname) {
4d476f
+		char *domainname;
4d476f
+		/* This should, but doesn't, take a const char ** */
4d476f
+		err = yp_get_default_domain(&domainname);
4d476f
+		if (err) {
4d476f
+			logerr(MODPREFIX
4d476f
+			      "map %s: %s", ctxt->mapname, yperr_string(err));
4d476f
+			free(ctxt);
4d476f
+			return 1;
4d476f
+		}
4d476f
+		ctxt->domainname = strdup(domainname);
4d476f
+		if (!ctxt->domainname) {
4d476f
+			char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
4d476f
+			logerr(MODPREFIX "strdup: %s", estr);
4d476f
+			free(ctxt);
4d476f
+			return 1;
4d476f
+		}
4d476f
 	}
4d476f
 
4d476f
 	ctxt->order = get_map_order(ctxt->domainname, ctxt->mapname);
4d476f
@@ -286,7 +298,12 @@ int yp_all_callback(int status, char *ypkey, int ypkeylen,
4d476f
 	if (*ypkey == '+')
4d476f
 		return 0;
4d476f
 
4d476f
-	key = sanitize_path(ypkey, ypkeylen, ap->type, ap->logopt);
4d476f
+	if (!(source->flags & MAP_FLAG_FORMAT_AMD))
4d476f
+		key = sanitize_path(ypkey, ypkeylen, ap->type, ap->logopt);
4d476f
+	else
4d476f
+		/* Don't fail on "/" in key => type == 0 */
4d476f
+		key = sanitize_path(ypkey, ypkeylen, 0, ap->logopt);
4d476f
+
4d476f
 	if (!key) {
4d476f
 		error(logopt, MODPREFIX "invalid path %s", ypkey);
4d476f
 		return 0;
4d476f
@@ -372,6 +389,9 @@ int lookup_read_map(struct autofs_point *ap, time_t age, void *context)
4d476f
 	}
4d476f
 
4d476f
 	source->age = age;
4d476f
+	pthread_mutex_lock(&ap->entry->current_mutex);
4d476f
+	ctxt->check_defaults = 0;
4d476f
+	pthread_mutex_unlock(&ap->entry->current_mutex);
4d476f
 
4d476f
 	return NSS_STATUS_SUCCESS;
4d476f
 }
4d476f
@@ -432,6 +452,66 @@ static int lookup_one(struct autofs_point *ap,
4d476f
 	return ret;
4d476f
 }
4d476f
 
4d476f
+static int match_key(struct autofs_point *ap,
4d476f
+		     struct map_source *source,
4d476f
+		     const char *key, int key_len,
4d476f
+		     struct lookup_context *ctxt)
4d476f
+{
4d476f
+	char buf[MAX_ERR_BUF];
4d476f
+	char *lkp_key;
4d476f
+	char *prefix;
4d476f
+	int ret;
4d476f
+
4d476f
+	ret = lookup_one(ap, source, key, strlen(key), ctxt);
4d476f
+	if (ret < 0)
4d476f
+		return ret;
4d476f
+	if (ret == CHE_OK || ret == CHE_UPDATED)
4d476f
+		return ret;
4d476f
+
4d476f
+	if (!(source->flags & MAP_FLAG_FORMAT_AMD))
4d476f
+		return CHE_FAIL;
4d476f
+
4d476f
+	lkp_key = strdup(key);
4d476f
+	if (!lkp_key) {
4d476f
+		char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
4d476f
+		error(ap->logopt, MODPREFIX "strdup: %s", estr);
4d476f
+		return CHE_FAIL;
4d476f
+	}
4d476f
+
4d476f
+	ret = CHE_MISSING;
4d476f
+
4d476f
+	/*
4d476f
+	 * Now strip successive directory components and try a
4d476f
+	 * match against map entries ending with a wildcard and
4d476f
+	 * finally try the wilcard entry itself.
4d476f
+	 */
4d476f
+	while ((prefix = strrchr(lkp_key, '/'))) {
4d476f
+		char *match;
4d476f
+		size_t len;
4d476f
+		*prefix = '\0';
4d476f
+		len = strlen(lkp_key) + 3;
4d476f
+		match = malloc(len);
4d476f
+		if (!match) {
4d476f
+			char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
4d476f
+			error(ap->logopt, MODPREFIX "malloc: %s", estr);
4d476f
+			ret = CHE_FAIL;
4d476f
+			goto done;
4d476f
+		}
4d476f
+		len--;
4d476f
+		strcpy(match, lkp_key);
4d476f
+		strcat(match, "/*");
4d476f
+		ret = lookup_one(ap, source, match, len, ctxt);
4d476f
+		free(match);
4d476f
+		if (ret < 0)
4d476f
+			goto done;
4d476f
+		if (ret == CHE_OK || ret == CHE_UPDATED)
4d476f
+			goto done;
4d476f
+	}
4d476f
+done:
4d476f
+	free(lkp_key);
4d476f
+	return ret;
4d476f
+}
4d476f
+
4d476f
 static int lookup_wild(struct autofs_point *ap,
4d476f
 		       struct map_source *source, struct lookup_context *ctxt)
4d476f
 {
4d476f
@@ -480,6 +560,52 @@ static int lookup_wild(struct autofs_point *ap,
4d476f
 	return ret;
4d476f
 }
4d476f
 
4d476f
+static int lookup_amd_defaults(struct autofs_point *ap,
4d476f
+			       struct map_source *source,
4d476f
+			       struct lookup_context *ctxt)
4d476f
+{
4d476f
+	struct mapent_cache *mc = source->mc;
4d476f
+	char *mapname;
4d476f
+	char *mapent;
4d476f
+	int mapent_len;
4d476f
+	int ret;
4d476f
+
4d476f
+	mapname = malloc(strlen(ctxt->mapname) + 1);
4d476f
+	if (!mapname)
4d476f
+		return 0;
4d476f
+
4d476f
+	strcpy(mapname, ctxt->mapname);
4d476f
+
4d476f
+	ret = yp_match((char *) ctxt->domainname, mapname,
4d476f
+		       (char *) "/defaults", 9, &mapent, &mapent_len);
4d476f
+
4d476f
+	if (ret != YPERR_SUCCESS) {
4d476f
+		if (ret == YPERR_MAP) {
4d476f
+			char *usc;
4d476f
+
4d476f
+			while ((usc = strchr(mapname, '_')))
4d476f
+				*usc = '.';
4d476f
+
4d476f
+			ret = yp_match((char *) ctxt->domainname, mapname,
4d476f
+				       "/defaults", 9, &mapent, &mapent_len);
4d476f
+		}
4d476f
+	}
4d476f
+	free(mapname);
4d476f
+
4d476f
+	/* No /defaults entry */
4d476f
+	if (ret == YPERR_KEY)
4d476f
+		return CHE_OK;
4d476f
+
4d476f
+	if (ret != YPERR_SUCCESS)
4d476f
+		return CHE_FAIL;
4d476f
+
4d476f
+	cache_writelock(mc);
4d476f
+	ret = cache_update(mc, source, "/defaults", mapent, time(NULL));
4d476f
+	cache_unlock(mc);
4d476f
+
4d476f
+	return ret;
4d476f
+}
4d476f
+
4d476f
 static int check_map_indirect(struct autofs_point *ap,
4d476f
 			      struct map_source *source,
4d476f
 			      char *key, int key_len,
4d476f
@@ -492,8 +618,28 @@ static int check_map_indirect(struct autofs_point *ap,
4d476f
 
4d476f
 	mc = source->mc;
4d476f
 
4d476f
+	/* Only read map if it has been modified */
4d476f
+	pthread_mutex_lock(&ap->entry->current_mutex);
4d476f
+	map_order = get_map_order(ctxt->domainname, ctxt->mapname);
4d476f
+	if (map_order > ctxt->order) {
4d476f
+		ctxt->order = map_order;
4d476f
+		source->stale = 1;
4d476f
+		ctxt->check_defaults = 1;
4d476f
+	}
4d476f
+
4d476f
+	if (source->flags & MAP_FLAG_FORMAT_AMD && ctxt->check_defaults) {
4d476f
+		/* Check for a /defaults entry to update the map source */
4d476f
+		if (lookup_amd_defaults(ap, source, ctxt) == CHE_FAIL) {
4d476f
+			warn(ap->logopt, MODPREFIX
4d476f
+			     "error getting /defaults from map %s",
4d476f
+			     ctxt->mapname);
4d476f
+		} else
4d476f
+			ctxt->check_defaults = 0;
4d476f
+	}
4d476f
+	pthread_mutex_unlock(&ap->entry->current_mutex);
4d476f
+
4d476f
 	/* check map and if change is detected re-read map */
4d476f
-	ret = lookup_one(ap, source, key, key_len, ctxt);
4d476f
+	ret = match_key(ap, source, key, key_len, ctxt);
4d476f
 	if (ret == CHE_FAIL)
4d476f
 		return NSS_STATUS_NOTFOUND;
4d476f
 
4d476f
@@ -503,7 +649,10 @@ static int check_map_indirect(struct autofs_point *ap,
4d476f
 		 * and belongs to this map return success and use the entry.
4d476f
 		 */
4d476f
 		cache_readlock(mc);
4d476f
-		exists = cache_lookup(mc, key);
4d476f
+		if (source->flags & MAP_FLAG_FORMAT_AMD)
4d476f
+			exists = match_cached_key(ap, MODPREFIX, source, key);
4d476f
+		else
4d476f
+			exists = cache_lookup(mc, key);
4d476f
 		if (exists && exists->source == source) {
4d476f
 			cache_unlock(mc);
4d476f
 			return NSS_STATUS_SUCCESS;
4d476f
@@ -517,15 +666,11 @@ static int check_map_indirect(struct autofs_point *ap,
4d476f
 		return NSS_STATUS_UNAVAIL;
4d476f
 	}
4d476f
 
4d476f
-	/* Only read map if it has been modified */
4d476f
-	map_order = get_map_order(ctxt->domainname, ctxt->mapname);
4d476f
-	if (map_order > ctxt->order) {
4d476f
-		ctxt->order = map_order;
4d476f
-		source->stale = 1;
4d476f
-	}
4d476f
-
4d476f
 	cache_writelock(mc);
4d476f
-	exists = cache_lookup_distinct(mc, key);
4d476f
+	if (source->flags & MAP_FLAG_FORMAT_AMD)
4d476f
+		exists = match_cached_key(ap, MODPREFIX, source, key);
4d476f
+	else
4d476f
+		exists = cache_lookup_distinct(mc, key);
4d476f
 	/* Not found in the map but found in the cache */
4d476f
 	if (exists && exists->source == source && ret & CHE_MISSING) {
4d476f
 		if (exists->mapent) {
4d476f
@@ -578,9 +723,11 @@ int lookup_mount(struct autofs_point *ap, const char *name, int name_len, void *
4d476f
 	struct mapent_cache *mc;
4d476f
 	char key[KEY_MAX_LEN + 1];
4d476f
 	int key_len;
4d476f
+	char *lkp_key;
4d476f
 	char *mapent = NULL;
4d476f
 	int mapent_len;
4d476f
 	struct mapent *me;
4d476f
+	char buf[MAX_ERR_BUF];
4d476f
 	int status = 0;
4d476f
 	int ret = 1;
4d476f
 
4d476f
@@ -592,9 +739,18 @@ int lookup_mount(struct autofs_point *ap, const char *name, int name_len, void *
4d476f
 
4d476f
 	debug(ap->logopt, MODPREFIX "looking up %s", name);
4d476f
 
4d476f
-	key_len = snprintf(key, KEY_MAX_LEN + 1, "%s", name);
4d476f
-	if (key_len > KEY_MAX_LEN)
4d476f
-		return NSS_STATUS_NOTFOUND;
4d476f
+	if (!(source->flags & MAP_FLAG_FORMAT_AMD)) {
4d476f
+		key_len = snprintf(key, KEY_MAX_LEN + 1, "%s", name);
4d476f
+		if (key_len > KEY_MAX_LEN)
4d476f
+			return NSS_STATUS_NOTFOUND;
4d476f
+	} else {
4d476f
+		key_len = expandamdent(name, NULL, NULL);
4d476f
+		if (key_len > KEY_MAX_LEN)
4d476f
+			return NSS_STATUS_NOTFOUND;
4d476f
+		memset(key, 0, KEY_MAX_LEN + 1);
4d476f
+		expandamdent(name, key, NULL);
4d476f
+		debug(ap->logopt, MODPREFIX "expanded key: \"%s\"", key);
4d476f
+	}
4d476f
 
4d476f
 	/* Check if we recorded a mount fail for this key anywhere */
4d476f
 	me = lookup_source_mapent(ap, key, LKP_DISTINCT);
4d476f
@@ -628,18 +784,26 @@ int lookup_mount(struct autofs_point *ap, const char *name, int name_len, void *
4d476f
 	  * we never know about it.
4d476f
 	  */
4d476f
         if (ap->type == LKP_INDIRECT && *key != '/') {
4d476f
-		char *lkp_key;
4d476f
-
4d476f
 		cache_readlock(mc);
4d476f
 		me = cache_lookup_distinct(mc, key);
4d476f
 		if (me && me->multi)
4d476f
 			lkp_key = strdup(me->multi->key);
4d476f
-		else
4d476f
+		else if (!ap->pref)
4d476f
 			lkp_key = strdup(key);
4d476f
+		else {
4d476f
+			lkp_key = malloc(strlen(ap->pref) + strlen(key) + 1);
4d476f
+			if (lkp_key) {
4d476f
+				strcpy(lkp_key, ap->pref);
4d476f
+				strcat(lkp_key, key);
4d476f
+			}
4d476f
+		}
4d476f
 		cache_unlock(mc);
4d476f
 
4d476f
-		if (!lkp_key)
4d476f
+		if (!lkp_key) {
4d476f
+			char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
4d476f
+			error(ap->logopt, MODPREFIX "malloc: %s", estr);
4d476f
 			return NSS_STATUS_UNKNOWN;
4d476f
+		}
4d476f
 
4d476f
 		status = check_map_indirect(ap, source,
4d476f
 					    lkp_key, strlen(lkp_key), ctxt);
4d476f
@@ -660,7 +824,25 @@ int lookup_mount(struct autofs_point *ap, const char *name, int name_len, void *
4d476f
 		cache_readlock(mc);
4d476f
 	else
4d476f
 		cache_writelock(mc);
4d476f
-	me = cache_lookup(mc, key);
4d476f
+
4d476f
+	if (!ap->pref)
4d476f
+		lkp_key = strdup(key);
4d476f
+	else {
4d476f
+		lkp_key = malloc(strlen(ap->pref) + strlen(key) + 1);
4d476f
+		if (lkp_key) {
4d476f
+			strcpy(lkp_key, ap->pref);
4d476f
+			strcat(lkp_key, key);
4d476f
+		}
4d476f
+	}
4d476f
+
4d476f
+	if (!lkp_key) {
4d476f
+		char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
4d476f
+		error(ap->logopt, MODPREFIX "malloc: %s", estr);
4d476f
+		cache_unlock(mc);
4d476f
+		return NSS_STATUS_UNKNOWN;
4d476f
+	}
4d476f
+
4d476f
+	me = match_cached_key(ap, MODPREFIX, source, lkp_key);
4d476f
 	/* Stale mapent => check for entry in alternate source or wildcard */
4d476f
 	if (me && !me->mapent) {
4d476f
 		while ((me = cache_lookup_key_next(me)))
4d476f
@@ -687,6 +869,7 @@ int lookup_mount(struct autofs_point *ap, const char *name, int name_len, void *
4d476f
 		}
4d476f
 	}
4d476f
 	cache_unlock(mc);
4d476f
+	free(lkp_key);
4d476f
 
4d476f
 	if (mapent) {
4d476f
 		master_source_current_wait(ap->entry);
4d476f
@@ -716,6 +899,7 @@ int lookup_done(void *context)
4d476f
 {
4d476f
 	struct lookup_context *ctxt = (struct lookup_context *) context;
4d476f
 	int rv = close_parse(ctxt->parse);
4d476f
+	free(ctxt->domainname);
4d476f
 	free(ctxt);
4d476f
 	return rv;
4d476f
 }