MEDIUM: map: dynamic manipulation of maps

This patch adds map manipulation commands to the socket interface.

add map <map> <key> <value>
  Add the value <value> in the map <map>, at the entry corresponding to
  the key <key>. This command does not verify if the entry already
  exists.

clear map <map>
  Remove entries from the map <map>

del map <map> <key>
  Delete all the map entries corresponding to the <key> value in the map
  <map>.

set map <map> <key> <value>
  Modify the value corresponding to each key <key> in a map <map>. The
  new value is <value>.

show map [<map>]
  Dump info about map converters. Without argument, the list of all
  available maps are returned. If a <map> is specified, is content is
  dumped.
diff --git a/doc/configuration.txt b/doc/configuration.txt
index 4043e5c..2080928 100644
--- a/doc/configuration.txt
+++ b/doc/configuration.txt
@@ -12292,6 +12292,11 @@
 all supported commands. Some commands support a more complex syntax, generally
 it will explain what part of the command is invalid when this happens.
 
+add map <map> <key> <value>
+  Add an entry into the map <map> to associate the value <value> to the key
+  <key>. This command does not verify if the entry already exists. It is
+  mainly used to fill a map after a clear operation.
+
 clear counters
   Clear the max values of the statistics counters in each proxy (frontend &
   backend) and in each server. The cumulated counters are not affected. This
@@ -12304,6 +12309,9 @@
   server. This has the same effect as restarting. This command is restricted
   and can only be issued on sockets configured for level "admin".
 
+clear map <map>
+  Remove all entries from the map <map>.
+
 clear table <table> [ data.<type> <operator> <value> ] | [ key <key> ]
   Remove entries from the stick-table <table>.
 
@@ -12374,6 +12382,9 @@
   This command is restricted and can only be issued on sockets configured for
   level "admin".
 
+del map <map> <key>
+  Delete all the map entries from the map <map> corresponding to the key <key>.
+
 disable frontend <frontend>
   Mark the frontend as temporarily stopped. This corresponds to the mode which
   is used during a soft restart : the frontend releases the port but can be
@@ -12462,6 +12473,10 @@
 quit
   Close the connection when in interactive mode.
 
+set map <map> <key> <value>
+  Modify the value corresponding to each key <key> in a map <map>. The new value
+  is <value>.
+
 set maxconn frontend <frontend> <value>
   Dynamically change the specified frontend's maxconn setting. Any positive
   value is allowed including zero, but setting values larger than the global
@@ -12577,6 +12592,10 @@
 show info
   Dump info about haproxy status on current process.
 
+show map [<map>]
+  Dump info about map converters. Without argument, the list of all available
+  maps is returned. If a <map> is specified, its contents are dumped.
+
 show sess
   Dump all known sessions. Avoid doing this on slow connections as this can
   be huge. This command is restricted and can only be issued on sockets
diff --git a/include/proto/dumpstats.h b/include/proto/dumpstats.h
index fd345d5..0da3091 100644
--- a/include/proto/dumpstats.h
+++ b/include/proto/dumpstats.h
@@ -39,20 +39,24 @@
 #define STATS_TYPE_SO  3
 
 /* unix stats socket states */
-#define STAT_CLI_INIT   0   /* initial state, must leave to zero ! */
-#define STAT_CLI_END    1   /* final state, let's close */
-#define STAT_CLI_GETREQ 2   /* wait for a request */
-#define STAT_CLI_OUTPUT 3   /* all states after this one are responses */
-#define STAT_CLI_PROMPT 3   /* display the prompt (first output, same code) */
-#define STAT_CLI_PRINT  4   /* display message in cli->msg */
+#define STAT_CLI_INIT    0   /* initial state, must leave to zero ! */
+#define STAT_CLI_END     1   /* final state, let's close */
+#define STAT_CLI_GETREQ  2   /* wait for a request */
+#define STAT_CLI_OUTPUT  3   /* all states after this one are responses */
+#define STAT_CLI_PROMPT  3   /* display the prompt (first output, same code) */
+#define STAT_CLI_PRINT   4   /* display message in cli->msg */
+
+#define STAT_CLI_O_INFO  5   /* dump info */
+#define STAT_CLI_O_SESS  6   /* dump sessions */
+#define STAT_CLI_O_ERR   7   /* dump errors */
+#define STAT_CLI_O_TAB   8   /* dump tables */
+#define STAT_CLI_O_CLR   9   /* clear tables */
+#define STAT_CLI_O_SET   10  /* set entries in tables */
+#define STAT_CLI_O_STAT  11  /* dump stats */
 
