Skip to content

Commit 39ad0fd

Browse files
committed
net/http: fix validHeaderValue yet again, excluding the DEL CTL byte
Third time's a charm. Thanks to Ralph Corderoy for noticing the DEL omission. Update #11207 Change-Id: I174fd01eaecceae1eb220f2c9136e12d40fbe943 Reviewed-on: https://go-review.googlesource.com/18375 Reviewed-by: Russ Cox <[email protected]> Run-TryBot: Brad Fitzpatrick <[email protected]> TryBot-Result: Gobot Gobot <[email protected]>
1 parent ecc01a7 commit 39ad0fd

File tree

3 files changed

+35
-2
lines changed

3 files changed

+35
-2
lines changed

src/net/http/lex.go

+14
Original file line numberDiff line numberDiff line change
@@ -167,3 +167,17 @@ func tokenEqual(t1, t2 string) bool {
167167
}
168168
return true
169169
}
170+
171+
// isLWS reports whether b is linear white space, according
172+
// to http://www.w3.org/Protocols/rfc2616/rfc2616-sec2.html#sec2.2
173+
// LWS = [CRLF] 1*( SP | HT )
174+
func isLWS(b byte) bool { return b == ' ' || b == '\t' }
175+
176+
// isCTL reports whether b is a control byte, according
177+
// to http://www.w3.org/Protocols/rfc2616/rfc2616-sec2.html#sec2.2
178+
// CTL = <any US-ASCII control character
179+
// (octets 0 - 31) and DEL (127)>
180+
func isCTL(b byte) bool {
181+
const del = 0x7f // a CTL
182+
return b < ' ' || b == del
183+
}

src/net/http/request.go

+17-1
Original file line numberDiff line numberDiff line change
@@ -1136,10 +1136,26 @@ func validHeaderName(v string) bool {
11361136
return strings.IndexFunc(v, isNotToken) == -1
11371137
}
11381138

1139+
// validHeaderValue reports whether v is a valid "field-value" according to
1140+
// http://www.w3.org/Protocols/rfc2616/rfc2616-sec4.html#sec4.2 :
1141+
//
1142+
// message-header = field-name ":" [ field-value ]
1143+
// field-value = *( field-content | LWS )
1144+
// field-content = <the OCTETs making up the field-value
1145+
// and consisting of either *TEXT or combinations
1146+
// of token, separators, and quoted-string>
1147+
//
1148+
// http://www.w3.org/Protocols/rfc2616/rfc2616-sec2.html#sec2.2 :
1149+
//
1150+
// TEXT = <any OCTET except CTLs,
1151+
// but including LWS>
1152+
// LWS = [CRLF] 1*( SP | HT )
1153+
// CTL = <any US-ASCII control character
1154+
// (octets 0 - 31) and DEL (127)>
11391155
func validHeaderValue(v string) bool {
11401156
for i := 0; i < len(v); i++ {
11411157
b := v[i]
1142-
if b < ' ' && b != '\t' {
1158+
if isCTL(b) && !isLWS(b) {
11431159
return false
11441160
}
11451161
}

src/net/http/serve_test.go

+4-1
Original file line numberDiff line numberDiff line change
@@ -3798,7 +3798,10 @@ func TestServerValidatesHeaders(t *testing.T) {
37983798
{"foo\xffbar: foo\r\n", 400}, // binary in header
37993799
{"foo\x00bar: foo\r\n", 400}, // binary in header
38003800

3801-
{"foo: foo\x00foo\r\n", 400}, // CTL in value is bad
3801+
{"foo: foo foo\r\n", 200}, // LWS space is okay
3802+
{"foo: foo\tfoo\r\n", 200}, // LWS tab is okay
3803+
{"foo: foo\x00foo\r\n", 400}, // CTL 0x00 in value is bad
3804+
{"foo: foo\x7ffoo\r\n", 400}, // CTL 0x7f in value is bad
38023805
{"foo: foo\xfffoo\r\n", 200}, // non-ASCII high octets in value are fine
38033806
}
38043807
for _, tt := range tests {

0 commit comments

Comments
 (0)