2022-01-20 18:27:56 +00:00
|
|
|
// This file Copyright © 2007-2022 Mnemosyne LLC.
|
2022-02-07 16:25:02 +00:00
|
|
|
// It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
|
2022-01-20 18:27:56 +00:00
|
|
|
// or any future license endorsed by Mnemosyne LLC.
|
|
|
|
// License text can be found in the licenses/ folder.
|
2007-07-29 18:11:21 +00:00
|
|
|
|
2022-08-26 18:35:28 +00:00
|
|
|
#include <array>
|
2021-10-17 20:17:18 +00:00
|
|
|
#include <cerrno>
|
2022-09-21 23:34:18 +00:00
|
|
|
#include <chrono>
|
2022-02-26 01:26:31 +00:00
|
|
|
#include <future>
|
|
|
|
#include <mutex>
|
2022-07-15 00:54:10 +00:00
|
|
|
#include <string>
|
2022-02-26 01:26:31 +00:00
|
|
|
#include <thread>
|
2022-08-17 16:08:36 +00:00
|
|
|
#include <utility>
|
2007-08-04 02:55:06 +00:00
|
|
|
|
2022-03-14 04:43:35 +00:00
|
|
|
#include <fmt/core.h>
|
2022-10-08 19:27:06 +00:00
|
|
|
#include <fmt/format.h>
|
2022-03-14 04:43:35 +00:00
|
|
|
|
2011-10-09 02:05:52 +00:00
|
|
|
#ifdef SYSTEM_MINIUPNP
|
2017-04-19 12:04:45 +00:00
|
|
|
#include <miniupnpc/miniupnpc.h>
|
|
|
|
#include <miniupnpc/upnpcommands.h>
|
2011-10-09 02:05:52 +00:00
|
|
|
#else
|
2017-04-19 12:04:45 +00:00
|
|
|
#include <miniupnp/miniupnpc.h>
|
|
|
|
#include <miniupnp/upnpcommands.h>
|
2011-10-09 02:05:52 +00:00
|
|
|
#endif
|
2007-07-14 16:29:21 +00:00
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
#define LIBTRANSMISSION_PORT_FORWARDING_MODULE
|
|
|
|
|
2006-09-25 18:37:45 +00:00
|
|
|
#include "transmission.h"
|
2022-09-30 13:59:10 +00:00
|
|
|
|
2013-01-25 23:34:20 +00:00
|
|
|
#include "log.h"
|
2022-09-30 13:59:10 +00:00
|
|
|
#include "port-forwarding-upnp.h"
|
2008-04-11 17:01:13 +00:00
|
|
|
#include "port-forwarding.h"
|
2017-06-08 07:24:12 +00:00
|
|
|
#include "tr-assert.h"
|
2022-09-30 13:59:10 +00:00
|
|
|
#include "utils.h" // for _(), tr_strerror()
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2022-02-26 01:26:31 +00:00
|
|
|
namespace
|
2017-04-19 12:04:45 +00:00
|
|
|
{
|
2008-03-07 03:26:59 +00:00
|
|
|
|
2022-02-26 01:26:31 +00:00
|
|
|
enum class UpnpState
|
2007-12-08 19:34:15 +00:00
|
|
|
{
|
2022-09-30 13:59:10 +00:00
|
|
|
Idle,
|
|
|
|
Failed,
|
|
|
|
WillDiscover, // next action is upnpDiscover()
|
|
|
|
Discovering, // currently making blocking upnpDiscover() call in a worker thread
|
|
|
|
WillMap, // next action is UPNP_AddPortMapping()
|
|
|
|
WillUnmap // next action is UPNP_DeletePortMapping()
|
2021-10-06 14:26:07 +00:00
|
|
|
};
|
2007-12-08 19:34:15 +00:00
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
constexpr auto portFwdState(UpnpState upnp_state, bool is_mapped)
|
2022-02-26 01:26:31 +00:00
|
|
|
{
|
|
|
|
switch (upnp_state)
|
|
|
|
{
|
2022-09-30 13:59:10 +00:00
|
|
|
case UpnpState::WillDiscover:
|
|
|
|
case UpnpState::Discovering:
|
2022-02-26 01:26:31 +00:00
|
|
|
return TR_PORT_UNMAPPED;
|
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
case UpnpState::WillMap:
|
2022-02-26 01:26:31 +00:00
|
|
|
return TR_PORT_MAPPING;
|
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
case UpnpState::WillUnmap:
|
2022-02-26 01:26:31 +00:00
|
|
|
return TR_PORT_UNMAPPING;
|
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
case UpnpState::Idle:
|
2022-02-26 01:26:31 +00:00
|
|
|
return is_mapped ? TR_PORT_MAPPED : TR_PORT_UNMAPPED;
|
|
|
|
|
|
|
|
default: // UpnpState::FAILED:
|
|
|
|
return TR_PORT_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2007-11-06 16:02:50 +00:00
|
|
|
struct tr_upnp
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2022-08-03 17:03:28 +00:00
|
|
|
tr_upnp() = default;
|
|
|
|
tr_upnp(tr_upnp&&) = delete;
|
|
|
|
tr_upnp(tr_upnp const&) = delete;
|
|
|
|
tr_upnp& operator=(tr_upnp&&) = delete;
|
|
|
|
tr_upnp& operator=(tr_upnp const&) = delete;
|
|
|
|
|
2022-01-23 17:16:36 +00:00
|
|
|
~tr_upnp()
|
|
|
|
{
|
|
|
|
TR_ASSERT(!isMapped);
|
2022-02-26 01:26:31 +00:00
|
|
|
TR_ASSERT(
|
2022-09-30 13:59:10 +00:00
|
|
|
state == UpnpState::Idle || state == UpnpState::Failed || state == UpnpState::WillDiscover ||
|
|
|
|
state == UpnpState::Discovering);
|
2022-01-23 17:16:36 +00:00
|
|
|
|
|
|
|
FreeUPNPUrls(&urls);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool hasDiscovered = false;
|
2022-02-26 01:26:31 +00:00
|
|
|
UPNPUrls urls = {};
|
|
|
|
IGDdatas data = {};
|
2022-04-21 15:58:13 +00:00
|
|
|
tr_port port;
|
2022-08-26 18:35:28 +00:00
|
|
|
std::string lanaddr;
|
2022-01-23 17:16:36 +00:00
|
|
|
bool isMapped = false;
|
2022-09-30 13:59:10 +00:00
|
|
|
UpnpState state = UpnpState::WillDiscover;
|
2022-02-26 01:26:31 +00:00
|
|
|
|
|
|
|
// Used to return the results of upnpDiscover() from a worker thread
|
|
|
|
// to be processed without blocking in tr_upnpPulse().
|
|
|
|
// This will be pending while the state is UpnpState::DISCOVERING.
|
|
|
|
std::optional<std::future<UPNPDev*>> discover_future;
|
2006-09-25 18:37:45 +00:00
|
|
|
};
|
|
|
|
|
2007-11-06 16:02:50 +00:00
|
|
|
/**
|
|
|
|
***
|
|
|
|
**/
|
2007-06-10 22:26:59 +00:00
|
|
|
|
2022-01-24 19:07:55 +00:00
|
|
|
tr_upnp* tr_upnpInit()
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2022-01-23 17:16:36 +00:00
|
|
|
return new tr_upnp();
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_upnpClose(tr_upnp* handle)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2022-01-23 17:16:36 +00:00
|
|
|
delete handle;
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2011-10-09 02:05:52 +00:00
|
|
|
/**
|
|
|
|
*** Wrappers for miniupnpc functions
|
|
|
|
**/
|
|
|
|
|
2022-01-21 15:44:45 +00:00
|
|
|
static struct UPNPDev* tr_upnpDiscover(int msec, char const* bindaddr)
|
2011-10-09 02:05:52 +00:00
|
|
|
{
|
2021-10-19 16:09:38 +00:00
|
|
|
UPNPDev* ret = nullptr;
|
|
|
|
auto have_err = bool{};
|
2014-04-27 19:31:10 +00:00
|
|
|
|
|
|
|
#if (MINIUPNPC_API_VERSION >= 8) /* adds ipv6 and error args */
|
2017-04-19 12:04:45 +00:00
|
|
|
int err = UPNPDISCOVER_SUCCESS;
|
|
|
|
|
|
|
|
#if (MINIUPNPC_API_VERSION >= 14) /* adds ttl */
|
2022-01-21 15:44:45 +00:00
|
|
|
ret = upnpDiscover(msec, bindaddr, nullptr, 0, 0, 2, &err);
|
2017-04-19 12:04:45 +00:00
|
|
|
#else
|
2022-01-21 15:44:45 +00:00
|
|
|
ret = upnpDiscover(msec, bindaddr, nullptr, 0, 0, &err);
|
2017-04-19 12:04:45 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
have_err = err != UPNPDISCOVER_SUCCESS;
|
2014-04-27 19:31:10 +00:00
|
|
|
#else
|
2022-01-21 15:44:45 +00:00
|
|
|
ret = upnpDiscover(msec, bindaddr, nullptr, 0);
|
2021-09-15 00:18:09 +00:00
|
|
|
have_err = ret == nullptr;
|
2011-10-09 02:05:52 +00:00
|
|
|
#endif
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
if (have_err)
|
|
|
|
{
|
2022-03-14 04:43:35 +00:00
|
|
|
tr_logAddDebug(fmt::format("upnpDiscover failed: {} ({})", tr_strerror(errno), errno));
|
2017-04-19 12:04:45 +00:00
|
|
|
}
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
return ret;
|
2011-10-09 02:05:52 +00:00
|
|
|
}
|
|
|
|
|
2022-09-06 17:52:58 +00:00
|
|
|
static int tr_upnpGetSpecificPortMappingEntry(tr_upnp const* handle, char const* proto)
|
2011-10-09 02:05:52 +00:00
|
|
|
{
|
2022-08-26 18:35:28 +00:00
|
|
|
auto int_client = std::array<char, 16>{};
|
|
|
|
auto int_port = std::array<char, 16>{};
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2022-04-21 15:58:13 +00:00
|
|
|
auto const port_str = fmt::format(FMT_STRING("{:d}"), handle->port.host());
|
2014-04-27 19:31:10 +00:00
|
|
|
|
|
|
|
#if (MINIUPNPC_API_VERSION >= 10) /* adds remoteHost arg */
|
2021-10-19 16:09:38 +00:00
|
|
|
int const err = UPNP_GetSpecificPortMappingEntry(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
2022-04-03 18:23:00 +00:00
|
|
|
port_str.c_str(),
|
2021-08-15 09:41:48 +00:00
|
|
|
proto,
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr /*remoteHost*/,
|
2022-08-26 18:35:28 +00:00
|
|
|
std::data(int_client),
|
|
|
|
std::data(int_port),
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr /*desc*/,
|
|
|
|
nullptr /*enabled*/,
|
|
|
|
nullptr /*duration*/);
|
2014-04-27 19:31:10 +00:00
|
|
|
#elif (MINIUPNPC_API_VERSION >= 8) /* adds desc, enabled and leaseDuration args */
|
2021-10-19 16:09:38 +00:00
|
|
|
int const err = UPNP_GetSpecificPortMappingEntry(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
2022-04-03 18:23:00 +00:00
|
|
|
port_str.c_str(),
|
2021-08-15 09:41:48 +00:00
|
|
|
proto,
|
2022-08-26 18:35:28 +00:00
|
|
|
std::data(int_client),
|
|
|
|
std::data(int_port),
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr /*desc*/,
|
|
|
|
nullptr /*enabled*/,
|
|
|
|
nullptr /*duration*/);
|
2011-10-09 02:05:52 +00:00
|
|
|
#else
|
2021-10-19 16:09:38 +00:00
|
|
|
int const err = UPNP_GetSpecificPortMappingEntry(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
2022-04-03 18:23:00 +00:00
|
|
|
port_str.c_str(),
|
2021-08-15 09:41:48 +00:00
|
|
|
proto,
|
2022-08-26 18:35:28 +00:00
|
|
|
std::data(int_client),
|
|
|
|
std::data(int_port));
|
2011-10-09 02:05:52 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static int tr_upnpAddPortMapping(tr_upnp const* handle, char const* proto, tr_port port, char const* desc)
|
2011-10-09 02:05:52 +00:00
|
|
|
{
|
2017-04-20 16:02:19 +00:00
|
|
|
int const old_errno = errno;
|
2011-10-09 02:05:52 +00:00
|
|
|
errno = 0;
|
|
|
|
|
2022-04-21 15:58:13 +00:00
|
|
|
auto const port_str = fmt::format(FMT_STRING("{:d}"), port.host());
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2014-04-27 19:31:10 +00:00
|
|
|
#if (MINIUPNPC_API_VERSION >= 8)
|
2022-07-27 14:03:13 +00:00
|
|
|
int const err = UPNP_AddPortMapping(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
2022-04-03 18:23:00 +00:00
|
|
|
port_str.c_str(),
|
|
|
|
port_str.c_str(),
|
2022-08-26 18:35:28 +00:00
|
|
|
handle->lanaddr.c_str(),
|
2021-08-15 09:41:48 +00:00
|
|
|
desc,
|
|
|
|
proto,
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr,
|
|
|
|
nullptr);
|
2011-10-09 02:05:52 +00:00
|
|
|
#else
|
2022-07-27 14:03:13 +00:00
|
|
|
int const err = UPNP_AddPortMapping(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
2022-04-03 18:23:00 +00:00
|
|
|
port_str.c_str(),
|
|
|
|
port_str.c_str(),
|
2022-08-26 18:35:28 +00:00
|
|
|
handle->lanaddr.c_str(),
|
2021-08-15 09:41:48 +00:00
|
|
|
desc,
|
|
|
|
proto,
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr);
|
2011-10-09 02:05:52 +00:00
|
|
|
#endif
|
|
|
|
|
2017-04-30 16:25:26 +00:00
|
|
|
if (err != 0)
|
2017-04-19 12:04:45 +00:00
|
|
|
{
|
2022-03-14 04:43:35 +00:00
|
|
|
tr_logAddDebug(fmt::format("{} Port forwarding failed with error {}: {} ({})", proto, err, tr_strerror(errno), errno));
|
2017-04-19 12:04:45 +00:00
|
|
|
}
|
2011-10-09 02:05:52 +00:00
|
|
|
|
|
|
|
errno = old_errno;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static void tr_upnpDeletePortMapping(tr_upnp const* handle, char const* proto, tr_port port)
|
2011-10-09 02:05:52 +00:00
|
|
|
{
|
2022-04-21 15:58:13 +00:00
|
|
|
auto const port_str = fmt::format(FMT_STRING("{:d}"), port.host());
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2022-04-03 18:23:00 +00:00
|
|
|
UPNP_DeletePortMapping(handle->urls.controlURL, handle->data.first.servicetype, port_str.c_str(), proto, nullptr);
|
2011-10-09 02:05:52 +00:00
|
|
|
}
|
|
|
|
|
2007-11-06 16:02:50 +00:00
|
|
|
/**
|
|
|
|
***
|
|
|
|
**/
|
2007-04-15 07:36:24 +00:00
|
|
|
|
2010-06-27 15:29:23 +00:00
|
|
|
enum
|
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
UPNP_IGD_NONE = 0,
|
|
|
|
UPNP_IGD_VALID_CONNECTED = 1,
|
|
|
|
UPNP_IGD_VALID_NOT_CONNECTED = 2,
|
|
|
|
UPNP_IGD_INVALID = 3
|
2010-06-27 15:29:23 +00:00
|
|
|
};
|
|
|
|
|
2022-08-02 23:34:53 +00:00
|
|
|
static auto* discoverThreadfunc(std::string bindaddr) // NOLINT performance-unnecessary-value-param
|
2022-02-26 01:26:31 +00:00
|
|
|
{
|
2022-09-30 13:59:10 +00:00
|
|
|
// If multicastif is not NULL, it will be used instead of the default
|
|
|
|
// multicast interface for sending SSDP discover packets.
|
|
|
|
char const* multicastif = std::empty(bindaddr) ? nullptr : bindaddr.c_str();
|
|
|
|
return tr_upnpDiscover(2000, multicastif);
|
2022-02-26 01:26:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
static bool isFutureReady(std::future<T> const& future)
|
|
|
|
{
|
|
|
|
return future.wait_for(std::chrono::seconds(0)) == std::future_status::ready;
|
|
|
|
}
|
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
tr_port_forwarding_state tr_upnpPulse(tr_upnp* handle, tr_port port, bool is_enabled, bool do_port_check, std::string bindaddr)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2022-09-30 13:59:10 +00:00
|
|
|
if (is_enabled && handle->state == UpnpState::WillDiscover)
|
2007-11-09 16:10:48 +00:00
|
|
|
{
|
2022-02-26 01:26:31 +00:00
|
|
|
TR_ASSERT(!handle->discover_future);
|
|
|
|
|
2022-07-10 01:03:40 +00:00
|
|
|
auto task = std::packaged_task<UPNPDev*(std::string)>{ discoverThreadfunc };
|
2022-02-26 01:26:31 +00:00
|
|
|
handle->discover_future = task.get_future();
|
2022-09-30 13:59:10 +00:00
|
|
|
handle->state = UpnpState::Discovering;
|
2022-02-26 01:26:31 +00:00
|
|
|
|
2022-07-10 01:03:40 +00:00
|
|
|
std::thread(std::move(task), std::move(bindaddr)).detach();
|
2022-02-26 01:26:31 +00:00
|
|
|
}
|
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
if (is_enabled && handle->state == UpnpState::Discovering && handle->discover_future &&
|
2022-02-26 01:26:31 +00:00
|
|
|
isFutureReady(*handle->discover_future))
|
|
|
|
{
|
|
|
|
auto* const devlist = handle->discover_future->get();
|
|
|
|
handle->discover_future.reset();
|
2017-04-19 12:04:45 +00:00
|
|
|
|
2022-01-23 17:16:36 +00:00
|
|
|
FreeUPNPUrls(&handle->urls);
|
2022-08-26 18:35:28 +00:00
|
|
|
auto lanaddr = std::array<char, TR_ADDRSTRLEN>{};
|
|
|
|
if (UPNP_GetValidIGD(devlist, &handle->urls, &handle->data, std::data(lanaddr), std::size(lanaddr)) ==
|
2021-08-15 09:41:48 +00:00
|
|
|
UPNP_IGD_VALID_CONNECTED)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2022-03-14 04:43:35 +00:00
|
|
|
tr_logAddInfo(fmt::format(_("Found Internet Gateway Device '{url}'"), fmt::arg("url", handle->urls.controlURL)));
|
2022-11-22 00:08:06 +00:00
|
|
|
tr_logAddInfo(fmt::format(_("Local Address is '{address}'"), fmt::arg("address", handle->lanaddr.data())));
|
2022-09-30 13:59:10 +00:00
|
|
|
handle->state = UpnpState::Idle;
|
2014-11-30 19:38:47 +00:00
|
|
|
handle->hasDiscovered = true;
|
2022-08-26 18:35:28 +00:00
|
|
|
handle->lanaddr = std::data(lanaddr);
|
2008-09-23 19:11:04 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-09-30 13:59:10 +00:00
|
|
|
handle->state = UpnpState::Failed;
|
2022-03-14 04:43:35 +00:00
|
|
|
tr_logAddDebug(fmt::format("UPNP_GetValidIGD failed: {} ({})", tr_strerror(errno), errno));
|
2022-03-11 21:09:22 +00:00
|
|
|
tr_logAddDebug("If your router supports UPnP, please make sure UPnP is enabled!");
|
2007-11-09 16:10:48 +00:00
|
|
|
}
|
2017-04-19 12:04:45 +00:00
|
|
|
|
|
|
|
freeUPNPDevlist(devlist);
|
2007-11-09 16:10:48 +00:00
|
|
|
}
|
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
if ((handle->state == UpnpState::Idle) && (handle->isMapped) && (!is_enabled || handle->port != port))
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2022-09-30 13:59:10 +00:00
|
|
|
handle->state = UpnpState::WillUnmap;
|
2007-06-10 22:26:59 +00:00
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2022-09-07 16:04:28 +00:00
|
|
|
if (is_enabled && handle->isMapped && do_port_check &&
|
2020-11-01 21:47:57 +00:00
|
|
|
((tr_upnpGetSpecificPortMappingEntry(handle, "TCP") != UPNPCOMMAND_SUCCESS) ||
|
2021-08-15 09:41:48 +00:00
|
|
|
(tr_upnpGetSpecificPortMappingEntry(handle, "UDP") != UPNPCOMMAND_SUCCESS)))
|
2009-04-22 16:00:45 +00:00
|
|
|
{
|
2022-04-21 15:58:13 +00:00
|
|
|
tr_logAddInfo(fmt::format(_("Port {port} is not forwarded"), fmt::arg("port", handle->port.host())));
|
2020-11-01 21:47:57 +00:00
|
|
|
handle->isMapped = false;
|
2009-04-22 16:00:45 +00:00
|
|
|
}
|
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
if (handle->state == UpnpState::WillUnmap)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_upnpDeletePortMapping(handle, "TCP", handle->port);
|
|
|
|
tr_upnpDeletePortMapping(handle, "UDP", handle->port);
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2022-03-14 04:43:35 +00:00
|
|
|
tr_logAddInfo(fmt::format(
|
|
|
|
_("Stopping port forwarding through '{url}', service '{type}'"),
|
|
|
|
fmt::arg("url", handle->urls.controlURL),
|
|
|
|
fmt::arg("type", handle->data.first.servicetype)));
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2017-04-30 16:25:26 +00:00
|
|
|
handle->isMapped = false;
|
2022-09-30 13:59:10 +00:00
|
|
|
handle->state = UpnpState::Idle;
|
2022-04-21 15:58:13 +00:00
|
|
|
handle->port = {};
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
if ((handle->state == UpnpState::Idle) && is_enabled && !handle->isMapped)
|
2007-12-08 19:34:15 +00:00
|
|
|
{
|
2022-09-30 13:59:10 +00:00
|
|
|
handle->state = UpnpState::WillMap;
|
2007-12-08 19:34:15 +00:00
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2022-09-30 13:59:10 +00:00
|
|
|
if (handle->state == UpnpState::WillMap)
|
2007-12-08 19:34:15 +00:00
|
|
|
{
|
2007-12-09 17:50:05 +00:00
|
|
|
errno = 0;
|
2007-12-24 07:02:40 +00:00
|
|
|
|
2021-09-15 00:18:09 +00:00
|
|
|
if (handle->urls.controlURL == nullptr)
|
2017-04-19 12:04:45 +00:00
|
|
|
{
|
2019-07-14 12:40:41 +00:00
|
|
|
handle->isMapped = false;
|
2017-04-19 12:04:45 +00:00
|
|
|
}
|
2008-09-23 19:11:04 +00:00
|
|
|
else
|
|
|
|
{
|
2022-09-30 13:59:10 +00:00
|
|
|
auto const desc = fmt::format(FMT_STRING("Transmission at {:d}"), port.host());
|
2022-04-03 18:23:00 +00:00
|
|
|
int const err_tcp = tr_upnpAddPortMapping(handle, "TCP", port, desc.c_str());
|
|
|
|
int const err_udp = tr_upnpAddPortMapping(handle, "UDP", port, desc.c_str());
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2017-04-30 16:25:26 +00:00
|
|
|
handle->isMapped = err_tcp == 0 || err_udp == 0;
|
2007-12-24 07:02:40 +00:00
|
|
|
}
|
2017-04-19 12:04:45 +00:00
|
|
|
|
2022-04-14 04:19:18 +00:00
|
|
|
tr_logAddDebug(fmt::format(
|
2022-03-14 04:43:35 +00:00
|
|
|
_("Port forwarding through '{url}', service '{type}'. (local address: {address}:{port})"),
|
|
|
|
fmt::arg("url", handle->urls.controlURL),
|
|
|
|
fmt::arg("type", handle->data.first.servicetype),
|
|
|
|
fmt::arg("address", handle->lanaddr),
|
2022-04-21 15:58:13 +00:00
|
|
|
fmt::arg("port", port.host())));
|
2017-04-19 12:04:45 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (handle->isMapped)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2022-04-21 15:58:13 +00:00
|
|
|
tr_logAddInfo(fmt::format(_("Port {port} is forwarded"), fmt::arg("port", port.host())));
|
2007-12-08 19:34:15 +00:00
|
|
|
handle->port = port;
|
2022-09-30 13:59:10 +00:00
|
|
|
handle->state = UpnpState::Idle;
|
2008-09-23 19:11:04 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-03-14 04:43:35 +00:00
|
|
|
tr_logAddInfo(_("If your router supports UPnP, please make sure UPnP is enabled!"));
|
2022-04-21 15:58:13 +00:00
|
|
|
handle->port = {};
|
2022-09-30 13:59:10 +00:00
|
|
|
handle->state = UpnpState::Failed;
|
2007-12-08 19:34:15 +00:00
|
|
|
}
|
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2022-02-26 01:26:31 +00:00
|
|
|
return portFwdState(handle->state, handle->isMapped);
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|