HttpRequestCommand.cc 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. /* <!-- copyright */
  2. /*
  3. * aria2 - The high speed download utility
  4. *
  5. * Copyright (C) 2006 Tatsuhiro Tsujikawa
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  20. *
  21. * In addition, as a special exception, the copyright holders give
  22. * permission to link the code of portions of this program with the
  23. * OpenSSL library under certain conditions as described in each
  24. * individual source file, and distribute linked combinations
  25. * including the two.
  26. * You must obey the GNU General Public License in all respects
  27. * for all of the code used other than OpenSSL. If you modify
  28. * file(s) with this exception, you may extend this exception to your
  29. * version of the file(s), but you are not obligated to do so. If you
  30. * do not wish to do so, delete this exception statement from your
  31. * version. If you delete this exception statement from all source
  32. * files in the program, then also delete it here.
  33. */
  34. /* copyright --> */
  35. #include "HttpRequestCommand.h"
  36. #include <algorithm>
  37. #include "Request.h"
  38. #include "DownloadEngine.h"
  39. #include "RequestGroup.h"
  40. #include "HttpResponseCommand.h"
  41. #include "HttpConnection.h"
  42. #include "HttpRequest.h"
  43. #include "SegmentMan.h"
  44. #include "Segment.h"
  45. #include "Option.h"
  46. #include "Socket.h"
  47. #include "prefs.h"
  48. #include "a2functional.h"
  49. #include "util.h"
  50. #include "CookieStorage.h"
  51. #include "AuthConfigFactory.h"
  52. #include "AuthConfig.h"
  53. #include "DownloadContext.h"
  54. #include "PieceStorage.h"
  55. #include "DefaultBtProgressInfoFile.h"
  56. #include "Logger.h"
  57. #include "LogFactory.h"
  58. #include "fmt.h"
  59. #include "SocketRecvBuffer.h"
  60. namespace aria2 {
  61. HttpRequestCommand::HttpRequestCommand
  62. (cuid_t cuid,
  63. const SharedHandle<Request>& req,
  64. const SharedHandle<FileEntry>& fileEntry,
  65. RequestGroup* requestGroup,
  66. const HttpConnectionHandle& httpConnection,
  67. DownloadEngine* e,
  68. const SocketHandle& s)
  69. : AbstractCommand(cuid, req, fileEntry, requestGroup, e, s,
  70. httpConnection->getSocketRecvBuffer()),
  71. httpConnection_(httpConnection)
  72. {
  73. setTimeout(getOption()->getAsInt(PREF_CONNECT_TIMEOUT));
  74. disableReadCheckSocket();
  75. setWriteCheckSocket(getSocket());
  76. }
  77. HttpRequestCommand::~HttpRequestCommand() {}
  78. namespace {
  79. SharedHandle<HttpRequest>
  80. createHttpRequest(const SharedHandle<Request>& req,
  81. const SharedHandle<FileEntry>& fileEntry,
  82. const SharedHandle<Segment>& segment,
  83. int64_t totalLength,
  84. const SharedHandle<Option>& option,
  85. const RequestGroup* rg,
  86. const SharedHandle<CookieStorage>& cookieStorage,
  87. const SharedHandle<AuthConfigFactory>& authConfigFactory,
  88. const SharedHandle<Request>& proxyRequest,
  89. int64_t endOffset = 0)
  90. {
  91. SharedHandle<HttpRequest> httpRequest(new HttpRequest());
  92. httpRequest->setUserAgent(option->get(PREF_USER_AGENT));
  93. httpRequest->setRequest(req);
  94. httpRequest->setFileEntry(fileEntry);
  95. httpRequest->setSegment(segment);
  96. httpRequest->addHeader(option->get(PREF_HEADER));
  97. httpRequest->setCookieStorage(cookieStorage);
  98. httpRequest->setAuthConfigFactory(authConfigFactory, option.get());
  99. httpRequest->setProxyRequest(proxyRequest);
  100. httpRequest->addAcceptType(rg->getAcceptTypes().begin(),
  101. rg->getAcceptTypes().end());
  102. if(option->getAsBool(PREF_HTTP_ACCEPT_GZIP)) {
  103. httpRequest->enableAcceptGZip();
  104. } else {
  105. httpRequest->disableAcceptGZip();
  106. }
  107. if(option->getAsBool(PREF_HTTP_NO_CACHE)) {
  108. httpRequest->enableNoCache();
  109. } else {
  110. httpRequest->disableNoCache();
  111. }
  112. httpRequest->setEndOffsetOverride(endOffset);
  113. return httpRequest;
  114. }
  115. } // namespace
  116. bool HttpRequestCommand::executeInternal() {
  117. //socket->setBlockingMode();
  118. if(getRequest()->getProtocol() == Request::PROTO_HTTPS) {
  119. getSocket()->prepareSecureConnection();
  120. if(!getSocket()->initiateSecureConnection(getRequest()->getHost())) {
  121. setReadCheckSocketIf(getSocket(), getSocket()->wantRead());
  122. setWriteCheckSocketIf(getSocket(), getSocket()->wantWrite());
  123. getDownloadEngine()->addCommand(this);
  124. return false;
  125. }
  126. }
  127. if(httpConnection_->sendBufferIsEmpty()) {
  128. if(!checkIfConnectionEstablished
  129. (getSocket(), getRequest()->getConnectedHostname(),
  130. getRequest()->getConnectedAddr(), getRequest()->getConnectedPort())) {
  131. return true;
  132. }
  133. if(getSegments().empty()) {
  134. SharedHandle<HttpRequest> httpRequest
  135. (createHttpRequest(getRequest(),
  136. getFileEntry(),
  137. SharedHandle<Segment>(),
  138. getRequestGroup()->getTotalLength(),
  139. getOption(),
  140. getRequestGroup(),
  141. getDownloadEngine()->getCookieStorage(),
  142. getDownloadEngine()->getAuthConfigFactory(),
  143. proxyRequest_));
  144. if(getOption()->getAsBool(PREF_CONDITIONAL_GET) &&
  145. (getRequest()->getProtocol() == Request::PROTO_HTTP ||
  146. getRequest()->getProtocol() == Request::PROTO_HTTPS)) {
  147. if(getFileEntry()->getPath().empty() &&
  148. getRequest()->getFile().empty()) {
  149. A2_LOG_DEBUG("Conditional-Get is disabled because file name"
  150. " is not available.");
  151. } else {
  152. if(getFileEntry()->getPath().empty()) {
  153. getFileEntry()->setPath
  154. (util::createSafePath
  155. (getOption()->get(PREF_DIR),
  156. util::percentDecode(getRequest()->getFile().begin(),
  157. getRequest()->getFile().end())));
  158. }
  159. File ctrlfile(getFileEntry()->getPath()+
  160. DefaultBtProgressInfoFile::getSuffix());
  161. File file(getFileEntry()->getPath());
  162. if(!ctrlfile.exists() && file.exists()) {
  163. httpRequest->setIfModifiedSinceHeader
  164. (file.getModifiedTime().toHTTPDate());
  165. }
  166. }
  167. }
  168. httpConnection_->sendRequest(httpRequest);
  169. } else {
  170. for(std::vector<SharedHandle<Segment> >::const_iterator itr =
  171. getSegments().begin(), eoi = getSegments().end();
  172. itr != eoi; ++itr) {
  173. const SharedHandle<Segment>& segment = *itr;
  174. if(!httpConnection_->isIssued(segment)) {
  175. int64_t endOffset = 0;
  176. if(getRequestGroup()->getTotalLength() > 0 && getPieceStorage()) {
  177. size_t nextIndex =
  178. getPieceStorage()->getNextUsedIndex(segment->getIndex());
  179. endOffset = std::min
  180. (getFileEntry()->getLength(),
  181. getFileEntry()->gtoloff
  182. (static_cast<int64_t>(segment->getSegmentLength())*nextIndex));
  183. }
  184. SharedHandle<HttpRequest> httpRequest
  185. (createHttpRequest(getRequest(),
  186. getFileEntry(),
  187. segment,
  188. getRequestGroup()->getTotalLength(),
  189. getOption(),
  190. getRequestGroup(),
  191. getDownloadEngine()->getCookieStorage(),
  192. getDownloadEngine()->getAuthConfigFactory(),
  193. proxyRequest_,
  194. endOffset));
  195. httpConnection_->sendRequest(httpRequest);
  196. }
  197. }
  198. }
  199. } else {
  200. httpConnection_->sendPendingData();
  201. }
  202. if(httpConnection_->sendBufferIsEmpty()) {
  203. Command* command = new HttpResponseCommand(getCuid(),
  204. getRequest(),
  205. getFileEntry(),
  206. getRequestGroup(),
  207. httpConnection_,
  208. getDownloadEngine(),
  209. getSocket());
  210. getDownloadEngine()->addCommand(command);
  211. return true;
  212. } else {
  213. setReadCheckSocketIf(getSocket(), getSocket()->wantRead());
  214. setWriteCheckSocketIf(getSocket(), getSocket()->wantWrite());
  215. getDownloadEngine()->addCommand(this);
  216. return false;
  217. }
  218. }
  219. void HttpRequestCommand::setProxyRequest
  220. (const SharedHandle<Request>& proxyRequest)
  221. {
  222. proxyRequest_ = proxyRequest;
  223. }
  224. } // namespace aria2