Skip to content

Commit

Permalink
Add announce_port support
Browse files Browse the repository at this point in the history
  • Loading branch information
0xThiebaut committed Oct 27, 2024
1 parent 24e658a commit 84036c6
Show file tree
Hide file tree
Showing 5 changed files with 106 additions and 6 deletions.
14 changes: 14 additions & 0 deletions include/libtorrent/settings_pack.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -2068,6 +2068,20 @@ namespace aux {
i2p_inbound_length,
i2p_outbound_length,

// ``announce_port`` is the port passed along to trackers as the
// ``&port=`` parameter. This setting does not affect the effective
// listening port. If left as zero (default), the listening port
// value is used.
//
// .. note::
// This setting is only meant for very special cases where a
// seed's listening port differs from the external port. As an
// example, if a local proxy is used and that the proxy supports
// reverse tunnels through NAT-PMP, the tracker must connect to
// the external NAT-PMP port (configured using ``announce_port``)
// instead of the actual local listening port.
announce_port,

max_int_setting_internal
};

Expand Down
85 changes: 84 additions & 1 deletion simulation/test_tracker.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,6 @@ void test_interval(int interval)
// the number of announces seen by the tracker
default_settings.set_str(settings_pack::listen_interfaces, "0.0.0.0:6881");
lt::add_torrent_params default_add_torrent;

setup_swarm(1, swarm_test::upload, sim, default_settings, default_add_torrent
// add session
, [](lt::settings_pack&) {}
Expand Down Expand Up @@ -361,6 +360,90 @@ void on_alert_notify(lt::session* ses)
});
}

void test_announce()
{
using sim::asio::ip::address_v4;
sim::default_config network_cfg;
sim::simulation sim{network_cfg};

sim::asio::io_context web_server(sim, make_address_v4("2.2.2.2"));

// listen on port 8080
sim::http_server http(web_server, 8080);

int announces = 0;

// expect announced IP & port
std::string const expect_port = "&port=1234";
std::string const expect_ip = "&ip=1.2.3.4";

http.register_handler("/announce"
, [&announces, expect_port, expect_ip](std::string method, std::string req
, std::map<std::string, std::string>&)
{
++announces;
TEST_EQUAL(method, "GET");
TEST_CHECK(req.find(expect_port) != std::string::npos);
TEST_CHECK(req.find(expect_ip) != std::string::npos);
char response[500];
int const size = std::snprintf(response, sizeof(response), "d8:intervali1800e5:peers0:e");
return sim::send_response(200, "OK", size) + response;
});

{
lt::session_proxy zombie;

std::vector<asio::ip::address> ips;
ips.push_back(make_address("123.0.0.3"));

asio::io_context ios(sim, ips);
lt::settings_pack sett = settings();
sett.set_str(settings_pack::listen_interfaces, "0.0.0.0:6881");
sett.set_str(settings_pack::announce_ip, "1.2.3.4");
sett.set_int(settings_pack::announce_port, 1234);

auto ses = std::make_unique<lt::session>(sett, ios);

ses->set_alert_notify(std::bind(&on_alert_notify, ses.get()));

lt::add_torrent_params p;
p.name = "test-torrent";
p.save_path = ".";
p.info_hashes.v1.assign("abababababababababab");

p.trackers.push_back("http://2.2.2.2:8080/announce");
ses->async_add_torrent(p);

// stop the torrent 5 seconds in
sim::timer t1(sim, lt::seconds(5)
, [&ses](boost::system::error_code const&)
{
std::vector<lt::torrent_handle> torrents = ses->get_torrents();
for (auto const& t : torrents)
{
t.pause();
}
});

// then shut down 10 seconds in
sim::timer t2(sim, lt::seconds(10)
, [&ses,&zombie](boost::system::error_code const&)
{
zombie = ses->abort();
ses.reset();
});

sim.run();
}

TEST_EQUAL(announces, 2);
}

// this test makes sure that a seed can overwrite its announced IP & port
TORRENT_TEST(announce_ip_port) {
test_announce();
}

static const int num_interfaces = 3;

void test_ipv6_support(char const* listen_interfaces
Expand Down
8 changes: 5 additions & 3 deletions src/session_impl.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1301,9 +1301,11 @@ namespace {
#endif
req.ssl_ctx = &m_ssl_ctx;
#endif

auto ls = req.outgoing_socket.get();
if (ls)
if (const auto announce_port = std::uint16_t(m_settings.get_int(settings_pack::announce_port)))
{
req.listen_port = announce_port;
}
else if (auto ls = req.outgoing_socket.get())
{
req.listen_port =
#ifdef TORRENT_SSL_PEERS
Expand Down
3 changes: 2 additions & 1 deletion src/settings_pack.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -401,7 +401,8 @@ constexpr int DISK_WRITE_MODE = settings_pack::enable_os_cache;
SET(i2p_inbound_quantity, 3, nullptr),
SET(i2p_outbound_quantity, 3, nullptr),
SET(i2p_inbound_length, 3, nullptr),
SET(i2p_outbound_length, 3, nullptr)
SET(i2p_outbound_length, 3, nullptr),
SET(announce_port, 0, nullptr)
}});

#undef SET
Expand Down
2 changes: 1 addition & 1 deletion src/tracker_manager.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -279,7 +279,7 @@ constexpr tracker_request_flags_t tracker_request::i2p;
#ifndef TORRENT_DISABLE_LOGGING
std::shared_ptr<request_callback> cb = c.lock();
if (cb) cb->debug_log("*** QUEUE_TRACKER_REQUEST [ listen_port: %d ]"
, req.listen_port);
, req.listen_port);
#endif

std::string const protocol = req.url.substr(0, req.url.find(':'));
Expand Down

0 comments on commit 84036c6

Please sign in to comment.