-#define STAT_CLI_O_INFO 5   /* dump info */
-#define STAT_CLI_O_SESS 6   /* dump sessions */
-#define STAT_CLI_O_ERR  7   /* dump errors */
-#define STAT_CLI_O_TAB  8   /* dump tables */
-#define STAT_CLI_O_CLR  9   /* clear tables */
-#define STAT_CLI_O_SET  10  /* set entries in tables */
-#define STAT_CLI_O_STAT 11  /* dump stats */
+#define STAT_CLI_O_MAPS  12  /* list all maps */
+#define STAT_CLI_O_MAP   13  /* list all map entries of a map */
+#define STAT_CLI_O_MLOOK 14  /* lookup a map entry */
 
 /* HTTP stats : applet.st0 */
 enum {
diff --git a/include/types/stream_interface.h b/include/types/stream_interface.h
index f0b63c4..16c127d 100644
--- a/include/types/stream_interface.h
+++ b/include/types/stream_interface.h
@@ -141,6 +141,12 @@
 		struct {
 			void *ptr;              /* multi-purpose pointer for peers */
 		} peers;
+		struct {
+			struct map_reference *ref;
+			struct map_entry *ent;
+			struct map_descriptor *desc;
+			struct chunk chunk;
+		} map;
 	} ctx;					/* used by stats I/O handlers to dump the stats */
 };
 
diff --git a/src/dumpstats.c b/src/dumpstats.c
index 96c5cea..c63464d 100644
--- a/src/dumpstats.c
+++ b/src/dumpstats.c
@@ -46,8 +46,10 @@
 #include <proto/fd.h>
 #include <proto/freq_ctr.h>
 #include <proto/log.h>
+#include <proto/pattern.h>
 #include <proto/pipe.h>
 #include <proto/listener.h>
+#include <proto/map.h>
 #include <proto/proto_http.h>
 #include <proto/proto_uxst.h>
 #include <proto/proxy.h>
@@ -68,6 +70,9 @@
 static int stats_table_request(struct stream_interface *si, int show);
 static int stats_dump_proxy_to_buffer(struct stream_interface *si, struct proxy *px, struct uri_auth *uri);
 static int stats_dump_stat_to_buffer(struct stream_interface *si, struct uri_auth *uri);
