diff options
author | Max Romanov <max.romanov@nginx.com> | 2019-08-16 00:56:38 +0300 |
---|---|---|
committer | Max Romanov <max.romanov@nginx.com> | 2019-08-16 00:56:38 +0300 |
commit | 29911538ea91705fcdcbcf0e271cfbc5c8ed674b (patch) | |
tree | a3a63b97f073cfe67e1b31d2f8c87fdfad4e3520 | |
parent | caea9d3c07543fecf9035ff2b406c190b714989e (diff) | |
download | unit-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.
-rw-r--r-- | src/nxt_http_parse.c | 23 | ||||
-rw-r--r-- | src/nxt_http_parse.h | 24 | ||||
-rw-r--r-- | src/nxt_router.c | 26 |
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; |