ServerStatURISelector.cc 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  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 "ServerStatURISelector.h"
  36. #include "ServerStatMan.h"
  37. #include "ServerStat.h"
  38. #include "Request.h"
  39. #include "A2STR.h"
  40. #include <algorithm>
  41. namespace aria2 {
  42. ServerStatURISelector::ServerStatURISelector
  43. (const SharedHandle<ServerStatMan>& serverStatMan):
  44. _serverStatMan(serverStatMan) {}
  45. ServerStatURISelector::~ServerStatURISelector() {}
  46. class ServerStatFaster {
  47. public:
  48. bool operator()(const std::pair<SharedHandle<ServerStat>, std::string> lhs,
  49. const std::pair<SharedHandle<ServerStat>, std::string> rhs)
  50. const
  51. {
  52. return lhs.first->getDownloadSpeed() > rhs.first->getDownloadSpeed();
  53. }
  54. };
  55. std::string ServerStatURISelector::select(std::deque<std::string>& uris)
  56. {
  57. if(uris.empty()) {
  58. return A2STR::NIL;
  59. }
  60. // Use first 10 URIs to introduce some randomness.
  61. const int NUM_URI = 10;
  62. // Ignore low speed server
  63. const unsigned int SPEED_THRESHOLD = 20*1024;
  64. size_t max = std::min(uris.size(), static_cast<size_t>(NUM_URI));
  65. std::deque<std::string>::iterator urisLast = uris.begin()+max;
  66. std::deque<std::pair<SharedHandle<ServerStat>, std::string> > cands;
  67. for(std::deque<std::string>::iterator i = uris.begin();
  68. i != urisLast; ++i) {
  69. Request r;
  70. r.setUrl(*i);
  71. SharedHandle<ServerStat> ss = _serverStatMan->find(r.getHost(),
  72. r.getProtocol());
  73. if(!ss.isNull() && ss->isOK() && ss->getDownloadSpeed() > SPEED_THRESHOLD) {
  74. cands.push_back(std::pair<SharedHandle<ServerStat>, std::string>(ss, *i));
  75. }
  76. }
  77. if(cands.empty()) {
  78. for(std::deque<std::string>::iterator i = uris.begin();
  79. i != uris.end(); ++i) {
  80. Request r;
  81. r.setUrl(*i);
  82. SharedHandle<ServerStat> ss = _serverStatMan->find(r.getHost(),
  83. r.getProtocol());
  84. // Skip ERROR state URI
  85. if(ss.isNull() || ss->isOK()) {
  86. std::string nextURI = *i;
  87. uris.erase(uris.begin(), i+1);
  88. return nextURI;
  89. }
  90. }
  91. // All URIs are inspected but aria2 cannot find usable one.
  92. // Return first URI anyway in this case.
  93. std::string nextURI = uris.front();
  94. uris.pop_front();
  95. return nextURI;
  96. } else {
  97. std::sort(cands.begin(), cands.end(), ServerStatFaster());
  98. uris.erase(std::find(uris.begin(), uris.end(), cands.front().second));
  99. return cands.front().second;
  100. }
  101. }
  102. } // namespace aria2