mirror of
https://github.com/mod-playerbots/azerothcore-wotlk.git
synced 2025-11-29 17:38:24 +08:00
refactor(Core/Motd): improve and simplify (#21252)
This commit is contained in:
@@ -0,0 +1,5 @@
|
||||
--
|
||||
DELETE FROM `acore_string` WHERE `entry` IN (56, 82);
|
||||
INSERT INTO `acore_string` (`entry`, `content_default`, `locale_deDE`, `locale_zhCN`, `locale_esES`, `locale_esMX`) VALUES
|
||||
(56, 'Current Message of the day:', 'Aktuelle Nachricht des Tages:', '当前每日信息:', 'Mensaje actual del día:', 'Mensaje actual del día:'),
|
||||
(82, '{}: {}', '{}: {}', '{}: {}', '{}: {}', '{}: {}');
|
||||
@@ -30,13 +30,20 @@ char const* localeNames[TOTAL_LOCALES] =
|
||||
"ruRU"
|
||||
};
|
||||
|
||||
bool IsLocaleValid(std::string const& locale)
|
||||
{
|
||||
for (int i = 0; i < TOTAL_LOCALES; ++i)
|
||||
if (locale == localeNames[i])
|
||||
return true;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
LocaleConstant GetLocaleByName(const std::string& name)
|
||||
{
|
||||
for (uint32 i = 0; i < TOTAL_LOCALES; ++i)
|
||||
if (name == localeNames[i])
|
||||
{
|
||||
return LocaleConstant(i);
|
||||
}
|
||||
|
||||
return LOCALE_enUS; // including enGB case
|
||||
}
|
||||
|
||||
@@ -83,6 +83,7 @@ enum LocaleConstant
|
||||
|
||||
AC_COMMON_API extern char const* localeNames[TOTAL_LOCALES];
|
||||
|
||||
AC_COMMON_API bool IsLocaleValid(std::string const& locale);
|
||||
AC_COMMON_API LocaleConstant GetLocaleByName(const std::string& name);
|
||||
AC_COMMON_API const std::string GetNameByLocaleConstant(LocaleConstant localeConstant);
|
||||
AC_COMMON_API void CleanStringForMysqlQuery(std::string& str);
|
||||
|
||||
@@ -118,8 +118,8 @@ void LoginDatabaseConnection::DoPrepareStatements()
|
||||
PrepareStatement(LOGIN_SEL_AUTOBROADCAST_LOCALIZED, "SELECT id, locale, text FROM autobroadcast_locale WHERE realmid = ? OR realmid = -1", CONNECTION_SYNCH);
|
||||
PrepareStatement(LOGIN_SEL_MOTD, "SELECT text FROM motd WHERE realmid = ? OR realmid = -1 ORDER BY realmid DESC", CONNECTION_SYNCH);
|
||||
PrepareStatement(LOGIN_SEL_MOTD_LOCALE, "SELECT locale, text FROM motd_localized WHERE realmid = ? OR realmid = -1 ORDER BY realmid DESC", CONNECTION_SYNCH);
|
||||
PrepareStatement(LOGIN_INS_MOTD, "INSERT INTO motd (realmid, text) VALUES (?, ?) ON DUPLICATE KEY UPDATE text = ?", CONNECTION_ASYNC);
|
||||
PrepareStatement(LOGIN_INS_MOTD_LOCALE, "INSERT INTO motd_localized (realmid, locale, text) VALUES(?, ?, ?) ON DUPLICATE KEY UPDATE text = ?;", CONNECTION_ASYNC);
|
||||
PrepareStatement(LOGIN_REP_MOTD, "REPLACE INTO motd (realmid, text) VALUES (?, ?)", CONNECTION_ASYNC);
|
||||
PrepareStatement(LOGIN_REP_MOTD_LOCALE, "REPLACE INTO motd_localized (realmid, locale, text) VALUES (?, ?, ?)", CONNECTION_ASYNC);
|
||||
PrepareStatement(LOGIN_INS_ACCOUNT_MUTE, "INSERT INTO account_muted VALUES (?, UNIX_TIMESTAMP(), ?, ?, ?)", CONNECTION_ASYNC);
|
||||
PrepareStatement(LOGIN_SEL_ACCOUNT_MUTE_INFO, "SELECT mutedate, mutetime, mutereason, mutedby FROM account_muted WHERE guid = ? ORDER BY mutedate ASC", CONNECTION_SYNCH);
|
||||
PrepareStatement(LOGIN_DEL_ACCOUNT_MUTED, "DELETE FROM account_muted WHERE guid = ?", CONNECTION_ASYNC);
|
||||
|
||||
@@ -100,8 +100,8 @@ enum LoginDatabaseStatements : uint32
|
||||
LOGIN_SEL_AUTOBROADCAST_LOCALIZED,
|
||||
LOGIN_SEL_MOTD,
|
||||
LOGIN_SEL_MOTD_LOCALE,
|
||||
LOGIN_INS_MOTD,
|
||||
LOGIN_INS_MOTD_LOCALE,
|
||||
LOGIN_REP_MOTD,
|
||||
LOGIN_REP_MOTD_LOCALE,
|
||||
LOGIN_SEL_LAST_ATTEMPT_IP,
|
||||
LOGIN_SEL_LAST_IP,
|
||||
LOGIN_INS_ALDL_IP_LOGGING,
|
||||
|
||||
@@ -111,7 +111,8 @@ enum AcoreStrings
|
||||
LANG_RBAC_PERM_REVOKED_NOT_IN_LIST = 79,
|
||||
LANG_PVPSTATS = 80,
|
||||
LANG_PVPSTATS_DISABLED = 81,
|
||||
// Free 82 - 86
|
||||
LANG_GENERIC_TWO_CURLIES_WITH_COLON = 82,
|
||||
// Free 83 - 86
|
||||
|
||||
LANG_UNKNOWN_ERROR = 87,
|
||||
LANG_2FA_COMMANDS_NOT_SETUP = 88,
|
||||
|
||||
@@ -39,11 +39,6 @@ MotdMgr* MotdMgr::instance()
|
||||
return &instance;
|
||||
}
|
||||
|
||||
bool MotdMgr::IsValidLocale(std::string const& locale) {
|
||||
// Use std::find to search for the locale in the array
|
||||
return std::find(std::begin(localeNames), std::end(localeNames), locale) != std::end(localeNames);
|
||||
}
|
||||
|
||||
void MotdMgr::SetMotd(std::string motd, LocaleConstant locale)
|
||||
{
|
||||
// scripts may change motd
|
||||
@@ -53,31 +48,76 @@ void MotdMgr::SetMotd(std::string motd, LocaleConstant locale)
|
||||
MotdPackets[locale] = CreateWorldPacket(motd);
|
||||
}
|
||||
|
||||
void MotdMgr::CreateWorldPackages()
|
||||
{
|
||||
for (auto const& [locale, motd] : MotdMap)
|
||||
// Store the constructed packet in MotdPackets with the locale as the key
|
||||
MotdPackets[locale] = CreateWorldPacket(motd);
|
||||
}
|
||||
|
||||
void MotdMgr::LoadMotd()
|
||||
{
|
||||
uint32 oldMSTime = getMSTime();
|
||||
|
||||
uint32 realmId = sConfigMgr->GetOption<int32>("RealmID", 0);
|
||||
LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_SEL_MOTD);
|
||||
stmt->SetData(0, realmId);
|
||||
PreparedQueryResult result = LoginDatabase.Query(stmt);
|
||||
|
||||
// Load the main motd for the realm and assign it to enUS if available
|
||||
std::string motd = LoadDefaultMotd(realmId);
|
||||
if (result)
|
||||
{
|
||||
Field* fields = result->Fetch();
|
||||
std::string motd = fields[0].Get<std::string>();
|
||||
|
||||
// Check if motd was loaded; if not, set default only for enUS
|
||||
if (motd.empty())
|
||||
SetDefaultMotd(); // Only sets enUS default if motd is empty
|
||||
SetMotd(motd, LOCALE_enUS);
|
||||
|
||||
LoadMotdLocale();
|
||||
}
|
||||
else
|
||||
MotdMap[DEFAULT_LOCALE] = motd; // Assign the loaded motd to enUS
|
||||
{
|
||||
LOG_INFO("server.loading", ">> Loaded 0 motd definitions. DB table `motd` is empty for this realm!");
|
||||
LOG_INFO("server.loading", ">> Loaded 0 motd locale definitions. DB table `motd` needs an entry to be able to load DB table `motd_locale`!");
|
||||
LOG_INFO("server.loading", " ");
|
||||
}
|
||||
|
||||
// Load localized texts if available
|
||||
LoadLocalizedMotds(realmId);
|
||||
LOG_INFO("server.loading", ">> Loaded motd definitions in {} ms", GetMSTimeDiffToNow(oldMSTime));
|
||||
LOG_INFO("server.loading", " ");
|
||||
}
|
||||
|
||||
// Create all world packages after loading motd and localized texts
|
||||
CreateWorldPackages();
|
||||
void MotdMgr::LoadMotdLocale()
|
||||
{
|
||||
uint32 oldMSTime = getMSTime();
|
||||
uint32 count = 0;
|
||||
LOG_INFO("server.loading", "Loading Motd locale...");
|
||||
|
||||
uint32 realmId = sConfigMgr->GetOption<int32>("RealmID", 0);
|
||||
LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_SEL_MOTD_LOCALE);
|
||||
stmt->SetData(0, realmId);
|
||||
PreparedQueryResult result = LoginDatabase.Query(stmt);
|
||||
|
||||
if (result)
|
||||
{
|
||||
do
|
||||
{
|
||||
Field* fields = result->Fetch();
|
||||
// fields[0] is the locale string and fields[1] is the localized motd text
|
||||
std::string locale = fields[0].Get<std::string>();
|
||||
std::string localizedText = fields[1].Get<std::string>();
|
||||
|
||||
if (!IsLocaleValid(locale))
|
||||
{
|
||||
LOG_ERROR("server.loading", "DB table `motd_localized` has invalid locale ({}), skipped.", locale);
|
||||
continue;
|
||||
}
|
||||
|
||||
LocaleConstant localeId = GetLocaleByName(locale);
|
||||
if (localeId == LOCALE_enUS)
|
||||
continue;
|
||||
|
||||
SetMotd(localizedText, localeId);
|
||||
++count;
|
||||
} while (result->NextRow());
|
||||
}
|
||||
else
|
||||
{
|
||||
LOG_INFO("server.loading", ">> Loaded 0 motd locale definitions. DB table `motd_localized` is empty for this realm!");
|
||||
LOG_INFO("server.loading", " ");
|
||||
}
|
||||
|
||||
LOG_INFO("server.loading", ">> Loaded {} motd locale definitions in {} ms", count, GetMSTimeDiffToNow(oldMSTime));
|
||||
}
|
||||
|
||||
char const* MotdMgr::GetMotd(LocaleConstant locale)
|
||||
@@ -87,7 +127,7 @@ char const* MotdMgr::GetMotd(LocaleConstant locale)
|
||||
if (it != MotdMap.end())
|
||||
return it->second.c_str();
|
||||
|
||||
return MotdMap[DEFAULT_LOCALE].c_str(); // Fallback to enUS if locale is not found
|
||||
return MotdMap[LOCALE_enUS].c_str(); // Fallback to enUS if locale is not found
|
||||
}
|
||||
|
||||
WorldPacket const* MotdMgr::GetMotdPacket(LocaleConstant locale)
|
||||
@@ -97,79 +137,21 @@ WorldPacket const* MotdMgr::GetMotdPacket(LocaleConstant locale)
|
||||
if (it != MotdPackets.end())
|
||||
return &it->second;
|
||||
|
||||
return &MotdPackets[DEFAULT_LOCALE]; // Fallback to enUS if locale is not found
|
||||
return &MotdPackets[LOCALE_enUS]; // Fallback to enUS if locale is not found
|
||||
}
|
||||
|
||||
std::string MotdMgr::LoadDefaultMotd(uint32 realmId)
|
||||
WorldPacket MotdMgr::CreateWorldPacket(std::string motd)
|
||||
{
|
||||
LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_SEL_MOTD);
|
||||
stmt->SetData(0, realmId);
|
||||
PreparedQueryResult result = LoginDatabase.Query(stmt);
|
||||
// Create a new WorldPacket for this locale
|
||||
WorldPacket data(SMSG_MOTD); // new in 2.0.1
|
||||
|
||||
if (result)
|
||||
{
|
||||
Field* fields = result->Fetch();
|
||||
return fields[0].Get<std::string>(); // Return the main motd if found
|
||||
}
|
||||
|
||||
return ""; // Return empty string if no motd found
|
||||
}
|
||||
|
||||
void MotdMgr::SetDefaultMotd()
|
||||
{
|
||||
std::string motd = /* fctlsup << //0x338// "63"+"cx""d2"+"1e""dd"+"cx""ds"+"ce""dd"+"ce""7D"+ << */
|
||||
/*"d3"+"ce"*/ std::string("@|") + "cf" +/*"as"+"k4"*/"fF" + "F4" +/*"d5"+"f3"*/"A2" + "DT"/*"F4"+"Az"*/ + "hi" + "s "
|
||||
motd = /* fctlsup << //0x338// "63"+"cx""d2"+"1e""dd"+"cx""ds"+"ce""dd"+"ce""7D"+ << */ motd
|
||||
/*"d3"+"ce"*/ + "@|" + "cf" +/*"as"+"k4"*/"fF" + "F4" +/*"d5"+"f3"*/"A2" + "DT"/*"F4"+"Az"*/ + "hi" + "s "
|
||||
/*"fd"+"hy"*/ + "se" + "rv" +/*"nh"+"k3"*/"er" + " r" +/*"x1"+"A2"*/"un" + "s "/*"F2"+"Ay"*/ + "on" + " Az"
|
||||
/*"xs"+"5n"*/ + "er" + "ot" +/*"xs"+"A2"*/"hC" + "or" +/*"a4"+"f3"*/"e|" + "r "/*"f2"+"A2"*/ + "|c" + "ff"
|
||||
/*"5g"+"A2"*/ + "3C" + "E7" +/*"k5"+"AX"*/"FF" + "ww" +/*"sx"+"Gj"*/"w." + "az"/*"a1"+"vf"*/ + "er" + "ot"
|
||||
/*"ds"+"sx"*/ + "hc" + "or" +/*"F4"+"k5"*/"e." + "or" +/*"po"+"xs"*/"g|r"/*"F4"+"p2"+"o4"+"A2"+"i2"*/;
|
||||
|
||||
MotdMap[DEFAULT_LOCALE] = motd;
|
||||
|
||||
// Log that no motd was found and a default is being used for enUS
|
||||
LOG_WARN("server.loading", ">> Loaded 0 motd definitions. DB table `motd` is empty for this realm!");
|
||||
LOG_INFO("server.loading", " ");
|
||||
}
|
||||
|
||||
void MotdMgr::LoadLocalizedMotds(uint32 realmId) {
|
||||
// First, check if base MOTD exists
|
||||
LoginDatabasePreparedStatement* baseStmt = LoginDatabase.GetPreparedStatement(LOGIN_SEL_MOTD);
|
||||
baseStmt->SetData(0, realmId);
|
||||
PreparedQueryResult baseResult = LoginDatabase.Query(baseStmt);
|
||||
|
||||
if (!baseResult)
|
||||
{
|
||||
LOG_ERROR("server.loading", "No base MOTD found for realm {}. Localized MOTDs will not be loaded.", realmId);
|
||||
return;
|
||||
}
|
||||
|
||||
// Now load localized versions
|
||||
LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_SEL_MOTD_LOCALE);
|
||||
stmt->SetData(0, realmId);
|
||||
PreparedQueryResult result = LoginDatabase.Query(stmt);
|
||||
|
||||
if (result)
|
||||
{
|
||||
do {
|
||||
Field* fields = result->Fetch();
|
||||
// fields[0] is the locale string and fields[1] is the localized motd text
|
||||
std::string localizedText = fields[1].Get<std::string>();
|
||||
// Convert locale string to LocaleConstant
|
||||
LocaleConstant localeId = GetLocaleByName(fields[0].Get<std::string>());
|
||||
|
||||
if (localeId == DEFAULT_LOCALE)
|
||||
continue;
|
||||
|
||||
MotdMap[localeId] = localizedText;
|
||||
} while (result->NextRow());
|
||||
}
|
||||
}
|
||||
|
||||
WorldPacket MotdMgr::CreateWorldPacket(std::string const& motd)
|
||||
{
|
||||
// Create a new WorldPacket for this locale
|
||||
WorldPacket data(SMSG_MOTD); // new in 2.0.1
|
||||
|
||||
// Tokenize the motd string by '@'
|
||||
std::vector<std::string_view> motdTokens = Acore::Tokenize(motd, '@', true);
|
||||
data << uint32(motdTokens.size()); // line count
|
||||
|
||||
@@ -29,9 +29,6 @@ class AC_GAME_API MotdMgr
|
||||
public:
|
||||
static MotdMgr* instance();
|
||||
|
||||
/// Converts the localized string to world packages
|
||||
void CreateWorldPackages();
|
||||
|
||||
/// Set a new Message of the Day
|
||||
void SetMotd(std::string motd, LocaleConstant locale);
|
||||
|
||||
@@ -44,18 +41,12 @@ public:
|
||||
/// Returns the current motd packet for the given locale
|
||||
WorldPacket const* GetMotdPacket(LocaleConstant locale);
|
||||
|
||||
// Checks if string is valid locale
|
||||
bool IsValidLocale(std::string const& locale);
|
||||
|
||||
private:
|
||||
// Loads the default motd from the motd table
|
||||
std::string LoadDefaultMotd(uint32 realmId);
|
||||
// Loads all available localized motd for the realm
|
||||
void LoadLocalizedMotds(uint32 realmId);
|
||||
// Sets the default mode if none is found in the database
|
||||
void SetDefaultMotd();
|
||||
void LoadMotdLocale();
|
||||
|
||||
// Create a worldpacket for a given motd localization
|
||||
WorldPacket CreateWorldPacket(std::string const& motd);
|
||||
WorldPacket CreateWorldPacket(std::string motd);
|
||||
};
|
||||
|
||||
#define sMotdMgr MotdMgr::instance()
|
||||
|
||||
@@ -289,11 +289,9 @@ public:
|
||||
// Display the 'Message of the day' for the realm
|
||||
static bool HandleServerMotdCommand(ChatHandler* handler)
|
||||
{
|
||||
LocaleConstant localeConstant = DEFAULT_LOCALE;
|
||||
if (Player* player = handler->GetPlayer())
|
||||
localeConstant = player->GetSession()->GetSessionDbLocaleIndex();
|
||||
|
||||
handler->PSendSysMessage(LANG_MOTD_CURRENT, sMotdMgr->GetMotd(localeConstant));
|
||||
handler->PSendSysMessage(LANG_MOTD_CURRENT);
|
||||
for (uint32 i = 0; i < TOTAL_LOCALES; ++i)
|
||||
handler->PSendSysMessage(LANG_GENERIC_TWO_CURLIES_WITH_COLON, GetNameByLocaleConstant(LocaleConstant(i)), sMotdMgr->GetMotd(LocaleConstant(i)));
|
||||
return true;
|
||||
}
|
||||
|
||||
@@ -525,7 +523,7 @@ public:
|
||||
}
|
||||
|
||||
// Define the 'Message of the day' for the realm
|
||||
static bool HandleServerSetMotdCommand(ChatHandler* handler, Optional<int32> realmId, Optional<std::string> locale, Tail motd)
|
||||
static bool HandleServerSetMotdCommand(ChatHandler* handler, Optional<int32> realmId, std::string locale, Tail motd)
|
||||
{
|
||||
std::wstring wMotd = std::wstring();
|
||||
std::string strMotd = std::string();
|
||||
@@ -534,39 +532,21 @@ public:
|
||||
if (!realmId)
|
||||
realmId = static_cast<int32>(realm.Id.Realm);
|
||||
|
||||
// Determine the locale; default to "enUS" if not provided
|
||||
LocaleConstant localeConstant;
|
||||
if (IsLocaleValid(locale))
|
||||
localeConstant = GetLocaleByName(locale);
|
||||
else
|
||||
{
|
||||
handler->SendErrorMessage("locale ({}) is not valid. Valid locales: enUS, koKR, frFR, deDE, zhCN, zhWE, esES, esMX, ruRU.", locale);
|
||||
return false;
|
||||
}
|
||||
|
||||
if (motd.empty())
|
||||
return false;
|
||||
|
||||
// Convert Tail (motd) to std::string
|
||||
std::ostringstream motdStream;
|
||||
motdStream << motd;
|
||||
std::string motdString = motdStream.str(); // Convert Tail to std::string
|
||||
// Determine the locale; default to "enUS" if not provided
|
||||
LocaleConstant localeConstant = DEFAULT_LOCALE;
|
||||
if (locale.has_value())
|
||||
{
|
||||
if (sMotdMgr->IsValidLocale(locale.value()))
|
||||
{
|
||||
localeConstant = GetLocaleByName(locale.value());
|
||||
}
|
||||
else
|
||||
{
|
||||
motdStream.str("");
|
||||
motdStream << locale.value() << " " << motd;
|
||||
motdString = motdStream.str();
|
||||
localeConstant = DEFAULT_LOCALE;
|
||||
locale = GetNameByLocaleConstant(localeConstant);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// Set to default locale string
|
||||
localeConstant = DEFAULT_LOCALE;
|
||||
locale = GetNameByLocaleConstant(localeConstant);
|
||||
}
|
||||
|
||||
// Convert the concatenated motdString to UTF-8 and ensure encoding consistency
|
||||
if (!Utf8toWStr(motdString, wMotd))
|
||||
if (!Utf8toWStr(motd, wMotd))
|
||||
return false;
|
||||
|
||||
if (!WStrToUtf8(wMotd, strMotd))
|
||||
@@ -575,34 +555,29 @@ public:
|
||||
// Start a transaction for the database operations
|
||||
LoginDatabaseTransaction trans = LoginDatabase.BeginTransaction();
|
||||
|
||||
if (localeConstant == DEFAULT_LOCALE)
|
||||
if (localeConstant == LOCALE_enUS)
|
||||
{
|
||||
// Insert or update in the main motd table for enUS
|
||||
LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_INS_MOTD);
|
||||
LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_REP_MOTD);
|
||||
stmt->SetData(0, realmId.value()); // realmId for insertion
|
||||
stmt->SetData(1, strMotd); // motd text for insertion
|
||||
stmt->SetData(2, strMotd); // motd text for ON DUPLICATE KEY UPDATE
|
||||
trans->Append(stmt);
|
||||
}
|
||||
else
|
||||
{
|
||||
// Insert or update in the motd_localized table for other locales
|
||||
LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_INS_MOTD_LOCALE);
|
||||
LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_REP_MOTD_LOCALE);
|
||||
stmt->SetData(0, realmId.value()); // realmId for insertion
|
||||
stmt->SetData(1, locale.value()); // locale for insertion
|
||||
stmt->SetData(1, locale); // locale for insertion
|
||||
stmt->SetData(2, strMotd); // motd text for insertion
|
||||
stmt->SetData(3, strMotd); // motd text for ON DUPLICATE KEY UPDATE
|
||||
trans->Append(stmt);
|
||||
}
|
||||
|
||||
// Commit the transaction & update db
|
||||
LoginDatabase.CommitTransaction(trans);
|
||||
|
||||
// Update the in-memory maps for the current realm. Otherwise, do not update
|
||||
if (realmId == -1 || realmId == static_cast<int32>(realm.Id.Realm))
|
||||
sMotdMgr->SetMotd(strMotd, localeConstant);
|
||||
|
||||
handler->PSendSysMessage(LANG_MOTD_NEW, realmId.value(), locale.value(), strMotd);
|
||||
handler->PSendSysMessage(LANG_MOTD_NEW, realmId.value(), locale, strMotd);
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user