| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265 | /* <!-- copyright *//* * aria2 - The high speed download utility * * Copyright (C) 2006 Tatsuhiro Tsujikawa * * 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 "ServerStatMan.h"#include <cstring>#include <cstdio>#include <algorithm>#include <iterator>#include <vector>#include "ServerStat.h"#include "util.h"#include "RecoverableException.h"#include "a2functional.h"#include "BufferedFile.h"#include "message.h"#include "fmt.h"#include "LogFactory.h"#include "File.h"namespace aria2 {ServerStatMan::ServerStatMan() {}ServerStatMan::~ServerStatMan() {}std::shared_ptr<ServerStat> ServerStatMan::find(const std::string& hostname,                                             const std::string& protocol) const{  std::shared_ptr<ServerStat> ss(new ServerStat(hostname, protocol));  ServerStatSet::iterator i = serverStats_.find(ss);  if(i == serverStats_.end()) {    return std::shared_ptr<ServerStat>();  } else {    return *i;  }}bool ServerStatMan::add(const std::shared_ptr<ServerStat>& serverStat){  ServerStatSet::iterator i = serverStats_.lower_bound(serverStat);  if(i != serverStats_.end() && *(*i) == *serverStat) {    return false;  } else {    serverStats_.insert(i, serverStat);    return true;  }}bool ServerStatMan::save(const std::string& filename) const{  std::string tempfile = filename;  tempfile += "__temp";  {    BufferedFile fp(tempfile.c_str(), BufferedFile::WRITE);    if(!fp) {      A2_LOG_ERROR(fmt(MSG_OPENING_WRITABLE_SERVER_STAT_FILE_FAILED,                       filename.c_str()));      return false;    }    for(ServerStatSet::iterator i = serverStats_.begin(),          eoi = serverStats_.end(); i != eoi; ++i) {      std::string l = (*i)->toString();      l += "\n";      if(fp.write(l.data(), l.size()) != l.size()) {        A2_LOG_ERROR(fmt(MSG_WRITING_SERVER_STAT_FILE_FAILED,                         filename.c_str()));      }    }    if(fp.close() == EOF) {      A2_LOG_ERROR(fmt(MSG_WRITING_SERVER_STAT_FILE_FAILED, filename.c_str()));      return false;    }  }  if(File(tempfile).renameTo(filename)) {    A2_LOG_NOTICE(fmt(MSG_SERVER_STAT_SAVED, filename.c_str()));    return true;  } else {    A2_LOG_ERROR(fmt(MSG_WRITING_SERVER_STAT_FILE_FAILED, filename.c_str()));    return false;  }}namespace {// Field and FIELD_NAMES must have same order except for MAX_FIELD.enum Field {  S_COUNTER,  S_DL_SPEED,  S_HOST,  S_LAST_UPDATED,  S_MC_AVG_SPEED,  S_PROTOCOL,  S_SC_AVG_SPEED,  S_STATUS,  MAX_FIELD};const char* FIELD_NAMES[] = {  "counter",  "dl_speed",  "host",  "last_updated",  "mc_avg_speed",  "protocol",  "sc_avg_speed",  "status",};} // namespacenamespace {int idField(std::string::const_iterator first,            std::string::const_iterator last){  int i;  for(i = 0; i < MAX_FIELD; ++i) {    if(util::streq(first, last, FIELD_NAMES[i])) {      return i;    }  }  return i;}} // namespacebool ServerStatMan::load(const std::string& filename){  BufferedFile fp(filename.c_str(), BufferedFile::READ);  if(!fp) {    A2_LOG_ERROR(fmt(MSG_OPENING_READABLE_SERVER_STAT_FILE_FAILED,                     filename.c_str()));    return false;  }  while(1) {    std::string line = fp.getLine();    if(line.empty()) {      if(fp.eof()) {        break;      } else if(!fp) {        A2_LOG_ERROR(fmt(MSG_READING_SERVER_STAT_FILE_FAILED,                         filename.c_str()));        return false;      } else {        continue;      }    }    std::pair<std::string::const_iterator,              std::string::const_iterator> p =      util::stripIter(line.begin(), line.end());    if(p.first == p.second) {      continue;    }    std::vector<Scip> items;    util::splitIter(p.first, p.second, std::back_inserter(items), ',');    std::vector<std::string> m(MAX_FIELD);    for(std::vector<Scip>::const_iterator i = items.begin(),          eoi = items.end(); i != eoi; ++i) {      auto p = util::divide((*i).first, (*i).second, '=');      int id = idField(p.first.first, p.first.second);      if(id != MAX_FIELD) {        m[id].assign(p.second.first, p.second.second);      }    }    if(m[S_HOST].empty() || m[S_PROTOCOL].empty()) {      continue;    }    std::shared_ptr<ServerStat> sstat(new ServerStat(m[S_HOST], m[S_PROTOCOL]));    uint32_t uintval;    if(!util::parseUIntNoThrow(uintval, m[S_DL_SPEED])) {      continue;    }    sstat->setDownloadSpeed(uintval);    // Old serverstat file doesn't contains SC_AVG_SPEED    if(!m[S_SC_AVG_SPEED].empty()) {      if(!util::parseUIntNoThrow(uintval, m[S_SC_AVG_SPEED])) {        continue;      }      sstat->setSingleConnectionAvgSpeed(uintval);    }    // Old serverstat file doesn't contains MC_AVG_SPEED    if(!m[S_MC_AVG_SPEED].empty()) {      if(!util::parseUIntNoThrow(uintval, m[S_MC_AVG_SPEED])) {        continue;      }      sstat->setMultiConnectionAvgSpeed(uintval);    }    // Old serverstat file doesn't contains COUNTER_SPEED    if(!m[S_COUNTER].empty()) {      if(!util::parseUIntNoThrow(uintval, m[S_COUNTER])) {        continue;      }      sstat->setCounter(uintval);    }    int32_t intval;    if(!util::parseIntNoThrow(intval, m[S_LAST_UPDATED])) {      continue;    }    sstat->setLastUpdated(Time(intval));    sstat->setStatus(m[S_STATUS]);    add(sstat);  }  A2_LOG_NOTICE(fmt(MSG_SERVER_STAT_LOADED, filename.c_str()));  return true;}namespace {class FindStaleServerStat {private:  time_t timeout_;  Time time_;public:  FindStaleServerStat(time_t timeout):timeout_(timeout) {}  bool operator()(const std::shared_ptr<ServerStat>& ss) const  {    return ss->getLastUpdated().difference(time_) >= timeout_;  }};} // namespacevoid ServerStatMan::removeStaleServerStat(time_t timeout){  FindStaleServerStat finder(timeout);  for(ServerStatSet::iterator i = serverStats_.begin(),        eoi = serverStats_.end(); i != eoi;) {    if(finder(*i)) {      serverStats_.erase(i++);    } else {      ++i;    }  }}} // namespace aria2
 |