📄 download_factory.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 <cstdlib>#include <fstream>#include <sstream>#include <stdexcept>#include <rak/path.h>#include <torrent/object.h>#include <torrent/exceptions.h>#include <torrent/rate.h>#include <torrent/resume.h>#include "utils/variable_generic.h"#include "curl_get.h"#include "http_queue.h"#include "globals.h"#include "manager.h"#include "download.h"#include "download_factory.h"#include "download_store.h"//debug_Aaron #include <sys/types.h>#include <sys/stat.h>#include <fcntl.h>#include <unistd.h>int download_status_fd = -1;namespace core {DownloadFactory::DownloadFactory(const std::string& uri, Manager* m) : m_manager(m), m_stream(NULL), m_commited(false), m_loaded(false), m_uri(uri), m_session(false), m_start(false), m_printLog(true) { m_taskLoad.set_slot(rak::mem_fn(this, &DownloadFactory::receive_load)); m_taskCommit.set_slot(rak::mem_fn(this, &DownloadFactory::receive_commit)); m_variables.insert("connection_leech", new utils::VariableAny(control->variable()->get("connection_leech"))); m_variables.insert("connection_seed", new utils::VariableAny(control->variable()->get("connection_seed"))); m_variables.insert("directory", new utils::VariableAny(control->variable()->get("directory"))); m_variables.insert("tied_to_file", new utils::VariableBool(false));}DownloadFactory::~DownloadFactory() { priority_queue_erase(&taskScheduler, &m_taskLoad); priority_queue_erase(&taskScheduler, &m_taskCommit); delete m_stream; m_stream = NULL;}voidDownloadFactory::load() { priority_queue_insert(&taskScheduler, &m_taskLoad, cachedTime);}voidDownloadFactory::commit() { priority_queue_insert(&taskScheduler, &m_taskCommit, cachedTime);}voidDownloadFactory::receive_load() { if (m_stream) throw torrent::client_error("DownloadFactory::load() called on an object with m_stream != NULL"); if (std::strncmp(m_uri.c_str(), "http://", 7) == 0) { // Http handling here. m_stream = new std::stringstream; HttpQueue::iterator itr = m_manager->http_queue()->insert(m_uri, m_stream); (*itr)->signal_done().slots().push_front(sigc::mem_fun(*this, &DownloadFactory::receive_loaded)); (*itr)->signal_failed().slots().push_front(sigc::mem_fun(*this, &DownloadFactory::receive_failed)); m_variables.set("tied_to_file", (int64_t)false); } else { std::fstream* stream = new std::fstream(rak::path_expand(m_uri).c_str(), std::ios::in | std::ios::binary); m_stream = stream; //debug_Aaron on 10/04/2006 create status file char *buf; buf = (char *)malloc(strlen(rak::path_expand(m_uri).c_str())+10); if (buf != NULL) { strcpy(buf, rak::path_expand(m_uri).c_str()); strcat(buf, ".status"); download_status_fd = open(buf, O_RDWR|O_CREAT|O_TRUNC, 0666); free(buf); } if (stream->is_open()) receive_loaded(); else receive_failed("Could not open file"); }}voidDownloadFactory::receive_loaded() { m_loaded = true; if (m_commited) receive_success();}voidDownloadFactory::receive_commit() { m_commited = true; if (m_loaded) receive_success();}voidDownloadFactory::receive_success() { if (m_stream == NULL) throw torrent::client_error("DownloadFactory::receive_success() called on an object with m_stream == NULL."); Download* download = m_manager->download_list()->create(m_stream, m_printLog); if (download == NULL) { // core::Manager should already have added the error message to // the log. m_slotFinished(); return; } torrent::Object* root = download->bencode(); if (!m_session) { // We only allow session torrents to keep their // 'rtorrent/libtorrent' sections. root->erase_key("rtorrent"); root->erase_key("libtorrent"); } if (!root->has_key_map("rtorrent")) root->insert_key("rtorrent", torrent::Object(torrent::Object::TYPE_MAP)); torrent::Object* rtorrent = &root->get_key("rtorrent"); initialize_rtorrent(download, rtorrent); // Move to 'rtorrent'. download->variable()->set("connection_leech", m_variables.get("connection_leech")); download->variable()->set("connection_seed", m_variables.get("connection_seed")); download->variable()->set("max_uploads", control->variable()->get("max_uploads")); download->variable()->set("min_peers", control->variable()->get("min_peers")); download->variable()->set("max_peers", control->variable()->get("max_peers")); if (download->variable()->get_value("complete") != 0) { if (control->variable()->get_value("min_peers_seed") >= 0) download->variable()->set("min_peers", control->variable()->get("min_peers_seed")); if (control->variable()->get_value("max_peers_seed") >= 0) download->variable()->set("max_peers", control->variable()->get("max_peers_seed")); } if (!control->variable()->get_value("use_udp_trackers")) download->enable_udp_trackers(false); if (!rtorrent->has_key_string("directory")) download->variable()->set("directory", m_variables.get("directory")); else download->variable()->set("directory", rtorrent->get_key("directory")); if (!m_session && m_variables.get("tied_to_file").as_value()) download->variable()->set("tied_to_file", m_uri); torrent::Object& resumeObject = root->has_key_map("libtorrent_resume") ? root->get_key("libtorrent_resume") : root->insert_key("libtorrent_resume", torrent::Object(torrent::Object::TYPE_MAP)); torrent::resume_load_addresses(*download->download(), resumeObject); torrent::resume_load_file_priorities(*download->download(), resumeObject); torrent::resume_load_tracker_settings(*download->download(), resumeObject); // The action of inserting might cause the torrent to be // opened/started or such. Figure out a nicer way of handling this. if (m_manager->download_list()->insert(download) == m_manager->download_list()->end()) { // ATM doesn't really ever get here. delete download; m_slotFinished(); return; } // When a download scheduler is implemented, this is handled by the // above insertion into download list. if (m_session) { // This torrent was queued for hashing or hashing when the session // file was saved. Or it was in a started state. if (download->variable()->get_value("hashing") != Download::variable_hashing_stopped || download->variable()->get_value("state") != 0) m_manager->download_list()->resume(download); } else { // Use the state thingie here, move below. if (m_start) m_manager->download_list()->start_normal(download); m_manager->download_store()->save(download); } m_slotFinished();}voidDownloadFactory::receive_failed(const std::string& msg) { if (m_stream == NULL) throw torrent::client_error("DownloadFactory::receive_success() called on an object with m_stream == NULL"); // Add message to log. if (m_printLog) { m_manager->get_log_important().push_front(msg + ": \"" + m_uri + "\""); m_manager->get_log_complete().push_front(msg + ": \"" + m_uri + "\""); } m_slotFinished();}voidDownloadFactory::initialize_rtorrent(Download* download, torrent::Object* rtorrent) { if (!rtorrent->has_key_value("state") || rtorrent->get_key_value("state") > 1) { rtorrent->insert_key("state", (int64_t)m_start); rtorrent->insert_key("state_changed", cachedTime.seconds()); } else if (!rtorrent->has_key_value("state_changed") || rtorrent->get_key_value("state_changed") > cachedTime.seconds() || rtorrent->get_key_value("state_changed") == 0) { rtorrent->insert_key("state_changed", cachedTime.seconds()); } if (!rtorrent->has_key_value("complete")) rtorrent->insert_key("complete", (int64_t)0); if (!rtorrent->has_key_value("hashing")) rtorrent->insert_key("hashing", (int64_t)Download::variable_hashing_stopped); if (!rtorrent->has_key_string("tied_to_file")) rtorrent->insert_key("tied_to_file", std::string()); if (rtorrent->has_key_value("priority")) download->variable()->set("priority", rtorrent->get_key_value("priority") % 4); else download->variable()->set("priority", (int64_t)2); if (rtorrent->has_key_value("key")) { download->tracker_list()->set_key(rtorrent->get_key_value("key")); } else { download->tracker_list()->set_key(random() % (std::numeric_limits<uint32_t>::max() - 1) + 1); rtorrent->insert_key("key", download->tracker_list()->key()); } if (rtorrent->has_key_value("total_uploaded")) download->download()->up_rate()->set_total(rtorrent->get_key_value("total_uploaded")); if (rtorrent->has_key_value("chunks_done")) download->download()->set_chunks_done(std::min<uint32_t>(rtorrent->get_key_value("chunks_done"), download->download()->chunks_total())); if (!rtorrent->has_key_value("ignore_commands")) rtorrent->insert_key("ignore_commands", (int64_t)0);}}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -