[MINOR] http: add http_remove_header2() to remove a header value.
Calling this function after http_find_header2() automatically deletes
the current value of the header, and removes the header itself if the
value is the only one. The context is automatically adjusted for a
next call to http_find_header2() to return the next header. No other
change nor test should be made on the transient context though.
diff --git a/include/proto/proto_http.h b/include/proto/proto_http.h
index f4a57a5..cfaf985 100644
--- a/include/proto/proto_http.h
+++ b/include/proto/proto_http.h
@@ -84,7 +84,7 @@
int stats_check_uri_auth(struct session *t, struct proxy *backend);
void init_proto_http();
int http_find_header2(const char *name, int len,
- const char *sol, struct hdr_idx *idx,
+ char *sol, struct hdr_idx *idx,
struct hdr_ctx *ctx);
void http_sess_log(struct session *s);
void perform_http_redirect(struct session *s, struct stream_interface *si);
diff --git a/include/types/proto_http.h b/include/types/proto_http.h
index 89c8f88..f068d7f 100644
--- a/include/types/proto_http.h
+++ b/include/types/proto_http.h
@@ -300,13 +300,17 @@
};
/* This structure is used by http_find_header() to return values of headers.
- * The header starts at <line>, the value at <line>+<val> for <vlen> bytes.
+ * The header starts at <line>, the value at <line>+<val> for <vlen> bytes, and
+ * sets <line>+<del> to point to the last delimitor (colon or comma) before
+ * this value. <prev> points to the index of the header whose next is this one.
*/
struct hdr_ctx {
- const char *line;
+ char *line;
int idx;
int val; /* relative to line */
int vlen; /* relative to line+val */
+ int del; /* relative to line */
+ int prev; /* index of previous header */
};
#endif /* _TYPES_PROTO_HTTP_H */
diff --git a/src/proto_http.c b/src/proto_http.c
index 980ef39..b9ef020 100644
--- a/src/proto_http.c
+++ b/src/proto_http.c
@@ -452,11 +452,11 @@
return val - hdr;
}
-/* Find the end of the header value contained between <s> and <e>.
- * See RFC2616, par 2.2 for more information. Note that it requires
- * a valid header to return a valid result.
+/* Find the end of the header value contained between <s> and <e>. See RFC2616,
+ * par 2.2 for more information. Note that it requires a valid header to return
+ * a valid result. This works for headers defined as comma-separated lists.
*/
-const char *find_hdr_value_end(const char *s, const char *e)
+char *find_hdr_value_end(char *s, const char *e)
{
int quoted, qdpair;
@@ -476,29 +476,35 @@
* structure holds everything necessary to use the header and find next
* occurrence. If its <idx> member is 0, the header is searched from the
* beginning. Otherwise, the next occurrence is returned. The function returns
- * 1 when it finds a value, and 0 when there is no more.
+ * 1 when it finds a value, and 0 when there is no more. It is designed to work
+ * with headers defined as comma-separated lists. As a special case, if ctx->val
+ * is NULL when searching for a new values of a header, the current header is
+ * rescanned. This allows rescanning after a header deletion.
*/
int http_find_header2(const char *name, int len,
- const char *sol, struct hdr_idx *idx,
+ char *sol, struct hdr_idx *idx,
struct hdr_ctx *ctx)
{
- const char *eol, *sov;
- int cur_idx;
+ char *eol, *sov;
+ int cur_idx, old_idx;
- if (ctx->idx) {
+ cur_idx = ctx->idx;
+ if (cur_idx) {
/* We have previously returned a value, let's search
* another one on the same line.
*/
- cur_idx = ctx->idx;
sol = ctx->line;
- sov = sol + ctx->val + ctx->vlen;
+ ctx->del = ctx->val + ctx->vlen;
+ sov = sol + ctx->del;
eol = sol + idx->v[cur_idx].len;
if (sov >= eol)
/* no more values in this header */
goto next_hdr;
- /* values remaining for this header, skip the comma */
+ /* values remaining for this header, skip the comma but save it
+ * for later use (eg: for header deletion).
+ */
sov++;
while (sov < eol && http_is_lws[(unsigned char)*sov])
sov++;
@@ -508,8 +514,8 @@
/* first request for this header */
sol += hdr_idx_first_pos(idx);
+ old_idx = 0;
cur_idx = hdr_idx_first_idx(idx);
-
while (cur_idx) {
eol = sol + idx->v[cur_idx].len;
@@ -524,12 +530,14 @@
if ((len < eol - sol) &&
(sol[len] == ':') &&
(strncasecmp(sol, name, len) == 0)) {
-
+ ctx->del = len;
sov = sol + len + 1;
while (sov < eol && http_is_lws[(unsigned char)*sov])
sov++;
- return_hdr:
+
ctx->line = sol;
+ ctx->prev = old_idx;
+ return_hdr:
ctx->idx = cur_idx;
ctx->val = sov - sol;
@@ -539,18 +547,69 @@
}
next_hdr:
sol = eol + idx->v[cur_idx].cr + 1;
+ old_idx = cur_idx;
cur_idx = idx->v[cur_idx].next;
}
return 0;
}
int http_find_header(const char *name,
- const char *sol, struct hdr_idx *idx,
+ char *sol, struct hdr_idx *idx,
struct hdr_ctx *ctx)
{
return http_find_header2(name, strlen(name), sol, idx, ctx);
}
+/* Remove one value of a header. This only works on a <ctx> returned by one of
+ * the http_find_header functions. The value is removed, as well as surrounding
+ * commas if any. If the removed value was alone, the whole header is removed.
+ * The ctx is always updated accordingly, as well as buffer <buf> and HTTP
+ * message <msg>. The new index is returned. If it is zero, it means there is
+ * no more header, so any processing may stop. The ctx is always left in a form
+ * that can be handled by http_find_header2() to find next occurrence.
+ */
+int http_remove_header2(struct http_msg *msg, struct buffer *buf,
+ struct hdr_idx *idx, struct hdr_ctx *ctx)
+{
+ int cur_idx = ctx->idx;
+ char *sol = ctx->line;
+ struct hdr_idx_elem *hdr;
+ int delta, skip_comma;
+
+ if (!cur_idx)
+ return 0;
+
+ hdr = &idx->v[cur_idx];
+ if (sol[ctx->del] == ':' && ctx->val + ctx->vlen == hdr->len) {
+ /* This was the only value of the header, we must now remove it entirely. */
+ delta = buffer_replace2(buf, sol, sol + hdr->len + hdr->cr + 1, NULL, 0);
+ http_msg_move_end(msg, delta);
+ idx->used--;
+ hdr->len = 0; /* unused entry */
+ idx->v[ctx->prev].next = idx->v[ctx->idx].next;
+ ctx->idx = ctx->prev; /* walk back to the end of previous header */
+ ctx->line -= idx->v[ctx->idx].len + idx->v[cur_idx].cr + 1;
+ ctx->val = idx->v[ctx->idx].len; /* point to end of previous header */
+ ctx->vlen = 0;
+ return ctx->idx;
+ }
+
+ /* This was not the only value of this header. We have to remove between
+ * ctx->del+1 and ctx->val+ctx->vlen+1 included. If it is the last entry
+ * of the list, we remove the last separator.
+ */
+
+ skip_comma = (ctx->val + ctx->vlen == hdr->len) ? 0 : 1;
+ delta = buffer_replace2(buf, sol + ctx->del + skip_comma,
+ sol + ctx->val + ctx->vlen + skip_comma,
+ NULL, 0);
+ hdr->len += delta;
+ http_msg_move_end(msg, delta);
+ ctx->val = ctx->del;
+ ctx->vlen = 0;
+ return ctx->idx;
+}
+
/* This function handles a server error at the stream interface level. The
* stream interface is assumed to be already in a closed state. An optional
* message is copied into the input buffer, and an HTTP status code stored.