MEDIUM: http: add new "capture" action for http-request

This is only possible in frontends of course, but it will finally
make it possible to capture arbitrary http parts, including URL
parameters or parts of the message body.

It's worth noting that an ugly (char **) cast had to be done to
call sample_fetch_string() which is caused by a 5- or 6- levels
of inheritance of this type in the API. Here it's harmless since
the function uses it as a const, but this API madness must be
fixed, starting with the one or two rare functions that modify
the args and inflict this on each and every keyword parser.
(cherry picked from commit 484a4f38460593919a1c1d9a047a043198d69f45)
diff --git a/doc/configuration.txt b/doc/configuration.txt
index 9a6202d..98242d2 100644
--- a/doc/configuration.txt
+++ b/doc/configuration.txt
@@ -3243,6 +3243,7 @@
 
 http-request { allow | deny | tarpit | auth [realm <realm>] | redirect <rule> |
               add-header <name> <fmt> | set-header <name> <fmt> |
+              capture <sample> len <length> |
               del-header <name> | set-nice <nice> | set-log-level <level> |
               replace-header <name> <match-regex> <replace-fmt> |
               replace-value <name> <match-regex> <replace-fmt> |
@@ -3461,6 +3462,17 @@
       with large lists! It is the equivalent of the "set map" command from the
       stats socket, but can be triggered by an HTTP request.
 
+    - capture <sample> len <length> :
+      captures sample expression <sample> from the request buffer, and converts
+      it to a string of at most <len> characters. The resulting string is
+      stored into the next request "capture" slot, so it will possibly appear
+      next to some captured HTTP headers. It will then automatically appear in
+      the logs, and it will be possible to extract it using sample fetch rules
+      to feed it into headers or anything. The length should be limited given
+      that this size will be allocated for each capture during the whole
+      session life. Please check section 7.3 (Fetching samples) and "capture
+      request header" for more information.
+
     - { track-sc0 | track-sc1 | track-sc2 } <key> [table <table>] :
       enables tracking of sticky counters from current request. These rules
       do not stop evaluation and do not change default action. Three sets of
@@ -7898,7 +7910,7 @@
   accept the incoming connection. There is no specific limit to the number of
   rules which may be inserted.
 
-  Five types of actions are supported :
+  Four types of actions are supported :
     - accept :
         accepts the connection if the condition is true (when used with "if")
         or false (when used with "unless"). The first such rule executed ends
@@ -7936,8 +7948,8 @@
         logs, and it will be possible to extract it using sample fetch rules to
         feed it into headers or anything. The length should be limited given
         that this size will be allocated for each capture during the whole
-        session life. Since it applies to Please check section 7.3 (Fetching
-        samples) and "capture request header" for more information.
+        session life. Please check section 7.3 (Fetching samples) and "capture
+        request header" for more information.
 
     - { track-sc0 | track-sc1 | track-sc2 } <key> [table <table>] :
         enables tracking of sticky counters from current connection. These
diff --git a/src/proto_http.c b/src/proto_http.c
index b89d018..275f329 100644
--- a/src/proto_http.c
+++ b/src/proto_http.c
@@ -12120,6 +12120,126 @@
 	return 0;
 }
 
