| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
1
|
|
|
1
|
|
3
|
use strict; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
26
|
|
|
2
|
1
|
|
|
1
|
|
3
|
use warnings; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
24
|
|
|
3
|
|
|
|
|
|
|
package App::DubiousHTTP::Tests::Chunked; |
|
4
|
1
|
|
|
1
|
|
4
|
use App::DubiousHTTP::Tests::Common; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
137
|
|
|
5
|
1
|
|
|
1
|
|
4
|
use Compress::Raw::Zlib; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
1998
|
|
|
6
|
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
SETUP( |
|
8
|
|
|
|
|
|
|
'chunked', |
|
9
|
|
|
|
|
|
|
"Variations of server side chunked encoding", |
|
10
|
|
|
|
|
|
|
<<'DESC', |
|
11
|
|
|
|
|
|
|
Various tests with invalid or uncommon forms of setting or not setting the |
|
12
|
|
|
|
|
|
|
Transfer-Encoding: chunked header: |
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
chunked is not defined for HTTP/1.0, but some systems still interpret the header for HTTP/1.0 responses |
|
15
|
|
|
|
|
|
|
some systems do not support breaking HTTP header over multiple lines |
|
16
|
|
|
|
|
|
|
some systems are happy if 'chunked' is matched somewhere in the header |
|
17
|
|
|
|
|
|
|
some even interprete the existence of a Transfer-Encoding header as enough to expect chunked data |
|
18
|
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
DESC |
|
20
|
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
# ------------------------ Tests ----------------------------------- |
|
22
|
|
|
|
|
|
|
[ 'VALID: basic tests' ], |
|
23
|
|
|
|
|
|
|
[ SHOULDBE_VALID, 'chunked' => 'simple and valid chunking'], |
|
24
|
|
|
|
|
|
|
[ MUSTBE_VALID, 'clen' => 'content-length header, not chunked'], |
|
25
|
|
|
|
|
|
|
|
|
26
|
|
|
|
|
|
|
[ 'VALID: modification of chunk size' ], |
|
27
|
|
|
|
|
|
|
[ UNCOMMON_VALID, '0size' => "chunks size prefixed with 0" ], |
|
28
|
|
|
|
|
|
|
[ UNCOMMON_VALID, '00size' => "chunks size prefixed with 00" ], |
|
29
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'ucsize' => "upper case size" ], |
|
30
|
|
|
|
|
|
|
[ UNCOMMON_VALID, '0ucsize' => "upper case size prefix with 0" ], |
|
31
|
|
|
|
|
|
|
[ INVALID, '32-size' => "negative size for 32bit uint" ], |
|
32
|
|
|
|
|
|
|
[ INVALID, '64-size' => "negative size for 64bit uint" ], |
|
33
|
|
|
|
|
|
|
[ INVALID, 'size-space' => "size followed by space" ], |
|
34
|
|
|
|
|
|
|
[ INVALID, 'size-tab' => "size followed by tab" ], |
|
35
|
|
|
|
|
|
|
[ INVALID, 'size-cr' => "size followed by " ], |
|
36
|
|
|
|
|
|
|
[ INVALID, 'size-lf' => "size followed by " ], |
|
37
|
|
|
|
|
|
|
[ INVALID, 'size-x' => "size followed by char 'x'" ], |
|
38
|
|
|
|
|
|
|
[ INVALID, 'size-\054' => "size followed by char ','" ], |
|
39
|
|
|
|
|
|
|
[ INVALID, 'size-\000' => "size followed by char \\000" ], |
|
40
|
|
|
|
|
|
|
[ INVALID, 'size-\013' => "size followed by char \\v" ], |
|
41
|
|
|
|
|
|
|
[ INVALID, 'size-\014' => "size followed by char \\f" ], |
|
42
|
|
|
|
|
|
|
[ INVALID, 'size-spacex' => "size followed by space and char 'x'" ], |
|
43
|
|
|
|
|
|
|
[ INVALID, 'space-size' => "size prefixed by space" ], |
|
44
|
|
|
|
|
|
|
[ INVALID, 'tab-size' => "size prefixed by tab" ], |
|
45
|
|
|
|
|
|
|
[ INVALID, 'cr-size' => "size prefixed by cr" ], |
|
46
|
|
|
|
|
|
|
[ INVALID, 'lf-size' => "size prefixed by lf" ], |
|
47
|
|
|
|
|
|
|
[ INVALID, 'crlf-size' => 'size prefixed by "\r\n"' ], |
|
48
|
|
|
|
|
|
|
[ INVALID, 'crlf-crlf-size' => 'size prefixed by "\r\n\r\n"' ], |
|
49
|
|
|
|
|
|
|
[ INVALID, 'crlf-x-crlf-size' => 'size prefixed by "\r\nx\r\n"' ], |
|
50
|
|
|
|
|
|
|
[ INVALID, 'x-size' => "size prefixed by char 'x'" ], |
|
51
|
|
|
|
|
|
|
[ INVALID, '\054-size' => "size prefixed by char ','" ], |
|
52
|
|
|
|
|
|
|
[ INVALID, '\073-size' => "size prefixed by char ';'" ], |
|
53
|
|
|
|
|
|
|
[ INVALID, '\000-size' => "size prefixed by char \\000" ], |
|
54
|
|
|
|
|
|
|
[ INVALID, '\013-size' => "size prefixed by char \\v" ], |
|
55
|
|
|
|
|
|
|
[ INVALID, '\014-size' => "size prefixed by char \\f" ], |
|
56
|
|
|
|
|
|
|
[ INVALID, 'xspace-size' => "size prefixed by char 'x' and space" ], |
|
57
|
|
|
|
|
|
|
[ INVALID, '\053-size' => "size prefixed by char '+'" ], |
|
58
|
|
|
|
|
|
|
[ INVALID, '\060\170-size' => "size prefixed by '0x'" ], |
|
59
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'final=00' => 'final chunk size "00"' ], |
|
60
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'final=00000000000000000000' => 'final chunk size "00000000000000000000"' ], |
|
61
|
|
|
|
|
|
|
[ INVALID, 'final=0x' => 'final chunk size "0x"' ], |
|
62
|
|
|
|
|
|
|
[ INVALID, 'final=Foo' => 'final chunk size "Foo"' ], |
|
63
|
|
|
|
|
|
|
[ INVALID, 'finalchunk=0\012' => 'final chunk just "0\n"' ], |
|
64
|
|
|
|
|
|
|
[ INVALID, 'finalchunk=0\015' => 'final chunk just "0\r"' ], |
|
65
|
|
|
|
|
|
|
[ INVALID, 'finalchunk=0' => 'final chunk just "0"' ], |
|
66
|
|
|
|
|
|
|
[ INVALID, 'finalchunk=0\012\012' => 'final chunk "0\n\n"' ], |
|
67
|
|
|
|
|
|
|
[ INVALID, 'finalchunk=0\012\040\012' => 'final chunk "0\n\n"' ], |
|
68
|
|
|
|
|
|
|
[ INVALID, 'finalchunk=0\012\015\012' => 'final chunk "0\n\r\n"' ], |
|
69
|
|
|
|
|
|
|
[ INVALID, 'finalchunk=0\012\015\015\012' => 'final chunk "0\n\r\r\n"' ], |
|
70
|
|
|
|
|
|
|
[ INVALID, 'finalchunk=0\015\012foobar\015\012' => 'final chunk "0\r\nfoobar\r\n"' ], |
|
71
|
|
|
|
|
|
|
|
|
72
|
|
|
|
|
|
|
[ 'VALID: (but uncommon) use of extensions in chunked header' ], |
|
73
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'chunk-ext-junk' => "chunked with some junk chunk extension" ], |
|
74
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'chunk-ext-chunk' => "chunked with some junk chunk extension looking like a chunk" ], |
|
75
|
|
|
|
|
|
|
|
|
76
|
|
|
|
|
|
|
[ 'VALID: combined with content-length' ], |
|
77
|
|
|
|
|
|
|
# according to RFC2616 TE chunked has preference to clen |
|
78
|
|
|
|
|
|
|
[ VALID, 'chunked,clen' => 'chunked first then content-length, served chunked'], |
|
79
|
|
|
|
|
|
|
[ VALID, 'clen,chunked' => 'content-length first then chunked, served chunked'], |
|
80
|
|
|
|
|
|
|
# but some still expect clen bytes |
|
81
|
|
|
|
|
|
|
# safari does not like it, so mark it as uncommon |
|
82
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'chunked,clen200' => 'chunking and content-length header with double length, served chunked'], |
|
83
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'chunked,clen50' => 'chunking and content-length header with half length, served chunked'], |
|
84
|
|
|
|
|
|
|
[ INVALID, 'addjunk,chunked,clen50' => 'content+junk, chunked, content-length header includes content only' ], |
|
85
|
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
[ 'INVALID: chunking is only allowed with HTTP/1.1' ], |
|
87
|
|
|
|
|
|
|
[ INVALID, 'chunked,http10' => 'Chunked Header and HTTP/1.0. Served chunked.'], |
|
88
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,http10' => 'Chunked Header and Content-length and HTTP/1.0. Served chunked.'], |
|
89
|
|
|
|
|
|
|
[ INVALID, 'clen,chunked,http10' => 'Content-length Header and Chunked and HTTP/1.0. Served chunked.'], |
|
90
|
|
|
|
|
|
|
[ INVALID, 'chunked,http10,gzip' => 'Chunked Header and HTTP/1.0. Served chunked with gzip.'], |
|
91
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,http10,gzip' => 'Chunked Header and Content-length and HTTP/1.0. Served chunked with gzip.'], |
|
92
|
|
|
|
|
|
|
[ INVALID, 'clen,chunked,http10,gzip' => 'Content-length Header and Chunked and HTTP/1.0. Served chunked with gzip.'], |
|
93
|
|
|
|
|
|
|
|
|
94
|
|
|
|
|
|
|
[ 'VALID: chunked header should be ignored with HTTP/1.0' ], |
|
95
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'chunked,http10,do_clen' => 'Chunked Header and HTTP/1.0. Not served chunked.'], |
|
96
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'chunked,clen,http10,do_clen' => 'Chunked Header and Content-length and HTTP/1.0. Not served chunked.'], |
|
97
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'clen,chunked,http10,do_clen' => 'Content-length Header and Chunked and HTTP/1.0. Not served chunked.'], |
|
98
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'chunked,http10,do_clen,gzip' => 'Chunked Header and HTTP/1.0. Not served chunked. Compressed with gzip.'], |
|
99
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'chunked,clen,http10,do_clen,gzip' => 'Chunked Header and Content-length and HTTP/1.0. Not served chunked. Compressed with gzip.'], |
|
100
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'clen,chunked,http10,do_clen,gzip' => 'Content-length Header and Chunked and HTTP/1.0. Not served chunked. Compressed with gzip.'], |
|
101
|
|
|
|
|
|
|
|
|
102
|
|
|
|
|
|
|
[ 'INVALID: chunking with invalid HTTP versions' ], |
|
103
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/1.2' => 'Chunked Header and HTTP/1.2. Served chunked.'], |
|
104
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/1.2,do_clen' => 'Chunked Header and HTTP/1.2. Not served chunked.'], |
|
105
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/2.0' => 'Chunked Header and HTTP/2.0. Served chunked.'], |
|
106
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/2.0,do_clen' => 'Chunked Header and HTTP/2.0. erved chunked.'], |
|
107
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/2.1' => 'Chunked Header and HTTP/2.1. Served chunked.'], |
|
108
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/2.1,do_clen' => 'Chunked Header and HTTP/2.1. erved chunked.'], |
|
109
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/0.9' => 'Chunked Header and HTTP/0.9. Served chunked.'], |
|
110
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/0.9,do_clen' => 'Chunked Header and HTTP/0.9. Not served chunked.'], |
|
111
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/1.01' => 'Chunked Header and HTTP/1.01. Served chunked.'], |
|
112
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/1.01,do_clen' => 'Chunked Header and HTTP/1.01. Not served chunked.'], |
|
113
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/1.10' => 'Chunked Header and HTTP/1.10. Served chunked.'], |
|
114
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/1.10,do_clen' => 'Chunked Header and HTTP/1.10. Not served chunked.'], |
|
115
|
|
|
|
|
|
|
[ INVALID, 'chunked,http/1.1' => 'Chunked Header and http/1.1. Served chunked.'], |
|
116
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,http/1.1,do_clen' => 'Chunked Header and http/1.1. Not served chunked.'], |
|
117
|
|
|
|
|
|
|
[ INVALID, 'chunked,http/1.0' => 'Chunked Header and http/1.0. Served chunked.'], |
|
118
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,http/1.0,do_clen' => 'Chunked Header and http/1.0. Not served chunked.'], |
|
119
|
|
|
|
|
|
|
|
|
120
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/0.1,gzip' => 'Chunked Header and HTTP/0.1. Served chunked and with gzip.'], |
|
121
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/0.1,gzip,do_clen' => 'Chunked Header and HTTP/0.1. Not served chunked but with gzip.'], |
|
122
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/01.1,gzip' => 'Chunked Header and HTTP/01.1. Served chunked and with gzip.'], |
|
123
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/01.1,gzip,do_clen' => 'Chunked Header and HTTP/01.1. Not served chunked but with gzip.'], |
|
124
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/11.01,gzip' => 'Chunked Header and HTTP/11.01. Served chunked and with gzip.'], |
|
125
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/11.01,gzip,do_clen' => 'Chunked Header and HTTP/11.01. Not served chunked but with gzip.'], |
|
126
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/11.10,gzip' => 'Chunked Header and HTTP/11.10. Served chunked and with gzip.'], |
|
127
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/11.10,gzip,do_clen' => 'Chunked Header and HTTP/11.10. Not served chunked but with gzip.'], |
|
128
|
|
|
|
|
|
|
[ INVALID, 'chunked,HTTP/9.9,gzip' => 'Chunked Header and HTTP/9.9. Served chunked and with gzip.'], |
|
129
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,HTTP/9.9,gzip,do_clen' => 'Chunked Header and HTTP/9.9. Not served chunked but with gzip.'], |
|
130
|
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
[ 'VALID: valid variations on "chunked" value' ], |
|
132
|
|
|
|
|
|
|
[ VALID, 'chUnked' => 'mixed case "chUnked", served chunked'], |
|
133
|
|
|
|
|
|
|
[ UNCOMMON_VALID,'nl-chunked' => "chunked header with continuation line, served chunked"], |
|
134
|
|
|
|
|
|
|
[ UNCOMMON_VALID,'chunkednl-' => "chunked header followed by empty with continuation line, served chunked"], |
|
135
|
|
|
|
|
|
|
[ UNCOMMON_VALID,'nl-nl-chunked' => "chunked header with double continuation line, served chunked"], |
|
136
|
|
|
|
|
|
|
[ UNCOMMON_VALID,'nl-nl-chunked-nl-' => "chunked header with double continuation line and continuation afterwareds, served chunked"], |
|
137
|
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
[ 'INVALID: invalid variations on "chunked" value' ], |
|
139
|
|
|
|
|
|
|
[ INVALID, 'chu' => '"chu" not "chunked"'], |
|
140
|
|
|
|
|
|
|
[ INVALID, 'chunked-semicolon' => '"Transfer-Encoding: chunked;"' ], |
|
141
|
|
|
|
|
|
|
[ INVALID, 'xchunked' => '"xchunked" not "chunked"'], |
|
142
|
|
|
|
|
|
|
[ INVALID, 'chunkedx' => '"chunkedx" not "chunked"'], |
|
143
|
|
|
|
|
|
|
[ INVALID, 'chunked-x' => '"chunked x" not "chunked"'], |
|
144
|
|
|
|
|
|
|
[ INVALID, 'x-chunked' => '"x chunked" not "chunked"'], |
|
145
|
|
|
|
|
|
|
[ UNCOMMON_INVALID, 'chunked-x,do_clen' => '"chunked x" not "chunked", not served chunked'], |
|
146
|
|
|
|
|
|
|
[ UNCOMMON_INVALID, 'x-chunked,do_clen' => '"x chunked" not "chunked", not served chunked'], |
|
147
|
|
|
|
|
|
|
[ INVALID, 'x-nl-chunked' => '"x-folding-chunked" not "chunked"'], |
|
148
|
|
|
|
|
|
|
[ INVALID, 'chunked-nl-x' => '"chunked-folding-x" not "chunked"'], |
|
149
|
|
|
|
|
|
|
[ INVALID, 'rfc2047,do_chunked' => 'rfc2047/base64 encoded "chunked", served chunked' ], |
|
150
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'rfc2047,do_clen' => 'rfc2047/base64 encoded "chunked", not served chunked' ], |
|
151
|
|
|
|
|
|
|
[ UNCOMMON_VALID, 'rfc2047,clen,do_clen' => 'rfc2047/base64 encoded "chunked" and content-length, not served chunked' ], |
|
152
|
|
|
|
|
|
|
[ INVALID,'nl-chunked,do_clen' => "chunked header with continuation line. Not served chunked."], |
|
153
|
|
|
|
|
|
|
[ INVALID,'chunkednl-,do_clen' => "chunked header followed by empty continuation line. Not served chunked."], |
|
154
|
|
|
|
|
|
|
[ INVALID,'nl-nl-chunked,do_clen' => "chunked header with double continuation line, not served chunked"], |
|
155
|
|
|
|
|
|
|
[ INVALID,'crchunked,do_chunked' => "Transfer-Encoding:chunked. Served chunked."], |
|
156
|
|
|
|
|
|
|
[ INVALID,'crchunked,do_clen' => "Transfer-Encoding:chunked. Not served chunked."], |
|
157
|
|
|
|
|
|
|
[ INVALID,'cr-chunked,do_chunked' => "Transfer-Encoding:chunked. Served chunked."], |
|
158
|
|
|
|
|
|
|
[ INVALID,'cr-chunked,do_clen' => "Transfer-Encoding:chunked. Not served chunked."], |
|
159
|
|
|
|
|
|
|
[ INVALID,'chunkedcr-,do_chunked' => "Transfer-Encoding:chunked. Served chunked."], |
|
160
|
|
|
|
|
|
|
[ INVALID,'chunkedcr-,do_clen' => "Transfer-Encoding:chunked. Not served chunked."], |
|
161
|
|
|
|
|
|
|
[ INVALID,'ce-chunked,do_chunked' => "Content-encoding chunked instead of Transfer-encoding. Served chunked."], |
|
162
|
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
[ 'INVALID: hiding with another Transfer-Encoding header' ], |
|
164
|
|
|
|
|
|
|
[ INVALID, 'xte,chunked,do_chunked' => "double Transfer-Encoding: first junk, last chunked. Served chunked." ], |
|
165
|
|
|
|
|
|
|
[ INVALID, 'xte,chunked,do_chunked,gzip' => "double Transfer-Encoding: first junk, last chunked. Served chunked and gzipped." ], |
|
166
|
|
|
|
|
|
|
[ INVALID, 'chunked,xte,do_chunked' => "double Transfer-Encoding: first chunked, last junk. Served chunked." ], |
|
167
|
|
|
|
|
|
|
[ INVALID, 'chunked,xte,do_chunked,gzip' => "double Transfer-Encoding: first chunked, last junk. Served chunked and gzipped." ], |
|
168
|
|
|
|
|
|
|
[ INVALID, 'xte,chunked,xte,do_chunked' => "triple Transfer-Encoding: first junk, then chunked, then junk again. Served chunked." ], |
|
169
|
|
|
|
|
|
|
[ INVALID, 'xte,chunked,do_clen' => "double Transfer-Encoding: first junk, last chunked. Not served chunked." ], |
|
170
|
|
|
|
|
|
|
[ INVALID, 'chunked,xte,do_clen' => "double Transfer-Encoding: first chunked, last junk. Not served chunked." ], |
|
171
|
|
|
|
|
|
|
[ INVALID, 'chunked,xte,clen,do_chunked' => "double Transfer-Encoding: first chunked, last junk. Also Content-length header. Served chunked." ], |
|
172
|
|
|
|
|
|
|
[ INVALID, 'xte,chunked,clen,do_chunked' => "double Transfer-Encoding: first junk, last chunked. Also Content-length header. Served chunked." ], |
|
173
|
|
|
|
|
|
|
[ INVALID, 'xte,chunked,xte,clen,do_chunked' => "triple Transfer-Encoding: first junk, then chunked, then junk again. Also Content-length header. Served chunked." ], |
|
174
|
|
|
|
|
|
|
[ INVALID, 'chunked,xte,clen,do_clen' => "double Transfer-Encoding: first chunked, last junk. Also Content-length header. Not served chunked." ], |
|
175
|
|
|
|
|
|
|
[ INVALID, 'chunked,xte,clen,do_clen,gzip' => "double Transfer-Encoding: first chunked, last junk. Also Content-length header. Not served chunked. Compressed with gzip." ], |
|
176
|
|
|
|
|
|
|
[ INVALID, 'xte,chunked,clen,do_clen' => "double Transfer-Encoding: first junk, last chunked. Also Content-length header. Not served chunked." ], |
|
177
|
|
|
|
|
|
|
[ INVALID, 'xte,chunked,clen,do_clen,gzip' => "double Transfer-Encoding: first junk, last chunked. Also Content-length header. Not served chunked. Compressed with gzip." ], |
|
178
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,do_clen' => 'chunking and content-length, not served chunked'], |
|
179
|
|
|
|
|
|
|
[ INVALID, 'chunked,clen,do_clen,gzip' => 'chunking and content-length, not served chunked. Compressed with gzip.'], |
|
180
|
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
[ 'INVALID: hiding the Transfer-Encoding header' ], |
|
182
|
|
|
|
|
|
|
[ INVALID, 'space-colon-chunked,do_chunked' => '"Transfer-Encoding:", served chunked' ], |
|
183
|
|
|
|
|
|
|
[ INVALID, 'tab-colon-chunked,do_chunked' => '"Transfer-Encoding:", served chunked' ], |
|
184
|
|
|
|
|
|
|
[ INVALID, 'cr-colon-chunked,do_chunked' => '"Transfer-Encoding:", served chunked' ], |
|
185
|
|
|
|
|
|
|
[ UNCOMMON_INVALID, 'space-colon-chunked,do_clen' => '"Transfer-Encoding:", not served chunked' ], |
|
186
|
|
|
|
|
|
|
[ UNCOMMON_INVALID, 'tab-colon-chunked,do_clen' => '"Transfer-Encoding:", not served chunked' ], |
|
187
|
|
|
|
|
|
|
[ UNCOMMON_INVALID, 'cr-colon-chunked,do_clen' => '"Transfer-Encoding:", not served chunked' ], |
|
188
|
|
|
|
|
|
|
[ INVALID, 'colon-colon-chunked,do_chunked' => '"Transfer-Encoding::", served chunked' ], |
|
189
|
|
|
|
|
|
|
[ UNCOMMON_INVALID, 'colon-colon-chunked,do_clen' => '"Transfer-Encoding::", not served chunked' ], |
|
190
|
|
|
|
|
|
|
[ INVALID, 'cronly-chunked,do_chunked' => 'Transfer-Encoding with only as line delimiter before, served chunked' ], |
|
191
|
|
|
|
|
|
|
[ INVALID, 'crxonly-chunked,do_chunked' => 'Only as line delimiter followed by "xTransfer-Encoding", served chunked' ], |
|
192
|
|
|
|
|
|
|
[ UNCOMMON_INVALID, 'cronly-chunked,do_clen' => 'Transfer-Encoding with only as line delimiter before, not served chunked' ], |
|
193
|
|
|
|
|
|
|
[ UNCOMMON_INVALID, 'lfonly-chunked,do_chunked' => 'Transfer-Encoding with only as line delimiter before, served chunked' ], |
|
194
|
|
|
|
|
|
|
[ INVALID, 'lfonly-chunked,do_clen' => 'Transfer-Encoding with only as line delimiter before, not served chunked' ], |
|
195
|
|
|
|
|
|
|
|
|
196
|
|
|
|
|
|
|
[ 'INVALID: invalid chunks' ], |
|
197
|
|
|
|
|
|
|
[ INVALID, 'chunk-lf' => "chunk with LF as delimiter instead of CRLF" ], |
|
198
|
|
|
|
|
|
|
[ INVALID, 'chunk-cr' => "chunk with CR as delimiter instead of CRLF" ], |
|
199
|
|
|
|
|
|
|
[ INVALID, 'chunk-crcr' => "chunk with CRCR as delimiter instead of CRLF" ], |
|
200
|
|
|
|
|
|
|
[ INVALID, 'chunk-lflf' => "chunk with LFLF as delimiter instead of CRLF" ], |
|
201
|
|
|
|
|
|
|
[ INVALID, 'chunk-lfcr' => "chunk with LFCR as delimiter instead of CRLF" ], |
|
202
|
|
|
|
|
|
|
[ INVALID, 'nofinal' => 'missing final chunk' ], |
|
203
|
|
|
|
|
|
|
[ INVALID, 'eof-inchunk' => 'eof inside some chunk' ], |
|
204
|
|
|
|
|
|
|
[ INVALID, 'space-before-chunks' => 'space before chunks start' ], |
|
205
|
|
|
|
|
|
|
[ INVALID, 'lf-before-chunks' => ' before chunks start' ], |
|
206
|
|
|
|
|
|
|
[ INVALID, 'cr-before-chunks' => ' before chunks start' ], |
|
207
|
|
|
|
|
|
|
[ INVALID, 'crlf-before-chunks' => ' before chunks start' ], |
|
208
|
|
|
|
|
|
|
); |
|
209
|
|
|
|
|
|
|
|
|
210
|
|
|
|
|
|
|
|
|
211
|
|
|
|
|
|
|
sub make_response { |
|
212
|
0
|
|
|
0
|
0
|
|
my ($self,$page,$spec) = @_; |
|
213
|
0
|
0
|
|
|
|
|
return make_index_page() if $page eq ''; |
|
214
|
0
|
0
|
|
|
|
|
my ($hdr,$data) = content($page,$self->ID."-".$spec) or die "unknown page $page"; |
|
215
|
0
|
|
|
|
|
|
my $version = 'HTTP/1.1'; |
|
216
|
0
|
|
|
|
|
|
my ($te,@chunks,%chunkmod,$clen); |
|
217
|
0
|
|
|
|
|
|
my $sizefmt = '%x'; |
|
218
|
0
|
|
|
|
|
|
my $before_chunks = ''; |
|
219
|
0
|
|
|
|
|
|
my $final = '0'; |
|
220
|
0
|
|
|
|
|
|
my $finalchunk; |
|
221
|
0
|
|
|
|
|
|
for (split(',',$spec)) { |
|
222
|
0
|
0
|
|
|
|
|
if ( m{^(x|-|nl|lf|cr)*chunked(x|-|nl|lf|cr)*$}i ) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
223
|
0
|
|
|
|
|
|
s{-}{ }g; |
|
224
|
0
|
|
|
|
|
|
s{nl}{\r\n}g; |
|
225
|
0
|
|
|
|
|
|
s{lf}{\n}g; |
|
226
|
0
|
|
|
|
|
|
s{cr}{\r}g; |
|
227
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: $_\r\nConnection: close\r\n"; |
|
228
|
|
|
|
|
|
|
} elsif ( m{^(space|tab|cr|colon)-colon-chunked$} ) { |
|
229
|
0
|
|
|
|
|
|
my $c = $1; |
|
230
|
0
|
|
|
|
|
|
$c =~s{space}{ }g; |
|
231
|
0
|
|
|
|
|
|
$c =~s{colon}{:}g; |
|
232
|
0
|
|
|
|
|
|
$c =~s{tab}{\t}g; |
|
233
|
0
|
|
|
|
|
|
$c =~s{cr}{\r}g; |
|
234
|
0
|
|
|
|
|
|
$te = 'chunked'; |
|
235
|
0
|
|
|
|
|
|
$hdr .= "Connection: close\r\nTransfer-Encoding$c: chunked\r\n" |
|
236
|
|
|
|
|
|
|
} elsif ( my ($crlf) = m {^((?:lf|cr|x)+)only-chunked$} ) { |
|
237
|
0
|
|
|
|
|
|
$te = 'chunked'; |
|
238
|
0
|
0
|
|
|
|
|
$hdr = "X-Foo: bar" if $hdr !~s{\r\n\z}{}; |
|
239
|
0
|
|
|
|
|
|
$crlf =~s{lf}{\n}g; |
|
240
|
0
|
|
|
|
|
|
$crlf =~s{cr}{\r}g; |
|
241
|
0
|
|
|
|
|
|
$hdr .= $crlf . "Transfer-Encoding: chunked\r\nConnection: close\r\n"; |
|
242
|
|
|
|
|
|
|
} elsif ( $_ eq '1chunk' ) { |
|
243
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked\r\n"; |
|
244
|
0
|
|
|
|
|
|
@chunks = $data; |
|
245
|
|
|
|
|
|
|
} elsif ( $_ eq 'chu' ) { |
|
246
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chu\r\nConnection: close\r\n" |
|
247
|
|
|
|
|
|
|
} elsif ( $_ eq 'ce-chunked' ) { |
|
248
|
0
|
|
|
|
|
|
$hdr .= "Content-Encoding: chunked\r\nConnection: close\r\n" |
|
249
|
|
|
|
|
|
|
} elsif ( $_ =~ m{^clen(\d+)?$} ) { |
|
250
|
0
|
|
0
|
|
|
|
$clen = $1 || 100; |
|
251
|
|
|
|
|
|
|
} elsif ( $_ eq 'http10' ) { |
|
252
|
0
|
|
|
|
|
|
$version = "HTTP/1.0"; |
|
253
|
|
|
|
|
|
|
} elsif ( $_ =~m{^HTTP/\S+}i ) { |
|
254
|
0
|
|
|
|
|
|
$version = $_; |
|
255
|
|
|
|
|
|
|
} elsif ( $_ eq 'do_clen' ) { |
|
256
|
0
|
|
|
|
|
|
$te = 'clen' |
|
257
|
|
|
|
|
|
|
} elsif ( $_ eq 'do_chunked' ) { |
|
258
|
0
|
|
|
|
|
|
$te = 'chunked' |
|
259
|
|
|
|
|
|
|
} elsif ( $_ eq 'chunked-semicolon' ) { |
|
260
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked;\r\nConnection: close\r\n" |
|
261
|
|
|
|
|
|
|
} elsif ( $_ eq 'rfc2047' ) { |
|
262
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: =?UTF-8?B?Y2h1bmtlZAo=?=\r\nConnection: close\r\n"; |
|
263
|
|
|
|
|
|
|
} elsif ( $_ eq 'xte' ) { |
|
264
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: lalala\r\nConnection: close\r\n"; |
|
265
|
|
|
|
|
|
|
} elsif ( m{^(chunk-ext-|nofinal$|eof-inchunk$)} ) { |
|
266
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked\r\nConnection: close\r\n"; |
|
267
|
0
|
|
|
|
|
|
$chunkmod{$_} = 1; |
|
268
|
|
|
|
|
|
|
} elsif ( my ($eol) = m{^chunk-((?:lf|cr)+)$} ) { |
|
269
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked\r\nConnection: close\r\n"; |
|
270
|
0
|
|
|
|
|
|
$eol =~s{cr}{\r}g; |
|
271
|
0
|
|
|
|
|
|
$eol =~s{lf}{\n}g; |
|
272
|
0
|
|
|
|
|
|
$chunkmod{lineend} = $eol; |
|
273
|
|
|
|
|
|
|
} elsif (m{^(32|64)-size\z}) { |
|
274
|
0
|
0
|
|
|
|
|
my $o = ($1 == 64) ? 'ffffffff':''; |
|
275
|
0
|
|
|
0
|
|
|
$sizefmt = sub { sprintf("-$o%08x", 1+(0xffffffff & ~shift())) }; |
|
|
0
|
|
|
|
|
|
|
|
276
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked\r\nConnection: close\r\n"; |
|
277
|
|
|
|
|
|
|
} elsif ( m{^(-|space|cr|lf|tab|x|\\[0-7]{3})*(0*)(uc)?size(-|space|cr|lf|tab|x|\\[0-7]{3})*$}) { |
|
278
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked\r\nConnection: close\r\n"; |
|
279
|
0
|
0
|
|
|
|
|
@chunks = ( $data =~m{(.{1,15})}smg,'') if ! @chunks; |
|
280
|
0
|
|
|
|
|
|
s{ucsize}{%X}; |
|
281
|
0
|
|
|
|
|
|
s{size}{%x}; |
|
282
|
0
|
|
|
|
|
|
s{\\r}{\r}g; |
|
283
|
0
|
|
|
|
|
|
s{\\n}{\n}g; |
|
284
|
0
|
|
|
|
|
|
s{-}{}g; |
|
285
|
0
|
|
|
|
|
|
s{space}{ }g; |
|
286
|
0
|
|
|
|
|
|
s{tab}{\t}g; |
|
287
|
0
|
|
|
|
|
|
s{cr}{\r}g; |
|
288
|
0
|
|
|
|
|
|
s{lf}{\n}g; |
|
289
|
0
|
|
|
|
|
|
s{\\([0-7]{3})}{ chr(oct($1)) }eg; |
|
|
0
|
|
|
|
|
|
|
|
290
|
0
|
|
|
|
|
|
$sizefmt = $_; |
|
291
|
|
|
|
|
|
|
} elsif (m{((?:space|tab|cr|lf)*)-before-chunks}) { |
|
292
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked\r\nConnection: close\r\n"; |
|
293
|
0
|
|
|
|
|
|
$before_chunks = $1; |
|
294
|
0
|
|
|
|
|
|
$before_chunks =~ s{space}{ }g; |
|
295
|
0
|
|
|
|
|
|
$before_chunks =~ s{tab}{\t}g; |
|
296
|
0
|
|
|
|
|
|
$before_chunks =~ s{cr}{\r}g; |
|
297
|
0
|
|
|
|
|
|
$before_chunks =~ s{lf}{\n}g; |
|
298
|
|
|
|
|
|
|
} elsif (m{^final=(.*)$}) { |
|
299
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked\r\nConnection: close\r\n"; |
|
300
|
0
|
|
|
|
|
|
$final = $1; |
|
301
|
|
|
|
|
|
|
} elsif (m{^finalchunk=(.*)$}) { |
|
302
|
0
|
|
|
|
|
|
$hdr .= "Transfer-Encoding: chunked\r\nConnection: close\r\n"; |
|
303
|
0
|
|
|
|
|
|
(my $d = $1 ) =~ s{\\([0-7]{3})}{ chr(oct($1)) }eg; |
|
|
0
|
|
|
|
|
|
|
|
304
|
0
|
|
|
|
|
|
$finalchunk = $d; |
|
305
|
|
|
|
|
|
|
} elsif ( $_ eq 'addjunk' ) { |
|
306
|
0
|
|
|
|
|
|
$data .= "x" x length($data); |
|
307
|
|
|
|
|
|
|
} elsif ( $_ eq 'gzip' ) { |
|
308
|
0
|
|
|
|
|
|
$data = _compress($data,'gzip'); |
|
309
|
0
|
|
|
|
|
|
$hdr .= "Content-Encoding: gzip\r\n"; |
|
310
|
|
|
|
|
|
|
} else { |
|
311
|
0
|
|
|
|
|
|
die $_ |
|
312
|
|
|
|
|
|
|
} |
|
313
|
|
|
|
|
|
|
} |
|
314
|
0
|
0
|
|
|
|
|
$hdr .= "Content-length: ". int($clen/100*length($data)) ."\r\n" |
|
315
|
|
|
|
|
|
|
if defined $clen; |
|
316
|
0
|
|
|
|
|
|
$hdr = "$version 200 ok\r\n$hdr"; |
|
317
|
0
|
0
|
0
|
|
|
|
$te ||= $hdr =~m{^Transfer-Encoding:}im ? 'chunked':'clen'; |
|
318
|
0
|
0
|
0
|
|
|
|
@chunks = ( $data =~m{(.{1,5})}smg,'') if $te eq 'chunked' && ! @chunks; |
|
319
|
0
|
0
|
|
|
|
|
if (@chunks) { |
|
320
|
0
|
|
|
|
|
|
@chunks = map { [ length($_), $_ ] } @chunks; |
|
|
0
|
|
|
|
|
|
|
|
321
|
0
|
|
0
|
|
|
|
my $nl = $chunkmod{lineend} || "\r\n"; |
|
322
|
0
|
0
|
|
|
|
|
if ($chunkmod{'chunk-ext-chunk'}) { |
|
|
|
0
|
|
|
|
|
|
|
323
|
0
|
|
|
|
|
|
$_->[2] = sprintf("; %s %x","x" x $_->[0],$_->[0]) for @chunks; |
|
324
|
|
|
|
|
|
|
} elsif ($chunkmod{'chunk-ext-junk'}) { |
|
325
|
0
|
|
|
|
|
|
$_->[2] = "; foobar" for @chunks; |
|
326
|
|
|
|
|
|
|
} |
|
327
|
0
|
0
|
0
|
|
|
|
pop @chunks if $chunkmod{nofinal} && ! $chunks[-1][0]; |
|
328
|
|
|
|
|
|
|
|
|
329
|
0
|
|
|
|
|
|
my $end = ''; |
|
330
|
0
|
0
|
|
|
|
|
if ($chunkmod{'eof-inchunk'}) { |
|
331
|
0
|
0
|
|
|
|
|
pop @chunks if ! $chunks[-1][0]; # remove final chunk |
|
332
|
0
|
|
|
|
|
|
my $last = pop(@chunks); |
|
333
|
0
|
|
0
|
|
|
|
$end = sprintf("%x%s%s%s",$last->[0]+10,$last->[2]||'',$nl,$last->[1]); |
|
334
|
|
|
|
|
|
|
} |
|
335
|
|
|
|
|
|
|
|
|
336
|
0
|
0
|
|
|
|
|
$finalchunk = "$final$nl$nl" if ! defined $finalchunk; |
|
337
|
|
|
|
|
|
|
$data = join("",map { |
|
338
|
0
|
0
|
0
|
|
|
|
$_->[0] ? sprintf("%s%s%s%s%s", |
|
|
0
|
0
|
|
|
|
|
|
|
339
|
|
|
|
|
|
|
ref($sizefmt) ? $sizefmt->($_->[0]): sprintf($sizefmt,$_->[0]), # size |
|
340
|
|
|
|
|
|
|
$_->[2] || '', # ext |
|
341
|
|
|
|
|
|
|
$nl, |
|
342
|
|
|
|
|
|
|
$_->[1], |
|
343
|
|
|
|
|
|
|
$nl |
|
344
|
|
|
|
|
|
|
) : $finalchunk |
|
345
|
|
|
|
|
|
|
} @chunks).$end; |
|
346
|
|
|
|
|
|
|
} |
|
347
|
0
|
|
|
|
|
|
return "$hdr\r\n$before_chunks$data"; |
|
348
|
|
|
|
|
|
|
} |
|
349
|
|
|
|
|
|
|
|
|
350
|
|
|
|
|
|
|
sub _compress { |
|
351
|
0
|
|
|
0
|
|
|
my ($data,$w) = @_; |
|
352
|
0
|
0
|
|
|
|
|
my $zlib = Compress::Raw::Zlib::Deflate->new( |
|
353
|
|
|
|
|
|
|
-WindowBits => $w eq 'gzip' ? WANT_GZIP : -MAX_WBITS(), |
|
354
|
|
|
|
|
|
|
-AppendOutput => 1, |
|
355
|
|
|
|
|
|
|
); |
|
356
|
0
|
|
|
|
|
|
my $newdata = ''; |
|
357
|
0
|
|
|
|
|
|
$zlib->deflate( $data, $newdata); |
|
358
|
0
|
|
|
|
|
|
$zlib->flush($newdata,Z_FINISH); |
|
359
|
0
|
|
|
|
|
|
return $newdata; |
|
360
|
|
|
|
|
|
|
} |
|
361
|
|
|
|
|
|
|
|
|
362
|
|
|
|
|
|
|
1; |