/*
* This file is part of the TrinityCore Project. See AUTHORS file for Copyright information
*
* 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, see .
*/
#include "IPLocation.h"
#include "BigNumber.h"
#include "Config.h"
#include "Errors.h"
#include "IpAddress.h"
#include "Log.h"
#include "Util.h"
#include
#include
IpLocationStore::IpLocationStore() = default;
IpLocationStore::~IpLocationStore() = default;
void IpLocationStore::Load()
{
_ipLocationStore.clear();
TC_LOG_INFO("server.loading", "Loading IP Location Database...");
std::string databaseFilePath = sConfigMgr->GetStringDefault("IPLocationFile", "");
if (databaseFilePath.empty())
return;
// Check if file exists
std::ifstream databaseFile(databaseFilePath);
if (!databaseFile)
{
TC_LOG_ERROR("server.loading", "IPLocation: No ip database file exists ({}).", databaseFilePath);
return;
}
if (!databaseFile.is_open())
{
TC_LOG_ERROR("server.loading", "IPLocation: Ip database file ({}) can not be opened.", databaseFilePath);
return;
}
std::string ipFrom;
std::string ipTo;
std::string countryCode;
std::string countryName;
BigNumber bnParser;
BigNumber ipv4Max(0xFFFFFFFF);
BigNumber ipv6MappedMask(0xFFFF);
ipv6MappedMask <<= 32;
auto parseStringToIPv6 = [&](std::string const& str) -> Optional>
{
bnParser.SetDecStr(str);
if (!bnParser.SetDecStr(str))
return {};
// convert ipv4 to ipv6 v4 mapped value
if (bnParser <= ipv4Max)
bnParser += ipv6MappedMask;
return bnParser.ToByteArray<16>(false);
};
while (databaseFile.good())
{
// Read lines
if (!std::getline(databaseFile, ipFrom, ','))
break;
if (!std::getline(databaseFile, ipTo, ','))
break;
if (!std::getline(databaseFile, countryCode, ','))
break;
if (!std::getline(databaseFile, countryName, '\n'))
break;
// Remove new lines and return
std::erase_if(countryName, [](char c) { return c == '\r' || c == '\n'; });
// Remove quotation marks
std::erase(ipFrom, '"');
std::erase(ipTo, '"');
std::erase(countryCode, '"');
std::erase(countryName, '"');
if (countryCode == "-")
continue;
// Convert country code to lowercase
strToLower(countryCode);
Optional> from = parseStringToIPv6(ipFrom);
if (!from)
continue;
Optional> to = parseStringToIPv6(ipTo);
if (!to)
continue;
_ipLocationStore.emplace_back(*from, *to, std::move(countryCode), std::move(countryName));
}
std::ranges::sort(_ipLocationStore, {}, &IpLocationRecord::IpFrom);
ASSERT(std::ranges::is_sorted(_ipLocationStore, [](IpLocationRecord const& a, IpLocationRecord const& b) { return a.IpFrom < b.IpTo; }),
"Overlapping IP ranges detected in database file");
databaseFile.close();
TC_LOG_INFO("server.loading", ">> Loaded {} ip location entries.", _ipLocationStore.size());
}
IpLocationRecord const* IpLocationStore::GetLocationRecord(std::string const& ipAddress) const
{
boost::system::error_code error;
boost::asio::ip::address address = Trinity::Net::make_address(ipAddress, error);
if (error)
return nullptr;
std::array bytes = [&]() -> std::array
{
if (address.is_v6())
return address.to_v6().to_bytes();
if (address.is_v4())
return Trinity::Net::make_address_v6(Trinity::Net::v4_mapped, address.to_v4()).to_bytes();
return {};
}();
auto itr = std::ranges::upper_bound(_ipLocationStore, bytes, {}, &IpLocationRecord::IpTo);
if (itr == _ipLocationStore.end())
return nullptr;
if (bytes < itr->IpFrom)
return nullptr;
return &(*itr);
}
IpLocationStore* IpLocationStore::Instance()
{
static IpLocationStore instance;
return &instance;
}