/** * basecommands.sp * Implements basic admin commands. * This file is part of SourceMod, Copyright (C) 2004-2007 AlliedModders LLC * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * Version: $Id$ */ #include <sourcemod> #pragma semicolon 1 public Plugin:myinfo = { name = "Basic Commands", author = "AlliedModders LLC", description = "Basic Admin Commands", version = SOURCEMOD_VERSION, url = "http://www.sourcemod.net/" }; new Handle:hBanForward = INVALID_HANDLE; new Handle:hAddBanForward = INVALID_HANDLE; new Handle:hBanRemoved = INVALID_HANDLE; public OnPluginStart() { LoadTranslations("common.cfg"); LoadTranslations("plugin.basecommands.cfg"); RegAdminCmd("sm_kick", Command_Kick, ADMFLAG_KICK, "sm_kick <#userid|name> [reason]"); RegAdminCmd("sm_map", Command_Map, ADMFLAG_CHANGEMAP, "sm_map <map>"); RegAdminCmd("sm_rcon", Command_Rcon, ADMFLAG_RCON, "sm_rcon <args>"); RegAdminCmd("sm_cvar", Command_Cvar, ADMFLAG_CONVARS, "sm_cvar <cvar> [value]"); RegAdminCmd("sm_execcfg", Command_ExecCfg, ADMFLAG_CONFIG, "sm_execcfg <filename>"); RegAdminCmd("sm_who", Command_Who, ADMFLAG_GENERIC, "sm_who [#userid|name]"); RegAdminCmd("sm_ban", Command_Ban, ADMFLAG_BAN, "sm_ban <#userid|name> <minutes|0> [reason]"); RegAdminCmd("sm_unban", Command_Unban, ADMFLAG_UNBAN, "sm_unban <steamid>"); RegAdminCmd("sm_addban", Command_AddBan, ADMFLAG_ROOT, "sm_addban <time> <steamid> [reason]"); RegAdminCmd("sm_banip", Command_BanIp, ADMFLAG_ROOT, "sm_banip <time> <ip> [reason]"); hBanForward = CreateGlobalForward("OnClientBanned", ET_Ignore, Param_Cell, Param_Cell, Param_Cell, Param_String); hAddBanForward = CreateGlobalForward("OnBanAdded", ET_Ignore, Param_Cell, Param_String, Param_Cell, Param_String); hBanRemoved = CreateGlobalForward("OnBanRemoved", ET_Ignore, Param_Cell, Param_String); } public Action:Command_BanIp(client, args) { if (args < 2) { ReplyToCommand(client, "[SM] Usage: sm_banip <time> <ip> [reason]"); return Plugin_Handled; } new String:arg[50], String:time[20]; GetCmdArg(1, time, sizeof(time)); GetCmdArg(2, arg, sizeof(arg)); new minutes = StringToInt(time); new String:reason[128]; if (args >= 3) { GetCmdArg(3, reason, sizeof(reason)); } LogMessage("\"%L\" added ban (minutes \"%d\") (ip \"%s\") (reason \"%s\")", client, minutes, arg, reason); Call_StartForward(hAddBanForward); Call_PushCell(client); Call_PushString(arg); Call_PushCell(minutes); Call_PushString(reason); Call_Finish(); ServerCommand("banip %d %s", minutes, arg); ServerCommand("writeip"); ReplyToCommand(client, "[SM] %t", "Ban added"); return Plugin_Handled; } public Action:Command_AddBan(client, args) { if (args < 2) { ReplyToCommand(client, "[SM] Usage: sm_addban <time> <steamid> [reason]"); return Plugin_Handled; } new String:arg[50], String:time[20]; GetCmdArg(1, time, sizeof(time)); GetCmdArg(2, arg, sizeof(arg)); new minutes = StringToInt(time); new String:reason[128]; if (args >= 3) { GetCmdArg(3, reason, sizeof(reason)); } LogMessage("\"%L\" added ban (minutes \"%d\") (id \"%s\") (reason \"%s\")", client, minutes, arg, reason); Call_StartForward(hAddBanForward); Call_PushCell(client); Call_PushString(arg); Call_PushCell(minutes); Call_PushString(reason); Call_Finish(); ServerCommand("banid %d %s", minutes, arg); ServerCommand("writeid"); ReplyToCommand(client, "[SM] %t", "Ban added"); return Plugin_Handled; } public Action:Command_Unban(client, args) { if (args < 1) { ReplyToCommand(client, "[SM] Usage: sm_unban <steamid>"); return Plugin_Handled; } new String:arg[50], start=0; new String:new_arg[50]; GetCmdArgString(arg, sizeof(arg)); if (strncmp(arg, "STEAM_0:", 8) == 0) { start = 8; } else if (strncmp(arg, "0:1:", 4) == 0 || strncmp(arg, "0:0:", 4) == 0) { start = 2; } LogMessage("\"%L\" removed ban (filter \"%s\")", arg[start]); Format(new_arg, sizeof(new_arg), "STEAM_0:%s", arg[start]); Call_StartForward(hBanRemoved); Call_PushCell(client); Call_PushString(new_arg); Call_Finish(); ServerCommand("removeid \"%s\"", arg[start]); ServerCommand("writeid"); ReplyToCommand(client, "[SM] %t", "Removed bans matching", arg); return Plugin_Handled; } public Action:Command_Ban(client, args) { if (args < 2) { ReplyToCommand(client, "[SM] Usage: sm_ban <#userid|name> <minutes|0> [reason]"); return Plugin_Handled; } new String:arg[65]; GetCmdArg(1, arg, sizeof(arg)); new clients[2]; new numClients = SearchForClients(arg, clients, 2); if (numClients == 0) { ReplyToCommand(client, "[SM] %t", "No matching client"); return Plugin_Handled; } else if (numClients > 1) { ReplyToCommand(client, "[SM] %t", "More than one client matches", arg); return Plugin_Handled; } else if (!CanUserTarget(client, clients[0])) { ReplyToCommand(client, "[SM] %t", "Unable to target"); return Plugin_Handled; } else if (IsFakeClient(clients[0])) { ReplyToCommand(client, "[SM] %t", "Cannot target bot"); return Plugin_Handled; } new String:s_time[12]; GetCmdArg(2, s_time, sizeof(s_time)); new time = StringToInt(s_time); decl String:reason[128]; if (args >= 3) { GetCmdArg(3, reason, sizeof(reason)); } else { reason[0] = '\0'; } new userid = GetClientUserId(clients[0]); GetClientName(clients[0], arg, sizeof(arg)); if (time) { if (reason[0] == '\0') { ShowActivity(client, "%t", "Permabanned player", arg); } else { ShowActivity(client, "%t", "Permabanned player reason", arg, reason); } } else { if (reason[0] == '\0') { ShowActivity(client, "%t", "Banned player", arg, time); } else { ShowActivity(client, "%t", "Banned player reason", arg, time, reason); } } LogMessage("\"%L\" banned \"%L\" (minutes \"%d\") (reason \"%s\")", client, clients[0], time, reason); /* Fire the ban forward */ Call_StartForward(hBanForward); Call_PushCell(client); Call_PushCell(clients[0]); Call_PushCell(time); Call_PushString(reason); Call_Finish(); if (reason[0] == '\0') { strcopy(reason, sizeof(reason), "Banned"); } ServerCommand("banid %d %d", time, userid); ServerCommand("kickid %d \"%s\"", userid, reason); if (time == 0) { ServerCommand("writeid"); } return Plugin_Handled; } #define FLAG_STRINGS 14 new String:g_FlagNames[FLAG_STRINGS][20] = { "reservation", "admin", "kick", "ban", "unban", "slay", "map", "cvars", "cfg", "chat", "vote", "pass", "rcon", "cheat" }; FlagsToString(String:buffer[], maxlength, flags) { new String:joins[FLAG_STRINGS][20]; new total; for (new i=0; i<FLAG_STRINGS; i++) { if (flags & (1<<i)) { strcopy(joins[total++], 20, g_FlagNames[i]); } } ImplodeStrings(joins, total, ", ", buffer, maxlength); } public Action:Command_Who(client, args) { if (args < 1) { /* Display header */ new String:t_access[16], String:t_name[16]; Format(t_access, sizeof(t_access), "%t", "Access", client); Format(t_name, sizeof(t_name), "%t", "Name", client); PrintToConsole(client, "%-24.23s %s", t_name, t_access); /* List all players */ new maxClients = GetMaxClients(); new String:flagstring[255]; for (new i=1; i<=maxClients; i++) { if (!IsClientInGame(i)) { continue; } new flags = GetUserFlagBits(i); if (flags == 0) { strcopy(flagstring, sizeof(flagstring), "none"); } else if (flags & ADMFLAG_ROOT) { strcopy(flagstring, sizeof(flagstring), "root"); } else { FlagsToString(flagstring, sizeof(flagstring), flags); } decl String:name[65]; GetClientName(i, name, sizeof(name)); PrintToConsole(client, "%d. %-24.23s %s", i, name, flagstring); } if (GetCmdReplySource() == SM_REPLY_TO_CHAT) { ReplyToCommand(client, "[SM] %t", "See console for output"); } return Plugin_Handled; } new String:arg[65]; GetCmdArg(1, arg, sizeof(arg)); new clients[2]; new numClients = SearchForClients(arg, clients, 2); if (numClients == 0) { ReplyToCommand(client, "[SM] %t", "No matching client"); return Plugin_Handled; } else if (numClients > 1) { ReplyToCommand(client, "[SM] %t", "More than one client matches", arg); return Plugin_Handled; } new flags = GetUserFlagBits(clients[0]); new String:flagstring[255]; if (flags == 0) { strcopy(flagstring, sizeof(flagstring), "none"); } else if (flags & ADMFLAG_ROOT) { strcopy(flagstring, sizeof(flagstring), "root"); } else { FlagsToString(flagstring, sizeof(flagstring), flags); } ReplyToCommand(client, "[SM] %t: %s", "Access", flagstring); return Plugin_Handled; } public Action:Command_ExecCfg(client, args) { if (args < 1) { ReplyToCommand(client, "[SM] Usage: sm_execcfg <filename>"); return Plugin_Handled; } new String:path[64] = "cfg/"; GetCmdArg(1, path[4], sizeof(path)-4); if (!FileExists(path)) { ReplyToCommand(client, "[SM] %t", "Config not found", path[4]); return Plugin_Handled; } ShowActivity(client, "%t", "Executed config", path[4]); LogMessage("\"%L\" executed config (file \"%s\")", client, path[4]); ServerCommand("exec \"%s\"", path[4]); return Plugin_Handled; } public Action:Command_Cvar(client, args) { if (args < 1) { ReplyToCommand(client, "[SM] Usage: sm_cvar <cvar> [value]"); return Plugin_Handled; } new String:cvarname[33]; GetCmdArg(1, cvarname, sizeof(cvarname)); new Handle:hndl = FindConVar(cvarname); if (hndl == INVALID_HANDLE) { ReplyToCommand(client, "[SM] %t", "Unable to find cvar", cvarname); return Plugin_Handled; } new bool:allowed = false; if (GetConVarFlags(hndl) & FCVAR_PROTECTED) { /* If they're root, allow anything */ if ((GetUserFlagBits(client) & ADMFLAG_ROOT) == ADMFLAG_ROOT) { allowed = true; } /* If they're not root, and getting sv_password, see if they have ADMFLAG_PASSWORD */ else if (StrEqual(cvarname, "sv_password") && (GetUserFlagBits(client) & ADMFLAG_PASSWORD)) { allowed = true; } } /* Do a check for the cheat cvar */ else if (StrEqual(cvarname, "sv_cheats")) { if (GetUserFlagBits(client) & ADMFLAG_CHEATS) { allowed = true; } } /* If we drop down to here, it was a normal cvar. */ else { allowed = true; } if (!allowed) { ReplyToCommand(client, "[SM] %t", "No access to cvar"); return Plugin_Handled; } decl String:value[255]; if (args < 2) { GetConVarString(hndl, value, sizeof(value)); ReplyToCommand(client, "[SM] %t", "Value of cvar", cvarname, value); return Plugin_Handled; } GetCmdArg(2, value, sizeof(value)); if ((GetConVarFlags(hndl) & FCVAR_PROTECTED) != FCVAR_PROTECTED) { ShowActivity(client, "%t", "Cvar changed", cvarname, value); } else { ReplyToCommand(client, "[SM] %t", "Cvar changed", cvarname, value); } LogMessage("\"%L\" changed cvar (cvar \"%s\") (value \"%s\")", client, cvarname, value); SetConVarString(hndl, value); return Plugin_Handled; } public Action:Command_Rcon(client, args) { if (args < 1) { ReplyToCommand(client, "[SM] Usage: sm_rcon <args>"); return Plugin_Handled; } decl String:argstring[255]; GetCmdArgString(argstring, sizeof(argstring)); LogMessage("\"%L\" console command (cmdline \"%s\")", client, argstring); ServerCommand("%s", argstring); return Plugin_Handled; } public Action:Command_Map(client, args) { if (args < 1) { ReplyToCommand(client, "[SM] Usage: sm_map <map>"); return Plugin_Handled; } new String:map[64]; GetCmdArg(1, map, sizeof(map)); if (!IsMapValid(map)) { ReplyToCommand(client, "[SM] %t", "Map was not found", map); return Plugin_Handled; } ShowActivity(client, "%t", "Changing map", map); LogMessage("\"%L\" changed map to \"%s\"", client, map); new Handle:dp; CreateDataTimer(3.0, Timer_ChangeMap, dp); WritePackString(dp, map); return Plugin_Handled; } public Action:Timer_ChangeMap(Handle:timer, Handle:dp) { new String:map[65]; ResetPack(dp); ReadPackString(dp, map, sizeof(map)); ServerCommand("changelevel \"%s\"", map); return Plugin_Stop; } public Action:Command_Kick(client, args) { if (args < 1) { ReplyToCommand(client, "[SM] Usage: sm_kick <#userid|name> [reason]"); return Plugin_Handled; } new String:arg[65]; GetCmdArg(1, arg, sizeof(arg)); new clients[2]; new numClients = SearchForClients(arg, clients, 2); if (numClients == 0) { ReplyToCommand(client, "[SM] %t", "No matching client"); return Plugin_Handled; } else if (numClients > 1) { ReplyToCommand(client, "[SM] %t", "More than one client matches", arg); return Plugin_Handled; } else if (!CanUserTarget(client, clients[0])) { ReplyToCommand(client, "[SM] %t", "Unable to target"); return Plugin_Handled; } new userid = GetClientUserId(clients[0]); new String:name[65]; GetClientName(clients[0], name, sizeof(name)); decl String:reason[256]; if (args < 2) { /* Safely null terminate */ reason[0] = '\0'; } else { GetCmdArg(2, reason, sizeof(reason)); } ShowActivity(client, "%t", "Kicked player", name); LogMessage("\"%L\" kicked \"%L\" (reason \"%s\")", client, clients[0], reason); if (args < 2) { ServerCommand("kickid %d", userid); } else { ServerCommand("kickid %d \"%s\"", userid, reason); } return Plugin_Handled; }