HttpRequestCommand.cc 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  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 "SocketCore.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, const std::shared_ptr<Request>& req,
  63. const std::shared_ptr<FileEntry>& fileEntry, RequestGroup* requestGroup,
  64. const std::shared_ptr<HttpConnection>& httpConnection, DownloadEngine* e,
  65. const std::shared_ptr<SocketCore>& s)
  66. : AbstractCommand(cuid, req, fileEntry, requestGroup, e, s,
  67. httpConnection->getSocketRecvBuffer()),
  68. httpConnection_(httpConnection)
  69. {
  70. setTimeout(std::chrono::seconds(getOption()->getAsInt(PREF_CONNECT_TIMEOUT)));
  71. disableReadCheckSocket();
  72. setWriteCheckSocket(getSocket());
  73. }
  74. HttpRequestCommand::~HttpRequestCommand() {}
  75. namespace {
  76. std::unique_ptr<HttpRequest>
  77. createHttpRequest(const std::shared_ptr<Request>& req,
  78. const std::shared_ptr<FileEntry>& fileEntry,
  79. const std::shared_ptr<Segment>& segment,
  80. const std::shared_ptr<Option>& option, const RequestGroup* rg,
  81. const DownloadEngine* e,
  82. const std::shared_ptr<Request>& proxyRequest,
  83. int64_t endOffset = 0)
  84. {
  85. auto httpRequest = make_unique<HttpRequest>();
  86. httpRequest->setUserAgent(option->get(PREF_USER_AGENT));
  87. httpRequest->setRequest(req);
  88. httpRequest->setFileEntry(fileEntry);
  89. httpRequest->setSegment(segment);
  90. httpRequest->addHeader(option->get(PREF_HEADER));
  91. httpRequest->setCookieStorage(e->getCookieStorage().get());
  92. httpRequest->setAuthConfigFactory(e->getAuthConfigFactory().get());
  93. httpRequest->setOption(option.get());
  94. httpRequest->setProxyRequest(proxyRequest);
  95. httpRequest->setAcceptMetalink(rg->getDownloadContext()->getAcceptMetalink());
  96. if (option->getAsBool(PREF_HTTP_ACCEPT_GZIP)) {
  97. httpRequest->enableAcceptGZip();
  98. }
  99. else {
  100. httpRequest->disableAcceptGZip();
  101. }
  102. if (option->getAsBool(PREF_HTTP_NO_CACHE)) {
  103. httpRequest->enableNoCache();
  104. }
  105. else {
  106. httpRequest->disableNoCache();
  107. }
  108. if (endOffset > 0) {
  109. httpRequest->setEndOffsetOverride(endOffset);
  110. }
  111. return httpRequest;
  112. }
  113. } // namespace
  114. bool HttpRequestCommand::executeInternal()
  115. {
  116. // socket->setBlockingMode();
  117. if (httpConnection_->sendBufferIsEmpty()) {
  118. #ifdef ENABLE_SSL
  119. if (getRequest()->getProtocol() == "https") {
  120. if (!getSocket()->tlsConnect(getRequest()->getHost())) {
  121. setReadCheckSocketIf(getSocket(), getSocket()->wantRead());
  122. setWriteCheckSocketIf(getSocket(), getSocket()->wantWrite());
  123. addCommandSelf();
  124. return false;
  125. }
  126. }
  127. #endif // ENABLE_SSL
  128. if (getSegments().empty()) {
  129. auto httpRequest = createHttpRequest(
  130. getRequest(), getFileEntry(), std::shared_ptr<Segment>(), getOption(),
  131. getRequestGroup(), getDownloadEngine(), proxyRequest_);
  132. if (getOption()->getAsBool(PREF_CONDITIONAL_GET) &&
  133. (getRequest()->getProtocol() == "http" ||
  134. getRequest()->getProtocol() == "https")) {
  135. std::string path;
  136. if (getFileEntry()->getPath().empty()) {
  137. auto& file = getRequest()->getFile();
  138. // If filename part of URI is empty, we just use
  139. // Request::DEFAULT_FILE, since it is the name we use to
  140. // store file in disk.
  141. path = util::createSafePath(
  142. getOption()->get(PREF_DIR),
  143. (getRequest()->getFile().empty()
  144. ? Request::DEFAULT_FILE
  145. : util::percentDecode(std::begin(file), std::end(file))));
  146. }
  147. else {
  148. path = getFileEntry()->getPath();
  149. }
  150. File ctrlfile(path + DefaultBtProgressInfoFile::getSuffix());
  151. File file(path);
  152. if (!ctrlfile.exists() && file.exists()) {
  153. httpRequest->setIfModifiedSinceHeader(
  154. file.getModifiedTime().toHTTPDate());
  155. }
  156. }
  157. httpConnection_->sendRequest(std::move(httpRequest));
  158. }
  159. else {
  160. for (auto& segment : getSegments()) {
  161. if (!httpConnection_->isIssued(segment)) {
  162. int64_t endOffset = 0;
  163. // FTP via HTTP proxy does not support end byte marker
  164. if (getRequest()->getProtocol() != "ftp" &&
  165. getRequestGroup()->getTotalLength() > 0 && getPieceStorage()) {
  166. size_t nextIndex =
  167. getPieceStorage()->getNextUsedIndex(segment->getIndex());
  168. endOffset =
  169. std::min(getFileEntry()->getLength(),
  170. getFileEntry()->gtoloff(
  171. static_cast<int64_t>(segment->getSegmentLength()) *
  172. nextIndex));
  173. }
  174. httpConnection_->sendRequest(
  175. createHttpRequest(getRequest(), getFileEntry(), segment,
  176. getOption(), getRequestGroup(),
  177. getDownloadEngine(), proxyRequest_, endOffset));
  178. }
  179. }
  180. }
  181. }
  182. else {
  183. httpConnection_->sendPendingData();
  184. }
  185. if (httpConnection_->sendBufferIsEmpty()) {
  186. getDownloadEngine()->addCommand(make_unique<HttpResponseCommand>(
  187. getCuid(), getRequest(), getFileEntry(), getRequestGroup(),
  188. httpConnection_, getDownloadEngine(), getSocket()));
  189. return true;
  190. }
  191. else {
  192. setReadCheckSocketIf(getSocket(), getSocket()->wantRead());
  193. setWriteCheckSocketIf(getSocket(), getSocket()->wantWrite());
  194. addCommandSelf();
  195. return false;
  196. }
  197. }
  198. void HttpRequestCommand::setProxyRequest(
  199. const std::shared_ptr<Request>& proxyRequest)
  200. {
  201. proxyRequest_ = proxyRequest;
  202. }
  203. } // namespace aria2