⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 manager.cc

📁 rtorrent
💻 CC
字号:
// rTorrent - BitTorrent client// Copyright (C) 2005-2006, Jari Sundell//// 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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  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.//// Contact:  Jari Sundell <jaris@ifi.uio.no>////           Skomakerveien 33//           3185 Skoppum, NORWAY#include "config.h"#include <stdexcept>#include <cstdio>#include <cstring>#include <fstream>#include <unistd.h>#include <sys/select.h>#include <rak/address_info.h>#include <rak/error_number.h>#include <rak/regex.h>#include <rak/path.h>#include <rak/string_manip.h>#include <sigc++/bind.h>#include <sigc++/hide.h>#include <torrent/object.h>#include <torrent/connection_manager.h>#include <torrent/exceptions.h>#include <torrent/resume.h>#include <torrent/tracker_list.h>#include "utils/variable_map.h"#include "globals.h"#include "curl_get.h"#include "download.h"#include "download_factory.h"#include "download_store.h"#include "http_queue.h"#include "manager.h"#include "poll_manager_epoll.h"#include "poll_manager_kqueue.h"#include "poll_manager_select.h"#include "view.h"namespace core {static voidconnect_signal_network_log(Download* d, torrent::Download::slot_string_type s) {  d->download()->signal_network_log(s);}static voidconnect_signal_storage_log(Download* d, torrent::Download::slot_string_type s) {  d->download()->signal_storage_error(s);}// Need a proper logging class for this.static voidconnect_signal_tracker_dump(Download* d, torrent::Download::slot_dump_type s) {  if (!control->variable()->get_string("tracker_dump").empty())    d->download()->signal_tracker_dump(s);}static voidreceive_tracker_dump(const std::string& url, const char* data, size_t size) {  const std::string& filename = control->variable()->get_string("tracker_dump");  if (filename.empty())    return;  std::fstream fstr(filename.c_str(), std::ios::out | std::ios::app);  if (!fstr.is_open())    return;  fstr << "url: " << url << std::endl << "---" << std::endl;  fstr.write(data, size);  fstr << std::endl <<"---" << std::endl;}// Hmm... find some better place for all this.voidManager::delete_tied(Download* d) {  const std::string& tie = d->variable()->get_string("tied_to_file");  // This should be configurable, need to wait for the variable  // thingie to be implemented.  if (tie.empty())    return;  if (::unlink(rak::path_expand(tie).c_str()) == -1)    push_log("Could not unlink tied file: " + std::string(rak::error_number::current().c_str()));  d->variable()->set("tied_to_file", std::string());}Manager::Manager() :  m_hashingView(NULL),  m_pollManager(NULL),  m_portFirst(6890),  m_portLast(6999) {  m_downloadStore = new DownloadStore();  m_downloadList = new DownloadList();  m_httpQueue = new HttpQueue();}Manager::~Manager() {  delete m_downloadList;  delete m_downloadStore;  delete m_httpQueue;}voidManager::set_hashing_view(View* v) {  if (v == NULL || m_hashingView != NULL)    throw torrent::client_error("Manager::set_hashing_view(...) received NULL or is already set.");  m_hashingView = v;  v->signal_changed().connect(sigc::mem_fun(this, &Manager::receive_hashing_changed));}voidManager::initialize_first() {  if ((m_pollManager = PollManagerEPoll::create(sysconf(_SC_OPEN_MAX))) != NULL)    m_logImportant.push_front("Using 'epoll' based polling.");  else if ((m_pollManager = PollManagerKQueue::create(sysconf(_SC_OPEN_MAX))) != NULL)    m_logImportant.push_front("Using 'kqueue' based polling.");  else if ((m_pollManager = PollManagerSelect::create(sysconf(_SC_OPEN_MAX))) != NULL)    m_logImportant.push_front("Using 'select' based polling.");  else    throw std::runtime_error("Could not create any PollManager.");  // Need to initialize this before parseing options.  torrent::initialize(m_pollManager->get_torrent_poll());}// Most of this should be possible to move out.voidManager::initialize_second() {  torrent::Http::set_factory(m_pollManager->get_http_stack()->get_http_factory());  m_httpQueue->slot_factory(m_pollManager->get_http_stack()->get_http_factory());  CurlStack::global_init();  // Register slots to be called when a download is inserted/erased,  // opened or closed.  m_downloadList->slot_map_insert()["1_connect_network_log"]  = sigc::bind(sigc::ptr_fun(&connect_signal_network_log), sigc::mem_fun(m_logComplete, &Log::push_front));  m_downloadList->slot_map_insert()["1_connect_storage_log"]  = sigc::bind(sigc::ptr_fun(&connect_signal_storage_log), sigc::mem_fun(m_logComplete, &Log::push_front));  m_downloadList->slot_map_insert()["1_connect_tracker_dump"] = sigc::bind(sigc::ptr_fun(&connect_signal_tracker_dump), sigc::ptr_fun(&receive_tracker_dump));  m_downloadList->slot_map_erase()["1_delete_tied"] = sigc::mem_fun(this, &Manager::delete_tied);}voidManager::cleanup() {  // Need to disconnect log signals? Not really since we won't receive  // any more.  m_downloadList->clear();  torrent::cleanup();  CurlStack::global_cleanup();  delete m_pollManager;}voidManager::shutdown(bool force) {  // This doesn't trigger a compiler error on gcc-3.4.5 for some reason.//   if (!force)//     std::for_each(m_downloadList->begin(), m_downloadList->end(), std::bind1st(std::mem_fun(&DownloadList::pause), &m_downloadList));//   else//     std::for_each(m_downloadList->begin(), m_downloadList->end(), std::bind1st(std::mem_fun(&DownloadList::close), &m_downloadList));  if (!force)    std::for_each(m_downloadList->begin(), m_downloadList->end(), std::bind1st(std::mem_fun(&DownloadList::pause), m_downloadList));  else    std::for_each(m_downloadList->begin(), m_downloadList->end(), std::bind1st(std::mem_fun(&DownloadList::close), m_downloadList));}voidManager::listen_open() {  if (!control->variable()->get_value("port_open"))    return;  if (m_portFirst > m_portLast)    throw torrent::input_error("Invalid port range for listening");  if (control->variable()->get_value("port_random")) {    int boundary = m_portFirst + random() % (m_portLast - m_portFirst + 1);    if (torrent::connection_manager()->listen_open(boundary, m_portLast) ||        torrent::connection_manager()->listen_open(m_portFirst, boundary))      return;  } else {    if (torrent::connection_manager()->listen_open(m_portFirst, m_portLast))      return;  }  throw torrent::input_error("Could not open/bind a port for listening: " + std::string(rak::error_number::current().c_str()));}std::stringManager::bind_address() const {  return rak::socket_address::cast_from(torrent::connection_manager()->bind_address())->address_str();}voidManager::set_bind_address(const std::string& addr) {  int err;  rak::address_info* ai;  if ((err = rak::address_info::get_address_info(addr.c_str(), PF_INET, SOCK_STREAM, &ai)) != 0)    throw torrent::input_error("Could not set bind address: " + std::string(rak::address_info::strerror(err)) + ".");    try {    if (torrent::connection_manager()->listen_port() != 0) {      torrent::connection_manager()->listen_close();      torrent::connection_manager()->set_bind_address(ai->address()->c_sockaddr());      listen_open();    } else {      torrent::connection_manager()->set_bind_address(ai->address()->c_sockaddr());    }    m_pollManager->get_http_stack()->set_bind_address(!ai->address()->is_address_any() ? ai->address()->address_str() : std::string());    rak::address_info::free_address_info(ai);  } catch (torrent::input_error& e) {    rak::address_info::free_address_info(ai);    throw e;  }}std::stringManager::local_address() const {  return rak::socket_address::cast_from(torrent::connection_manager()->local_address())->address_str();}voidManager::set_local_address(const std::string& addr) {  int err;  rak::address_info* ai;  if ((err = rak::address_info::get_address_info(addr.c_str(), PF_INET, SOCK_STREAM, &ai)) != 0)    throw torrent::input_error("Could not set local address: " + std::string(rak::address_info::strerror(err)) + ".");    try {    torrent::connection_manager()->set_local_address(ai->address()->c_sockaddr());    rak::address_info::free_address_info(ai);  } catch (torrent::input_error& e) {    rak::address_info::free_address_info(ai);    throw e;  }}voidManager::receive_http_failed(std::string msg) {  m_logImportant.push_front("Http download error: \"" + msg + "\"");  m_logComplete.push_front("Http download error: \"" + msg + "\"");}voidManager::try_create_download(const std::string& uri, bool start, bool printLog, bool tied) {  // Adding download.  DownloadFactory* f = new DownloadFactory(uri, this);  f->variable()->set("tied_to_file", tied ? "yes" : "no");  f->set_start(start);  f->set_print_log(printLog);  f->slot_finished(sigc::bind(sigc::ptr_fun(&rak::call_delete_func<core::DownloadFactory>), f));  f->load();  f->commit();}// Move this somewhere better.voidpath_expand(std::vector<std::string>* paths, const std::string& pattern) {  std::vector<utils::Directory> currentCache;  std::vector<utils::Directory> nextCache;  rak::split_iterator_t<std::string> first = rak::split_iterator(pattern, '/');  rak::split_iterator_t<std::string> last = rak::split_iterator(pattern);      if (first == last)    return;  // Check for initial '/' that indicates the root.  if ((*first).empty()) {    currentCache.push_back(utils::Directory("/"));    ++first;  } else if (rak::trim(*first) == "~") {    currentCache.push_back(utils::Directory("~"));    ++first;  } else {    currentCache.push_back(utils::Directory("."));  }  // Might be an idea to use depth-first search instead.  for (; first != last; ++first) {    rak::regex r(*first);    if (r.pattern().empty())      continue;    // Special case for ".."?    for (std::vector<utils::Directory>::iterator itr = currentCache.begin(); itr != currentCache.end(); ++itr) {      // Only include filenames starting with '.' if the pattern      // starts with the same.      itr->update(r.pattern()[0] != '.');      itr->erase(std::remove_if(itr->begin(), itr->end(), std::not1(r)), itr->end());      std::transform(itr->begin(), itr->end(), std::back_inserter(nextCache), std::bind1st(std::plus<std::string>(), itr->get_path() + "/"));    }    currentCache.clear();    currentCache.swap(nextCache);  }  std::transform(currentCache.begin(), currentCache.end(), std::back_inserter(*paths), std::mem_fun_ref(&utils::Directory::get_path));}voidManager::try_create_download_expand(const std::string& uri, bool start, bool printLog, bool tied) {  std::vector<std::string> paths;  paths.reserve(256);  path_expand(&paths, uri);  if (tied)    for (std::vector<std::string>::iterator itr = paths.begin(); itr != paths.end(); )      if (std::find_if(m_downloadList->begin(), m_downloadList->end(), rak::equal(*itr, rak::bind2nd(std::mem_fun(&Download::variable_string), "tied_to_file")))          != m_downloadList->end())        itr = paths.erase(itr);      else        itr++;  if (!paths.empty())    for (std::vector<std::string>::iterator itr = paths.begin(); itr != paths.end(); ++itr)      try_create_download(*itr, start, printLog, tied);  else    try_create_download(uri, start, printLog, tied);}// DownloadList's hashing related functions don't actually start the// hashing, it only reacts to events. This functions checks the// hashing view and starts hashing if nessesary.voidManager::receive_hashing_changed() {  bool foundHashing = false;    // Try quick hashing all those with hashing == initial, set them to  // something else when failed.  for (View::iterator itr = m_hashingView->begin_visible(), last = m_hashingView->end_visible(); itr != last; ++itr)    try {      if ((*itr)->is_hash_checking()) {        foundHashing = true;        continue;      }      if ((*itr)->is_hash_checked())        throw torrent::client_error("core::Manager::receive_hashing_changed() hash already checked or checking.");        if ((*itr)->is_hash_failed() || (*itr)->variable()->get_value("hashing") != Download::variable_hashing_initial)        continue;      m_downloadList->open_throw(*itr);      torrent::resume_load_progress(*(*itr)->download(), (*itr)->download()->bencode()->get_key("libtorrent_resume"));      if ((*itr)->download()->hash_check(true)) {        // Set this to exit the function so that the delayed hash done        // signal gets triggered before we try any real hashing.        foundHashing = true;      } else {        // Temporary hack.        (*itr)->download()->hash_stop();      }      (*itr)->variable()->set_value("hashing", Download::variable_hashing_rehash);    } catch (torrent::local_error& e) {      (*itr)->set_hash_failed(true);      push_log(e.what());    }  if (foundHashing)    return;  for (View::iterator itr = m_hashingView->begin_visible(), last = m_hashingView->end_visible(); itr != last; ++itr)    try {      if ((*itr)->is_hash_checked() || (*itr)->is_hash_checking())        throw torrent::client_error("core::Manager::receive_hashing_changed() hash already checked or checking.");        if ((*itr)->is_hash_failed())        continue;      m_downloadList->open_throw(*itr);      torrent::resume_load_progress(*(*itr)->download(), (*itr)->download()->bencode()->get_key("libtorrent_resume"));      (*itr)->download()->hash_check(false);      return;    } catch (torrent::local_error& e) {      (*itr)->set_hash_failed(true);      push_log(e.what());    }}}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -