1
0
Fork 0
forked from lthn/blockchain
blockchain/src/common/miniupnp_helper.h
Snider 06e7780df5 updates licence
adds this to the bottom of whats there:
// Copyright (c) 2017-2025 Lethean (https://lt.hn)
//
// Licensed under the European Union Public Licence (EUPL) version 1.2.
// You may obtain a copy of the licence at:
//
//     https://joinup.ec.europa.eu/software/page/eupl/licence-eupl
//
// The EUPL is a copyleft licence that is compatible with the MIT/X11
// licence used by the original projects; the MIT terms are therefore
// considered “grandfathered” under the EUPL for this code.
//
// SPDX‑License‑Identifier: EUPL-1.2
//
2025-09-25 18:18:41 +01:00

196 lines
5.1 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

// Copyright (c) 2014-2018 Zano Project
// Copyright (c) 2014-2018 The Louisdor Project
// Copyright (c) 2012-2013 The Boolberry developers
// Copyright (c) 2017-2025 Lethean (https://lt.hn)
//
// Licensed under the European Union Public Licence (EUPL) version 1.2.
// You may obtain a copy of the licence at:
//
// https://joinup.ec.europa.eu/software/page/eupl/licence-eupl
//
// The EUPL is a copyleft licence that is compatible with the MIT/X11
// licence used by the original projects; the MIT terms are therefore
// considered “grandfathered” under the EUPL for this code.
//
// SPDXLicenseIdentifier: EUPL-1.2
//
#pragma once
#include <thread>
#include <string>
#include <boost/thread.hpp>
#include "include_base_utils.h"
extern "C" {
#include "miniupnpc/miniupnpc.h"
#include "miniupnpc/upnpcommands.h"
#include "miniupnpc/upnperrors.h"
}
#include "misc_language.h"
#include "currency_core/currency_config.h"
#include "version.h"
namespace tools
{
class miniupnp_helper
{
UPNPDev *m_devlist;
UPNPUrls m_urls;
IGDdatas m_data;
char m_lanaddr[64];
int m_IGD;
boost::thread m_mapper_thread;
boost::thread m_initializer_thread;
uint32_t m_external_port;
uint32_t m_internal_port;
uint32_t m_period_ms;
public:
miniupnp_helper():m_devlist(nullptr),
m_urls(AUTO_VAL_INIT(m_urls)),
m_data(AUTO_VAL_INIT(m_data)),
m_IGD(0),
m_external_port(0),
m_internal_port(0),
m_period_ms(0)
{
m_lanaddr[0] = 0;
}
~miniupnp_helper()
{
NESTED_TRY_ENTRY();
deinit();
NESTED_CATCH_ENTRY(__func__);
}
bool start_regular_mapping(uint32_t internal_port, uint32_t external_port, uint32_t period_ms)
{
m_external_port = external_port;
m_internal_port = internal_port;
m_period_ms = period_ms;
if(!init())
return false;
return true;
}
bool stop_mapping()
{
if(m_mapper_thread.joinable())
{
m_mapper_thread.interrupt();
m_mapper_thread.join();
}
if(m_IGD == 1)
{
do_port_unmapping();
}
return true;
}
private:
bool init()
{
m_initializer_thread = boost::thread([=]()
{
deinit();
int error = 0;
m_devlist = upnpDiscover(2000, nullptr, nullptr, 0, 0, 2, &error);
if (error)
{
LOG_PRINT_L0("Failed to call upnpDiscover");
return false;
}
m_IGD = UPNP_GetValidIGD(m_devlist, &m_urls, &m_data, m_lanaddr, sizeof(m_lanaddr));
if (m_IGD != 1)
{
LOG_PRINT_L2("IGD not found");
return false;
}
m_mapper_thread = boost::thread([=]() {run_port_mapping_loop(m_internal_port, m_external_port, m_period_ms); });
return true;
});
return true;
}
bool deinit()
{
if(m_initializer_thread.get_id() != boost::this_thread::get_id())
{
if (m_initializer_thread.joinable())
{
m_initializer_thread.interrupt();
m_initializer_thread.join();
}
}
stop_mapping();
if(m_devlist)
{
freeUPNPDevlist(m_devlist);
m_devlist = nullptr;
}
if(m_IGD > 0)
{
FreeUPNPUrls(&m_urls);
m_IGD = 0;
}
return true;
}
bool run_port_mapping_loop(uint32_t internal_port, uint32_t external_port, uint32_t period_ms)
{
while(true)
{
do_port_mapping(external_port, internal_port);
boost::this_thread::sleep_for(boost::chrono::milliseconds( period_ms ));
}
return true;
}
bool do_port_mapping(uint32_t external_port, uint32_t internal_port)
{
std::string internal_port_str = std::to_string(internal_port);
std::string external_port_str = std::to_string(external_port);
std::string str_desc = CURRENCY_NAME " v" PROJECT_VERSION_LONG;
int r = UPNP_AddPortMapping(m_urls.controlURL, m_data.first.servicetype,
external_port_str.c_str(), internal_port_str.c_str(), m_lanaddr, str_desc.c_str(), "TCP", nullptr, "0");
if(r!=UPNPCOMMAND_SUCCESS)
{
LOG_PRINT_L1("AddPortMapping with external_port_str= " << external_port_str <<
", internal_port_str=" << internal_port_str <<
", failed with code=" << r << "(" << strupnperror(r) << ")");
}else
{
LOG_PRINT_L0("[upnp] port mapped successful (ext: " << external_port_str << ", int:" << internal_port_str << ")");
}
return true;
}
void do_port_unmapping()
{
std::string external_port_str = std::to_string(m_external_port);
int r = UPNP_DeletePortMapping(m_urls.controlURL, m_data.first.servicetype, external_port_str.c_str(), "TCP", nullptr);
if(r!=UPNPCOMMAND_SUCCESS)
{
LOG_PRINT_L1("DeletePortMapping with external_port_str= " << external_port_str <<
", failed with code=" << r << "(" << strupnperror(r) << ")");
}else
{
LOG_PRINT_L0("[upnp] port unmapped successful (ext: " << external_port_str << ")");
}
}
};
}