216 lines
4.9 KiB
C++

#include <std_include.hpp>
#include "loader/component_loader.hpp"
#include "game/game.hpp"
#include "command.hpp"
#include "console.hpp"
#include "filesystem.hpp"
#include "network.hpp"
#include "party.hpp"
#include "scheduler.hpp"
#include "server_list.hpp"
#include <utils/hook.hpp>
#include <utils/string.hpp>
namespace bots
{
namespace
{
constexpr std::size_t MAX_NAME_LENGTH = 16;
bool can_add()
{
return party::get_client_count() < game::mp::svs->clientCount;
}
void bot_team_join(const int entity_num)
{
// schedule the select team call
scheduler::once([entity_num]()
{
game::SV_ExecuteClientCommand(&game::mp::svs->clients[entity_num],
utils::string::va("lui 68 2 %i", *game::mp::sv_serverId_value),
false);
// scheduler the select class call
scheduler::once([entity_num]()
{
game::SV_ExecuteClientCommand(&game::mp::svs->clients[entity_num],
utils::string::va("lui 5 %i %i", (rand() % 5) + 10,
*game::mp::sv_serverId_value), false);
}, scheduler::pipeline::server, 1s);
}, scheduler::pipeline::server, 1s);
}
void bot_team(const int entity_num)
{
if (game::SV_BotIsBot(game::mp::g_entities[entity_num].s.clientNum))
{
if (game::mp::g_entities[entity_num].client->sess.cs.team == game::mp::team_t::TEAM_SPECTATOR)
{
bot_team_join(entity_num);
}
scheduler::once([entity_num]()
{
bot_team(entity_num);
}, scheduler::pipeline::server, 3s);
}
}
void spawn_bot(const int entity_num)
{
scheduler::once([entity_num]()
{
game::SV_SpawnTestClient(&game::mp::g_entities[entity_num]);
bot_team(entity_num);
}, scheduler::pipeline::server, 1s);
}
void add_bot()
{
if (!can_add())
{
return;
}
const auto* bot_name = game::SV_BotGetRandomName();
const auto* bot_ent = game::SV_AddBot(bot_name, 26, 62, 0);
if (bot_ent)
{
spawn_bot(bot_ent->s.number);
}
}
utils::hook::detour get_bot_name_hook;
volatile bool bot_names_received = false;
std::vector<std::string> bot_names;
bool should_use_remote_bot_names()
{
#ifdef ALLOW_CUSTOM_BOT_NAMES
return !filesystem::exists("bots.txt");
#else
return true;
#endif
}
void parse_bot_names_from_file()
{
std::string data;
filesystem::read_file("bots.txt", &data);
if (data.empty())
{
return;
}
auto name_list = utils::string::split(data, '\n');
for (auto& entry : name_list)
{
// Take into account CR line endings
entry = utils::string::replace(entry, "\r", "");
if (entry.empty())
{
continue;
}
entry = entry.substr(0, MAX_NAME_LENGTH - 1);
bot_names.emplace_back(entry);
}
}
const char* get_random_bot_name()
{
if (!bot_names_received && bot_names.empty())
{
// last attempt to use custom names if they can be found
parse_bot_names_from_file();
}
if (bot_names.empty())
{
return get_bot_name_hook.invoke<const char*>();
}
const auto index = std::rand() % bot_names.size();
const auto& name = bot_names.at(index);
return utils::string::va("%.*s", static_cast<int>(name.size()), name.data());
}
void update_bot_names()
{
bot_names_received = false;
game::netadr_t master{};
if (server_list::get_master_server(master))
{
console::info("Getting bots...\n");
network::send(master, "getbots");
}
}
}
class component final : public component_interface
{
public:
void post_unpack() override
{
if (game::environment::is_sp())
{
return;
}
get_bot_name_hook.create(game::SV_BotGetRandomName, get_random_bot_name);
command::add("spawnBot", [](const command::params& params)
{
if (!game::SV_Loaded()) return;
std::size_t num_bots = 1;
if (params.size() == 2)
{
num_bots = std::strtoul(params.get(1), nullptr, 10);
}
num_bots = std::min(num_bots, static_cast<std::size_t>(game::mp::svs->clientCount));
console::info("Spawning %zu %s\n", num_bots, (num_bots == 1 ? "bot" : "bots"));
for (std::size_t i = 0; i < num_bots; ++i)
{
scheduler::once(add_bot, scheduler::pipeline::server, 100ms * i);
}
});
if (should_use_remote_bot_names())
{
scheduler::on_game_initialized([]()
{
update_bot_names();
scheduler::loop(update_bot_names, scheduler::main, 1h);
}, scheduler::main);
}
else
{
parse_bot_names_from_file();
}
network::on("getbotsResponse", [](const game::netadr_t& target, const std::string& data)
{
game::netadr_t master{};
if (server_list::get_master_server(master) && !bot_names_received && target == master)
{
bot_names = utils::string::split(data, '\n');
console::info("Got %zu names from the master server\n", bot_names.size());
bot_names_received = true;
}
});
}
};
}
REGISTER_COMPONENT(bots::component)