#include "HttpResponse.h" #include #include #include "prefs.h" #include "PiecedSegment.h" #include "Piece.h" #include "Request.h" #include "HttpHeader.h" #include "HttpRequest.h" #include "Exception.h" #include "A2STR.h" #include "DlRetryEx.h" #include "CookieStorage.h" #include "AuthConfigFactory.h" #include "AuthConfig.h" #include "StreamFilter.h" #include "MetalinkHttpEntry.h" #include "Option.h" #include "Checksum.h" namespace aria2 { class HttpResponseTest : public CppUnit::TestFixture { CPPUNIT_TEST_SUITE(HttpResponseTest); CPPUNIT_TEST(testGetContentLength_null); CPPUNIT_TEST(testGetContentLength_contentLength); //CPPUNIT_TEST(testGetContentLength_range); CPPUNIT_TEST(testGetEntityLength); CPPUNIT_TEST(testGetContentType); CPPUNIT_TEST(testDeterminFilename_without_ContentDisposition); CPPUNIT_TEST(testDeterminFilename_with_ContentDisposition_zero_length); CPPUNIT_TEST(testDeterminFilename_with_ContentDisposition); CPPUNIT_TEST(testGetRedirectURI_without_Location); CPPUNIT_TEST(testGetRedirectURI_with_Location); CPPUNIT_TEST(testIsRedirect); CPPUNIT_TEST(testIsTransferEncodingSpecified); CPPUNIT_TEST(testGetTransferEncoding); CPPUNIT_TEST(testGetTransferEncodingStreamFilter); CPPUNIT_TEST(testIsContentEncodingSpecified); CPPUNIT_TEST(testGetContentEncoding); CPPUNIT_TEST(testGetContentEncodingStreamFilter); CPPUNIT_TEST(testValidateResponse); CPPUNIT_TEST(testValidateResponse_good_range); CPPUNIT_TEST(testValidateResponse_bad_range); CPPUNIT_TEST(testValidateResponse_chunked); CPPUNIT_TEST(testValidateResponse_withIfModifiedSince); CPPUNIT_TEST(testHasRetryAfter); CPPUNIT_TEST(testProcessRedirect); CPPUNIT_TEST(testRetrieveCookie); CPPUNIT_TEST(testSupportsPersistentConnection); CPPUNIT_TEST(testGetMetalinKHttpEntries); #ifdef ENABLE_MESSAGE_DIGEST CPPUNIT_TEST(testGetDigest); #endif // ENABLE_MESSAGE_DIGEST CPPUNIT_TEST_SUITE_END(); private: public: void setUp() { } void testGetContentLength_null(); void testGetContentLength_contentLength(); void testGetEntityLength(); void testGetContentType(); void testDeterminFilename_without_ContentDisposition(); void testDeterminFilename_with_ContentDisposition_zero_length(); void testDeterminFilename_with_ContentDisposition(); void testGetRedirectURI_without_Location(); void testGetRedirectURI_with_Location(); void testIsRedirect(); void testIsTransferEncodingSpecified(); void testGetTransferEncoding(); void testGetTransferEncodingStreamFilter(); void testIsContentEncodingSpecified(); void testGetContentEncoding(); void testGetContentEncodingStreamFilter(); void testValidateResponse(); void testValidateResponse_good_range(); void testValidateResponse_bad_range(); void testValidateResponse_chunked(); void testValidateResponse_withIfModifiedSince(); void testHasRetryAfter(); void testProcessRedirect(); void testRetrieveCookie(); void testSupportsPersistentConnection(); void testGetMetalinKHttpEntries(); #ifdef ENABLE_MESSAGE_DIGEST void testGetDigest(); #endif // ENABLE_MESSAGE_DIGEST }; CPPUNIT_TEST_SUITE_REGISTRATION( HttpResponseTest ); void HttpResponseTest::testGetContentLength_null() { HttpResponse httpResponse; CPPUNIT_ASSERT_EQUAL((int64_t)0LL, httpResponse.getContentLength()); } void HttpResponseTest::testGetContentLength_contentLength() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpHeader->put(HttpHeader::CONTENT_LENGTH, "4294967296"); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT_EQUAL((int64_t)4294967296LL, httpResponse.getContentLength()); } void HttpResponseTest::testGetEntityLength() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpHeader->put(HttpHeader::CONTENT_LENGTH, "4294967296"); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT_EQUAL((int64_t)4294967296LL, httpResponse.getEntityLength()); httpHeader->put(HttpHeader::CONTENT_RANGE, "bytes 1-4294967296/4294967297"); CPPUNIT_ASSERT_EQUAL((int64_t)4294967297LL, httpResponse.getEntityLength()); } void HttpResponseTest::testGetContentType() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpHeader->put(HttpHeader::CONTENT_TYPE, "application/metalink+xml; charset=UTF-8"); httpResponse.setHttpHeader(httpHeader); // See paramter is ignored. CPPUNIT_ASSERT_EQUAL(std::string("application/metalink+xml"), httpResponse.getContentType()); } void HttpResponseTest::testDeterminFilename_without_ContentDisposition() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); SharedHandle httpRequest(new HttpRequest()); SharedHandle request(new Request()); request->setUri("http://localhost/archives/aria2-1.0.0.tar.bz2"); httpRequest->setRequest(request); httpResponse.setHttpHeader(httpHeader); httpResponse.setHttpRequest(httpRequest); CPPUNIT_ASSERT_EQUAL(std::string("aria2-1.0.0.tar.bz2"), httpResponse.determinFilename()); } void HttpResponseTest::testDeterminFilename_with_ContentDisposition_zero_length () { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpHeader->put(HttpHeader::CONTENT_DISPOSITION, "attachment; filename=\"\""); SharedHandle httpRequest(new HttpRequest()); SharedHandle request(new Request()); request->setUri("http://localhost/archives/aria2-1.0.0.tar.bz2"); httpRequest->setRequest(request); httpResponse.setHttpHeader(httpHeader); httpResponse.setHttpRequest(httpRequest); CPPUNIT_ASSERT_EQUAL(std::string("aria2-1.0.0.tar.bz2"), httpResponse.determinFilename()); } void HttpResponseTest::testDeterminFilename_with_ContentDisposition() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpHeader->put(HttpHeader::CONTENT_DISPOSITION, "attachment; filename=\"aria2-current.tar.bz2\""); SharedHandle httpRequest(new HttpRequest()); SharedHandle request(new Request()); request->setUri("http://localhost/archives/aria2-1.0.0.tar.bz2"); httpRequest->setRequest(request); httpResponse.setHttpHeader(httpHeader); httpResponse.setHttpRequest(httpRequest); CPPUNIT_ASSERT_EQUAL(std::string("aria2-current.tar.bz2"), httpResponse.determinFilename()); } void HttpResponseTest::testGetRedirectURI_without_Location() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT_EQUAL(std::string(""), httpResponse.getRedirectURI()); } void HttpResponseTest::testGetRedirectURI_with_Location() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpHeader->put(HttpHeader::LOCATION, "http://localhost/download/aria2-1.0.0.tar.bz2"); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT_EQUAL (std::string("http://localhost/download/aria2-1.0.0.tar.bz2"), httpResponse.getRedirectURI()); } void HttpResponseTest::testIsRedirect() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpHeader->setStatusCode(200); httpHeader->put(HttpHeader::LOCATION, "http://localhost/download/aria2-1.0.0.tar.bz2"); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT(!httpResponse.isRedirect()); httpHeader->setStatusCode(301); CPPUNIT_ASSERT(httpResponse.isRedirect()); } void HttpResponseTest::testIsTransferEncodingSpecified() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT(!httpResponse.isTransferEncodingSpecified()); httpHeader->put(HttpHeader::TRANSFER_ENCODING, "chunked"); CPPUNIT_ASSERT(httpResponse.isTransferEncodingSpecified()); } void HttpResponseTest::testGetTransferEncoding() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT_EQUAL(std::string(""), httpResponse.getTransferEncoding()); httpHeader->put(HttpHeader::TRANSFER_ENCODING, "chunked"); CPPUNIT_ASSERT_EQUAL(std::string("chunked"), httpResponse.getTransferEncoding()); } void HttpResponseTest::testGetTransferEncodingStreamFilter() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT(!httpResponse.getTransferEncodingStreamFilter()); httpHeader->put(HttpHeader::TRANSFER_ENCODING, "chunked"); CPPUNIT_ASSERT(httpResponse.getTransferEncodingStreamFilter()); } void HttpResponseTest::testIsContentEncodingSpecified() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT(!httpResponse.isContentEncodingSpecified()); httpHeader->put(HttpHeader::CONTENT_ENCODING, "gzip"); CPPUNIT_ASSERT(httpResponse.isContentEncodingSpecified()); } void HttpResponseTest::testGetContentEncoding() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT_EQUAL(A2STR::NIL, httpResponse.getContentEncoding()); httpHeader->put(HttpHeader::CONTENT_ENCODING, "gzip"); CPPUNIT_ASSERT_EQUAL(std::string("gzip"), httpResponse.getContentEncoding()); } void HttpResponseTest::testGetContentEncodingStreamFilter() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); CPPUNIT_ASSERT(!httpResponse.getContentEncodingStreamFilter()); #ifdef HAVE_ZLIB httpHeader->put(HttpHeader::CONTENT_ENCODING, "gzip"); { SharedHandle filter = httpResponse.getContentEncodingStreamFilter(); CPPUNIT_ASSERT(filter); CPPUNIT_ASSERT_EQUAL(std::string("GZipDecodingStreamFilter"), filter->getName()); } httpHeader.reset(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); httpHeader->put(HttpHeader::CONTENT_ENCODING, "deflate"); { SharedHandle filter = httpResponse.getContentEncodingStreamFilter(); CPPUNIT_ASSERT(filter); CPPUNIT_ASSERT_EQUAL(std::string("GZipDecodingStreamFilter"), filter->getName()); } #endif // HAVE_ZLIB httpHeader.reset(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); httpHeader->put(HttpHeader::CONTENT_ENCODING, "bzip2"); { SharedHandle filter = httpResponse.getContentEncodingStreamFilter(); CPPUNIT_ASSERT(!filter); } } void HttpResponseTest::testValidateResponse() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); httpHeader->setStatusCode(301); try { httpResponse.validateResponse(); CPPUNIT_FAIL("exception must be thrown."); } catch(Exception& e) { } httpHeader->put(HttpHeader::LOCATION, "http://localhost/archives/aria2-1.0.0.tar.bz2"); try { httpResponse.validateResponse(); } catch(Exception& e) { CPPUNIT_FAIL("exception must not be thrown."); } } void HttpResponseTest::testValidateResponse_good_range() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); SharedHandle httpRequest(new HttpRequest()); SharedHandle p(new Piece(1, 1024*1024)); SharedHandle segment(new PiecedSegment(1024*1024, p)); httpRequest->setSegment(segment); SharedHandle fileEntry(new FileEntry("file", 1024*1024*10, 0)); httpRequest->setFileEntry(fileEntry); SharedHandle request(new Request()); request->setUri("http://localhost/archives/aria2-1.0.0.tar.bz2"); httpRequest->setRequest(request); httpResponse.setHttpRequest(httpRequest); httpHeader->setStatusCode(206); httpHeader->put(HttpHeader::CONTENT_RANGE, "bytes 1048576-10485760/10485760"); try { httpResponse.validateResponse(); } catch(Exception& e) { std::cerr << e.stackTrace() << std::endl; CPPUNIT_FAIL("exception must not be thrown."); } } void HttpResponseTest::testValidateResponse_bad_range() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); SharedHandle httpRequest(new HttpRequest()); SharedHandle p(new Piece(1, 1024*1024)); SharedHandle segment(new PiecedSegment(1024*1024, p)); httpRequest->setSegment(segment); SharedHandle fileEntry(new FileEntry("file", 1024*1024*10, 0)); httpRequest->setFileEntry(fileEntry); SharedHandle request(new Request()); request->setUri("http://localhost/archives/aria2-1.0.0.tar.bz2"); httpRequest->setRequest(request); httpResponse.setHttpRequest(httpRequest); httpHeader->setStatusCode(206); httpHeader->put(HttpHeader::CONTENT_RANGE, "bytes 0-10485760/10485761"); try { httpResponse.validateResponse(); CPPUNIT_FAIL("exception must be thrown."); } catch(Exception& e) { } } void HttpResponseTest::testValidateResponse_chunked() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); SharedHandle httpRequest(new HttpRequest()); SharedHandle p(new Piece(1, 1024*1024)); SharedHandle segment(new PiecedSegment(1024*1024, p)); httpRequest->setSegment(segment); SharedHandle fileEntry(new FileEntry("file", 1024*1024*10, 0)); httpRequest->setFileEntry(fileEntry); SharedHandle request(new Request()); request->setUri("http://localhost/archives/aria2-1.0.0.tar.bz2"); httpRequest->setRequest(request); httpResponse.setHttpRequest(httpRequest); httpHeader->setStatusCode(206); httpHeader->put(HttpHeader::CONTENT_RANGE, "bytes 0-10485760/10485761"); httpHeader->put(HttpHeader::TRANSFER_ENCODING, "chunked"); // if transfer-encoding is specified, then range validation is skipped. try { httpResponse.validateResponse(); } catch(Exception& e) { CPPUNIT_FAIL("exception must not be thrown."); } } void HttpResponseTest::testValidateResponse_withIfModifiedSince() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); httpHeader->setStatusCode(304); SharedHandle httpRequest(new HttpRequest()); httpResponse.setHttpRequest(httpRequest); try { httpResponse.validateResponse(); CPPUNIT_FAIL("exception must be thrown."); } catch(Exception& e) { } httpRequest->setIfModifiedSinceHeader("Fri, 16 Jul 2010 12:56:59 GMT"); httpResponse.validateResponse(); } void HttpResponseTest::testHasRetryAfter() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); httpHeader->put(HttpHeader::RETRY_AFTER, "60"); CPPUNIT_ASSERT(httpResponse.hasRetryAfter()); CPPUNIT_ASSERT_EQUAL((time_t)60, httpResponse.getRetryAfter()); } void HttpResponseTest::testProcessRedirect() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); SharedHandle httpRequest(new HttpRequest()); SharedHandle request(new Request()); request->setUri("http://localhost/archives/aria2-1.0.0.tar.bz2"); httpRequest->setRequest(request); httpResponse.setHttpRequest(httpRequest); httpHeader->put(HttpHeader::LOCATION, "http://mirror/aria2-1.0.0.tar.bz2"); httpResponse.processRedirect(); httpHeader->clearField(); // Test for percent-encode httpHeader->put(HttpHeader::LOCATION, "http://example.org/white space#aria2"); httpResponse.processRedirect(); CPPUNIT_ASSERT_EQUAL(std::string("http://example.org/white%20space"), request->getCurrentUri()); httpHeader->clearField(); // Give unsupported scheme httpHeader->put(HttpHeader::LOCATION, "unsupported://mirror/aria2-1.0.0.tar.bz2"); try { httpResponse.processRedirect(); CPPUNIT_FAIL("DlRetryEx exception must be thrown."); } catch(DlRetryEx& e) { // Success } catch(...) { CPPUNIT_FAIL("DlRetryEx exception must be thrown."); } } void HttpResponseTest::testRetrieveCookie() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); SharedHandle httpRequest(new HttpRequest()); SharedHandle request(new Request()); request->setUri("http://www.aria2.org/archives/aria2-1.0.0.tar.bz2"); httpRequest->setRequest(request); SharedHandle st(new CookieStorage()); httpRequest->setCookieStorage(st); httpResponse.setHttpRequest(httpRequest); httpHeader->put(HttpHeader::SET_COOKIE, "k1=v1; expires=Sun, 10-Jun-2007 11:00:00 GMT;" "path=/; domain=.aria2.org;"); httpHeader->put(HttpHeader::SET_COOKIE, "k2=v2; expires=Sun, 01-Jan-38 00:00:00 GMT;" "path=/; domain=.aria2.org;"); httpHeader->put(HttpHeader::SET_COOKIE, "k3=v3;"); httpResponse.retrieveCookie(); CPPUNIT_ASSERT_EQUAL((size_t)2, st->size()); std::vector cookies; st->dumpCookie(std::back_inserter(cookies)); CPPUNIT_ASSERT_EQUAL(std::string("k2=v2"), cookies[0].toString()); CPPUNIT_ASSERT_EQUAL(std::string("k3=v3"), cookies[1].toString()); } void HttpResponseTest::testSupportsPersistentConnection() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); SharedHandle httpRequest(new HttpRequest()); httpResponse.setHttpRequest(httpRequest); httpHeader->setVersion("HTTP/1.1"); CPPUNIT_ASSERT(httpResponse.supportsPersistentConnection()); httpHeader->put(HttpHeader::CONNECTION, "close"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->put(HttpHeader::CONNECTION, "keep-alive"); CPPUNIT_ASSERT(httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->setVersion("HTTP/1.0"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->put(HttpHeader::CONNECTION, "close"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->put(HttpHeader::CONNECTION, "keep-alive"); CPPUNIT_ASSERT(httpResponse.supportsPersistentConnection()); httpHeader->clearField(); // test proxy connection SharedHandle proxyRequest(new Request()); httpRequest->setProxyRequest(proxyRequest); httpHeader->setVersion("HTTP/1.1"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->put(HttpHeader::CONNECTION, "close"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->put(HttpHeader::CONNECTION, "keep-alive"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->put(HttpHeader::PROXY_CONNECTION, "keep-alive"); CPPUNIT_ASSERT(httpResponse.supportsPersistentConnection()); httpHeader->put(HttpHeader::CONNECTION, "close"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->put(HttpHeader::PROXY_CONNECTION, "close"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->setVersion("HTTP/1.0"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->put(HttpHeader::CONNECTION, "close"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->put(HttpHeader::CONNECTION, "keep-alive"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->put(HttpHeader::PROXY_CONNECTION, "keep-alive"); CPPUNIT_ASSERT(httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->put(HttpHeader::PROXY_CONNECTION, "keep-alive"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); httpHeader->put(HttpHeader::PROXY_CONNECTION, "close"); CPPUNIT_ASSERT(!httpResponse.supportsPersistentConnection()); httpHeader->clearField(); } void HttpResponseTest::testGetMetalinKHttpEntries() { HttpResponse httpResponse; SharedHandle httpHeader(new HttpHeader()); httpResponse.setHttpHeader(httpHeader); SharedHandle