HttpRequest.cc 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434
  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 "HttpRequest.h"
  36. #include <cassert>
  37. #include <numeric>
  38. #include <vector>
  39. #include "Segment.h"
  40. #include "Range.h"
  41. #include "CookieStorage.h"
  42. #include "Option.h"
  43. #include "util.h"
  44. #include "Base64.h"
  45. #include "prefs.h"
  46. #include "AuthConfigFactory.h"
  47. #include "AuthConfig.h"
  48. #include "a2functional.h"
  49. #include "TimeA2.h"
  50. #include "array_fun.h"
  51. #include "Request.h"
  52. namespace aria2 {
  53. const std::string HttpRequest::USER_AGENT("aria2");
  54. HttpRequest::HttpRequest():contentEncodingEnabled_(true),
  55. userAgent_(USER_AGENT),
  56. noCache_(true),
  57. acceptGzip_(false),
  58. endOffsetOverride_(0)
  59. {}
  60. HttpRequest::~HttpRequest() {}
  61. void HttpRequest::setSegment(const SharedHandle<Segment>& segment)
  62. {
  63. segment_ = segment;
  64. }
  65. void HttpRequest::setRequest(const SharedHandle<Request>& request)
  66. {
  67. request_ = request;
  68. }
  69. off_t HttpRequest::getStartByte() const
  70. {
  71. if(!segment_) {
  72. return 0;
  73. } else {
  74. return fileEntry_->gtoloff(segment_->getPositionToWrite());
  75. }
  76. }
  77. off_t HttpRequest::getEndByte() const
  78. {
  79. if(!segment_ || !request_) {
  80. return 0;
  81. } else {
  82. if(request_->isPipeliningEnabled()) {
  83. off_t endByte =
  84. fileEntry_->gtoloff(segment_->getPosition()+segment_->getLength()-1);
  85. return std::min(endByte, static_cast<off_t>(fileEntry_->getLength()-1));
  86. } else {
  87. return 0;
  88. }
  89. }
  90. }
  91. RangeHandle HttpRequest::getRange() const
  92. {
  93. // content-length is always 0
  94. if(!segment_) {
  95. return SharedHandle<Range>(new Range());
  96. } else {
  97. return SharedHandle<Range>(new Range(getStartByte(), getEndByte(),
  98. fileEntry_->getLength()));
  99. }
  100. }
  101. bool HttpRequest::isRangeSatisfied(const RangeHandle& range) const
  102. {
  103. if(!segment_) {
  104. return true;
  105. }
  106. if((getStartByte() == range->getStartByte()) &&
  107. ((getEndByte() == 0) ||
  108. (getEndByte() == range->getEndByte())) &&
  109. ((fileEntry_->getLength() == 0) ||
  110. (fileEntry_->getLength() == range->getEntityLength()))) {
  111. return true;
  112. } else {
  113. return false;
  114. }
  115. }
  116. namespace {
  117. std::string getHostText(const std::string& host, uint16_t port)
  118. {
  119. std::string hosttext = host;
  120. if(!(port == 80 || port == 443)) {
  121. strappend(hosttext, ":", util::uitos(port));
  122. }
  123. return hosttext;
  124. }
  125. } // namespace
  126. std::string HttpRequest::createRequest()
  127. {
  128. authConfig_ = authConfigFactory_->createAuthConfig(request_, option_);
  129. std::string requestLine = request_->getMethod();
  130. requestLine += " ";
  131. if(proxyRequest_) {
  132. if(getProtocol() == Request::PROTO_FTP &&
  133. request_->getUsername().empty() && authConfig_) {
  134. // Insert user into URI, like ftp://USER@host/
  135. std::string uri = getCurrentURI();
  136. assert(uri.size() >= 6);
  137. uri.insert(6, util::percentEncode(authConfig_->getUser())+"@");
  138. requestLine += uri;
  139. } else {
  140. requestLine += getCurrentURI();
  141. }
  142. } else {
  143. if(getDir() == A2STR::SLASH_C) {
  144. requestLine += getDir();
  145. } else {
  146. requestLine += getDir();
  147. requestLine += A2STR::SLASH_C;
  148. }
  149. requestLine += getFile();
  150. requestLine += getQuery();
  151. }
  152. requestLine += " HTTP/1.1\r\n";
  153. std::vector<std::pair<std::string, std::string> > builtinHds;
  154. builtinHds.reserve(20);
  155. builtinHds.push_back(std::make_pair("User-Agent:", userAgent_));
  156. std::string acceptTypes = "*/*";
  157. for(std::vector<std::string>::const_iterator i = acceptTypes_.begin(),
  158. eoi = acceptTypes_.end(); i != eoi; ++i) {
  159. strappend(acceptTypes, ",", (*i));
  160. }
  161. builtinHds.push_back(std::make_pair("Accept:", acceptTypes));
  162. if(contentEncodingEnabled_) {
  163. std::string acceptableEncodings;
  164. #ifdef HAVE_LIBZ
  165. if(acceptGzip_) {
  166. acceptableEncodings += "deflate, gzip";
  167. }
  168. #endif // HAVE_LIBZ
  169. if(!acceptableEncodings.empty()) {
  170. builtinHds.push_back
  171. (std::make_pair("Accept-Encoding:", acceptableEncodings));
  172. }
  173. }
  174. builtinHds.push_back
  175. (std::make_pair("Host:", getHostText(getURIHost(), getPort())));
  176. if(noCache_) {
  177. builtinHds.push_back(std::make_pair("Pragma:", "no-cache"));
  178. builtinHds.push_back(std::make_pair("Cache-Control:", "no-cache"));
  179. }
  180. if(!request_->isKeepAliveEnabled() && !request_->isPipeliningEnabled()) {
  181. builtinHds.push_back(std::make_pair("Connection:", "close"));
  182. }
  183. if(segment_ && segment_->getLength() > 0 &&
  184. (request_->isPipeliningEnabled() || getStartByte() > 0)) {
  185. std::string rangeHeader = "bytes=";
  186. rangeHeader += util::itos(getStartByte());
  187. rangeHeader += "-";
  188. if(request_->isPipeliningEnabled()) {
  189. rangeHeader += util::itos(getEndByte());
  190. } else if(getProtocol() != Request::PROTO_FTP && endOffsetOverride_ > 0) {
  191. // FTP via http proxy does not support endbytes
  192. rangeHeader += util::itos(endOffsetOverride_-1);
  193. }
  194. builtinHds.push_back(std::make_pair("Range:", rangeHeader));
  195. }
  196. if(proxyRequest_) {
  197. if(request_->isKeepAliveEnabled() || request_->isPipeliningEnabled()) {
  198. builtinHds.push_back(std::make_pair("Proxy-Connection:", "Keep-Alive"));
  199. } else {
  200. builtinHds.push_back(std::make_pair("Proxy-Connection:", "close"));
  201. }
  202. }
  203. if(proxyRequest_ && !proxyRequest_->getUsername().empty()) {
  204. builtinHds.push_back(getProxyAuthString());
  205. }
  206. if(authConfig_) {
  207. builtinHds.push_back
  208. (std::make_pair("Authorization:",
  209. strconcat("Basic ",
  210. Base64::encode(authConfig_->getAuthText()))));
  211. }
  212. if(getPreviousURI().size()) {
  213. builtinHds.push_back(std::make_pair("Referer:", getPreviousURI()));
  214. }
  215. if(cookieStorage_) {
  216. std::string cookiesValue;
  217. std::vector<Cookie> cookies =
  218. cookieStorage_->criteriaFind
  219. (getHost(),
  220. getDir() == A2STR::SLASH_C?
  221. getDir()+getFile():strconcat(getDir(), A2STR::SLASH_C, getFile()),
  222. Time().getTime(),
  223. getProtocol() == Request::PROTO_HTTPS ?
  224. true : false);
  225. for(std::vector<Cookie>::const_iterator itr = cookies.begin(),
  226. eoi = cookies.end(); itr != eoi; ++itr) {
  227. strappend(cookiesValue, (*itr).toString(), ";");
  228. }
  229. if(!cookiesValue.empty()) {
  230. builtinHds.push_back(std::make_pair("Cookie:", cookiesValue));
  231. }
  232. }
  233. if(!ifModSinceHeader_.empty()) {
  234. builtinHds.push_back
  235. (std::make_pair("If-Modified-Since:", ifModSinceHeader_));
  236. }
  237. for(std::vector<std::pair<std::string, std::string> >::const_iterator i =
  238. builtinHds.begin(), eoi = builtinHds.end(); i != eoi; ++i) {
  239. std::vector<std::string>::const_iterator j = headers_.begin();
  240. std::vector<std::string>::const_iterator jend = headers_.end();
  241. for(; j != jend; ++j) {
  242. if(util::startsWith(*j, (*i).first)) {
  243. break;
  244. }
  245. }
  246. if(j == jend) {
  247. strappend(requestLine, (*i).first, " ", (*i).second, A2STR::CRLF);
  248. }
  249. }
  250. // append additional headers given by user.
  251. for(std::vector<std::string>::const_iterator i = headers_.begin(),
  252. eoi = headers_.end(); i != eoi; ++i) {
  253. strappend(requestLine, (*i), A2STR::CRLF);
  254. }
  255. requestLine += A2STR::CRLF;
  256. return requestLine;
  257. }
  258. std::string HttpRequest::createProxyRequest() const
  259. {
  260. assert(proxyRequest_);
  261. std::string hostport = getURIHost();
  262. strappend(hostport, ":", util::uitos(getPort()));
  263. std::string requestLine = "CONNECT ";
  264. strappend(requestLine, hostport, " HTTP/1.1\r\n");
  265. strappend(requestLine, "User-Agent: ", userAgent_, "\r\n");
  266. strappend(requestLine, "Host: ", hostport, "\r\n");
  267. // TODO Is "Proxy-Connection" needed here?
  268. // if(request->isKeepAliveEnabled() || request->isPipeliningEnabled()) {
  269. // requestLine += "Proxy-Connection: Keep-Alive\r\n";
  270. // }else {
  271. // requestLine += "Proxy-Connection: close\r\n";
  272. // }
  273. if(!proxyRequest_->getUsername().empty()) {
  274. std::pair<std::string, std::string> auth = getProxyAuthString();
  275. strappend(requestLine, auth.first, " ", auth.second, A2STR::CRLF);
  276. }
  277. requestLine += A2STR::CRLF;
  278. return requestLine;
  279. }
  280. std::pair<std::string, std::string> HttpRequest::getProxyAuthString() const
  281. {
  282. return std::make_pair
  283. ("Proxy-Authorization:",
  284. strconcat("Basic ",
  285. Base64::encode(strconcat(proxyRequest_->getUsername(),
  286. ":",
  287. proxyRequest_->getPassword()))));
  288. }
  289. void HttpRequest::enableContentEncoding()
  290. {
  291. contentEncodingEnabled_ = true;
  292. }
  293. void HttpRequest::disableContentEncoding()
  294. {
  295. contentEncodingEnabled_ = false;
  296. }
  297. void HttpRequest::addHeader(const std::string& headersString)
  298. {
  299. std::vector<std::string> headers;
  300. util::split(headersString, std::back_inserter(headers), "\n", true);
  301. headers_.insert(headers_.end(), headers.begin(), headers.end());
  302. }
  303. void HttpRequest::addAcceptType(const std::string& type)
  304. {
  305. acceptTypes_.push_back(type);
  306. }
  307. void HttpRequest::setCookieStorage
  308. (const SharedHandle<CookieStorage>& cookieStorage)
  309. {
  310. cookieStorage_ = cookieStorage;
  311. }
  312. void HttpRequest::setAuthConfigFactory
  313. (const SharedHandle<AuthConfigFactory>& factory, const Option* option)
  314. {
  315. authConfigFactory_ = factory;
  316. option_ = option;
  317. }
  318. void HttpRequest::setProxyRequest(const SharedHandle<Request>& proxyRequest)
  319. {
  320. proxyRequest_ = proxyRequest;
  321. }
  322. bool HttpRequest::isProxyRequestSet() const
  323. {
  324. return proxyRequest_;
  325. }
  326. bool HttpRequest::authenticationUsed() const
  327. {
  328. return authConfig_;
  329. }
  330. const SharedHandle<AuthConfig>& HttpRequest::getAuthConfig() const
  331. {
  332. return authConfig_;
  333. }
  334. uint64_t HttpRequest::getEntityLength() const
  335. {
  336. assert(fileEntry_);
  337. return fileEntry_->getLength();
  338. }
  339. const std::string& HttpRequest::getHost() const
  340. {
  341. return request_->getHost();
  342. }
  343. uint16_t HttpRequest::getPort() const
  344. {
  345. return request_->getPort();
  346. }
  347. const std::string& HttpRequest::getMethod() const
  348. {
  349. return request_->getMethod();
  350. }
  351. const std::string& HttpRequest::getProtocol() const
  352. {
  353. return request_->getProtocol();
  354. }
  355. const std::string& HttpRequest::getCurrentURI() const
  356. {
  357. return request_->getCurrentUri();
  358. }
  359. const std::string& HttpRequest::getDir() const
  360. {
  361. return request_->getDir();
  362. }
  363. const std::string& HttpRequest::getFile() const
  364. {
  365. return request_->getFile();
  366. }
  367. const std::string& HttpRequest::getQuery() const
  368. {
  369. return request_->getQuery();
  370. }
  371. const std::string& HttpRequest::getPreviousURI() const
  372. {
  373. return request_->getPreviousUri();
  374. }
  375. std::string HttpRequest::getURIHost() const
  376. {
  377. return request_->getURIHost();
  378. }
  379. void HttpRequest::setUserAgent(const std::string& userAgent)
  380. {
  381. userAgent_ = userAgent;
  382. }
  383. void HttpRequest::setFileEntry(const SharedHandle<FileEntry>& fileEntry)
  384. {
  385. fileEntry_ = fileEntry;
  386. }
  387. void HttpRequest::setIfModifiedSinceHeader(const std::string& hd)
  388. {
  389. ifModSinceHeader_ = hd;
  390. }
  391. } // namespace aria2