2006-02-17 13:35:04 +00:00
|
|
|
/* <!-- copyright */
|
|
|
|
/*
|
2006-09-21 15:31:24 +00:00
|
|
|
* aria2 - The high speed download utility
|
2006-02-17 13:35:04 +00:00
|
|
|
*
|
|
|
|
* Copyright (C) 2006 Tatsuhiro Tsujikawa
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
2010-01-05 16:01:46 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
2006-09-21 15:31:24 +00:00
|
|
|
*
|
|
|
|
* In addition, as a special exception, the copyright holders give
|
|
|
|
* permission to link the code of portions of this program with the
|
|
|
|
* OpenSSL library under certain conditions as described in each
|
|
|
|
* individual source file, and distribute linked combinations
|
|
|
|
* including the two.
|
|
|
|
* You must obey the GNU General Public License in all respects
|
|
|
|
* for all of the code used other than OpenSSL. If you modify
|
|
|
|
* file(s) with this exception, you may extend this exception to your
|
|
|
|
* version of the file(s), but you are not obligated to do so. If you
|
|
|
|
* do not wish to do so, delete this exception statement from your
|
|
|
|
* version. If you delete this exception statement from all source
|
|
|
|
* files in the program, then also delete it here.
|
2006-02-17 13:35:04 +00:00
|
|
|
*/
|
|
|
|
/* copyright --> */
|
|
|
|
#include "HttpDownloadCommand.h"
|
2007-10-11 16:58:24 +00:00
|
|
|
#include "RequestGroup.h"
|
|
|
|
#include "DownloadEngine.h"
|
|
|
|
#include "Request.h"
|
2006-02-17 13:35:04 +00:00
|
|
|
#include "HttpRequestCommand.h"
|
2007-10-17 16:26:51 +00:00
|
|
|
#include "HttpConnection.h"
|
2008-02-08 15:53:45 +00:00
|
|
|
#include "HttpRequest.h"
|
|
|
|
#include "Segment.h"
|
|
|
|
#include "Socket.h"
|
2008-04-22 08:52:47 +00:00
|
|
|
#include "prefs.h"
|
|
|
|
#include "Option.h"
|
2008-12-15 15:38:07 +00:00
|
|
|
#include "HttpResponse.h"
|
|
|
|
#include "HttpHeader.h"
|
|
|
|
#include "Range.h"
|
2009-06-28 10:37:15 +00:00
|
|
|
#include "DownloadContext.h"
|
2010-07-12 11:55:23 +00:00
|
|
|
#include "Logger.h"
|
2010-09-06 14:29:36 +00:00
|
|
|
#include "StreamFilter.h"
|
|
|
|
#include "SinkStreamFilter.h"
|
|
|
|
#include "util.h"
|
2011-01-16 07:55:41 +00:00
|
|
|
#include "SocketRecvBuffer.h"
|
2010-09-06 14:29:36 +00:00
|
|
|
|
2008-02-08 15:53:45 +00:00
|
|
|
namespace aria2 {
|
2006-02-17 13:35:04 +00:00
|
|
|
|
2008-12-15 15:38:07 +00:00
|
|
|
HttpDownloadCommand::HttpDownloadCommand
|
2010-03-20 14:30:36 +00:00
|
|
|
(cuid_t cuid,
|
2010-02-28 12:30:11 +00:00
|
|
|
const SharedHandle<Request>& req,
|
2009-06-23 15:35:45 +00:00
|
|
|
const SharedHandle<FileEntry>& fileEntry,
|
2008-12-15 15:38:07 +00:00
|
|
|
RequestGroup* requestGroup,
|
|
|
|
const SharedHandle<HttpResponse>& httpResponse,
|
|
|
|
const HttpConnectionHandle& httpConnection,
|
|
|
|
DownloadEngine* e,
|
|
|
|
const SocketHandle& socket)
|
2011-01-16 07:55:41 +00:00
|
|
|
: DownloadCommand(cuid, req, fileEntry, requestGroup, e, socket,
|
|
|
|
httpConnection->getSocketRecvBuffer()),
|
|
|
|
httpResponse_(httpResponse),
|
|
|
|
httpConnection_(httpConnection)
|
|
|
|
{}
|
2006-02-17 13:35:04 +00:00
|
|
|
|
2007-03-15 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>
To handle Segment as SegmentHandle:
* src/AbstractCommand.cc (execute): Rewritten.
* src/SegmentMan.h: Segment -> SegmentHandle
Introducded HttpResponse class, HttpRequest class to improve
code
extensiveness and make it clear:
* src/HttpDownloadCommand.cc: transfer encoders are now managed
by
HttpResponse class.
* src/HttpRequest.h, src/HttpRequest.cc: New class.
* src/HttpResponse.h, src/HttpResponse.cc: New class.
* src/HttpConnection.cc: Contruction of http request were moved
to
HttpRequest class.
* src/HttpResponseCommand.h, src/HttpResponseCommand.cc:
Refactored.
* src/HttpRequestCommand.cc (executeInternal): Rewritten.
* src/HttpAuthConfig.h: New class.
* src/Range.h: New class.
To make FtpTunnel{Request, Response}Command and
HttpProxy{Request, Response}Command derived from
AbstractProxy{Request, Response}Command:
* src/FtpTunnelResponseCommand.h,
src/FtpTunnelResponseCommand.cc:
Derived from AbstractProxyRequestCommand class.
* src/FtpTunnelRequestCommand.h, src/FtpTunnelRequestCommand.cc:
Derived from AbstractProxyResponseCommand class.
* src/HttpProxyRequestCommand.h, src/HttpProxyRequestCommand.cc:
Derived from AbstractProxyRequestCommand class.
* src/HttpProxyResponseCommand.h,
src/HttpProxyResponseCommand.cc:
Derived from AbstractProxyResponseCommand class.
* src/AbstractProxyRequestCommand.h,
src/AbstractProxyRequestCommand.cc
: New class.
* src/AbstractProxyResponseCommand.h,
src/AbstractProxyResponseCommand.cc: New class.
To add netrc support:
* src/Netrc.h, src/Netrc.cc: New class.
* src/Util.h, src/Util.cc (split): New function.
* src/HttpHeader.cc (getRange): Fixed so that it inspects
"Content-Range" header instead of "Range" header.
* src/HttpHeader.h
(getStatus): Removed.
(setStatus): Removed.
* src/Segment.h
(getPositionToWrite): New function.
2007-03-15 15:07:18 +00:00
|
|
|
HttpDownloadCommand::~HttpDownloadCommand() {}
|
2006-09-19 14:52:59 +00:00
|
|
|
|
2007-03-15 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>
To handle Segment as SegmentHandle:
* src/AbstractCommand.cc (execute): Rewritten.
* src/SegmentMan.h: Segment -> SegmentHandle
Introducded HttpResponse class, HttpRequest class to improve
code
extensiveness and make it clear:
* src/HttpDownloadCommand.cc: transfer encoders are now managed
by
HttpResponse class.
* src/HttpRequest.h, src/HttpRequest.cc: New class.
* src/HttpResponse.h, src/HttpResponse.cc: New class.
* src/HttpConnection.cc: Contruction of http request were moved
to
HttpRequest class.
* src/HttpResponseCommand.h, src/HttpResponseCommand.cc:
Refactored.
* src/HttpRequestCommand.cc (executeInternal): Rewritten.
* src/HttpAuthConfig.h: New class.
* src/Range.h: New class.
To make FtpTunnel{Request, Response}Command and
HttpProxy{Request, Response}Command derived from
AbstractProxy{Request, Response}Command:
* src/FtpTunnelResponseCommand.h,
src/FtpTunnelResponseCommand.cc:
Derived from AbstractProxyRequestCommand class.
* src/FtpTunnelRequestCommand.h, src/FtpTunnelRequestCommand.cc:
Derived from AbstractProxyResponseCommand class.
* src/HttpProxyRequestCommand.h, src/HttpProxyRequestCommand.cc:
Derived from AbstractProxyRequestCommand class.
* src/HttpProxyResponseCommand.h,
src/HttpProxyResponseCommand.cc:
Derived from AbstractProxyResponseCommand class.
* src/AbstractProxyRequestCommand.h,
src/AbstractProxyRequestCommand.cc
: New class.
* src/AbstractProxyResponseCommand.h,
src/AbstractProxyResponseCommand.cc: New class.
To add netrc support:
* src/Netrc.h, src/Netrc.cc: New class.
* src/Util.h, src/Util.cc (split): New function.
* src/HttpHeader.cc (getRange): Fixed so that it inspects
"Content-Range" header instead of "Range" header.
* src/HttpHeader.h
(getStatus): Removed.
(setStatus): Removed.
* src/Segment.h
(getPositionToWrite): New function.
2007-03-15 15:07:18 +00:00
|
|
|
bool HttpDownloadCommand::prepareForNextSegment() {
|
2010-06-09 12:43:44 +00:00
|
|
|
bool downloadFinished = getRequestGroup()->downloadFinished();
|
|
|
|
if(getRequest()->isPipeliningEnabled() && !downloadFinished) {
|
2008-12-15 15:38:07 +00:00
|
|
|
HttpRequestCommand* command =
|
2010-06-09 12:43:44 +00:00
|
|
|
new HttpRequestCommand(getCuid(), getRequest(), getFileEntry(),
|
2010-06-21 13:51:56 +00:00
|
|
|
getRequestGroup(), httpConnection_,
|
2010-06-09 12:43:44 +00:00
|
|
|
getDownloadEngine(), getSocket());
|
2008-12-15 15:38:07 +00:00
|
|
|
// Set proxy request here. aria2 sends the HTTP request specialized for
|
|
|
|
// proxy.
|
2010-06-09 12:43:44 +00:00
|
|
|
if(resolveProxyMethod(getRequest()->getProtocol()) == V_GET) {
|
2008-12-15 15:38:07 +00:00
|
|
|
command->setProxyRequest(createProxyRequest());
|
|
|
|
}
|
2010-06-09 12:43:44 +00:00
|
|
|
getDownloadEngine()->addCommand(command);
|
2007-10-17 16:26:51 +00:00
|
|
|
return true;
|
2007-05-31 15:56:20 +00:00
|
|
|
} else {
|
2011-11-04 15:25:24 +00:00
|
|
|
const std::string& streamFilterName = getStreamFilter()->getName();
|
2010-06-09 12:43:44 +00:00
|
|
|
if(getRequest()->isPipeliningEnabled() ||
|
|
|
|
(getRequest()->isKeepAliveEnabled() &&
|
2010-01-06 13:02:10 +00:00
|
|
|
(
|
2010-09-06 14:29:36 +00:00
|
|
|
// Make sure that all filters are finished to pool socket
|
2011-11-04 15:25:24 +00:00
|
|
|
(!util::endsWith(streamFilterName.begin(),
|
|
|
|
streamFilterName.end(),
|
|
|
|
SinkStreamFilter::NAME.begin(),
|
|
|
|
SinkStreamFilter::NAME.end()) &&
|
2010-09-06 14:29:36 +00:00
|
|
|
getStreamFilter()->finished()) ||
|
2010-07-12 11:55:23 +00:00
|
|
|
getRequestEndOffset() ==
|
|
|
|
getFileEntry()->gtoloff(getSegments().front()->getPositionToWrite())
|
2010-01-06 13:02:10 +00:00
|
|
|
)
|
|
|
|
)
|
|
|
|
) {
|
2010-09-06 14:29:36 +00:00
|
|
|
// TODO What if server sends EOF when non-SinkStreamFilter is
|
2010-01-06 13:02:10 +00:00
|
|
|
// 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.
|
2010-06-09 12:43:44 +00:00
|
|
|
getDownloadEngine()->poolSocket
|
|
|
|
(getRequest(), createProxyRequest(), getSocket());
|
2008-04-22 08:52:47 +00:00
|
|
|
}
|
2008-12-15 15:38:07 +00:00
|
|
|
// 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.
|
2010-06-09 12:43:44 +00:00
|
|
|
if(!getRequest()->isPipeliningEnabled() &&
|
|
|
|
getRequest()->isPipeliningHint() &&
|
2009-06-30 17:03:57 +00:00
|
|
|
!downloadFinished) {
|
2010-06-09 12:43:44 +00:00
|
|
|
const SharedHandle<Segment>& segment = getSegments().front();
|
2009-06-30 17:03:57 +00:00
|
|
|
|
2010-06-09 12:43:44 +00:00
|
|
|
off_t lastOffset =getFileEntry()->gtoloff
|
|
|
|
(std::min(static_cast<off_t>
|
|
|
|
(segment->getPosition()+segment->getLength()),
|
|
|
|
getFileEntry()->getLastOffset()));
|
2009-06-30 17:03:57 +00:00
|
|
|
|
|
|
|
if(lastOffset ==
|
2010-06-21 13:51:56 +00:00
|
|
|
httpResponse_->getHttpHeader()->getRange()->getEndByte()+1) {
|
2010-01-05 16:01:46 +00:00
|
|
|
return prepareForRetry(0);
|
2008-12-15 15:38:07 +00:00
|
|
|
}
|
|
|
|
}
|
2007-05-31 15:56:20 +00:00
|
|
|
return DownloadCommand::prepareForNextSegment();
|
2006-09-19 14:52:59 +00:00
|
|
|
}
|
|
|
|
}
|
2008-02-08 15:53:45 +00:00
|
|
|
|
2010-07-11 14:21:17 +00:00
|
|
|
off_t HttpDownloadCommand::getRequestEndOffset() const
|
|
|
|
{
|
2010-09-06 14:29:36 +00:00
|
|
|
off_t endByte = httpResponse_->getHttpHeader()->getRange()->getEndByte();
|
|
|
|
if(endByte > 0) {
|
|
|
|
return endByte+1;
|
|
|
|
} else {
|
|
|
|
return endByte;
|
|
|
|
}
|
2010-07-11 14:21:17 +00:00
|
|
|
}
|
|
|
|
|
2008-02-08 15:53:45 +00:00
|
|
|
} // namespace aria2
|