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.
F4KXL_DStarGateway/Common/HostsFilesManager.cpp

211 lines
6.1 KiB

/*
* Copyright (c) 2024 by Geoffrey F4FXL / KC3FRA
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <cassert>
#include "HostsFilesManager.h"
#include "HostFile.h"
#include "StringUtils.h"
#include "UDPReaderWriter.h"
#include "Log.h"
std::string CHostsFilesManager::m_hostFilesDirectory("");
std::string CHostsFilesManager::m_customFilesDirectory("");
bool CHostsFilesManager::m_dextraEnabled = false;
bool CHostsFilesManager::m_dcsEnabled = false;
bool CHostsFilesManager::m_dplusEnabled = false;
bool CHostsFilesManager::m_xlxEnabled = false;
std::string CHostsFilesManager::m_dextraUrl("");
std::string CHostsFilesManager::m_dcsUrl("");
std::string CHostsFilesManager::m_dplusUrl("");
std::string CHostsFilesManager::m_xlxUrl("");
CCacheManager * CHostsFilesManager::m_cache = nullptr;
CTimer CHostsFilesManager::m_downloadTimer(1000U, 60 * 60 * 24);
HostFileDownloadCallback CHostsFilesManager::m_downloadCallback = nullptr;
void CHostsFilesManager::setHostFilesDirectories(const std::string & hostFilesDir, const std::string & customHostFilesDir)
{
m_hostFilesDirectory.assign(hostFilesDir);
m_customFilesDirectory.assign(customHostFilesDir);
}
void CHostsFilesManager::setDownloadCallback(HostFileDownloadCallback callback)
{
m_downloadCallback = callback;
}
void CHostsFilesManager::setDextra(bool enabled, const std::string & url)
{
m_dextraEnabled = enabled;
m_dextraUrl.assign(url);
}
void CHostsFilesManager::setDPlus(bool enabled, const std::string & url)
{
m_dplusEnabled = enabled;
m_dplusUrl.assign(url);
}
void CHostsFilesManager::setDCS(bool enabled, const std::string & url)
{
m_dcsEnabled = enabled;
m_dcsUrl.assign(url);
}
void CHostsFilesManager::setXLX(bool enabled, const std::string & url)
{
m_xlxEnabled = enabled;
m_xlxUrl.assign(url);
}
void CHostsFilesManager::setCache(CCacheManager * cache)
{
assert(cache != nullptr);
m_cache = cache;
}
void CHostsFilesManager::clock(unsigned int ms)
{
m_downloadTimer.clock(ms);
if(m_downloadTimer.hasExpired()) {
CLog::logInfo("Downloading hosts files after %u hours", m_downloadTimer.getTimeout() / 3600U);
UpdateHostsAsync(); // call and forget
m_downloadTimer.start();
}
}
void CHostsFilesManager::setDownloadTimeout(unsigned int seconds)
{
m_downloadTimer.start(seconds);
}
bool CHostsFilesManager::UpdateHostsFromInternet()
{
CLog::logInfo("Updating hosts files from internet");
bool ret = true;
if(m_dextraEnabled && !m_dextraUrl.empty()) ret = m_downloadCallback(m_dextraUrl, m_hostFilesDirectory + "/" + DEXTRA_HOSTS_FILE_NAME) && ret;
if(m_dcsEnabled && !m_dcsUrl.empty()) ret = m_downloadCallback(m_dcsUrl, m_hostFilesDirectory + "/" + DCS_HOSTS_FILE_NAME) && ret;
if(m_dplusEnabled && !m_dplusUrl.empty()) ret = m_downloadCallback(m_dplusUrl, m_hostFilesDirectory + "/" + DPLUS_HOSTS_FILE_NAME) && ret;
if(m_xlxEnabled && !m_xlxUrl.empty()) ret = m_downloadCallback(m_xlxUrl, m_hostFilesDirectory + "/" + XLX_HOSTS_FILE_NAME) && ret;
if(!ret) CLog::logWarning("Some hosts files failed to downlaod");
CHostsFilesManager::loadReflectors(m_hostFilesDirectory);
return ret;
}
bool CHostsFilesManager::UpdateHostsFromLocal()
{
CHostsFilesManager::loadReflectors(m_customFilesDirectory);
return true;
}
bool CHostsFilesManager::UpdateHosts()
{
bool ret = UpdateHostsFromInternet();
UpdateHostsFromLocal() && ret;
return ret;
}
std::future<bool> CHostsFilesManager::UpdateHostsAsync()
{
auto fut = std::async(std::launch::async, UpdateHosts);
return fut;
}
void CHostsFilesManager::loadReflectors(const std::string & directory)
{
if (m_xlxEnabled) {
std::string fileName = directory + "/" + XLX_HOSTS_FILE_NAME;
loadReflectors(fileName, DP_DCS);
}
if (m_dplusEnabled) {
std::string fileName = directory + "/" + DPLUS_HOSTS_FILE_NAME;
loadReflectors(fileName, DP_DPLUS);
}
if (m_dextraEnabled) {
std::string fileName = directory + "/" + DEXTRA_HOSTS_FILE_NAME;
loadReflectors(fileName, DP_DEXTRA);
}
if (m_dcsEnabled) {
std::string fileName = directory + "/" + DCS_HOSTS_FILE_NAME;
loadReflectors(fileName, DP_DCS);
}
}
void CHostsFilesManager::loadReflectors(const std::string & hostFileName, DSTAR_PROTOCOL proto)
{
if(m_cache == nullptr) {
CLog::logDebug("HostsFilesManager cache not initilized");
return;
}
unsigned int count = 0U;
CHostFile hostFile(hostFileName, false);
for (unsigned int i = 0U; i < hostFile.getCount(); i++) {
std::string reflector = hostFile.getName(i);
in_addr address = CUDPReaderWriter::lookup(hostFile.getAddress(i));
bool lock = hostFile.getLock(i);
if (address.s_addr != INADDR_NONE) {
unsigned char* ucp = (unsigned char*)&address;
std::string addrText;
addrText = CStringUtils::string_format("%u.%u.%u.%u", ucp[0U] & 0xFFU, ucp[1U] & 0xFFU, ucp[2U] & 0xFFU, ucp[3U] & 0xFFU);
if (lock)
CLog::logInfo("Locking %s to %s", reflector.c_str(), addrText.c_str());
reflector.resize(LONG_CALLSIGN_LENGTH - 1U, ' ');
reflector += "G";
m_cache->updateGateway(reflector, addrText, proto, lock, true);
count++;
}
}
std::string protoString;
switch (proto)
{
case DP_DEXTRA:
protoString = "DExtra";
break;
case DP_DCS:
protoString = "DCS";
break;
case DP_DPLUS:
protoString = "DPlus";
break;
default:
// ???
break;
}
CLog::logInfo("Loaded %u of %u %s hosts from %s", count, hostFile.getCount(), protoString.c_str() , hostFileName.c_str());
}

Powered by TurnKey Linux.