/* * This file is part of the AzerothCore 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 Affero General Public License as published by the * Free Software Foundation; either version 3 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 Affero 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 . */ /** \file \ingroup world */ #include "World.h" #include "AccountMgr.h" #include "AchievementMgr.h" #include "AddonMgr.h" #include "ArenaTeamMgr.h" #include "AsyncAuctionListing.h" #include "AuctionHouseMgr.h" #include "AutobroadcastMgr.h" #include "BattlefieldMgr.h" #include "BattlegroundMgr.h" #include "CalendarMgr.h" #include "Channel.h" #include "ChannelMgr.h" #include "CharacterDatabaseCleaner.h" #include "Chat.h" #include "ChatPackets.h" #include "Common.h" #include "ConditionMgr.h" #include "Config.h" #include "CreatureAIRegistry.h" #include "CreatureGroups.h" #include "CreatureTextMgr.h" #include "DBCStores.h" #include "DatabaseEnv.h" #include "DisableMgr.h" #include "DynamicVisibility.h" #include "GameEventMgr.h" #include "GameGraveyard.h" #include "GameTime.h" #include "GitRevision.h" #include "GridNotifiersImpl.h" #include "GroupMgr.h" #include "GuildMgr.h" #include "IPLocation.h" #include "InstanceSaveMgr.h" #include "ItemEnchantmentMgr.h" #include "LFGMgr.h" #include "Log.h" #include "LootItemStorage.h" #include "LootMgr.h" #include "MMapFactory.h" #include "MapMgr.h" #include "Metric.h" #include "M2Stores.h" #include "ObjectMgr.h" #include "Opcodes.h" #include "OutdoorPvPMgr.h" #include "PetitionMgr.h" #include "Player.h" #include "PlayerDump.h" #include "PoolMgr.h" #include "Realm.h" #include "ScriptMgr.h" #include "MotdMgr.h" #include "SkillDiscovery.h" #include "SkillExtraItems.h" #include "SmartAI.h" #include "SpellMgr.h" #include "TaskScheduler.h" #include "TicketMgr.h" #include "Transport.h" #include "TransportMgr.h" #include "UpdateTime.h" #include "Util.h" #include "VMapFactory.h" #include "VMapMgr2.h" #include "Vehicle.h" #include "Warden.h" #include "WardenCheckMgr.h" #include "WaypointMovementGenerator.h" #include "WeatherMgr.h" #include "WhoListCacheMgr.h" #include "WorldPacket.h" #include "WorldSession.h" #include #include namespace { TaskScheduler playersSaveScheduler; } std::atomic_long World::_stopEvent = false; uint8 World::_exitCode = SHUTDOWN_EXIT_CODE; uint32 World::m_worldLoopCounter = 0; float World::_maxVisibleDistanceOnContinents = DEFAULT_VISIBILITY_DISTANCE; float World::_maxVisibleDistanceInInstances = DEFAULT_VISIBILITY_INSTANCE; float World::_maxVisibleDistanceInBGArenas = DEFAULT_VISIBILITY_BGARENAS; int32 World::m_visibility_notify_periodOnContinents = DEFAULT_VISIBILITY_NOTIFY_PERIOD; int32 World::m_visibility_notify_periodInInstances = DEFAULT_VISIBILITY_NOTIFY_PERIOD; int32 World::m_visibility_notify_periodInBGArenas = DEFAULT_VISIBILITY_NOTIFY_PERIOD; Realm realm; /// World constructor World::World() { _playerLimit = 0; _allowedSecurityLevel = SEC_PLAYER; _allowMovement = true; _shutdownMask = 0; _shutdownTimer = 0; _maxActiveSessionCount = 0; _maxQueuedSessionCount = 0; _playerCount = 0; _maxPlayerCount = 0; _nextDailyQuestReset = 0s; _nextWeeklyQuestReset = 0s; _nextMonthlyQuestReset = 0s; _nextRandomBGReset = 0s; _nextCalendarOldEventsDeletionTime = 0s; _nextGuildReset = 0s; _defaultDbcLocale = LOCALE_enUS; _mail_expire_check_timer = 0s; _isClosed = false; _cleaningFlags = 0; memset(_rate_values, 0, sizeof(_rate_values)); memset(_int_configs, 0, sizeof(_int_configs)); memset(_bool_configs, 0, sizeof(_bool_configs)); memset(_float_configs, 0, sizeof(_float_configs)); } /// World destructor World::~World() { ///- Empty the kicked session set while (!_sessions.empty()) { // not remove from queue, prevent loading new sessions delete _sessions.begin()->second; _sessions.erase(_sessions.begin()); } while (!_offlineSessions.empty()) { delete _offlineSessions.begin()->second; _offlineSessions.erase(_offlineSessions.begin()); } CliCommandHolder* command = nullptr; while (_cliCmdQueue.next(command)) delete command; VMAP::VMapFactory::clear(); MMAP::MMapFactory::clear(); //TODO free addSessQueue } std::unique_ptr& getWorldInstance() { static std::unique_ptr instance = std::make_unique(); return instance; } /// Find a player in a specified zone Player* World::FindPlayerInZone(uint32 zone) { ///- circle through active sessions and return the first player found in the zone SessionMap::const_iterator itr; for (itr = _sessions.begin(); itr != _sessions.end(); ++itr) { if (!itr->second) continue; Player* player = itr->second->GetPlayer(); if (!player) continue; if (player->IsInWorld() && player->GetZoneId() == zone) return player; } return nullptr; } bool World::IsClosed() const { return _isClosed; } void World::SetClosed(bool val) { _isClosed = val; // Invert the value, for simplicity for scripters. sScriptMgr->OnOpenStateChange(!val); } /// Find a session by its id WorldSession* World::FindSession(uint32 id) const { SessionMap::const_iterator itr = _sessions.find(id); if (itr != _sessions.end()) return itr->second; // also can return nullptr for kicked session else return nullptr; } WorldSession* World::FindOfflineSession(uint32 id) const { SessionMap::const_iterator itr = _offlineSessions.find(id); if (itr != _offlineSessions.end()) return itr->second; else return nullptr; } WorldSession* World::FindOfflineSessionForCharacterGUID(ObjectGuid::LowType guidLow) const { if (_offlineSessions.empty()) return nullptr; for (SessionMap::const_iterator itr = _offlineSessions.begin(); itr != _offlineSessions.end(); ++itr) if (itr->second->GetGuidLow() == guidLow) return itr->second; return nullptr; } /// Remove a given session bool World::KickSession(uint32 id) { ///- Find the session, kick the user, but we can't delete session at this moment to prevent iterator invalidation SessionMap::const_iterator itr = _sessions.find(id); if (itr != _sessions.end() && itr->second) { if (itr->second->PlayerLoading()) return false; itr->second->KickPlayer("KickSession", false); } return true; } void World::AddSession(WorldSession* s) { _addSessQueue.add(s); } void World::AddSession_(WorldSession* s) { ASSERT (s); // kick existing session with same account (if any) // if character on old session is being loaded, then return if (!KickSession(s->GetAccountId())) { s->KickPlayer("kick existing session with same account"); delete s; // session not added yet in session list, so not listed in queue return; } SessionMap::const_iterator old = _sessions.find(s->GetAccountId()); if (old != _sessions.end()) { WorldSession* oldSession = old->second; if (!RemoveQueuedPlayer(oldSession) && getIntConfig(CONFIG_INTERVAL_DISCONNECT_TOLERANCE)) _disconnects[s->GetAccountId()] = GameTime::GetGameTime().count(); // pussywizard: if (oldSession->HandleSocketClosed()) { // there should be no offline session if current one is logged onto a character SessionMap::iterator iter; if ((iter = _offlineSessions.find(oldSession->GetAccountId())) != _offlineSessions.end()) { WorldSession* tmp = iter->second; _offlineSessions.erase(iter); delete tmp; } oldSession->SetOfflineTime(GameTime::GetGameTime().count()); _offlineSessions[oldSession->GetAccountId()] = oldSession; } else { delete oldSession; } } _sessions[s->GetAccountId()] = s; uint32 Sessions = GetActiveAndQueuedSessionCount(); uint32 pLimit = GetPlayerAmountLimit(); // don't count this session when checking player limit --Sessions; if (pLimit > 0 && Sessions >= pLimit && AccountMgr::IsPlayerAccount(s->GetSecurity()) && !s->CanSkipQueue() && !HasRecentlyDisconnected(s)) { AddQueuedPlayer(s); UpdateMaxSessionCounters(); return; } s->InitializeSession(); UpdateMaxSessionCounters(); } bool World::HasRecentlyDisconnected(WorldSession* session) { if (!session) return false; if (uint32 tolerance = getIntConfig(CONFIG_INTERVAL_DISCONNECT_TOLERANCE)) { for (DisconnectMap::iterator i = _disconnects.begin(); i != _disconnects.end();) { if ((GameTime::GetGameTime().count() - i->second) < tolerance) { if (i->first == session->GetAccountId()) return true; ++i; } else _disconnects.erase(i++); } } return false; } int32 World::GetQueuePos(WorldSession* sess) { uint32 position = 1; for (Queue::const_iterator iter = _queuedPlayer.begin(); iter != _queuedPlayer.end(); ++iter, ++position) if ((*iter) == sess) return position; return 0; } void World::AddQueuedPlayer(WorldSession* sess) { sess->SetInQueue(true); _queuedPlayer.push_back(sess); // The 1st SMSG_AUTH_RESPONSE needs to contain other info too. sess->SendAuthResponse(AUTH_WAIT_QUEUE, false, GetQueuePos(sess)); } bool World::RemoveQueuedPlayer(WorldSession* sess) { uint32 sessions = GetActiveSessionCount(); uint32 position = 1; Queue::iterator iter = _queuedPlayer.begin(); // search to remove and count skipped positions bool found = false; for (; iter != _queuedPlayer.end(); ++iter, ++position) { if (*iter == sess) { sess->SetInQueue(false); sess->ResetTimeOutTime(false); iter = _queuedPlayer.erase(iter); found = true; break; } } // if session not queued then it was an active session if (!found) { ASSERT(sessions > 0); --sessions; } // accept first in queue if ((!GetPlayerAmountLimit() || sessions < GetPlayerAmountLimit()) && !_queuedPlayer.empty()) { WorldSession* pop_sess = _queuedPlayer.front(); pop_sess->InitializeSession(); _queuedPlayer.pop_front(); // update iter to point first queued socket or end() if queue is empty now iter = _queuedPlayer.begin(); position = 1; } // update queue position from iter to end() for (; iter != _queuedPlayer.end(); ++iter, ++position) (*iter)->SendAuthWaitQueue(position); return found; } /// Initialize config values void World::LoadConfigSettings(bool reload) { if (reload) { if (!sConfigMgr->Reload()) { LOG_ERROR("server.loading", "World settings reload fail: can't read settings."); return; } sLog->LoadFromConfig(); sMetric->LoadFromConfigs(); } // Set realm id and enable db logging sLog->SetRealmId(realm.Id.Realm); sScriptMgr->OnBeforeConfigLoad(reload); // load update time related configs sWorldUpdateTime.LoadFromConfig(); ///- Read the player limit and the Message of the day from the config file if (!reload) { SetPlayerAmountLimit(sConfigMgr->GetOption("PlayerLimit", 1000)); } ///- Read ticket system setting from the config file _bool_configs[CONFIG_ALLOW_TICKETS] = sConfigMgr->GetOption("AllowTickets", true); _bool_configs[CONFIG_DELETE_CHARACTER_TICKET_TRACE] = sConfigMgr->GetOption("DeletedCharacterTicketTrace", false); ///- Get string for new logins (newly created characters) SetNewCharString(sConfigMgr->GetOption("PlayerStart.String", "")); ///- Send server info on login? _int_configs[CONFIG_ENABLE_SINFO_LOGIN] = sConfigMgr->GetOption("Server.LoginInfo", 0); ///- Read all rates from the config file _rate_values[RATE_HEALTH] = sConfigMgr->GetOption("Rate.Health", 1); if (_rate_values[RATE_HEALTH] < 0) { LOG_ERROR("server.loading", "Rate.Health ({}) must be > 0. Using 1 instead.", _rate_values[RATE_HEALTH]); _rate_values[RATE_HEALTH] = 1; } _rate_values[RATE_POWER_MANA] = sConfigMgr->GetOption("Rate.Mana", 1); if (_rate_values[RATE_POWER_MANA] < 0) { LOG_ERROR("server.loading", "Rate.Mana ({}) must be > 0. Using 1 instead.", _rate_values[RATE_POWER_MANA]); _rate_values[RATE_POWER_MANA] = 1; } _rate_values[RATE_POWER_RAGE_INCOME] = sConfigMgr->GetOption("Rate.Rage.Income", 1); _rate_values[RATE_POWER_RAGE_LOSS] = sConfigMgr->GetOption("Rate.Rage.Loss", 1); if (_rate_values[RATE_POWER_RAGE_LOSS] < 0) { LOG_ERROR("server.loading", "Rate.Rage.Loss ({}) must be > 0. Using 1 instead.", _rate_values[RATE_POWER_RAGE_LOSS]); _rate_values[RATE_POWER_RAGE_LOSS] = 1; } _rate_values[RATE_POWER_RUNICPOWER_INCOME] = sConfigMgr->GetOption("Rate.RunicPower.Income", 1); _rate_values[RATE_POWER_RUNICPOWER_LOSS] = sConfigMgr->GetOption("Rate.RunicPower.Loss", 1); if (_rate_values[RATE_POWER_RUNICPOWER_LOSS] < 0) { LOG_ERROR("server.loading", "Rate.RunicPower.Loss ({}) must be > 0. Using 1 instead.", _rate_values[RATE_POWER_RUNICPOWER_LOSS]); _rate_values[RATE_POWER_RUNICPOWER_LOSS] = 1; } _rate_values[RATE_POWER_FOCUS] = sConfigMgr->GetOption("Rate.Focus", 1.0f); _rate_values[RATE_POWER_ENERGY] = sConfigMgr->GetOption("Rate.Energy", 1.0f); _rate_values[RATE_SKILL_DISCOVERY] = sConfigMgr->GetOption("Rate.Skill.Discovery", 1.0f); _rate_values[RATE_DROP_ITEM_POOR] = sConfigMgr->GetOption("Rate.Drop.Item.Poor", 1.0f); _rate_values[RATE_DROP_ITEM_NORMAL] = sConfigMgr->GetOption("Rate.Drop.Item.Normal", 1.0f); _rate_values[RATE_DROP_ITEM_UNCOMMON] = sConfigMgr->GetOption("Rate.Drop.Item.Uncommon", 1.0f); _rate_values[RATE_DROP_ITEM_RARE] = sConfigMgr->GetOption("Rate.Drop.Item.Rare", 1.0f); _rate_values[RATE_DROP_ITEM_EPIC] = sConfigMgr->GetOption("Rate.Drop.Item.Epic", 1.0f); _rate_values[RATE_DROP_ITEM_LEGENDARY] = sConfigMgr->GetOption("Rate.Drop.Item.Legendary", 1.0f); _rate_values[RATE_DROP_ITEM_ARTIFACT] = sConfigMgr->GetOption("Rate.Drop.Item.Artifact", 1.0f); _rate_values[RATE_DROP_ITEM_REFERENCED] = sConfigMgr->GetOption("Rate.Drop.Item.Referenced", 1.0f); _rate_values[RATE_DROP_ITEM_REFERENCED_AMOUNT] = sConfigMgr->GetOption("Rate.Drop.Item.ReferencedAmount", 1.0f); _rate_values[RATE_DROP_ITEM_GROUP_AMOUNT] = sConfigMgr->GetOption("Rate.Drop.Item.GroupAmount", 1.0f); _rate_values[RATE_DROP_MONEY] = sConfigMgr->GetOption("Rate.Drop.Money", 1.0f); _rate_values[RATE_REWARD_BONUS_MONEY] = sConfigMgr->GetOption("Rate.RewardBonusMoney", 1.0f); _rate_values[RATE_XP_KILL] = sConfigMgr->GetOption("Rate.XP.Kill", 1.0f); _rate_values[RATE_XP_BG_KILL_AV] = sConfigMgr->GetOption("Rate.XP.BattlegroundKillAV", 1.0f); _rate_values[RATE_XP_BG_KILL_WSG] = sConfigMgr->GetOption("Rate.XP.BattlegroundKillWSG", 1.0f); _rate_values[RATE_XP_BG_KILL_AB] = sConfigMgr->GetOption("Rate.XP.BattlegroundKillAB", 1.0f); _rate_values[RATE_XP_BG_KILL_EOTS] = sConfigMgr->GetOption("Rate.XP.BattlegroundKillEOTS", 1.0f); _rate_values[RATE_XP_BG_KILL_SOTA] = sConfigMgr->GetOption("Rate.XP.BattlegroundKillSOTA", 1.0f); _rate_values[RATE_XP_BG_KILL_IC] = sConfigMgr->GetOption("Rate.XP.BattlegroundKillIC", 1.0f); _rate_values[RATE_XP_QUEST] = sConfigMgr->GetOption("Rate.XP.Quest", 1.0f); _rate_values[RATE_XP_QUEST_DF] = sConfigMgr->GetOption("Rate.XP.Quest.DF", 1.0f); _rate_values[RATE_XP_EXPLORE] = sConfigMgr->GetOption("Rate.XP.Explore", 1.0f); _rate_values[RATE_XP_PET] = sConfigMgr->GetOption("Rate.XP.Pet", 1.0f); _rate_values[RATE_XP_PET_NEXT_LEVEL] = sConfigMgr->GetOption("Rate.Pet.LevelXP", 0.05f); _rate_values[RATE_REPAIRCOST] = sConfigMgr->GetOption("Rate.RepairCost", 1.0f); _rate_values[RATE_SELLVALUE_ITEM_POOR] = sConfigMgr->GetOption("Rate.SellValue.Item.Poor", 1.0f); _rate_values[RATE_SELLVALUE_ITEM_NORMAL] = sConfigMgr->GetOption("Rate.SellValue.Item.Normal", 1.0f); _rate_values[RATE_SELLVALUE_ITEM_UNCOMMON] = sConfigMgr->GetOption("Rate.SellValue.Item.Uncommon", 1.0f); _rate_values[RATE_SELLVALUE_ITEM_RARE] = sConfigMgr->GetOption("Rate.SellValue.Item.Rare", 1.0f); _rate_values[RATE_SELLVALUE_ITEM_EPIC] = sConfigMgr->GetOption("Rate.SellValue.Item.Epic", 1.0f); _rate_values[RATE_SELLVALUE_ITEM_LEGENDARY] = sConfigMgr->GetOption("Rate.SellValue.Item.Legendary", 1.0f); _rate_values[RATE_SELLVALUE_ITEM_ARTIFACT] = sConfigMgr->GetOption("Rate.SellValue.Item.Artifact", 1.0f); _rate_values[RATE_SELLVALUE_ITEM_HEIRLOOM] = sConfigMgr->GetOption("Rate.SellValue.Item.Heirloom", 1.0f); _rate_values[ RATE_BUYVALUE_ITEM_POOR] = sConfigMgr->GetOption("Rate.BuyValue.Item.Poor", 1.0f); _rate_values[ RATE_BUYVALUE_ITEM_NORMAL] = sConfigMgr->GetOption("Rate.BuyValue.Item.Normal", 1.0f); _rate_values[ RATE_BUYVALUE_ITEM_UNCOMMON] = sConfigMgr->GetOption("Rate.BuyValue.Item.Uncommon", 1.0f); _rate_values[ RATE_BUYVALUE_ITEM_RARE] = sConfigMgr->GetOption("Rate.BuyValue.Item.Rare", 1.0f); _rate_values[ RATE_BUYVALUE_ITEM_EPIC] = sConfigMgr->GetOption("Rate.BuyValue.Item.Epic", 1.0f); _rate_values[ RATE_BUYVALUE_ITEM_LEGENDARY] = sConfigMgr->GetOption("Rate.BuyValue.Item.Legendary", 1.0f); _rate_values[RATE_BUYVALUE_ITEM_ARTIFACT] = sConfigMgr->GetOption("Rate.BuyValue.Item.Artifact", 1.0f); _rate_values[RATE_BUYVALUE_ITEM_HEIRLOOM] = sConfigMgr->GetOption("Rate.BuyValue.Item.Heirloom", 1.0f); if (_rate_values[RATE_REPAIRCOST] < 0.0f) { LOG_ERROR("server.loading", "Rate.RepairCost ({}) must be >=0. Using 0.0 instead.", _rate_values[RATE_REPAIRCOST]); _rate_values[RATE_REPAIRCOST] = 0.0f; } _rate_values[RATE_REPUTATION_GAIN] = sConfigMgr->GetOption("Rate.Reputation.Gain", 1.0f); _rate_values[RATE_REPUTATION_LOWLEVEL_KILL] = sConfigMgr->GetOption("Rate.Reputation.LowLevel.Kill", 1.0f); _rate_values[RATE_REPUTATION_LOWLEVEL_QUEST] = sConfigMgr->GetOption("Rate.Reputation.LowLevel.Quest", 1.0f); _rate_values[RATE_REPUTATION_RECRUIT_A_FRIEND_BONUS] = sConfigMgr->GetOption("Rate.Reputation.RecruitAFriendBonus", 0.1f); _rate_values[RATE_CREATURE_NORMAL_DAMAGE] = sConfigMgr->GetOption("Rate.Creature.Normal.Damage", 1.0f); _rate_values[RATE_CREATURE_ELITE_ELITE_DAMAGE] = sConfigMgr->GetOption("Rate.Creature.Elite.Elite.Damage", 1.0f); _rate_values[RATE_CREATURE_ELITE_RAREELITE_DAMAGE] = sConfigMgr->GetOption("Rate.Creature.Elite.RAREELITE.Damage", 1.0f); _rate_values[RATE_CREATURE_ELITE_WORLDBOSS_DAMAGE] = sConfigMgr->GetOption("Rate.Creature.Elite.WORLDBOSS.Damage", 1.0f); _rate_values[RATE_CREATURE_ELITE_RARE_DAMAGE] = sConfigMgr->GetOption("Rate.Creature.Elite.RARE.Damage", 1.0f); _rate_values[RATE_CREATURE_NORMAL_HP] = sConfigMgr->GetOption("Rate.Creature.Normal.HP", 1.0f); _rate_values[RATE_CREATURE_ELITE_ELITE_HP] = sConfigMgr->GetOption("Rate.Creature.Elite.Elite.HP", 1.0f); _rate_values[RATE_CREATURE_ELITE_RAREELITE_HP] = sConfigMgr->GetOption("Rate.Creature.Elite.RAREELITE.HP", 1.0f); _rate_values[RATE_CREATURE_ELITE_WORLDBOSS_HP] = sConfigMgr->GetOption("Rate.Creature.Elite.WORLDBOSS.HP", 1.0f); _rate_values[RATE_CREATURE_ELITE_RARE_HP] = sConfigMgr->GetOption("Rate.Creature.Elite.RARE.HP", 1.0f); _rate_values[RATE_CREATURE_NORMAL_SPELLDAMAGE] = sConfigMgr->GetOption("Rate.Creature.Normal.SpellDamage", 1.0f); _rate_values[RATE_CREATURE_ELITE_ELITE_SPELLDAMAGE] = sConfigMgr->GetOption("Rate.Creature.Elite.Elite.SpellDamage", 1.0f); _rate_values[RATE_CREATURE_ELITE_RAREELITE_SPELLDAMAGE] = sConfigMgr->GetOption("Rate.Creature.Elite.RAREELITE.SpellDamage", 1.0f); _rate_values[RATE_CREATURE_ELITE_WORLDBOSS_SPELLDAMAGE] = sConfigMgr->GetOption("Rate.Creature.Elite.WORLDBOSS.SpellDamage", 1.0f); _rate_values[RATE_CREATURE_ELITE_RARE_SPELLDAMAGE] = sConfigMgr->GetOption("Rate.Creature.Elite.RARE.SpellDamage", 1.0f); _rate_values[RATE_CREATURE_AGGRO] = sConfigMgr->GetOption("Rate.Creature.Aggro", 1.0f); _rate_values[RATE_REST_INGAME] = sConfigMgr->GetOption("Rate.Rest.InGame", 1.0f); _rate_values[RATE_REST_OFFLINE_IN_TAVERN_OR_CITY] = sConfigMgr->GetOption("Rate.Rest.Offline.InTavernOrCity", 1.0f); _rate_values[RATE_REST_OFFLINE_IN_WILDERNESS] = sConfigMgr->GetOption("Rate.Rest.Offline.InWilderness", 1.0f); _rate_values[RATE_DAMAGE_FALL] = sConfigMgr->GetOption("Rate.Damage.Fall", 1.0f); _rate_values[RATE_AUCTION_TIME] = sConfigMgr->GetOption("Rate.Auction.Time", 1.0f); _rate_values[RATE_AUCTION_DEPOSIT] = sConfigMgr->GetOption("Rate.Auction.Deposit", 1.0f); _rate_values[RATE_AUCTION_CUT] = sConfigMgr->GetOption("Rate.Auction.Cut", 1.0f); _rate_values[RATE_HONOR] = sConfigMgr->GetOption("Rate.Honor", 1.0f); _rate_values[RATE_ARENA_POINTS] = sConfigMgr->GetOption("Rate.ArenaPoints", 1.0f); _rate_values[RATE_INSTANCE_RESET_TIME] = sConfigMgr->GetOption("Rate.InstanceResetTime", 1.0f); _rate_values[RATE_MISS_CHANCE_MULTIPLIER_TARGET_CREATURE] = sConfigMgr->GetOption("Rate.MissChanceMultiplier.TargetCreature", 11.0f); _rate_values[RATE_MISS_CHANCE_MULTIPLIER_TARGET_PLAYER] = sConfigMgr->GetOption("Rate.MissChanceMultiplier.TargetPlayer", 7.0f); _bool_configs[CONFIG_MISS_CHANCE_MULTIPLIER_ONLY_FOR_PLAYERS] = sConfigMgr->GetOption("Rate.MissChanceMultiplier.OnlyAffectsPlayer", false); _rate_values[RATE_TALENT] = sConfigMgr->GetOption("Rate.Talent", 1.0f); if (_rate_values[RATE_TALENT] < 0.0f) { LOG_ERROR("server.loading", "Rate.Talent ({}) must be > 0. Using 1 instead.", _rate_values[RATE_TALENT]); _rate_values[RATE_TALENT] = 1.0f; } _rate_values[RATE_MOVESPEED] = sConfigMgr->GetOption("Rate.MoveSpeed", 1.0f); if (_rate_values[RATE_MOVESPEED] < 0) { LOG_ERROR("server.loading", "Rate.MoveSpeed ({}) must be > 0. Using 1 instead.", _rate_values[RATE_MOVESPEED]); _rate_values[RATE_MOVESPEED] = 1.0f; } for (uint8 i = 0; i < MAX_MOVE_TYPE; ++i) playerBaseMoveSpeed[i] = baseMoveSpeed[i] * _rate_values[RATE_MOVESPEED]; _rate_values[RATE_CORPSE_DECAY_LOOTED] = sConfigMgr->GetOption("Rate.Corpse.Decay.Looted", 0.5f); _rate_values[RATE_TARGET_POS_RECALCULATION_RANGE] = sConfigMgr->GetOption("TargetPosRecalculateRange", 1.5f); if (_rate_values[RATE_TARGET_POS_RECALCULATION_RANGE] < CONTACT_DISTANCE) { LOG_ERROR("server.loading", "TargetPosRecalculateRange ({}) must be >= {}. Using {} instead.", _rate_values[RATE_TARGET_POS_RECALCULATION_RANGE], CONTACT_DISTANCE, CONTACT_DISTANCE); _rate_values[RATE_TARGET_POS_RECALCULATION_RANGE] = CONTACT_DISTANCE; } else if (_rate_values[RATE_TARGET_POS_RECALCULATION_RANGE] > NOMINAL_MELEE_RANGE) { LOG_ERROR("server.loading", "TargetPosRecalculateRange ({}) must be <= {}. Using {} instead.", _rate_values[RATE_TARGET_POS_RECALCULATION_RANGE], NOMINAL_MELEE_RANGE, NOMINAL_MELEE_RANGE); _rate_values[RATE_TARGET_POS_RECALCULATION_RANGE] = NOMINAL_MELEE_RANGE; } _rate_values[RATE_DURABILITY_LOSS_ON_DEATH] = sConfigMgr->GetOption("DurabilityLoss.OnDeath", 10.0f); if (_rate_values[RATE_DURABILITY_LOSS_ON_DEATH] < 0.0f) { LOG_ERROR("server.loading", "DurabilityLoss.OnDeath ({}) must be >=0. Using 0.0 instead.", _rate_values[RATE_DURABILITY_LOSS_ON_DEATH]); _rate_values[RATE_DURABILITY_LOSS_ON_DEATH] = 0.0f; } if (_rate_values[RATE_DURABILITY_LOSS_ON_DEATH] > 100.0f) { LOG_ERROR("server.loading", "DurabilityLoss.OnDeath ({}) must be <= 100. Using 100.0 instead.", _rate_values[RATE_DURABILITY_LOSS_ON_DEATH]); _rate_values[RATE_DURABILITY_LOSS_ON_DEATH] = 0.0f; } _rate_values[RATE_DURABILITY_LOSS_ON_DEATH] = _rate_values[RATE_DURABILITY_LOSS_ON_DEATH] / 100.0f; _rate_values[RATE_DURABILITY_LOSS_DAMAGE] = sConfigMgr->GetOption("DurabilityLossChance.Damage", 0.5f); if (_rate_values[RATE_DURABILITY_LOSS_DAMAGE] < 0.0f) { LOG_ERROR("server.loading", "DurabilityLossChance.Damage ({}) must be >=0. Using 0.0 instead.", _rate_values[RATE_DURABILITY_LOSS_DAMAGE]); _rate_values[RATE_DURABILITY_LOSS_DAMAGE] = 0.0f; } _rate_values[RATE_DURABILITY_LOSS_ABSORB] = sConfigMgr->GetOption("DurabilityLossChance.Absorb", 0.5f); if (_rate_values[RATE_DURABILITY_LOSS_ABSORB] < 0.0f) { LOG_ERROR("server.loading", "DurabilityLossChance.Absorb ({}) must be >=0. Using 0.0 instead.", _rate_values[RATE_DURABILITY_LOSS_ABSORB]); _rate_values[RATE_DURABILITY_LOSS_ABSORB] = 0.0f; } _rate_values[RATE_DURABILITY_LOSS_PARRY] = sConfigMgr->GetOption("DurabilityLossChance.Parry", 0.05f); if (_rate_values[RATE_DURABILITY_LOSS_PARRY] < 0.0f) { LOG_ERROR("server.loading", "DurabilityLossChance.Parry ({}) must be >=0. Using 0.0 instead.", _rate_values[RATE_DURABILITY_LOSS_PARRY]); _rate_values[RATE_DURABILITY_LOSS_PARRY] = 0.0f; } _rate_values[RATE_DURABILITY_LOSS_BLOCK] = sConfigMgr->GetOption("DurabilityLossChance.Block", 0.05f); if (_rate_values[RATE_DURABILITY_LOSS_BLOCK] < 0.0f) { LOG_ERROR("server.loading", "DurabilityLossChance.Block ({}) must be >=0. Using 0.0 instead.", _rate_values[RATE_DURABILITY_LOSS_BLOCK]); _rate_values[RATE_DURABILITY_LOSS_BLOCK] = 0.0f; } ///- Read other configuration items from the config file _bool_configs[CONFIG_DURABILITY_LOSS_IN_PVP] = sConfigMgr->GetOption("DurabilityLoss.InPvP", false); _int_configs[CONFIG_COMPRESSION] = sConfigMgr->GetOption("Compression", 1); if (_int_configs[CONFIG_COMPRESSION] < 1 || _int_configs[CONFIG_COMPRESSION] > 9) { LOG_ERROR("server.loading", "Compression level ({}) must be in range 1..9. Using default compression level (1).", _int_configs[CONFIG_COMPRESSION]); _int_configs[CONFIG_COMPRESSION] = 1; } _bool_configs[CONFIG_ADDON_CHANNEL] = sConfigMgr->GetOption("AddonChannel", true); _bool_configs[CONFIG_CLEAN_CHARACTER_DB] = sConfigMgr->GetOption("CleanCharacterDB", false); _int_configs[CONFIG_PERSISTENT_CHARACTER_CLEAN_FLAGS] = sConfigMgr->GetOption("PersistentCharacterCleanFlags", 0); _int_configs[CONFIG_CHAT_CHANNEL_LEVEL_REQ] = sConfigMgr->GetOption("ChatLevelReq.Channel", 1); _int_configs[CONFIG_CHAT_WHISPER_LEVEL_REQ] = sConfigMgr->GetOption("ChatLevelReq.Whisper", 1); _int_configs[CONFIG_CHAT_SAY_LEVEL_REQ] = sConfigMgr->GetOption("ChatLevelReq.Say", 1); _int_configs[CONFIG_PARTY_LEVEL_REQ] = sConfigMgr->GetOption("PartyLevelReq", 1); _int_configs[CONFIG_TRADE_LEVEL_REQ] = sConfigMgr->GetOption("LevelReq.Trade", 1); _int_configs[CONFIG_TICKET_LEVEL_REQ] = sConfigMgr->GetOption("LevelReq.Ticket", 1); _int_configs[CONFIG_AUCTION_LEVEL_REQ] = sConfigMgr->GetOption("LevelReq.Auction", 1); _int_configs[CONFIG_MAIL_LEVEL_REQ] = sConfigMgr->GetOption("LevelReq.Mail", 1); _bool_configs[CONFIG_ALLOW_PLAYER_COMMANDS] = sConfigMgr->GetOption("AllowPlayerCommands", 1); _bool_configs[CONFIG_PRESERVE_CUSTOM_CHANNELS] = sConfigMgr->GetOption("PreserveCustomChannels", false); _int_configs[CONFIG_PRESERVE_CUSTOM_CHANNEL_DURATION] = sConfigMgr->GetOption("PreserveCustomChannelDuration", 14); _int_configs[CONFIG_INTERVAL_SAVE] = sConfigMgr->GetOption("PlayerSaveInterval", 15 * MINUTE * IN_MILLISECONDS); _int_configs[CONFIG_INTERVAL_DISCONNECT_TOLERANCE] = sConfigMgr->GetOption("DisconnectToleranceInterval", 0); _bool_configs[CONFIG_STATS_SAVE_ONLY_ON_LOGOUT] = sConfigMgr->GetOption("PlayerSave.Stats.SaveOnlyOnLogout", true); _int_configs[CONFIG_MIN_LEVEL_STAT_SAVE] = sConfigMgr->GetOption("PlayerSave.Stats.MinLevel", 0); if (_int_configs[CONFIG_MIN_LEVEL_STAT_SAVE] > MAX_LEVEL || int32(_int_configs[CONFIG_MIN_LEVEL_STAT_SAVE]) < 0) { LOG_ERROR("server.loading", "PlayerSave.Stats.MinLevel ({}) must be in range 0..80. Using default, do not save character stats (0).", _int_configs[CONFIG_MIN_LEVEL_STAT_SAVE]); _int_configs[CONFIG_MIN_LEVEL_STAT_SAVE] = 0; } _int_configs[CONFIG_INTERVAL_GRIDCLEAN] = sConfigMgr->GetOption("GridCleanUpDelay", 5 * MINUTE * IN_MILLISECONDS); if (_int_configs[CONFIG_INTERVAL_GRIDCLEAN] < MIN_GRID_DELAY) { LOG_ERROR("server.loading", "GridCleanUpDelay (%i) must be greater %u. Use this minimal value.", _int_configs[CONFIG_INTERVAL_GRIDCLEAN], MIN_GRID_DELAY); _int_configs[CONFIG_INTERVAL_GRIDCLEAN] = MIN_GRID_DELAY; } if (reload) sMapMgr->SetGridCleanUpDelay(_int_configs[CONFIG_INTERVAL_GRIDCLEAN]); _int_configs[CONFIG_INTERVAL_MAPUPDATE] = sConfigMgr->GetOption("MapUpdateInterval", 10); if (_int_configs[CONFIG_INTERVAL_MAPUPDATE] < MIN_MAP_UPDATE_DELAY) { LOG_ERROR("server.loading", "MapUpdateInterval ({}) must be greater {}. Use this minimal value.", _int_configs[CONFIG_INTERVAL_MAPUPDATE], MIN_MAP_UPDATE_DELAY); _int_configs[CONFIG_INTERVAL_MAPUPDATE] = MIN_MAP_UPDATE_DELAY; } if (reload) sMapMgr->SetMapUpdateInterval(_int_configs[CONFIG_INTERVAL_MAPUPDATE]); _int_configs[CONFIG_INTERVAL_CHANGEWEATHER] = sConfigMgr->GetOption("ChangeWeatherInterval", 10 * MINUTE * IN_MILLISECONDS); if (reload) { uint32 val = sConfigMgr->GetOption("WorldServerPort", 8085); if (val != _int_configs[CONFIG_PORT_WORLD]) LOG_ERROR("server.loading", "WorldServerPort option can't be changed at worldserver.conf reload, using current value ({}).", _int_configs[CONFIG_PORT_WORLD]); } else _int_configs[CONFIG_PORT_WORLD] = sConfigMgr->GetOption("WorldServerPort", 8085); _bool_configs[CONFIG_CLOSE_IDLE_CONNECTIONS] = sConfigMgr->GetOption("CloseIdleConnections", true); _int_configs[CONFIG_SOCKET_TIMEOUTTIME] = sConfigMgr->GetOption("SocketTimeOutTime", 900000); _int_configs[CONFIG_SOCKET_TIMEOUTTIME_ACTIVE] = sConfigMgr->GetOption("SocketTimeOutTimeActive", 60000); _int_configs[CONFIG_SESSION_ADD_DELAY] = sConfigMgr->GetOption("SessionAddDelay", 10000); _float_configs[CONFIG_GROUP_XP_DISTANCE] = sConfigMgr->GetOption("MaxGroupXPDistance", 74.0f); _float_configs[CONFIG_MAX_RECRUIT_A_FRIEND_DISTANCE] = sConfigMgr->GetOption("MaxRecruitAFriendBonusDistance", 100.0f); /// \todo Add MonsterSight in worldserver.conf or put it as define _float_configs[CONFIG_SIGHT_MONSTER] = sConfigMgr->GetOption("MonsterSight", 50); if (reload) { uint32 val = sConfigMgr->GetOption("GameType", 0); if (val != _int_configs[CONFIG_GAME_TYPE]) LOG_ERROR("server.loading", "GameType option can't be changed at worldserver.conf reload, using current value ({}).", _int_configs[CONFIG_GAME_TYPE]); } else _int_configs[CONFIG_GAME_TYPE] = sConfigMgr->GetOption("GameType", 0); if (reload) { uint32 val = sConfigMgr->GetOption("RealmZone", REALM_ZONE_DEVELOPMENT); if (val != _int_configs[CONFIG_REALM_ZONE]) LOG_ERROR("server.loading", "RealmZone option can't be changed at worldserver.conf reload, using current value ({}).", _int_configs[CONFIG_REALM_ZONE]); } else _int_configs[CONFIG_REALM_ZONE] = sConfigMgr->GetOption("RealmZone", REALM_ZONE_DEVELOPMENT); _bool_configs[CONFIG_STRICT_NAMES_RESERVED] = sConfigMgr->GetOption ("StrictNames.Reserved", true); _bool_configs[CONFIG_STRICT_NAMES_PROFANITY] = sConfigMgr->GetOption ("StrictNames.Profanity", true); _int_configs[CONFIG_STRICT_PLAYER_NAMES] = sConfigMgr->GetOption ("StrictPlayerNames", 0); _int_configs[CONFIG_STRICT_CHARTER_NAMES] = sConfigMgr->GetOption ("StrictCharterNames", 0); _int_configs[CONFIG_STRICT_CHANNEL_NAMES] = sConfigMgr->GetOption ("StrictChannelNames", 0); _int_configs[CONFIG_STRICT_PET_NAMES] = sConfigMgr->GetOption ("StrictPetNames", 0); _bool_configs[CONFIG_ALLOW_TWO_SIDE_ACCOUNTS] = sConfigMgr->GetOption("AllowTwoSide.Accounts", true); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_CALENDAR] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Calendar", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_CHAT] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Chat", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_CHANNEL] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Channel", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_GROUP] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Group", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_GUILD] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Guild", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_ARENA] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Arena", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_AUCTION] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Auction", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_MAIL] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Mail", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_WHO_LIST] = sConfigMgr->GetOption("AllowTwoSide.WhoList", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_ADD_FRIEND] = sConfigMgr->GetOption("AllowTwoSide.AddFriend", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_TRADE] = sConfigMgr->GetOption("AllowTwoSide.Trade", false); _bool_configs[CONFIG_ALLOW_TWO_SIDE_INTERACTION_EMOTE] = sConfigMgr->GetOption("AllowTwoSide.Interaction.Emote", false); _int_configs[CONFIG_MIN_PLAYER_NAME] = sConfigMgr->GetOption ("MinPlayerName", 2); if (_int_configs[CONFIG_MIN_PLAYER_NAME] < 1 || _int_configs[CONFIG_MIN_PLAYER_NAME] > MAX_PLAYER_NAME) { LOG_ERROR("server.loading", "MinPlayerName ({}) must be in range 1..{}. Set to 2.", _int_configs[CONFIG_MIN_PLAYER_NAME], MAX_PLAYER_NAME); _int_configs[CONFIG_MIN_PLAYER_NAME] = 2; } _int_configs[CONFIG_MIN_CHARTER_NAME] = sConfigMgr->GetOption ("MinCharterName", 2); if (_int_configs[CONFIG_MIN_CHARTER_NAME] < 1 || _int_configs[CONFIG_MIN_CHARTER_NAME] > MAX_CHARTER_NAME) { LOG_ERROR("server.loading", "MinCharterName ({}) must be in range 1..{}. Set to 2.", _int_configs[CONFIG_MIN_CHARTER_NAME], MAX_CHARTER_NAME); _int_configs[CONFIG_MIN_CHARTER_NAME] = 2; } _int_configs[CONFIG_MIN_PET_NAME] = sConfigMgr->GetOption ("MinPetName", 2); if (_int_configs[CONFIG_MIN_PET_NAME] < 1 || _int_configs[CONFIG_MIN_PET_NAME] > MAX_PET_NAME) { LOG_ERROR("server.loading", "MinPetName ({}) must be in range 1..{}. Set to 2.", _int_configs[CONFIG_MIN_PET_NAME], MAX_PET_NAME); _int_configs[CONFIG_MIN_PET_NAME] = 2; } _int_configs[CONFIG_CHARTER_COST_GUILD] = sConfigMgr->GetOption("Guild.CharterCost", 1000); _int_configs[CONFIG_CHARTER_COST_ARENA_2v2] = sConfigMgr->GetOption("ArenaTeam.CharterCost.2v2", 800000); _int_configs[CONFIG_CHARTER_COST_ARENA_3v3] = sConfigMgr->GetOption("ArenaTeam.CharterCost.3v3", 1200000); _int_configs[CONFIG_CHARTER_COST_ARENA_5v5] = sConfigMgr->GetOption("ArenaTeam.CharterCost.5v5", 2000000); _int_configs[CONFIG_MAX_WHO_LIST_RETURN] = sConfigMgr->GetOption("MaxWhoListReturns", 49); _int_configs[CONFIG_CHARACTER_CREATING_DISABLED] = sConfigMgr->GetOption("CharacterCreating.Disabled", 0); _int_configs[CONFIG_CHARACTER_CREATING_DISABLED_RACEMASK] = sConfigMgr->GetOption("CharacterCreating.Disabled.RaceMask", 0); _int_configs[CONFIG_CHARACTER_CREATING_DISABLED_CLASSMASK] = sConfigMgr->GetOption("CharacterCreating.Disabled.ClassMask", 0); _int_configs[CONFIG_CHARACTERS_PER_REALM] = sConfigMgr->GetOption("CharactersPerRealm", 10); if (_int_configs[CONFIG_CHARACTERS_PER_REALM] < 1 || _int_configs[CONFIG_CHARACTERS_PER_REALM] > 10) { LOG_ERROR("server.loading", "CharactersPerRealm ({}) must be in range 1..10. Set to 10.", _int_configs[CONFIG_CHARACTERS_PER_REALM]); _int_configs[CONFIG_CHARACTERS_PER_REALM] = 10; } // must be after CONFIG_CHARACTERS_PER_REALM _int_configs[CONFIG_CHARACTERS_PER_ACCOUNT] = sConfigMgr->GetOption("CharactersPerAccount", 50); if (_int_configs[CONFIG_CHARACTERS_PER_ACCOUNT] < _int_configs[CONFIG_CHARACTERS_PER_REALM]) { LOG_ERROR("server.loading", "CharactersPerAccount ({}) can't be less than CharactersPerRealm ({}).", _int_configs[CONFIG_CHARACTERS_PER_ACCOUNT], _int_configs[CONFIG_CHARACTERS_PER_REALM]); _int_configs[CONFIG_CHARACTERS_PER_ACCOUNT] = _int_configs[CONFIG_CHARACTERS_PER_REALM]; } _int_configs[CONFIG_HEROIC_CHARACTERS_PER_REALM] = sConfigMgr->GetOption("HeroicCharactersPerRealm", 1); if (int32(_int_configs[CONFIG_HEROIC_CHARACTERS_PER_REALM]) < 0 || _int_configs[CONFIG_HEROIC_CHARACTERS_PER_REALM] > 10) { LOG_ERROR("server.loading", "HeroicCharactersPerRealm ({}) must be in range 0..10. Set to 1.", _int_configs[CONFIG_HEROIC_CHARACTERS_PER_REALM]); _int_configs[CONFIG_HEROIC_CHARACTERS_PER_REALM] = 1; } _int_configs[CONFIG_CHARACTER_CREATING_MIN_LEVEL_FOR_HEROIC_CHARACTER] = sConfigMgr->GetOption("CharacterCreating.MinLevelForHeroicCharacter", 55); _int_configs[CONFIG_SKIP_CINEMATICS] = sConfigMgr->GetOption("SkipCinematics", 0); if (int32(_int_configs[CONFIG_SKIP_CINEMATICS]) < 0 || _int_configs[CONFIG_SKIP_CINEMATICS] > 2) { LOG_ERROR("server.loading", "SkipCinematics ({}) must be in range 0..2. Set to 0.", _int_configs[CONFIG_SKIP_CINEMATICS]); _int_configs[CONFIG_SKIP_CINEMATICS] = 0; } if (reload) { uint32 val = sConfigMgr->GetOption("MaxPlayerLevel", DEFAULT_MAX_LEVEL); if (val != _int_configs[CONFIG_MAX_PLAYER_LEVEL]) LOG_ERROR("server.loading", "MaxPlayerLevel option can't be changed at config reload, using current value ({}).", _int_configs[CONFIG_MAX_PLAYER_LEVEL]); } else _int_configs[CONFIG_MAX_PLAYER_LEVEL] = sConfigMgr->GetOption("MaxPlayerLevel", DEFAULT_MAX_LEVEL); if (_int_configs[CONFIG_MAX_PLAYER_LEVEL] > MAX_LEVEL || _int_configs[CONFIG_MAX_PLAYER_LEVEL] < 1) { LOG_ERROR("server.loading", "MaxPlayerLevel ({}) must be in range 1..{}. Set to {}.", _int_configs[CONFIG_MAX_PLAYER_LEVEL], MAX_LEVEL, MAX_LEVEL); _int_configs[CONFIG_MAX_PLAYER_LEVEL] = MAX_LEVEL; } _int_configs[CONFIG_MIN_DUALSPEC_LEVEL] = sConfigMgr->GetOption("MinDualSpecLevel", 40); _int_configs[CONFIG_START_PLAYER_LEVEL] = sConfigMgr->GetOption("StartPlayerLevel", 1); if (_int_configs[CONFIG_START_PLAYER_LEVEL] < 1 || _int_configs[CONFIG_START_PLAYER_LEVEL] > _int_configs[CONFIG_MAX_PLAYER_LEVEL]) { LOG_ERROR("server.loading", "StartPlayerLevel ({}) must be in range 1..MaxPlayerLevel({}). Set to 1.", _int_configs[CONFIG_START_PLAYER_LEVEL], _int_configs[CONFIG_MAX_PLAYER_LEVEL]); _int_configs[CONFIG_START_PLAYER_LEVEL] = 1; } _int_configs[CONFIG_START_HEROIC_PLAYER_LEVEL] = sConfigMgr->GetOption("StartHeroicPlayerLevel", 55); if (_int_configs[CONFIG_START_HEROIC_PLAYER_LEVEL] < 1 || _int_configs[CONFIG_START_HEROIC_PLAYER_LEVEL] > _int_configs[CONFIG_MAX_PLAYER_LEVEL]) { LOG_ERROR("server.loading", "StartHeroicPlayerLevel ({}) must be in range 1..MaxPlayerLevel({}). Set to 55.", _int_configs[CONFIG_START_HEROIC_PLAYER_LEVEL], _int_configs[CONFIG_MAX_PLAYER_LEVEL]); _int_configs[CONFIG_START_HEROIC_PLAYER_LEVEL] = 55; } _int_configs[CONFIG_START_PLAYER_MONEY] = sConfigMgr->GetOption("StartPlayerMoney", 0); if (int32(_int_configs[CONFIG_START_PLAYER_MONEY]) < 0 || int32(_int_configs[CONFIG_START_PLAYER_MONEY]) > MAX_MONEY_AMOUNT) { LOG_ERROR("server.loading", "StartPlayerMoney ({}) must be in range 0..{}. Set to {}.", _int_configs[CONFIG_START_PLAYER_MONEY], MAX_MONEY_AMOUNT, 0); _int_configs[CONFIG_START_PLAYER_MONEY] = 0; } _int_configs[CONFIG_START_HEROIC_PLAYER_MONEY] = sConfigMgr->GetOption("StartHeroicPlayerMoney", 2000); if (int32(_int_configs[CONFIG_START_HEROIC_PLAYER_MONEY]) < 0 || int32(_int_configs[CONFIG_START_HEROIC_PLAYER_MONEY]) > MAX_MONEY_AMOUNT) { LOG_ERROR("server.loading", "StartHeroicPlayerMoney ({}) must be in range 0..{}. Set to {}.", _int_configs[CONFIG_START_HEROIC_PLAYER_MONEY], MAX_MONEY_AMOUNT, 2000); _int_configs[CONFIG_START_HEROIC_PLAYER_MONEY] = 2000; } _int_configs[CONFIG_MAX_HONOR_POINTS] = sConfigMgr->GetOption("MaxHonorPoints", 75000); if (int32(_int_configs[CONFIG_MAX_HONOR_POINTS]) < 0) { LOG_ERROR("server.loading", "MaxHonorPoints ({}) can't be negative. Set to 0.", _int_configs[CONFIG_MAX_HONOR_POINTS]); _int_configs[CONFIG_MAX_HONOR_POINTS] = 0; } _int_configs[CONFIG_MAX_HONOR_POINTS_MONEY_PER_POINT] = sConfigMgr->GetOption("MaxHonorPointsMoneyPerPoint", 0); if (int32(_int_configs[CONFIG_MAX_HONOR_POINTS_MONEY_PER_POINT]) < 0) { LOG_ERROR("server.loading", "MaxHonorPointsMoneyPerPoint ({}) can't be negative. Set to 0.", _int_configs[CONFIG_MAX_HONOR_POINTS_MONEY_PER_POINT]); _int_configs[CONFIG_MAX_HONOR_POINTS_MONEY_PER_POINT] = 0; } _int_configs[CONFIG_START_HONOR_POINTS] = sConfigMgr->GetOption("StartHonorPoints", 0); if (int32(_int_configs[CONFIG_START_HONOR_POINTS]) < 0 || int32(_int_configs[CONFIG_START_HONOR_POINTS]) > int32(_int_configs[CONFIG_MAX_HONOR_POINTS])) { LOG_ERROR("server.loading", "StartHonorPoints ({}) must be in range 0..MaxHonorPoints({}). Set to {}.", _int_configs[CONFIG_START_HONOR_POINTS], _int_configs[CONFIG_MAX_HONOR_POINTS], 0); _int_configs[CONFIG_START_HONOR_POINTS] = 0; } _int_configs[CONFIG_MAX_ARENA_POINTS] = sConfigMgr->GetOption("MaxArenaPoints", 10000); if (int32(_int_configs[CONFIG_MAX_ARENA_POINTS]) < 0) { LOG_ERROR("server.loading", "MaxArenaPoints ({}) can't be negative. Set to 0.", _int_configs[CONFIG_MAX_ARENA_POINTS]); _int_configs[CONFIG_MAX_ARENA_POINTS] = 0; } _int_configs[CONFIG_START_ARENA_POINTS] = sConfigMgr->GetOption("StartArenaPoints", 0); if (int32(_int_configs[CONFIG_START_ARENA_POINTS]) < 0 || int32(_int_configs[CONFIG_START_ARENA_POINTS]) > int32(_int_configs[CONFIG_MAX_ARENA_POINTS])) { LOG_ERROR("server.loading", "StartArenaPoints ({}) must be in range 0..MaxArenaPoints({}). Set to {}.", _int_configs[CONFIG_START_ARENA_POINTS], _int_configs[CONFIG_MAX_ARENA_POINTS], 0); _int_configs[CONFIG_START_ARENA_POINTS] = 0; } _int_configs[CONFIG_MAX_RECRUIT_A_FRIEND_BONUS_PLAYER_LEVEL] = sConfigMgr->GetOption("RecruitAFriend.MaxLevel", 60); if (_int_configs[CONFIG_MAX_RECRUIT_A_FRIEND_BONUS_PLAYER_LEVEL] > _int_configs[CONFIG_MAX_PLAYER_LEVEL] || int32(_int_configs[CONFIG_MAX_RECRUIT_A_FRIEND_BONUS_PLAYER_LEVEL]) < 0) { LOG_ERROR("server.loading", "RecruitAFriend.MaxLevel ({}) must be in the range 0..MaxLevel({}). Set to {}.", _int_configs[CONFIG_MAX_RECRUIT_A_FRIEND_BONUS_PLAYER_LEVEL], _int_configs[CONFIG_MAX_PLAYER_LEVEL], 60); _int_configs[CONFIG_MAX_RECRUIT_A_FRIEND_BONUS_PLAYER_LEVEL] = 60; } _int_configs[CONFIG_MAX_RECRUIT_A_FRIEND_BONUS_PLAYER_LEVEL_DIFFERENCE] = sConfigMgr->GetOption("RecruitAFriend.MaxDifference", 4); _bool_configs[CONFIG_ALL_TAXI_PATHS] = sConfigMgr->GetOption("AllFlightPaths", false); _int_configs[CONFIG_INSTANT_TAXI] = sConfigMgr->GetOption("InstantFlightPaths", 0); _bool_configs[CONFIG_INSTANCE_IGNORE_LEVEL] = sConfigMgr->GetOption("Instance.IgnoreLevel", false); _bool_configs[CONFIG_INSTANCE_IGNORE_RAID] = sConfigMgr->GetOption("Instance.IgnoreRaid", false); _bool_configs[CONFIG_INSTANCE_GMSUMMON_PLAYER] = sConfigMgr->GetOption("Instance.GMSummonPlayer", false); _bool_configs[CONFIG_INSTANCE_SHARED_ID] = sConfigMgr->GetOption("Instance.SharedNormalHeroicId", false); _int_configs[CONFIG_INSTANCE_RESET_TIME_HOUR] = sConfigMgr->GetOption("Instance.ResetTimeHour", 4); _int_configs[CONFIG_INSTANCE_RESET_TIME_RELATIVE_TIMESTAMP] = sConfigMgr->GetOption("Instance.ResetTimeRelativeTimestamp", 1135814400); _int_configs[CONFIG_INSTANCE_UNLOAD_DELAY] = sConfigMgr->GetOption("Instance.UnloadDelay", 30 * MINUTE * IN_MILLISECONDS); _int_configs[CONFIG_MAX_PRIMARY_TRADE_SKILL] = sConfigMgr->GetOption("MaxPrimaryTradeSkill", 2); _int_configs[CONFIG_MIN_PETITION_SIGNS] = sConfigMgr->GetOption("MinPetitionSigns", 9); if (_int_configs[CONFIG_MIN_PETITION_SIGNS] > 9 || int32(_int_configs[CONFIG_MIN_PETITION_SIGNS]) < 0) { LOG_ERROR("server.loading", "MinPetitionSigns ({}) must be in range 0..9. Set to 9.", _int_configs[CONFIG_MIN_PETITION_SIGNS]); _int_configs[CONFIG_MIN_PETITION_SIGNS] = 9; } _int_configs[CONFIG_GM_LOGIN_STATE] = sConfigMgr->GetOption("GM.LoginState", 2); _int_configs[CONFIG_GM_VISIBLE_STATE] = sConfigMgr->GetOption("GM.Visible", 2); _int_configs[CONFIG_GM_CHAT] = sConfigMgr->GetOption("GM.Chat", 2); _int_configs[CONFIG_GM_WHISPERING_TO] = sConfigMgr->GetOption("GM.WhisperingTo", 2); _int_configs[CONFIG_GM_LEVEL_IN_GM_LIST] = sConfigMgr->GetOption("GM.InGMList.Level", SEC_ADMINISTRATOR); _int_configs[CONFIG_GM_LEVEL_IN_WHO_LIST] = sConfigMgr->GetOption("GM.InWhoList.Level", SEC_ADMINISTRATOR); _int_configs[CONFIG_START_GM_LEVEL] = sConfigMgr->GetOption("GM.StartLevel", 1); if (_int_configs[CONFIG_START_GM_LEVEL] < _int_configs[CONFIG_START_PLAYER_LEVEL]) { LOG_ERROR("server.loading", "GM.StartLevel ({}) must be in range StartPlayerLevel({})..{}. Set to {}.", _int_configs[CONFIG_START_GM_LEVEL], _int_configs[CONFIG_START_PLAYER_LEVEL], MAX_LEVEL, _int_configs[CONFIG_START_PLAYER_LEVEL]); _int_configs[CONFIG_START_GM_LEVEL] = _int_configs[CONFIG_START_PLAYER_LEVEL]; } else if (_int_configs[CONFIG_START_GM_LEVEL] > MAX_LEVEL) { LOG_ERROR("server.loading", "GM.StartLevel ({}) must be in range 1..{}. Set to {}.", _int_configs[CONFIG_START_GM_LEVEL], MAX_LEVEL, MAX_LEVEL); _int_configs[CONFIG_START_GM_LEVEL] = MAX_LEVEL; } _bool_configs[CONFIG_ALLOW_GM_GROUP] = sConfigMgr->GetOption("GM.AllowInvite", false); _bool_configs[CONFIG_ALLOW_GM_FRIEND] = sConfigMgr->GetOption("GM.AllowFriend", false); _bool_configs[CONFIG_GM_LOWER_SECURITY] = sConfigMgr->GetOption("GM.LowerSecurity", false); _float_configs[CONFIG_CHANCE_OF_GM_SURVEY] = sConfigMgr->GetOption("GM.TicketSystem.ChanceOfGMSurvey", 50.0f); _int_configs[CONFIG_GROUP_VISIBILITY] = sConfigMgr->GetOption("Visibility.GroupMode", 1); _bool_configs[CONFIG_OBJECT_SPARKLES] = sConfigMgr->GetOption("Visibility.ObjectSparkles", true); _bool_configs[CONFIG_LOW_LEVEL_REGEN_BOOST] = sConfigMgr->GetOption("EnableLowLevelRegenBoost", true); _bool_configs[CONFIG_OBJECT_QUEST_MARKERS] = sConfigMgr->GetOption("Visibility.ObjectQuestMarkers", true); _int_configs[CONFIG_MAIL_DELIVERY_DELAY] = sConfigMgr->GetOption("MailDeliveryDelay", HOUR); _int_configs[CONFIG_UPTIME_UPDATE] = sConfigMgr->GetOption("UpdateUptimeInterval", 10); if (int32(_int_configs[CONFIG_UPTIME_UPDATE]) <= 0) { LOG_ERROR("server.loading", "UpdateUptimeInterval ({}) must be > 0, set to default 10.", _int_configs[CONFIG_UPTIME_UPDATE]); _int_configs[CONFIG_UPTIME_UPDATE] = 1; } if (reload) { _timers[WUPDATE_UPTIME].SetInterval(_int_configs[CONFIG_UPTIME_UPDATE]*MINUTE * IN_MILLISECONDS); _timers[WUPDATE_UPTIME].Reset(); } // log db cleanup interval _int_configs[CONFIG_LOGDB_CLEARINTERVAL] = sConfigMgr->GetOption("LogDB.Opt.ClearInterval", 10); if (int32(_int_configs[CONFIG_LOGDB_CLEARINTERVAL]) <= 0) { LOG_ERROR("server.loading", "LogDB.Opt.ClearInterval ({}) must be > 0, set to default 10.", _int_configs[CONFIG_LOGDB_CLEARINTERVAL]); _int_configs[CONFIG_LOGDB_CLEARINTERVAL] = 10; } if (reload) { _timers[WUPDATE_CLEANDB].SetInterval(_int_configs[CONFIG_LOGDB_CLEARINTERVAL] * MINUTE * IN_MILLISECONDS); _timers[WUPDATE_CLEANDB].Reset(); } _int_configs[CONFIG_LOGDB_CLEARTIME] = sConfigMgr->GetOption("LogDB.Opt.ClearTime", 1209600); // 14 days default LOG_INFO("server.loading", "Will clear `logs` table of entries older than {} seconds every {} minutes.", _int_configs[CONFIG_LOGDB_CLEARTIME], _int_configs[CONFIG_LOGDB_CLEARINTERVAL]); _int_configs[CONFIG_TELEPORT_TIMEOUT_NEAR] = sConfigMgr->GetOption("TeleportTimeoutNear", 25); // pussywizard _int_configs[CONFIG_TELEPORT_TIMEOUT_FAR] = sConfigMgr->GetOption("TeleportTimeoutFar", 45); // pussywizard _int_configs[CONFIG_MAX_ALLOWED_MMR_DROP] = sConfigMgr->GetOption("MaxAllowedMMRDrop", 500); // pussywizard _bool_configs[CONFIG_ENABLE_LOGIN_AFTER_DC] = sConfigMgr->GetOption("EnableLoginAfterDC", true); // pussywizard _bool_configs[CONFIG_DONT_CACHE_RANDOM_MOVEMENT_PATHS] = sConfigMgr->GetOption("DontCacheRandomMovementPaths", true); // pussywizard _int_configs[CONFIG_SKILL_CHANCE_ORANGE] = sConfigMgr->GetOption("SkillChance.Orange", 100); _int_configs[CONFIG_SKILL_CHANCE_YELLOW] = sConfigMgr->GetOption("SkillChance.Yellow", 75); _int_configs[CONFIG_SKILL_CHANCE_GREEN] = sConfigMgr->GetOption("SkillChance.Green", 25); _int_configs[CONFIG_SKILL_CHANCE_GREY] = sConfigMgr->GetOption("SkillChance.Grey", 0); _int_configs[CONFIG_SKILL_CHANCE_MINING_STEPS] = sConfigMgr->GetOption("SkillChance.MiningSteps", 75); _int_configs[CONFIG_SKILL_CHANCE_SKINNING_STEPS] = sConfigMgr->GetOption("SkillChance.SkinningSteps", 75); _bool_configs[CONFIG_SKILL_PROSPECTING] = sConfigMgr->GetOption("SkillChance.Prospecting", false); _bool_configs[CONFIG_SKILL_MILLING] = sConfigMgr->GetOption("SkillChance.Milling", false); _int_configs[CONFIG_SKILL_GAIN_CRAFTING] = sConfigMgr->GetOption("SkillGain.Crafting", 1); _int_configs[CONFIG_SKILL_GAIN_DEFENSE] = sConfigMgr->GetOption("SkillGain.Defense", 1); _int_configs[CONFIG_SKILL_GAIN_GATHERING] = sConfigMgr->GetOption("SkillGain.Gathering", 1); _int_configs[CONFIG_SKILL_GAIN_WEAPON] = sConfigMgr->GetOption("SkillGain.Weapon", 1); _int_configs[CONFIG_MAX_OVERSPEED_PINGS] = sConfigMgr->GetOption("MaxOverspeedPings", 2); if (_int_configs[CONFIG_MAX_OVERSPEED_PINGS] != 0 && _int_configs[CONFIG_MAX_OVERSPEED_PINGS] < 2) { LOG_ERROR("server.loading", "MaxOverspeedPings ({}) must be in range 2..infinity (or 0 to disable check). Set to 2.", _int_configs[CONFIG_MAX_OVERSPEED_PINGS]); _int_configs[CONFIG_MAX_OVERSPEED_PINGS] = 2; } _bool_configs[CONFIG_SAVE_RESPAWN_TIME_IMMEDIATELY] = sConfigMgr->GetOption("SaveRespawnTimeImmediately", true); _bool_configs[CONFIG_WEATHER] = sConfigMgr->GetOption("ActivateWeather", true); _int_configs[CONFIG_DISABLE_BREATHING] = sConfigMgr->GetOption("DisableWaterBreath", SEC_CONSOLE); _bool_configs[CONFIG_ALWAYS_MAX_SKILL_FOR_LEVEL] = sConfigMgr->GetOption("AlwaysMaxSkillForLevel", false); if (reload) { uint32 val = sConfigMgr->GetOption("Expansion", 2); if (val != _int_configs[CONFIG_EXPANSION]) LOG_ERROR("server.loading", "Expansion option can't be changed at worldserver.conf reload, using current value ({}).", _int_configs[CONFIG_EXPANSION]); } else _int_configs[CONFIG_EXPANSION] = sConfigMgr->GetOption("Expansion", 2); _int_configs[CONFIG_CHATFLOOD_MESSAGE_COUNT] = sConfigMgr->GetOption("ChatFlood.MessageCount", 10); _int_configs[CONFIG_CHATFLOOD_MESSAGE_DELAY] = sConfigMgr->GetOption("ChatFlood.MessageDelay", 1); _int_configs[CONFIG_CHATFLOOD_ADDON_MESSAGE_COUNT] = sConfigMgr->GetOption("ChatFlood.AddonMessageCount", 100); _int_configs[CONFIG_CHATFLOOD_ADDON_MESSAGE_DELAY] = sConfigMgr->GetOption("ChatFlood.AddonMessageDelay", 1); _int_configs[CONFIG_CHATFLOOD_MUTE_TIME] = sConfigMgr->GetOption("ChatFlood.MuteTime", 10); _bool_configs[CONFIG_CHAT_MUTE_FIRST_LOGIN] = sConfigMgr->GetOption("Chat.MuteFirstLogin", false); _int_configs[CONFIG_CHAT_TIME_MUTE_FIRST_LOGIN] = sConfigMgr->GetOption("Chat.MuteTimeFirstLogin", 120); _int_configs[CONFIG_EVENT_ANNOUNCE] = sConfigMgr->GetOption("Event.Announce", 0); _float_configs[CONFIG_CREATURE_FAMILY_FLEE_ASSISTANCE_RADIUS] = sConfigMgr->GetOption("CreatureFamilyFleeAssistanceRadius", 30.0f); _float_configs[CONFIG_CREATURE_FAMILY_ASSISTANCE_RADIUS] = sConfigMgr->GetOption("CreatureFamilyAssistanceRadius", 10.0f); _int_configs[CONFIG_CREATURE_FAMILY_ASSISTANCE_DELAY] = sConfigMgr->GetOption("CreatureFamilyAssistanceDelay", 2000); _int_configs[CONFIG_CREATURE_FAMILY_ASSISTANCE_PERIOD] = sConfigMgr->GetOption("CreatureFamilyAssistancePeriod", 3000); _int_configs[CONFIG_CREATURE_FAMILY_FLEE_DELAY] = sConfigMgr->GetOption("CreatureFamilyFleeDelay", 7000); _int_configs[CONFIG_WORLD_BOSS_LEVEL_DIFF] = sConfigMgr->GetOption("WorldBossLevelDiff", 3); _bool_configs[CONFIG_QUEST_ENABLE_QUEST_TRACKER] = sConfigMgr->GetOption("Quests.EnableQuestTracker", false); // note: disable value (-1) will assigned as 0xFFFFFFF, to prevent overflow at calculations limit it to max possible player level MAX_LEVEL(100) _int_configs[CONFIG_QUEST_LOW_LEVEL_HIDE_DIFF] = sConfigMgr->GetOption("Quests.LowLevelHideDiff", 4); if (_int_configs[CONFIG_QUEST_LOW_LEVEL_HIDE_DIFF] > MAX_LEVEL) _int_configs[CONFIG_QUEST_LOW_LEVEL_HIDE_DIFF] = MAX_LEVEL; _int_configs[CONFIG_QUEST_HIGH_LEVEL_HIDE_DIFF] = sConfigMgr->GetOption("Quests.HighLevelHideDiff", 7); if (_int_configs[CONFIG_QUEST_HIGH_LEVEL_HIDE_DIFF] > MAX_LEVEL) _int_configs[CONFIG_QUEST_HIGH_LEVEL_HIDE_DIFF] = MAX_LEVEL; _bool_configs[CONFIG_QUEST_IGNORE_RAID] = sConfigMgr->GetOption("Quests.IgnoreRaid", false); _bool_configs[CONFIG_QUEST_IGNORE_AUTO_ACCEPT] = sConfigMgr->GetOption("Quests.IgnoreAutoAccept", false); _bool_configs[CONFIG_QUEST_IGNORE_AUTO_COMPLETE] = sConfigMgr->GetOption("Quests.IgnoreAutoComplete", false); _int_configs[CONFIG_RANDOM_BG_RESET_HOUR] = sConfigMgr->GetOption("Battleground.Random.ResetHour", 6); if (_int_configs[CONFIG_RANDOM_BG_RESET_HOUR] > 23) { LOG_ERROR("server.loading", "Battleground.Random.ResetHour ({}) can't be load. Set to 6.", _int_configs[CONFIG_RANDOM_BG_RESET_HOUR]); _int_configs[CONFIG_RANDOM_BG_RESET_HOUR] = 6; } _int_configs[CONFIG_CALENDAR_DELETE_OLD_EVENTS_HOUR] = sConfigMgr->GetOption("Calendar.DeleteOldEventsHour", 6); if (_int_configs[CONFIG_CALENDAR_DELETE_OLD_EVENTS_HOUR] > 23 || int32(_int_configs[CONFIG_CALENDAR_DELETE_OLD_EVENTS_HOUR]) < 0) { LOG_ERROR("server.loading", "Calendar.DeleteOldEventsHour ({}) can't be load. Set to 6.", _int_configs[CONFIG_CALENDAR_DELETE_OLD_EVENTS_HOUR]); _int_configs[CONFIG_CALENDAR_DELETE_OLD_EVENTS_HOUR] = 6; } _int_configs[CONFIG_GUILD_RESET_HOUR] = sConfigMgr->GetOption("Guild.ResetHour", 6); if (_int_configs[CONFIG_GUILD_RESET_HOUR] > 23) { LOG_ERROR("server.loading", "Guild.ResetHour ({}) can't be load. Set to 6.", _int_configs[CONFIG_GUILD_RESET_HOUR]); _int_configs[CONFIG_GUILD_RESET_HOUR] = 6; } _int_configs[CONFIG_GUILD_BANK_INITIAL_TABS] = sConfigMgr->GetOption("Guild.BankInitialTabs", 0); _int_configs[CONFIG_GUILD_BANK_TAB_COST_0] = sConfigMgr->GetOption("Guild.BankTabCost0", 1000000); _int_configs[CONFIG_GUILD_BANK_TAB_COST_1] = sConfigMgr->GetOption("Guild.BankTabCost1", 2500000); _int_configs[CONFIG_GUILD_BANK_TAB_COST_2] = sConfigMgr->GetOption("Guild.BankTabCost2", 5000000); _int_configs[CONFIG_GUILD_BANK_TAB_COST_3] = sConfigMgr->GetOption("Guild.BankTabCost3", 10000000); _int_configs[CONFIG_GUILD_BANK_TAB_COST_4] = sConfigMgr->GetOption("Guild.BankTabCost4", 25000000); _int_configs[CONFIG_GUILD_BANK_TAB_COST_5] = sConfigMgr->GetOption("Guild.BankTabCost5", 50000000); _bool_configs[CONFIG_DETECT_POS_COLLISION] = sConfigMgr->GetOption("DetectPosCollision", true); _bool_configs[CONFIG_RESTRICTED_LFG_CHANNEL] = sConfigMgr->GetOption("Channel.RestrictedLfg", true); _bool_configs[CONFIG_SILENTLY_GM_JOIN_TO_CHANNEL] = sConfigMgr->GetOption("Channel.SilentlyGMJoin", false); _bool_configs[CONFIG_TALENTS_INSPECTING] = sConfigMgr->GetOption("TalentsInspecting", true); _bool_configs[CONFIG_CHAT_FAKE_MESSAGE_PREVENTING] = sConfigMgr->GetOption("ChatFakeMessagePreventing", false); _int_configs[CONFIG_CHAT_STRICT_LINK_CHECKING_SEVERITY] = sConfigMgr->GetOption("ChatStrictLinkChecking.Severity", 0); _int_configs[CONFIG_CHAT_STRICT_LINK_CHECKING_KICK] = sConfigMgr->GetOption("ChatStrictLinkChecking.Kick", 0); _int_configs[CONFIG_CORPSE_DECAY_NORMAL] = sConfigMgr->GetOption("Corpse.Decay.NORMAL", 60); _int_configs[CONFIG_CORPSE_DECAY_RARE] = sConfigMgr->GetOption("Corpse.Decay.RARE", 300); _int_configs[CONFIG_CORPSE_DECAY_ELITE] = sConfigMgr->GetOption("Corpse.Decay.ELITE", 300); _int_configs[CONFIG_CORPSE_DECAY_RAREELITE] = sConfigMgr->GetOption("Corpse.Decay.RAREELITE", 300); _int_configs[CONFIG_CORPSE_DECAY_WORLDBOSS] = sConfigMgr->GetOption("Corpse.Decay.WORLDBOSS", 3600); _int_configs[CONFIG_DEATH_SICKNESS_LEVEL] = sConfigMgr->GetOption ("Death.SicknessLevel", 11); _bool_configs[CONFIG_DEATH_CORPSE_RECLAIM_DELAY_PVP] = sConfigMgr->GetOption("Death.CorpseReclaimDelay.PvP", true); _bool_configs[CONFIG_DEATH_CORPSE_RECLAIM_DELAY_PVE] = sConfigMgr->GetOption("Death.CorpseReclaimDelay.PvE", true); _bool_configs[CONFIG_DEATH_BONES_WORLD] = sConfigMgr->GetOption("Death.Bones.World", true); _bool_configs[CONFIG_DEATH_BONES_BG_OR_ARENA] = sConfigMgr->GetOption("Death.Bones.BattlegroundOrArena", true); _bool_configs[CONFIG_DIE_COMMAND_MODE] = sConfigMgr->GetOption("Die.Command.Mode", true); // always use declined names in the russian client _bool_configs[CONFIG_DECLINED_NAMES_USED] = (_int_configs[CONFIG_REALM_ZONE] == REALM_ZONE_RUSSIAN) ? true : sConfigMgr->GetOption("DeclinedNames", false); _float_configs[CONFIG_LISTEN_RANGE_SAY] = sConfigMgr->GetOption("ListenRange.Say", 25.0f); _float_configs[CONFIG_LISTEN_RANGE_TEXTEMOTE] = sConfigMgr->GetOption("ListenRange.TextEmote", 25.0f); _float_configs[CONFIG_LISTEN_RANGE_YELL] = sConfigMgr->GetOption("ListenRange.Yell", 300.0f); _bool_configs[CONFIG_BATTLEGROUND_DISABLE_QUEST_SHARE_IN_BG] = sConfigMgr->GetOption("Battleground.DisableQuestShareInBG", false); _bool_configs[CONFIG_BATTLEGROUND_DISABLE_READY_CHECK_IN_BG] = sConfigMgr->GetOption("Battleground.DisableReadyCheckInBG", false); _bool_configs[CONFIG_BATTLEGROUND_CAST_DESERTER] = sConfigMgr->GetOption("Battleground.CastDeserter", true); _bool_configs[CONFIG_BATTLEGROUND_QUEUE_ANNOUNCER_ENABLE] = sConfigMgr->GetOption("Battleground.QueueAnnouncer.Enable", false); _int_configs[CONFIG_BATTLEGROUND_QUEUE_ANNOUNCER_LIMIT_MIN_LEVEL] = sConfigMgr->GetOption("Battleground.QueueAnnouncer.Limit.MinLevel", 0); _int_configs[CONFIG_BATTLEGROUND_QUEUE_ANNOUNCER_LIMIT_MIN_PLAYERS] = sConfigMgr->GetOption("Battleground.QueueAnnouncer.Limit.MinPlayers", 3); _int_configs[CONFIG_BATTLEGROUND_QUEUE_ANNOUNCER_SPAM_DELAY] = sConfigMgr->GetOption("Battleground.QueueAnnouncer.SpamProtection.Delay", 30); _bool_configs[CONFIG_BATTLEGROUND_QUEUE_ANNOUNCER_PLAYERONLY] = sConfigMgr->GetOption("Battleground.QueueAnnouncer.PlayerOnly", false); _bool_configs[CONFIG_BATTLEGROUND_QUEUE_ANNOUNCER_TIMED] = sConfigMgr->GetOption("Battleground.QueueAnnouncer.Timed", false); _int_configs[CONFIG_BATTLEGROUND_QUEUE_ANNOUNCER_TIMER] = sConfigMgr->GetOption("Battleground.QueueAnnouncer.Timer", 30000); _bool_configs[CONFIG_BATTLEGROUND_STORE_STATISTICS_ENABLE] = sConfigMgr->GetOption("Battleground.StoreStatistics.Enable", false); _bool_configs[CONFIG_BATTLEGROUND_TRACK_DESERTERS] = sConfigMgr->GetOption("Battleground.TrackDeserters.Enable", false); _int_configs[CONFIG_BATTLEGROUND_PREMATURE_FINISH_TIMER] = sConfigMgr->GetOption ("Battleground.PrematureFinishTimer", 5 * MINUTE * IN_MILLISECONDS); _int_configs[CONFIG_BATTLEGROUND_INVITATION_TYPE] = sConfigMgr->GetOption("Battleground.InvitationType", 0); _int_configs[CONFIG_BATTLEGROUND_PREMADE_GROUP_WAIT_FOR_MATCH] = sConfigMgr->GetOption ("Battleground.PremadeGroupWaitForMatch", 30 * MINUTE * IN_MILLISECONDS); _bool_configs[CONFIG_BG_XP_FOR_KILL] = sConfigMgr->GetOption("Battleground.GiveXPForKills", false); _int_configs[CONFIG_BATTLEGROUND_REPORT_AFK_TIMER] = sConfigMgr->GetOption("Battleground.ReportAFK.Timer", 4); _int_configs[CONFIG_BATTLEGROUND_REPORT_AFK] = sConfigMgr->GetOption("Battleground.ReportAFK", 3); if (_int_configs[CONFIG_BATTLEGROUND_REPORT_AFK] < 1) { LOG_ERROR("server.loading", "Battleground.ReportAFK ({}) must be >0. Using 3 instead.", _int_configs[CONFIG_BATTLEGROUND_REPORT_AFK]); _int_configs[CONFIG_BATTLEGROUND_REPORT_AFK] = 3; } else if (_int_configs[CONFIG_BATTLEGROUND_REPORT_AFK] > 9) { LOG_ERROR("server.loading", "Battleground.ReportAFK ({}) must be <10. Using 3 instead.", _int_configs[CONFIG_BATTLEGROUND_REPORT_AFK]); _int_configs[CONFIG_BATTLEGROUND_REPORT_AFK] = 3; } _int_configs[CONFIG_BATTLEGROUND_PLAYER_RESPAWN] = sConfigMgr->GetOption("Battleground.PlayerRespawn", 30); if (_int_configs[CONFIG_BATTLEGROUND_PLAYER_RESPAWN] < 3) { LOG_ERROR("server.loading", "Battleground.PlayerRespawn ({}) must be >2. Using 30 instead.", _int_configs[CONFIG_BATTLEGROUND_PLAYER_RESPAWN]); _int_configs[CONFIG_BATTLEGROUND_PLAYER_RESPAWN] = 30; } _int_configs[CONFIG_BATTLEGROUND_RESTORATION_BUFF_RESPAWN] = sConfigMgr->GetOption("Battleground.RestorationBuffRespawn", 20); if (_int_configs[CONFIG_BATTLEGROUND_RESTORATION_BUFF_RESPAWN] < 1) { LOG_ERROR("server.loading", "Battleground.RestorationBuffRespawn ({}) must be > 0. Using 20 instead.", _int_configs[CONFIG_BATTLEGROUND_RESTORATION_BUFF_RESPAWN]); _int_configs[CONFIG_BATTLEGROUND_RESTORATION_BUFF_RESPAWN] = 20; } _int_configs[CONFIG_BATTLEGROUND_BERSERKING_BUFF_RESPAWN] = sConfigMgr->GetOption("Battleground.BerserkingBuffRespawn", 120); if (_int_configs[CONFIG_BATTLEGROUND_BERSERKING_BUFF_RESPAWN] < 1) { LOG_ERROR("server.loading", "Battleground.BerserkingBuffRespawn ({}) must be > 0. Using 120 instead.", _int_configs[CONFIG_BATTLEGROUND_BERSERKING_BUFF_RESPAWN]); _int_configs[CONFIG_BATTLEGROUND_BERSERKING_BUFF_RESPAWN] = 120; } _int_configs[CONFIG_BATTLEGROUND_SPEED_BUFF_RESPAWN] = sConfigMgr->GetOption("Battleground.SpeedBuffRespawn", 150); if (_int_configs[CONFIG_BATTLEGROUND_SPEED_BUFF_RESPAWN] < 1) { LOG_ERROR("server.loading", "Battleground.SpeedBuffRespawn ({}) must be > 0. Using 150 instead.", _int_configs[CONFIG_BATTLEGROUND_SPEED_BUFF_RESPAWN]); _int_configs[CONFIG_BATTLEGROUND_SPEED_BUFF_RESPAWN] = 150; } _int_configs[CONFIG_ARENA_MAX_RATING_DIFFERENCE] = sConfigMgr->GetOption("Arena.MaxRatingDifference", 150); _int_configs[CONFIG_ARENA_RATING_DISCARD_TIMER] = sConfigMgr->GetOption("Arena.RatingDiscardTimer", 10 * MINUTE * IN_MILLISECONDS); _int_configs[CONFIG_ARENA_PREV_OPPONENTS_DISCARD_TIMER] = sConfigMgr->GetOption("Arena.PreviousOpponentsDiscardTimer", 2 * MINUTE * IN_MILLISECONDS); _bool_configs[CONFIG_ARENA_AUTO_DISTRIBUTE_POINTS] = sConfigMgr->GetOption("Arena.AutoDistributePoints", false); _int_configs[CONFIG_ARENA_AUTO_DISTRIBUTE_INTERVAL_DAYS] = sConfigMgr->GetOption("Arena.AutoDistributeInterval", 7); // pussywizard: spoiled by implementing constant day and hour, always 7 now _int_configs[CONFIG_ARENA_GAMES_REQUIRED] = sConfigMgr->GetOption("Arena.GamesRequired", 10); _int_configs[CONFIG_ARENA_SEASON_ID] = sConfigMgr->GetOption("Arena.ArenaSeason.ID", 1); _int_configs[CONFIG_ARENA_START_RATING] = sConfigMgr->GetOption("Arena.ArenaStartRating", 0); _int_configs[CONFIG_LEGACY_ARENA_POINTS_CALC] = sConfigMgr->GetOption("Arena.LegacyArenaPoints", 0); _int_configs[CONFIG_ARENA_START_PERSONAL_RATING] = sConfigMgr->GetOption("Arena.ArenaStartPersonalRating", 1000); _int_configs[CONFIG_ARENA_START_MATCHMAKER_RATING] = sConfigMgr->GetOption("Arena.ArenaStartMatchmakerRating", 1500); _bool_configs[CONFIG_ARENA_SEASON_IN_PROGRESS] = sConfigMgr->GetOption("Arena.ArenaSeason.InProgress", true); _float_configs[CONFIG_ARENA_WIN_RATING_MODIFIER_1] = sConfigMgr->GetOption("Arena.ArenaWinRatingModifier1", 48.0f); _float_configs[CONFIG_ARENA_WIN_RATING_MODIFIER_2] = sConfigMgr->GetOption("Arena.ArenaWinRatingModifier2", 24.0f); _float_configs[CONFIG_ARENA_LOSE_RATING_MODIFIER] = sConfigMgr->GetOption("Arena.ArenaLoseRatingModifier", 24.0f); _float_configs[CONFIG_ARENA_MATCHMAKER_RATING_MODIFIER] = sConfigMgr->GetOption("Arena.ArenaMatchmakerRatingModifier", 24.0f); _bool_configs[CONFIG_ARENA_QUEUE_ANNOUNCER_ENABLE] = sConfigMgr->GetOption("Arena.QueueAnnouncer.Enable", false); _bool_configs[CONFIG_ARENA_QUEUE_ANNOUNCER_PLAYERONLY] = sConfigMgr->GetOption("Arena.QueueAnnouncer.PlayerOnly", false); _int_configs[CONFIG_ARENA_QUEUE_ANNOUNCER_DETAIL] = sConfigMgr->GetOption("Arena.QueueAnnouncer.Detail", 3); _bool_configs[CONFIG_OFFHAND_CHECK_AT_SPELL_UNLEARN] = sConfigMgr->GetOption("OffhandCheckAtSpellUnlearn", true); _int_configs[CONFIG_CREATURE_STOP_FOR_PLAYER] = sConfigMgr->GetOption("Creature.MovingStopTimeForPlayer", 3 * MINUTE * IN_MILLISECONDS); _int_configs[CONFIG_WATER_BREATH_TIMER] = sConfigMgr->GetOption("WaterBreath.Timer", 180000); if (_int_configs[CONFIG_WATER_BREATH_TIMER] <= 0) { LOG_ERROR("server.loading", "WaterBreath.Timer ({}) must be > 0. Using 180000 instead.", _int_configs[CONFIG_WATER_BREATH_TIMER]); _int_configs[CONFIG_WATER_BREATH_TIMER] = 180000; } if (int32 clientCacheId = sConfigMgr->GetOption("ClientCacheVersion", 0)) { // overwrite DB/old value if (clientCacheId > 0) { _int_configs[CONFIG_CLIENTCACHE_VERSION] = clientCacheId; LOG_INFO("server.loading", "Client cache version set to: {}", clientCacheId); } else LOG_ERROR("server.loading", "ClientCacheVersion can't be negative {}, ignored.", clientCacheId); } _int_configs[CONFIG_INSTANT_LOGOUT] = sConfigMgr->GetOption("InstantLogout", SEC_MODERATOR); _int_configs[CONFIG_GUILD_EVENT_LOG_COUNT] = sConfigMgr->GetOption("Guild.EventLogRecordsCount", GUILD_EVENTLOG_MAX_RECORDS); if (_int_configs[CONFIG_GUILD_EVENT_LOG_COUNT] > GUILD_EVENTLOG_MAX_RECORDS) _int_configs[CONFIG_GUILD_EVENT_LOG_COUNT] = GUILD_EVENTLOG_MAX_RECORDS; _int_configs[CONFIG_GUILD_BANK_EVENT_LOG_COUNT] = sConfigMgr->GetOption("Guild.BankEventLogRecordsCount", GUILD_BANKLOG_MAX_RECORDS); if (_int_configs[CONFIG_GUILD_BANK_EVENT_LOG_COUNT] > GUILD_BANKLOG_MAX_RECORDS) _int_configs[CONFIG_GUILD_BANK_EVENT_LOG_COUNT] = GUILD_BANKLOG_MAX_RECORDS; //visibility on continents _maxVisibleDistanceOnContinents = sConfigMgr->GetOption("Visibility.Distance.Continents", DEFAULT_VISIBILITY_DISTANCE); if (_maxVisibleDistanceOnContinents < 45 * sWorld->getRate(RATE_CREATURE_AGGRO)) { LOG_ERROR("server.loading", "Visibility.Distance.Continents can't be less max aggro radius {}", 45 * sWorld->getRate(RATE_CREATURE_AGGRO)); _maxVisibleDistanceOnContinents = 45 * sWorld->getRate(RATE_CREATURE_AGGRO); } else if (_maxVisibleDistanceOnContinents > MAX_VISIBILITY_DISTANCE) { LOG_ERROR("server.loading", "Visibility.Distance.Continents can't be greater {}", MAX_VISIBILITY_DISTANCE); _maxVisibleDistanceOnContinents = MAX_VISIBILITY_DISTANCE; } //visibility in instances _maxVisibleDistanceInInstances = sConfigMgr->GetOption("Visibility.Distance.Instances", DEFAULT_VISIBILITY_INSTANCE); if (_maxVisibleDistanceInInstances < 45 * sWorld->getRate(RATE_CREATURE_AGGRO)) { LOG_ERROR("server.loading", "Visibility.Distance.Instances can't be less max aggro radius {}", 45 * sWorld->getRate(RATE_CREATURE_AGGRO)); _maxVisibleDistanceInInstances = 45 * sWorld->getRate(RATE_CREATURE_AGGRO); } else if (_maxVisibleDistanceInInstances > MAX_VISIBILITY_DISTANCE) { LOG_ERROR("server.loading", "Visibility.Distance.Instances can't be greater {}", MAX_VISIBILITY_DISTANCE); _maxVisibleDistanceInInstances = MAX_VISIBILITY_DISTANCE; } //visibility in BG/Arenas _maxVisibleDistanceInBGArenas = sConfigMgr->GetOption("Visibility.Distance.BGArenas", DEFAULT_VISIBILITY_BGARENAS); if (_maxVisibleDistanceInBGArenas < 45 * sWorld->getRate(RATE_CREATURE_AGGRO)) { LOG_ERROR("server.loading", "Visibility.Distance.BGArenas can't be less max aggro radius {}", 45 * sWorld->getRate(RATE_CREATURE_AGGRO)); _maxVisibleDistanceInBGArenas = 45 * sWorld->getRate(RATE_CREATURE_AGGRO); } else if (_maxVisibleDistanceInBGArenas > MAX_VISIBILITY_DISTANCE) { LOG_ERROR("server.loading", "Visibility.Distance.BGArenas can't be greater {}", MAX_VISIBILITY_DISTANCE); _maxVisibleDistanceInBGArenas = MAX_VISIBILITY_DISTANCE; } m_visibility_notify_periodOnContinents = sConfigMgr->GetOption("Visibility.Notify.Period.OnContinents", DEFAULT_VISIBILITY_NOTIFY_PERIOD); m_visibility_notify_periodInInstances = sConfigMgr->GetOption("Visibility.Notify.Period.InInstances", DEFAULT_VISIBILITY_NOTIFY_PERIOD); m_visibility_notify_periodInBGArenas = sConfigMgr->GetOption("Visibility.Notify.Period.InBGArenas", DEFAULT_VISIBILITY_NOTIFY_PERIOD); ///- Load the CharDelete related config options _int_configs[CONFIG_CHARDELETE_METHOD] = sConfigMgr->GetOption("CharDelete.Method", 0); _int_configs[CONFIG_CHARDELETE_MIN_LEVEL] = sConfigMgr->GetOption("CharDelete.MinLevel", 0); _int_configs[CONFIG_CHARDELETE_KEEP_DAYS] = sConfigMgr->GetOption("CharDelete.KeepDays", 30); ///- Load the ItemDelete related config options _bool_configs[CONFIG_ITEMDELETE_METHOD] = sConfigMgr->GetOption("ItemDelete.Method", 0); _bool_configs[CONFIG_ITEMDELETE_VENDOR] = sConfigMgr->GetOption("ItemDelete.Vendor", 0); _int_configs[CONFIG_ITEMDELETE_QUALITY] = sConfigMgr->GetOption("ItemDelete.Quality", 3); _int_configs[CONFIG_ITEMDELETE_ITEM_LEVEL] = sConfigMgr->GetOption("ItemDelete.ItemLevel", 80); _int_configs[CONFIG_FFA_PVP_TIMER] = sConfigMgr->GetOption("FFAPvPTimer", 30); _int_configs[CONFIG_LOOT_NEED_BEFORE_GREED_ILVL_RESTRICTION] = sConfigMgr->GetOption("LootNeedBeforeGreedILvlRestriction", 70); _bool_configs[CONFIG_PLAYER_SETTINGS_ENABLED] = sConfigMgr->GetOption("EnablePlayerSettings", 0); _bool_configs[CONFIG_ALLOW_JOIN_BG_AND_LFG] = sConfigMgr->GetOption("JoinBGAndLFG.Enable", false); _bool_configs[CONFIG_LEAVE_GROUP_ON_LOGOUT] = sConfigMgr->GetOption("LeaveGroupOnLogout.Enabled", true); _bool_configs[CONFIG_QUEST_POI_ENABLED] = sConfigMgr->GetOption("QuestPOI.Enabled", true); _int_configs[CONFIG_CHANGE_FACTION_MAX_MONEY] = sConfigMgr->GetOption("ChangeFaction.MaxMoney", 0); _bool_configs[CONFIG_ALLOWS_RANK_MOD_FOR_PET_HEALTH] = sConfigMgr->GetOption("Pet.RankMod.Health", true); _int_configs[CONFIG_AUCTION_HOUSE_SEARCH_TIMEOUT] = sConfigMgr->GetOption("AuctionHouse.SearchTimeout", 1000); ///- Read the "Data" directory from the config file std::string dataPath = sConfigMgr->GetOption("DataDir", "./"); if (dataPath.empty() || (dataPath.at(dataPath.length() - 1) != '/' && dataPath.at(dataPath.length() - 1) != '\\')) dataPath.push_back('/'); #if AC_PLATFORM == AC_PLATFORM_UNIX || AC_PLATFORM == AC_PLATFORM_APPLE if (dataPath[0] == '~') { const char* home = getenv("HOME"); if (home) dataPath.replace(0, 1, home); } #endif if (reload) { if (dataPath != _dataPath) LOG_ERROR("server.loading", "DataDir option can't be changed at worldserver.conf reload, using current value ({}).", _dataPath); } else { _dataPath = dataPath; LOG_INFO("server.loading", "Using DataDir {}", _dataPath); } _bool_configs[CONFIG_VMAP_INDOOR_CHECK] = sConfigMgr->GetOption("vmap.enableIndoorCheck", 0); bool enableIndoor = sConfigMgr->GetOption("vmap.enableIndoorCheck", true); bool enableLOS = sConfigMgr->GetOption("vmap.enableLOS", true); bool enableHeight = sConfigMgr->GetOption("vmap.enableHeight", true); bool enablePetLOS = sConfigMgr->GetOption("vmap.petLOS", true); _bool_configs[CONFIG_VMAP_BLIZZLIKE_PVP_LOS] = sConfigMgr->GetOption("vmap.BlizzlikePvPLOS", true); if (!enableHeight) LOG_ERROR("server.loading", "VMap height checking disabled! Creatures movements and other various things WILL be broken! Expect no support."); VMAP::VMapFactory::createOrGetVMapMgr()->setEnableLineOfSightCalc(enableLOS); VMAP::VMapFactory::createOrGetVMapMgr()->setEnableHeightCalc(enableHeight); LOG_INFO("server.loading", "WORLD: VMap support included. LineOfSight:{}, getHeight:{}, indoorCheck:{} PetLOS:{}", enableLOS, enableHeight, enableIndoor, enablePetLOS); _bool_configs[CONFIG_PET_LOS] = sConfigMgr->GetOption("vmap.petLOS", true); _bool_configs[CONFIG_START_CUSTOM_SPELLS] = sConfigMgr->GetOption("PlayerStart.CustomSpells", false); _int_configs[CONFIG_HONOR_AFTER_DUEL] = sConfigMgr->GetOption("HonorPointsAfterDuel", 0); _bool_configs[CONFIG_START_ALL_EXPLORED] = sConfigMgr->GetOption("PlayerStart.MapsExplored", false); _bool_configs[CONFIG_START_ALL_REP] = sConfigMgr->GetOption("PlayerStart.AllReputation", false); _bool_configs[CONFIG_ALWAYS_MAXSKILL] = sConfigMgr->GetOption("AlwaysMaxWeaponSkill", false); _bool_configs[CONFIG_PVP_TOKEN_ENABLE] = sConfigMgr->GetOption("PvPToken.Enable", false); _int_configs[CONFIG_PVP_TOKEN_MAP_TYPE] = sConfigMgr->GetOption("PvPToken.MapAllowType", 4); _int_configs[CONFIG_PVP_TOKEN_ID] = sConfigMgr->GetOption("PvPToken.ItemID", 29434); _int_configs[CONFIG_PVP_TOKEN_COUNT] = sConfigMgr->GetOption("PvPToken.ItemCount", 1); if (_int_configs[CONFIG_PVP_TOKEN_COUNT] < 1) _int_configs[CONFIG_PVP_TOKEN_COUNT] = 1; _bool_configs[CONFIG_NO_RESET_TALENT_COST] = sConfigMgr->GetOption("NoResetTalentsCost", false); _int_configs[CONFIG_TOGGLE_XP_COST] = sConfigMgr->GetOption("ToggleXP.Cost", 100000); _bool_configs[CONFIG_SHOW_KICK_IN_WORLD] = sConfigMgr->GetOption("ShowKickInWorld", false); _bool_configs[CONFIG_SHOW_MUTE_IN_WORLD] = sConfigMgr->GetOption("ShowMuteInWorld", false); _bool_configs[CONFIG_SHOW_BAN_IN_WORLD] = sConfigMgr->GetOption("ShowBanInWorld", false); _int_configs[CONFIG_NUMTHREADS] = sConfigMgr->GetOption("MapUpdate.Threads", 1); _int_configs[CONFIG_MAX_RESULTS_LOOKUP_COMMANDS] = sConfigMgr->GetOption("Command.LookupMaxResults", 0); // Warden _bool_configs[CONFIG_WARDEN_ENABLED] = sConfigMgr->GetOption("Warden.Enabled", true); _int_configs[CONFIG_WARDEN_NUM_MEM_CHECKS] = sConfigMgr->GetOption("Warden.NumMemChecks", 3); _int_configs[CONFIG_WARDEN_NUM_LUA_CHECKS] = sConfigMgr->GetOption("Warden.NumLuaChecks", 1); _int_configs[CONFIG_WARDEN_NUM_OTHER_CHECKS] = sConfigMgr->GetOption("Warden.NumOtherChecks", 7); _int_configs[CONFIG_WARDEN_CLIENT_BAN_DURATION] = sConfigMgr->GetOption("Warden.BanDuration", 86400); _int_configs[CONFIG_WARDEN_CLIENT_CHECK_HOLDOFF] = sConfigMgr->GetOption("Warden.ClientCheckHoldOff", 30); _int_configs[CONFIG_WARDEN_CLIENT_FAIL_ACTION] = sConfigMgr->GetOption("Warden.ClientCheckFailAction", 0); _int_configs[CONFIG_WARDEN_CLIENT_RESPONSE_DELAY] = sConfigMgr->GetOption("Warden.ClientResponseDelay", 600); // Dungeon finder _int_configs[CONFIG_LFG_OPTIONSMASK] = sConfigMgr->GetOption("DungeonFinder.OptionsMask", 5); // DBC_ItemAttributes _bool_configs[CONFIG_DBC_ENFORCE_ITEM_ATTRIBUTES] = sConfigMgr->GetOption("DBC.EnforceItemAttributes", true); // Max instances per hour _int_configs[CONFIG_MAX_INSTANCES_PER_HOUR] = sConfigMgr->GetOption("AccountInstancesPerHour", 5); // AutoBroadcast _bool_configs[CONFIG_AUTOBROADCAST] = sConfigMgr->GetOption("AutoBroadcast.On", false); _int_configs[CONFIG_AUTOBROADCAST_CENTER] = sConfigMgr->GetOption("AutoBroadcast.Center", 0); _int_configs[CONFIG_AUTOBROADCAST_INTERVAL] = sConfigMgr->GetOption("AutoBroadcast.Timer", 60000); _int_configs[CONFIG_AUTOBROADCAST_MIN_LEVEL_DISABLE] = sConfigMgr->GetOption("AutoBroadcast.MinDisableLevel", 0); if (reload) { _timers[WUPDATE_AUTOBROADCAST].SetInterval(_int_configs[CONFIG_AUTOBROADCAST_INTERVAL]); _timers[WUPDATE_AUTOBROADCAST].Reset(); } // MySQL ping time interval _int_configs[CONFIG_DB_PING_INTERVAL] = sConfigMgr->GetOption("MaxPingTime", 30); // misc _bool_configs[CONFIG_PDUMP_NO_PATHS] = sConfigMgr->GetOption("PlayerDump.DisallowPaths", true); _bool_configs[CONFIG_PDUMP_NO_OVERWRITE] = sConfigMgr->GetOption("PlayerDump.DisallowOverwrite", true); _bool_configs[CONFIG_ENABLE_MMAPS] = sConfigMgr->GetOption("MoveMaps.Enable", true); MMAP::MMapFactory::InitializeDisabledMaps(); // Wintergrasp _int_configs[CONFIG_WINTERGRASP_ENABLE] = sConfigMgr->GetOption("Wintergrasp.Enable", 1); _int_configs[CONFIG_WINTERGRASP_PLR_MAX] = sConfigMgr->GetOption("Wintergrasp.PlayerMax", 100); _int_configs[CONFIG_WINTERGRASP_PLR_MIN] = sConfigMgr->GetOption("Wintergrasp.PlayerMin", 0); _int_configs[CONFIG_WINTERGRASP_PLR_MIN_LVL] = sConfigMgr->GetOption("Wintergrasp.PlayerMinLvl", 77); _int_configs[CONFIG_WINTERGRASP_BATTLETIME] = sConfigMgr->GetOption("Wintergrasp.BattleTimer", 30); _int_configs[CONFIG_WINTERGRASP_NOBATTLETIME] = sConfigMgr->GetOption("Wintergrasp.NoBattleTimer", 150); _int_configs[CONFIG_WINTERGRASP_RESTART_AFTER_CRASH] = sConfigMgr->GetOption("Wintergrasp.CrashRestartTimer", 10); _int_configs[CONFIG_BIRTHDAY_TIME] = sConfigMgr->GetOption("BirthdayTime", 1222964635); _bool_configs[CONFIG_MINIGOB_MANABONK] = sConfigMgr->GetOption("Minigob.Manabonk.Enable", true); _bool_configs[CONFIG_ENABLE_CONTINENT_TRANSPORT] = sConfigMgr->GetOption("IsContinentTransport.Enabled", true); _bool_configs[CONFIG_ENABLE_CONTINENT_TRANSPORT_PRELOADING] = sConfigMgr->GetOption("IsPreloadedContinentTransport.Enabled", false); _bool_configs[CONFIG_IP_BASED_ACTION_LOGGING] = sConfigMgr->GetOption("Allow.IP.Based.Action.Logging", false); // Whether to use LoS from game objects _bool_configs[CONFIG_CHECK_GOBJECT_LOS] = sConfigMgr->GetOption("CheckGameObjectLoS", true); _bool_configs[CONFIG_CALCULATE_CREATURE_ZONE_AREA_DATA] = sConfigMgr->GetOption("Calculate.Creature.Zone.Area.Data", false); _bool_configs[CONFIG_CALCULATE_GAMEOBJECT_ZONE_AREA_DATA] = sConfigMgr->GetOption("Calculate.Gameoject.Zone.Area.Data", false); // Player can join LFG anywhere _bool_configs[CONFIG_LFG_LOCATION_ALL] = sConfigMgr->GetOption("LFG.Location.All", false); // Prevent players AFK from being logged out _int_configs[CONFIG_AFK_PREVENT_LOGOUT] = sConfigMgr->GetOption("PreventAFKLogout", 0); // Preload all grids of all non-instanced maps _bool_configs[CONFIG_PRELOAD_ALL_NON_INSTANCED_MAP_GRIDS] = sConfigMgr->GetOption("PreloadAllNonInstancedMapGrids", false); // ICC buff override _int_configs[CONFIG_ICC_BUFF_HORDE] = sConfigMgr->GetOption("ICC.Buff.Horde", 73822); _int_configs[CONFIG_ICC_BUFF_ALLIANCE] = sConfigMgr->GetOption("ICC.Buff.Alliance", 73828); _bool_configs[CONFIG_SET_ALL_CREATURES_WITH_WAYPOINT_MOVEMENT_ACTIVE] = sConfigMgr->GetOption("SetAllCreaturesWithWaypointMovementActive", false); // packet spoof punishment _int_configs[CONFIG_PACKET_SPOOF_POLICY] = sConfigMgr->GetOption("PacketSpoof.Policy", (uint32)WorldSession::DosProtection::POLICY_KICK); _int_configs[CONFIG_PACKET_SPOOF_BANMODE] = sConfigMgr->GetOption("PacketSpoof.BanMode", (uint32)0); if (_int_configs[CONFIG_PACKET_SPOOF_BANMODE] > 1) _int_configs[CONFIG_PACKET_SPOOF_BANMODE] = (uint32)0; _int_configs[CONFIG_PACKET_SPOOF_BANDURATION] = sConfigMgr->GetOption("PacketSpoof.BanDuration", 86400); // Random Battleground Rewards _int_configs[CONFIG_BG_REWARD_WINNER_HONOR_FIRST] = sConfigMgr->GetOption("Battleground.RewardWinnerHonorFirst", 30); _int_configs[CONFIG_BG_REWARD_WINNER_ARENA_FIRST] = sConfigMgr->GetOption("Battleground.RewardWinnerArenaFirst", 25); _int_configs[CONFIG_BG_REWARD_WINNER_HONOR_LAST] = sConfigMgr->GetOption("Battleground.RewardWinnerHonorLast", 15); _int_configs[CONFIG_BG_REWARD_WINNER_ARENA_LAST] = sConfigMgr->GetOption("Battleground.RewardWinnerArenaLast", 0); _int_configs[CONFIG_BG_REWARD_LOSER_HONOR_FIRST] = sConfigMgr->GetOption("Battleground.RewardLoserHonorFirst", 5); _int_configs[CONFIG_BG_REWARD_LOSER_HONOR_LAST] = sConfigMgr->GetOption("Battleground.RewardLoserHonorLast", 5); _int_configs[CONFIG_WAYPOINT_MOVEMENT_STOP_TIME_FOR_PLAYER] = sConfigMgr->GetOption("WaypointMovementStopTimeForPlayer", 120); _int_configs[CONFIG_DUNGEON_ACCESS_REQUIREMENTS_PRINT_MODE] = sConfigMgr->GetOption("DungeonAccessRequirements.PrintMode", 1); _bool_configs[CONFIG_DUNGEON_ACCESS_REQUIREMENTS_PORTAL_CHECK_ILVL] = sConfigMgr->GetOption("DungeonAccessRequirements.PortalAvgIlevelCheck", false); _bool_configs[CONFIG_DUNGEON_ACCESS_REQUIREMENTS_LFG_DBC_LEVEL_OVERRIDE] = sConfigMgr->GetOption("DungeonAccessRequirements.LFGLevelDBCOverride", false); _int_configs[CONFIG_DUNGEON_ACCESS_REQUIREMENTS_OPTIONAL_STRING_ID] = sConfigMgr->GetOption("DungeonAccessRequirements.OptionalStringID", 0); _int_configs[CONFIG_NPC_EVADE_IF_NOT_REACHABLE] = sConfigMgr->GetOption("NpcEvadeIfTargetIsUnreachable", 5); _int_configs[CONFIG_NPC_REGEN_TIME_IF_NOT_REACHABLE_IN_RAID] = sConfigMgr->GetOption("NpcRegenHPTimeIfTargetIsUnreachable", 10); _bool_configs[CONFIG_REGEN_HP_CANNOT_REACH_TARGET_IN_RAID] = sConfigMgr->GetOption("NpcRegenHPIfTargetIsUnreachable", true); //Debug _bool_configs[CONFIG_DEBUG_BATTLEGROUND] = sConfigMgr->GetOption("Debug.Battleground", false); _bool_configs[CONFIG_DEBUG_ARENA] = sConfigMgr->GetOption("Debug.Arena", false); _int_configs[CONFIG_GM_LEVEL_CHANNEL_MODERATION] = sConfigMgr->GetOption("Channel.ModerationGMLevel", 1); _bool_configs[CONFIG_SET_BOP_ITEM_TRADEABLE] = sConfigMgr->GetOption("Item.SetItemTradeable", true); // Specifies if IP addresses can be logged to the database _bool_configs[CONFIG_ALLOW_LOGGING_IP_ADDRESSES_IN_DATABASE] = sConfigMgr->GetOption("AllowLoggingIPAddressesInDatabase", true, true); // LFG group mechanics. _int_configs[CONFIG_LFG_MAX_KICK_COUNT] = sConfigMgr->GetOption("LFG.MaxKickCount", 2); if (_int_configs[CONFIG_LFG_MAX_KICK_COUNT] > 3) { _int_configs[CONFIG_LFG_MAX_KICK_COUNT] = 3; LOG_ERROR("server.loading", "LFG.MaxKickCount can't be higher than 3."); } _int_configs[CONFIG_LFG_KICK_PREVENTION_TIMER] = sConfigMgr->GetOption("LFG.KickPreventionTimer", 15 * MINUTE * IN_MILLISECONDS) * IN_MILLISECONDS; if (_int_configs[CONFIG_LFG_KICK_PREVENTION_TIMER] > 15 * MINUTE * IN_MILLISECONDS) { _int_configs[CONFIG_LFG_KICK_PREVENTION_TIMER] = 15 * MINUTE * IN_MILLISECONDS; LOG_ERROR("server.loading", "LFG.KickPreventionTimer can't be higher than 15 minutes."); } // Realm Availability _bool_configs[CONFIG_REALM_LOGIN_ENABLED] = sConfigMgr->GetOption("World.RealmAvailability", true); // call ScriptMgr if we're reloading the configuration sScriptMgr->OnAfterConfigLoad(reload); } /// Initialize the World void World::SetInitialWorldSettings() { ///- Server startup begin uint32 startupBegin = getMSTime(); ///- Initialize the random number generator srand((unsigned int)GameTime::GetGameTime().count()); ///- Initialize detour memory management dtAllocSetCustom(dtCustomAlloc, dtCustomFree); ///- Initialize VMapMgr function pointers (to untangle game/collision circular deps) VMAP::VMapMgr2* vmmgr2 = VMAP::VMapFactory::createOrGetVMapMgr(); vmmgr2->GetLiquidFlagsPtr = &GetLiquidFlags; vmmgr2->IsVMAPDisabledForPtr = &DisableMgr::IsVMAPDisabledFor; ///- Initialize config settings LoadConfigSettings(); ///- Initialize Allowed Security Level LoadDBAllowedSecurityLevel(); ///- Init highest guids before any table loading to prevent using not initialized guids in some code. sObjectMgr->SetHighestGuids(); if (!sConfigMgr->isDryRun()) { ///- Check the existence of the map files for all starting areas. if (!MapMgr::ExistMapAndVMap(0, -6240.32f, 331.033f) || !MapMgr::ExistMapAndVMap(0, -8949.95f, -132.493f) || !MapMgr::ExistMapAndVMap(1, -618.518f, -4251.67f) || !MapMgr::ExistMapAndVMap(0, 1676.35f, 1677.45f) || !MapMgr::ExistMapAndVMap(1, 10311.3f, 832.463f) || !MapMgr::ExistMapAndVMap(1, -2917.58f, -257.98f) || (_int_configs[CONFIG_EXPANSION] && ( !MapMgr::ExistMapAndVMap(530, 10349.6f, -6357.29f) || !MapMgr::ExistMapAndVMap(530, -3961.64f, -13931.2f)))) { exit(1); } } ///- Initialize pool manager sPoolMgr->Initialize(); ///- Initialize game event manager sGameEventMgr->Initialize(); ///- Loading strings. Getting no records means core load has to be canceled because no error message can be output. LOG_INFO("server.loading", " "); LOG_INFO("server.loading", "Loading Acore Strings..."); if (!sObjectMgr->LoadAcoreStrings()) exit(1); // Error message displayed in function already ///- Update the realm entry in the database with the realm type from the config file //No SQL injection as values are treated as integers // not send custom type REALM_FFA_PVP to realm list uint32 server_type; if (IsFFAPvPRealm()) server_type = REALM_TYPE_PVP; else server_type = getIntConfig(CONFIG_GAME_TYPE); uint32 realm_zone = getIntConfig(CONFIG_REALM_ZONE); LoginDatabase.Execute("UPDATE realmlist SET icon = {}, timezone = {} WHERE id = '{}'", server_type, realm_zone, realm.Id.Realm); // One-time query ///- Custom Hook for loading DB items sScriptMgr->OnLoadCustomDatabaseTable(); ///- Load the DBC files LOG_INFO("server.loading", "Initialize Data Stores..."); LoadDBCStores(_dataPath); DetectDBCLang(); // Load cinematic cameras LoadM2Cameras(_dataPath); // Load IP Location Database sIPLocation->Load(); std::vector mapIds; for (auto const& map : sMapStore) { mapIds.emplace_back(map->MapID); } vmmgr2->InitializeThreadUnsafe(mapIds); MMAP::MMapMgr* mmmgr = MMAP::MMapFactory::createOrGetMMapMgr(); mmmgr->InitializeThreadUnsafe(mapIds); LOG_INFO("server.loading", "Loading Game Graveyard..."); sGraveyard->LoadGraveyardFromDB(); LOG_INFO("server.loading", "Initializing PlayerDump Tables..."); PlayerDump::InitializeTables(); ///- Initilize static helper structures AIRegistry::Initialize(); LOG_INFO("server.loading", "Loading SpellInfo Store..."); sSpellMgr->LoadSpellInfoStore(); LOG_INFO("server.loading", "Loading Spell Cooldown Overrides..."); sSpellMgr->LoadSpellCooldownOverrides(); LOG_INFO("server.loading", "Loading SpellInfo Data Corrections..."); sSpellMgr->LoadSpellInfoCorrections(); LOG_INFO("server.loading", "Loading Spell Rank Data..."); sSpellMgr->LoadSpellRanks(); LOG_INFO("server.loading", "Loading Spell Specific And Aura State..."); sSpellMgr->LoadSpellSpecificAndAuraState(); LOG_INFO("server.loading", "Loading SkillLineAbilityMultiMap Data..."); sSpellMgr->LoadSkillLineAbilityMap(); LOG_INFO("server.loading", "Loading SpellInfo Custom Attributes..."); sSpellMgr->LoadSpellInfoCustomAttributes(); LOG_INFO("server.loading", "Loading GameObject Models..."); LoadGameObjectModelList(_dataPath); LOG_INFO("server.loading", "Loading Script Names..."); sObjectMgr->LoadScriptNames(); LOG_INFO("server.loading", "Loading Instance Template..."); sObjectMgr->LoadInstanceTemplate(); LOG_INFO("server.loading", "Loading Instance Saved Gameobject State Data..."); sObjectMgr->LoadInstanceSavedGameobjectStateData(); LOG_INFO("server.loading", "Loading Character Cache..."); sCharacterCache->LoadCharacterCacheStorage(); // Must be called before `creature_respawn`/`gameobject_respawn` tables LOG_INFO("server.loading", "Loading Instances..."); sInstanceSaveMgr->LoadInstances(); LOG_INFO("server.loading", "Loading Broadcast Texts..."); sObjectMgr->LoadBroadcastTexts(); sObjectMgr->LoadBroadcastTextLocales(); LOG_INFO("server.loading", "Loading Localization Strings..."); uint32 oldMSTime = getMSTime(); sObjectMgr->LoadCreatureLocales(); sObjectMgr->LoadGameObjectLocales(); sObjectMgr->LoadItemLocales(); sObjectMgr->LoadItemSetNameLocales(); sObjectMgr->LoadQuestLocales(); sObjectMgr->LoadQuestOfferRewardLocale(); sObjectMgr->LoadQuestRequestItemsLocale(); sObjectMgr->LoadNpcTextLocales(); sObjectMgr->LoadPageTextLocales(); sObjectMgr->LoadGossipMenuItemsLocales(); sObjectMgr->LoadPointOfInterestLocales(); sObjectMgr->LoadPetNamesLocales(); sObjectMgr->SetDBCLocaleIndex(GetDefaultDbcLocale()); // Get once for all the locale index of DBC language (console/broadcasts) LOG_INFO("server.loading", ">> Localization Strings loaded in {} ms", GetMSTimeDiffToNow(oldMSTime)); LOG_INFO("server.loading", " "); LOG_INFO("server.loading", "Loading Page Texts..."); sObjectMgr->LoadPageTexts(); LOG_INFO("server.loading", "Loading Game Object Templates..."); // must be after LoadPageTexts sObjectMgr->LoadGameObjectTemplate(); LOG_INFO("server.loading", "Loading Game Object Template Addons..."); sObjectMgr->LoadGameObjectTemplateAddons(); LOG_INFO("server.loading", "Loading Transport Templates..."); sTransportMgr->LoadTransportTemplates(); LOG_INFO("server.loading", "Loading Spell Required Data..."); sSpellMgr->LoadSpellRequired(); LOG_INFO("server.loading", "Loading Spell Group Types..."); sSpellMgr->LoadSpellGroups(); LOG_INFO("server.loading", "Loading Spell Learn Skills..."); sSpellMgr->LoadSpellLearnSkills(); // must be after LoadSpellRanks LOG_INFO("server.loading", "Loading Spell Proc Event Conditions..."); sSpellMgr->LoadSpellProcEvents(); LOG_INFO("server.loading", "Loading Spell Proc Conditions and Data..."); sSpellMgr->LoadSpellProcs(); LOG_INFO("server.loading", "Loading Spell Bonus Data..."); sSpellMgr->LoadSpellBonuses(); LOG_INFO("server.loading", "Loading Aggro Spells Definitions..."); sSpellMgr->LoadSpellThreats(); LOG_INFO("server.loading", "Loading Mixology Bonuses..."); sSpellMgr->LoadSpellMixology(); LOG_INFO("server.loading", "Loading Spell Group Stack Rules..."); sSpellMgr->LoadSpellGroupStackRules(); LOG_INFO("server.loading", "Loading NPC Texts..."); sObjectMgr->LoadGossipText(); LOG_INFO("server.loading", "Loading Enchant Spells Proc Datas..."); sSpellMgr->LoadSpellEnchantProcData(); LOG_INFO("server.loading", "Loading Item Random Enchantments Table..."); LoadRandomEnchantmentsTable(); LOG_INFO("server.loading", "Loading Disables"); DisableMgr::LoadDisables(); // must be before loading quests and items LOG_INFO("server.loading", "Loading Items..."); // must be after LoadRandomEnchantmentsTable and LoadPageTexts sObjectMgr->LoadItemTemplates(); LOG_INFO("server.loading", "Loading Item Set Names..."); // must be after LoadItemPrototypes sObjectMgr->LoadItemSetNames(); LOG_INFO("server.loading", "Loading Creature Model Based Info Data..."); sObjectMgr->LoadCreatureModelInfo(); LOG_INFO("server.loading", "Loading Creature Custom IDs Config..."); sObjectMgr->LoadCreatureCustomIDs(); LOG_INFO("server.loading", "Loading Creature Templates..."); sObjectMgr->LoadCreatureTemplates(); LOG_INFO("server.loading", "Loading Equipment Templates..."); // must be after LoadCreatureTemplates sObjectMgr->LoadEquipmentTemplates(); LOG_INFO("server.loading", "Loading Creature Template Addons..."); sObjectMgr->LoadCreatureTemplateAddons(); LOG_INFO("server.loading", "Loading Reputation Reward Rates..."); sObjectMgr->LoadReputationRewardRate(); LOG_INFO("server.loading", "Loading Creature Reputation OnKill Data..."); sObjectMgr->LoadReputationOnKill(); LOG_INFO("server.loading", "Loading Reputation Spillover Data..." ); sObjectMgr->LoadReputationSpilloverTemplate(); LOG_INFO("server.loading", "Loading Points Of Interest Data..."); sObjectMgr->LoadPointsOfInterest(); LOG_INFO("server.loading", "Loading Creature Base Stats..."); sObjectMgr->LoadCreatureClassLevelStats(); LOG_INFO("server.loading", "Loading Creature Data..."); sObjectMgr->LoadCreatures(); LOG_INFO("server.loading", "Loading Temporary Summon Data..."); sObjectMgr->LoadTempSummons(); // must be after LoadCreatureTemplates() and LoadGameObjectTemplates() LOG_INFO("server.loading", "Loading Pet Levelup Spells..."); sSpellMgr->LoadPetLevelupSpellMap(); LOG_INFO("server.loading", "Loading Pet default Spells additional to Levelup Spells..."); sSpellMgr->LoadPetDefaultSpells(); LOG_INFO("server.loading", "Loading Creature Addon Data..."); sObjectMgr->LoadCreatureAddons(); // must be after LoadCreatureTemplates() and LoadCreatures() LOG_INFO("server.loading", "Loading Creature Movement Overrides..."); sObjectMgr->LoadCreatureMovementOverrides(); // must be after LoadCreatures() LOG_INFO("server.loading", "Loading Gameobject Data..."); sObjectMgr->LoadGameobjects(); LOG_INFO("server.loading", "Loading GameObject Addon Data..."); sObjectMgr->LoadGameObjectAddons(); // must be after LoadGameObjectTemplate() and LoadGameobjects() LOG_INFO("server.loading", "Loading GameObject Quest Items..."); sObjectMgr->LoadGameObjectQuestItems(); LOG_INFO("server.loading", "Loading Creature Quest Items..."); sObjectMgr->LoadCreatureQuestItems(); LOG_INFO("server.loading", "Loading Creature Linked Respawn..."); sObjectMgr->LoadLinkedRespawn(); // must be after LoadCreatures(), LoadGameObjects() LOG_INFO("server.loading", "Loading Weather Data..."); WeatherMgr::LoadWeatherData(); LOG_INFO("server.loading", "Loading Quests..."); sObjectMgr->LoadQuests(); // must be loaded after DBCs, creature_template, item_template, gameobject tables LOG_INFO("server.loading", "Checking Quest Disables"); DisableMgr::CheckQuestDisables(); // must be after loading quests LOG_INFO("server.loading", "Loading Quest POI"); sObjectMgr->LoadQuestPOI(); LOG_INFO("server.loading", "Loading Quests Starters and Enders..."); sObjectMgr->LoadQuestStartersAndEnders(); // must be after quest load LOG_INFO("server.loading", "Loading Quest Greetings..."); sObjectMgr->LoadQuestGreetings(); // must be loaded after creature_template, gameobject_template tables LOG_INFO("server.loading", "Loading Quest Greeting Locales..."); sObjectMgr->LoadQuestGreetingsLocales(); // must be loaded after creature_template, gameobject_template tables LOG_INFO("server.loading", "Loading Quest Money Rewards..."); sObjectMgr->LoadQuestMoneyRewards(); LOG_INFO("server.loading", "Loading Objects Pooling Data..."); sPoolMgr->LoadFromDB(); LOG_INFO("server.loading", "Loading Game Event Data..."); // must be after loading pools fully sGameEventMgr->LoadHolidayDates(); // Must be after loading DBC sGameEventMgr->LoadFromDB(); // Must be after loading holiday dates LOG_INFO("server.loading", "Loading UNIT_NPC_FLAG_SPELLCLICK Data..."); // must be after LoadQuests sObjectMgr->LoadNPCSpellClickSpells(); LOG_INFO("server.loading", "Loading Vehicle Template Accessories..."); sObjectMgr->LoadVehicleTemplateAccessories(); // must be after LoadCreatureTemplates() and LoadNPCSpellClickSpells() LOG_INFO("server.loading", "Loading Vehicle Accessories..."); sObjectMgr->LoadVehicleAccessories(); // must be after LoadCreatureTemplates() and LoadNPCSpellClickSpells() LOG_INFO("server.loading", "Loading SpellArea Data..."); // must be after quest load sSpellMgr->LoadSpellAreas(); LOG_INFO("server.loading", "Loading Area Trigger Definitions"); sObjectMgr->LoadAreaTriggers(); LOG_INFO("server.loading", "Loading Area Trigger Teleport Definitions..."); sObjectMgr->LoadAreaTriggerTeleports(); LOG_INFO("server.loading", "Loading Access Requirements..."); sObjectMgr->LoadAccessRequirements(); // must be after item template load LOG_INFO("server.loading", "Loading Quest Area Triggers..."); sObjectMgr->LoadQuestAreaTriggers(); // must be after LoadQuests LOG_INFO("server.loading", "Loading Tavern Area Triggers..."); sObjectMgr->LoadTavernAreaTriggers(); LOG_INFO("server.loading", "Loading AreaTrigger Script Names..."); sObjectMgr->LoadAreaTriggerScripts(); LOG_INFO("server.loading", "Loading LFG Entrance Positions..."); // Must be after areatriggers sLFGMgr->LoadLFGDungeons(); LOG_INFO("server.loading", "Loading Dungeon Boss Data..."); sObjectMgr->LoadInstanceEncounters(); LOG_INFO("server.loading", "Loading LFG Rewards..."); sLFGMgr->LoadRewards(); LOG_INFO("server.loading", "Loading Graveyard-Zone Links..."); sGraveyard->LoadGraveyardZones(); LOG_INFO("server.loading", "Loading Spell Pet Auras..."); sSpellMgr->LoadSpellPetAuras(); LOG_INFO("server.loading", "Loading Spell Target Coordinates..."); sSpellMgr->LoadSpellTargetPositions(); LOG_INFO("server.loading", "Loading Enchant Custom Attributes..."); sSpellMgr->LoadEnchantCustomAttr(); LOG_INFO("server.loading", "Loading linked Spells..."); sSpellMgr->LoadSpellLinked(); LOG_INFO("server.loading", "Loading Player Create Data..."); sObjectMgr->LoadPlayerInfo(); LOG_INFO("server.loading", "Loading Exploration BaseXP Data..."); sObjectMgr->LoadExplorationBaseXP(); LOG_INFO("server.loading", "Loading Pet Name Parts..."); sObjectMgr->LoadPetNames(); CharacterDatabaseCleaner::CleanDatabase(); LOG_INFO("server.loading", "Loading The Max Pet Number..."); sObjectMgr->LoadPetNumber(); LOG_INFO("server.loading", "Loading Pet Level Stats..."); sObjectMgr->LoadPetLevelInfo(); LOG_INFO("server.loading", "Loading Player Level Dependent Mail Rewards..."); sObjectMgr->LoadMailLevelRewards(); LOG_INFO("server.loading", "Load Mail Server Template..."); sObjectMgr->LoadMailServerTemplates(); // Loot tables LoadLootTables(); LOG_INFO("server.loading", "Loading Skill Discovery Table..."); LoadSkillDiscoveryTable(); LOG_INFO("server.loading", "Loading Skill Extra Item Table..."); LoadSkillExtraItemTable(); LOG_INFO("server.loading", "Loading Skill Perfection Data Table..."); LoadSkillPerfectItemTable(); LOG_INFO("server.loading", "Loading Skill Fishing Base Level Requirements..."); sObjectMgr->LoadFishingBaseSkillLevel(); LOG_INFO("server.loading", "Loading Achievements..."); sAchievementMgr->LoadAchievementReferenceList(); LOG_INFO("server.loading", "Loading Achievement Criteria Lists..."); sAchievementMgr->LoadAchievementCriteriaList(); LOG_INFO("server.loading", "Loading Achievement Criteria Data..."); sAchievementMgr->LoadAchievementCriteriaData(); LOG_INFO("server.loading", "Loading Achievement Rewards..."); sAchievementMgr->LoadRewards(); LOG_INFO("server.loading", "Loading Achievement Reward Locales..."); sAchievementMgr->LoadRewardLocales(); LOG_INFO("server.loading", "Loading Completed Achievements..."); sAchievementMgr->LoadCompletedAchievements(); ///- Load dynamic data tables from the database LOG_INFO("server.loading", "Loading Item Auctions..."); sAuctionMgr->LoadAuctionItems(); LOG_INFO("server.loading", "Loading Auctions..."); sAuctionMgr->LoadAuctions(); sGuildMgr->LoadGuilds(); LOG_INFO("server.loading", "Loading ArenaTeams..."); sArenaTeamMgr->LoadArenaTeams(); LOG_INFO("server.loading", "Loading Groups..."); sGroupMgr->LoadGroups(); LOG_INFO("server.loading", "Loading Reserved Names..."); sObjectMgr->LoadReservedPlayersNames(); LOG_INFO("server.loading", "Loading Profanity Names..."); sObjectMgr->LoadProfanityPlayersNames(); LOG_INFO("server.loading", "Loading GameObjects for Quests..."); sObjectMgr->LoadGameObjectForQuests(); LOG_INFO("server.loading", "Loading BattleMasters..."); sBattlegroundMgr->LoadBattleMastersEntry(); LOG_INFO("server.loading", "Loading GameTeleports..."); sObjectMgr->LoadGameTele(); LOG_INFO("server.loading", "Loading Gossip Menu..."); sObjectMgr->LoadGossipMenu(); LOG_INFO("server.loading", "Loading Gossip Menu Options..."); sObjectMgr->LoadGossipMenuItems(); LOG_INFO("server.loading", "Loading Vendors..."); sObjectMgr->LoadVendors(); // must be after load CreatureTemplate and ItemTemplate LOG_INFO("server.loading", "Loading Trainers..."); sObjectMgr->LoadTrainerSpell(); // must be after load CreatureTemplate LOG_INFO("server.loading", "Loading Waypoints..."); sWaypointMgr->Load(); LOG_INFO("server.loading", "Loading SmartAI Waypoints..."); sSmartWaypointMgr->LoadFromDB(); LOG_INFO("server.loading", "Loading Creature Formations..."); sFormationMgr->LoadCreatureFormations(); LOG_INFO("server.loading", "Loading World States..."); // must be loaded before battleground, outdoor PvP and conditions LoadWorldStates(); LOG_INFO("server.loading", "Loading Conditions..."); sConditionMgr->LoadConditions(); LOG_INFO("server.loading", "Loading Faction Change Achievement Pairs..."); sObjectMgr->LoadFactionChangeAchievements(); LOG_INFO("server.loading", "Loading Faction Change Spell Pairs..."); sObjectMgr->LoadFactionChangeSpells(); LOG_INFO("server.loading", "Loading Faction Change Item Pairs..."); sObjectMgr->LoadFactionChangeItems(); LOG_INFO("server.loading", "Loading Faction Change Reputation Pairs..."); sObjectMgr->LoadFactionChangeReputations(); LOG_INFO("server.loading", "Loading Faction Change Title Pairs..."); sObjectMgr->LoadFactionChangeTitles(); LOG_INFO("server.loading", "Loading Faction Change Quest Pairs..."); sObjectMgr->LoadFactionChangeQuests(); LOG_INFO("server.loading", "Loading GM Tickets..."); sTicketMgr->LoadTickets(); LOG_INFO("server.loading", "Loading GM Surveys..."); sTicketMgr->LoadSurveys(); LOG_INFO("server.loading", "Loading Client Addons..."); AddonMgr::LoadFromDB(); // pussywizard: LOG_INFO("server.loading", "Deleting Invalid Mail Items..."); LOG_INFO("server.loading", " "); CharacterDatabase.Execute("DELETE mi FROM mail_items mi LEFT JOIN item_instance ii ON mi.item_guid = ii.guid WHERE ii.guid IS NULL"); CharacterDatabase.Execute("DELETE mi FROM mail_items mi LEFT JOIN mail m ON mi.mail_id = m.id WHERE m.id IS NULL"); CharacterDatabase.Execute("UPDATE mail m LEFT JOIN mail_items mi ON m.id = mi.mail_id SET m.has_items=0 WHERE m.has_items<>0 AND mi.mail_id IS NULL"); ///- Handle outdated emails (delete/return) LOG_INFO("server.loading", "Returning Old Mails..."); LOG_INFO("server.loading", " "); sObjectMgr->ReturnOrDeleteOldMails(false); ///- Load AutoBroadCast LOG_INFO("server.loading", "Loading Autobroadcasts..."); sAutobroadcastMgr->LoadAutobroadcasts(); ///- Load Motd LOG_INFO("server.loading", "Loading Motd..."); sMotdMgr->LoadMotd(); ///- Load and initialize scripts sObjectMgr->LoadSpellScripts(); // must be after load Creature/Gameobject(Template/Data) sObjectMgr->LoadEventScripts(); // must be after load Creature/Gameobject(Template/Data) sObjectMgr->LoadWaypointScripts(); LOG_INFO("server.loading", "Loading Spell Script Names..."); sObjectMgr->LoadSpellScriptNames(); LOG_INFO("server.loading", "Loading Creature Texts..."); sCreatureTextMgr->LoadCreatureTexts(); LOG_INFO("server.loading", "Loading Creature Text Locales..."); sCreatureTextMgr->LoadCreatureTextLocales(); LOG_INFO("server.loading", "Loading Scripts..."); sScriptMgr->LoadDatabase(); LOG_INFO("server.loading", "Validating Spell Scripts..."); sObjectMgr->ValidateSpellScripts(); LOG_INFO("server.loading", "Loading SmartAI Scripts..."); sSmartScriptMgr->LoadSmartAIFromDB(); LOG_INFO("server.loading", "Loading Calendar Data..."); sCalendarMgr->LoadFromDB(); LOG_INFO("server.loading", "Initializing SpellInfo Precomputed Data..."); // must be called after loading items, professions, spells and pretty much anything LOG_INFO("server.loading", " "); sObjectMgr->InitializeSpellInfoPrecomputedData(); LOG_INFO("server.loading", "Initialize Commands..."); Acore::ChatCommands::LoadCommandMap(); ///- Initialize game time and timers LOG_INFO("server.loading", "Initialize Game Time and Timers"); LOG_INFO("server.loading", " "); LoginDatabase.Execute("INSERT INTO uptime (realmid, starttime, uptime, revision) VALUES ({}, {}, 0, '{}')", realm.Id.Realm, uint32(GameTime::GetStartTime().count()), GitRevision::GetFullVersion()); // One-time query _timers[WUPDATE_WEATHERS].SetInterval(1 * IN_MILLISECONDS); _timers[WUPDATE_AUCTIONS].SetInterval(MINUTE * IN_MILLISECONDS); _timers[WUPDATE_AUCTIONS].SetCurrent(MINUTE * IN_MILLISECONDS); _timers[WUPDATE_UPTIME].SetInterval(_int_configs[CONFIG_UPTIME_UPDATE]*MINUTE * IN_MILLISECONDS); //Update "uptime" table based on configuration entry in minutes. _timers[WUPDATE_CORPSES].SetInterval(20 * MINUTE * IN_MILLISECONDS); //erase corpses every 20 minutes _timers[WUPDATE_CLEANDB].SetInterval(_int_configs[CONFIG_LOGDB_CLEARINTERVAL]*MINUTE * IN_MILLISECONDS); // clean logs table every 14 days by default _timers[WUPDATE_AUTOBROADCAST].SetInterval(getIntConfig(CONFIG_AUTOBROADCAST_INTERVAL)); _timers[WUPDATE_PINGDB].SetInterval(getIntConfig(CONFIG_DB_PING_INTERVAL)*MINUTE * IN_MILLISECONDS); // Mysql ping time in minutes // our speed up _timers[WUPDATE_5_SECS].SetInterval(5 * IN_MILLISECONDS); _timers[WUPDATE_WHO_LIST].SetInterval(5 * IN_MILLISECONDS); // update who list cache every 5 seconds _mail_expire_check_timer = GameTime::GetGameTime() + 6h; ///- Initialize MapMgr LOG_INFO("server.loading", "Starting Map System"); LOG_INFO("server.loading", " "); sMapMgr->Initialize(); LOG_INFO("server.loading", "Starting Game Event system..."); LOG_INFO("server.loading", " "); uint32 nextGameEvent = sGameEventMgr->StartSystem(); _timers[WUPDATE_EVENTS].SetInterval(nextGameEvent); //depend on next event // Delete all characters which have been deleted X days before Player::DeleteOldCharacters(); // Delete all custom channels which haven't been used for PreserveCustomChannelDuration days. Channel::CleanOldChannelsInDB(); LOG_INFO("server.loading", "Initializing Opcodes..."); opcodeTable.Initialize(); LOG_INFO("server.loading", "Starting Arena Season..."); LOG_INFO("server.loading", " "); sGameEventMgr->StartArenaSeason(); sTicketMgr->Initialize(); ///- Initialize Battlegrounds LOG_INFO("server.loading", "Starting Battleground System"); sBattlegroundMgr->LoadBattlegroundTemplates(); sBattlegroundMgr->InitAutomaticArenaPointDistribution(); ///- Initialize outdoor pvp LOG_INFO("server.loading", "Starting Outdoor PvP System"); sOutdoorPvPMgr->InitOutdoorPvP(); ///- Initialize Battlefield LOG_INFO("server.loading", "Starting Battlefield System"); sBattlefieldMgr->InitBattlefield(); LOG_INFO("server.loading", "Loading Transports..."); sTransportMgr->SpawnContinentTransports(); ///- Initialize Warden LOG_INFO("server.loading", "Loading Warden Checks..." ); sWardenCheckMgr->LoadWardenChecks(); LOG_INFO("server.loading", "Loading Warden Action Overrides..." ); sWardenCheckMgr->LoadWardenOverrides(); LOG_INFO("server.loading", "Deleting Expired Bans..."); LoginDatabase.Execute("DELETE FROM ip_banned WHERE unbandate <= UNIX_TIMESTAMP() AND unbandate<>bandate"); // One-time query LOG_INFO("server.loading", "Calculate Next Daily Quest Reset Time..."); InitDailyQuestResetTime(); LOG_INFO("server.loading", "Calculate Next Weekly Quest Reset Time..." ); InitWeeklyQuestResetTime(); LOG_INFO("server.loading", "Calculate Next Monthly Quest Reset Time..."); InitMonthlyQuestResetTime(); LOG_INFO("server.loading", "Calculate Random Battleground Reset Time..." ); InitRandomBGResetTime(); LOG_INFO("server.loading", "Calculate Deletion Of Old Calendar Events Time..."); InitCalendarOldEventsDeletionTime(); LOG_INFO("server.loading", "Calculate Guild Cap Reset Time..."); LOG_INFO("server.loading", " "); InitGuildResetTime(); LOG_INFO("server.loading", "Load Petitions..."); sPetitionMgr->LoadPetitions(); LOG_INFO("server.loading", "Load Petition Signs..."); sPetitionMgr->LoadSignatures(); LOG_INFO("server.loading", "Load Stored Loot Items..."); sLootItemStorage->LoadStorageFromDB(); LOG_INFO("server.loading", "Load Channel Rights..."); ChannelMgr::LoadChannelRights(); LOG_INFO("server.loading", "Load Channels..."); ChannelMgr::LoadChannels(); sScriptMgr->OnBeforeWorldInitialized(); if (sWorld->getBoolConfig(CONFIG_PRELOAD_ALL_NON_INSTANCED_MAP_GRIDS)) { LOG_INFO("server.loading", "Loading All Grids For All Non-Instanced Maps..."); for (uint32 i = 0; i < sMapStore.GetNumRows(); ++i) { MapEntry const* mapEntry = sMapStore.LookupEntry(i); if (mapEntry && !mapEntry->Instanceable()) { Map* map = sMapMgr->CreateBaseMap(mapEntry->MapID); if (map) { LOG_INFO("server.loading", ">> Loading All Grids For Map {}", map->GetId()); map->LoadAllCells(); } } } } uint32 startupDuration = GetMSTimeDiffToNow(startupBegin); LOG_INFO("server.loading", " "); LOG_INFO("server.loading", "WORLD: World Initialized In {} Minutes {} Seconds", (startupDuration / 60000), ((startupDuration % 60000) / 1000)); // outError for red color in console LOG_INFO("server.loading", " "); METRIC_EVENT("events", "World initialized", "World Initialized In " + std::to_string(startupDuration / 60000) + " Minutes " + std::to_string((startupDuration % 60000) / 1000) + " Seconds"); if (sConfigMgr->isDryRun()) { sMapMgr->UnloadAll(); LOG_INFO("server.loading", "AzerothCore Dry Run Completed, Terminating."); exit(0); } } void World::DetectDBCLang() { uint8 m_lang_confid = sConfigMgr->GetOption("DBC.Locale", 255); if (m_lang_confid != 255 && m_lang_confid >= TOTAL_LOCALES) { LOG_ERROR("server.loading", "Incorrect DBC.Locale! Must be >= 0 and < {} (set to 0)", TOTAL_LOCALES); m_lang_confid = LOCALE_enUS; } ChrRacesEntry const* race = sChrRacesStore.LookupEntry(1); std::string availableLocalsStr; uint8 default_locale = TOTAL_LOCALES; for (uint8 i = default_locale - 1; i < TOTAL_LOCALES; --i) // -1 will be 255 due to uint8 { if (race->name[i][0] != '\0') // check by race names { default_locale = i; _availableDbcLocaleMask |= (1 << i); availableLocalsStr += localeNames[i]; availableLocalsStr += " "; } } if (default_locale != m_lang_confid && m_lang_confid < TOTAL_LOCALES && (_availableDbcLocaleMask & (1 << m_lang_confid))) { default_locale = m_lang_confid; } if (default_locale >= TOTAL_LOCALES) { LOG_ERROR("server.loading", "Unable to determine your DBC Locale! (corrupt DBC?)"); exit(1); } _defaultDbcLocale = LocaleConstant(default_locale); LOG_INFO("server.loading", "Using {} DBC Locale As Default. All Available DBC locales: {}", localeNames[GetDefaultDbcLocale()], availableLocalsStr.empty() ? "" : availableLocalsStr); LOG_INFO("server.loading", " "); } /// Update the World ! void World::Update(uint32 diff) { METRIC_TIMER("world_update_time_total"); ///- Update the game time and check for shutdown time _UpdateGameTime(); Seconds currentGameTime = GameTime::GetGameTime(); sWorldUpdateTime.UpdateWithDiff(diff); // Record update if recording set in log and diff is greater then minimum set in log sWorldUpdateTime.RecordUpdateTime(GameTime::GetGameTimeMS(), diff, GetActiveSessionCount()); DynamicVisibilityMgr::Update(GetActiveSessionCount()); ///- Update the different timers for (int i = 0; i < WUPDATE_COUNT; ++i) { if (_timers[i].GetCurrent() >= 0) _timers[i].Update(diff); else _timers[i].SetCurrent(0); } // pussywizard: our speed up and functionality if (_timers[WUPDATE_5_SECS].Passed()) { _timers[WUPDATE_5_SECS].Reset(); // moved here from HandleCharEnumOpcode CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_DEL_EXPIRED_BANS); CharacterDatabase.Execute(stmt); } ///- Update Who List Cache if (_timers[WUPDATE_WHO_LIST].Passed()) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update who list")); _timers[WUPDATE_WHO_LIST].Reset(); sWhoListCacheMgr->Update(); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Check quest reset times")); /// Handle daily quests reset time if (currentGameTime > _nextDailyQuestReset) { ResetDailyQuests(); } /// Handle weekly quests reset time if (currentGameTime > _nextWeeklyQuestReset) { ResetWeeklyQuests(); } /// Handle monthly quests reset time if (currentGameTime > _nextMonthlyQuestReset) { ResetMonthlyQuests(); } } if (currentGameTime > _nextRandomBGReset) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Reset random BG")); ResetRandomBG(); } if (currentGameTime > _nextCalendarOldEventsDeletionTime) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Delete old calendar events")); CalendarDeleteOldEvents(); } if (currentGameTime > _nextGuildReset) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Reset guild cap")); ResetGuildCap(); } // pussywizard: handle auctions when the timer has passed if (_timers[WUPDATE_AUCTIONS].Passed()) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update expired auctions")); _timers[WUPDATE_AUCTIONS].Reset(); // pussywizard: handle expired auctions, auctions expired when realm was offline are also handled here (not during loading when many required things aren't loaded yet) sAuctionMgr->Update(); } AsyncAuctionListingMgr::Update(Milliseconds(diff)); if (currentGameTime > _mail_expire_check_timer) { sObjectMgr->ReturnOrDeleteOldMails(true); _mail_expire_check_timer = currentGameTime + 6h; } METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update sessions")); UpdateSessions(diff); ///
  • Handle weather updates when the timer has passed if (_timers[WUPDATE_WEATHERS].Passed()) { _timers[WUPDATE_WEATHERS].Reset(); WeatherMgr::Update(uint32(_timers[WUPDATE_WEATHERS].GetInterval())); } ///
  • Clean logs table if (sWorld->getIntConfig(CONFIG_LOGDB_CLEARTIME) > 0) // if not enabled, ignore the timer { if (_timers[WUPDATE_CLEANDB].Passed()) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Clean logs table")); _timers[WUPDATE_CLEANDB].Reset(); LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_DEL_OLD_LOGS); stmt->SetData(0, sWorld->getIntConfig(CONFIG_LOGDB_CLEARTIME)); stmt->SetData(1, uint32(currentGameTime.count())); LoginDatabase.Execute(stmt); } } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update LFG 0")); sLFGMgr->Update(diff, 0); // pussywizard: remove obsolete stuff before finding compatibility during map update } { ///- Update objects when the timer has passed (maps, transport, creatures, ...) METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update maps")); sMapMgr->Update(diff); } if (sWorld->getBoolConfig(CONFIG_AUTOBROADCAST)) { if (_timers[WUPDATE_AUTOBROADCAST].Passed()) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Send autobroadcast")); _timers[WUPDATE_AUTOBROADCAST].Reset(); sAutobroadcastMgr->SendAutobroadcasts(); } } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update battlegrounds")); sBattlegroundMgr->Update(diff); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update outdoor pvp")); sOutdoorPvPMgr->Update(diff); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update battlefields")); sBattlefieldMgr->Update(diff); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update LFG 2")); sLFGMgr->Update(diff, 2); // pussywizard: handle created proposals } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Process query callbacks")); // execute callbacks from sql queries that were queued recently ProcessQueryCallbacks(); } ///
  • Update uptime table if (_timers[WUPDATE_UPTIME].Passed()) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update uptime")); _timers[WUPDATE_UPTIME].Reset(); LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_UPD_UPTIME_PLAYERS); stmt->SetData(0, uint32(GameTime::GetUptime().count())); stmt->SetData(1, uint16(GetMaxPlayerCount())); stmt->SetData(2, realm.Id.Realm); stmt->SetData(3, uint32(GameTime::GetStartTime().count())); LoginDatabase.Execute(stmt); } ///- Erase corpses once every 20 minutes if (_timers[WUPDATE_CORPSES].Passed()) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Remove old corpses")); _timers[WUPDATE_CORPSES].Reset(); sMapMgr->DoForAllMaps([](Map* map) { map->RemoveOldCorpses(); }); } ///- Process Game events when necessary if (_timers[WUPDATE_EVENTS].Passed()) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update game events")); _timers[WUPDATE_EVENTS].Reset(); // to give time for Update() to be processed uint32 nextGameEvent = sGameEventMgr->Update(); _timers[WUPDATE_EVENTS].SetInterval(nextGameEvent); _timers[WUPDATE_EVENTS].Reset(); } ///- Ping to keep MySQL connections alive if (_timers[WUPDATE_PINGDB].Passed()) { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Ping MySQL")); _timers[WUPDATE_PINGDB].Reset(); LOG_DEBUG("sql.driver", "Ping MySQL to keep connection alive"); CharacterDatabase.KeepAlive(); LoginDatabase.KeepAlive(); WorldDatabase.KeepAlive(); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update instance reset times")); // update the instance reset times sInstanceSaveMgr->Update(); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Process cli commands")); // And last, but not least handle the issued cli commands ProcessCliCommands(); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update world scripts")); sScriptMgr->OnWorldUpdate(diff); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update playersSaveScheduler")); playersSaveScheduler.Update(diff); } { METRIC_TIMER("world_update_time", METRIC_TAG("type", "Update metrics")); // Stats logger update sMetric->Update(); METRIC_VALUE("update_time_diff", diff); } } void World::ForceGameEventUpdate() { _timers[WUPDATE_EVENTS].Reset(); // to give time for Update() to be processed uint32 nextGameEvent = sGameEventMgr->Update(); _timers[WUPDATE_EVENTS].SetInterval(nextGameEvent); _timers[WUPDATE_EVENTS].Reset(); } /// Send a packet to all players (except self if mentioned) void World::SendGlobalMessage(WorldPacket const* packet, WorldSession* self, TeamId teamId) { SessionMap::const_iterator itr; for (itr = _sessions.begin(); itr != _sessions.end(); ++itr) { if (itr->second && itr->second->GetPlayer() && itr->second->GetPlayer()->IsInWorld() && itr->second != self && (teamId == TEAM_NEUTRAL || itr->second->GetPlayer()->GetTeamId() == teamId)) { itr->second->SendPacket(packet); } } } /// Send a packet to all GMs (except self if mentioned) void World::SendGlobalGMMessage(WorldPacket const* packet, WorldSession* self, TeamId teamId) { SessionMap::iterator itr; for (itr = _sessions.begin(); itr != _sessions.end(); ++itr) { if (itr->second && itr->second->GetPlayer() && itr->second->GetPlayer()->IsInWorld() && itr->second != self && !AccountMgr::IsPlayerAccount(itr->second->GetSecurity()) && (teamId == TEAM_NEUTRAL || itr->second->GetPlayer()->GetTeamId() == teamId)) { itr->second->SendPacket(packet); } } } namespace Acore { class WorldWorldTextBuilder { public: typedef std::vector WorldPacketList; explicit WorldWorldTextBuilder(uint32 textId, va_list* args = nullptr) : i_textId(textId), i_args(args) {} void operator()(WorldPacketList& data_list, LocaleConstant loc_idx) { char const* text = sObjectMgr->GetAcoreString(i_textId, loc_idx); if (i_args) { // we need copy va_list before use or original va_list will corrupted va_list ap; va_copy(ap, *i_args); char str[2048]; vsnprintf(str, 2048, text, ap); va_end(ap); do_helper(data_list, &str[0]); } else do_helper(data_list, (char*)text); } private: char* lineFromMessage(char*& pos) { char* start = strtok(pos, "\n"); pos = nullptr; return start; } void do_helper(WorldPacketList& data_list, char* text) { char* pos = text; while (char* line = lineFromMessage(pos)) { WorldPacket* data = new WorldPacket(); ChatHandler::BuildChatPacket(*data, CHAT_MSG_SYSTEM, LANG_UNIVERSAL, nullptr, nullptr, line); data_list.push_back(data); } } uint32 i_textId; va_list* i_args; }; } // namespace Acore /// Send a System Message to all players (except self if mentioned) void World::SendWorldText(uint32 string_id, ...) { va_list ap; va_start(ap, string_id); Acore::WorldWorldTextBuilder wt_builder(string_id, &ap); Acore::LocalizedPacketListDo wt_do(wt_builder); for (SessionMap::const_iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) { if (!itr->second || !itr->second->GetPlayer() || !itr->second->GetPlayer()->IsInWorld()) continue; wt_do(itr->second->GetPlayer()); } va_end(ap); } void World::SendWorldTextOptional(uint32 string_id, uint32 flag, ...) { va_list ap; va_start(ap, flag); Acore::WorldWorldTextBuilder wt_builder(string_id, &ap); Acore::LocalizedPacketListDo wt_do(wt_builder); for (auto const& itr : _sessions) { if (!itr.second || !itr.second->GetPlayer() || !itr.second->GetPlayer()->IsInWorld()) { continue; } if (sWorld->getBoolConfig(CONFIG_PLAYER_SETTINGS_ENABLED)) { if (itr.second->GetPlayer()->GetPlayerSetting(AzerothcorePSSource, SETTING_ANNOUNCER_FLAGS).HasFlag(flag)) { continue; } } wt_do(itr.second->GetPlayer()); } va_end(ap); } /// Send a System Message to all GMs (except self if mentioned) void World::SendGMText(uint32 string_id, ...) { va_list ap; va_start(ap, string_id); Acore::WorldWorldTextBuilder wt_builder(string_id, &ap); Acore::LocalizedPacketListDo wt_do(wt_builder); for (SessionMap::iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) { // Session should have permissions to receive global gm messages WorldSession* session = itr->second; if (!session || AccountMgr::IsPlayerAccount(session->GetSecurity())) continue; // Player should be in world Player* player = session->GetPlayer(); if (!player || !player->IsInWorld()) continue; wt_do(session->GetPlayer()); } va_end(ap); } /// @deprecated only for debug purpose. Send a System Message to all players (except self if mentioned) void World::SendGlobalText(const char* text, WorldSession* self) { WorldPacket data; // need copy to prevent corruption by strtok call in LineFromMessage original string char* buf = strdup(text); char* pos = buf; while (char* line = ChatHandler::LineFromMessage(pos)) { ChatHandler::BuildChatPacket(data, CHAT_MSG_SYSTEM, LANG_UNIVERSAL, nullptr, nullptr, line); SendGlobalMessage(&data, self); } free(buf); } /// Send a packet to all players (or players selected team) in the zone (except self if mentioned) bool World::SendZoneMessage(uint32 zone, WorldPacket const* packet, WorldSession* self, TeamId teamId) { bool foundPlayerToSend = false; SessionMap::const_iterator itr; for (itr = _sessions.begin(); itr != _sessions.end(); ++itr) { if (itr->second && itr->second->GetPlayer() && itr->second->GetPlayer()->IsInWorld() && itr->second->GetPlayer()->GetZoneId() == zone && itr->second != self && (teamId == TEAM_NEUTRAL || itr->second->GetPlayer()->GetTeamId() == teamId)) { itr->second->SendPacket(packet); foundPlayerToSend = true; } } return foundPlayerToSend; } /// Send a System Message to all players in the zone (except self if mentioned) void World::SendZoneText(uint32 zone, const char* text, WorldSession* self, TeamId teamId) { WorldPacket data; ChatHandler::BuildChatPacket(data, CHAT_MSG_SYSTEM, LANG_UNIVERSAL, nullptr, nullptr, text); SendZoneMessage(zone, &data, self, teamId); } /// Kick (and save) all players void World::KickAll() { _queuedPlayer.clear(); // prevent send queue update packet and login queued sessions // session not removed at kick and will removed in next update tick for (SessionMap::const_iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) itr->second->KickPlayer("KickAll sessions"); // pussywizard: kick offline sessions for (SessionMap::const_iterator itr = _offlineSessions.begin(); itr != _offlineSessions.end(); ++itr) itr->second->KickPlayer("KickAll offline sessions"); } /// Kick (and save) all players with security level less `sec` void World::KickAllLess(AccountTypes sec) { // session not removed at kick and will removed in next update tick for (SessionMap::const_iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) if (itr->second->GetSecurity() < sec) itr->second->KickPlayer("KickAllLess"); } /// Update the game time void World::_UpdateGameTime() { ///- update the time Seconds lastGameTime = GameTime::GetGameTime(); GameTime::UpdateGameTimers(); Seconds elapsed = GameTime::GetGameTime() - lastGameTime; ///- if there is a shutdown timer if (!IsStopped() && _shutdownTimer > 0 && elapsed > 0s) { ///- ... and it is overdue, stop the world (set m_stopEvent) if (_shutdownTimer <= elapsed.count()) { if (!(_shutdownMask & SHUTDOWN_MASK_IDLE) || GetActiveAndQueuedSessionCount() == 0) _stopEvent = true; // exist code already set else _shutdownTimer = 1; // minimum timer value to wait idle state } ///- ... else decrease it and if necessary display a shutdown countdown to the users else { _shutdownTimer -= elapsed.count(); ShutdownMsg(); } } } /// Shutdown the server void World::ShutdownServ(uint32 time, uint32 options, uint8 exitcode, const std::string& reason) { // ignore if server shutdown at next tick if (IsStopped()) return; _shutdownMask = options; _exitCode = exitcode; auto const& playersOnline = GetActiveSessionCount(); if (time < 5 && playersOnline) { // Set time to 5s for save all players time = 5; } playersSaveScheduler.CancelAll(); if (time >= 5) { playersSaveScheduler.Schedule(Seconds(time - 5), [this](TaskContext /*context*/) { if (!GetActiveSessionCount()) { LOG_INFO("server", "> No players online. Skip save before shutdown"); return; } LOG_INFO("server", "> Save players before shutdown server"); ObjectAccessor::SaveAllPlayers(); }); } LOG_WARN("server", "Time left until shutdown/restart: {}", time); ///- If the shutdown time is 0, set m_stopEvent (except if shutdown is 'idle' with remaining sessions) if (time == 0) { if (!(options & SHUTDOWN_MASK_IDLE) || GetActiveAndQueuedSessionCount() == 0) _stopEvent = true; // exist code already set else _shutdownTimer = 1; //So that the session count is re-evaluated at next world tick } ///- Else set the shutdown timer and warn users else { _shutdownTimer = time; ShutdownMsg(true, nullptr, reason); } sScriptMgr->OnShutdownInitiate(ShutdownExitCode(exitcode), ShutdownMask(options)); } /// Display a shutdown message to the user(s) void World::ShutdownMsg(bool show, Player* player, const std::string& reason) { // not show messages for idle shutdown mode if (_shutdownMask & SHUTDOWN_MASK_IDLE) return; ///- Display a message every 12 hours, hours, 5 minutes, minute, 5 seconds and finally seconds if (show || (_shutdownTimer < 5 * MINUTE && (_shutdownTimer % 15) == 0) || // < 5 min; every 15 sec (_shutdownTimer < 15 * MINUTE && (_shutdownTimer % MINUTE) == 0) || // < 15 min ; every 1 min (_shutdownTimer < 30 * MINUTE && (_shutdownTimer % (5 * MINUTE)) == 0) || // < 30 min ; every 5 min (_shutdownTimer < 12 * HOUR && (_shutdownTimer % HOUR) == 0) || // < 12 h ; every 1 h (_shutdownTimer > 12 * HOUR && (_shutdownTimer % (12 * HOUR)) == 0)) // > 12 h ; every 12 h { std::string str = secsToTimeString(_shutdownTimer).append("."); if (!reason.empty()) { str += " - " + reason; } ServerMessageType msgid = (_shutdownMask & SHUTDOWN_MASK_RESTART) ? SERVER_MSG_RESTART_TIME : SERVER_MSG_SHUTDOWN_TIME; SendServerMessage(msgid, str, player); LOG_DEBUG("server.worldserver", "Server is {} in {}", (_shutdownMask & SHUTDOWN_MASK_RESTART ? "restart" : "shuttingdown"), str); } } /// Cancel a planned server shutdown void World::ShutdownCancel() { // nothing cancel or too later if (!_shutdownTimer || _stopEvent) return; ServerMessageType msgid = (_shutdownMask & SHUTDOWN_MASK_RESTART) ? SERVER_MSG_RESTART_CANCELLED : SERVER_MSG_SHUTDOWN_CANCELLED; _shutdownMask = 0; _shutdownTimer = 0; _exitCode = SHUTDOWN_EXIT_CODE; // to default value SendServerMessage(msgid); LOG_DEBUG("server.worldserver", "Server {} cancelled.", (_shutdownMask & SHUTDOWN_MASK_RESTART ? "restart" : "shuttingdown")); sScriptMgr->OnShutdownCancel(); } /// Send a server message to the user(s) void World::SendServerMessage(ServerMessageType messageID, std::string stringParam /*= ""*/, Player* player /*= nullptr*/) { WorldPackets::Chat::ChatServerMessage chatServerMessage; chatServerMessage.MessageID = int32(messageID); if (messageID <= SERVER_MSG_STRING) chatServerMessage.StringParam = stringParam; if (player) player->SendDirectMessage(chatServerMessage.Write()); else SendGlobalMessage(chatServerMessage.Write()); } void World::UpdateSessions(uint32 diff) { { METRIC_DETAILED_NO_THRESHOLD_TIMER("world_update_time", METRIC_TAG("type", "Add sessions"), METRIC_TAG("parent_type", "Update sessions")); ///- Add new sessions WorldSession* sess = nullptr; while (_addSessQueue.next(sess)) { AddSession_(sess); } } ///- Then send an update signal to remaining ones for (SessionMap::iterator itr = _sessions.begin(), next; itr != _sessions.end(); itr = next) { next = itr; ++next; ///- and remove not active sessions from the list WorldSession* pSession = itr->second; WorldSessionFilter updater(pSession); // pussywizard: if (pSession->HandleSocketClosed()) { if (!RemoveQueuedPlayer(pSession) && getIntConfig(CONFIG_INTERVAL_DISCONNECT_TOLERANCE)) _disconnects[pSession->GetAccountId()] = GameTime::GetGameTime().count(); _sessions.erase(itr); // there should be no offline session if current one is logged onto a character SessionMap::iterator iter; if ((iter = _offlineSessions.find(pSession->GetAccountId())) != _offlineSessions.end()) { WorldSession* tmp = iter->second; _offlineSessions.erase(iter); delete tmp; } pSession->SetOfflineTime(GameTime::GetGameTime().count()); _offlineSessions[pSession->GetAccountId()] = pSession; continue; } [[maybe_unused]] uint32 currentSessionId = itr->first; METRIC_DETAILED_TIMER("world_update_sessions_time", METRIC_TAG("account_id", std::to_string(currentSessionId))); if (!pSession->Update(diff, updater)) { if (!RemoveQueuedPlayer(pSession) && getIntConfig(CONFIG_INTERVAL_DISCONNECT_TOLERANCE)) _disconnects[pSession->GetAccountId()] = GameTime::GetGameTime().count(); _sessions.erase(itr); delete pSession; } } // pussywizard: if (_offlineSessions.empty()) return; uint32 currTime = GameTime::GetGameTime().count(); for (SessionMap::iterator itr = _offlineSessions.begin(), next; itr != _offlineSessions.end(); itr = next) { next = itr; ++next; WorldSession* pSession = itr->second; if (!pSession->GetPlayer() || pSession->GetOfflineTime() + 60 < currTime || pSession->IsKicked()) { _offlineSessions.erase(itr); delete pSession; } } } // This handles the issued and queued CLI commands void World::ProcessCliCommands() { CliCommandHolder::Print zprint = nullptr; void* callbackArg = nullptr; CliCommandHolder* command = nullptr; while (_cliCmdQueue.next(command)) { LOG_DEBUG("server.worldserver", "CLI command under processing..."); zprint = command->m_print; callbackArg = command->m_callbackArg; CliHandler handler(callbackArg, zprint); handler.ParseCommands(command->m_command); if (command->m_commandFinished) command->m_commandFinished(callbackArg, !handler.HasSentErrorMessage()); delete command; } } void World::UpdateRealmCharCount(uint32 accountId) { CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_SEL_CHARACTER_COUNT); stmt->SetData(0, accountId); _queryProcessor.AddCallback(CharacterDatabase.AsyncQuery(stmt).WithPreparedCallback(std::bind(&World::_UpdateRealmCharCount, this, std::placeholders::_1))); } void World::_UpdateRealmCharCount(PreparedQueryResult resultCharCount) { if (resultCharCount) { Field* fields = resultCharCount->Fetch(); uint32 accountId = fields[0].Get(); uint8 charCount = uint8(fields[1].Get()); LoginDatabaseTransaction trans = LoginDatabase.BeginTransaction(); LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_REP_REALM_CHARACTERS); stmt->SetData(0, charCount); stmt->SetData(1, accountId); stmt->SetData(2, realm.Id.Realm); trans->Append(stmt); LoginDatabase.CommitTransaction(trans); } } void World::InitWeeklyQuestResetTime() { Seconds wstime = Seconds(sWorld->getWorldState(WS_WEEKLY_QUEST_RESET_TIME)); _nextWeeklyQuestReset = wstime > 0s ? wstime : Seconds(Acore::Time::GetNextTimeWithDayAndHour(4, 6)); if (wstime == 0s) { sWorld->setWorldState(WS_WEEKLY_QUEST_RESET_TIME, _nextWeeklyQuestReset.count()); } } void World::InitDailyQuestResetTime() { Seconds wstime = Seconds(sWorld->getWorldState(WS_DAILY_QUEST_RESET_TIME)); _nextDailyQuestReset = wstime > 0s ? wstime : Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, 6)); if (wstime == 0s) { sWorld->setWorldState(WS_DAILY_QUEST_RESET_TIME, _nextDailyQuestReset.count()); } } void World::InitMonthlyQuestResetTime() { Seconds wstime = Seconds(sWorld->getWorldState(WS_MONTHLY_QUEST_RESET_TIME)); _nextMonthlyQuestReset = wstime > 0s ? wstime : Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, 6)); if (wstime == 0s) { sWorld->setWorldState(WS_MONTHLY_QUEST_RESET_TIME, _nextMonthlyQuestReset.count()); } } void World::InitRandomBGResetTime() { Seconds wstime = Seconds(sWorld->getWorldState(WS_BG_DAILY_RESET_TIME)); _nextRandomBGReset = wstime > 0s ? wstime : Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, 6)); if (wstime == 0s) { sWorld->setWorldState(WS_BG_DAILY_RESET_TIME, _nextRandomBGReset.count()); } } void World::InitCalendarOldEventsDeletionTime() { Seconds currentDeletionTime = Seconds(getWorldState(WS_DAILY_CALENDAR_DELETION_OLD_EVENTS_TIME)); Seconds nextDeletionTime = currentDeletionTime > 0s ? currentDeletionTime : Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, getIntConfig(CONFIG_CALENDAR_DELETE_OLD_EVENTS_HOUR))); if (currentDeletionTime == 0s) { sWorld->setWorldState(WS_DAILY_CALENDAR_DELETION_OLD_EVENTS_TIME, nextDeletionTime.count()); } } void World::InitGuildResetTime() { Seconds wstime = Seconds(getWorldState(WS_GUILD_DAILY_RESET_TIME)); _nextGuildReset = wstime > 0s ? wstime : Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, 6)); if (wstime == 0s) { sWorld->setWorldState(WS_GUILD_DAILY_RESET_TIME, _nextGuildReset.count()); } } void World::ResetDailyQuests() { CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_DEL_QUEST_STATUS_DAILY); CharacterDatabase.Execute(stmt); for (SessionMap::const_iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) if (itr->second->GetPlayer()) itr->second->GetPlayer()->ResetDailyQuestStatus(); _nextDailyQuestReset = Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, 6)); sWorld->setWorldState(WS_DAILY_QUEST_RESET_TIME, _nextDailyQuestReset.count()); // change available dailies sPoolMgr->ChangeDailyQuests(); } void World::LoadDBAllowedSecurityLevel() { LoginDatabasePreparedStatement* stmt = LoginDatabase.GetPreparedStatement(LOGIN_SEL_REALMLIST_SECURITY_LEVEL); stmt->SetData(0, int32(realm.Id.Realm)); PreparedQueryResult result = LoginDatabase.Query(stmt); if (result) SetPlayerSecurityLimit(AccountTypes(result->Fetch()->Get())); } void World::SetPlayerSecurityLimit(AccountTypes _sec) { AccountTypes sec = _sec < SEC_CONSOLE ? _sec : SEC_PLAYER; bool update = sec > _allowedSecurityLevel; _allowedSecurityLevel = sec; if (update) KickAllLess(_allowedSecurityLevel); } void World::ResetWeeklyQuests() { CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_DEL_QUEST_STATUS_WEEKLY); CharacterDatabase.Execute(stmt); for (SessionMap::const_iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) if (itr->second->GetPlayer()) itr->second->GetPlayer()->ResetWeeklyQuestStatus(); _nextWeeklyQuestReset = Seconds(Acore::Time::GetNextTimeWithDayAndHour(4, 6)); sWorld->setWorldState(WS_WEEKLY_QUEST_RESET_TIME, _nextWeeklyQuestReset.count()); // change available weeklies sPoolMgr->ChangeWeeklyQuests(); } void World::ResetMonthlyQuests() { LOG_INFO("server.worldserver", "Monthly quests reset for all characters."); CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_DEL_QUEST_STATUS_MONTHLY); CharacterDatabase.Execute(stmt); for (SessionMap::const_iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) if (itr->second->GetPlayer()) itr->second->GetPlayer()->ResetMonthlyQuestStatus(); _nextMonthlyQuestReset = Seconds(Acore::Time::GetNextTimeWithMonthAndHour(-1, 6)); sWorld->setWorldState(WS_MONTHLY_QUEST_RESET_TIME, _nextMonthlyQuestReset.count()); } void World::ResetEventSeasonalQuests(uint16 event_id) { CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_DEL_QUEST_STATUS_SEASONAL); stmt->SetData(0, event_id); CharacterDatabase.Execute(stmt); for (SessionMap::const_iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) if (itr->second->GetPlayer()) itr->second->GetPlayer()->ResetSeasonalQuestStatus(event_id); } void World::ResetRandomBG() { LOG_DEBUG("server.worldserver", "Random BG status reset for all characters."); CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_DEL_BATTLEGROUND_RANDOM); CharacterDatabase.Execute(stmt); for (SessionMap::const_iterator itr = _sessions.begin(); itr != _sessions.end(); ++itr) if (itr->second->GetPlayer()) itr->second->GetPlayer()->SetRandomWinner(false); _nextRandomBGReset = Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, 6)); sWorld->setWorldState(WS_BG_DAILY_RESET_TIME, _nextRandomBGReset.count()); } void World::CalendarDeleteOldEvents() { LOG_INFO("server.worldserver", "Calendar deletion of old events."); _nextCalendarOldEventsDeletionTime = Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, getIntConfig(CONFIG_CALENDAR_DELETE_OLD_EVENTS_HOUR))); sWorld->setWorldState(WS_DAILY_CALENDAR_DELETION_OLD_EVENTS_TIME, _nextCalendarOldEventsDeletionTime.count()); sCalendarMgr->DeleteOldEvents(); } void World::ResetGuildCap() { LOG_INFO("server.worldserver", "Guild Daily Cap reset."); _nextGuildReset = Seconds(Acore::Time::GetNextTimeWithDayAndHour(-1, 6)); sWorld->setWorldState(WS_GUILD_DAILY_RESET_TIME, _nextGuildReset.count()); sGuildMgr->ResetTimes(); } void World::UpdateMaxSessionCounters() { _maxActiveSessionCount = std::max(_maxActiveSessionCount, uint32(_sessions.size() - _queuedPlayer.size())); _maxQueuedSessionCount = std::max(_maxQueuedSessionCount, uint32(_queuedPlayer.size())); } void World::LoadDBVersion() { QueryResult result = WorldDatabase.Query("SELECT db_version, cache_id FROM version LIMIT 1"); if (result) { Field* fields = result->Fetch(); _dbVersion = fields[0].Get(); // will be overwrite by config values if different and non-0 _int_configs[CONFIG_CLIENTCACHE_VERSION] = fields[1].Get(); } if (_dbVersion.empty()) _dbVersion = "Unknown world database."; } void World::UpdateAreaDependentAuras() { SessionMap::const_iterator itr; for (itr = _sessions.begin(); itr != _sessions.end(); ++itr) if (itr->second && itr->second->GetPlayer() && itr->second->GetPlayer()->IsInWorld()) { itr->second->GetPlayer()->UpdateAreaDependentAuras(itr->second->GetPlayer()->GetAreaId()); itr->second->GetPlayer()->UpdateZoneDependentAuras(itr->second->GetPlayer()->GetZoneId()); } } void World::LoadWorldStates() { uint32 oldMSTime = getMSTime(); QueryResult result = CharacterDatabase.Query("SELECT entry, value FROM worldstates"); if (!result) { LOG_WARN("server.loading", ">> Loaded 0 world states. DB table `worldstates` is empty!"); LOG_INFO("server.loading", " "); return; } do { Field* fields = result->Fetch(); _worldstates[fields[0].Get()] = fields[1].Get(); } while (result->NextRow()); LOG_INFO("server.loading", ">> Loaded {} World States in {} ms", _worldstates.size(), GetMSTimeDiffToNow(oldMSTime)); LOG_INFO("server.loading", " "); } // Setting a worldstate will save it to DB void World::setWorldState(uint32 index, uint64 timeValue) { auto const& it = _worldstates.find(index); if (it != _worldstates.end()) { CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_UPD_WORLDSTATE); stmt->SetData(0, uint32(timeValue)); stmt->SetData(1, index); CharacterDatabase.Execute(stmt); } else { CharacterDatabasePreparedStatement* stmt = CharacterDatabase.GetPreparedStatement(CHAR_INS_WORLDSTATE); stmt->SetData(0, index); stmt->SetData(1, uint32(timeValue)); CharacterDatabase.Execute(stmt); } _worldstates[index] = timeValue; } uint64 World::getWorldState(uint32 index) const { auto const& itr = _worldstates.find(index); return itr != _worldstates.end() ? itr->second : 0; } void World::ProcessQueryCallbacks() { _queryProcessor.ProcessReadyCallbacks(); } void World::RemoveOldCorpses() { _timers[WUPDATE_CORPSES].SetCurrent(_timers[WUPDATE_CORPSES].GetInterval()); } bool World::IsPvPRealm() const { return getIntConfig(CONFIG_GAME_TYPE) == REALM_TYPE_PVP || getIntConfig(CONFIG_GAME_TYPE) == REALM_TYPE_RPPVP || getIntConfig(CONFIG_GAME_TYPE) == REALM_TYPE_FFA_PVP; } bool World::IsFFAPvPRealm() const { return getIntConfig(CONFIG_GAME_TYPE) == REALM_TYPE_FFA_PVP; } uint32 World::GetNextWhoListUpdateDelaySecs() { if (_timers[WUPDATE_5_SECS].Passed()) return 1; uint32 t = _timers[WUPDATE_5_SECS].GetInterval() - _timers[WUPDATE_5_SECS].GetCurrent(); t = std::min(t, (uint32)_timers[WUPDATE_5_SECS].GetInterval()); return uint32(std::ceil(t / 1000.0f)); } CliCommandHolder::CliCommandHolder(void* callbackArg, char const* command, Print zprint, CommandFinished commandFinished) : m_callbackArg(callbackArg), m_command(strdup(command)), m_print(zprint), m_commandFinished(commandFinished) { } CliCommandHolder::~CliCommandHolder() { free(m_command); }