// This file Copyright © 2007-2022 Mnemosyne LLC. // It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only), // or any future license endorsed by Mnemosyne LLC. // License text can be found in the licenses/ folder. #include #include #include // uint32_t #include #include /* evutil_inet_ntop() */ #include #define ENABLE_STRNATPMPERR #include "natpmp.h" #define LIBTRANSMISSION_PORT_FORWARDING_MODULE #include "transmission.h" #include "log.h" #include "port-forwarding-natpmp.h" #include "port-forwarding.h" #include "utils.h" static void logVal(char const* func, int ret) { if (ret == NATPMP_TRYAGAIN) { return; } if (ret >= 0) { tr_logAddDebug(fmt::format("{} succeeded ({})", func, ret)); } else { tr_logAddDebug(fmt::format( "{} failed. Natpmp returned {} ({}); errno is {} ({})", func, ret, strnatpmperr(ret), errno, tr_strerror(errno))); } } bool tr_natpmp::canSendCommand() const { return tr_time() >= command_time_; } void tr_natpmp::setCommandTime() { command_time_ = tr_time() + CommandWaitSecs; } tr_natpmp::PulseResult tr_natpmp::pulse(tr_port private_port, bool is_enabled) { if (is_enabled && state_ == State::Discover) { int val = initnatpmp(&natpmp_, 0, 0); logVal("initnatpmp", val); val = sendpublicaddressrequest(&natpmp_); logVal("sendpublicaddressrequest", val); state_ = val < 0 ? State::Err : State::RecvPub; has_discovered_ = true; setCommandTime(); } if (state_ == State::RecvPub && canSendCommand()) { natpmpresp_t response; auto const val = readnatpmpresponseorretry(&natpmp_, &response); logVal("readnatpmpresponseorretry", val); if (val >= 0) { auto str = std::array{}; evutil_inet_ntop(AF_INET, &response.pnu.publicaddress.addr, std::data(str), std::size(str)); tr_logAddInfo(fmt::format(_("Found public address '{address}'"), fmt::arg("address", std::data(str)))); state_ = State::Idle; } else if (val != NATPMP_TRYAGAIN) { state_ = State::Err; } } if ((state_ == State::Idle || state_ == State::Err) && is_mapped_ && (!is_enabled || private_port_ != private_port)) { state_ = State::SendUnmap; } if (state_ == State::SendUnmap && canSendCommand()) { auto const val = sendnewportmappingrequest(&natpmp_, NATPMP_PROTOCOL_TCP, private_port_.host(), public_port_.host(), 0); logVal("sendnewportmappingrequest", val); state_ = val < 0 ? State::Err : State::RecvUnmap; setCommandTime(); } if (state_ == State::RecvUnmap) { auto resp = natpmpresp_t{}; auto const val = readnatpmpresponseorretry(&natpmp_, &resp); logVal("readnatpmpresponseorretry", val); if (val >= 0) { auto const unmapped_port = tr_port::fromHost(resp.pnu.newportmapping.privateport); tr_logAddInfo(fmt::format(_("Port {port} is no longer forwarded"), fmt::arg("port", unmapped_port.host()))); if (private_port_ == unmapped_port) { private_port_.clear(); public_port_.clear(); state_ = State::Idle; is_mapped_ = false; } } else if (val != NATPMP_TRYAGAIN) { state_ = State::Err; } } if (state_ == State::Idle) { if (is_enabled && !is_mapped_ && has_discovered_) { state_ = State::SendMap; } else if (is_mapped_ && tr_time() >= renew_time_) { state_ = State::SendMap; } } if (state_ == State::SendMap && canSendCommand()) { auto const val = sendnewportmappingrequest( &natpmp_, NATPMP_PROTOCOL_TCP, private_port.host(), private_port.host(), LifetimeSecs); logVal("sendnewportmappingrequest", val); state_ = val < 0 ? State::Err : State::RecvMap; setCommandTime(); } if (state_ == State::RecvMap) { auto resp = natpmpresp_t{}; auto const val = readnatpmpresponseorretry(&natpmp_, &resp); logVal("readnatpmpresponseorretry", val); if (val >= 0) { state_ = State::Idle; is_mapped_ = true; renew_time_ = tr_time() + (resp.pnu.newportmapping.lifetime / 2); private_port_ = tr_port::fromHost(resp.pnu.newportmapping.privateport); public_port_ = tr_port::fromHost(resp.pnu.newportmapping.mappedpublicport); tr_logAddInfo(fmt::format(_("Port {port} forwarded successfully"), fmt::arg("port", private_port_.host()))); } else if (val != NATPMP_TRYAGAIN) { state_ = State::Err; } } switch (state_) { case State::Idle: return { is_mapped_ ? TR_PORT_MAPPED : TR_PORT_UNMAPPED, public_port_, private_port_ }; case State::Discover: return { TR_PORT_UNMAPPED, {}, {} }; case State::RecvPub: case State::SendMap: case State::RecvMap: return { TR_PORT_MAPPING, {}, {} }; case State::SendUnmap: case State::RecvUnmap: return { TR_PORT_UNMAPPING, {}, {} }; default: return { TR_PORT_ERROR, {}, {} }; } }