+static int stats_maps_list(struct stream_interface *si);
+static int stats_map_list(struct stream_interface *si);
+static int stats_map_lookup(struct stream_interface *si);
 
 /*
  * cli_io_handler()
@@ -103,6 +108,7 @@
 	"Unknown command. Please enter one of the following commands only :\n"
 	"  clear counters : clear max statistics counters (add 'all' for all counters)\n"
 	"  clear table    : remove an entry from a table\n"
+	"  clear map [id] : clear the content of this map\n"
 	"  help           : this message\n"
 	"  prompt         : toggle interactive mode with prompt\n"
 	"  quit           : disconnect\n"
@@ -111,12 +117,16 @@
 	"  show errors    : report last request and response errors for each proxy\n"
 	"  show sess [id] : report the list of current sessions or dump this session\n"
 	"  show table [id]: report table usage stats or dump this table's contents\n"
+	"  show map [id]  : report avalaible maps or dump this map's contents\n"
 	"  get weight     : report a server's current weight\n"
 	"  set weight     : change a server's weight\n"
 	"  set table [id] : update or create a table entry's data\n"
 	"  set timeout    : change a timeout setting\n"
 	"  set maxconn    : change a maxconn setting\n"
 	"  set rate-limit : change a rate limiting value\n"
+	"  set map [id] [key] [value] : modify map entry\n"
+	"  add map [id] [key] [value] : add map entry\n"
+	"  del map [id] [key] : delete map entry\n"
 	"  disable        : put a server or frontend in maintenance mode\n"
 	"  enable         : re-enable a server or frontend which is in maintenance mode\n"
 	"  shutdown       : kill a session or a frontend (eg:to release listening ports)\n"
@@ -914,6 +924,42 @@
 	return sv;
 }
 
+/* This function is used with map management. It permits to browse each
+ * really allocated descriptors of one map reference. The variable
+ * <appctx->ctx.map.ref> must contain the map reference to browse.
+ * The variable <appctx->ctx.map.desc> contain the descriptor of the
+ * current allocated map descriptor. This variable must be initialized
+ * to NULL.
+ */
+static inline void stats_map_lookup_next(struct stream_interface *si)
+{
+	struct appctx *appctx = __objt_appctx(si->end);
+
+	/* search the next allocated map */
+	while (1) {
+		/* get next descriptor */
+		if (!appctx->ctx.map.desc)
+			appctx->ctx.map.desc = LIST_NEXT(&appctx->ctx.map.ref->maps,
+			                                 struct map_descriptor *, list);
+		else
+			appctx->ctx.map.desc = LIST_NEXT(&appctx->ctx.map.desc->list,
+			                                 struct map_descriptor *, list);
+
+		/* detect end of list */
+		if (&appctx->ctx.map.desc->list == &appctx->ctx.map.ref->maps) {
+			appctx->ctx.map.desc = NULL;
+			return;
+		}
+
+		/* do not lookup this entry */
+		if (!appctx->ctx.map.desc->do_free)
+			continue;
+
+		/* avalaible descriptor */
+		return;
+	}
+}
+
 /* Processes the stats interpreter on the statistics socket. This function is
  * called from an applet running in a stream interface. The function returns 1
  * if the request was understood, otherwise zero. It sets appctx->st0 to a value
@@ -1021,6 +1067,25 @@
 		else if (strcmp(args[1], "table") == 0) {
 			stats_sock_table_request(si, args, STAT_CLI_O_TAB);
 		}
+		else if (strcmp(args[1], "map") == 0) {
+
+			/* no parameter: display all map avalaible */
+			if (!*args[2]) {
+				appctx->st2 = STAT_ST_INIT;
+				appctx->st0 = STAT_CLI_O_MAPS;
+				return 1;
+			}
+
+			/* lookup into the maps */
+			appctx->ctx.map.ref = map_get_reference(args[2]);
+			if (!appctx->ctx.map.ref) {
+				appctx->ctx.cli.msg = "Unknown map reference.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+			appctx->st2 = STAT_ST_INIT;
+			appctx->st0 = STAT_CLI_O_MAP;
+		}
 		else { /* neither "stat" nor "info" nor "sess" nor "errors" nor "table" */
 			return 0;
 		}
@@ -1088,6 +1153,43 @@
 			/* end of processing */
 			return 1;
 		}
