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
Yimura 335e9325b0
feat(logger): use AsyncLogger instead of g3log (#882)
* feat(exception_handler): skip certain exception codes
* feat(zydis): Disable BUILD_DOXYGEN and FEATURE_ENCODER
* feat(cmake): link dbghelp
* feat(exception_handler): implement stack dumper
* feat(logger): differentiate between wine and non-wine
* feat(logger): add NO_COLOR env var support
* fix(logger): fix wine logging (#960)
* feat(exception_handler): added string logging of exception
* fix(logger): exception access violation NO_COLOR

--------

Co-authored-by: Aure7138 <100095051+Aure7138@users.noreply.github.com>
Co-authored-by: tupoy-ya <72797377+tupoy-ya@users.noreply.github.com>
2023-02-08 22:36:55 +00:00

414 lines
11 KiB
C++

#include "hooking.hpp"
#include "gta_util.hpp"
#include "util/session.hpp"
#include "gta/net_game_event.hpp"
#include "backend/player_command.hpp"
#include "gta/script_handler.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;
}
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);
switch (hash)
{
case eRemoteEvent::Bounty:
if (g.protections.script_events.bounty && args[2] == 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[2] > 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[3]) || isnan(*(float*)&args[4]))
{
g.reactions.crash.process(plyr);
return true;
}
break;
}
case eRemoteEvent::Notification:
switch (static_cast<eRemoteEvent>(args[2]))
{
case eRemoteEvent::NotificationMoneyBanked:
case eRemoteEvent::NotificationMoneyRemoved:
case eRemoteEvent::NotificationMoneyStolen:
if (g.protections.script_events.fake_deposit)
{
g.reactions.fake_deposit.process(plyr);
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[3] <= 32)
{
g.reactions.mc_teleport.process(plyr);
return true;
}
else if (args[3] > 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 && player->m_player_id != scr_globals::gpbd_fm_3.as<GPBD_FM_3*>()->Entries[self::id].BossGoon.Boss)
{
g.reactions.remote_off_radar.process(plyr);
return true;
}
break;
case eRemoteEvent::TSECommand:
if (g.protections.script_events.rotate_cam && static_cast<eRemoteEvent>(args[2]) == eRemoteEvent::TSECommandRotateCam && !gta_util::get_network()->m_is_activity_session)
{
g.reactions.rotate_cam.process(plyr);
return true;
}
break;
case eRemoteEvent::SendToCayoPerico:
if (g.protections.script_events.send_to_location && args[3] == 0)
{
g.reactions.send_to_location.process(plyr);
return true;
}
break;
case eRemoteEvent::SendToCutscene:
if (g.protections.script_events.send_to_cutscene && player->m_player_id != scr_globals::gpbd_fm_3.as<GPBD_FM_3*>()->Entries[self::id].BossGoon.Boss)
{
g.reactions.send_to_cutscene.process(plyr);
return true;
}
break;
case eRemoteEvent::SendToLocation:
{
bool known_location = false;
if (args[2] == 0 && args[3] == 0)
{
if (args[4] == 4 && args[5] == 0)
{
known_location = true;
if (g.protections.script_events.send_to_location)
{
g.reactions.send_to_location.process(plyr);
return true;
}
}
else if ((args[4] == 3 || args[4] == 4) && args[5] == 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:
{
auto plyr = g_player_service->get_by_id(player->m_player_id);
if (g.protections.script_events.sound_spam && (!plyr || plyr->m_invites_rate_limit.process()))
{
if (plyr && 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:
if (auto plyr = g_player_service->get_by_id(player->m_player_id))
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[2]);
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[2] == 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[2];
if (interior < 0 || interior > 161) // 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;
}
break;
}
case eRemoteEvent::SMS:
if (g.protections.script_events.send_sms)
{
if (g.session.kick_chat_spammers)
{
if (auto plyr = g_player_service->get_by_id(player->m_player_id))
{
((player_command*)command::get(RAGE_JOAAT("breakup")))->call(plyr, {});
}
}
return true;
}
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 && script->m_net_component->m_host && 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 && script->m_net_component->m_host && 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;
}
return false;
}
}