summaryrefslogtreecommitdiffhomepage
path: root/src
diff options
context:
space:
mode:
authorMax Romanov <max.romanov@nginx.com>2019-08-16 00:56:38 +0300
committerMax Romanov <max.romanov@nginx.com>2019-08-16 00:56:38 +0300
commit29911538ea91705fcdcbcf0e271cfbc5c8ed674b (patch)
treea3a63b97f073cfe67e1b31d2f8c87fdfad4e3520 /src
parentcaea9d3c07543fecf9035ff2b406c190b714989e (diff)
downloadunit-29911538ea91705fcdcbcf0e271cfbc5c8ed674b.tar.gz
unit-29911538ea91705fcdcbcf0e271cfbc5c8ed674b.tar.bz2
Improving response header fields processing.
Fields are filtered one by one before being added to fields list. This avoids adding and then skipping connection-specific fields.
Diffstat (limited to '')
-rw-r--r--src/nxt_http_parse.c23
-rw-r--r--src/nxt_http_parse.h24
-rw-r--r--src/nxt_router.c26
3 files changed, 46 insertions, 27 deletions
diff --git a/src/nxt_http_parse.c b/src/nxt_http_parse.c
index 05df245e..8b4bf47c 100644
--- a/src/nxt_http_parse.c
+++ b/src/nxt_http_parse.c
@@ -1110,7 +1110,7 @@ done:
}
-static const nxt_lvlhsh_proto_t nxt_http_fields_hash_proto nxt_aligned(64) = {
+const nxt_lvlhsh_proto_t nxt_http_fields_hash_proto nxt_aligned(64) = {
NXT_LVLHSH_BUCKET_SIZE(64),
{ NXT_HTTP_FIELD_LVLHSH_SHIFT, 0, 0, 0, 0, 0, 0, 0 },
nxt_http_field_hash_test,
@@ -1240,27 +1240,12 @@ nxt_http_fields_hash_collisions(nxt_lvlhsh_t *hash, nxt_mp_t *mp,
nxt_int_t
nxt_http_fields_process(nxt_list_t *fields, nxt_lvlhsh_t *hash, void *ctx)
{
- nxt_int_t ret;
- nxt_http_field_t *field;
- nxt_lvlhsh_query_t lhq;
- nxt_http_field_proc_t *proc;
-
- lhq.proto = &nxt_http_fields_hash_proto;
+ nxt_int_t ret;
+ nxt_http_field_t *field;
nxt_list_each(field, fields) {
- lhq.key_hash = field->hash;
- lhq.key.length = field->name_length;
- lhq.key.start = field->name;
-
- if (nxt_lvlhsh_find(hash, &lhq) != NXT_OK) {
- continue;
- }
-
- proc = lhq.value;
-
- ret = proc->handler(ctx, field, proc->data);
-
+ ret = nxt_http_field_process(field, hash, ctx);
if (nxt_slow_path(ret != NXT_OK)) {
return ret;
}
diff --git a/src/nxt_http_parse.h b/src/nxt_http_parse.h
index 6c629936..c5b11bf3 100644
--- a/src/nxt_http_parse.h
+++ b/src/nxt_http_parse.h
@@ -113,4 +113,28 @@ nxt_int_t nxt_http_fields_process(nxt_list_t *fields, nxt_lvlhsh_t *hash,
void *ctx);
+const nxt_lvlhsh_proto_t nxt_http_fields_hash_proto;
+
+nxt_inline nxt_int_t
+nxt_http_field_process(nxt_http_field_t *field, nxt_lvlhsh_t *hash, void *ctx)
+{
+ nxt_lvlhsh_query_t lhq;
+ nxt_http_field_proc_t *proc;
+
+ lhq.proto = &nxt_http_fields_hash_proto;
+
+ lhq.key_hash = field->hash;
+ lhq.key.length = field->name_length;
+ lhq.key.start = field->name;
+
+ if (nxt_lvlhsh_find(hash, &lhq) != NXT_OK) {
+ return NXT_OK;
+ }
+
+ proc = lhq.value;
+
+ return proc->handler(ctx, field, proc->data);
+}
+
+
#endif /* _NXT_HTTP_PARSER_H_INCLUDED_ */
diff --git a/src/nxt_router.c b/src/nxt_router.c
index f09779bc..509f487a 100644
--- a/src/nxt_router.c
+++ b/src/nxt_router.c
@@ -3483,7 +3483,13 @@ nxt_router_response_ready_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg,
goto fail;
}
+ field = NULL;
+
for (f = resp->fields; f < resp->fields + resp->fields_count; f++) {
+ if (f->skip) {
+ continue;
+ }
+
field = nxt_list_add(r->resp.fields);
if (nxt_slow_path(field == NULL)) {
@@ -3491,26 +3497,30 @@ nxt_router_response_ready_handler(nxt_task_t *task, nxt_port_recv_msg_t *msg,
}
field->hash = f->hash;
- field->skip = f->skip;
+ field->skip = 0;
field->name_length = f->name_length;
field->value_length = f->value_length;
field->name = nxt_unit_sptr_get(&f->name);
field->value = nxt_unit_sptr_get(&f->value);
- nxt_debug(task, "header: %*s: %*s",
+ ret = nxt_http_field_process(field, &nxt_response_fields_hash, r);
+ if (nxt_slow_path(ret != NXT_OK)) {
+ goto fail;
+ }
+
+ nxt_debug(task, "header%s: %*s: %*s",
+ (field->skip ? " skipped" : ""),
(size_t) field->name_length, field->name,
(size_t) field->value_length, field->value);
+
+ if (field->skip) {
+ r->resp.fields->last->nelts--;
+ }
}
r->status = resp->status;
- ret = nxt_http_fields_process(r->resp.fields,
- &nxt_response_fields_hash, r);
- if (nxt_slow_path(ret != NXT_OK)) {
- goto fail;
- }
-
if (resp->piggyback_content_length != 0) {
b->mem.pos = nxt_unit_sptr_get(&resp->piggyback_content);
b->mem.free = b->mem.pos + resp->piggyback_content_length;