/* * 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 "ClientBuildInfo.h" #include "DatabaseEnv.h" #include "Log.h" #include "Util.h" #include #include namespace { std::vector Builds; } namespace ClientBuild { std::array ToCharArray(uint32 value) { auto normalize = [](uint8 c) -> char { if (!c || std::isprint(c)) return char(c); return ' '; }; std::array chars = { char((value >> 24) & 0xFF), char((value >> 16) & 0xFF), char((value >> 8) & 0xFF), char(value & 0xFF), '\0' }; auto firstNonZero = std::ranges::find_if(chars, [](char c) { return c != '\0'; }); if (firstNonZero != chars.end()) { // move leading zeros to end std::rotate(chars.begin(), firstNonZero, chars.end()); // ensure we only have printable characters remaining std::ranges::transform(chars, chars.begin(), normalize); } return chars; } bool Platform::IsValid(std::string_view platform) { if (platform.length() > sizeof(uint32)) return false; switch (ToFourCC(platform)) { case Win_x86: case Win_x64: case Win_arm64: case Mac_x86: case Mac_x64: case Mac_arm64: return true; default: break; } return false; } bool PlatformType::IsValid(std::string_view platformType) { if (platformType.length() > sizeof(uint32)) return false; switch (ToFourCC(platformType)) { case Windows: case macOS: return true; default: break; } return false; } bool Arch::IsValid(std::string_view arch) { if (arch.length() > sizeof(uint32)) return false; switch (ToFourCC(arch)) { case x86: case x64: case Arm32: case Arm64: case WA32: return true; default: break; } return false; } bool Type::IsValid(std::string_view type) { if (type.length() > sizeof(uint32)) return false; switch (ToFourCC(type)) { case Retail: case RetailChina: case Beta: case BetaRelease: case Ptr: case PtrRelease: return true; default: break; } return false; } void LoadBuildInfo() { Builds.clear(); // 0 1 2 3 4 if (QueryResult result = LoginDatabase.Query("SELECT majorVersion, minorVersion, bugfixVersion, hotfixVersion, build FROM build_info ORDER BY build ASC")) { do { Field* fields = result->Fetch(); Info& build = Builds.emplace_back(); build.MajorVersion = fields[0].GetUInt32(); build.MinorVersion = fields[1].GetUInt32(); build.BugfixVersion = fields[2].GetUInt32(); std::string hotfixVersion = fields[3].GetString(); if (hotfixVersion.length() < build.HotfixVersion.size()) std::ranges::copy(hotfixVersion, build.HotfixVersion.begin()); else build.HotfixVersion = { }; build.Build = fields[4].GetUInt32(); } while (result->NextRow()); } // 0 1 2 3 4 if (QueryResult result = LoginDatabase.Query("SELECT `build`, `platform`, `arch`, `type`, `key` FROM `build_auth_key`")) { do { Field* fields = result->Fetch(); uint32 build = fields[0].GetInt32(); auto buildInfo = std::ranges::find(Builds, build, &Info::Build); if (buildInfo == Builds.end()) { TC_LOG_ERROR("sql.sql", "ClientBuild::LoadBuildInfo: Unknown `build` {} in `build_auth_key` - missing from `build_info`, skipped.", build); continue; } std::string_view platformType = fields[1].GetStringView(); if (!PlatformType::IsValid(platformType)) { TC_LOG_ERROR("sql.sql", "ClientBuild::LoadBuildInfo: Invalid platform {} for `build` {} in `build_auth_key`, skipped.", platformType, build); continue; } std::string_view arch = fields[2].GetStringView(); if (!Arch::IsValid(arch)) { TC_LOG_ERROR("sql.sql", "ClientBuild::LoadBuildInfo: Invalid `arch` {} for `build` {} in `build_auth_key`, skipped.", arch, build); continue; } std::string_view type = fields[3].GetStringView(); if (!Type::IsValid(type)) { TC_LOG_ERROR("sql.sql", "ClientBuild::LoadBuildInfo: Invalid `type` {} for `build` {} in `build_auth_key`, skipped.", type, build); continue; } AuthKey& buildKey = buildInfo->AuthKeys.emplace_back(); buildKey.Variant = { .Platform = ToFourCC(platformType), .Arch = ToFourCC(arch), .Type = ToFourCC(type) }; buildKey.Key = fields[4].GetBinary(); } while (result->NextRow()); } } Info const* GetBuildInfo(uint32 build) { auto buildInfo = std::ranges::find(Builds, build, &Info::Build); return buildInfo != Builds.end() ? &*buildInfo : nullptr; } uint32 GetMinorMajorBugfixVersionForBuild(uint32 build) { auto buildInfo = std::ranges::lower_bound(Builds, build, {}, &Info::Build); return buildInfo != Builds.end() ? (buildInfo->MajorVersion * 10000 + buildInfo->MinorVersion * 100 + buildInfo->BugfixVersion) : 0; } }