#include #include "Bots.hpp" #include "ClanTags.hpp" #include "GSC/Script.hpp" // From Quake-III #define ANGLE2SHORT(x) ((int)((x) * (USHRT_MAX + 1) / 360.0f) & USHRT_MAX) #define SHORT2ANGLE(x) ((x)* (360.0f / (USHRT_MAX + 1))) namespace Components { constexpr std::size_t MAX_NAME_LENGTH = 16; std::vector Bots::BotNames; const Game::dvar_t* Bots::sv_randomBotNames; const Game::dvar_t* Bots::sv_replaceBots; struct BotMovementInfo { std::int32_t buttons; // Actions std::int8_t forward; std::int8_t right; std::uint16_t weapon; bool active; }; static BotMovementInfo g_botai[Game::MAX_CLIENTS]; struct BotAction { std::string action; std::int32_t key; }; static const BotAction BotActions[] = { { "gostand", Game::CMD_BUTTON_UP }, { "gocrouch", Game::CMD_BUTTON_CROUCH }, { "goprone", Game::CMD_BUTTON_PRONE }, { "fire", Game::CMD_BUTTON_ATTACK }, { "melee", Game::CMD_BUTTON_MELEE }, { "frag", Game::CMD_BUTTON_FRAG }, { "smoke", Game::CMD_BUTTON_OFFHAND_SECONDARY }, { "reload", Game::CMD_BUTTON_RELOAD }, { "sprint", Game::CMD_BUTTON_SPRINT }, { "leanleft", Game::CMD_BUTTON_LEAN_LEFT }, { "leanright", Game::CMD_BUTTON_LEAN_RIGHT }, { "ads", Game::CMD_BUTTON_ADS }, { "holdbreath", Game::CMD_BUTTON_BREATH }, { "usereload", Game::CMD_BUTTON_USE_RELOAD }, { "activate", Game::CMD_BUTTON_ACTIVATE }, }; void Bots::RandomizeBotNames() { std::random_device rd; std::mt19937 gen(rd()); std::ranges::shuffle(BotNames, gen); } std::string Bots::TruncBotString(const std::string& input, const std::size_t length) { if (length > input.size()) { return input; } return input.substr(length); } void Bots::LoadBotNames() { FileSystem::File bots("bots.txt"); if (!bots.exists()) { return; } auto data = Utils::String::Split(bots.getBuffer(), '\n'); for (auto& entry : data) { // Take into account for CR line endings Utils::String::Replace(entry, "\r", ""); // Remove whitespace Utils::String::Trim(entry); if (entry.empty()) { continue; } std::string clanAbbrev; // Check if there is a clan tag if (const auto pos = entry.find(','); pos != std::string::npos) { // Only start copying over from non-null characters (otherwise it can be "<=") if ((pos + 1) < entry.size()) { clanAbbrev = entry.substr(pos + 1); } entry = entry.substr(0, pos); } entry = TruncBotString(entry, MAX_NAME_LENGTH - 1); clanAbbrev = TruncBotString(clanAbbrev, ClanTags::MAX_CLAN_NAME_LENGTH - 1); BotNames.emplace_back(entry, clanAbbrev); } if (sv_randomBotNames->current.enabled) { RandomizeBotNames(); } } int Bots::BuildConnectString(char* buffer, const char* connectString, int num, int, int protocol, int checksum, int statVer, int statStuff, int port) { static std::size_t botId = 0; // Loop over the BotNames vector static bool loadedNames = false; // Load file only once std::string botName; std::string clanName; if (!loadedNames) { loadedNames = true; LoadBotNames(); } if (!BotNames.empty()) { botId %= BotNames.size(); const auto index = botId++; botName = BotNames[index].first; clanName = BotNames[index].second; } else { botName = std::format("bot{}", ++botId); clanName = "BOT"s; } return _snprintf_s(buffer, 0x400, _TRUNCATE, connectString, num, botName.data(), clanName.data(), protocol, checksum, statVer, statStuff, port); } void Bots::Spawn(unsigned int count) { for (std::size_t i = 0; i < count; ++i) { Scheduler::Once([] { auto* ent = Game::SV_AddTestClient(); if (!ent) { return; } Scheduler::Once([ent] { Game::Scr_AddString("autoassign"); Game::Scr_AddString("team_marinesopfor"); Game::Scr_Notify(ent, static_cast(Game::SL_GetString("menuresponse", 0)), 2); Scheduler::Once([ent] { Game::Scr_AddString(Utils::String::Format("class{}", std::rand() % 5)); Game::Scr_AddString("changeclass"); Game::Scr_Notify(ent, static_cast(Game::SL_GetString("menuresponse", 0)), 2); }, Scheduler::Pipeline::SERVER, 1s); }, Scheduler::Pipeline::SERVER, 1s); }, Scheduler::Pipeline::SERVER, 500ms * (i + 1)); } } void Bots::GScr_isTestClient(const Game::scr_entref_t entref) { const auto* ent = Game::GetEntity(entref); if (!ent->client) { Game::Scr_Error("isTestClient: entity must be a player entity"); return; } Game::Scr_AddBool(Game::SV_IsTestClient(ent->s.number) != 0); } void Bots::AddScriptMethods() { GSC::Script::AddMethMultiple(GScr_isTestClient, false, {"IsTestClient", "IsBot"}); // Usage: self IsTestClient(); GSC::Script::AddMethod("BotStop", [](const Game::scr_entref_t entref) // Usage: BotStop(); { const auto* ent = GSC::Script::Scr_GetPlayerEntity(entref); if (!Game::SV_IsTestClient(ent->s.number)) { Game::Scr_Error("BotStop: Can only call on a bot!"); return; } ZeroMemory(&g_botai[entref.entnum], sizeof(BotMovementInfo)); g_botai[entref.entnum].weapon = 1; g_botai[entref.entnum].active = true; }); GSC::Script::AddMethod("BotWeapon", [](const Game::scr_entref_t entref) // Usage: BotWeapon(); { const auto* ent = GSC::Script::Scr_GetPlayerEntity(entref); if (!Game::SV_IsTestClient(ent->s.number)) { Game::Scr_Error("BotWeapon: Can only call on a bot!"); return; } const auto* weapon = Game::Scr_GetString(0); if (!weapon || !*weapon) { g_botai[entref.entnum].weapon = 1; return; } const auto weapId = Game::G_GetWeaponIndexForName(weapon); g_botai[entref.entnum].weapon = static_cast(weapId); g_botai[entref.entnum].active = true; }); GSC::Script::AddMethod("BotAction", [](const Game::scr_entref_t entref) // Usage: BotAction(); { const auto* ent = GSC::Script::Scr_GetPlayerEntity(entref); if (!Game::SV_IsTestClient(ent->s.number)) { Game::Scr_Error("BotAction: Can only call on a bot!"); return; } const auto* action = Game::Scr_GetString(0); if (!action) { Game::Scr_ParamError(0, "BotAction: Illegal parameter!"); return; } if (action[0] != '+' && action[0] != '-') { Game::Scr_ParamError(0, "BotAction: Sign for action must be '+' or '-'"); return; } for (std::size_t i = 0; i < std::extent_v; ++i) { if (Utils::String::ToLower(&action[1]) != BotActions[i].action) continue; if (action[0] == '+') g_botai[entref.entnum].buttons |= BotActions[i].key; else g_botai[entref.entnum].buttons &= ~BotActions[i].key; g_botai[entref.entnum].active = true; return; } Game::Scr_ParamError(0, "BotAction: Unknown action"); }); GSC::Script::AddMethod("BotMovement", [](const Game::scr_entref_t entref) // Usage: BotMovement(, ); { const auto* ent = GSC::Script::Scr_GetPlayerEntity(entref); if (!Game::SV_IsTestClient(ent->s.number)) { Game::Scr_Error("BotMovement: Can only call on a bot!"); return; } const auto forwardInt = std::clamp(Game::Scr_GetInt(0), std::numeric_limits::min(), std::numeric_limits::max()); const auto rightInt = std::clamp(Game::Scr_GetInt(1), std::numeric_limits::min(), std::numeric_limits::max()); g_botai[entref.entnum].forward = static_cast(forwardInt); g_botai[entref.entnum].right = static_cast(rightInt); g_botai[entref.entnum].active = true; }); GSC::Script::AddMethod("SetPing", []([[maybe_unused]] const Game::scr_entref_t entref) {}); } void Bots::BotAiAction(Game::client_t* cl) { if (!cl->gentity) { return; } // Keep test client functionality if (!g_botai[cl - Game::svs_clients].active) { Game::SV_BotUserMove(cl); return; } Game::usercmd_s userCmd; ZeroMemory(&userCmd, sizeof(Game::usercmd_s)); userCmd.serverTime = *Game::svs_time; userCmd.buttons = g_botai[cl - Game::svs_clients].buttons; userCmd.forwardmove = g_botai[cl - Game::svs_clients].forward; userCmd.rightmove = g_botai[cl - Game::svs_clients].right; userCmd.weapon = g_botai[cl - Game::svs_clients].weapon; userCmd.angles[0] = ANGLE2SHORT((cl->gentity->client->ps.viewangles[0] - cl->gentity->client->ps.delta_angles[0])); userCmd.angles[1] = ANGLE2SHORT((cl->gentity->client->ps.viewangles[1] - cl->gentity->client->ps.delta_angles[1])); userCmd.angles[2] = ANGLE2SHORT((cl->gentity->client->ps.viewangles[2] - cl->gentity->client->ps.delta_angles[2])); Game::SV_ClientThink(cl, &userCmd); } __declspec(naked) void Bots::SV_BotUserMove_Hk() { __asm { pushad push edi call BotAiAction add esp, 4 popad ret } } void Bots::G_SelectWeaponIndex(int clientNum, int iWeaponIndex) { if (g_botai[clientNum].active) { g_botai[clientNum].weapon = static_cast(iWeaponIndex); } } __declspec(naked) void Bots::G_SelectWeaponIndex_Hk() { __asm { pushad push [esp + 0x20 + 0x8] push [esp + 0x20 + 0x8] call G_SelectWeaponIndex add esp, 0x8 popad // Code skipped by hook mov eax, [esp + 0x8] push eax push 0x441B85 retn } } int Bots::SV_GetClientPing_Hk(const int clientNum) { AssertIn(clientNum, Game::MAX_CLIENTS); if (Game::SV_IsTestClient(clientNum)) { return -1; } return Game::svs_clients[clientNum].ping; } void Bots::SV_DirectConnect_Full_Check() { if (!sv_replaceBots->current.enabled) { return; } for (auto i = 0; i < (*Game::sv_maxclients)->current.integer; ++i) { auto* cl = &Game::svs_clients[i]; if (cl->bIsTestClient) { Game::SV_DropClient(cl, "EXE_DISCONNECTED", false); return; } } } Bots::Bots() { AssertOffset(Game::client_t, bIsTestClient, 0x41AF0); AssertOffset(Game::client_t, ping, 0x212C8); AssertOffset(Game::client_t, gentity, 0x212A0); // Replace connect string Utils::Hook::Set(0x48ADA6, "connect bot%d \"\\cg_predictItems\\1\\cl_anonymous\\0\\color\\4\\head\\default\\model\\multi\\snaps\\20\\rate\\5000\\name\\%s\\clanAbbrev\\%s\\protocol\\%d\\checksum\\%d\\statver\\%d %u\\qport\\%d\""); // Intercept sprintf for the connect string Utils::Hook(0x48ADAB, BuildConnectString, HOOK_CALL).install()->quick(); Utils::Hook(0x627021, SV_BotUserMove_Hk, HOOK_CALL).install()->quick(); Utils::Hook(0x627241, SV_BotUserMove_Hk, HOOK_CALL).install()->quick(); Utils::Hook(0x441B80, G_SelectWeaponIndex_Hk, HOOK_JUMP).install()->quick(); Utils::Hook(0x459654, SV_GetClientPing_Hk, HOOK_CALL).install()->quick(); sv_randomBotNames = Game::Dvar_RegisterBool("sv_randomBotNames", false, Game::DVAR_NONE, "Randomize the bots' names"); sv_replaceBots = Game::Dvar_RegisterBool("sv_replaceBots", false, Game::DVAR_NONE, "Test clients will be replaced by connecting players when the server is full."); // Reset BotMovementInfo.active when client is dropped Events::OnClientDisconnect([](const int clientNum) -> void { g_botai[clientNum].active = false; }); // Zero the bot command array for (std::size_t i = 0; i < std::extent_v; ++i) { ZeroMemory(&g_botai[i], sizeof(BotMovementInfo)); g_botai[i].weapon = 1; // Prevent the bots from defaulting to the 'none' weapon } Command::Add("spawnBot", [](Command::Params* params) { if (!Dedicated::IsRunning()) { Logger::Print("Server is not running.\n"); return; } std::size_t count = 1; if (params->size() > 1) { if (params->get(1) == "all"s) { count = Game::MAX_CLIENTS; } else { char* end; const auto* input = params->get(1); count = std::strtoul(input, &end, 10); if (input == end) { Logger::Warning(Game::CON_CHANNEL_DONT_FILTER, "{} is not a valid input\nUsage: {} optional or optional <\"all\">\n", input, params->get(0)); return; } } } count = std::min(Game::MAX_CLIENTS, count); Logger::Print("Spawning {} {}", count, (count == 1 ? "bot" : "bots")); Spawn(count); }); AddScriptMethods(); // In case a loaded mod didn't call "BotStop" before the VM shutdown Events::OnVMShutdown([] { for (std::size_t i = 0; i < std::extent_v; ++i) { g_botai[i].active = false; } }); } }