Skip to content

Commit 49ca841

Browse files
icingbagder
authored andcommitted
websockets: check for negative payload lengths
- in en- and decoding, check the websocket frame payload lengths for negative values (from curl_off_t) and error the operation in that case - add test 2307 to verify Closes #12707
1 parent 9034a16 commit 49ca841

File tree

4 files changed

+85
-1
lines changed

4 files changed

+85
-1
lines changed

lib/ws.c

+11
Original file line numberDiff line numberDiff line change
@@ -225,6 +225,10 @@ static CURLcode ws_dec_read_head(struct ws_decoder *dec,
225225
dec->payload_len = (dec->head[2] << 8) | dec->head[3];
226226
break;
227227
case 10:
228+
if(dec->head[2] > 127) {
229+
failf(data, "WS: frame length longer than 64 signed not supported");
230+
return CURLE_RECV_ERROR;
231+
}
228232
dec->payload_len = ((curl_off_t)dec->head[2] << 56) |
229233
(curl_off_t)dec->head[3] << 48 |
230234
(curl_off_t)dec->head[4] << 40 |
@@ -410,6 +414,13 @@ static ssize_t ws_enc_write_head(struct Curl_easy *data,
410414
size_t hlen;
411415
ssize_t n;
412416

417+
if(payload_len < 0) {
418+
failf(data, "WS: starting new frame with negative payload length %"
419+
CURL_FORMAT_CURL_OFF_T, payload_len);
420+
*err = CURLE_SEND_ERROR;
421+
return -1;
422+
}
423+
413424
if(enc->payload_remain > 0) {
414425
/* trying to write a new frame before the previous one is finished */
415426
failf(data, "WS: starting new frame with %zd bytes from last one"

tests/data/DISABLED

+2
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,8 @@
8181
2301
8282
2302
8383
2305
84+
# response body seem not to be handled by hyper
85+
2307
8486
%endif
8587
2043
8688
# The CRL test (313) doesn't work with rustls because rustls doesn't support

tests/data/Makefile.inc

+1-1
Original file line numberDiff line numberDiff line change
@@ -244,7 +244,7 @@ test2100 \
244244
\
245245
test2200 test2201 test2202 test2203 test2204 test2205 \
246246
\
247-
test2300 test2301 test2302 test2303 test2304 test2305 test2306 \
247+
test2300 test2301 test2302 test2303 test2304 test2305 test2306 test2307 \
248248
\
249249
test2400 test2401 test2402 test2403 test2404 \
250250
\

tests/data/test2307

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
<testcase>
2+
<info>
3+
<keywords>
4+
WebSockets
5+
</keywords>
6+
</info>
7+
8+
#
9+
# Sends a PING with overlong payload
10+
<reply>
11+
<data nocheck="yes" nonewline="yes">
12+
HTTP/1.1 101 Switching to WebSockets
13+
Server: test-server/fake
14+
Upgrade: websocket
15+
Connection: Upgrade
16+
Something: else
17+
Sec-WebSocket-Accept: HkPsVga7+8LuxM4RGQ5p9tZHeYs=
18+
19+
%hex[%19%7f%ff%30%30%30%30%30%30%30%30%30%30%30%30]hex%
20+
</data>
21+
# allow upgrade
22+
<servercmd>
23+
upgrade
24+
</servercmd>
25+
</reply>
26+
27+
#
28+
# Client-side
29+
<client>
30+
# require debug for the forced CURL_ENTROPY
31+
<features>
32+
debug
33+
ws
34+
!hyper
35+
</features>
36+
<server>
37+
http
38+
</server>
39+
<name>
40+
WebSockets, overlong PING payload
41+
</name>
42+
<tool>
43+
lib2302
44+
</tool>
45+
<command>
46+
ws://%HOSTIP:%HTTPPORT/%TESTNUMBER
47+
</command>
48+
</client>
49+
50+
#
51+
# PONG with no data and the 32 bit mask
52+
#
53+
<verify>
54+
<protocol nocheck="yes" nonewline="yes">
55+
GET /%TESTNUMBER HTTP/1.1
56+
Host: %HOSTIP:%HTTPPORT
57+
User-Agent: webbie-sox/3
58+
Accept: */*
59+
Upgrade: websocket
60+
Connection: Upgrade
61+
Sec-WebSocket-Version: 13
62+
Sec-WebSocket-Key: NDMyMTUzMjE2MzIxNzMyMQ==
63+
64+
65+
</protocol>
66+
# 23 == CURLE_WRITE_ERROR
67+
<errorcode>
68+
23
69+
</errorcode>
70+
</verify>
71+
</testcase>

0 commit comments

Comments
 (0)