Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 1 | #!/usr/bin/perl |
| 2 | |
| 3 | # (C) Sergey Kandaurov |
| 4 | # (C) Nginx, Inc. |
| 5 | |
| 6 | # Tests for HTTP/2 protocol with headers. |
| 7 | # various HEADERS compression/encoding, see hpack() for mode details. |
| 8 | |
| 9 | ############################################################################### |
| 10 | |
| 11 | use warnings; |
| 12 | use strict; |
| 13 | |
| 14 | use Test::More; |
| 15 | |
Sergey Kandaurov | 62a669e | 2016-10-16 13:01:36 +0300 | [diff] [blame] | 16 | use Config; |
| 17 | |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 18 | BEGIN { use FindBin; chdir($FindBin::Bin); } |
| 19 | |
| 20 | use lib 'lib'; |
| 21 | use Test::Nginx; |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 22 | use Test::Nginx::HTTP2; |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 23 | |
| 24 | ############################################################################### |
| 25 | |
| 26 | select STDERR; $| = 1; |
| 27 | select STDOUT; $| = 1; |
| 28 | |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 29 | my $t = Test::Nginx->new()->has(qw/http http_v2 proxy rewrite/)->plan(93) |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 30 | ->write_file_expand('nginx.conf', <<'EOF'); |
| 31 | |
| 32 | %%TEST_GLOBALS%% |
| 33 | |
| 34 | daemon off; |
| 35 | |
| 36 | events { |
| 37 | } |
| 38 | |
| 39 | http { |
| 40 | %%TEST_GLOBALS_HTTP%% |
| 41 | |
| 42 | server { |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 43 | listen 127.0.0.1:8080 http2; |
| 44 | listen 127.0.0.1:8081; |
| 45 | listen 127.0.0.1:8082 http2 sndbuf=128; |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 46 | server_name localhost; |
| 47 | |
| 48 | http2_max_field_size 128k; |
| 49 | http2_max_header_size 128k; |
| 50 | |
| 51 | location / { |
| 52 | add_header X-Sent-Foo $http_x_foo; |
| 53 | add_header X-Referer $http_referer; |
| 54 | return 200; |
| 55 | } |
| 56 | location /frame_size { |
| 57 | add_header X-LongHeader $arg_h; |
| 58 | add_header X-LongHeader $arg_h; |
| 59 | add_header X-LongHeader $arg_h; |
| 60 | alias %%TESTDIR%%/t2.html; |
| 61 | } |
| 62 | location /continuation { |
| 63 | add_header X-LongHeader $arg_h; |
| 64 | add_header X-LongHeader $arg_h; |
| 65 | add_header X-LongHeader $arg_h; |
| 66 | return 200 body; |
| 67 | |
| 68 | location /continuation/204 { |
| 69 | return 204; |
| 70 | } |
| 71 | } |
| 72 | location /proxy/ { |
| 73 | add_header X-UC-a $upstream_cookie_a; |
| 74 | add_header X-UC-c $upstream_cookie_c; |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 75 | proxy_pass http://127.0.0.1:8083/; |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 76 | proxy_set_header X-Cookie-a $cookie_a; |
| 77 | proxy_set_header X-Cookie-c $cookie_c; |
| 78 | } |
| 79 | location /proxy2/ { |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 80 | proxy_pass http://127.0.0.1:8081/; |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 81 | } |
| 82 | location /set-cookie { |
| 83 | add_header Set-Cookie a=b; |
| 84 | add_header Set-Cookie c=d; |
| 85 | return 200; |
| 86 | } |
| 87 | location /cookie { |
| 88 | add_header X-Cookie $http_cookie; |
| 89 | add_header X-Cookie-a $cookie_a; |
| 90 | add_header X-Cookie-c $cookie_c; |
| 91 | return 200; |
| 92 | } |
| 93 | } |
| 94 | |
| 95 | server { |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 96 | listen 127.0.0.1:8084 http2; |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 97 | server_name localhost; |
| 98 | |
| 99 | http2_max_field_size 22; |
| 100 | } |
| 101 | |
| 102 | server { |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 103 | listen 127.0.0.1:8085 http2; |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 104 | server_name localhost; |
| 105 | |
| 106 | http2_max_header_size 64; |
| 107 | } |
| 108 | } |
| 109 | |
| 110 | EOF |
| 111 | |
| 112 | $t->run_daemon(\&http_daemon); |
Sergey Kandaurov | 3591a68 | 2017-06-16 15:18:40 +0300 | [diff] [blame] | 113 | $t->run()->waitforsocket('127.0.0.1:' . port(8083)); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 114 | |
| 115 | # file size is slightly beyond initial window size: 2**16 + 80 bytes |
| 116 | |
| 117 | $t->write_file('t1.html', |
| 118 | join('', map { sprintf "X%04dXXX", $_ } (1 .. 8202))); |
| 119 | |
| 120 | $t->write_file('t2.html', 'SEE-THIS'); |
| 121 | |
| 122 | ############################################################################### |
| 123 | |
| 124 | # 6.1. Indexed Header Field Representation |
| 125 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 126 | my $s = Test::Nginx::HTTP2->new(); |
| 127 | my $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 128 | { name => ':method', value => 'GET', mode => 0 }, |
| 129 | { name => ':scheme', value => 'http', mode => 0 }, |
| 130 | { name => ':path', value => '/', mode => 0 }, |
| 131 | { name => ':authority', value => 'localhost', mode => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 132 | my $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 133 | |
| 134 | my ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 135 | is($frame->{headers}->{':status'}, 200, 'indexed header field'); |
| 136 | |
| 137 | # 6.2.1. Literal Header Field with Incremental Indexing |
| 138 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 139 | $s = Test::Nginx::HTTP2->new(); |
| 140 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 141 | { name => ':method', value => 'GET', mode => 1, huff => 0 }, |
| 142 | { name => ':scheme', value => 'http', mode => 1, huff => 0 }, |
| 143 | { name => ':path', value => '/', mode => 1, huff => 0 }, |
| 144 | { name => ':authority', value => 'localhost', mode => 1, huff => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 145 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 146 | |
| 147 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 148 | is($frame->{headers}->{':status'}, 200, 'literal with indexing'); |
| 149 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 150 | $s = Test::Nginx::HTTP2->new(); |
| 151 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 152 | { name => ':method', value => 'GET', mode => 1, huff => 1 }, |
| 153 | { name => ':scheme', value => 'http', mode => 1, huff => 1 }, |
| 154 | { name => ':path', value => '/', mode => 1, huff => 1 }, |
| 155 | { name => ':authority', value => 'localhost', mode => 1, huff => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 156 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 157 | |
| 158 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 159 | is($frame->{headers}->{':status'}, 200, 'literal with indexing - huffman'); |
| 160 | |
| 161 | # 6.2.1. Literal Header Field with Incremental Indexing -- New Name |
| 162 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 163 | $s = Test::Nginx::HTTP2->new(); |
| 164 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 165 | { name => ':method', value => 'GET', mode => 2, huff => 0 }, |
| 166 | { name => ':scheme', value => 'http', mode => 2, huff => 0 }, |
| 167 | { name => ':path', value => '/', mode => 2, huff => 0 }, |
| 168 | { name => ':authority', value => 'localhost', mode => 2, huff => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 169 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 170 | |
| 171 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 172 | is($frame->{headers}->{':status'}, 200, 'literal with indexing - new'); |
| 173 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 174 | $s = Test::Nginx::HTTP2->new(); |
| 175 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 176 | { name => ':method', value => 'GET', mode => 2, huff => 1 }, |
| 177 | { name => ':scheme', value => 'http', mode => 2, huff => 1 }, |
| 178 | { name => ':path', value => '/', mode => 2, huff => 1 }, |
| 179 | { name => ':authority', value => 'localhost', mode => 2, huff => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 180 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 181 | |
| 182 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 183 | is($frame->{headers}->{':status'}, 200, 'literal with indexing - new huffman'); |
| 184 | |
| 185 | # 6.2.2. Literal Header Field without Indexing |
| 186 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 187 | $s = Test::Nginx::HTTP2->new(); |
| 188 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 189 | { name => ':method', value => 'GET', mode => 3, huff => 0 }, |
| 190 | { name => ':scheme', value => 'http', mode => 3, huff => 0 }, |
| 191 | { name => ':path', value => '/', mode => 3, huff => 0 }, |
| 192 | { name => ':authority', value => 'localhost', mode => 3, huff => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 193 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 194 | |
| 195 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 196 | is($frame->{headers}->{':status'}, 200, 'literal without indexing'); |
| 197 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 198 | $s = Test::Nginx::HTTP2->new(); |
| 199 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 200 | { name => ':method', value => 'GET', mode => 3, huff => 1 }, |
| 201 | { name => ':scheme', value => 'http', mode => 3, huff => 1 }, |
| 202 | { name => ':path', value => '/', mode => 3, huff => 1 }, |
| 203 | { name => ':authority', value => 'localhost', mode => 3, huff => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 204 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 205 | |
| 206 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 207 | is($frame->{headers}->{':status'}, 200, 'literal without indexing - huffman'); |
| 208 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 209 | $s = Test::Nginx::HTTP2->new(); |
| 210 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 211 | { name => ':method', value => 'GET', mode => 3, huff => 0 }, |
| 212 | { name => ':scheme', value => 'http', mode => 3, huff => 0 }, |
| 213 | { name => ':path', value => '/', mode => 3, huff => 0 }, |
| 214 | { name => ':authority', value => 'localhost', mode => 3, huff => 0 }, |
| 215 | { name => 'referer', value => 'foo', mode => 3, huff => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 216 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 217 | |
| 218 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 219 | is($frame->{headers}->{':status'}, 200, |
| 220 | 'literal without indexing - multibyte index'); |
| 221 | is($frame->{headers}->{'x-referer'}, 'foo', |
| 222 | 'literal without indexing - multibyte index value'); |
| 223 | |
| 224 | # 6.2.2. Literal Header Field without Indexing -- New Name |
| 225 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 226 | $s = Test::Nginx::HTTP2->new(); |
| 227 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 228 | { name => ':method', value => 'GET', mode => 4, huff => 0 }, |
| 229 | { name => ':scheme', value => 'http', mode => 4, huff => 0 }, |
| 230 | { name => ':path', value => '/', mode => 4, huff => 0 }, |
| 231 | { name => ':authority', value => 'localhost', mode => 4, huff => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 232 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 233 | |
| 234 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 235 | is($frame->{headers}->{':status'}, 200, 'literal without indexing - new'); |
| 236 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 237 | $s = Test::Nginx::HTTP2->new(); |
| 238 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 239 | { name => ':method', value => 'GET', mode => 4, huff => 1 }, |
| 240 | { name => ':scheme', value => 'http', mode => 4, huff => 1 }, |
| 241 | { name => ':path', value => '/', mode => 4, huff => 1 }, |
| 242 | { name => ':authority', value => 'localhost', mode => 4, huff => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 243 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 244 | |
| 245 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 246 | is($frame->{headers}->{':status'}, 200, |
| 247 | 'literal without indexing - new huffman'); |
| 248 | |
| 249 | # 6.2.3. Literal Header Field Never Indexed |
| 250 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 251 | $s = Test::Nginx::HTTP2->new(); |
| 252 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 253 | { name => ':method', value => 'GET', mode => 5, huff => 0 }, |
| 254 | { name => ':scheme', value => 'http', mode => 5, huff => 0 }, |
| 255 | { name => ':path', value => '/', mode => 5, huff => 0 }, |
| 256 | { name => ':authority', value => 'localhost', mode => 5, huff => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 257 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 258 | |
| 259 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 260 | is($frame->{headers}->{':status'}, 200, 'literal never indexed'); |
| 261 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 262 | $s = Test::Nginx::HTTP2->new(); |
| 263 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 264 | { name => ':method', value => 'GET', mode => 5, huff => 1 }, |
| 265 | { name => ':scheme', value => 'http', mode => 5, huff => 1 }, |
| 266 | { name => ':path', value => '/', mode => 5, huff => 1 }, |
| 267 | { name => ':authority', value => 'localhost', mode => 5, huff => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 268 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 269 | |
| 270 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 271 | is($frame->{headers}->{':status'}, 200, 'literal never indexed - huffman'); |
| 272 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 273 | $s = Test::Nginx::HTTP2->new(); |
| 274 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 275 | { name => ':method', value => 'GET', mode => 5, huff => 0 }, |
| 276 | { name => ':scheme', value => 'http', mode => 5, huff => 0 }, |
| 277 | { name => ':path', value => '/', mode => 5, huff => 0 }, |
| 278 | { name => ':authority', value => 'localhost', mode => 5, huff => 0 }, |
| 279 | { name => 'referer', value => 'foo', mode => 5, huff => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 280 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 281 | |
| 282 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 283 | is($frame->{headers}->{':status'}, 200, |
| 284 | 'literal never indexed - multibyte index'); |
| 285 | is($frame->{headers}->{'x-referer'}, 'foo', |
| 286 | 'literal never indexed - multibyte index value'); |
| 287 | |
| 288 | # 6.2.3. Literal Header Field Never Indexed -- New Name |
| 289 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 290 | $s = Test::Nginx::HTTP2->new(); |
| 291 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 292 | { name => ':method', value => 'GET', mode => 6, huff => 0 }, |
| 293 | { name => ':scheme', value => 'http', mode => 6, huff => 0 }, |
| 294 | { name => ':path', value => '/', mode => 6, huff => 0 }, |
| 295 | { name => ':authority', value => 'localhost', mode => 6, huff => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 296 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 297 | |
| 298 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 299 | is($frame->{headers}->{':status'}, 200, 'literal never indexed - new'); |
| 300 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 301 | $s = Test::Nginx::HTTP2->new(); |
| 302 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 303 | { name => ':method', value => 'GET', mode => 6, huff => 1 }, |
| 304 | { name => ':scheme', value => 'http', mode => 6, huff => 1 }, |
| 305 | { name => ':path', value => '/', mode => 6, huff => 1 }, |
| 306 | { name => ':authority', value => 'localhost', mode => 6, huff => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 307 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 308 | |
| 309 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 310 | is($frame->{headers}->{':status'}, 200, 'literal never indexed - new huffman'); |
| 311 | |
| 312 | # reuse literal with multibyte indexing |
| 313 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 314 | $s = Test::Nginx::HTTP2->new(); |
| 315 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 316 | { name => ':method', value => 'GET', mode => 0 }, |
| 317 | { name => ':scheme', value => 'http', mode => 0 }, |
| 318 | { name => ':path', value => '/', mode => 0 }, |
| 319 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 320 | { name => 'referer', value => 'foo', mode => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 321 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 322 | |
| 323 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 324 | is($frame->{headers}->{'x-referer'}, 'foo', 'value with indexing - new'); |
| 325 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 326 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 327 | { name => ':method', value => 'GET', mode => 0 }, |
| 328 | { name => ':scheme', value => 'http', mode => 0 }, |
| 329 | { name => ':path', value => '/', mode => 0 }, |
| 330 | { name => ':authority', value => 'localhost', mode => 0 }, |
| 331 | { name => 'referer', value => 'foo', mode => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 332 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 333 | |
| 334 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 335 | is($frame->{headers}->{'x-referer'}, 'foo', 'value with indexing - indexed'); |
| 336 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 337 | $s = Test::Nginx::HTTP2->new(); |
| 338 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 339 | { name => ':method', value => 'GET', mode => 0 }, |
| 340 | { name => ':scheme', value => 'http', mode => 0 }, |
| 341 | { name => ':path', value => '/', mode => 0 }, |
| 342 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 343 | { name => 'x-foo', value => 'X-Bar', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 344 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 345 | |
| 346 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 347 | is($frame->{headers}->{'x-sent-foo'}, 'X-Bar', 'name with indexing - new'); |
| 348 | |
| 349 | # reuse literal with multibyte indexing - reused name |
| 350 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 351 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 352 | { name => ':method', value => 'GET', mode => 0 }, |
| 353 | { name => ':scheme', value => 'http', mode => 0 }, |
| 354 | { name => ':path', value => '/', mode => 0 }, |
| 355 | { name => ':authority', value => 'localhost', mode => 0 }, |
| 356 | { name => 'x-foo', value => 'X-Bar', mode => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 357 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 358 | |
| 359 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 360 | is($frame->{headers}->{'x-sent-foo'}, 'X-Bar', 'name with indexing - indexed'); |
| 361 | |
| 362 | # reuse literal with multibyte indexing - reused name only |
| 363 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 364 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 365 | { name => ':method', value => 'GET', mode => 0 }, |
| 366 | { name => ':scheme', value => 'http', mode => 0 }, |
| 367 | { name => ':path', value => '/', mode => 0 }, |
| 368 | { name => ':authority', value => 'localhost', mode => 0 }, |
| 369 | { name => 'x-foo', value => 'X-Baz', mode => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 370 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 371 | |
| 372 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 373 | is($frame->{headers}->{'x-sent-foo'}, 'X-Baz', |
| 374 | 'name with indexing - indexed name'); |
| 375 | |
| 376 | # response header field with characters not suitable for huffman encoding |
| 377 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 378 | $s = Test::Nginx::HTTP2->new(); |
| 379 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 380 | { name => ':method', value => 'GET', mode => 0 }, |
| 381 | { name => ':scheme', value => 'http', mode => 0 }, |
| 382 | { name => ':path', value => '/', mode => 0 }, |
| 383 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 384 | { name => 'x-foo', value => '{{{{{', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 385 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 386 | |
| 387 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 388 | is($frame->{headers}->{'x-sent-foo'}, '{{{{{', 'rare chars'); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 389 | like($s->{headers}, qr/\Q{{{{{/, 'rare chars - no huffman encoding'); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 390 | |
| 391 | # response header field with huffman encoding |
| 392 | # NB: implementation detail, not obligated |
| 393 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 394 | $s = Test::Nginx::HTTP2->new(); |
| 395 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 396 | { name => ':method', value => 'GET', mode => 0 }, |
| 397 | { name => ':scheme', value => 'http', mode => 0 }, |
| 398 | { name => ':path', value => '/', mode => 0 }, |
| 399 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 400 | { name => 'x-foo', value => 'aaaaa', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 401 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 402 | |
| 403 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 404 | is($frame->{headers}->{'x-sent-foo'}, 'aaaaa', 'well known chars'); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 405 | unlike($s->{headers}, qr/aaaaa/, 'well known chars - huffman encoding'); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 406 | |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 407 | # response header field with huffman encoding - complete table mod \0, CR, LF |
| 408 | # first saturate with short-encoded characters (NB: implementation detail) |
| 409 | |
| 410 | my $field = pack "C*", ((map { 97 } (1 .. 862)), 1 .. 9, 11, 12, 14 .. 255); |
| 411 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 412 | $s = Test::Nginx::HTTP2->new(); |
| 413 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 414 | { name => ':method', value => 'GET', mode => 0 }, |
| 415 | { name => ':scheme', value => 'http', mode => 0 }, |
| 416 | { name => ':path', value => '/', mode => 0 }, |
| 417 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 418 | { name => 'x-foo', value => $field, mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 419 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 420 | |
| 421 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 422 | is($frame->{headers}->{'x-sent-foo'}, $field, 'all chars'); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 423 | unlike($s->{headers}, qr/abcde/, 'all chars - huffman encoding'); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 424 | |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 425 | # 6.3. Dynamic Table Size Update |
| 426 | |
| 427 | # remove some indexed headers from the dynamic table |
| 428 | # by maintaining dynamic table space only for index 0 |
| 429 | # 'x-foo' has index 0, and 'referer' has index 1 |
| 430 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 431 | $s = Test::Nginx::HTTP2->new(); |
| 432 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 433 | { name => ':method', value => 'GET', mode => 0 }, |
| 434 | { name => ':scheme', value => 'http', mode => 0 }, |
| 435 | { name => ':path', value => '/', mode => 0 }, |
| 436 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 437 | { name => 'referer', value => 'foo', mode => 1 }, |
| 438 | { name => 'x-foo', value => 'X-Bar', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 439 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 440 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 441 | $sid = $s->new_stream({ table_size => 61, headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 442 | { name => ':method', value => 'GET', mode => 0 }, |
| 443 | { name => ':scheme', value => 'http', mode => 0 }, |
| 444 | { name => ':path', value => '/', mode => 0 }, |
| 445 | { name => 'x-foo', value => 'X-Bar', mode => 0 }, |
| 446 | { name => ':authority', value => 'localhost', mode => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 447 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 448 | |
| 449 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 450 | isnt($frame, undef, 'updated table size - remaining index'); |
| 451 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 452 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 453 | { name => ':method', value => 'GET', mode => 0 }, |
| 454 | { name => ':scheme', value => 'http', mode => 0 }, |
| 455 | { name => ':path', value => '/', mode => 0 }, |
| 456 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 457 | { name => 'referer', value => 'foo', mode => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 458 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 459 | |
| 460 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 461 | is($frame, undef, 'invalid index'); |
| 462 | |
| 463 | # 5.4.1. Connection Error Handling |
| 464 | # An endpoint that encounters a connection error SHOULD first send a |
| 465 | # GOAWAY frame <..> |
| 466 | |
| 467 | ($frame) = grep { $_->{type} eq "GOAWAY" } @$frames; |
| 468 | ok($frame, 'invalid index - GOAWAY'); |
| 469 | |
| 470 | # RFC 7541, 2.3.3. Index Address Space |
| 471 | # Indices strictly greater than the sum of the lengths of both tables |
| 472 | # MUST be treated as a decoding error. |
| 473 | |
| 474 | # 4.3. Header Compression and Decompression |
| 475 | # A decoding error in a header block MUST be treated |
| 476 | # as a connection error of type COMPRESSION_ERROR. |
| 477 | |
| 478 | is($frame->{last_sid}, $sid, 'invalid index - GOAWAY last stream'); |
| 479 | is($frame->{code}, 9, 'invalid index - GOAWAY COMPRESSION_ERROR'); |
| 480 | |
| 481 | # HPACK zero index |
| 482 | |
| 483 | # RFC 7541, 6.1 Indexed Header Field Representation |
| 484 | # The index value of 0 is not used. It MUST be treated as a decoding |
| 485 | # error if found in an indexed header field representation. |
| 486 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 487 | $s = Test::Nginx::HTTP2->new(); |
| 488 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 489 | { name => ':method', value => 'GET', mode => 0 }, |
| 490 | { name => ':scheme', value => 'http', mode => 0 }, |
| 491 | { name => ':path', value => '/', mode => 0 }, |
| 492 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 493 | { name => '', value => '', mode => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 494 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 495 | |
| 496 | ok($frame, 'zero index - GOAWAY'); |
| 497 | is($frame->{code}, 9, 'zero index - GOAWAY COMPRESSION_ERROR'); |
| 498 | |
| 499 | # invalid table size update |
| 500 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 501 | $s = Test::Nginx::HTTP2->new(); |
| 502 | $sid = $s->new_stream({ table_size => 4097, headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 503 | { name => ':method', value => 'GET', mode => 0 }, |
| 504 | { name => ':scheme', value => 'http', mode => 0 }, |
| 505 | { name => ':path', value => '/', mode => 0 }, |
| 506 | { name => 'x-foo', value => 'X-Bar', mode => 0 }, |
| 507 | { name => ':authority', value => 'localhost', mode => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 508 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 509 | |
| 510 | ($frame) = grep { $_->{type} eq "GOAWAY" } @$frames; |
| 511 | ok($frame, 'invalid table size - GOAWAY'); |
| 512 | is($frame->{last_sid}, $sid, 'invalid table size - GOAWAY last stream'); |
| 513 | is($frame->{code}, 9, 'invalid table size - GOAWAY COMPRESSION_ERROR'); |
| 514 | |
| 515 | # request header field with multiple values |
| 516 | |
| 517 | # 8.1.2.5. Compressing the Cookie Header Field |
| 518 | # To allow for better compression efficiency, the Cookie header field |
| 519 | # MAY be split into separate header fields <..>. |
| 520 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 521 | $s = Test::Nginx::HTTP2->new(); |
| 522 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 523 | { name => ':method', value => 'GET', mode => 0 }, |
| 524 | { name => ':scheme', value => 'http', mode => 0 }, |
| 525 | { name => ':path', value => '/cookie', mode => 2 }, |
| 526 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 527 | { name => 'cookie', value => 'a=b', mode => 2}, |
| 528 | { name => 'cookie', value => 'c=d', mode => 2}]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 529 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 530 | |
| 531 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 532 | is($frame->{headers}->{'x-cookie-a'}, 'b', |
| 533 | 'multiple request header fields - cookie'); |
| 534 | is($frame->{headers}->{'x-cookie-c'}, 'd', |
| 535 | 'multiple request header fields - cookie 2'); |
| 536 | is($frame->{headers}->{'x-cookie'}, 'a=b; c=d', |
| 537 | 'multiple request header fields - semi-colon'); |
| 538 | |
| 539 | # request header field with multiple values to HTTP backend |
| 540 | |
| 541 | # 8.1.2.5. Compressing the Cookie Header Field |
| 542 | # these MUST be concatenated into a single octet string |
| 543 | # using the two-octet delimiter of 0x3B, 0x20 (the ASCII string "; ") |
| 544 | # before being passed into a non-HTTP/2 context, such as an HTTP/1.1 |
| 545 | # connection <..> |
| 546 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 547 | $s = Test::Nginx::HTTP2->new(); |
| 548 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 549 | { name => ':method', value => 'GET', mode => 0 }, |
| 550 | { name => ':scheme', value => 'http', mode => 0 }, |
| 551 | { name => ':path', value => '/proxy/cookie', mode => 2 }, |
| 552 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 553 | { name => 'cookie', value => 'a=b', mode => 2 }, |
| 554 | { name => 'cookie', value => 'c=d', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 555 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 556 | |
| 557 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 558 | is($frame->{headers}->{'x-sent-cookie'}, 'a=b; c=d', |
| 559 | 'multiple request header fields proxied - semi-colon'); |
| 560 | is($frame->{headers}->{'x-sent-cookie2'}, '', |
| 561 | 'multiple request header fields proxied - dublicate cookie'); |
| 562 | is($frame->{headers}->{'x-sent-cookie-a'}, 'b', |
| 563 | 'multiple request header fields proxied - cookie 1'); |
| 564 | is($frame->{headers}->{'x-sent-cookie-c'}, 'd', |
| 565 | 'multiple request header fields proxied - cookie 2'); |
| 566 | |
| 567 | # response header field with multiple values |
| 568 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 569 | $s = Test::Nginx::HTTP2->new(); |
| 570 | $sid = $s->new_stream({ path => '/set-cookie' }); |
| 571 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 572 | |
| 573 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 574 | is($frame->{headers}->{'set-cookie'}[0], 'a=b', |
| 575 | 'multiple response header fields - cookie'); |
| 576 | is($frame->{headers}->{'set-cookie'}[1], 'c=d', |
| 577 | 'multiple response header fields - cookie 2'); |
| 578 | |
| 579 | # response header field with multiple values from HTTP backend |
| 580 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 581 | $s = Test::Nginx::HTTP2->new(); |
| 582 | $sid = $s->new_stream({ path => '/proxy/set-cookie' }); |
| 583 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 584 | |
| 585 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 586 | is($frame->{headers}->{'set-cookie'}[0], 'a=b', |
| 587 | 'multiple response header proxied - cookie'); |
| 588 | is($frame->{headers}->{'set-cookie'}[1], 'c=d', |
| 589 | 'multiple response header proxied - cookie 2'); |
| 590 | is($frame->{headers}->{'x-uc-a'}, 'b', |
| 591 | 'multiple response header proxied - upstream cookie'); |
| 592 | is($frame->{headers}->{'x-uc-c'}, 'd', |
| 593 | 'multiple response header proxied - upstream cookie 2'); |
| 594 | |
| 595 | # CONTINUATION in response |
| 596 | # put three long header fields (not less than SETTINGS_MAX_FRAME_SIZE/2) |
| 597 | # to break header block into separate frames, one such field per frame |
| 598 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 599 | $s = Test::Nginx::HTTP2->new(); |
| 600 | $sid = $s->new_stream({ path => '/continuation?h=' . 'x' x 2**13 }); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 601 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 602 | $frames = $s->read(all => [{ sid => $sid, fin => 0x4 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 603 | my @data = grep { $_->{type} =~ "HEADERS|CONTINUATION" } @$frames; |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 604 | my $data = $data[-1]; |
| 605 | is(@{$data->{headers}{'x-longheader'}}, 3, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 606 | 'response CONTINUATION - headers'); |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 607 | is($data->{headers}{'x-longheader'}[0], 'x' x 2**13, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 608 | 'response CONTINUATION - header 1'); |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 609 | is($data->{headers}{'x-longheader'}[1], 'x' x 2**13, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 610 | 'response CONTINUATION - header 2'); |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 611 | is($data->{headers}{'x-longheader'}[2], 'x' x 2**13, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 612 | 'response CONTINUATION - header 3'); |
| 613 | @data = sort { $a <=> $b } map { $_->{length} } @data; |
| 614 | cmp_ok($data[-1], '<=', 2**14, 'response CONTINUATION - max frame size'); |
| 615 | |
| 616 | # same but without response DATA frames |
| 617 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 618 | $s = Test::Nginx::HTTP2->new(); |
| 619 | $sid = $s->new_stream({ path => '/continuation/204?h=' . 'x' x 2**13 }); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 620 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 621 | $frames = $s->read(all => [{ sid => $sid, fin => 0x4 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 622 | @data = grep { $_->{type} =~ "HEADERS|CONTINUATION" } @$frames; |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 623 | $data = $data[-1]; |
| 624 | is(@{$data->{headers}{'x-longheader'}}, 3, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 625 | 'no body CONTINUATION - headers'); |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 626 | is($data->{headers}{'x-longheader'}[0], 'x' x 2**13, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 627 | 'no body CONTINUATION - header 1'); |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 628 | is($data->{headers}{'x-longheader'}[1], 'x' x 2**13, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 629 | 'no body CONTINUATION - header 2'); |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 630 | is($data->{headers}{'x-longheader'}[2], 'x' x 2**13, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 631 | 'no body CONTINUATION - header 3'); |
| 632 | @data = sort { $a <=> $b } map { $_->{length} } @data; |
| 633 | cmp_ok($data[-1], '<=', 2**14, 'no body CONTINUATION - max frame size'); |
| 634 | |
| 635 | # response header block is always split by SETTINGS_MAX_FRAME_SIZE |
| 636 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 637 | $s = Test::Nginx::HTTP2->new(); |
| 638 | $sid = $s->new_stream({ path => '/continuation?h=' . 'x' x 2**15 }); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 639 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 640 | $frames = $s->read(all => [{ sid => $sid, fin => 0x4 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 641 | @data = grep { $_->{type} =~ "HEADERS|CONTINUATION" } @$frames; |
| 642 | @data = sort { $a <=> $b } map { $_->{length} } @data; |
| 643 | cmp_ok($data[-1], '<=', 2**14, 'response header frames limited'); |
| 644 | |
| 645 | # response header frame sent in parts |
| 646 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 647 | $s = Test::Nginx::HTTP2->new(port(8082)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 648 | $s->h2_settings(0, 0x5 => 2**17); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 649 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 650 | $sid = $s->new_stream({ path => '/frame_size?h=' . 'x' x 2**15 }); |
| 651 | $frames = $s->read(all => [{ sid => $sid, fin => 0x4 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 652 | |
| 653 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 654 | ok($frame, 'response header - parts'); |
| 655 | |
| 656 | SKIP: { |
| 657 | skip 'response header failed', 1 unless $frame; |
Sergey Kandaurov | 62a669e | 2016-10-16 13:01:36 +0300 | [diff] [blame] | 658 | skip 'broken sendfile', 1 if $^O eq 'freebsd' and |
| 659 | $Config{osvers} =~ '11.0-release' and |
| 660 | $t->read_file('nginx.conf') =~ /sendfile on/; |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 661 | |
| 662 | is(length join('', @{$frame->{headers}->{'x-longheader'}}), 98304, |
| 663 | 'response header - headers'); |
| 664 | |
| 665 | } |
| 666 | |
| 667 | # response header block split and sent in parts |
| 668 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 669 | $s = Test::Nginx::HTTP2->new(port(8082)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 670 | $sid = $s->new_stream({ path => '/continuation?h=' . 'x' x 2**15 }); |
| 671 | $frames = $s->read(all => [{ sid => $sid, fin => 0x4 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 672 | |
| 673 | @data = grep { $_->{type} =~ "HEADERS|CONTINUATION" } @$frames; |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 674 | ok(@data, 'response header split'); |
| 675 | |
| 676 | SKIP: { |
| 677 | skip 'response header split failed', 2 unless @data; |
| 678 | |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 679 | my ($lengths) = sort { $b <=> $a } map { $_->{length} } @data; |
| 680 | cmp_ok($lengths, '<=', 16384, 'response header split - max size'); |
| 681 | |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 682 | is(length join('', @{$data[-1]->{headers}->{'x-longheader'}}), 98304, |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 683 | 'response header split - headers'); |
| 684 | |
Sergey Kandaurov | 1e210ec | 2017-05-03 13:36:40 +0300 | [diff] [blame] | 685 | } |
| 686 | |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 687 | # max_field_size - header field name |
| 688 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 689 | $s = Test::Nginx::HTTP2->new(port(8084)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 690 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 691 | { name => ':method', value => 'GET', mode => 0 }, |
| 692 | { name => ':scheme', value => 'http', mode => 0 }, |
| 693 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 694 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 695 | { name => 'longname10' x 2 . 'x', value => 'value', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 696 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 697 | |
| 698 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 699 | ok($frame, 'field name size less'); |
| 700 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 701 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 702 | { name => ':method', value => 'GET', mode => 0 }, |
| 703 | { name => ':scheme', value => 'http', mode => 0 }, |
| 704 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 705 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 706 | { name => 'longname10' x 2 . 'x', value => 'value', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 707 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 708 | |
| 709 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 710 | ok($frame, 'field name size second'); |
| 711 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 712 | $s = Test::Nginx::HTTP2->new(port(8084)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 713 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 714 | { name => ':method', value => 'GET', mode => 0 }, |
| 715 | { name => ':scheme', value => 'http', mode => 0 }, |
| 716 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 717 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 718 | { name => 'longname10' x 2 . 'xx', value => 'value', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 719 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 720 | |
| 721 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 722 | ok($frame, 'field name size equal'); |
| 723 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 724 | $s = Test::Nginx::HTTP2->new(port(8084)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 725 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 726 | { name => ':method', value => 'GET', mode => 0 }, |
| 727 | { name => ':scheme', value => 'http', mode => 0 }, |
| 728 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 729 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 730 | { name => 'longname10' x 2 . 'xxx', value => 'value', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 731 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 732 | |
| 733 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 734 | is($frame, undef, 'field name size greater'); |
| 735 | |
| 736 | # max_field_size - header field value |
| 737 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 738 | $s = Test::Nginx::HTTP2->new(port(8084)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 739 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 740 | { name => ':method', value => 'GET', mode => 0 }, |
| 741 | { name => ':scheme', value => 'http', mode => 0 }, |
| 742 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 743 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 744 | { name => 'name', value => 'valu5' x 4 . 'x', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 745 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 746 | |
| 747 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 748 | ok($frame, 'field value size less'); |
| 749 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 750 | $s = Test::Nginx::HTTP2->new(port(8084)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 751 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 752 | { name => ':method', value => 'GET', mode => 0 }, |
| 753 | { name => ':scheme', value => 'http', mode => 0 }, |
| 754 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 755 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 756 | { name => 'name', value => 'valu5' x 4 . 'xx', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 757 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 758 | |
| 759 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 760 | ok($frame, 'field value size equal'); |
| 761 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 762 | $s = Test::Nginx::HTTP2->new(port(8084)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 763 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 764 | { name => ':method', value => 'GET', mode => 0 }, |
| 765 | { name => ':scheme', value => 'http', mode => 0 }, |
| 766 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 767 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 768 | { name => 'name', value => 'valu5' x 4 . 'xxx', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 769 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 770 | |
| 771 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 772 | is($frame, undef, 'field value size greater'); |
| 773 | |
| 774 | # max_header_size |
| 775 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 776 | $s = Test::Nginx::HTTP2->new(port(8085)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 777 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 778 | { name => ':method', value => 'GET', mode => 0 }, |
| 779 | { name => ':scheme', value => 'http', mode => 0 }, |
| 780 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 781 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 782 | { name => 'longname9', value => 'x', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 783 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 784 | |
| 785 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 786 | ok($frame, 'header size less'); |
| 787 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 788 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 789 | { name => ':method', value => 'GET', mode => 0 }, |
| 790 | { name => ':scheme', value => 'http', mode => 0 }, |
| 791 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 792 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 793 | { name => 'longname9', value => 'x', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 794 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 795 | |
| 796 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 797 | ok($frame, 'header size second'); |
| 798 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 799 | $s = Test::Nginx::HTTP2->new(port(8085)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 800 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 801 | { name => ':method', value => 'GET', mode => 0 }, |
| 802 | { name => ':scheme', value => 'http', mode => 0 }, |
| 803 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 804 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 805 | { name => 'longname9', value => 'xx', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 806 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 807 | |
| 808 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 809 | ok($frame, 'header size equal'); |
| 810 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 811 | $s = Test::Nginx::HTTP2->new(port(8085)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 812 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 813 | { name => ':method', value => 'GET', mode => 0 }, |
| 814 | { name => ':scheme', value => 'http', mode => 0 }, |
| 815 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 816 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 817 | { name => 'longname9', value => 'xxx', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 818 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 819 | |
| 820 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 821 | is($frame, undef, 'header size greater'); |
| 822 | |
| 823 | # header size is based on (decompressed) header list |
| 824 | # two extra 1-byte indices would otherwise fit in max_header_size |
| 825 | |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 826 | $s = Test::Nginx::HTTP2->new(port(8085)); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 827 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 828 | { name => ':method', value => 'GET', mode => 0 }, |
| 829 | { name => ':scheme', value => 'http', mode => 0 }, |
| 830 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 831 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 832 | { name => 'longname9', value => 'x', mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 833 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 834 | |
| 835 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 836 | ok($frame, 'header size new index'); |
| 837 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 838 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 839 | { name => ':method', value => 'GET', mode => 0 }, |
| 840 | { name => ':scheme', value => 'http', mode => 0 }, |
| 841 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 842 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 843 | { name => 'longname9', value => 'x', mode => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 844 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 845 | |
| 846 | ($frame) = grep { $_->{type} eq 'DATA' } @$frames; |
| 847 | ok($frame, 'header size indexed'); |
| 848 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 849 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 850 | { name => ':method', value => 'GET', mode => 0 }, |
| 851 | { name => ':scheme', value => 'http', mode => 0 }, |
| 852 | { name => ':path', value => '/t2.html', mode => 1 }, |
| 853 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 854 | { name => 'longname9', value => 'x', mode => 0 }, |
| 855 | { name => 'longname9', value => 'x', mode => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 856 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 857 | |
| 858 | ($frame) = grep { $_->{type} eq 'GOAWAY' } @$frames; |
| 859 | is($frame->{code}, 0xb, 'header size indexed greater'); |
| 860 | |
| 861 | # HPACK table boundary |
| 862 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 863 | $s = Test::Nginx::HTTP2->new(); |
| 864 | $s->read(all => [{ sid => $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 865 | { name => ':method', value => 'GET', mode => 0 }, |
| 866 | { name => ':scheme', value => 'http', mode => 0 }, |
| 867 | { name => ':path', value => '/', mode => 0 }, |
| 868 | { name => ':authority', value => '', mode => 0 }, |
| 869 | { name => 'x' x 2016, value => 'x' x 2048, mode => 2 }]}), fin => 1 }]); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 870 | $frames = $s->read(all => [{ sid => $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 871 | { name => ':method', value => 'GET', mode => 0 }, |
| 872 | { name => ':scheme', value => 'http', mode => 0 }, |
| 873 | { name => ':path', value => '/', mode => 0 }, |
| 874 | { name => ':authority', value => '', mode => 0 }, |
| 875 | { name => 'x' x 2016, value => 'x' x 2048, mode => 0 }]}), fin => 1 }]); |
| 876 | |
| 877 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 878 | ok($frame, 'HPACK table boundary'); |
| 879 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 880 | $s->read(all => [{ sid => $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 881 | { name => ':method', value => 'GET', mode => 0 }, |
| 882 | { name => ':scheme', value => 'http', mode => 0 }, |
| 883 | { name => ':path', value => '/', mode => 0 }, |
| 884 | { name => ':authority', value => '', mode => 0 }, |
| 885 | { name => 'x' x 33, value => 'x' x 4031, mode => 2 }]}), fin => 1 }]); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 886 | $frames = $s->read(all => [{ sid => $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 887 | { name => ':method', value => 'GET', mode => 0 }, |
| 888 | { name => ':scheme', value => 'http', mode => 0 }, |
| 889 | { name => ':path', value => '/', mode => 0 }, |
| 890 | { name => ':authority', value => '', mode => 0 }, |
| 891 | { name => 'x' x 33, value => 'x' x 4031, mode => 0 }]}), fin => 1 }]); |
| 892 | |
| 893 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 894 | ok($frame, 'HPACK table boundary - header field name'); |
| 895 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 896 | $s->read(all => [{ sid => $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 897 | { name => ':method', value => 'GET', mode => 0 }, |
| 898 | { name => ':scheme', value => 'http', mode => 0 }, |
| 899 | { name => ':path', value => '/', mode => 0 }, |
| 900 | { name => ':authority', value => '', mode => 0 }, |
| 901 | { name => 'x', value => 'x' x 64, mode => 2 }]}), fin => 1 }]); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 902 | $frames = $s->read(all => [{ sid => $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 903 | { name => ':method', value => 'GET', mode => 0 }, |
| 904 | { name => ':scheme', value => 'http', mode => 0 }, |
| 905 | { name => ':path', value => '/', mode => 0 }, |
| 906 | { name => ':authority', value => '', mode => 0 }, |
| 907 | { name => 'x', value => 'x' x 64, mode => 0 }]}), fin => 1 }]); |
| 908 | |
| 909 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 910 | ok($frame, 'HPACK table boundary - header field value'); |
| 911 | |
| 912 | # ensure that request header field value with newline doesn't get split |
| 913 | # |
| 914 | # 10.3. Intermediary Encapsulation Attacks |
| 915 | # Any request or response that contains a character not permitted |
| 916 | # in a header field value MUST be treated as malformed. |
| 917 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 918 | $s = Test::Nginx::HTTP2->new(); |
| 919 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 920 | { name => ':method', value => 'GET', mode => 0 }, |
| 921 | { name => ':scheme', value => 'http', mode => 0 }, |
| 922 | { name => ':path', value => '/proxy2/', mode => 1 }, |
| 923 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 924 | { name => 'x-foo', value => "x-bar\r\nreferer:see-this", mode => 2 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 925 | $frames = $s->read(all => [{ type => 'RST_STREAM' }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 926 | |
| 927 | # 10.3. Intermediary Encapsulation Attacks |
| 928 | # An intermediary therefore cannot translate an HTTP/2 request or response |
| 929 | # containing an invalid field name into an HTTP/1.1 message. |
| 930 | |
| 931 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 932 | isnt($frame->{headers}->{'x-referer'}, 'see-this', 'newline in request header'); |
| 933 | |
| 934 | # 8.1.2.6. Malformed Requests and Responses |
| 935 | # Malformed requests or responses that are detected MUST be treated |
| 936 | # as a stream error (Section 5.4.2) of type PROTOCOL_ERROR. |
| 937 | |
| 938 | ($frame) = grep { $_->{type} eq "RST_STREAM" } @$frames; |
| 939 | is($frame->{sid}, $sid, 'newline in request header - RST_STREAM sid'); |
| 940 | is($frame->{length}, 4, 'newline in request header - RST_STREAM length'); |
| 941 | is($frame->{flags}, 0, 'newline in request header - RST_STREAM flags'); |
| 942 | is($frame->{code}, 1, 'newline in request header - RST_STREAM code'); |
| 943 | |
| 944 | # invalid header name as seen with underscore should not lead to ignoring rest |
| 945 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 946 | $s = Test::Nginx::HTTP2->new(); |
| 947 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 948 | { name => ':method', value => 'GET', mode => 0 }, |
| 949 | { name => ':scheme', value => 'http', mode => 0 }, |
| 950 | { name => ':path', value => '/', mode => 0 }, |
| 951 | { name => ':authority', value => 'localhost', mode => 1 }, |
| 952 | { name => 'x_foo', value => "x-bar", mode => 2 }, |
| 953 | { name => 'referer', value => "see-this", mode => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 954 | $frames = $s->read(all => [{ type => 'HEADERS' }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 955 | |
| 956 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 957 | is($frame->{headers}->{'x-referer'}, 'see-this', 'after invalid header name'); |
| 958 | |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 959 | # missing mandatory request header ':scheme' |
| 960 | |
| 961 | TODO: { |
Sergey Kandaurov | d3af7fb | 2017-06-14 12:38:45 +0300 | [diff] [blame] | 962 | local $TODO = 'not yet' unless $t->has_version('1.13.2'); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 963 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 964 | $s = Test::Nginx::HTTP2->new(); |
| 965 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 966 | { name => ':method', value => 'GET', mode => 0 }, |
| 967 | { name => ':path', value => '/', mode => 0 }, |
| 968 | { name => ':authority', value => 'localhost', mode => 1 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 969 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 970 | |
| 971 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 972 | is($frame->{headers}->{':status'}, 400, 'incomplete headers'); |
| 973 | |
| 974 | } |
| 975 | |
| 976 | # empty request header ':authority' |
| 977 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 978 | $s = Test::Nginx::HTTP2->new(); |
| 979 | $sid = $s->new_stream({ headers => [ |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 980 | { name => ':method', value => 'GET', mode => 0 }, |
| 981 | { name => ':scheme', value => 'http', mode => 0 }, |
| 982 | { name => ':path', value => '/', mode => 0 }, |
| 983 | { name => ':authority', value => '', mode => 0 }]}); |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 984 | $frames = $s->read(all => [{ sid => $sid, fin => 1 }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 985 | |
| 986 | ($frame) = grep { $_->{type} eq "HEADERS" } @$frames; |
| 987 | is($frame->{headers}->{':status'}, 400, 'empty authority'); |
| 988 | |
| 989 | # client sent invalid :path header |
| 990 | |
Sergey Kandaurov | 67599f4 | 2016-06-17 11:36:33 +0300 | [diff] [blame] | 991 | $sid = $s->new_stream({ path => 't1.html' }); |
| 992 | $frames = $s->read(all => [{ type => 'RST_STREAM' }]); |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 993 | |
| 994 | ($frame) = grep { $_->{type} eq "RST_STREAM" } @$frames; |
| 995 | is($frame->{code}, 1, 'invalid path'); |
| 996 | |
| 997 | ############################################################################### |
| 998 | |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 999 | sub http_daemon { |
| 1000 | my $server = IO::Socket::INET->new( |
| 1001 | Proto => 'tcp', |
| 1002 | LocalHost => '127.0.0.1', |
Andrey Zelenkov | e59bf36 | 2016-07-12 17:39:03 +0300 | [diff] [blame] | 1003 | LocalPort => port(8083), |
Andrey Zelenkov | 82aff46 | 2016-03-23 17:23:08 +0300 | [diff] [blame] | 1004 | Listen => 5, |
| 1005 | Reuse => 1 |
| 1006 | ) |
| 1007 | or die "Can't create listening socket: $!\n"; |
| 1008 | |
| 1009 | local $SIG{PIPE} = 'IGNORE'; |
| 1010 | |
| 1011 | while (my $client = $server->accept()) { |
| 1012 | $client->autoflush(1); |
| 1013 | |
| 1014 | my $headers = ''; |
| 1015 | my $uri = ''; |
| 1016 | |
| 1017 | while (<$client>) { |
| 1018 | $headers .= $_; |
| 1019 | last if (/^\x0d?\x0a?$/); |
| 1020 | } |
| 1021 | |
| 1022 | next if $headers eq ''; |
| 1023 | $uri = $1 if $headers =~ /^\S+\s+([^ ]+)\s+HTTP/i; |
| 1024 | |
| 1025 | if ($uri eq '/cookie') { |
| 1026 | |
| 1027 | my ($cookie, $cookie2) = $headers =~ /Cookie: (.+)/ig; |
| 1028 | $cookie2 = '' unless defined $cookie2; |
| 1029 | |
| 1030 | my ($cookie_a, $cookie_c) = ('', ''); |
| 1031 | $cookie_a = $1 if $headers =~ /X-Cookie-a: (.+)/i; |
| 1032 | $cookie_c = $1 if $headers =~ /X-Cookie-c: (.+)/i; |
| 1033 | |
| 1034 | print $client <<EOF; |
| 1035 | HTTP/1.1 200 OK |
| 1036 | Connection: close |
| 1037 | X-Sent-Cookie: $cookie |
| 1038 | X-Sent-Cookie2: $cookie2 |
| 1039 | X-Sent-Cookie-a: $cookie_a |
| 1040 | X-Sent-Cookie-c: $cookie_c |
| 1041 | |
| 1042 | EOF |
| 1043 | |
| 1044 | } elsif ($uri eq '/set-cookie') { |
| 1045 | |
| 1046 | print $client <<EOF; |
| 1047 | HTTP/1.1 200 OK |
| 1048 | Connection: close |
| 1049 | Set-Cookie: a=b |
| 1050 | Set-Cookie: c=d |
| 1051 | |
| 1052 | EOF |
| 1053 | |
| 1054 | } |
| 1055 | |
| 1056 | } continue { |
| 1057 | close $client; |
| 1058 | } |
| 1059 | } |
| 1060 | |
| 1061 | ############################################################################### |