Remove content-length and content-range if transfer-encoding is given

See GH-473
This commit is contained in:
Tatsuhiro Tsujikawa 2015-11-04 00:28:44 +09:00
parent 4bbc71c8e6
commit 5ccd5b6953
6 changed files with 64 additions and 2 deletions

View file

@ -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);

View file

@ -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;

View file

@ -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)

View file

@ -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);

View file

@ -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);

View file

@ -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