/* Copyright (c) 2003-2017, Arvid Norberg All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of the author nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include // for snprintf #include // for atoi #include #include #include #include #include #include // for min()/max() #include "libtorrent/config.hpp" #ifdef TORRENT_WINDOWS #include // for _mkdir and _getcwd #include // for _stat #include #endif #ifdef TORRENT_UTP_LOG_ENABLE #include "libtorrent/utp_stream.hpp" #endif #include "libtorrent/torrent_info.hpp" #include "libtorrent/announce_entry.hpp" #include "libtorrent/entry.hpp" #include "libtorrent/bencode.hpp" #include "libtorrent/session.hpp" #include "libtorrent/identify_client.hpp" #include "libtorrent/alert_types.hpp" #include "libtorrent/ip_filter.hpp" #include "libtorrent/magnet_uri.hpp" #include "libtorrent/peer_info.hpp" #include "libtorrent/bdecode.hpp" #include "libtorrent/add_torrent_params.hpp" #include "libtorrent/time.hpp" #include "libtorrent/read_resume_data.hpp" #include "libtorrent/write_resume_data.hpp" #include "libtorrent/string_view.hpp" #include "libtorrent/disk_interface.hpp" // for open_file_state #include "torrent_view.hpp" #include "session_view.hpp" #include "print.hpp" using lt::total_milliseconds; using lt::alert; using lt::piece_index_t; using lt::file_index_t; using lt::torrent_handle; using lt::add_torrent_params; using lt::cache_status; using lt::total_seconds; using lt::torrent_flags_t; using lt::seconds; using lt::operator "" _sv; using lt::address_v4; using lt::address_v6; using std::chrono::duration_cast; using std::stoi; #ifdef _WIN32 #include #include bool sleep_and_input(int* c, lt::time_duration const sleep) { for (int i = 0; i < 2; ++i) { if (_kbhit()) { *c = _getch(); return true; } std::this_thread::sleep_for(sleep / 2); } return false; } #else #include #include #include #include #include struct set_keypress { enum terminal_mode { echo = 1, canonical = 2 }; explicit set_keypress(std::uint8_t const mode = 0) { termios new_settings; tcgetattr(0, &stored_settings); new_settings = stored_settings; // Disable canonical mode, and set buffer size to 1 byte // and disable echo if (mode & echo) new_settings.c_lflag |= ECHO; else new_settings.c_lflag &= ~ECHO; if (mode & canonical) new_settings.c_lflag |= ICANON; else new_settings.c_lflag &= ~ICANON; new_settings.c_cc[VTIME] = 0; new_settings.c_cc[VMIN] = 1; tcsetattr(0,TCSANOW,&new_settings); } ~set_keypress() { tcsetattr(0, TCSANOW, &stored_settings); } private: termios stored_settings; }; bool sleep_and_input(int* c, lt::time_duration const sleep) { lt::time_point const done = lt::clock_type::now() + sleep; int ret = 0; retry: fd_set set; FD_ZERO(&set); FD_SET(0, &set); int const delay = total_milliseconds(done - lt::clock_type::now()); timeval tv = {delay / 1000, (delay % 1000) * 1000 }; ret = select(1, &set, nullptr, nullptr, &tv); if (ret > 0) { *c = getc(stdin); return true; } if (errno == EINTR) { if (lt::clock_type::now() < done) goto retry; return false; } if (ret < 0 && errno != 0 && errno != ETIMEDOUT) { std::fprintf(stderr, "select failed: %s\n", strerror(errno)); std::this_thread::sleep_for(lt::milliseconds(500)); } return false; } #endif bool print_trackers = false; bool print_peers = false; bool print_connecting_peers = false; bool print_log = false; bool print_downloads = false; bool print_matrix = false; bool print_file_progress = false; bool show_pad_files = false; bool show_dht_status = false; bool sequential_download = false; bool print_ip = true; bool print_local_ip = false; bool print_timers = false; bool print_block = false; bool print_fails = false; bool print_send_bufs = true; bool print_disk_stats = false; // the number of times we've asked to save resume data // without having received a response (successful or failure) int num_outstanding_resume_data = 0; #ifndef TORRENT_DISABLE_DHT std::vector dht_active_requests; std::vector dht_routing_table; #endif std::string to_hex(lt::sha1_hash const& s) { std::stringstream ret; ret << s; return ret.str(); } bool load_file(std::string const& filename, std::vector& v , int limit = 8000000) { std::fstream f(filename, std::ios_base::in | std::ios_base::binary); f.seekg(0, std::ios_base::end); auto const s = f.tellg(); if (s > limit || s < 0) return false; f.seekg(0, std::ios_base::beg); v.resize(static_cast(s)); if (s == std::fstream::pos_type(0)) return !f.fail(); f.read(v.data(), v.size()); return !f.fail(); } bool is_absolute_path(std::string const& f) { if (f.empty()) return false; #if defined(TORRENT_WINDOWS) || defined(TORRENT_OS2) int i = 0; // match the xx:\ or xx:/ form while (f[i] && strchr("abcdefghijklmnopqrstuvxyz", f[i])) ++i; if (i < int(f.size()-1) && f[i] == ':' && (f[i+1] == '\\' || f[i+1] == '/')) return true; // match the \\ form if (int(f.size()) >= 2 && f[0] == '\\' && f[1] == '\\') return true; return false; #else if (f[0] == '/') return true; return false; #endif } std::string path_append(std::string const& lhs, std::string const& rhs) { if (lhs.empty() || lhs == ".") return rhs; if (rhs.empty() || rhs == ".") return lhs; #if defined(TORRENT_WINDOWS) || defined(TORRENT_OS2) #define TORRENT_SEPARATOR "\\" bool need_sep = lhs[lhs.size()-1] != '\\' && lhs[lhs.size()-1] != '/'; #else #define TORRENT_SEPARATOR "/" bool need_sep = lhs[lhs.size()-1] != '/'; #endif return lhs + (need_sep?TORRENT_SEPARATOR:"") + rhs; } std::string make_absolute_path(std::string const& p) { if (is_absolute_path(p)) return p; std::string ret; #if defined TORRENT_WINDOWS char* cwd = ::_getcwd(nullptr, 0); ret = path_append(cwd, p); std::free(cwd); #else char* cwd = ::getcwd(nullptr, 0); ret = path_append(cwd, p); std::free(cwd); #endif return ret; } std::string print_endpoint(lt::tcp::endpoint const& ep) { using namespace lt; lt::error_code ec; char buf[200]; address const& addr = ep.address(); if (addr.is_v6()) std::snprintf(buf, sizeof(buf), "[%s]:%d", addr.to_string(ec).c_str(), ep.port()); else std::snprintf(buf, sizeof(buf), "%s:%d", addr.to_string(ec).c_str(), ep.port()); return buf; } using lt::torrent_status; FILE* g_log_file = nullptr; int peer_index(lt::tcp::endpoint addr, std::vector const& peers) { using namespace lt; auto i = std::find_if(peers.begin(), peers.end() , [&addr](peer_info const& pi) { return pi.ip == addr; }); if (i == peers.end()) return -1; return int(i - peers.begin()); } // returns the number of lines printed int print_peer_info(std::string& out , std::vector const& peers, int max_lines) { using namespace lt; int pos = 0; if (print_ip) out += "IP "; if (print_local_ip) out += "local IP "; out += "progress down (total | peak ) up (total | peak ) sent-req tmo bsy rcv flags dn up source "; if (print_fails) out += "fail hshf "; if (print_send_bufs) out += "rq sndb (recvb |alloc | wmrk ) q-bytes "; if (print_timers) out += "inactive wait timeout q-time "; out += " v disk ^ rtt "; if (print_block) out += "block-progress "; out += "client \x1b[K\n"; ++pos; char str[500]; for (std::vector::const_iterator i = peers.begin(); i != peers.end(); ++i) { if ((i->flags & (peer_info::handshake | peer_info::connecting) && !print_connecting_peers)) { continue; } if (print_ip) { std::snprintf(str, sizeof(str), "%-30s ", (::print_endpoint(i->ip) + (i->flags & peer_info::utp_socket ? " [uTP]" : "") + (i->flags & peer_info::i2p_socket ? " [i2p]" : "") ).c_str()); out += str; } if (print_local_ip) { std::snprintf(str, sizeof(str), "%-30s ", ::print_endpoint(i->local_endpoint).c_str()); out += str; } char temp[10]; std::snprintf(temp, sizeof(temp), "%d/%d" , i->download_queue_length , i->target_dl_queue_length); temp[7] = 0; char peer_progress[10]; std::snprintf(peer_progress, sizeof(peer_progress), "%.1f%%", i->progress_ppm / 10000.f); std::snprintf(str, sizeof(str) , "%s %s%s (%s|%s) %s%s (%s|%s) %s%7s %4d%4d%4d %s%s%s%s%s%s%s%s%s%s%s%s%s %s%s%s %s%s%s %s%s%s%s%s%s " , progress_bar(i->progress_ppm / 1000, 15, col_green, '#', '-', peer_progress).c_str() , esc("32"), add_suffix(i->down_speed, "/s").c_str() , add_suffix(i->total_download).c_str(), add_suffix(i->download_rate_peak, "/s").c_str() , esc("31"), add_suffix(i->up_speed, "/s").c_str(), add_suffix(i->total_upload).c_str() , add_suffix(i->upload_rate_peak, "/s").c_str(), esc("0") , temp // sent requests and target number of outstanding reqs. , i->timed_out_requests , i->busy_requests , i->upload_queue_length , color("I", (i->flags & peer_info::interesting)?col_white:col_blue).c_str() , color("C", (i->flags & peer_info::choked)?col_white:col_blue).c_str() , color("i", (i->flags & peer_info::remote_interested)?col_white:col_blue).c_str() , color("c", (i->flags & peer_info::remote_choked)?col_white:col_blue).c_str() , color("x", (i->flags & peer_info::supports_extensions)?col_white:col_blue).c_str() , color("o", (i->flags & peer_info::local_connection)?col_white:col_blue).c_str() , color("p", (i->flags & peer_info::on_parole)?col_white:col_blue).c_str() , color("O", (i->flags & peer_info::optimistic_unchoke)?col_white:col_blue).c_str() , color("S", (i->flags & peer_info::snubbed)?col_white:col_blue).c_str() , color("U", (i->flags & peer_info::upload_only)?col_white:col_blue).c_str() , color("e", (i->flags & peer_info::endgame_mode)?col_white:col_blue).c_str() , color("E", (i->flags & peer_info::rc4_encrypted)?col_white:(i->flags & peer_info::plaintext_encrypted)?col_cyan:col_blue).c_str() , color("h", (i->flags & peer_info::holepunched)?col_white:col_blue).c_str() , color("d", (i->read_state & peer_info::bw_disk)?col_white:col_blue).c_str() , color("l", (i->read_state & peer_info::bw_limit)?col_white:col_blue).c_str() , color("n", (i->read_state & peer_info::bw_network)?col_white:col_blue).c_str() , color("d", (i->write_state & peer_info::bw_disk)?col_white:col_blue).c_str() , color("l", (i->write_state & peer_info::bw_limit)?col_white:col_blue).c_str() , color("n", (i->write_state & peer_info::bw_network)?col_white:col_blue).c_str() , color("t", (i->source & peer_info::tracker)?col_white:col_blue).c_str() , color("p", (i->source & peer_info::pex)?col_white:col_blue).c_str() , color("d", (i->source & peer_info::dht)?col_white:col_blue).c_str() , color("l", (i->source & peer_info::lsd)?col_white:col_blue).c_str() , color("r", (i->source & peer_info::resume_data)?col_white:col_blue).c_str() , color("i", (i->source & peer_info::incoming)?col_white:col_blue).c_str()); out += str; if (print_fails) { std::snprintf(str, sizeof(str), "%4d %4d " , i->failcount, i->num_hashfails); out += str; } if (print_send_bufs) { std::snprintf(str, sizeof(str), "%2d %6d %6d|%6d|%6d%5dkB " , i->requests_in_buffer, i->used_send_buffer , i->used_receive_buffer , i->receive_buffer_size , i->receive_buffer_watermark , i->queue_bytes / 1000); out += str; } if (print_timers) { char req_timeout[20] = "-"; // timeout is only meaningful if there is at least one outstanding // request to the peer if (i->download_queue_length > 0) std::snprintf(req_timeout, sizeof(req_timeout), "%d", i->request_timeout); std::snprintf(str, sizeof(str), "%8d %4d %7s %6d " , int(total_seconds(i->last_active)) , int(total_seconds(i->last_request)) , req_timeout , int(total_seconds(i->download_queue_time))); out += str; } std::snprintf(str, sizeof(str), "%s|%s %5d " , add_suffix(i->pending_disk_bytes).c_str() , add_suffix(i->pending_disk_read_bytes).c_str() , i->rtt); out += str; if (print_block) { if (i->downloading_piece_index >= piece_index_t(0)) { char buf[50]; std::snprintf(buf, sizeof(buf), "%d:%d" , static_cast(i->downloading_piece_index), i->downloading_block_index); out += progress_bar( i->downloading_progress * 1000 / i->downloading_total, 14, col_green, '-', '#', buf); } else { out += progress_bar(0, 14); } } out += " "; if (i->flags & lt::peer_info::handshake) { out += esc("31"); out += " waiting for handshake"; out += esc("0"); } else if (i->flags & lt::peer_info::connecting) { out += esc("31"); out += " connecting to peer"; out += esc("0"); } else { out += " "; out += i->client; } out += "\x1b[K\n"; ++pos; if (pos >= max_lines) break; } return pos; } lt::storage_mode_t allocation_mode = lt::storage_mode_sparse; std::string save_path("."); int torrent_upload_limit = 0; int torrent_download_limit = 0; std::string monitor_dir; int poll_interval = 5; int max_connections_per_torrent = 50; bool seed_mode = false; bool stats_enabled = false; int cache_size = -1; bool share_mode = false; bool disable_storage = false; bool quit = false; void signal_handler(int) { // make the main loop terminate quit = true; } // if non-empty, a peer that will be added to all torrents std::string peer; void print_settings(int const start, int const num , char const* const fmt) { for (int i = start; i < start + num; ++i) { char const* name = lt::name_for_setting(i); if (!name || name[0] == '\0') continue; std::printf(fmt, name); } } void assign_setting(lt::settings_pack& settings, std::string const& key, char const* value) { int const sett_name = lt::setting_by_name(key); if (sett_name < 0) { std::fprintf(stderr, "unknown setting: \"%s\"\n", key.c_str()); std::exit(1); } using lt::settings_pack; switch (sett_name & settings_pack::type_mask) { case settings_pack::string_type_base: settings.set_str(sett_name, value); break; case settings_pack::bool_type_base: if (value == "1"_sv || value == "on"_sv || value == "true"_sv) { settings.set_bool(sett_name, true); } else if (value == "0"_sv || value == "off"_sv || value == "false"_sv) { settings.set_bool(sett_name, false); } else { std::fprintf(stderr, "invalid value for \"%s\". expected 0 or 1\n" , key.c_str()); std::exit(1); } break; case settings_pack::int_type_base: using namespace libtorrent::literals; static std::map const enums = { {"no_piece_suggestions"_sv, settings_pack::no_piece_suggestions}, {"suggest_read_cache"_sv, settings_pack::suggest_read_cache}, {"fixed_slots_choker"_sv, settings_pack::fixed_slots_choker}, {"rate_based_choker"_sv, settings_pack::rate_based_choker}, {"round_robin"_sv, settings_pack::round_robin}, {"fastest_upload"_sv, settings_pack::fastest_upload}, {"anti_leech"_sv, settings_pack::anti_leech}, {"enable_os_cache"_sv, settings_pack::enable_os_cache}, {"disable_os_cache"_sv, settings_pack::disable_os_cache}, {"prefer_tcp"_sv, settings_pack::prefer_tcp}, {"peer_proportional"_sv, settings_pack::peer_proportional}, {"pe_forced"_sv, settings_pack::pe_forced}, {"pe_enabled"_sv, settings_pack::pe_enabled}, {"pe_disabled"_sv, settings_pack::pe_disabled}, {"pe_plaintext"_sv, settings_pack::pe_plaintext}, {"pe_rc4"_sv, settings_pack::pe_rc4}, {"pe_both"_sv, settings_pack::pe_both}, {"none"_sv, settings_pack::none}, {"socks4"_sv, settings_pack::socks4}, {"socks5"_sv, settings_pack::socks5}, {"socks5_pw"_sv, settings_pack::socks5_pw}, {"http"_sv, settings_pack::http}, {"http_pw"_sv, settings_pack::http_pw}, {"i2p_proxy"_sv, settings_pack::i2p_proxy}, }; auto const it = enums.find(lt::string_view(value)); if (it != enums.end()) { settings.set_int(sett_name, it->second); break; } static std::map const alert_categories = { {"error"_sv, lt::alert_category::error}, {"peer"_sv, lt::alert_category::peer}, {"port_mapping"_sv, lt::alert_category::port_mapping}, {"storage"_sv, lt::alert_category::storage}, {"tracker"_sv, lt::alert_category::tracker}, {"connect"_sv, lt::alert_category::connect}, {"status"_sv, lt::alert_category::status}, {"ip_block"_sv, lt::alert_category::ip_block}, {"performance_warning"_sv, lt::alert_category::performance_warning}, {"dht"_sv, lt::alert_category::dht}, {"session_log"_sv, lt::alert_category::session_log}, {"torrent_log"_sv, lt::alert_category::torrent_log}, {"peer_log"_sv, lt::alert_category::peer_log}, {"incoming_request"_sv, lt::alert_category::incoming_request}, {"dht_log"_sv, lt::alert_category::dht_log}, {"dht_operation"_sv, lt::alert_category::dht_operation}, {"port_mapping_log"_sv, lt::alert_category::port_mapping_log}, {"picker_log"_sv, lt::alert_category::picker_log}, {"file_progress"_sv, lt::alert_category::file_progress}, {"piece_progress"_sv, lt::alert_category::piece_progress}, {"upload"_sv, lt::alert_category::upload}, {"block_progress"_sv, lt::alert_category::block_progress}, {"all"_sv, lt::alert_category::all}, }; std::stringstream flags(value); std::string f; lt::alert_category_t val; while (std::getline(flags, f, ',')) try { auto const it = alert_categories.find(f); if (it == alert_categories.end()) val |= lt::alert_category_t{unsigned(std::stoi(f))}; else val |= it->second; } catch (std::invalid_argument const&) { std::fprintf(stderr, "invalid value for \"%s\". expected integer or enum value\n" , key.c_str()); std::exit(1); } settings.set_int(sett_name, val); break; } } std::string resume_file(lt::sha1_hash const& info_hash) { return path_append(save_path, path_append(".resume" , to_hex(info_hash) + ".resume")); } void set_torrent_params(lt::add_torrent_params& p) { p.max_connections = max_connections_per_torrent; p.max_uploads = -1; p.upload_limit = torrent_upload_limit; p.download_limit = torrent_download_limit; if (seed_mode) p.flags |= lt::torrent_flags::seed_mode; if (disable_storage) p.storage = lt::disabled_storage_constructor; if (share_mode) p.flags |= lt::torrent_flags::share_mode; p.save_path = save_path; p.storage_mode = allocation_mode; } void add_magnet(lt::session& ses, lt::string_view uri) { lt::error_code ec; lt::add_torrent_params p = lt::parse_magnet_uri(uri.to_string(), ec); if (ec) { std::printf("invalid magnet link \"%s\": %s\n" , uri.to_string().c_str(), ec.message().c_str()); return; } std::vector resume_data; if (load_file(resume_file(p.info_hash), resume_data)) { p = lt::read_resume_data(resume_data, ec); if (ec) std::printf(" failed to load resume data: %s\n", ec.message().c_str()); } set_torrent_params(p); std::printf("adding magnet: %s\n", uri.to_string().c_str()); ses.async_add_torrent(std::move(p)); } // return false on failure bool add_torrent(lt::session& ses, std::string torrent) { using lt::add_torrent_params; using lt::storage_mode_t; static int counter = 0; std::printf("[%d] %s\n", counter++, torrent.c_str()); lt::error_code ec; auto ti = std::make_shared(torrent, ec); if (ec) { std::printf("failed to load torrent \"%s\": %s\n" , torrent.c_str(), ec.message().c_str()); return false; } add_torrent_params p; std::vector resume_data; if (load_file(resume_file(ti->info_hash()), resume_data)) { p = lt::read_resume_data(resume_data, ec); if (ec) std::printf(" failed to load resume data: %s\n", ec.message().c_str()); } set_torrent_params(p); p.ti = ti; p.flags &= ~lt::torrent_flags::duplicate_is_error; ses.async_add_torrent(std::move(p)); return true; } std::vector list_dir(std::string path , bool (*filter_fun)(lt::string_view) , lt::error_code& ec) { std::vector ret; #ifdef TORRENT_WINDOWS if (!path.empty() && path[path.size()-1] != '\\') path += "\\*"; else path += "*"; WIN32_FIND_DATAA fd; HANDLE handle = FindFirstFileA(path.c_str(), &fd); if (handle == INVALID_HANDLE_VALUE) { ec.assign(GetLastError(), boost::system::system_category()); return ret; } do { lt::string_view p = fd.cFileName; if (filter_fun(p)) ret.push_back(p.to_string()); } while (FindNextFileA(handle, &fd)); FindClose(handle); #else if (!path.empty() && path[path.size()-1] == '/') path.resize(path.size()-1); DIR* handle = opendir(path.c_str()); if (handle == nullptr) { ec.assign(errno, boost::system::system_category()); return ret; } struct dirent* de; while ((de = readdir(handle))) { lt::string_view p(de->d_name); if (filter_fun(p)) ret.push_back(p.to_string()); } closedir(handle); #endif return ret; } void scan_dir(std::string const& dir_path, lt::session& ses) { using namespace lt; error_code ec; std::vector ents = list_dir(dir_path , [](lt::string_view p) { return p.size() > 8 && p.substr(p.size() - 8) == ".torrent"; }, ec); if (ec) { std::fprintf(stderr, "failed to list directory: (%s : %d) %s\n" , ec.category().name(), ec.value(), ec.message().c_str()); return; } for (auto const& e : ents) { std::string const file = path_append(dir_path, e); // there's a new file in the monitor directory, load it up if (add_torrent(ses, file)) { if (::remove(file.c_str()) < 0) { std::fprintf(stderr, "failed to remove torrent file: \"%s\"\n" , file.c_str()); } } } } char const* timestamp() { time_t t = std::time(nullptr); tm* timeinfo = std::localtime(&t); static char str[200]; std::strftime(str, 200, "%b %d %X", timeinfo); return str; } void print_alert(lt::alert const* a, std::string& str) { using namespace lt; if (a->category() & alert_category::error) { str += esc("31"); } else if (a->category() & (alert_category::peer | alert_category::storage)) { str += esc("33"); } str += "["; str += timestamp(); str += "] "; str += a->message(); str += esc("0"); if (g_log_file) std::fprintf(g_log_file, "[%s] %s\n", timestamp(), a->message().c_str()); } int save_file(std::string const& filename, std::vector const& v) { std::fstream f(filename, std::ios_base::trunc | std::ios_base::out | std::ios_base::binary); f.write(v.data(), v.size()); return !f.fail(); } // returns true if the alert was handled (and should not be printed to the log) // returns false if the alert was not handled bool handle_alert(torrent_view& view, session_view& ses_view , lt::session&, lt::alert* a) { using namespace lt; if (session_stats_alert* s = alert_cast(a)) { ses_view.update_counters(s->counters() , duration_cast(s->timestamp().time_since_epoch()).count()); return !stats_enabled; } #ifndef TORRENT_DISABLE_DHT if (dht_stats_alert* p = alert_cast(a)) { dht_active_requests = p->active_requests; dht_routing_table = p->routing_table; return true; } #endif #ifdef TORRENT_USE_OPENSSL if (torrent_need_cert_alert* p = alert_cast(a)) { torrent_handle h = p->handle; std::string base_name = path_append("certificates", to_hex(h.info_hash())); std::string cert = base_name + ".pem"; std::string priv = base_name + "_key.pem"; #ifdef TORRENT_WINDOWS struct ::_stat st; int ret = ::_stat(cert.c_str(), &st); if (ret < 0 || (st.st_mode & _S_IFREG) == 0) #else struct ::stat st; int ret = ::stat(cert.c_str(), &st); if (ret < 0 || (st.st_mode & S_IFREG) == 0) #endif { char msg[256]; std::snprintf(msg, sizeof(msg), "ERROR. could not load certificate %s: %s\n" , cert.c_str(), std::strerror(errno)); if (g_log_file) std::fprintf(g_log_file, "[%s] %s\n", timestamp(), msg); return true; } #ifdef TORRENT_WINDOWS ret = ::_stat(priv.c_str(), &st); if (ret < 0 || (st.st_mode & _S_IFREG) == 0) #else ret = ::stat(priv.c_str(), &st); if (ret < 0 || (st.st_mode & S_IFREG) == 0) #endif { char msg[256]; std::snprintf(msg, sizeof(msg), "ERROR. could not load private key %s: %s\n" , priv.c_str(), std::strerror(errno)); if (g_log_file) std::fprintf(g_log_file, "[%s] %s\n", timestamp(), msg); return true; } char msg[256]; std::snprintf(msg, sizeof(msg), "loaded certificate %s and key %s\n", cert.c_str(), priv.c_str()); if (g_log_file) std::fprintf(g_log_file, "[%s] %s\n", timestamp(), msg); h.set_ssl_certificate(cert, priv, "certificates/dhparams.pem", "1234"); h.resume(); } #endif // don't log every peer we try to connect to if (alert_cast(a)) return true; if (peer_disconnected_alert* pd = alert_cast(a)) { // ignore failures to connect and peers not responding with a // handshake. The peers that we successfully connect to and then // disconnect is more interesting. if (pd->op == operation_t::connect || pd->error == errors::timed_out_no_handshake) return true; } #ifdef _MSC_VER // it seems msvc makes the definitions of 'p' escape the if-statement here #pragma warning(push) #pragma warning(disable: 4456) #endif if (metadata_received_alert* p = alert_cast(a)) { torrent_handle h = p->handle; h.save_resume_data(torrent_handle::save_info_dict); ++num_outstanding_resume_data; } else if (add_torrent_alert* p = alert_cast(a)) { if (p->error) { std::fprintf(stderr, "failed to add torrent: %s %s\n" , p->params.ti ? p->params.ti->name().c_str() : p->params.name.c_str() , p->error.message().c_str()); } else { torrent_handle h = p->handle; h.save_resume_data(torrent_handle::save_info_dict | torrent_handle::only_if_modified); ++num_outstanding_resume_data; // if we have a peer specified, connect to it if (!peer.empty()) { char* port = (char*) strrchr((char*)peer.c_str(), ':'); if (port != nullptr) { *port++ = 0; char const* ip = peer.c_str(); int peer_port = atoi(port); error_code ec; if (peer_port > 0) h.connect_peer(tcp::endpoint(address::from_string(ip, ec), std::uint16_t(peer_port))); } } } } else if (torrent_finished_alert* p = alert_cast(a)) { p->handle.set_max_connections(max_connections_per_torrent / 2); // write resume data for the finished torrent // the alert handler for save_resume_data_alert // will save it to disk torrent_handle h = p->handle; h.save_resume_data(torrent_handle::save_info_dict); ++num_outstanding_resume_data; } else if (save_resume_data_alert* p = alert_cast(a)) { --num_outstanding_resume_data; auto const buf = write_resume_data_buf(p->params); save_file(resume_file(p->params.info_hash), buf); } else if (save_resume_data_failed_alert* p = alert_cast(a)) { --num_outstanding_resume_data; // don't print the error if it was just that we didn't need to save resume // data. Returning true means "handled" and not printed to the log return p->error == lt::errors::resume_data_not_modified; } else if (torrent_paused_alert* p = alert_cast(a)) { // write resume data for the finished torrent // the alert handler for save_resume_data_alert // will save it to disk torrent_handle h = p->handle; h.save_resume_data(torrent_handle::save_info_dict); ++num_outstanding_resume_data; } else if (state_update_alert* p = alert_cast(a)) { view.update_torrents(std::move(p->status)); return true; } else if (torrent_removed_alert* p = alert_cast(a)) { view.remove_torrent(std::move(p->handle)); } return false; #ifdef _MSC_VER #pragma warning(pop) #endif } void pop_alerts(torrent_view& view, session_view& ses_view , lt::session& ses, std::deque& events) { std::vector alerts; ses.pop_alerts(&alerts); for (auto a : alerts) { if (::handle_alert(view, ses_view, ses, a)) continue; // if we didn't handle the alert, print it to the log std::string event_string; print_alert(a, event_string); events.push_back(event_string); if (events.size() >= 20) events.pop_front(); } } void print_piece(lt::partial_piece_info const* pp , lt::cached_piece_info const* cs , std::vector const& peers , std::string& out) { using namespace lt; char str[1024]; assert(pp == nullptr || cs == nullptr || cs->piece == pp->piece_index); int piece = static_cast(pp ? pp->piece_index : cs->piece); int num_blocks = pp ? pp->blocks_in_piece : int(cs->blocks.size()); std::snprintf(str, sizeof(str), "%5d:[", piece); out += str; string_view last_color; for (int j = 0; j < num_blocks; ++j) { int const index = pp ? peer_index(pp->blocks[j].peer(), peers) % 36 : -1; char const* chr = " "; bool const snubbed = index >= 0 ? bool(peers[index].flags & lt::peer_info::snubbed) : false; char const* color = ""; if (pp == nullptr) { color = cs->blocks[j] ? esc("34;7") : esc("0"); chr = " "; } else { if (cs && cs->blocks[j] && pp->blocks[j].state != block_info::finished) color = esc("36;7"); else if (pp->blocks[j].bytes_progress > 0 && pp->blocks[j].state == block_info::requested) { if (pp->blocks[j].num_peers > 1) color = esc("0;1"); else color = snubbed ? esc("0;35") : esc("0;33"); #ifndef TORRENT_WINDOWS static char const* const progress[] = { "\u2581", "\u2582", "\u2583", "\u2584", "\u2585", "\u2586", "\u2587", "\u2588" }; chr = progress[pp->blocks[j].bytes_progress * 8 / pp->blocks[j].block_size]; #else static char const* const progress[] = { "\xb0", "\xb1", "\xb2" }; chr = progress[pp->blocks[j].bytes_progress * 3 / pp->blocks[j].block_size]; #endif } else if (pp->blocks[j].state == block_info::finished) color = esc("32;7"); else if (pp->blocks[j].state == block_info::writing) color = esc("36;7"); else if (pp->blocks[j].state == block_info::requested) { color = snubbed ? esc("0;35") : esc("0"); chr = "="; } else { color = esc("0"); chr = " "; } } if (last_color != color) { out += color; last_color = color; } out += chr; } out += esc("0"); out += "]"; } bool is_resume_file(std::string const& s) { static std::string const hex_digit = "0123456789abcdef"; if (s.size() != 40 + 7) return false; if (s.substr(40) != ".resume") return false; for (char const c : s.substr(0, 40)) { if (hex_digit.find(c) == std::string::npos) return false; } return true; } int main(int argc, char* argv[]) { #ifndef _WIN32 // sets the terminal to single-character mode // and resets when destructed set_keypress s; #endif if (argc == 1) { std::fprintf(stderr, R"(usage: client_test [OPTIONS] [TORRENT|MAGNETURL] OPTIONS: CLIENT OPTIONS -f logs all events to the given file -s sets the save path for downloads. This also determines the resume data save directory. Torrents from the resume directory are automatically added to the session on startup. -m sets the .torrent monitor directory. torrent files dropped in the directory are added the session and the resume data directory, and removed from the monitor dir. -t sets the scan interval of the monitor dir -F sets the UI refresh rate. This is the number of milliseconds between screen refreshes. -k enable high performance settings. This overwrites any other previous command line options, so be sure to specify this first -G Add torrents in seed-mode (i.e. assume all pieces are present and check hashes on-demand) -e exit client after the specified number of iterations through the main loop -O print session stats counters to the log)" #ifdef TORRENT_UTP_LOG_ENABLE R"( -q Enable uTP transport-level verbose logging )" #endif R"( LIBTORRENT SETTINGS --= set the libtorrent setting to --list-settings print all libtorrent settings and exit BITTORRENT OPTIONS -T sets the max number of connections per torrent -U sets per-torrent upload rate -D sets per-torrent download rate -Q enables share mode. Share mode attempts to maximize share ratio rather than downloading -r connect to specified peer NETWORK OPTIONS -x loads an emule IP-filter file -Y Rate limit local peers)" #if TORRENT_USE_I2P R"( -i the hostname to an I2P SAM bridge to use )" #endif R"( DISK OPTIONS -a sets the allocation mode. [sparse|allocate] -0 disable disk I/O, read garbage and don't flush to disk TORRENT is a path to a .torrent file MAGNETURL is a magnet link alert mask flags: error peer port_mapping storage tracker connect status ip_block performance_warning dht session_log torrent_log peer_log incoming_request dht_log dht_operation port_mapping_log picker_log file_progress piece_progress upload block_progress all examples: --alert_mask=error,port_mapping,tracker,connect,session_log --alert_mask=error,session_log,torrent_log,peer_log --alert_mask=error,dht,dht_log,dht_operation --alert_mask=all )") ; return 0; } using lt::settings_pack; using lt::session_handle; torrent_view view; session_view ses_view; lt::session_params params; #ifndef TORRENT_DISABLE_DHT params.dht_settings.privacy_lookups = true; std::vector in; if (load_file(".ses_state", in)) { lt::error_code ec; lt::bdecode_node e = lt::bdecode(in, ec); if (!ec) params = read_session_params(e, session_handle::save_dht_state); } #endif auto& settings = params.settings; settings.set_int(settings_pack::cache_size, cache_size); settings.set_int(settings_pack::choking_algorithm, settings_pack::rate_based_choker); settings.set_str(settings_pack::user_agent, "client_test/" LIBTORRENT_VERSION); settings.set_int(settings_pack::alert_mask , lt::alert_category::error | lt::alert_category::peer | lt::alert_category::port_mapping | lt::alert_category::storage | lt::alert_category::tracker | lt::alert_category::connect | lt::alert_category::status | lt::alert_category::ip_block | lt::alert_category::performance_warning | lt::alert_category::dht | lt::alert_category::incoming_request | lt::alert_category::dht_operation | lt::alert_category::port_mapping_log | lt::alert_category::file_progress); lt::time_duration refresh_delay = lt::milliseconds(500); bool rate_limit_locals = false; std::deque events; int loop_limit = -1; lt::time_point next_dir_scan = lt::clock_type::now(); // load the torrents given on the commandline std::vector torrents; lt::ip_filter loaded_ip_filter; for (int i = 1; i < argc; ++i) { if (argv[i][0] != '-') { torrents.push_back(argv[i]); continue; } if (argv[i] == "--list-settings"_sv) { // print all libtorrent settings and exit print_settings(settings_pack::string_type_base , settings_pack::num_string_settings , "%s=\n"); print_settings(settings_pack::bool_type_base , settings_pack::num_bool_settings , "%s=\n"); print_settings(settings_pack::int_type_base , settings_pack::num_int_settings , "%s=\n"); return 0; } // maybe this is an assignment of a libtorrent setting if (argv[i][1] == '-' && strchr(argv[i], '=') != nullptr) { char const* equal = strchr(argv[i], '='); char const* start = argv[i]+2; // +2 is to skip the -- std::string const key(start, equal - start); char const* value = equal + 1; assign_setting(settings, key, value); continue; } // if there's a flag but no argument following, ignore it if (argc == i) continue; char const* arg = argv[i+1]; if (arg == nullptr) arg = ""; switch (argv[i][1]) { case 'f': g_log_file = std::fopen(arg, "w+"); break; case 'k': settings = lt::high_performance_seed(); --i; break; case 'G': seed_mode = true; --i; break; case 's': save_path = make_absolute_path(arg); break; case 'O': stats_enabled = true; --i; break; #ifdef TORRENT_UTP_LOG_ENABLE case 'q': lt::set_utp_stream_logging(true); break; #endif case 'U': torrent_upload_limit = atoi(arg) * 1000; break; case 'D': torrent_download_limit = atoi(arg) * 1000; break; case 'm': monitor_dir = make_absolute_path(arg); break; case 'Q': share_mode = true; --i; break; case 't': poll_interval = atoi(arg); break; case 'F': refresh_delay = lt::milliseconds(atoi(arg)); break; case 'a': allocation_mode = (arg == std::string("sparse")) ? lt::storage_mode_sparse : lt::storage_mode_allocate; break; case 'x': { std::fstream filter(arg, std::ios_base::in); if (!filter.fail()) { std::regex regex(R"(^\s*([0-9]+)\.([0-9]+)\.([0-9]+)\.([0-9]+)\s*-\s*([0-9]+)\.([0-9]+)\.([0-9]+)\.([0-9]+)\s+([0-9]+)$)"); std::string line; while (std::getline(filter, line)) { std::smatch m; if (std::regex_match(line, m, regex)) { address_v4 start((stoi(m[1]) << 24) | (stoi(m[2]) << 16) | (stoi(m[3]) << 8) | stoi(m[4])); address_v4 last((stoi(m[5]) << 24) | (stoi(m[6]) << 16) | (stoi(m[7]) << 8) | stoi(m[8])); loaded_ip_filter.add_rule(start, last , stoi(m[9]) <= 127 ? lt::ip_filter::blocked : 0); } } } } break; case 'T': max_connections_per_torrent = atoi(arg); break; case 'r': peer = arg; break; case 'Y': { --i; rate_limit_locals = true; break; } case '0': disable_storage = true; --i; break; case 'e': { loop_limit = atoi(arg); break; } } ++i; // skip the argument } // create directory for resume files #ifdef TORRENT_WINDOWS int mkdir_ret = _mkdir(path_append(save_path, ".resume").c_str()); #else int mkdir_ret = mkdir(path_append(save_path, ".resume").c_str(), 0777); #endif if (mkdir_ret < 0 && errno != EEXIST) { std::fprintf(stderr, "failed to create resume file directory: (%d) %s\n" , errno, strerror(errno)); } lt::session ses(std::move(params)); if (rate_limit_locals) { lt::ip_filter pcf; pcf.add_rule(address_v4::from_string("0.0.0.0") , address_v4::from_string("255.255.255.255") , 1 << static_cast(lt::session::global_peer_class_id)); pcf.add_rule(address_v6::from_string("::") , address_v6::from_string("ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff"), 1); ses.set_peer_class_filter(pcf); } ses.set_ip_filter(loaded_ip_filter); for (auto const& i : torrents) { if (i.substr(0, 7) == "magnet:") add_magnet(ses, i); else add_torrent(ses, i.to_string()); } std::thread resume_data_loader([&ses] { // load resume files lt::error_code ec; std::string const resume_dir = path_append(save_path, ".resume"); std::vector ents = list_dir(resume_dir , [](lt::string_view p) { return p.size() > 7 && p.substr(p.size() - 7) == ".resume"; }, ec); if (ec) { std::fprintf(stderr, "failed to list resume directory \"%s\": (%s : %d) %s\n" , resume_dir.c_str(), ec.category().name(), ec.value(), ec.message().c_str()); } else { for (auto const& e : ents) { // only load resume files of the form .resume if (!is_resume_file(e)) continue; std::string const file = path_append(resume_dir, e); std::vector resume_data; if (!load_file(file, resume_data)) { std::printf(" failed to load resume file \"%s\": %s\n" , file.c_str(), ec.message().c_str()); continue; } add_torrent_params p = lt::read_resume_data(resume_data, ec); if (ec) { std::printf(" failed to parse resume data \"%s\": %s\n" , file.c_str(), ec.message().c_str()); continue; } ses.async_add_torrent(std::move(p)); } } }); // main loop std::vector peers; std::vector queue; #ifndef _WIN32 signal(SIGTERM, signal_handler); signal(SIGINT, signal_handler); #endif while (!quit && loop_limit != 0) { if (loop_limit > 0) --loop_limit; ses.post_torrent_updates(); ses.post_session_stats(); ses.post_dht_stats(); int terminal_width = 80; int terminal_height = 50; std::tie(terminal_width, terminal_height) = terminal_size(); // the ratio of torrent-list and details below depend on the number of // torrents we have in the session int const height = std::min(terminal_height / 2 , std::max(5, view.num_visible_torrents() + 2)); view.set_size(terminal_width, height); ses_view.set_pos(height); ses_view.set_width(terminal_width); int c = 0; if (sleep_and_input(&c, refresh_delay)) { #ifdef _WIN32 constexpr int escape_seq = 224; constexpr int left_arrow = 75; constexpr int right_arrow = 77; constexpr int up_arrow = 72; constexpr int down_arrow = 80; #else constexpr int escape_seq = 27; constexpr int left_arrow = 68; constexpr int right_arrow = 67; constexpr int up_arrow = 65; constexpr int down_arrow = 66; #endif torrent_handle h = view.get_active_handle(); if (c == EOF) { quit = true; break; } do { if (c == escape_seq) { // escape code, read another character #ifdef _WIN32 int c2 = _getch(); #else int c2 = getc(stdin); if (c2 == EOF) { quit = true; break; } if (c2 != '[') continue; c2 = getc(stdin); #endif if (c2 == EOF) { quit = true; break; } if (c2 == left_arrow) { int const filter = view.filter(); if (filter > 0) { view.set_filter(filter - 1); h = view.get_active_handle(); } } else if (c2 == right_arrow) { int const filter = view.filter(); if (filter < torrent_view::torrents_max - 1) { view.set_filter(filter + 1); h = view.get_active_handle(); } } else if (c2 == up_arrow) { view.arrow_up(); h = view.get_active_handle(); } else if (c2 == down_arrow) { view.arrow_down(); h = view.get_active_handle(); } } if (c == ' ') { if (ses.is_paused()) ses.resume(); else ses.pause(); } if (c == '[' && h.is_valid()) { h.queue_position_up(); } if (c == ']' && h.is_valid()) { h.queue_position_down(); } // add magnet link if (c == 'm') { char url[4096]; url[0] = '\0'; puts("Enter magnet link:\n"); #ifndef _WIN32 // enable terminal echo temporarily set_keypress s(set_keypress::echo | set_keypress::canonical); #endif if (std::scanf("%4095s", url) == 1) add_magnet(ses, url); else std::printf("failed to read magnet link\n"); } if (c == 'q') { quit = true; break; } if (c == 'W' && h.is_valid()) { std::set seeds = h.url_seeds(); for (std::set::iterator i = seeds.begin() , end(seeds.end()); i != end; ++i) { h.remove_url_seed(*i); } seeds = h.http_seeds(); for (std::set::iterator i = seeds.begin() , end(seeds.end()); i != end; ++i) { h.remove_http_seed(*i); } } if (c == 'D' && h.is_valid()) { torrent_status const& st = view.get_active_torrent(); std::printf("\n\nARE YOU SURE YOU WANT TO DELETE THE FILES FOR '%s'. THIS OPERATION CANNOT BE UNDONE. (y/N)" , st.name.c_str()); #ifndef _WIN32 // enable terminal echo temporarily set_keypress s(set_keypress::echo | set_keypress::canonical); #endif char response = 'n'; int scan_ret = std::scanf("%c", &response); if (scan_ret == 1 && response == 'y') { // also delete the resume file std::string const rpath = resume_file(st.info_hash); if (::remove(rpath.c_str()) < 0) std::printf("failed to delete resume file (\"%s\")\n" , rpath.c_str()); if (st.handle.is_valid()) { ses.remove_torrent(st.handle, lt::session::delete_files); } else { std::printf("failed to delete torrent, invalid handle: %s\n" , st.name.c_str()); } } } if (c == 'j' && h.is_valid()) { h.force_recheck(); } if (c == 'r' && h.is_valid()) { h.force_reannounce(); } if (c == 's' && h.is_valid()) { torrent_status const& ts = view.get_active_torrent(); h.set_flags(~ts.flags, lt::torrent_flags::sequential_download); } if (c == 'R') { // save resume data for all torrents std::vector const torr = ses.get_torrent_status( [](torrent_status const& st) { return st.need_save_resume; }, {}); for (torrent_status const& st : torr) { st.handle.save_resume_data(torrent_handle::save_info_dict); ++num_outstanding_resume_data; } } if (c == 'o' && h.is_valid()) { torrent_status const& ts = view.get_active_torrent(); int num_pieces = ts.num_pieces; if (num_pieces > 300) num_pieces = 300; for (piece_index_t i(0); i < piece_index_t(num_pieces); ++i) { h.set_piece_deadline(i, (static_cast(i)+5) * 1000 , torrent_handle::alert_when_available); } } if (c == 'v' && h.is_valid()) { h.scrape_tracker(); } if (c == 'p' && h.is_valid()) { torrent_status const& ts = view.get_active_torrent(); if ((ts.flags & (lt::torrent_flags::auto_managed | lt::torrent_flags::paused)) == lt::torrent_flags::paused) { h.set_flags(lt::torrent_flags::auto_managed); } else { h.unset_flags(lt::torrent_flags::auto_managed); h.pause(torrent_handle::graceful_pause); } } // toggle force-start if (c == 'k' && h.is_valid()) { torrent_status const& ts = view.get_active_torrent(); h.set_flags( ~(ts.flags & lt::torrent_flags::auto_managed), lt::torrent_flags::auto_managed); if ((ts.flags & lt::torrent_flags::auto_managed) && (ts.flags & lt::torrent_flags::paused)) { h.resume(); } } if (c == 'c' && h.is_valid()) { h.clear_error(); } // toggle displays if (c == 't') print_trackers = !print_trackers; if (c == 'i') print_peers = !print_peers; if (c == 'l') print_log = !print_log; if (c == 'd') print_downloads = !print_downloads; if (c == 'y') print_matrix = !print_matrix; if (c == 'f') print_file_progress = !print_file_progress; if (c == 'P') show_pad_files = !show_pad_files; if (c == 'g') show_dht_status = !show_dht_status; if (c == 'x') print_disk_stats = !print_disk_stats; // toggle columns if (c == '1') print_ip = !print_ip; if (c == '2') print_connecting_peers = !print_connecting_peers; if (c == '3') print_timers = !print_timers; if (c == '4') print_block = !print_block; if (c == '6') print_fails = !print_fails; if (c == '7') print_send_bufs = !print_send_bufs; if (c == '8') print_local_ip = !print_local_ip; if (c == 'C') { cache_size = (cache_size == 0) ? -1 : 0; settings_pack p; p.set_int(settings_pack::cache_size, cache_size); ses.apply_settings(std::move(p)); } if (c == 'h') { clear_screen(); set_cursor_pos(0,0); print( R"(HELP SCREEN (press any key to dismiss) CLIENT OPTIONS [q] quit client [m] add magnet link TORRENT ACTIONS [p] pause/resume selected torrent [C] toggle disk cache [s] toggle sequential download [j] force recheck [space] toggle session pause [c] clear error [v] scrape [D] delete torrent and data [r] force reannounce [R] save resume data for all torrents [o] set piece deadlines (sequential dl) [P] toggle auto-managed [k] toggle force-started [W] remove all web seeds [ move queue position closer to beginning ] move queue position closer to end DISPLAY OPTIONS left/right arrow keys: select torrent filter up/down arrow keys: select torrent [i] toggle show peers [d] toggle show downloading pieces [P] show pad files (in file list) [f] toggle show files [g] show DHT [x] toggle disk cache stats [t] show trackers [l] toggle show log [y] toggle show piece matrix COLUMN OPTIONS [1] toggle IP column [2] toggle show peer connection attempts [3] toggle timers column [4] toggle block progress column [6] toggle failures column [7] toggle send buffers column [8] toggle local IP column )"); int tmp; while (sleep_and_input(&tmp, lt::milliseconds(500)) == false); } } while (sleep_and_input(&c, lt::milliseconds(0))); if (c == 'q') { quit = true; break; } } pop_alerts(view, ses_view, ses, events); std::string out; char str[500]; int pos = view.height() + ses_view.height(); set_cursor_pos(0, pos); int cache_flags = print_downloads ? 0 : lt::session::disk_cache_no_pieces; torrent_handle h = view.get_active_handle(); cache_status cs; ses.get_cache_info(&cs, h, cache_flags); #ifndef TORRENT_DISABLE_DHT if (show_dht_status) { // TODO: 3 expose these counters as performance counters /* std::snprintf(str, sizeof(str), "DHT nodes: %d DHT cached nodes: %d " "total DHT size: %" PRId64 " total observers: %d\n" , sess_stat.dht_nodes, sess_stat.dht_node_cache, sess_stat.dht_global_nodes , sess_stat.dht_total_allocations); out += str; */ int bucket = 0; for (lt::dht_routing_bucket const& n : dht_routing_table) { char const* progress_bar = "################################" "################################" "################################" "################################"; char const* short_progress_bar = "--------"; std::snprintf(str, sizeof(str) , "%3d [%3d, %d] %s%s\x1b[K\n" , bucket, n.num_nodes, n.num_replacements , progress_bar + (128 - n.num_nodes) , short_progress_bar + (8 - std::min(8, n.num_replacements))); out += str; pos += 1; ++bucket; } for (lt::dht_lookup const& l : dht_active_requests) { std::snprintf(str, sizeof(str) , " %10s target: %s " "[limit: %2d] " "in-flight: %-2d " "left: %-3d " "1st-timeout: %-2d " "timeouts: %-2d " "responses: %-2d " "last_sent: %-2d " "\x1b[K\n" , l.type , to_hex(l.target).c_str() , l.branch_factor , l.outstanding_requests , l.nodes_left , l.first_timeout , l.timeouts , l.responses , l.last_sent); out += str; pos += 1; } } #endif lt::time_point const now = lt::clock_type::now(); if (h.is_valid()) { torrent_status const& s = view.get_active_torrent(); print((piece_bar(s.pieces, terminal_width - 2) + "\x1b[K\n").c_str()); pos += 1; if ((print_downloads && s.state != torrent_status::seeding) || print_peers) h.get_peer_info(peers); if (print_peers && !peers.empty()) { using lt::peer_info; // sort connecting towards the bottom of the list, and by peer_id // otherwise, to keep the list as stable as possible std::sort(peers.begin(), peers.end() , [](peer_info const& lhs, peer_info const& rhs) { { bool const l = bool(lhs.flags & peer_info::connecting); bool const r = bool(rhs.flags & peer_info::connecting); if (l != r) return l < r; } { bool const l = bool(lhs.flags & peer_info::handshake); bool const r = bool(rhs.flags & peer_info::handshake); if (l != r) return l < r; } return lhs.pid < rhs.pid; }); pos += print_peer_info(out, peers, terminal_height - pos - 2); } if (print_trackers) { snprintf(str, sizeof(str), "next_announce: %4" PRId64 " | current tracker: %s\x1b[K\n" , std::int64_t(duration_cast(s.next_announce).count()) , s.current_tracker.c_str()); out += str; pos += 1; std::vector tr = h.trackers(); for (lt::announce_entry const& ae : h.trackers()) { std::snprintf(str, sizeof(str), "%2d %-55s %s\x1b[K\n" , ae.tier, ae.url.c_str(), ae.verified?"OK ":"- "); out += str; pos += 1; int idx = 0; for (auto const& ep : ae.endpoints) { ++idx; if (!ep.enabled) continue; if (pos + 1 >= terminal_height) break; std::snprintf(str, sizeof(str), " [%2d] fails: %-3d (%-3d) %s %5d \"%s\" %s\x1b[K\n" , idx , ep.fails, ae.fail_limit , to_string(int(total_seconds(ep.next_announce - now)), 8).c_str() , ep.min_announce > now ? int(total_seconds(ep.min_announce - now)) : 0 , ep.last_error ? ep.last_error.message().c_str() : "" , ep.message.c_str()); out += str; pos += 1; // we only need to show this error once, not for every // endpoint if (ep.last_error == boost::asio::error::host_not_found) break; } if (pos + 1 >= terminal_height) break; } } if (print_matrix) { int height_out = 0; print(piece_matrix(s.pieces, terminal_width, &height_out).c_str()); pos += height_out; } if (print_downloads) { h.get_download_queue(queue); std::sort(queue.begin(), queue.end() , [] (lt::partial_piece_info const& lhs, lt::partial_piece_info const& rhs) { return lhs.piece_index < rhs.piece_index; }); std::sort(cs.pieces.begin(), cs.pieces.end() , [](lt::cached_piece_info const& lhs, lt::cached_piece_info const& rhs) { return lhs.piece < rhs.piece; }); int p = 0; // this is horizontal position for (lt::cached_piece_info const& i : cs.pieces) { if (pos + 3 >= terminal_height) break; lt::partial_piece_info* pp = nullptr; lt::partial_piece_info tmp; tmp.piece_index = i.piece; std::vector::iterator ppi = std::lower_bound(queue.begin(), queue.end(), tmp , [](lt::partial_piece_info const& lhs, lt::partial_piece_info const& rhs) { return lhs.piece_index < rhs.piece_index; }); if (ppi != queue.end() && ppi->piece_index == i.piece) pp = &*ppi; print_piece(pp, &i, peers, out); int num_blocks = pp ? pp->blocks_in_piece : int(i.blocks.size()); p += num_blocks + 8; bool continuous_mode = 8 + num_blocks > terminal_width; if (continuous_mode) { while (p > terminal_width) { p -= terminal_width; ++pos; } } else if (p + num_blocks + 8 > terminal_width) { out += "\x1b[K\n"; pos += 1; p = 0; } if (pp) queue.erase(ppi); } for (lt::partial_piece_info const& i : queue) { if (pos + 3 >= terminal_height) break; print_piece(&i, nullptr, peers, out); int num_blocks = i.blocks_in_piece; p += num_blocks + 8; bool continuous_mode = 8 + num_blocks > terminal_width; if (continuous_mode) { while (p > terminal_width) { p -= terminal_width; ++pos; } } else if (p + num_blocks + 8 > terminal_width) { out += "\x1b[K\n"; pos += 1; p = 0; } } if (p != 0) { out += "\x1b[K\n"; pos += 1; } std::snprintf(str, sizeof(str), "%s %s read cache | %s %s downloading | %s %s cached | %s %s flushed | %s %s snubbed | = requested\x1b[K\n" , esc("34;7"), esc("0") // read cache , esc("33;7"), esc("0") // downloading , esc("36;7"), esc("0") // cached , esc("32;7"), esc("0") // flushed , esc("35;7"), esc("0") // snubbed ); out += str; pos += 1; } if (print_file_progress && s.has_metadata) { std::vector file_progress; h.file_progress(file_progress); std::vector file_status = h.file_status(); std::vector file_prio = h.get_file_priorities(); auto f = file_status.begin(); std::shared_ptr ti = h.torrent_file(); int p = 0; // this is horizontal position for (file_index_t i(0); i < file_index_t(ti->num_files()); ++i) { int const idx = static_cast(i); if (pos + 1 >= terminal_height) break; bool const pad_file = ti->files().pad_file_at(i); if (pad_file && !show_pad_files) continue; int const progress = ti->files().file_size(i) > 0 ? int(file_progress[idx] * 1000 / ti->files().file_size(i)) : 1000; assert(file_progress[idx] <= ti->files().file_size(i)); bool const complete = file_progress[idx] == ti->files().file_size(i); std::string title = ti->files().file_name(i).to_string(); if (!complete) { std::snprintf(str, sizeof(str), " (%.1f%%)", progress / 10.f); title += str; } if (f != file_status.end() && f->file_index == i) { title += " [ "; if ((f->open_mode & lt::file_open_mode::rw_mask) == lt::file_open_mode::read_write) title += "read/write "; else if ((f->open_mode & lt::file_open_mode::rw_mask) == lt::file_open_mode::read_only) title += "read "; else if ((f->open_mode & lt::file_open_mode::rw_mask) == lt::file_open_mode::write_only) title += "write "; if (f->open_mode & lt::file_open_mode::random_access) title += "random_access "; if (f->open_mode & lt::file_open_mode::sparse) title += "sparse "; title += "]"; ++f; } const int file_progress_width = pad_file ? 10 : 65; // do we need to line-break? if (p + file_progress_width + 13 > terminal_width) { out += "\x1b[K\n"; pos += 1; p = 0; } std::snprintf(str, sizeof(str), "%s %7s p: %d ", progress_bar(progress, file_progress_width , pad_file ? col_blue : complete ? col_green : col_yellow , '-', '#', title.c_str()).c_str() , add_suffix(file_progress[idx]).c_str() , static_cast(file_prio[idx])); p += file_progress_width + 13; out += str; } if (p != 0) { out += "\x1b[K\n"; pos += 1; } } } if (print_log) { for (auto const& e : events) { if (pos + 1 >= terminal_height) break; out += e; out += "\x1b[K\n"; pos += 1; } } // clear rest of screen out += "\x1b[J"; print(out.c_str()); std::fflush(stdout); if (!monitor_dir.empty() && next_dir_scan < now) { scan_dir(monitor_dir, ses); next_dir_scan = now + seconds(poll_interval); } } resume_data_loader.join(); ses.pause(); std::printf("saving resume data\n"); // get all the torrent handles that we need to save resume data for std::vector const temp = ses.get_torrent_status( [](torrent_status const& st) { if (!st.handle.is_valid() || !st.has_metadata || !st.need_save_resume) return false; return true; }, {}); int idx = 0; for (auto const& st : temp) { // save_resume_data will generate an alert when it's done st.handle.save_resume_data(torrent_handle::save_info_dict); ++num_outstanding_resume_data; ++idx; if ((idx % 32) == 0) { std::printf("\r%d ", num_outstanding_resume_data); pop_alerts(view, ses_view, ses, events); } } std::printf("\nwaiting for resume data [%d]\n", num_outstanding_resume_data); while (num_outstanding_resume_data > 0) { alert const* a = ses.wait_for_alert(seconds(10)); if (a == nullptr) continue; pop_alerts(view, ses_view, ses, events); } if (g_log_file) std::fclose(g_log_file); // we're just saving the DHT state #ifndef TORRENT_DISABLE_DHT std::printf("\nsaving session state\n"); { lt::entry session_state; ses.save_state(session_state, lt::session::save_dht_state); std::vector out; bencode(std::back_inserter(out), session_state); save_file(".ses_state", out); } #endif std::printf("closing session\n"); return 0; }