+/* This function executes the "capture" action. It executes a fetch expression,
+ * turns the result into a string and puts it in a capture slot. It always
+ * returns 1. If an error occurs the action is cancelled, but the rule
+ * processing continues.
+ */
+int http_action_req_capture(struct http_req_rule *rule, struct proxy *px, struct stream *s)
+{
+	struct session *sess = s->sess;
+	struct sample *key;
+	struct sample_expr *expr = rule->arg.act.p[0];
+	struct cap_hdr *h = rule->arg.act.p[1];
+	char **cap = s->req_cap;
+	int len;
+
+	key = sample_fetch_string(s->be, sess, s, SMP_OPT_DIR_REQ|SMP_OPT_FINAL, expr);
+	if (!key)
+		return 1;
+
+	if (cap[h->index] == NULL)
+		cap[h->index] = pool_alloc2(h->pool);
+
+	if (cap[h->index] == NULL) /* no more capture memory */
+		return 1;
+
+	len = key->data.str.len;
+	if (len > h->len)
+		len = h->len;
+
+	memcpy(cap[h->index], key->data.str.str, len);
+	cap[h->index][len] = 0;
+	return 1;
+}
+
+/* parse an "http-request capture" action. It takes a single argument which is
+ * a sample fetch expression. It stores the expression into arg->act.p[0] and
+ * the allocated hdr_cap struct into arg->act.p[1].
+ * It returns 0 on success, < 0 on error.
+ */
+int parse_http_req_capture(const char **args, int *orig_arg, struct proxy *px, struct http_req_rule *rule, char **err)
+{
+	struct sample_expr *expr;
+	struct cap_hdr *hdr;
+	int cur_arg;
+	int len;
+
+	for (cur_arg = *orig_arg; cur_arg < *orig_arg + 3 && *args[cur_arg]; cur_arg++)
+		if (strcmp(args[cur_arg], "if") == 0 ||
+		    strcmp(args[cur_arg], "unless") == 0)
+			break;
+
+	if (cur_arg < *orig_arg + 3) {
+		memprintf(err, "expects <expression> 'len' <length> ");
+		return -1;
+	}
+
+	if (!(px->cap & PR_CAP_FE)) {
+		memprintf(err, "proxy '%s' has no frontend capability", px->id);
+		return -1;
+	}
+
+	LIST_INIT((struct list *)&rule->arg.act.p[0]);
+	proxy->conf.args.ctx = ARGC_CAP;
+
+	cur_arg = *orig_arg;
+	expr = sample_parse_expr((char **)args, &cur_arg, px->conf.args.file, px->conf.args.line, err, &px->conf.args);
+	if (!expr)
+		return -1;
+
+	if (!(expr->fetch->val & SMP_VAL_FE_HRQ_HDR)) {
+		memprintf(err,
+			  "fetch method '%s' extracts information from '%s', none of which is available here",
+			  args[cur_arg-1], sample_src_names(expr->fetch->use));
+		free(expr);
+		return -1;
+	}
+
+	if (strcmp(args[cur_arg], "len") == 0) {
+		cur_arg++;
+		if (!args[cur_arg]) {
+			memprintf(err, "missing length value");
+			free(expr);
+			return -1;
+		}
+		/* we copy the table name for now, it will be resolved later */
+		len = atoi(args[cur_arg]);
+		if (len <= 0) {
+			memprintf(err, "length must be > 0");
+			free(expr);
+			return -1;
+		}
+		cur_arg++;
+	}
+
+	if (!len) {
+		memprintf(err, "a positive 'len' argument is mandatory");
+		free(expr);
+		return -1;
+	}
+
+
+	hdr = calloc(sizeof(struct cap_hdr), 1);
+	hdr->next = px->req_cap;
+	hdr->name = NULL; /* not a header capture */
+	hdr->namelen = 0;
+	hdr->len = len;
+	hdr->pool = create_pool("caphdr", hdr->len + 1, MEM_F_SHARED);
+	hdr->index = px->nb_req_cap++;
+
+	px->req_cap = hdr;
+	px->to_log |= LW_REQHDR;
+
+	rule->action       = HTTP_REQ_ACT_CUSTOM_CONT;
+	rule->action_ptr   = http_action_req_capture;
+	rule->arg.act.p[0] = expr;
+	rule->arg.act.p[1] = hdr;
+
+	*orig_arg = cur_arg;
+	return 0;
+}
+
 /*
  * Return the struct http_req_action_kw associated to a keyword.
  */
@@ -12381,6 +12501,7 @@
 struct http_req_action_kw_list http_req_actions = {
 	.scope = "http",
 	.kw = {
+		{ "capture",    parse_http_req_capture },
 		{ "set-method", parse_set_req_line },
 		{ "set-path",   parse_set_req_line },
 		{ "set-query",  parse_set_req_line },