626 lines
16 KiB
C++
626 lines
16 KiB
C++
/*
|
|
|
|
Copyright (c) 2008, 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 "setup_swarm.hpp"
|
|
#include "test.hpp"
|
|
#include "utils.hpp"
|
|
#include "libtorrent/alert.hpp"
|
|
#include "libtorrent/alert_types.hpp"
|
|
#include "libtorrent/session.hpp"
|
|
#include "libtorrent/session_stats.hpp"
|
|
#include "libtorrent/aux_/path.hpp"
|
|
#include "libtorrent/torrent_info.hpp"
|
|
#include "settings.hpp"
|
|
|
|
using namespace libtorrent;
|
|
|
|
TORRENT_TEST(seed_mode)
|
|
{
|
|
// with seed mode
|
|
setup_swarm(3, swarm_test::upload
|
|
// add session
|
|
, [](lt::settings_pack&) {}
|
|
// add torrent
|
|
, [](lt::add_torrent_params& params) {
|
|
params.flags |= add_torrent_params::flag_seed_mode;
|
|
}
|
|
// on alert
|
|
, [](lt::alert const*, lt::session&) {}
|
|
// terminate
|
|
, [](int, lt::session&) -> bool
|
|
{ return false; });
|
|
}
|
|
|
|
TORRENT_TEST(seed_mode_disable_hash_checks)
|
|
{
|
|
// all nodes need to disable hash checking, otherwise the downloader would
|
|
// just fail
|
|
settings_pack swarm_settings = settings();
|
|
swarm_settings.set_bool(settings_pack::disable_hash_checks, true);
|
|
|
|
dsl_config network_cfg;
|
|
sim::simulation sim{network_cfg};
|
|
|
|
// with seed mode
|
|
setup_swarm(2, swarm_test::upload, sim, swarm_settings, add_torrent_params()
|
|
// add session
|
|
, [](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::suggest_mode, settings_pack::suggest_read_cache);
|
|
}
|
|
// add torrent
|
|
, [](lt::add_torrent_params& params) {
|
|
params.flags |= add_torrent_params::flag_seed_mode;
|
|
// just to make sure the disable_hash_checks really work, we
|
|
// shouldn't be verifying anything from the storage
|
|
params.storage = disabled_storage_constructor;
|
|
}
|
|
// on alert
|
|
, [](lt::alert const*, lt::session&) {}
|
|
// terminate
|
|
, [](int, lt::session&) -> bool
|
|
{ return false; });
|
|
}
|
|
|
|
TORRENT_TEST(plain)
|
|
{
|
|
setup_swarm(2, swarm_test::download
|
|
// add session
|
|
, [](lt::settings_pack&) {}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [](lt::alert const*, lt::session&) {}
|
|
// terminate
|
|
, [](int const ticks, lt::session& ses) -> bool
|
|
{
|
|
if (ticks > 80)
|
|
{
|
|
TEST_ERROR("timeout");
|
|
return true;
|
|
}
|
|
if (!is_seed(ses)) return false;
|
|
std::printf("completed in %d ticks\n", ticks);
|
|
return true;
|
|
});
|
|
}
|
|
|
|
TORRENT_TEST(session_stats)
|
|
{
|
|
std::vector<stats_metric> stats = session_stats_metrics();
|
|
int const downloading_idx = find_metric_idx("ses.num_downloading_torrents");
|
|
TEST_CHECK(downloading_idx >= 0);
|
|
int const incoming_extended_idx = find_metric_idx("ses.num_incoming_extended");
|
|
TEST_CHECK(incoming_extended_idx >= 0);
|
|
|
|
setup_swarm(2, swarm_test::download
|
|
// add session
|
|
, [](lt::settings_pack&) {}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [=](lt::alert const* a, lt::session&)
|
|
{
|
|
auto const* ss = lt::alert_cast<session_stats_alert>(a);
|
|
if (!ss) return;
|
|
|
|
// there's one downloading torrent
|
|
TEST_EQUAL(ss->values[downloading_idx], 1);
|
|
TEST_EQUAL(ss->values[incoming_extended_idx], 1);
|
|
}
|
|
// terminate
|
|
, [](int const ticks, lt::session& ses) -> bool
|
|
{
|
|
ses.post_session_stats();
|
|
if (ticks > 80)
|
|
{
|
|
TEST_ERROR("timeout");
|
|
return true;
|
|
}
|
|
if (!is_seed(ses)) return false;
|
|
std::printf("completed in %d ticks\n", ticks);
|
|
return true;
|
|
});
|
|
}
|
|
|
|
|
|
TORRENT_TEST(suggest)
|
|
{
|
|
int num_suggests = 0;
|
|
setup_swarm(10, swarm_test::upload
|
|
// add session
|
|
, [](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::suggest_mode, settings_pack::suggest_read_cache);
|
|
pack.set_int(settings_pack::max_suggest_pieces, 10);
|
|
}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [&num_suggests](lt::alert const* a, lt::session&) {
|
|
if (auto pl = alert_cast<peer_log_alert>(a))
|
|
{
|
|
if (pl->direction == peer_log_alert::outgoing_message
|
|
&& pl->event_type == std::string("SUGGEST"))
|
|
{
|
|
++num_suggests;
|
|
}
|
|
}
|
|
}
|
|
// terminate
|
|
, [](int const ticks, lt::session&) -> bool
|
|
{
|
|
if (ticks > 500)
|
|
{
|
|
return true;
|
|
}
|
|
return false;
|
|
});
|
|
|
|
// for now, just make sure we send any suggests at all. This feature is
|
|
// experimental and it's not entirely clear it's correct or how to verify
|
|
// that it does what it's supposed to do.
|
|
// perhaps a better way would be to look at piece upload distribution over
|
|
// time
|
|
TEST_CHECK(num_suggests > 0);
|
|
}
|
|
|
|
TORRENT_TEST(utp_only)
|
|
{
|
|
setup_swarm(2, swarm_test::download
|
|
// add session
|
|
, [](lt::settings_pack& pack) {
|
|
pack.set_bool(settings_pack::enable_incoming_utp, true);
|
|
pack.set_bool(settings_pack::enable_outgoing_utp, true);
|
|
pack.set_bool(settings_pack::enable_incoming_tcp, false);
|
|
pack.set_bool(settings_pack::enable_outgoing_tcp, false);
|
|
}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [](lt::alert const*, lt::session&) {}
|
|
// terminate
|
|
, [](int const ticks, lt::session& ses) -> bool
|
|
{
|
|
if (ticks > 80)
|
|
{
|
|
TEST_ERROR("timeout");
|
|
return true;
|
|
}
|
|
if (!is_seed(ses)) return false;
|
|
return true;
|
|
});
|
|
}
|
|
|
|
void test_stop_start_download(swarm_test type, bool graceful)
|
|
{
|
|
bool paused_once = false;
|
|
bool resumed = false;
|
|
|
|
setup_swarm(3, type
|
|
// add session
|
|
, [](lt::settings_pack& pack) {
|
|
// this test will pause and resume the torrent immediately, we expect
|
|
// to reconnect immediately too, so disable the min reconnect time
|
|
// limit.
|
|
pack.set_int(settings_pack::min_reconnect_time, 0);
|
|
}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {
|
|
|
|
}
|
|
// on alert
|
|
, [&](lt::alert const* a, lt::session& ses) {
|
|
|
|
if (lt::alert_cast<lt::add_torrent_alert>(a))
|
|
add_extra_peers(ses);
|
|
|
|
if (auto tp = lt::alert_cast<lt::torrent_paused_alert>(a))
|
|
{
|
|
TEST_EQUAL(resumed, false);
|
|
std::printf("\nSTART\n\n");
|
|
tp->handle.resume();
|
|
resumed = true;
|
|
}
|
|
}
|
|
// terminate
|
|
, [&](int const ticks, lt::session& ses) -> bool
|
|
{
|
|
if (paused_once == false)
|
|
{
|
|
auto st = get_status(ses);
|
|
const bool limit_reached = (type == swarm_test::download)
|
|
? st.total_wanted_done > st.total_wanted / 2
|
|
: st.total_payload_upload >= 3 * 16 * 1024;
|
|
|
|
if (limit_reached)
|
|
{
|
|
std::printf("\nSTOP\n\n");
|
|
auto h = ses.get_torrents()[0];
|
|
h.pause(graceful ? torrent_handle::graceful_pause : 0);
|
|
paused_once = true;
|
|
}
|
|
}
|
|
|
|
std::printf("tick: %d\n", ticks);
|
|
|
|
const int timeout = type == swarm_test::download ? 20 : 91;
|
|
if (ticks > timeout)
|
|
{
|
|
TEST_ERROR("timeout");
|
|
return true;
|
|
}
|
|
if (type == swarm_test::upload) return false;
|
|
if (!is_seed(ses)) return false;
|
|
std::printf("completed in %d ticks\n", ticks);
|
|
return true;
|
|
});
|
|
|
|
TEST_EQUAL(paused_once, true);
|
|
TEST_EQUAL(resumed, true);
|
|
}
|
|
|
|
TORRENT_TEST(stop_start_download)
|
|
{
|
|
test_stop_start_download(swarm_test::download, false);
|
|
}
|
|
|
|
TORRENT_TEST(stop_start_download_graceful)
|
|
{
|
|
test_stop_start_download(swarm_test::download, true);
|
|
}
|
|
|
|
TORRENT_TEST(stop_start_download_graceful_no_peers)
|
|
{
|
|
bool paused_once = false;
|
|
bool resumed = false;
|
|
|
|
setup_swarm(1, swarm_test::download
|
|
// add session
|
|
, [](lt::settings_pack&) {}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [&](lt::alert const* a, lt::session&) {
|
|
if (auto tp = lt::alert_cast<lt::torrent_paused_alert>(a))
|
|
{
|
|
TEST_EQUAL(resumed, false);
|
|
std::printf("\nSTART\n\n");
|
|
tp->handle.resume();
|
|
resumed = true;
|
|
}
|
|
}
|
|
// terminate
|
|
, [&](int const ticks, lt::session& ses) -> bool
|
|
{
|
|
if (paused_once == false
|
|
&& ticks == 6)
|
|
{
|
|
std::printf("\nSTOP\n\n");
|
|
auto h = ses.get_torrents()[0];
|
|
h.pause(torrent_handle::graceful_pause);
|
|
paused_once = true;
|
|
}
|
|
|
|
std::printf("tick: %d\n", ticks);
|
|
|
|
// when there's only one node (i.e. no peers) we won't ever download
|
|
// the torrent. It's just a test to make sure we still get the
|
|
// torrent_paused_alert
|
|
return ticks > 60;
|
|
});
|
|
|
|
TEST_EQUAL(paused_once, true);
|
|
TEST_EQUAL(resumed, true);
|
|
}
|
|
|
|
|
|
TORRENT_TEST(stop_start_seed)
|
|
{
|
|
test_stop_start_download(swarm_test::upload, false);
|
|
}
|
|
|
|
TORRENT_TEST(stop_start_seed_graceful)
|
|
{
|
|
test_stop_start_download(swarm_test::upload, true);
|
|
}
|
|
|
|
TORRENT_TEST(shutdown)
|
|
{
|
|
setup_swarm(2, swarm_test::download
|
|
// add session
|
|
, [](lt::settings_pack&) {}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [](lt::alert const*, lt::session&) {}
|
|
// terminate
|
|
, [](int, lt::session& ses) -> bool
|
|
{
|
|
if (completed_pieces(ses) == 0) return false;
|
|
TEST_EQUAL(is_seed(ses), false);
|
|
return true;
|
|
});
|
|
}
|
|
|
|
TORRENT_TEST(delete_files)
|
|
{
|
|
std::string save_path;
|
|
|
|
setup_swarm(2, swarm_test::download
|
|
// add session
|
|
, [](lt::settings_pack&) {}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [](lt::alert const*, lt::session&) {}
|
|
// terminate
|
|
, [&save_path](int, lt::session& ses) -> bool
|
|
{
|
|
if (completed_pieces(ses) == 0) return false;
|
|
|
|
auto h = ses.get_torrents()[0];
|
|
save_path = h.status().save_path;
|
|
ses.remove_torrent(h, session::delete_files);
|
|
return true;
|
|
});
|
|
|
|
// assert the file is no longer there
|
|
file_status st;
|
|
error_code ec;
|
|
stat_file(combine_path(save_path, "temporary"), &st, ec);
|
|
std::printf("expecting \"%s/temporary\" to NOT exist [%s | %s]\n"
|
|
, save_path.c_str()
|
|
, ec.category().name()
|
|
, ec.message().c_str());
|
|
TEST_EQUAL(ec, error_code(boost::system::errc::no_such_file_or_directory, system_category()));
|
|
}
|
|
|
|
TORRENT_TEST(delete_partfile)
|
|
{
|
|
std::string save_path;
|
|
setup_swarm(2, swarm_test::download
|
|
// add session
|
|
, [](lt::settings_pack&) {}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [](lt::alert const* a, lt::session&) {}
|
|
// terminate
|
|
, [&save_path](int, lt::session& ses) -> bool
|
|
{
|
|
if (completed_pieces(ses) == 0) return false;
|
|
|
|
auto h = ses.get_torrents()[0];
|
|
save_path = h.status().save_path;
|
|
ses.remove_torrent(h, session::delete_partfile);
|
|
return true;
|
|
});
|
|
// assert the file *is* still there
|
|
file_status st;
|
|
error_code ec;
|
|
stat_file(combine_path(save_path, "temporary"), &st, ec);
|
|
std::printf("expecting \"%s/temporary\" to exist [%s]\n", save_path.c_str()
|
|
, ec.message().c_str());
|
|
TEST_CHECK(!ec);
|
|
}
|
|
|
|
TORRENT_TEST(torrent_completed_alert)
|
|
{
|
|
int num_file_completed = false;
|
|
|
|
setup_swarm(2, swarm_test::download
|
|
// add session
|
|
, [](lt::settings_pack& pack)
|
|
{
|
|
pack.set_int(lt::settings_pack::alert_mask, alert::progress_notification);
|
|
}
|
|
// add torrent
|
|
, [](lt::add_torrent_params&) {}
|
|
// on alert
|
|
, [&](lt::alert const* a, lt::session&)
|
|
{
|
|
auto tc = alert_cast<lt::file_completed_alert>(a);
|
|
if (tc == nullptr) return;
|
|
++num_file_completed;
|
|
}
|
|
// terminate
|
|
, [](int ticks, lt::session& ses) -> bool
|
|
{
|
|
if (ticks > 80)
|
|
{
|
|
TEST_ERROR("timeout");
|
|
return true;
|
|
}
|
|
if (!is_seed(ses)) return false;
|
|
printf("completed in %d ticks\n", ticks);
|
|
return true;
|
|
});
|
|
|
|
TEST_EQUAL(num_file_completed, 1);
|
|
}
|
|
|
|
// template for testing running swarms with edge case settings
|
|
template <typename SettingsFun>
|
|
void test_settings(SettingsFun fun)
|
|
{
|
|
setup_swarm(2, swarm_test::download
|
|
// add session
|
|
, fun
|
|
// add torrent
|
|
, [](lt::add_torrent_params& params) {}
|
|
// on alert
|
|
, [](lt::alert const* a, lt::session& ses) {}
|
|
// terminate
|
|
, [](int ticks, lt::session& ses) -> bool
|
|
{
|
|
if (ticks > 80)
|
|
{
|
|
TEST_ERROR("timeout");
|
|
return true;
|
|
}
|
|
if (!is_seed(ses)) return false;
|
|
return true;
|
|
});
|
|
}
|
|
|
|
TORRENT_TEST(unlimited_connections)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::connections_limit, std::numeric_limits<int>::max()); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(default_connections_limit)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::connections_limit, 0); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(default_connections_limit_negative)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::connections_limit, -1); }
|
|
);
|
|
}
|
|
|
|
|
|
TORRENT_TEST(redundant_have)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_bool(settings_pack::send_redundant_have, false); }
|
|
);
|
|
}
|
|
|
|
#ifndef TORRENT_NO_DEPRECATE
|
|
TORRENT_TEST(lazy_bitfields)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_bool(settings_pack::lazy_bitfields, true); }
|
|
);
|
|
}
|
|
#endif
|
|
|
|
TORRENT_TEST(prioritize_partial_pieces)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_bool(settings_pack::prioritize_partial_pieces, true); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(active_downloads)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::active_downloads, std::numeric_limits<int>::max()); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(active_seeds)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::active_seeds, std::numeric_limits<int>::max()); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(active_seeds_negative)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::active_seeds, -1); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(active_limit)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::active_limit, std::numeric_limits<int>::max()); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(active_limit_negative)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::active_limit, -1); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(upload_rate_limit)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::upload_rate_limit, std::numeric_limits<int>::max()); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(upload_rate_limit_negative)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::upload_rate_limit, -1); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(download_rate_limit)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::download_rate_limit, std::numeric_limits<int>::max()); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(download_rate_limit_negative)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::download_rate_limit, -1); }
|
|
);
|
|
}
|
|
|
|
|
|
TORRENT_TEST(unchoke_slots_limit)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::unchoke_slots_limit, std::numeric_limits<int>::max()); }
|
|
);
|
|
}
|
|
|
|
TORRENT_TEST(unchoke_slots_limit_negative)
|
|
{
|
|
test_settings([](lt::settings_pack& pack) {
|
|
pack.set_int(settings_pack::unchoke_slots_limit, -1);
|
|
pack.set_int(settings_pack::choking_algorithm, settings_pack::fixed_slots_choker);
|
|
});
|
|
}
|
|
|
|
// TODO: add test that makes sure a torrent in graceful pause mode won't make
|
|
// outgoing connections
|
|
// TODO: add test that makes sure a torrent in graceful pause mode won't accept
|
|
// incoming connections
|
|
// TODO: test the different storage allocation modes
|
|
// TODO: test contiguous buffers
|
|
|
|
|