mirror of
https://github.com/curl/curl.git
synced 2024-12-09 06:30:06 +08:00
http: restore header folding behavior
Folded header lines will now get passed through like before. The headers API is adapted and will provide the content unfolded. Added test 1274 and extended test 1940 to verify. Reported-by: Petr Pisar Fixes #8844 Closes #8899
This commit is contained in:
parent
16a58e9f93
commit
c9b60f0053
@ -94,7 +94,9 @@ but it might have a different case.
|
||||
|
||||
The data \fBvalue\fP field points to, comes exactly as delivered over the
|
||||
network but with leading and trailing whitespace and newlines stripped
|
||||
off. The `value` data is nul-terminated.
|
||||
off. The `value` data is nul-terminated. For legacy HTTP/1 "folded headers",
|
||||
this API provides the full single value in an unfolded manner with a single
|
||||
whitespace between the lines.
|
||||
|
||||
\fBamount\fP is how many headers using this name that exist, within the origin
|
||||
and request scope asked for.
|
||||
|
@ -216,6 +216,54 @@ static CURLcode namevalue(char *header, size_t hlen, unsigned int type,
|
||||
return CURLE_OK;
|
||||
}
|
||||
|
||||
static CURLcode append_value(struct Curl_easy *data, const char *value,
|
||||
size_t vlen) /* length of the incoming header */
|
||||
{
|
||||
struct Curl_header_store *hs;
|
||||
struct Curl_header_store *newhs;
|
||||
size_t olen; /* length of the old value */
|
||||
size_t offset;
|
||||
DEBUGASSERT(data->state.prevhead);
|
||||
hs = data->state.prevhead;
|
||||
olen = strlen(hs->value);
|
||||
offset = hs->value - hs->buffer;
|
||||
|
||||
/* skip all trailing space letters */
|
||||
while(vlen && ISSPACE(value[vlen - 1]))
|
||||
vlen--;
|
||||
|
||||
/* save only one leading space */
|
||||
while((vlen > 1) && ISSPACE(value[0]) && ISSPACE(value[1])) {
|
||||
vlen--;
|
||||
value++;
|
||||
}
|
||||
|
||||
/* since this header block might move in the realloc below, it needs to
|
||||
first be unlinked from the list and then re-added again after the
|
||||
realloc */
|
||||
Curl_llist_remove(&data->state.httphdrs, &hs->node, NULL);
|
||||
|
||||
newhs = Curl_saferealloc(hs, sizeof(*hs) + vlen + olen + 1);
|
||||
if(!newhs)
|
||||
return CURLE_OUT_OF_MEMORY;
|
||||
/* ->name' and ->value point into ->buffer (to keep the header allocation
|
||||
in a single memory block), which now potentially have moved. Adjust
|
||||
them. */
|
||||
newhs->name = newhs->buffer;
|
||||
newhs->value = &newhs->buffer[offset];
|
||||
|
||||
/* put the data at the end of the previous data, not the newline */
|
||||
memcpy(&newhs->value[olen], value, vlen);
|
||||
newhs->value[olen + vlen] = 0; /* zero terminate at newline */
|
||||
|
||||
/* insert this node into the list of headers */
|
||||
Curl_llist_insert_next(&data->state.httphdrs, data->state.httphdrs.tail,
|
||||
newhs, &newhs->node);
|
||||
data->state.prevhead = newhs;
|
||||
return CURLE_OK;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* Curl_headers_push() gets passed a full HTTP header to store. It gets called
|
||||
* immediately before the header callback. The header is CRLF terminated.
|
||||
@ -242,6 +290,10 @@ CURLcode Curl_headers_push(struct Curl_easy *data, const char *header,
|
||||
}
|
||||
hlen = end - header + 1;
|
||||
|
||||
if((header[0] == ' ') || (header[0] == '\t'))
|
||||
/* line folding, append value to the previous header's value */
|
||||
return append_value(data, header, hlen);
|
||||
|
||||
hs = calloc(1, sizeof(*hs) + hlen);
|
||||
if(!hs)
|
||||
return CURLE_OUT_OF_MEMORY;
|
||||
@ -260,7 +312,7 @@ CURLcode Curl_headers_push(struct Curl_easy *data, const char *header,
|
||||
/* insert this node into the list of headers */
|
||||
Curl_llist_insert_next(&data->state.httphdrs, data->state.httphdrs.tail,
|
||||
hs, &hs->node);
|
||||
|
||||
data->state.prevhead = hs;
|
||||
return CURLE_OK;
|
||||
fail:
|
||||
free(hs);
|
||||
|
15
lib/http.c
15
lib/http.c
@ -3799,11 +3799,16 @@ static CURLcode verify_header(struct Curl_easy *data)
|
||||
if(k->headerline < 2)
|
||||
/* the first "header" is the status-line and it has no colon */
|
||||
return CURLE_OK;
|
||||
ptr = memchr(header, ':', hlen);
|
||||
if(!ptr) {
|
||||
/* this is bad, bail out */
|
||||
failf(data, "Header without colon");
|
||||
return CURLE_WEIRD_SERVER_REPLY;
|
||||
if(((header[0] == ' ') || (header[0] == '\t')) && k->headerline > 2)
|
||||
/* line folding, can't happen on line 2 */
|
||||
;
|
||||
else {
|
||||
ptr = memchr(header, ':', hlen);
|
||||
if(!ptr) {
|
||||
/* this is bad, bail out */
|
||||
failf(data, "Header without colon");
|
||||
return CURLE_WEIRD_SERVER_REPLY;
|
||||
}
|
||||
}
|
||||
return CURLE_OK;
|
||||
}
|
||||
|
@ -1423,6 +1423,7 @@ struct UrlState {
|
||||
headers */
|
||||
struct Curl_llist httphdrs; /* received headers */
|
||||
struct curl_header headerout; /* for external purposes */
|
||||
struct Curl_header_store *prevhead; /* the latest added header */
|
||||
#endif
|
||||
trailers_state trailers_state; /* whether we are sending trailers
|
||||
and what stage are we at */
|
||||
|
@ -160,7 +160,7 @@ test1240 test1241 test1242 test1243 test1244 test1245 test1246 test1247 \
|
||||
test1248 test1249 test1250 test1251 test1252 test1253 test1254 test1255 \
|
||||
test1256 test1257 test1258 test1259 test1260 test1261 test1262 test1263 \
|
||||
test1264 test1265 test1266 test1267 test1268 test1269 test1270 test1271 \
|
||||
test1272 test1273 \
|
||||
test1272 test1273 test1274 \
|
||||
\
|
||||
test1280 test1281 test1282 test1283 test1284 test1285 test1286 test1287 \
|
||||
test1288 test1289 test1290 test1291 test1292 test1293 test1294 test1295 \
|
||||
|
65
tests/data/test1274
Normal file
65
tests/data/test1274
Normal file
@ -0,0 +1,65 @@
|
||||
<testcase>
|
||||
<info>
|
||||
<keywords>
|
||||
HTTP
|
||||
HTTP GET
|
||||
header line folding
|
||||
</keywords>
|
||||
</info>
|
||||
|
||||
#
|
||||
# Server-side
|
||||
<reply>
|
||||
<data>
|
||||
HTTP/1.1 200 OK
|
||||
Date: Tue, 09 Nov 2010 14:49:00 GMT
|
||||
Server: test-server/
|
||||
fake
|
||||
folded
|
||||
Last-Modified: Tue, 13 Jun 2000 12:10:00 GMT
|
||||
ETag: "21025-dc7-39462498"
|
||||
Content-Length: 6
|
||||
Connection: close
|
||||
|
||||
-foo-
|
||||
</data>
|
||||
</reply>
|
||||
|
||||
#
|
||||
# Client-side
|
||||
<client>
|
||||
<server>
|
||||
http
|
||||
</server>
|
||||
<name>
|
||||
HTTP header line folding
|
||||
</name>
|
||||
<command>
|
||||
http://%HOSTIP:%HTTPPORT/%TESTNUMBER -D log/out%TESTNUMBER
|
||||
</command>
|
||||
</client>
|
||||
|
||||
#
|
||||
# Verify data after the test has been "shot"
|
||||
<verify>
|
||||
<protocol>
|
||||
GET /%TESTNUMBER HTTP/1.1
|
||||
Host: %HOSTIP:%HTTPPORT
|
||||
User-Agent: curl/%VERSION
|
||||
Accept: */*
|
||||
|
||||
</protocol>
|
||||
<file name="log/out%TESTNUMBER">
|
||||
HTTP/1.1 200 OK
|
||||
Date: Tue, 09 Nov 2010 14:49:00 GMT
|
||||
Server: test-server/
|
||||
fake
|
||||
folded
|
||||
Last-Modified: Tue, 13 Jun 2000 12:10:00 GMT
|
||||
ETag: "21025-dc7-39462498"
|
||||
Content-Length: 6
|
||||
Connection: close
|
||||
|
||||
</file>
|
||||
</verify>
|
||||
</testcase>
|
@ -12,6 +12,9 @@ HTTP/1.1 200 OK
|
||||
Date: Thu, 09 Nov 2010 14:49:00 GMT
|
||||
Server: test with trailing space
|
||||
Content-Type: text/html
|
||||
Fold: is
|
||||
folding a
|
||||
line
|
||||
Content-Length: 0
|
||||
Set-Cookie: onecookie=data;
|
||||
Set-Cookie: secondcookie=2data;
|
||||
@ -44,7 +47,7 @@ http://%HOSTIP:%HTTPPORT/%TESTNUMBER
|
||||
|
||||
# Verify data after the test has been "shot"
|
||||
<verify>
|
||||
<stdout>
|
||||
<stdout mode="text">
|
||||
Date == Thu, 09 Nov 2010 14:49:00 GMT
|
||||
Server == test with trailing space
|
||||
Content-Type == text/html
|
||||
@ -53,6 +56,7 @@ http://%HOSTIP:%HTTPPORT/%TESTNUMBER
|
||||
- Set-Cookie == onecookie=data; (0/3)
|
||||
- Set-Cookie == secondcookie=2data; (1/3)
|
||||
- Set-Cookie == cookie3=data3; (2/3)
|
||||
Fold == is folding a line
|
||||
</stdout>
|
||||
</verify>
|
||||
</testcase>
|
||||
|
@ -32,6 +32,7 @@ static const char *show[]={
|
||||
"location",
|
||||
"set-cookie",
|
||||
"silly-thing",
|
||||
"fold",
|
||||
NULL
|
||||
};
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user