| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369 | /* <!-- copyright *//* * aria2 - The high speed download utility * * Copyright (C) 2006 Tatsuhiro Tsujikawa * Copyright (C) 2008 Aurelien Lefebvre, Mandriva * * 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 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * 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. *//* copyright --> */#include "AdaptiveURISelector.h"#include <cstdlib>#include <cmath>#include <algorithm>#include "DownloadCommand.h"#include "DownloadContext.h"#include "ServerStatMan.h"#include "ServerStat.h"#include "RequestGroup.h"#include "LogFactory.h"#include "A2STR.h"#include "prefs.h"#include "Option.h"#include "SimpleRandomizer.h"#include "SocketCore.h"#include "FileEntry.h"#include "uri.h"namespace aria2 {/* In that URI Selector, select method returns one of the bests  * mirrors for first and reserved connections. For supplementary  * ones, it returns mirrors which has not been tested yet, and  * if each of them already tested, returns mirrors which has to  * be tested again. Otherwise, it doesn't return anymore mirrors. */AdaptiveURISelector::AdaptiveURISelector(const SharedHandle<ServerStatMan>& serverStatMan, RequestGroup* requestGroup):  serverStatMan_(serverStatMan),  requestGroup_(requestGroup),  logger_(LogFactory::getInstance()){  resetCounters();}AdaptiveURISelector::~AdaptiveURISelector() {}std::string AdaptiveURISelector::select(FileEntry* fileEntry, const std::vector<std::pair<size_t, std::string> >& usedHosts){  if(logger_->debug()) {    logger_->debug("AdaptiveURISelector: called %d",                   requestGroup_->getNumConnection());  }  std::deque<std::string>& uris = fileEntry->getRemainingUris();  if (uris.empty() && requestGroup_->getNumConnection() <= 1) {    // here we know the download will fail, trying to find previously    // failed uris that may succeed with more permissive values    mayRetryWithIncreasedTimeout(fileEntry);  }   std::string selected = selectOne(uris);  if(selected != A2STR::NIL)    uris.erase(std::find(uris.begin(), uris.end(), selected));  return selected;}void AdaptiveURISelector::mayRetryWithIncreasedTimeout(FileEntry* fileEntry){  if (requestGroup_->getTimeout()*2 >= MAX_TIMEOUT) return;  requestGroup_->setTimeout(requestGroup_->getTimeout()*2);  std::deque<std::string>& uris = fileEntry->getRemainingUris();  // looking for retries  std::deque<URIResult> timeouts;  fileEntry->extractURIResult(timeouts, downloadresultcode::TIME_OUT);  std::transform(timeouts.begin(), timeouts.end(), std::back_inserter(uris),                 std::mem_fun_ref(&URIResult::getURI));  if(logger_->debug()) {    for(std::deque<std::string>::const_iterator i = uris.begin(),          eoi = uris.end(); i != eoi; ++i) {      logger_->debug("AdaptiveURISelector: will retry server with increased"                     " timeout (%ld s): %s",                     static_cast<long int>(requestGroup_->getTimeout()),                     (*i).c_str());    }  }}std::string AdaptiveURISelector::selectOne(const std::deque<std::string>& uris){  if(uris.empty()) {    return A2STR::NIL;  } else {    const unsigned int numPieces =      requestGroup_->getDownloadContext()->getNumPieces();    bool reservedContext = numPieces > 0 &&       nbConnections_ > std::min(numPieces,                                requestGroup_->getNumConcurrentCommand());    bool selectBest = numPieces == 0 || reservedContext;        if(numPieces > 0)      ++nbConnections_;    /* At least, 3 mirrors must be tested */    if(getNbTestedServers(uris) < 3) {      std::string notTested = getFirstNotTestedUri(uris);      if(notTested != A2STR::NIL) {        if(logger_->debug()) {          logger_->debug("AdaptiveURISelector: choosing the first non tested"                         " mirror: %s", notTested.c_str());        }        --nbServerToEvaluate_;        return notTested;      }    }        if(!selectBest && nbConnections_ > 1 && nbServerToEvaluate_ > 0) {      nbServerToEvaluate_--;      std::string notTested = getFirstNotTestedUri(uris);      if(notTested != A2STR::NIL) {        /* Here we return the first untested mirror */        if(logger_->debug()) {          logger_->debug("AdaptiveURISelector: choosing non tested mirror %s"                         " for connection #%d",                         notTested.c_str(), nbConnections_);        }        return notTested;      } else {        /* Here we return a mirror which need to be tested again */        std::string toReTest = getFirstToTestUri(uris);        if(toReTest != A2STR::NIL) {          if(logger_->debug()) {            logger_->debug("AdaptiveURISelector: choosing mirror %s which has"                           " not been tested recently for connection #%d",                           toReTest.c_str(), nbConnections_);          }          return toReTest;        } else {          return getBestMirror(uris);        }      }    }    else {      return getBestMirror(uris);    }  }}std::string AdaptiveURISelector::getBestMirror(const std::deque<std::string>& uris) const{  /* Here we return one of the bests mirrors */  unsigned int max = getMaxDownloadSpeed(uris);  unsigned int min = max-(int)(max*0.25);  std::deque<std::string> bests = getUrisBySpeed(uris, min);    if (bests.size() < 2) {    std::string uri = getMaxDownloadSpeedUri(uris);    if(logger_->debug()) {      logger_->debug("AdaptiveURISelector: choosing the best mirror :"                     " %.2fKB/s %s (other mirrors are at least 25%% slower)",                     (float) max/1024, uri.c_str());    }    return uri;  } else {    std::string uri = selectRandomUri(bests);    if(logger_->debug()) {      logger_->debug("AdaptiveURISelector: choosing randomly one of the best"                     " mirrors (range [%.2fKB/s, %.2fKB/s]): %s",                     (float) min/1024, (float) max/1024, uri.c_str());    }    return uri;  }}void AdaptiveURISelector::resetCounters(){  nbConnections_ = 1;  nbServerToEvaluate_ =    requestGroup_->getOption()->getAsInt(PREF_METALINK_SERVERS) - 1;}void AdaptiveURISelector::tuneDownloadCommand(const std::deque<std::string>& uris, DownloadCommand* command){  adjustLowestSpeedLimit(uris, command);}void AdaptiveURISelector::adjustLowestSpeedLimit(const std::deque<std::string>& uris, DownloadCommand* command) const{  unsigned int lowest =    requestGroup_->getOption()->getAsInt(PREF_LOWEST_SPEED_LIMIT);  if (lowest > 0) {    unsigned int low_lowest = 4 * 1024;    unsigned int max = getMaxDownloadSpeed(uris);    if (max > 0 && lowest > max / 4) {      logger_->notice("Lowering lowest-speed-limit since known max speed is too"                      " near (new:%d was:%d max:%d)", max / 4, lowest, max);      command->setLowestDownloadSpeedLimit(max / 4);    } else if (max == 0 && lowest > low_lowest) {      logger_->notice("Lowering lowest-speed-limit since we have no clue about"                      " available speed (now:%d was:%d)", low_lowest, lowest);      command->setLowestDownloadSpeedLimit(low_lowest);    }  }}namespace {unsigned int getUriMaxSpeed(SharedHandle<ServerStat> ss){  return std::max(ss->getSingleConnectionAvgSpeed(),                  ss->getMultiConnectionAvgSpeed());}} // namespaceunsigned int AdaptiveURISelector::getMaxDownloadSpeed(const std::deque<std::string>& uris) const{  std::string uri = getMaxDownloadSpeedUri(uris);  if(uri == A2STR::NIL)    return 0;  return getUriMaxSpeed(getServerStats(uri));}std::string AdaptiveURISelector::getMaxDownloadSpeedUri(const std::deque<std::string>& uris) const{  int max = -1;  std::string uri = A2STR::NIL;  for(std::deque<std::string>::const_iterator i = uris.begin(),        eoi = uris.end(); i != eoi; ++i) {    SharedHandle<ServerStat> ss = getServerStats(*i);    if(!ss)      continue;    if((int)ss->getSingleConnectionAvgSpeed() > max) {      max = ss->getSingleConnectionAvgSpeed();      uri = (*i);    }    if((int)ss->getMultiConnectionAvgSpeed() > max) {      max = ss->getMultiConnectionAvgSpeed();      uri = (*i);    }  }  return uri;}std::deque<std::string> AdaptiveURISelector::getUrisBySpeed(const std::deque<std::string>& uris, unsigned int min) const{  std::deque<std::string> bests;  for(std::deque<std::string>::const_iterator i = uris.begin(),        eoi = uris.end(); i != eoi; ++i) {    SharedHandle<ServerStat> ss = getServerStats(*i);    if(!ss)      continue;    if(ss->getSingleConnectionAvgSpeed() > min ||       ss->getMultiConnectionAvgSpeed() > min) {      bests.push_back(*i);    }  }  return bests;}std::string AdaptiveURISelector::selectRandomUri(const std::deque<std::string>& uris) const{  int pos = SimpleRandomizer::getInstance()->getRandomNumber(uris.size());  std::deque<std::string>::const_iterator i = uris.begin();  i = i+pos;  return *i;}std::string AdaptiveURISelector::getFirstNotTestedUri(const std::deque<std::string>& uris) const{  for(std::deque<std::string>::const_iterator i = uris.begin(),        eoi = uris.end(); i != eoi; ++i) {    SharedHandle<ServerStat> ss = getServerStats(*i);    if(!ss)      return *i;  }  return A2STR::NIL;}std::string AdaptiveURISelector::getFirstToTestUri(const std::deque<std::string>& uris) const{  unsigned int counter;  int power;  for(std::deque<std::string>::const_iterator i = uris.begin(),        eoi = uris.end(); i != eoi; ++i) {    SharedHandle<ServerStat> ss = getServerStats(*i);    if(!ss)      continue;    counter = ss->getCounter();    if(counter > 8)      continue;    power = (int)pow(2.0, (float)counter);    /* We test the mirror another time if it has not been     * tested since 2^counter days */    if(ss->getLastUpdated().difference() > power*24*60*60) {      return *i;    }  }  return A2STR::NIL;}SharedHandle<ServerStat> AdaptiveURISelector::getServerStats(const std::string& uri) const{  uri::UriStruct us;  if(uri::parse(us, uri)) {    return serverStatMan_->find(us.host, us.protocol);  } else {    return SharedHandle<ServerStat>();  }}unsigned int AdaptiveURISelector::getNbTestedServers(const std::deque<std::string>& uris) const{  unsigned int counter = 0;  for(std::deque<std::string>::const_iterator i = uris.begin(),        eoi = uris.end(); i != eoi; ++i) {    SharedHandle<ServerStat> ss = getServerStats(*i);    if(!ss)      ++counter;  }  return uris.size() - counter;}} // namespace aria2
 |