twisterp2pbittorrentblockchainnetworkipv6microbloggingdecentralizedsocial-networkdhttwister-coretwisterarmyp2p-networktwister-servertwister-ipv6
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
486 lines
13 KiB
486 lines
13 KiB
/* |
|
|
|
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 "test.hpp" |
|
|
|
#include "libtorrent/bandwidth_manager.hpp" |
|
#include "libtorrent/bandwidth_queue_entry.hpp" |
|
#include "libtorrent/bandwidth_limit.hpp" |
|
#include "libtorrent/bandwidth_socket.hpp" |
|
#include "libtorrent/socket.hpp" |
|
#include "libtorrent/stat.hpp" |
|
#include "libtorrent/time.hpp" |
|
|
|
#include <boost/function.hpp> |
|
#include <boost/bind.hpp> |
|
#include <iostream> |
|
#include <math.h> |
|
|
|
struct torrent; |
|
struct peer_connection; |
|
|
|
using namespace libtorrent; |
|
|
|
const float sample_time = 20.f; // seconds |
|
|
|
//#define VERBOSE_LOGGING |
|
|
|
bandwidth_channel global_bwc; |
|
|
|
struct peer_connection: bandwidth_socket |
|
{ |
|
peer_connection(bandwidth_manager& bwm |
|
, bandwidth_channel& torrent_bwc, int prio, bool ignore_limits, std::string name) |
|
: m_bwm(bwm) |
|
, m_torrent_bandwidth_channel(torrent_bwc) |
|
, m_priority(prio) |
|
, m_ignore_limits(ignore_limits) |
|
, m_name(name) |
|
, m_quota(0) |
|
{} |
|
|
|
bool is_disconnecting() const { return false; } |
|
bool ignore_bandwidth_limits() { return m_ignore_limits; } |
|
void assign_bandwidth(int channel, int amount); |
|
|
|
void throttle(int limit) { m_bandwidth_channel.throttle(limit); } |
|
|
|
void start(); |
|
|
|
bandwidth_manager& m_bwm; |
|
|
|
bandwidth_channel m_bandwidth_channel; |
|
bandwidth_channel& m_torrent_bandwidth_channel; |
|
|
|
int m_priority; |
|
bool m_ignore_limits; |
|
std::string m_name; |
|
int m_quota; |
|
}; |
|
|
|
void peer_connection::assign_bandwidth(int channel, int amount) |
|
{ |
|
m_quota += amount; |
|
#ifdef VERBOSE_LOGGING |
|
std::cerr << " [" << m_name |
|
<< "] assign bandwidth, " << amount << std::endl; |
|
#endif |
|
TEST_CHECK(amount > 0); |
|
start(); |
|
} |
|
|
|
void peer_connection::start() |
|
{ |
|
m_bwm.request_bandwidth(self(), 150000, m_priority |
|
, &m_bandwidth_channel |
|
, &m_torrent_bandwidth_channel |
|
, &global_bwc); |
|
} |
|
|
|
|
|
typedef std::vector<boost::intrusive_ptr<peer_connection> > connections_t; |
|
|
|
void do_change_rate(bandwidth_channel& t1, bandwidth_channel& t2, int limit) |
|
{ |
|
static int counter = 10; |
|
--counter; |
|
if (counter == 0) |
|
{ |
|
t1.throttle(limit); |
|
t2.throttle(limit); |
|
return; |
|
} |
|
|
|
t1.throttle(limit + limit / 2 * ((counter & 1)?-1:1)); |
|
t2.throttle(limit + limit / 2 * ((counter & 1)?1:-1)); |
|
} |
|
|
|
void do_change_peer_rate(connections_t& v, int limit) |
|
{ |
|
static int count = 10; |
|
--count; |
|
if (count == 0) |
|
{ |
|
std::for_each(v.begin(), v.end() |
|
, boost::bind(&peer_connection::throttle, _1, limit)); |
|
return; |
|
} |
|
|
|
int c = count; |
|
for (connections_t::iterator i = v.begin(); i != v.end(); ++i, ++c) |
|
i->get()->throttle(limit + limit / 2 * ((c & 1)?-1:1)); |
|
} |
|
|
|
void nop() {} |
|
|
|
void run_test(connections_t& v |
|
, bandwidth_manager& manager |
|
, boost::function<void()> f = &nop) |
|
{ |
|
std::cerr << "-------------" << std::endl; |
|
|
|
std::for_each(v.begin(), v.end() |
|
, boost::bind(&peer_connection::start, _1)); |
|
|
|
for (int i = 0; i < int(sample_time * 10); ++i) |
|
{ |
|
manager.update_quotas(milliseconds(100)); |
|
if ((i % 15) == 0) f(); |
|
} |
|
} |
|
|
|
bool close_to(float val, float comp, float err) |
|
{ |
|
return fabs(val - comp) <= err; |
|
} |
|
|
|
void spawn_connections(connections_t& v, bandwidth_manager& bwm |
|
, bandwidth_channel& bwc, int num, char const* prefix) |
|
{ |
|
for (int i = 0; i < num; ++i) |
|
{ |
|
char name[200]; |
|
snprintf(name, sizeof(name), "%s%d", prefix, i); |
|
v.push_back(new peer_connection(bwm, bwc, 200, false, name)); |
|
} |
|
} |
|
|
|
void test_equal_connections(int num, int limit) |
|
{ |
|
std::cerr << "\ntest equal connections " << num << " " << limit << std::endl; |
|
bandwidth_manager manager(0); |
|
global_bwc.throttle(limit); |
|
|
|
bandwidth_channel t1; |
|
|
|
connections_t v; |
|
spawn_connections(v, manager, t1, num, "p"); |
|
run_test(v, manager); |
|
|
|
float sum = 0.f; |
|
float err = (std::max)(limit / num * 0.3f, 1000.f); |
|
for (connections_t::iterator i = v.begin() |
|
, end(v.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
|
|
std::cerr << (*i)->m_quota / sample_time |
|
<< " target: " << (limit / num) << " eps: " << err << std::endl; |
|
TEST_CHECK(close_to((*i)->m_quota / sample_time, limit / num, err)); |
|
} |
|
sum /= sample_time; |
|
std::cerr << "sum: " << sum << " target: " << limit << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit, 50)); |
|
} |
|
|
|
void test_connections_variable_rate(int num, int limit, int torrent_limit) |
|
{ |
|
std::cerr << "\ntest connections variable rate" << num |
|
<< " l: " << limit |
|
<< " t: " << torrent_limit |
|
<< std::endl; |
|
bandwidth_manager manager(0); |
|
global_bwc.throttle(0); |
|
|
|
bandwidth_channel t1; |
|
if (torrent_limit) |
|
t1.throttle(torrent_limit); |
|
|
|
connections_t v; |
|
spawn_connections(v, manager, t1, num, "p"); |
|
std::for_each(v.begin(), v.end() |
|
, boost::bind(&peer_connection::throttle, _1, limit)); |
|
|
|
run_test(v, manager, boost::bind(&do_change_peer_rate |
|
, boost::ref(v), limit)); |
|
|
|
if (torrent_limit > 0 && limit * num > torrent_limit) |
|
limit = torrent_limit / num; |
|
|
|
float sum = 0.f; |
|
float err = limit * 0.3f; |
|
for (connections_t::iterator i = v.begin() |
|
, end(v.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
|
|
std::cerr << (*i)->m_quota / sample_time |
|
<< " target: " << limit << " eps: " << err << std::endl; |
|
TEST_CHECK(close_to((*i)->m_quota / sample_time, limit, err)); |
|
} |
|
sum /= sample_time; |
|
std::cerr << "sum: " << sum << " target: " << (limit * num) << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit * num, limit * 0.3f * num)); |
|
} |
|
|
|
void test_single_peer(int limit, bool torrent_limit) |
|
{ |
|
std::cerr << "\ntest single peer " << limit << " " << torrent_limit << std::endl; |
|
bandwidth_manager manager(0); |
|
bandwidth_channel t1; |
|
global_bwc.throttle(0); |
|
|
|
if (torrent_limit) |
|
t1.throttle(limit); |
|
else |
|
global_bwc.throttle(limit); |
|
|
|
connections_t v; |
|
spawn_connections(v, manager, t1, 1, "p"); |
|
run_test(v, manager); |
|
|
|
float sum = 0.f; |
|
for (connections_t::iterator i = v.begin() |
|
, end(v.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
} |
|
sum /= sample_time; |
|
std::cerr << sum << " target: " << limit << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit, 1000)); |
|
} |
|
|
|
void test_torrents(int num, int limit1, int limit2, int global_limit) |
|
{ |
|
std::cerr << "\ntest equal torrents " << num |
|
<< " l1: " << limit1 |
|
<< " l2: " << limit2 |
|
<< " g: " << global_limit << std::endl; |
|
bandwidth_manager manager(0); |
|
global_bwc.throttle(global_limit); |
|
|
|
bandwidth_channel t1; |
|
bandwidth_channel t2; |
|
|
|
t1.throttle(limit1); |
|
t2.throttle(limit2); |
|
|
|
connections_t v1; |
|
spawn_connections(v1, manager, t1, num, "t1p"); |
|
connections_t v2; |
|
spawn_connections(v2, manager, t2, num, "t2p"); |
|
connections_t v; |
|
std::copy(v1.begin(), v1.end(), std::back_inserter(v)); |
|
std::copy(v2.begin(), v2.end(), std::back_inserter(v)); |
|
run_test(v, manager); |
|
|
|
if (global_limit > 0 && global_limit < limit1 + limit2) |
|
{ |
|
limit1 = (std::min)(limit1, global_limit / 2); |
|
limit2 = global_limit - limit1; |
|
} |
|
float sum = 0.f; |
|
for (connections_t::iterator i = v1.begin() |
|
, end(v1.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
} |
|
sum /= sample_time; |
|
std::cerr << sum << " target: " << limit1 << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit1, 1000)); |
|
|
|
sum = 0.f; |
|
for (connections_t::iterator i = v2.begin() |
|
, end(v2.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
} |
|
sum /= sample_time; |
|
std::cerr << sum << " target: " << limit2 << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit2, 1000)); |
|
} |
|
|
|
void test_torrents_variable_rate(int num, int limit, int global_limit) |
|
{ |
|
std::cerr << "\ntest torrents variable rate" << num |
|
<< " l: " << limit |
|
<< " g: " << global_limit << std::endl; |
|
bandwidth_manager manager(0); |
|
global_bwc.throttle(global_limit); |
|
|
|
bandwidth_channel t1; |
|
bandwidth_channel t2; |
|
|
|
t1.throttle(limit); |
|
t2.throttle(limit); |
|
|
|
connections_t v1; |
|
spawn_connections(v1, manager, t1, num, "t1p"); |
|
connections_t v2; |
|
spawn_connections(v2, manager, t2, num, "t2p"); |
|
connections_t v; |
|
std::copy(v1.begin(), v1.end(), std::back_inserter(v)); |
|
std::copy(v2.begin(), v2.end(), std::back_inserter(v)); |
|
|
|
run_test(v, manager, boost::bind(&do_change_rate, boost::ref(t1), boost::ref(t2), limit)); |
|
|
|
if (global_limit > 0 && global_limit < 2 * limit) |
|
limit = global_limit / 2; |
|
|
|
float sum = 0.f; |
|
for (connections_t::iterator i = v1.begin() |
|
, end(v1.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
} |
|
sum /= sample_time; |
|
std::cerr << sum << " target: " << limit << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit, 1000)); |
|
|
|
sum = 0.f; |
|
for (connections_t::iterator i = v2.begin() |
|
, end(v2.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
} |
|
sum /= sample_time; |
|
std::cerr << sum << " target: " << limit << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit, 1000)); |
|
} |
|
|
|
void test_peer_priority(int limit, bool torrent_limit) |
|
{ |
|
std::cerr << "\ntest peer priority " << limit << " " << torrent_limit << std::endl; |
|
bandwidth_manager manager(0); |
|
bandwidth_channel t1; |
|
global_bwc.throttle(0); |
|
|
|
if (torrent_limit) |
|
t1.throttle(limit); |
|
else |
|
global_bwc.throttle(limit); |
|
|
|
connections_t v1; |
|
spawn_connections(v1, manager, t1, 10, "p"); |
|
connections_t v; |
|
std::copy(v1.begin(), v1.end(), std::back_inserter(v)); |
|
boost::intrusive_ptr<peer_connection> p( |
|
new peer_connection(manager, t1, 1, false, "no-priority")); |
|
v.push_back(p); |
|
run_test(v, manager); |
|
|
|
float sum = 0.f; |
|
for (connections_t::iterator i = v1.begin() |
|
, end(v1.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
} |
|
sum /= sample_time; |
|
std::cerr << sum << " target: " << limit << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit, 50)); |
|
|
|
std::cerr << "non-prioritized rate: " << p->m_quota / sample_time |
|
<< " target: " << (limit / 200 / 10) << std::endl; |
|
TEST_CHECK(close_to(p->m_quota / sample_time, limit / 200 / 10, 5)); |
|
} |
|
|
|
void test_no_starvation(int limit) |
|
{ |
|
std::cerr << "\ntest no starvation " << limit << std::endl; |
|
bandwidth_manager manager(0); |
|
bandwidth_channel t1; |
|
bandwidth_channel t2; |
|
|
|
global_bwc.throttle(limit); |
|
|
|
const int num_peers = 20; |
|
|
|
connections_t v1; |
|
spawn_connections(v1, manager, t1, num_peers, "p"); |
|
connections_t v; |
|
std::copy(v1.begin(), v1.end(), std::back_inserter(v)); |
|
boost::intrusive_ptr<peer_connection> p( |
|
new peer_connection(manager, t2, 1, false, "no-priority")); |
|
v.push_back(p); |
|
run_test(v, manager); |
|
|
|
float sum = 0.f; |
|
for (connections_t::iterator i = v.begin() |
|
, end(v.end()); i != end; ++i) |
|
{ |
|
sum += (*i)->m_quota; |
|
} |
|
sum /= sample_time; |
|
std::cerr << sum << " target: " << limit << std::endl; |
|
TEST_CHECK(sum > 0); |
|
TEST_CHECK(close_to(sum, limit, 50)); |
|
|
|
std::cerr << "non-prioritized rate: " << p->m_quota / sample_time |
|
<< " target: " << (limit / 200 / num_peers) << std::endl; |
|
TEST_CHECK(close_to(p->m_quota / sample_time, limit / 200 / num_peers, 5)); |
|
} |
|
|
|
int test_main() |
|
{ |
|
using namespace libtorrent; |
|
|
|
test_equal_connections(2, 20); |
|
test_equal_connections(2, 2000); |
|
test_equal_connections(2, 20000); |
|
test_equal_connections(3, 20000); |
|
test_equal_connections(5, 20000); |
|
test_equal_connections(7, 20000); |
|
test_equal_connections(33, 60000); |
|
test_equal_connections(33, 500000); |
|
test_connections_variable_rate(2, 20, 0); |
|
test_connections_variable_rate(5, 20000, 0); |
|
test_connections_variable_rate(3, 2000, 6000); |
|
test_connections_variable_rate(5, 2000, 30000); |
|
test_connections_variable_rate(33, 500000, 0); |
|
test_torrents(2, 400, 400, 0); |
|
test_torrents(2, 100, 500, 0); |
|
test_torrents(2, 3000, 3000, 6000); |
|
test_torrents(1, 40000, 40000, 0); |
|
test_torrents(24, 50000, 50000, 0); |
|
test_torrents(5, 6000, 6000, 3000); |
|
test_torrents(5, 6000, 5000, 4000); |
|
test_torrents(5, 20000, 20000, 30000); |
|
test_torrents_variable_rate(5, 6000, 3000); |
|
test_torrents_variable_rate(5, 20000, 30000); |
|
test_single_peer(40000, true); |
|
test_single_peer(40000, false); |
|
test_peer_priority(40000, false); |
|
test_peer_priority(40000, true); |
|
test_no_starvation(40000); |
|
|
|
return 0; |
|
} |
|
|
|
|
|
|