/* */ #include "HttpDownloadCommand.h" #include "RequestGroup.h" #include "DownloadEngine.h" #include "Request.h" #include "HttpRequestCommand.h" #include "HttpConnection.h" #include "HttpRequest.h" #include "Segment.h" #include "Socket.h" #include "prefs.h" #include "Option.h" #include "HttpResponse.h" #include "HttpHeader.h" #include "Range.h" #include "DownloadContext.h" #include "Logger.h" #include "StreamFilter.h" #include "SinkStreamFilter.h" #include "util.h" namespace aria2 { HttpDownloadCommand::HttpDownloadCommand (cuid_t cuid, const SharedHandle& req, const SharedHandle& fileEntry, RequestGroup* requestGroup, const SharedHandle& httpResponse, const HttpConnectionHandle& httpConnection, DownloadEngine* e, const SocketHandle& socket) :DownloadCommand(cuid, req, fileEntry, requestGroup, e, socket), httpResponse_(httpResponse), httpConnection_(httpConnection) {} HttpDownloadCommand::~HttpDownloadCommand() {} bool HttpDownloadCommand::prepareForNextSegment() { bool downloadFinished = getRequestGroup()->downloadFinished(); if(getRequest()->isPipeliningEnabled() && !downloadFinished) { HttpRequestCommand* command = new HttpRequestCommand(getCuid(), getRequest(), getFileEntry(), getRequestGroup(), httpConnection_, getDownloadEngine(), getSocket()); // Set proxy request here. aria2 sends the HTTP request specialized for // proxy. if(resolveProxyMethod(getRequest()->getProtocol()) == V_GET) { command->setProxyRequest(createProxyRequest()); } getDownloadEngine()->addCommand(command); return true; } else { if(getRequest()->isPipeliningEnabled() || (getRequest()->isKeepAliveEnabled() && ( // Make sure that all filters are finished to pool socket (!util::endsWith(getStreamFilter()->getName(), SinkStreamFilter::NAME) && getStreamFilter()->finished()) || getRequestEndOffset() == getFileEntry()->gtoloff(getSegments().front()->getPositionToWrite()) ) ) ) { // TODO What if server sends EOF when non-SinkStreamFilter is // used and server didn't send Connection: close? We end up to // pool terminated socket. In HTTP/1.1, keep-alive is default, // so closing connection without Connection: close header means // that server is broken or not configured properly. getDownloadEngine()->poolSocket (getRequest(), createProxyRequest(), getSocket()); } // The request was sent assuming that server supported pipelining, but // it turned out that server didn't support it. // We detect this situation by comparing the end byte in range header // of the response with the end byte of segment. // If it is the same, HTTP negotiation is necessary for the next request. if(!getRequest()->isPipeliningEnabled() && getRequest()->isPipeliningHint() && !downloadFinished) { const SharedHandle& segment = getSegments().front(); off_t lastOffset =getFileEntry()->gtoloff (std::min(static_cast (segment->getPosition()+segment->getLength()), getFileEntry()->getLastOffset())); if(lastOffset == httpResponse_->getHttpHeader()->getRange()->getEndByte()+1) { return prepareForRetry(0); } } return DownloadCommand::prepareForNextSegment(); } } off_t HttpDownloadCommand::getRequestEndOffset() const { off_t endByte = httpResponse_->getHttpHeader()->getRange()->getEndByte(); if(endByte > 0) { return endByte+1; } else { return endByte; } } } // namespace aria2