+		else if (strcmp(args[1], "map") == 0) {
+			struct map_reference *mref;
+			struct map_descriptor *mdesc;
+			struct map_entry *ent, *nent;
+
+			/* no parameter */
+			if (!*args[2]) {
+				appctx->ctx.cli.msg = "Expect map reference.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* lookup into the maps */
+			mref = map_get_reference(args[2]);
+			if (!mref) {
+				appctx->ctx.cli.msg = "Unknown map reference.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* clear all maps */
+			list_for_each_entry(mdesc, &mref->maps, list)
+				if (mdesc->do_free)
+					pattern_prune_expr(mdesc->pat);
+
+			/* clear map reference */
+			list_for_each_entry_safe(ent, nent, &mref->entries, list) {
+				LIST_DEL(&ent->list);
+				free(ent->key);
+				free(ent->value);
+				free(ent);
+			}
+
+			/* return response */
+			appctx->ctx.cli.msg = "Done.\n";
+			appctx->st0 = STAT_CLI_PRINT;
+		}
 		else {
 			/* unknown "clear" argument */
 			return 0;
@@ -1122,6 +1224,37 @@
 			bi_putstr(si->ib, trash.str);
 			return 1;
 		}
+		else if (strcmp(args[1], "map") == 0) {
+
+			/* no parameter */
+			if (!*args[2] || !*args[3]) {
+				appctx->ctx.cli.msg = "Expect map reference and required key.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* lookup into the maps */
+			appctx->ctx.map.ref = map_get_reference(args[2]);
+			if (!appctx->ctx.map.ref) {
+				appctx->ctx.cli.msg = "Unknown map reference.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* copy input string */
+			appctx->ctx.map.chunk.len = strlen(args[3]);
+			appctx->ctx.map.chunk.size = appctx->ctx.map.chunk.len + 1;
+			appctx->ctx.map.chunk.str = strdup(args[3]);
+			if (!appctx->ctx.map.chunk.str) {
+				appctx->ctx.cli.msg = "Out of memory error.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* prepare response */
+			appctx->st2 = STAT_ST_INIT;
+			appctx->st0 = STAT_CLI_O_MLOOK;
+		}
 		else { /* not "get weight" */
 			return 0;
 		}
@@ -1319,6 +1452,70 @@
 		else if (strcmp(args[1], "table") == 0) {
 			stats_sock_table_request(si, args, STAT_CLI_O_SET);
 		}
+		else if (strcmp(args[1], "map") == 0) {
+			struct pattern *pat_elt;
+			struct pat_idx_elt *idx_elt;
+			char *value;
+
+			/* Expect three parameters: map name, key and new value. */
+			if (!*args[2] || !*args[3] || !*args[4]) {
+				appctx->ctx.cli.msg = "'set map' expect three parameters: map name, key and value.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* Lookup the reference in the maps. */
+			appctx->ctx.map.ref = map_get_reference(args[2]);
+			if (!appctx->ctx.map.ref) {
+				appctx->ctx.cli.msg = "Unknown map reference.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* Lookup the entry in the reference values. */
+			list_for_each_entry(appctx->ctx.map.ent, &appctx->ctx.map.ref->entries, list)
+				if (strcmp(args[3], appctx->ctx.map.ent->key) == 0)
+					break;
+
+			if (&appctx->ctx.map.ent->list == &appctx->ctx.map.ref->entries) {
+				appctx->ctx.cli.msg = "Entry not found.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* Update each reference entries. */
+			list_for_each_entry(appctx->ctx.map.ent, &appctx->ctx.map.ref->entries, list) {
+				if (strcmp(args[3], appctx->ctx.map.ent->key) == 0) {
+					value = strdup(args[4]);
+					if (!value) {
+						appctx->ctx.cli.msg = "Out of memory error.\n";
+						appctx->st0 = STAT_CLI_PRINT;
+						return 1;
+					}
+					free(appctx->ctx.map.ent->value);
+					appctx->ctx.map.ent->value = value;
+				}
+			}
+
+			/* Change the sample. The lookup juste return the first entry, other
+			 * entries are not changed, but are never matched.
+			 */
+			appctx->ctx.map.desc = NULL;
+			for (stats_map_lookup_next(si);
+			     appctx->ctx.map.desc;
+			     stats_map_lookup_next(si)) {
+				pattern_lookup(args[3], appctx->ctx.map.desc->pat, &pat_elt, &idx_elt, NULL);
+				if (pat_elt != NULL)
+					appctx->ctx.map.desc->parse(appctx->ctx.map.ent->value, pat_elt->smp);
+				if (idx_elt != NULL)
+					appctx->ctx.map.desc->parse(appctx->ctx.map.ent->value, idx_elt->smp);
+			}
+
+			/* The set is done, send message. */
+			appctx->ctx.cli.msg = "Done.\n";
+			appctx->st0 = STAT_CLI_PRINT;
+			return 1;
+		}
 		else { /* unknown "set" parameter */
 			return 0;
 		}
@@ -1535,6 +1732,190 @@
 			appctx->st0 = STAT_CLI_PRINT;
 			return 1;
 		}
+	}
+	else if (strcmp(args[0], "del") == 0) {
+		if (strcmp(args[1], "map") == 0) {
+			struct pattern *pat_elt;
+			struct pat_idx_elt *idx_elt;
+			struct map_entry *ent;
+
+			/* Expect two parameters: map name and key. */
+			if (!*args[2] || !*args[3]) {
+				appctx->ctx.cli.msg = "'del map' expect two parameters: map name and key.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* Lookup the reference in the maps. */
+			appctx->ctx.map.ref = map_get_reference(args[2]);
+			if (!appctx->ctx.map.ref) {
+				appctx->ctx.cli.msg = "Unknown map reference.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* Lookup the entry in the reference values.
+			 * If the entry is not found in the reference, return error message.
+			 */
+			list_for_each_entry(appctx->ctx.map.ent, &appctx->ctx.map.ref->entries, list)
+				if (strcmp(args[3], appctx->ctx.map.ent->key) == 0)
+					break;
+
+			if (&appctx->ctx.map.ent->list == &appctx->ctx.map.ref->entries) {
+				appctx->ctx.cli.msg = "Entry not found.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* Delete each enties from reference. */
+			list_for_each_entry_safe(appctx->ctx.map.ent, ent, &appctx->ctx.map.ref->entries, list) {
+				if (strcmp(args[3], appctx->ctx.map.ent->key) == 0) {
+					LIST_DEL(&appctx->ctx.map.ent->list);
+					free(appctx->ctx.map.ent->key);
+					free(appctx->ctx.map.ent->value);
+					free(appctx->ctx.map.ent);
+				}
+			}
+
+			/* Delete all matching entries for each map descritor. */
+			appctx->ctx.map.desc = NULL;
+			stats_map_lookup_next(si);
+			while (appctx->ctx.map.desc) {
+				while (pattern_lookup(args[3], appctx->ctx.map.desc->pat, &pat_elt, &idx_elt, NULL)) {
+					if (pat_elt != NULL) {
+						LIST_DEL(&pat_elt->list);
+						pattern_free(pat_elt);
+					}
+					if (idx_elt != NULL) {
+						ebmb_delete(&idx_elt->node);
+						free(idx_elt);
+					}
+				}
+				stats_map_lookup_next(si);
+			}
+
+			/* The deletion is done, send message. */
+			appctx->ctx.cli.msg = "Done.\n";
+			appctx->st0 = STAT_CLI_PRINT;
+			return 1;
+		}
+		else { /* unknown "del" parameter */
+			appctx->ctx.cli.msg = "'del' only supports 'map'.\n";
+			appctx->st0 = STAT_CLI_PRINT;
+			return 1;
+		}
+	}
+	else if (strcmp(args[0], "add") == 0) {
+		if (strcmp(args[1], "map") == 0) {
+			const char *params[2];
+			struct pattern *pat;
+			struct map_entry *ent;
+			struct sample_storage *smp;
+
+			/* Expect three parameters: map name, key and new value. */
+			if (!*args[2] || !*args[3] || !*args[4]) {
+				appctx->ctx.cli.msg = "'add map' expect three parameters: map name, key and value.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			params[0] = args[3];
+			params[1] = "";
+
+			/* Lookup the reference in the maps. */
+			appctx->ctx.map.ref = map_get_reference(args[2]);
+			if (!appctx->ctx.map.ref) {
+				appctx->ctx.cli.msg = "Unknown map reference.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+
+			/* Prepare and link the new map_entry element. If out of memory
+			 * error the action is cancelled and the descriptor are left
+			 * coherents.
+			 */
+			ent = malloc(sizeof(*ent));
+			if (!ent) {
+				appctx->ctx.cli.msg = "Out of memory error.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+			ent->key = strdup(args[3]);
+			if (!ent->key) {
+				free(ent);
+				appctx->ctx.cli.msg = "Out of memory error.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+			ent->value = strdup(args[4]);
+			if (!ent->value) {
+				free(ent->key);
+				free(ent);
+				appctx->ctx.cli.msg = "Out of memory error.\n";
+				appctx->st0 = STAT_CLI_PRINT;
+				return 1;
+			}
+			LIST_ADDQ(&appctx->ctx.map.ref->entries, &ent->list);
+
+			/* Browse each map descritor and try to insert this new value. */
+			appctx->ctx.map.desc = NULL;
+			for (stats_map_lookup_next(si);
+			     appctx->ctx.map.desc;
+			     stats_map_lookup_next(si)) {
+
+				/* Create new sample. Return out of memory error
+				 * if the memory cannot be allocated. The 'add' process
+				 * is aborted, but the already inserted entries are not
+				 * deleted.
+				 */
+				smp = calloc(1, sizeof(*smp));
+				if (!smp) {
+					appctx->ctx.cli.msg = "Out of memory error. The value is not added in all maps.\n";
+					appctx->st0 = STAT_CLI_PRINT;
+					return 1;
+				}
+
+				/* Create sample. If this function fails, the insertion
+				 * is canceled for this 'descriptor', but continue, for
+				 * the other descriptors.
+				 */
+				if (!appctx->ctx.map.desc->parse(ent->value, smp)) {
+					free(smp);
+					continue;
+				}
+
+				/* If the value can be indexed, get the first pattern. If
+				 * the return entry is not the indexed entry, new 'pattern' is
+				 * created by the function pattern_register(). If the 'pattern'
+				 * is NULL, new entry is created. This is ugly because the
+				 * following code interfers with the own code of the function
+				 * pattern_register().
+				 */
+				if (appctx->ctx.map.desc->pat->match == pat_match_str ||
+				    appctx->ctx.map.desc->pat->match == pat_match_ip) {
+					pat = LIST_NEXT(&appctx->ctx.map.desc->pat->patterns, struct pattern *, list);
+					if (&pat->list == &appctx->ctx.map.desc->pat->patterns)
+						pat = NULL;
+				}
+				else
+					pat = NULL;
+
+				if (!pattern_register(appctx->ctx.map.desc->pat, params, smp, &pat, 0, NULL)) {
+					free(smp);
+					continue;
+				}
+			}
+
+			/* The add is done, send message. */
+			appctx->ctx.cli.msg = "Done.\n";
+			appctx->st0 = STAT_CLI_PRINT;
+			return 1;
+		}
+		else { /* unknown "del" parameter */
+			appctx->ctx.cli.msg = "'add' only supports 'map'.\n";
+			appctx->st0 = STAT_CLI_PRINT;
+			return 1;
+		}
 	}
 	else { /* not "show" nor "clear" nor "get" nor "set" nor "enable" nor "disable" */
 		return 0;
@@ -1676,6 +2057,17 @@
 				if (stats_table_request(si, appctx->st0))
 					appctx->st0 = STAT_CLI_PROMPT;
 				break;
+			case STAT_CLI_O_MAPS:
+				if (stats_maps_list(si))
+					appctx->st0 = STAT_CLI_PROMPT;
+				break;
+			case STAT_CLI_O_MAP:
+				if (stats_map_list(si))
+					appctx->st0 = STAT_CLI_PROMPT;
+				break;
+			case STAT_CLI_O_MLOOK:
+				if (stats_map_lookup(si))
+					appctx->st0 = STAT_CLI_PROMPT;
 			default: /* abnormal state */
 				appctx->st0 = STAT_CLI_PROMPT;
 				break;
@@ -4222,6 +4614,252 @@
 	return 1;
 }
 
+static int stats_maps_list(struct stream_interface *si)
+{
+	struct appctx *appctx = __objt_appctx(si->end);
+
+	switch (appctx->st2) {
+	case STAT_ST_INIT:
+		/* Init to the first entry. The list cannot be change */
+		appctx->ctx.map.ref = LIST_NEXT(&maps, struct map_reference *, list);
+		appctx->st2 = STAT_ST_LIST;
+		/* fall through */
+
+	case STAT_ST_LIST:
+		while (appctx->ctx.map.ref) {
+
+			chunk_reset(&trash);
+
+			/* build messages */
+			chunk_appendf(&trash, "%s\n", appctx->ctx.map.ref->reference);
+
+			if (bi_putchk(si->ib, &trash) == -1) {
+				/* let's try again later from this session. We add ourselves into
+				 * this session's users so that it can remove us upon termination.
+				 */
+				return 0;
+			}
+
+			/* get next list entry and check the end of the list */
+			appctx->ctx.map.ref = LIST_NEXT(&appctx->ctx.map.ref->list,
+			                                 struct map_reference *, list);
+			if (&appctx->ctx.map.ref->list == &maps)
+				break;
+		}
+
+		appctx->st2 = STAT_ST_FIN;
+		/* fall through */
+
+	default:
+		appctx->st2 = STAT_ST_FIN;
+		return 1;
+	}
+}
+
+static const char *smp_to_type[SMP_TYPES] = {
+	[SMP_T_BOOL] = "bool",
+	[SMP_T_UINT] = "uint",
+	[SMP_T_SINT] = "sint",
+	[SMP_T_ADDR] = "addr",
+	[SMP_T_IPV4] = "ipv4",
+	[SMP_T_IPV6] = "ipv6",
+	[SMP_T_STR]  = "str",
+	[SMP_T_BIN]  = "bin",
+	[SMP_T_CSTR] = "cstr",
+	[SMP_T_CBIN] = "cbin",
+};
+
+static int stats_map_lookup(struct stream_interface *si)
+{
+	struct appctx *appctx = __objt_appctx(si->end);
+	struct sample_storage *smp;
+	struct sample sample;
+	struct pattern *pat;
+	struct pat_idx_elt *elt;
+	enum pat_match_res res;
+	struct sockaddr_in addr;
+	char s_addr[INET_ADDRSTRLEN];
+
+	switch (appctx->st2) {
+	case STAT_ST_INIT:
+		appctx->ctx.map.desc = NULL;
+		stats_map_lookup_next(si);
+		appctx->st2 = STAT_ST_LIST;
+		/* fall through */
+
+	case STAT_ST_LIST:
+		/* for each lookup type */
+		while (appctx->ctx.map.desc) {
+			/* initialise chunk to build new message */
+			chunk_reset(&trash);
+
+			/* execute pattern matching */
+			sample.type = SMP_T_CSTR;
+			sample.data.str.len = appctx->ctx.map.chunk.len;
+			sample.data.str.str = appctx->ctx.map.chunk.str;
+			pat = NULL;
+			elt = NULL;
+			res = pattern_exec_match(appctx->ctx.map.desc->pat, &sample, &smp, &pat, &elt);
+
+			/* build return message: set type of match */
+			/**/ if (appctx->ctx.map.desc->pat->match == NULL)
+				chunk_appendf(&trash, "found, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_nothing)
+				chunk_appendf(&trash, "bool, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_int)
+				chunk_appendf(&trash, "int, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_ip)
+				chunk_appendf(&trash, "ip, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_bin)
+				chunk_appendf(&trash, "bin, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_len)
+				chunk_appendf(&trash, "len, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_str)
+				chunk_appendf(&trash, "str, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_beg)
+				chunk_appendf(&trash, "beg, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_sub)
+				chunk_appendf(&trash, "sub, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_dir)
+				chunk_appendf(&trash, "dir, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_dom)
+				chunk_appendf(&trash, "dom, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_end)
+				chunk_appendf(&trash, "end, ");
+			else if (appctx->ctx.map.desc->pat->match == pat_match_reg)
+				chunk_appendf(&trash, "reg, ");
+			else /* The never appens case */
+				chunk_appendf(&trash, "unknown(%p), ", appctx->ctx.map.desc->pat->match);
+
+			/* Display no match, and set default value */
+			if (res == PAT_NOMATCH) {
+				chunk_appendf(&trash, "no-match, ");
+				smp = appctx->ctx.map.desc->def;
+			}
+
+			/* Display match and match info */
+			else {
+				/* display match */
+				chunk_appendf(&trash, "match, ");
+
+				/* display search mode */
+				if (elt)
+					chunk_appendf(&trash, "tree, ");
+				else
+					chunk_appendf(&trash, "list, ");
+
+				/* display search options */
+				if (pat) {
+					/* case sensitive */
+					if (pat->flags & PAT_F_IGNORE_CASE)
+						chunk_appendf(&trash, "case-insensitive, ");
+					else
+						chunk_appendf(&trash, "case-sensitive, ");
+
+					/* display source */
+					if (pat->flags & PAT_F_FROM_FILE)
+						chunk_appendf(&trash, "from-file, ");
+				}
+
+				/* display match expresion */
+				if (elt) {
+					if (appctx->ctx.map.desc->pat->match == pat_match_str) {
+						chunk_appendf(&trash, "match=\"%s\", ", elt->node.key);
+					}
+					/* only IPv4 */
+					else if (appctx->ctx.map.desc->pat->match == pat_match_ip) {
+						/* convert ip */
+						memcpy(&addr.sin_addr, elt->node.key, 4);
+						addr.sin_family = AF_INET;
+						if (addr_to_str((struct sockaddr_storage *)&addr, s_addr, INET_ADDRSTRLEN))
+							chunk_appendf(&trash, "match=\"%s/%d\", ", s_addr, elt->node.node.pfx);
+					}
+				}
+			}
+
+			/* display return value */
+			if (!smp) {
+				chunk_appendf(&trash, "return=nothing\n");
+			}
+			else {
+				memcpy(&sample.data, &smp->data, sizeof(sample.data));
+				sample.type = smp->type;
+				if (sample_casts[sample.type][SMP_T_CSTR] &&
+				    sample_casts[sample.type][SMP_T_CSTR](&sample))
+					chunk_appendf(&trash, "return=\"%s\", type=\"%s\"\n",
+					              sample.data.str.str, smp_to_type[smp->type]);
+				else
+					chunk_appendf(&trash, "return=cannot-display, type=\"%s\"\n",
+					              smp_to_type[smp->type]);
+			}
+
+			/* display response */
+			if (bi_putchk(si->ib, &trash) == -1) {
+				/* let's try again later from this session. We add ourselves into
+				 * this session's users so that it can remove us upon termination.
+				 */
+				return 0;
+			}
+
+			/* get next entry */
+			stats_map_lookup_next(si);
+		}
+
+		appctx->st2 = STAT_ST_FIN;
+		/* fall through */
+
+	default:
+		appctx->st2 = STAT_ST_FIN;
+		free(appctx->ctx.map.chunk.str);
+		return 1;
+	}
+}
+
+static int stats_map_list(struct stream_interface *si)
+{
+	struct appctx *appctx = __objt_appctx(si->end);
+
+	switch (appctx->st2) {
+
+	case STAT_ST_INIT:
+		/* Init to the first entry. The list cannot be change */
+		appctx->ctx.map.ent = LIST_NEXT(&appctx->ctx.map.ref->entries,
+		                                struct map_entry *, list);
+		if (&appctx->ctx.map.ent->list == &appctx->ctx.map.ref->entries)
+			appctx->ctx.map.ent = NULL;
+		appctx->st2 = STAT_ST_LIST;
+		/* fall through */
+
+	case STAT_ST_LIST:
+		while (appctx->ctx.map.ent) {
+			chunk_reset(&trash);
+
+			/* build messages */
+			chunk_appendf(&trash, "%s %s\n", appctx->ctx.map.ent->key, appctx->ctx.map.ent->value);
+
+			if (bi_putchk(si->ib, &trash) == -1) {
+				/* let's try again later from this session. We add ourselves into
+				 * this session's users so that it can remove us upon termination.
+				 */
+				return 0;
+			}
+
+			/* get next list entry and check the end of the list */
+			appctx->ctx.map.ent = LIST_NEXT(&appctx->ctx.map.ent->list,
+			                                struct map_entry *, list);
+			if (&appctx->ctx.map.ent->list == &appctx->ctx.map.ref->entries)
+				break;
+		}
+
+		appctx->st2 = STAT_ST_FIN;
+		/* fall through */
+
+	default:
+		appctx->st2 = STAT_ST_FIN;
+		return 1;
+	}
+}
+
 /* This function dumps all sessions' states onto the stream interface's
  * read buffer. It returns 0 if the output buffer is full and it needs
  * to be called again, otherwise non-zero. It is designed to be called