mirror of
https://github.com/aria2/aria2.git
synced 2025-04-05 21:47:37 +03:00
Remove content-length and content-range if transfer-encoding is given
See GH-473
This commit is contained in:
parent
4bbc71c8e6
commit
5ccd5b6953
6 changed files with 64 additions and 2 deletions
|
@ -53,6 +53,11 @@ void HttpHeader::put(int hdKey, const std::string& value)
|
||||||
table_.insert(vt);
|
table_.insert(vt);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HttpHeader::remove(int hdKey)
|
||||||
|
{
|
||||||
|
table_.erase(hdKey);
|
||||||
|
}
|
||||||
|
|
||||||
bool HttpHeader::defined(int hdKey) const
|
bool HttpHeader::defined(int hdKey) const
|
||||||
{
|
{
|
||||||
return table_.count(hdKey);
|
return table_.count(hdKey);
|
||||||
|
|
|
@ -106,6 +106,8 @@ public:
|
||||||
std::multimap<int, std::string>::const_iterator>
|
std::multimap<int, std::string>::const_iterator>
|
||||||
equalRange(int hdKey) const;
|
equalRange(int hdKey) const;
|
||||||
|
|
||||||
|
void remove(int hdKey);
|
||||||
|
|
||||||
Range getRange() const;
|
Range getRange() const;
|
||||||
|
|
||||||
int getStatusCode() const;
|
int getStatusCode() const;
|
||||||
|
|
|
@ -446,7 +446,24 @@ fin:
|
||||||
|
|
||||||
lastBytesProcessed_ = i;
|
lastBytesProcessed_ = i;
|
||||||
headers_.append(&data[0], &data[i]);
|
headers_.append(&data[0], &data[i]);
|
||||||
return state_ == HEADERS_COMPLETE;
|
|
||||||
|
if (state_ != HEADERS_COMPLETE) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If both transfer-encoding and (content-length or content-range)
|
||||||
|
// are present, delete content-length and content-range. RFC 7230
|
||||||
|
// says that sender must not send both transfer-encoding and
|
||||||
|
// content-length. If both present, transfer-encoding overrides
|
||||||
|
// content-length. There is no text about transfer-encoding and
|
||||||
|
// content-range. But there is no reason to send transfer-encoding
|
||||||
|
// when range is set.
|
||||||
|
if (result_->defined(HttpHeader::TRANSFER_ENCODING)) {
|
||||||
|
result_->remove(HttpHeader::CONTENT_LENGTH);
|
||||||
|
result_->remove(HttpHeader::CONTENT_RANGE);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HttpHeaderProcessor::parse(const std::string& data)
|
bool HttpHeaderProcessor::parse(const std::string& data)
|
||||||
|
|
|
@ -263,7 +263,8 @@ bool HttpResponseCommand::executeInternal()
|
||||||
updateLastModifiedTime(httpResponse->getLastModifiedTime());
|
updateLastModifiedTime(httpResponse->getLastModifiedTime());
|
||||||
|
|
||||||
// If both transfer-encoding and total length is specified, we
|
// If both transfer-encoding and total length is specified, we
|
||||||
// assume we can do segmented downloading
|
// should have ignored total length. In this case, we can not do
|
||||||
|
// segmented downloading
|
||||||
if (totalLength == 0 || shouldInflateContentEncoding(httpResponse.get())) {
|
if (totalLength == 0 || shouldInflateContentEncoding(httpResponse.get())) {
|
||||||
// we ignore content-length when inflate is required
|
// we ignore content-length when inflate is required
|
||||||
fe->setLength(0);
|
fe->setLength(0);
|
||||||
|
|
|
@ -22,6 +22,7 @@ class HttpHeaderProcessorTest:public CppUnit::TestFixture {
|
||||||
CPPUNIT_TEST(testGetHttpResponseHeader_statusOnly);
|
CPPUNIT_TEST(testGetHttpResponseHeader_statusOnly);
|
||||||
CPPUNIT_TEST(testGetHttpResponseHeader_insufficientStatusLength);
|
CPPUNIT_TEST(testGetHttpResponseHeader_insufficientStatusLength);
|
||||||
CPPUNIT_TEST(testGetHttpResponseHeader_nameStartsWs);
|
CPPUNIT_TEST(testGetHttpResponseHeader_nameStartsWs);
|
||||||
|
CPPUNIT_TEST(testGetHttpResponseHeader_teAndCl);
|
||||||
CPPUNIT_TEST(testBeyondLimit);
|
CPPUNIT_TEST(testBeyondLimit);
|
||||||
CPPUNIT_TEST(testGetHeaderString);
|
CPPUNIT_TEST(testGetHeaderString);
|
||||||
CPPUNIT_TEST(testGetHttpRequestHeader);
|
CPPUNIT_TEST(testGetHttpRequestHeader);
|
||||||
|
@ -37,6 +38,7 @@ public:
|
||||||
void testGetHttpResponseHeader_statusOnly();
|
void testGetHttpResponseHeader_statusOnly();
|
||||||
void testGetHttpResponseHeader_insufficientStatusLength();
|
void testGetHttpResponseHeader_insufficientStatusLength();
|
||||||
void testGetHttpResponseHeader_nameStartsWs();
|
void testGetHttpResponseHeader_nameStartsWs();
|
||||||
|
void testGetHttpResponseHeader_teAndCl();
|
||||||
void testBeyondLimit();
|
void testBeyondLimit();
|
||||||
void testGetHeaderString();
|
void testGetHeaderString();
|
||||||
void testGetHttpRequestHeader();
|
void testGetHttpRequestHeader();
|
||||||
|
@ -210,6 +212,26 @@ void HttpHeaderProcessorTest::testGetHttpResponseHeader_nameStartsWs()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HttpHeaderProcessorTest::testGetHttpResponseHeader_teAndCl()
|
||||||
|
{
|
||||||
|
HttpHeaderProcessor proc(HttpHeaderProcessor::CLIENT_PARSER);
|
||||||
|
|
||||||
|
std::string hd =
|
||||||
|
"HTTP/1.1 200\r\n"
|
||||||
|
"Content-Length: 200\r\n"
|
||||||
|
"Transfer-Encoding: chunked\r\n"
|
||||||
|
"Content-Range: 1-200/300\r\n"
|
||||||
|
"\r\n";
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(proc.parse(hd));
|
||||||
|
|
||||||
|
auto httpHeader = proc.getResult();
|
||||||
|
CPPUNIT_ASSERT_EQUAL(std::string("chunked"),
|
||||||
|
httpHeader->find(HttpHeader::TRANSFER_ENCODING));
|
||||||
|
CPPUNIT_ASSERT(!httpHeader->defined(HttpHeader::CONTENT_LENGTH));
|
||||||
|
CPPUNIT_ASSERT(!httpHeader->defined(HttpHeader::CONTENT_RANGE));
|
||||||
|
}
|
||||||
|
|
||||||
void HttpHeaderProcessorTest::testBeyondLimit()
|
void HttpHeaderProcessorTest::testBeyondLimit()
|
||||||
{
|
{
|
||||||
HttpHeaderProcessor proc(HttpHeaderProcessor::CLIENT_PARSER);
|
HttpHeaderProcessor proc(HttpHeaderProcessor::CLIENT_PARSER);
|
||||||
|
|
|
@ -14,6 +14,7 @@ class HttpHeaderTest:public CppUnit::TestFixture {
|
||||||
CPPUNIT_TEST(testFindAll);
|
CPPUNIT_TEST(testFindAll);
|
||||||
CPPUNIT_TEST(testClearField);
|
CPPUNIT_TEST(testClearField);
|
||||||
CPPUNIT_TEST(testFieldContains);
|
CPPUNIT_TEST(testFieldContains);
|
||||||
|
CPPUNIT_TEST(testRemove);
|
||||||
CPPUNIT_TEST_SUITE_END();
|
CPPUNIT_TEST_SUITE_END();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
@ -21,6 +22,7 @@ public:
|
||||||
void testFindAll();
|
void testFindAll();
|
||||||
void testClearField();
|
void testClearField();
|
||||||
void testFieldContains();
|
void testFieldContains();
|
||||||
|
void testRemove();
|
||||||
};
|
};
|
||||||
|
|
||||||
CPPUNIT_TEST_SUITE_REGISTRATION( HttpHeaderTest );
|
CPPUNIT_TEST_SUITE_REGISTRATION( HttpHeaderTest );
|
||||||
|
@ -167,4 +169,17 @@ void HttpHeaderTest::testFieldContains()
|
||||||
CPPUNIT_ASSERT(!h.fieldContains(HttpHeader::SEC_WEBSOCKET_VERSION, "6"));
|
CPPUNIT_ASSERT(!h.fieldContains(HttpHeader::SEC_WEBSOCKET_VERSION, "6"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HttpHeaderTest::testRemove()
|
||||||
|
{
|
||||||
|
HttpHeader h;
|
||||||
|
h.put(HttpHeader::CONNECTION, "close");
|
||||||
|
h.put(HttpHeader::TRANSFER_ENCODING, "chunked");
|
||||||
|
h.put(HttpHeader::TRANSFER_ENCODING, "gzip");
|
||||||
|
|
||||||
|
h.remove(HttpHeader::TRANSFER_ENCODING);
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(!h.defined(HttpHeader::TRANSFER_ENCODING));
|
||||||
|
CPPUNIT_ASSERT(h.defined(HttpHeader::CONNECTION));
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace aria2
|
} // namespace aria2
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue