This repository has been archived on 2024-10-22. You can view files and clone it, but cannot push or open issues or pull requests.
YimMenu/src/hooks/protections/script_event_handler.cpp
gir489 695a6dd20e
Fixes for b3095 (#2537)
* Refactored TRIGGER_SCRIPT_EVENT for the duplicate f_2 argument they added in 3095.
Fixed OOB gadget spam caused by the tow truck.
Increased max interiors in InteriorControl.
* Updated scripted_game_event.
* Update GTAV-Classes repo.
* Fixed locals for 3095.
* Bump project to Build 3095.
2023-12-14 00:56:40 +01:00

454 lines
13 KiB
C++

#include "backend/player_command.hpp"
#include "gta/net_game_event.hpp"
#include "gta/script_handler.hpp"
#include "gta_util.hpp"
#include "hooking.hpp"
#include "lua/lua_manager.hpp"
#include "util/math.hpp"
#include "util/session.hpp"
#include <network/CNetGamePlayer.hpp>
#include <network/Network.hpp>
#include <script/globals/GPBD_FM_3.hpp>
#include <script/globals/GlobalPlayerBD.hpp>
namespace big
{
void format_string(std::string_view player_name, std::string_view protection_type, bool should_log, bool should_notify)
{
if (should_log)
LOG(WARNING) << "BLOCKED_SCRIPT_EVENT From: " << player_name << " Event Type: " << protection_type;
if (should_notify)
g_notification_service->push_warning("Script Event Protection",
std::format("From: {}\nEvent Type: {}", player_name.data(), protection_type.data()));
}
inline bool is_player_driver_of_local_vehicle(Player sender)
{
auto plyr = g_player_service->get_by_id(sender);
if (!plyr || !plyr->get_current_vehicle() || !g_player_service->get_self()->get_current_vehicle())
return false;
return g_player_service->get_self()->get_current_vehicle()->m_driver == plyr->get_ped();
}
inline bool is_player_our_goon(Player sender)
{
auto& boss_goon = scr_globals::gpbd_fm_3.as<GPBD_FM_3*>()->Entries[self::id].BossGoon;
if (boss_goon.Boss != self::id)
return false;
for (int i = 0; i < boss_goon.Goons.Size; i++)
{
if (boss_goon.Goons[i] == sender)
{
return true;
}
}
return false;
}
inline bool is_player_our_boss(Player sender)
{
return sender == scr_globals::gpbd_fm_3.as<GPBD_FM_3*>()->Entries[self::id].BossGoon.Boss;
}
bool hooks::scripted_game_event(CScriptedGameEvent* scripted_game_event, CNetGamePlayer* player)
{
const auto args = scripted_game_event->m_args;
const auto hash = static_cast<eRemoteEvent>(args[0]);
const auto player_name = player->get_name();
auto plyr = g_player_service->get_by_id(player->m_player_id);
if (g_lua_manager && g_lua_manager->get_module_count() > 0)
{
std::vector<int32_t> script_event_args;
for (int i = 0; i < scripted_game_event->m_args_size; i++)
script_event_args.push_back(args[i]);
auto event_ret = g_lua_manager->trigger_event<menu_event::ScriptedGameEventReceived, bool>((int)player->m_player_id, script_event_args);
if (event_ret.has_value())
return true; // don't care, block event if any bool is returned
}
switch (hash)
{
case eRemoteEvent::Bounty:
if (g.protections.script_events.bounty && args[3] == self::id)
{
g.reactions.bounty.process(plyr);
return true;
}
break;
case eRemoteEvent::CeoKick:
if (player->m_player_id != scr_globals::gpbd_fm_3.as<GPBD_FM_3*>()->Entries[self::id].BossGoon.Boss)
{
g.reactions.ceo_kick.process(plyr);
return true;
}
break;
case eRemoteEvent::CeoMoney:
if (g.protections.script_events.ceo_money
&& player->m_player_id != scr_globals::gpbd_fm_3.as<GPBD_FM_3*>()->Entries[self::id].BossGoon.Boss)
{
g.reactions.ceo_money.process(plyr);
return true;
}
break;
case eRemoteEvent::ClearWantedLevel:
if (g.protections.script_events.clear_wanted_level && !is_player_driver_of_local_vehicle(player->m_player_id))
{
g.reactions.clear_wanted_level.process(plyr);
return true;
}
break;
case eRemoteEvent::Crash: g.reactions.crash.process(plyr); return true;
case eRemoteEvent::Crash2:
if (args[3] > 32) // actual crash condition is if args[2] is above 255
{
g.reactions.crash.process(plyr);
return true;
}
break;
case eRemoteEvent::Crash3:
{
if (isnan(*(float*)&args[4]) || isnan(*(float*)&args[5]))
{
g.reactions.crash.process(plyr);
return true;
}
break;
}
case eRemoteEvent::Notification:
{
switch (static_cast<eRemoteEvent>(args[3]))
{
case eRemoteEvent::NotificationMoneyBanked: // never used
case eRemoteEvent::NotificationMoneyRemoved:
case eRemoteEvent::NotificationMoneyStolen: g.reactions.fake_deposit.process(plyr); return true;
case eRemoteEvent::NotificationCrash1: // this isn't used by the game
session::add_infraction(plyr, Infraction::TRIED_CRASH_PLAYER); // stand user detected
return true;
case eRemoteEvent::NotificationCrash2:
if (!gta_util::find_script_thread(RAGE_JOAAT("gb_salvage")))
{
// This looks like it's meant to trigger a sound crash by spamming too many notifications. We've already patched it, but the notifications are still annoying
session::add_infraction(plyr, Infraction::TRIED_CRASH_PLAYER); // stand user detected
return true;
}
break;
}
break;
}
case eRemoteEvent::ForceMission:
if (g.protections.script_events.force_mission)
{
g.reactions.force_mission.process(plyr);
return true;
}
break;
case eRemoteEvent::GiveCollectible:
if (g.protections.script_events.give_collectible)
{
g.reactions.give_collectible.process(plyr);
return true;
}
break;
case eRemoteEvent::GtaBanner:
if (g.protections.script_events.gta_banner)
{
g.reactions.gta_banner.process(plyr);
return true;
}
break;
case eRemoteEvent::MCTeleport:
if (g.protections.script_events.mc_teleport && args[4] <= 32 && !is_player_our_boss(plyr->id()))
{
for (int i = 0; i < 32; i++)
{
if (args[5 + i] == NETWORK::NETWORK_HASH_FROM_PLAYER_HANDLE(self::id))
{
g.reactions.mc_teleport.process(plyr);
return true;
}
}
}
else if (args[4] > 32)
{
g.reactions.crash.process(plyr);
return true;
}
break;
case eRemoteEvent::PersonalVehicleDestroyed:
if (g.protections.script_events.personal_vehicle_destroyed)
{
g.reactions.personal_vehicle_destroyed.process(plyr);
return true;
}
break;
case eRemoteEvent::RemoteOffradar:
if (g.protections.script_events.remote_off_radar && !is_player_our_boss(plyr->id()) && !is_player_driver_of_local_vehicle(plyr->id()))
{
g.reactions.remote_off_radar.process(plyr);
return true;
}
break;
case eRemoteEvent::TSECommand:
if (g.protections.script_events.rotate_cam && static_cast<eRemoteEvent>(args[3]) == eRemoteEvent::TSECommandRotateCam && !NETWORK::NETWORK_IS_ACTIVITY_SESSION())
{
g.reactions.rotate_cam.process(plyr);
return true;
}
break;
case eRemoteEvent::SendToCayoPerico:
if (g.protections.script_events.send_to_location && args[4] == 0)
{
g.reactions.send_to_location.process(plyr);
return true;
}
break;
case eRemoteEvent::SendToCutscene:
if (g.protections.script_events.send_to_cutscene && !is_player_our_boss(plyr->id()))
{
g.reactions.send_to_cutscene.process(plyr);
return true;
}
break;
case eRemoteEvent::SendToLocation:
{
if (is_player_our_boss(plyr->id()))
break;
bool known_location = false;
if (args[3] == 0 && args[4] == 0)
{
if (args[5] == 4 && args[6] == 0)
{
known_location = true;
if (g.protections.script_events.send_to_location)
{
g.reactions.send_to_location.process(plyr);
return true;
}
}
else if ((args[5] == 3 || args[5] == 4) && args[6] == 1)
{
known_location = true;
if (g.protections.script_events.send_to_location)
{
g.reactions.send_to_location.process(plyr);
return true;
}
}
}
if (!known_location)
{
g.reactions.tse_freeze.process(plyr);
return true;
}
break;
}
case eRemoteEvent::SoundSpam:
{
if (g.protections.script_events.sound_spam && (!plyr || plyr->m_invites_rate_limit.process()))
{
if (plyr->m_invites_rate_limit.exceeded_last_process())
g.reactions.sound_spam.process(plyr);
return true;
}
break;
}
case eRemoteEvent::Spectate:
if (g.protections.script_events.spectate)
{
g.reactions.spectate_notification.process(plyr);
return true;
}
break;
case eRemoteEvent::Teleport:
if (g.protections.script_events.force_teleport && !is_player_driver_of_local_vehicle(player->m_player_id))
{
g.reactions.force_teleport.process(plyr);
return true;
}
break;
case eRemoteEvent::TransactionError: g.reactions.transaction_error.process(plyr); return true;
case eRemoteEvent::VehicleKick:
if (g.protections.script_events.vehicle_kick)
{
g.reactions.vehicle_kick.process(plyr);
return true;
}
break;
case eRemoteEvent::NetworkBail:
session::add_infraction(plyr, Infraction::TRIED_KICK_PLAYER);
g.reactions.network_bail.process(plyr);
return true;
case eRemoteEvent::TeleportToWarehouse:
if (g.protections.script_events.teleport_to_warehouse && !is_player_driver_of_local_vehicle(player->m_player_id))
{
g.reactions.teleport_to_warehouse.process(plyr);
return true;
}
break;
case eRemoteEvent::StartActivity:
{
eActivityType activity = static_cast<eActivityType>(args[3]);
if (g.protections.script_events.start_activity)
{
if (activity == eActivityType::Survival || activity == eActivityType::Mission || activity == eActivityType::Deathmatch || activity == eActivityType::BaseJump || activity == eActivityType::Race)
{
g.reactions.tse_freeze.process(plyr);
return true;
}
else if (activity == eActivityType::Darts)
{
g.reactions.start_activity.process(plyr);
return true;
}
else if (activity == eActivityType::PilotSchool)
{
g.reactions.start_activity.process(plyr);
return true;
}
else if (activity == eActivityType::ImpromptuDeathmatch)
{
g.reactions.start_activity.process(plyr);
return true;
}
else if (activity == eActivityType::DefendSpecialCargo || activity == eActivityType::GunrunningDefend || activity == eActivityType::BikerDefend || args[3] == 238)
{
g.reactions.trigger_business_raid.process(plyr);
return true;
}
}
else if (activity == eActivityType::Tennis)
{
g.reactions.crash.process(plyr);
return true;
}
if (g.protections.script_events.start_activity && !is_player_our_goon(player->m_player_id))
{
g.reactions.start_activity.process(plyr);
return true;
}
break;
}
case eRemoteEvent::InteriorControl:
{
int interior = (int)args[3];
if (interior < 0 || interior > 166) // the upper bound will change after an update
{
if (auto plyr = g_player_service->get_by_id(player->m_player_id))
session::add_infraction(plyr, Infraction::TRIED_KICK_PLAYER);
g.reactions.null_function_kick.process(plyr);
return true;
}
if (NETWORK::NETWORK_IS_ACTIVITY_SESSION())
break;
if (!g_local_player)
break;
if (is_player_our_boss(plyr->id()))
break;
if (is_player_driver_of_local_vehicle(plyr->id()))
break;
if (!plyr->get_ped() || math::distance_between_vectors(*plyr->get_ped()->get_position(), *g_local_player->get_position()) > 75.0f)
{
// g.reactions.send_to_interior.process(plyr); false positives
return true; // this is fine, the game will reject our false positives anyway
}
break;
}
case eRemoteEvent::DestroyPersonalVehicle:
g.reactions.destroy_personal_vehicle.process(plyr);
return true;
case eRemoteEvent::KickFromInterior:
if (scr_globals::globalplayer_bd.as<GlobalPlayerBD*>()->Entries[self::id].SimpleInteriorData.Owner != plyr->id())
{
g.reactions.kick_from_interior.process(plyr);
return true;
}
break;
case eRemoteEvent::TriggerCEORaid:
{
if (auto script = gta_util::find_script_thread(RAGE_JOAAT("freemode")))
{
if (script->m_net_component && ((CGameScriptHandlerNetComponent*)script->m_net_component)->m_host
&& ((CGameScriptHandlerNetComponent*)script->m_net_component)->m_host->m_net_game_player != player)
{
g.reactions.trigger_business_raid.process(plyr);
}
}
return true;
}
case eRemoteEvent::StartScriptProceed:
{
// TODO: Breaks stuff
if (auto script = gta_util::find_script_thread(RAGE_JOAAT("freemode")))
{
if (script->m_net_component && ((CGameScriptHandlerNetComponent*)script->m_net_component)->m_host
&& ((CGameScriptHandlerNetComponent*)script->m_net_component)->m_host->m_net_game_player != player)
{
g.reactions.start_script.process(plyr);
return true;
}
}
break;
}
}
// detect pasted menus setting args[1] to something other than PLAYER_ID()
if (*(int*)&args[1] != player->m_player_id && player->m_player_id != -1)
{
LOG(INFO) << "Hash = " << (int)args[0];
LOG(INFO) << "Sender = " << args[1];
g.reactions.tse_sender_mismatch.process(plyr);
return true;
}
if (g.debug.logs.script_event.logs
&& (!g.debug.logs.script_event.filter_player || g.debug.logs.script_event.player_id == player->m_player_id))
{
std::string script_args = "{ ";
for (std::size_t i = 0; i < scripted_game_event->m_args_size; i++)
{
if (i)
script_args += ", ";
script_args += std::to_string((int)args[i]);
}
script_args += " };";
LOG(VERBOSE) << "Script Event:\n"
<< "\tPlayer: " << player->get_name() << "\n"
<< "\tArgs: " << script_args;
}
if (g.debug.logs.script_event.block_all)
return true;
return false;
}
}