From: Rudolf Polzer Date: Mon, 24 Oct 2011 11:19:33 +0000 (+0200) Subject: add a SV_ParseClientCommand hook X-Git-Tag: xonotic-v0.6.0~40^2~16 X-Git-Url: https://git.rm.cloudns.org/?a=commitdiff_plain;h=18d6084f14448cd0d90512b5f3533ff1d9d7b3a8;p=xonotic%2Fxonotic-data.pk3dir.git add a SV_ParseClientCommand hook --- diff --git a/qcsrc/server/clientcommands.qc b/qcsrc/server/clientcommands.qc index 2ffa4afd9..92336234a 100644 --- a/qcsrc/server/clientcommands.qc +++ b/qcsrc/server/clientcommands.qc @@ -145,33 +145,40 @@ float cmd_floodcheck() .float checkfail; void SV_ParseClientCommand(string s) { - string cmd; - float tokens; float i; entity e; - tokens = tokenize_console(s); - - cmd = strtolower(argv(0)); - if(cmd != "reportcvar") - if(cmd != "sentcvar") - if(cmd != "pause") - if(cmd != "prespawn") - if(cmd != "spawn") - if(cmd != "begin") + cmd_argc = tokenize_console(s); + cmd_string = s; + cmd_name = strtolower(argv(0)); + if(cmd_name != "reportcvar") + if(cmd_name != "sentcvar") + if(cmd_name != "pause") + if(cmd_name != "prespawn") + if(cmd_name != "spawn") + if(cmd_name != "begin") { if(cmd_floodcheck()) + { + cmd_string = cmd_name = string_null; // unreference tempstrings return; + } } + if(MUTATOR_CALLHOOK(SV_ParseClientCommand)) { + cmd_string = cmd_name = string_null; // unreference tempstrings + return; // already handled + } + cmd_string = cmd_name = string_null; // unreference tempstrings + if(GameCommand_Vote(s, self)) { return; } else if(GameCommand_MapVote(argv(0))) { return; - } else if(cmd == "checkfail") { + } else if(cmd_name == "checkfail") { print(sprintf("CHECKFAIL: %s (%s) epically failed check %s\n", self.netname, self.netaddress, substring(s, argv_start_index(1), argv_end_index(-1) - argv_start_index(1)))); self.checkfail = 1; - } else if(cmd == "autoswitch") { + } else if(cmd_name == "autoswitch") { // be backwards compatible with older clients (enabled) self.autoswitch = ("0" != argv(1)); string autoswitchmsg; @@ -181,7 +188,7 @@ void SV_ParseClientCommand(string s) { autoswitchmsg = "off"; } sprint(self, strcat("^1autoswitch turned ", autoswitchmsg, "\n")); - } else if(cmd == "clientversion") { + } else if(cmd_name == "clientversion") { if not(self.flags & FL_CLIENT) return; if (argv(1) == "$gameversion") { @@ -201,21 +208,21 @@ void SV_ParseClientCommand(string s) { self.classname = "observer"; stuffcmd(self,"menu_showteamselect\n"); } - } else if(cmd == "reportcvar") { // old system + } else if(cmd_name == "reportcvar") { // old system if(substring(argv(2), 0, 1) == "$") // undefined cvar: use the default value on the server then { s = strcat(substring(s, argv_start_index(0), argv_end_index(1) - argv_start_index(0)), " \"", cvar_defstring(argv(1)), "\""); - tokens = tokenize_console(s); + cmd_argc = tokenize_console(s); } GetCvars(1); - } else if(cmd == "sentcvar") { // new system - if(tokens == 2) // undefined cvar: use the default value on the server then + } else if(cmd_name == "sentcvar") { // new system + if(cmd_argc == 2) // undefined cvar: use the default value on the server then { s = strcat(substring(s, argv_start_index(0), argv_end_index(1) - argv_start_index(0)), " \"", cvar_defstring(argv(1)), "\""); - tokens = tokenize_console(s); + cmd_argc = tokenize_console(s); } GetCvars(1); - } else if(cmd == "spectate") { + } else if(cmd_name == "spectate") { if(cmd_floodcheck()) return; if not(self.flags & FL_CLIENT) @@ -244,7 +251,7 @@ void SV_ParseClientCommand(string s) { sprint(self, "WARNING: you will spectate in the next round.\n"); self.caplayer = 0; } - } else if(cmd == "join") { + } else if(cmd_name == "join") { if not(self.flags & FL_CLIENT) return; if(!g_arena) @@ -265,7 +272,7 @@ void SV_ParseClientCommand(string s) { centerprint(self, PREVENT_JOIN_TEXT); } } - } else if( cmd == "selectteam" ) { + } else if( cmd_name == "selectteam" ) { if not(self.flags & FL_CLIENT) return; if( !teamplay ) { @@ -301,7 +308,7 @@ void SV_ParseClientCommand(string s) { } else { sprint( self, strcat( "selectteam none/red/blue/yellow/pink/auto - \"", argv(1), "\" not recognised\n" ) ); } - } else if(cmd == "ready") { + } else if(cmd_name == "ready") { if not(self.flags & FL_CLIENT) return; @@ -328,54 +335,54 @@ void SV_ParseClientCommand(string s) { sprint(self, "^1Game has already been restarted\n"); } } - } else if(cmd == "maplist") { + } else if(cmd_name == "maplist") { sprint(self, maplist_reply); - } else if(cmd == "lsmaps") { + } else if(cmd_name == "lsmaps") { sprint(self, lsmaps_reply); - } else if(cmd == "lsnewmaps") { + } else if(cmd_name == "lsnewmaps") { sprint(self, lsnewmaps_reply); - } else if(cmd == "records") { + } else if(cmd_name == "records") { for(i = 0; i < 10; ++i) sprint(self, records_reply[i]); - } else if(cmd == "ladder") { + } else if(cmd_name == "ladder") { sprint(self, ladder_reply); - } else if(cmd == "rankings") { + } else if(cmd_name == "rankings") { sprint(self, rankings_reply); - } else if(cmd == "voice") { - if(tokens >= 3) + } else if(cmd_name == "voice") { + if(cmd_argc >= 3) VoiceMessage(argv(1), substring(s, argv_start_index(2), argv_end_index(-1) - argv_start_index(2))); else VoiceMessage(argv(1), ""); - } else if(cmd == "say") { - if(tokens >= 2) + } else if(cmd_name == "say") { + if(cmd_argc >= 2) Say(self, FALSE, world, substring(s, argv_start_index(1), argv_end_index(-1) - argv_start_index(1)), 1); //clientcommand(self, formatmessage(s)); - } else if(cmd == "say_team") { - if(tokens >= 2) + } else if(cmd_name == "say_team") { + if(cmd_argc >= 2) Say(self, TRUE, world, substring(s, argv_start_index(1), argv_end_index(-1) - argv_start_index(1)), 1); //clientcommand(self, formatmessage(s)); - } else if(cmd == "tell") { - e = GetCommandPlayerSlotTargetFromTokenizedCommand(tokens, 1); - if(e && tokens > ParseCommandPlayerSlotTarget_firsttoken) + } else if(cmd_name == "tell") { + e = GetCommandPlayerSlotTargetFromTokenizedCommand(cmd_argc, 1); + if(e && cmd_argc > ParseCommandPlayerSlotTarget_firsttoken) { Say(self, FALSE, e, substring(s, argv_start_index(ParseCommandPlayerSlotTarget_firsttoken), argv_end_index(-1) - argv_start_index(ParseCommandPlayerSlotTarget_firsttoken)), TRUE); } else { - if(tokens > ParseCommandPlayerSlotTarget_firsttoken) + if(cmd_argc > ParseCommandPlayerSlotTarget_firsttoken) trigger_magicear_processmessage_forallears(self, -1, world, substring(s, argv_start_index(ParseCommandPlayerSlotTarget_firsttoken), argv_end_index(-1) - argv_start_index(ParseCommandPlayerSlotTarget_firsttoken))); sprint(self, "ERROR: usage: tell # playerid text...\n"); } //clientcommand(self, formatmessage(s)); - } else if(cmd == "info") { - cmd = cvar_string_builtin(strcat("sv_info_", argv(1))); // This needed fixed for the cvar check - if(cmd == "") + } else if(cmd_name == "info") { + cmd_name = cvar_string_builtin(strcat("sv_info_", argv(1))); // This needed fixed for the cvar check + if(cmd_name == "") sprint(self, "ERROR: unsupported info command\n"); else - wordwrap_sprint(cmd, 1111); - } else if(cmd == "suggestmap") { + wordwrap_sprint(cmd_name, 1111); + } else if(cmd_name == "suggestmap") { sprint(self, strcat(MapVote_Suggest(argv(1)), "\n")); - } else if(cmd == "timeout") { + } else if(cmd_name == "timeout") { if not(self.flags & FL_CLIENT) return; if(autocvar_sv_timeout) { @@ -388,42 +395,42 @@ void SV_ParseClientCommand(string s) { else sprint(self, "^7Error: only players can call a timeout!\n"); } - } else if(cmd == "timein") { + } else if(cmd_name == "timein") { if not(self.flags & FL_CLIENT) return; if(autocvar_sv_timeout) { evaluateTimein(); } - } else if(cmd == "teamstatus") { + } else if(cmd_name == "teamstatus") { Score_NicePrint(self); - } else if(cmd == "cvar_changes") { + } else if(cmd_name == "cvar_changes") { sprint(self, cvar_changes); - } else if(cmd == "cvar_purechanges") { + } else if(cmd_name == "cvar_purechanges") { sprint(self, cvar_purechanges); - } else if(CheatCommand(tokens)) { + } else if(CheatCommand(cmd_argc)) { } else { #if 0 //if(ctf_clientcommand()) // return; // grep for Cmd_AddCommand_WithClientCommand to find them all - if(cmd != "status") - //if(cmd != "say") // handled above - //if(cmd != "say_team") // handled above - if(cmd != "kill") - if(cmd != "pause") - if(cmd != "ping") - if(cmd != "name") - if(cmd != "color") - if(cmd != "rate") - if(cmd != "pmodel") - if(cmd != "playermodel") - if(cmd != "playerskin") - if(cmd != "prespawn") - if(cmd != "spawn") - if(cmd != "begin") - if(cmd != "pings") - if(cmd != "sv_startdownload") - if(cmd != "download") + if(cmd_name != "status") + //if(cmd_name != "say") // handled above + //if(cmd_name != "say_team") // handled above + if(cmd_name != "kill") + if(cmd_name != "pause") + if(cmd_name != "ping") + if(cmd_name != "name") + if(cmd_name != "color") + if(cmd_name != "rate") + if(cmd_name != "pmodel") + if(cmd_name != "playermodel") + if(cmd_name != "playerskin") + if(cmd_name != "prespawn") + if(cmd_name != "spawn") + if(cmd_name != "begin") + if(cmd_name != "pings") + if(cmd_name != "sv_startdownload") + if(cmd_name != "download") { print("WARNING: Invalid clientcommand by ", self.netname, ": ", s, "\n"); return; @@ -431,7 +438,7 @@ void SV_ParseClientCommand(string s) { #endif if(self.jointime > 0 && time > self.jointime + 10 && time > self.nickspamtime) // allow any changes in the first 10 seconds since joining - if(cmd == "name" || cmd == "playermodel") // TODO also playerskin and color? + if(cmd_name == "name" || cmd_name == "playermodel") // TODO also playerskin and color? { if(self.nickspamtime == 0 || time > self.nickspamtime + autocvar_g_nick_flood_timeout) // good, no serious flood diff --git a/qcsrc/server/mutators/base.qh b/qcsrc/server/mutators/base.qh index e26280e61..a088e1c05 100644 --- a/qcsrc/server/mutators/base.qh +++ b/qcsrc/server/mutators/base.qh @@ -159,3 +159,13 @@ MUTATOR_HOOKABLE(PlayerUseKey); // called when the use key is pressed // if MUTATOR_RETURNVALUE is 1, don't do anything // return 1 if the use key actually did something + +MUTATOR_HOOKABLE(SV_ParseClientCommand); + // called when a client command is parsed + // NOTE: hooks MUST start with if(MUTATOR_RETURNVALUE) return MUTATOR_RETURNVALUE; + // NOTE: return 1 if you handled the command, return 0 to continue handling + // NOTE: THESE HOOKS MUST NEVER EVER CALL tokenize() + // INPUT + string cmd_name; // command name + string cmd_string; // whole command, use only if you really have to + float cmd_argc; // also, argv() can be used