/** * vim: set ts=4 : * ============================================================================= * SourceMod * Copyright (C) 2004-2008 AlliedModders LLC. All rights reserved. * ============================================================================= * * This program is free software; you can redistribute it and/or modify it under * the terms of the GNU General Public License, version 3.0, as published by the * Free Software Foundation. * * 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, see . * * As a special exception, AlliedModders LLC gives you permission to link the * code of this program (as well as its derivative works) to "Half-Life 2," the * "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software * by the Valve Corporation. You must obey the GNU General Public License in * all respects for all other code used. Additionally, AlliedModders LLC grants * this exception to all derivative works. AlliedModders LLC defines further * exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007), * or . * * Version: $Id$ */ #include #include #include #include "AdminCache.h" #include "ShareSys.h" #include "ForwardSys.h" #include "PlayerManager.h" #include "ConCmdManager.h" #include "Logger.h" #include "sourcemod.h" #include "sm_stringutil.h" #include "sourcemm_api.h" #include "sm_srvcmds.h" #define LEVEL_STATE_NONE 0 #define LEVEL_STATE_LEVELS 1 #define LEVEL_STATE_FLAGS 2 AdminCache g_Admins; char g_ReverseFlags[26]; AdminFlag g_FlagLetters[26]; bool g_FlagSet[26]; ConVar sm_immunity_mode("sm_immunity_mode", "1", FCVAR_SPONLY, "Mode for deciding immunity protection"); /* Default flags */ AdminFlag g_DefaultFlags[26] = { Admin_Reservation, Admin_Generic, Admin_Kick, Admin_Ban, Admin_Unban, Admin_Slay, Admin_Changemap, Admin_Convars, Admin_Config, Admin_Chat, Admin_Vote, Admin_Password, Admin_RCON, Admin_Cheats, Admin_Custom1, Admin_Custom2, Admin_Custom3, Admin_Custom4, Admin_Custom5, Admin_Custom6, Admin_Generic, Admin_Generic, Admin_Generic, Admin_Generic, Admin_Generic, Admin_Root }; class FlagReader : public ITextListener_SMC { public: void LoadLevels() { if (!Parse()) { memcpy(g_FlagLetters, g_DefaultFlags, sizeof(AdminFlag) * 26); for (unsigned int i=0; i<20; i++) { g_FlagSet[i] = true; } g_FlagSet[25] = true; } } private: bool Parse() { SMCStates states; SMCError error; m_bFileNameLogged = false; g_SourceMod.BuildPath(Path_SM, m_File, sizeof(m_File), "configs/admin_levels.cfg"); if ((error = textparsers->ParseFile_SMC(m_File, this, &states)) != SMCError_Okay) { const char *err_string = textparsers->GetSMCErrorString(error); if (!err_string) { err_string = "Unknown error"; } ParseError(NULL, "Error %d (%s)", error, err_string); return false; } return true; } void ReadSMC_ParseStart() { m_LevelState = LEVEL_STATE_NONE; m_IgnoreLevel = 0; memset(g_FlagSet, 0, sizeof(g_FlagSet)); } SMCResult ReadSMC_NewSection(const SMCStates *states, const char *name) { if (m_IgnoreLevel) { m_IgnoreLevel++; return SMCResult_Continue; } if (m_LevelState == LEVEL_STATE_NONE) { if (strcmp(name, "Levels") == 0) { m_LevelState = LEVEL_STATE_LEVELS; } else { m_IgnoreLevel++; } } else if (m_LevelState == LEVEL_STATE_LEVELS) { if (strcmp(name, "Flags") == 0) { m_LevelState = LEVEL_STATE_FLAGS; } else { m_IgnoreLevel++; } } else { m_IgnoreLevel++; } return SMCResult_Continue; } SMCResult ReadSMC_KeyValue(const SMCStates *states, const char *key, const char *value) { if (m_LevelState != LEVEL_STATE_FLAGS || m_IgnoreLevel) { return SMCResult_Continue; } unsigned char c = (unsigned)value[0]; if (c < (unsigned)'a' || c > (unsigned)'z') { ParseError(states, "Flag \"%c\" is not a lower-case ASCII letter", c); return SMCResult_Continue; } c -= (unsigned)'a'; if (!g_Admins.FindFlag(key, &g_FlagLetters[c])) { ParseError(states, "Unrecognized admin level \"%s\"", key); return SMCResult_Continue; } g_FlagSet[c] = true; return SMCResult_Continue; } SMCResult ReadSMC_LeavingSection(const SMCStates *states) { if (m_IgnoreLevel) { m_IgnoreLevel--; return SMCResult_Continue; } if (m_LevelState == LEVEL_STATE_FLAGS) { m_LevelState = LEVEL_STATE_LEVELS; return SMCResult_Halt; } else if (m_LevelState == LEVEL_STATE_LEVELS) { m_LevelState = LEVEL_STATE_NONE; } return SMCResult_Continue; } void ParseError(const SMCStates *states, const char *message, ...) { va_list ap; char buffer[256]; va_start(ap, message); UTIL_FormatArgs(buffer, sizeof(buffer), message, ap); va_end(ap); if (!m_bFileNameLogged) { g_Logger.LogError("[SM] Parse error(s) detected in file \"%s\":", m_File); m_bFileNameLogged = true; } g_Logger.LogError("[SM] (Line %d): %s", states ? states->line : 0, buffer); } private: bool m_bFileNameLogged; char m_File[PLATFORM_MAX_PATH]; int m_LevelState; int m_IgnoreLevel; } s_FlagReader; AdminCache::AdminCache() { m_pCmdOverrides = sm_trie_create(); m_pCmdGrpOverrides = sm_trie_create(); m_pStrings = new BaseStringTable(1024); m_pMemory = m_pStrings->GetMemTable(); m_FreeGroupList = m_FirstGroup = m_LastGroup = INVALID_GROUP_ID; m_FreeUserList = m_FirstUser = m_LastUser = INVALID_ADMIN_ID; m_pGroups = sm_trie_create(); m_pCacheFwd = NULL; m_FirstGroup = -1; m_pAuthTables = sm_trie_create(); m_InvalidatingAdmins = false; m_destroying = false; m_pLevelNames = sm_trie_create(); } AdminCache::~AdminCache() { m_destroying = true; DumpAdminCache(AdminCache_Overrides, false); DumpAdminCache(AdminCache_Groups, false); sm_trie_destroy(m_pCmdGrpOverrides); sm_trie_destroy(m_pCmdOverrides); if (m_pGroups) { sm_trie_destroy(m_pGroups); } List::iterator iter; for (iter=m_AuthMethods.begin(); iter!=m_AuthMethods.end(); iter++) { sm_trie_destroy((*iter).table); } sm_trie_destroy(m_pAuthTables); delete m_pStrings; sm_trie_destroy(m_pLevelNames); } void AdminCache::OnSourceModStartup(bool late) { RegisterAuthIdentType("steam"); RegisterAuthIdentType("name"); RegisterAuthIdentType("ip"); NameFlag("reservation", Admin_Reservation); NameFlag("kick", Admin_Kick); NameFlag("generic", Admin_Generic); NameFlag("ban", Admin_Ban); NameFlag("unban", Admin_Unban); NameFlag("slay", Admin_Slay); NameFlag("changemap", Admin_Changemap); NameFlag("cvars", Admin_Convars); NameFlag("config", Admin_Config); NameFlag("chat", Admin_Chat); NameFlag("vote", Admin_Vote); NameFlag("password", Admin_Password); NameFlag("rcon", Admin_RCON); NameFlag("cheats", Admin_Cheats); NameFlag("root", Admin_Root); NameFlag("custom1", Admin_Custom1); NameFlag("custom2", Admin_Custom2); NameFlag("custom3", Admin_Custom3); NameFlag("custom4", Admin_Custom4); NameFlag("custom5", Admin_Custom5); NameFlag("custom6", Admin_Custom6); } void AdminCache::OnSourceModAllInitialized() { m_pCacheFwd = g_Forwards.CreateForward("OnRebuildAdminCache", ET_Ignore, 1, NULL, Param_Cell); g_ShareSys.AddInterface(NULL, this); } void AdminCache::OnSourceModLevelChange(const char *mapName) { int i; AdminFlag flag; /* For now, we only read these once per level. */ s_FlagReader.LoadLevels(); for (i = 0; i < 26; i++) { if (FindFlag('a' + i, &flag)) { g_ReverseFlags[flag] = 'a' + i; } else { g_ReverseFlags[flag] = '?'; } } } void AdminCache::OnSourceModShutdown() { g_Forwards.ReleaseForward(m_pCacheFwd); m_pCacheFwd = NULL; } void AdminCache::OnSourceModPluginsLoaded() { DumpAdminCache(AdminCache_Overrides, true); DumpAdminCache(AdminCache_Groups, true); } void AdminCache::NameFlag(const char *str, AdminFlag flag) { sm_trie_insert(m_pLevelNames, str, (void *)flag); } bool AdminCache::FindFlag(const char *str, AdminFlag *pFlag) { void *obj; if (!sm_trie_retrieve(m_pLevelNames, str, &obj)) { return false; } if (pFlag) { *pFlag = (AdminFlag)(int)obj; } return true; } void AdminCache::AddCommandOverride(const char *cmd, OverrideType type, FlagBits flags) { Trie *pTrie = NULL; if (type == Override_Command) { pTrie = m_pCmdOverrides; } else if (type == Override_CommandGroup) { pTrie = m_pCmdGrpOverrides; } else { return; } sm_trie_insert(pTrie, cmd, (void *)(unsigned int)flags); g_ConCmds.UpdateAdminCmdFlags(cmd, type, flags, false); } bool AdminCache::GetCommandOverride(const char *cmd, OverrideType type, FlagBits *pFlags) { Trie *pTrie = NULL; if (type == Override_Command) { pTrie = m_pCmdOverrides; } else if (type == Override_CommandGroup) { pTrie = m_pCmdGrpOverrides; } else { return false; } void *object; if (sm_trie_retrieve(pTrie, cmd, &object)) { if (pFlags) { *pFlags = (FlagBits)object; } return true; } return false; } void AdminCache::UnsetCommandOverride(const char *cmd, OverrideType type) { if (type == Override_Command) { return _UnsetCommandOverride(cmd); } else if (type == Override_CommandGroup) { return _UnsetCommandGroupOverride(cmd); } } void AdminCache::_UnsetCommandGroupOverride(const char *group) { if (!m_pCmdGrpOverrides) { return; } sm_trie_delete(m_pCmdGrpOverrides, group); g_ConCmds.UpdateAdminCmdFlags(group, Override_CommandGroup, 0, true); } void AdminCache::_UnsetCommandOverride(const char *cmd) { if (!m_pCmdOverrides) { return; } sm_trie_delete(m_pCmdOverrides, cmd); g_ConCmds.UpdateAdminCmdFlags(cmd, Override_Command, 0, true); } void AdminCache::DumpCommandOverrideCache(OverrideType type) { if (type == Override_Command && m_pCmdOverrides) { sm_trie_clear(m_pCmdOverrides); } else if (type == Override_CommandGroup && m_pCmdGrpOverrides) { sm_trie_clear(m_pCmdGrpOverrides); } } AdminId AdminCache::CreateAdmin(const char *name) { AdminId id; AdminUser *pUser; if (m_FreeUserList != INVALID_ADMIN_ID) { pUser = (AdminUser *)m_pMemory->GetAddress(m_FreeUserList); assert(pUser->magic == USR_MAGIC_UNSET); id = m_FreeUserList; m_FreeUserList = pUser->next_user; } else { id = m_pMemory->CreateMem(sizeof(AdminUser), (void **)&pUser); pUser->grp_size = 0; pUser->grp_table = -1; } pUser->flags = 0; pUser->eflags = 0; pUser->grp_count = 0; pUser->password = -1; pUser->magic = USR_MAGIC_SET; pUser->auth.identidx = -1; pUser->auth.index = 0; pUser->immunity_level = 0; pUser->serialchange = 1; if (m_FirstUser == INVALID_ADMIN_ID) { m_FirstUser = id; m_LastUser = id; } else { AdminUser *pPrev = (AdminUser *)m_pMemory->GetAddress(m_LastUser); pPrev->next_user = id; pUser->prev_user = m_LastUser; m_LastUser = id; } /* Since we always append to the tail, we should invalidate their next */ pUser->next_user = -1; if (name && name[0] != '\0') { int nameidx = m_pStrings->AddString(name); pUser = (AdminUser *)m_pMemory->GetAddress(id); pUser->nameidx = nameidx; } else { pUser->nameidx = -1; } return id; } GroupId AdminCache::AddGroup(const char *group_name) { if (sm_trie_retrieve(m_pGroups, group_name, NULL)) { return INVALID_GROUP_ID; } GroupId id; AdminGroup *pGroup; if (m_FreeGroupList != INVALID_GROUP_ID) { pGroup = (AdminGroup *)m_pMemory->GetAddress(m_FreeGroupList); assert(pGroup->magic == GRP_MAGIC_UNSET); id = m_FreeGroupList; m_FreeGroupList = pGroup->next_grp; } else { id = m_pMemory->CreateMem(sizeof(AdminGroup), (void **)&pGroup); } pGroup->immunity_level = 0; pGroup->immune_table = -1; pGroup->magic = GRP_MAGIC_SET; pGroup->next_grp = INVALID_GROUP_ID; pGroup->pCmdGrpTable = NULL; pGroup->pCmdTable = NULL; pGroup->addflags = 0; if (m_FirstGroup == INVALID_GROUP_ID) { m_FirstGroup = id; m_LastGroup = id; pGroup->prev_grp = INVALID_GROUP_ID; } else { AdminGroup *pPrev = (AdminGroup *)m_pMemory->GetAddress(m_LastGroup); assert(pPrev->magic == GRP_MAGIC_SET); pPrev->next_grp = id; pGroup->prev_grp = m_LastGroup; m_LastGroup = id; } int nameidx = m_pStrings->AddString(group_name); pGroup = (AdminGroup *)m_pMemory->GetAddress(id); pGroup->nameidx = nameidx; sm_trie_insert(m_pGroups, group_name, (void *)id); return id; } GroupId AdminCache::FindGroupByName(const char *group_name) { void *object; if (!sm_trie_retrieve(m_pGroups, group_name, &object)) { return INVALID_GROUP_ID; } GroupId id = (GroupId)object; AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return INVALID_GROUP_ID; } return id; } void AdminCache::SetGroupAddFlag(GroupId id, AdminFlag flag, bool enabled) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return; } if (flag < Admin_Reservation || flag >= AdminFlags_TOTAL) { return; } FlagBits bits = (1<<(FlagBits)flag); if (enabled) { pGroup->addflags |= bits; } else { pGroup->addflags &= ~bits; } } bool AdminCache::GetGroupAddFlag(GroupId id, AdminFlag flag) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return false; } if (flag < Admin_Reservation || flag >= AdminFlags_TOTAL) { return false; } FlagBits bit = 1<<(FlagBits)flag; return ((pGroup->addflags & bit) == bit); } FlagBits AdminCache::GetGroupAddFlags(GroupId id) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return 0; } return pGroup->addflags; } const char *AdminCache::GetGroupName(GroupId gid) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(gid); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return NULL; } return m_pStrings->GetString(pGroup->nameidx); } void AdminCache::SetGroupGenericImmunity(GroupId id, ImmunityType type, bool enabled) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return; } unsigned int level = 0; if (enabled) { if (type == Immunity_Default) { level = 1; } else if (type == Immunity_Global) { level = 2; } if (level > pGroup->immunity_level) { pGroup->immunity_level = level; } } else { pGroup->immunity_level = 0; } } bool AdminCache::GetGroupGenericImmunity(GroupId id, ImmunityType type) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return false; } if (type == Immunity_Default && pGroup->immunity_level >= 1) { return true; } else if (type == Immunity_Global && pGroup->immunity_level >= 2) { return true; } return false; } void AdminCache::AddGroupImmunity(GroupId id, GroupId other_id) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(other_id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return; } pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return; } /* We always need to resize the immunity table */ int *table, tblidx; if (pGroup->immune_table == -1) { tblidx = m_pMemory->CreateMem(sizeof(int) * 2, (void **)&table); pGroup = (AdminGroup *)m_pMemory->GetAddress(id); table[0] = 0; } else { int *old_table = (int *)m_pMemory->GetAddress(pGroup->immune_table); /* Break out if this group is already in the list */ for (int i=0; iCreateMem(sizeof(int) * (old_table[0] + 2), (void **)&table); /* Get the old address again in case of resize */ pGroup = (AdminGroup *)m_pMemory->GetAddress(id); old_table = (int *)m_pMemory->GetAddress(pGroup->immune_table); table[0] = old_table[0]; for (unsigned int i=1; i<=(unsigned int)old_table[0]; i++) { table[i] = old_table[i]; } } /* Assign */ pGroup->immune_table = tblidx; /* Add to the array */ table[0]++; table[table[0]] = other_id; } unsigned int AdminCache::GetGroupImmunityCount(GroupId id) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return 0; } if (pGroup->immune_table == -1) { return 0; } int *table = (int *)m_pMemory->GetAddress(pGroup->immune_table); return table[0]; } GroupId AdminCache::GetGroupImmunity(GroupId id, unsigned int number) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return INVALID_GROUP_ID; } if (pGroup->immune_table == -1) { return INVALID_GROUP_ID; } int *table = (int *)m_pMemory->GetAddress(pGroup->immune_table); if (number >= (unsigned int)table[0]) { return INVALID_GROUP_ID; } return table[1 + number]; } void AdminCache::AddGroupCommandOverride(GroupId id, const char *name, OverrideType type, OverrideRule rule) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return; } Trie *pTrie = NULL; if (type == Override_Command) { if (pGroup->pCmdTable == NULL) { pGroup->pCmdTable = sm_trie_create(); } pTrie = pGroup->pCmdTable; } else if (type == Override_CommandGroup) { if (pGroup->pCmdGrpTable == NULL) { pGroup->pCmdGrpTable = sm_trie_create(); } pTrie = pGroup->pCmdGrpTable; } else { return; } sm_trie_insert(pTrie, name, (void *)(int)rule); } bool AdminCache::GetGroupCommandOverride(GroupId id, const char *name, OverrideType type, OverrideRule *pRule) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return false; } Trie *pTrie = NULL; if (type == Override_Command) { if (pGroup->pCmdTable == NULL) { return false; } pTrie = pGroup->pCmdTable; } else if (type == Override_CommandGroup) { if (pGroup->pCmdGrpTable == NULL) { return false; } pTrie = pGroup->pCmdGrpTable; } else { return false; } void *object; if (!sm_trie_retrieve(pTrie, name, &object)) { return false; } if (pRule) { *pRule = (OverrideRule)(int)object; } return true; } Trie *AdminCache::GetMethodByIndex(unsigned int index) { List::iterator iter; for (iter=m_AuthMethods.begin(); iter!=m_AuthMethods.end(); iter++) { if (index-- == 0) { return (*iter).table; } } return NULL; } bool AdminCache::InvalidateAdmin(AdminId id) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); AdminUser *pOther; if (!pUser || pUser->magic != USR_MAGIC_SET) { return false; } if (!m_InvalidatingAdmins && !m_destroying) { g_Players.ClearAdminId(id); } /* Unlink from the dbl link list */ if (id == m_FirstUser && id == m_LastUser) { m_FirstUser = INVALID_ADMIN_ID; m_LastUser = INVALID_ADMIN_ID; } else if (id == m_FirstUser) { m_FirstUser = pUser->next_user; pOther = (AdminUser *)m_pMemory->GetAddress(m_FirstUser); pOther->prev_user = INVALID_ADMIN_ID; } else if (id == m_LastUser) { m_LastUser = pUser->prev_user; pOther = (AdminUser *)m_pMemory->GetAddress(m_LastUser); pOther->next_user = INVALID_ADMIN_ID; } else { pOther = (AdminUser *)m_pMemory->GetAddress(pUser->prev_user); pOther->next_user = pUser->next_user; pOther = (AdminUser *)m_pMemory->GetAddress(pUser->next_user); pOther->prev_user = pUser->prev_user; } /* Unlink from auth tables */ if (pUser->auth.identidx != -1) { Trie *pTrie = GetMethodByIndex(pUser->auth.index); if (pTrie) { sm_trie_delete(pTrie, m_pStrings->GetString(pUser->auth.identidx)); } } /* Clear table counts */ pUser->grp_count = 0; /* Link into free list */ pUser->magic = USR_MAGIC_UNSET; pUser->next_user = m_FreeUserList; m_FreeUserList = id; /* Unset serial change */ pUser->serialchange = 0; return true; } void AdminCache::InvalidateGroup(GroupId id) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(id); AdminGroup *pOther; if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return; } const char *str = m_pStrings->GetString(pGroup->nameidx); sm_trie_delete(m_pGroups, str); /* Unlink from the live dbllink list */ if (id == m_FirstGroup && id == m_LastGroup) { m_LastGroup = INVALID_GROUP_ID; m_FirstGroup = INVALID_GROUP_ID; } else if (id == m_FirstGroup) { m_FirstGroup = pGroup->next_grp; pOther = (AdminGroup *)m_pMemory->GetAddress(m_FirstGroup); pOther->prev_grp = INVALID_GROUP_ID; } else if (id == m_LastGroup) { m_LastGroup = pGroup->prev_grp; pOther = (AdminGroup *)m_pMemory->GetAddress(m_LastGroup); pOther->next_grp = INVALID_GROUP_ID; } else { pOther = (AdminGroup *)m_pMemory->GetAddress(pGroup->prev_grp); pOther->next_grp = pGroup->next_grp; pOther = (AdminGroup *)m_pMemory->GetAddress(pGroup->next_grp); pOther->prev_grp = pGroup->prev_grp; } /* Free any used memory to be safe */ if (pGroup->pCmdGrpTable) { sm_trie_destroy(pGroup->pCmdGrpTable); pGroup->pCmdGrpTable = NULL; } if (pGroup->pCmdTable) { sm_trie_destroy(pGroup->pCmdTable); pGroup->pCmdTable = NULL; } /* Link into the free list */ pGroup->magic = GRP_MAGIC_UNSET; pGroup->next_grp = m_FreeGroupList; m_FreeGroupList = id; int idx = m_FirstUser; AdminUser *pUser; int *table; while (idx != INVALID_ADMIN_ID) { pUser = (AdminUser *)m_pMemory->GetAddress(idx); if (pUser->grp_count > 0) { table = (int *)m_pMemory->GetAddress(pUser->grp_table); for (unsigned int i=0; igrp_count; i++) { if (table[i] == id) { /* We have to remove this entry */ for (unsigned int j=i+1; jgrp_count; j++) { /* Move everything down by one */ table[j-1] = table[j]; } /* Decrease count */ pUser->grp_count--; /* Recalculate effective flags */ pUser->eflags = pUser->flags; for (unsigned int j=0; jgrp_count; j++) { pOther = (AdminGroup *)m_pMemory->GetAddress(table[j]); pUser->eflags |= pOther->addflags; } /* Mark as changed */ pUser->serialchange++; /* Break now, duplicates aren't allowed */ break; } } } idx = pUser->next_user; } } void AdminCache::InvalidateGroupCache() { /* Nuke the free list */ m_FreeGroupList = -1; /* Nuke reverse lookups */ sm_trie_clear(m_pGroups); /* Free memory on groups */ GroupId cur = m_FirstGroup; AdminGroup *pGroup; while (cur != INVALID_GROUP_ID) { pGroup = (AdminGroup *)m_pMemory->GetAddress(cur); assert(pGroup->magic == GRP_MAGIC_SET); if (pGroup->pCmdGrpTable) { sm_trie_destroy(pGroup->pCmdGrpTable); } if (pGroup->pCmdTable) { sm_trie_destroy(pGroup->pCmdTable); } cur = pGroup->next_grp; } m_FirstGroup = INVALID_GROUP_ID; m_LastGroup = INVALID_GROUP_ID; InvalidateAdminCache(false); /* Reset the memory table */ m_pMemory->Reset(); } void AdminCache::AddAdminListener(IAdminListener *pListener) { m_hooks.push_back(pListener); } void AdminCache::RemoveAdminListener(IAdminListener *pListener) { m_hooks.remove(pListener); } void AdminCache::RegisterAuthIdentType(const char *name) { if (sm_trie_retrieve(m_pAuthTables, name, NULL)) { return; } Trie *pAuth = sm_trie_create(); AuthMethod method; method.name.assign(name); method.table = pAuth; m_AuthMethods.push_back(method); sm_trie_insert(m_pAuthTables, name, pAuth); } void AdminCache::InvalidateAdminCache(bool unlink_admins) { m_InvalidatingAdmins = true; if (!m_destroying) { g_Players.ClearAllAdmins(); } /* Wipe the identity cache first */ List::iterator iter; for (iter=m_AuthMethods.begin(); iter!=m_AuthMethods.end(); iter++) { sm_trie_clear((*iter).table); } if (unlink_admins) { while (m_FirstUser != INVALID_ADMIN_ID) { InvalidateAdmin(m_FirstUser); } } else { m_FirstUser = -1; m_LastUser = -1; m_FreeUserList = -1; } m_InvalidatingAdmins = false; } void AdminCache::DumpAdminCache(AdminCachePart part, bool rebuild) { List::iterator iter; IAdminListener *pListener; cell_t result; if (part == AdminCache_Overrides) { DumpCommandOverrideCache(Override_Command); DumpCommandOverrideCache(Override_CommandGroup); if (rebuild && !m_destroying) { for (iter=m_hooks.begin(); iter!=m_hooks.end(); iter++) { pListener = (*iter); pListener->OnRebuildOverrideCache(); } m_pCacheFwd->PushCell(part); m_pCacheFwd->Execute(&result); } } else if (part == AdminCache_Groups || part == AdminCache_Admins) { if (part == AdminCache_Groups) { InvalidateGroupCache(); if (rebuild && !m_destroying) { for (iter=m_hooks.begin(); iter!=m_hooks.end(); iter++) { pListener = (*iter); pListener->OnRebuildGroupCache(); } m_pCacheFwd->PushCell(part); m_pCacheFwd->Execute(&result); } } InvalidateAdminCache(true); if (rebuild && !m_destroying) { for (iter=m_hooks.begin(); iter!=m_hooks.end(); iter++) { pListener = (*iter); pListener->OnRebuildAdminCache((part == AdminCache_Groups)); } m_pCacheFwd->PushCell(AdminCache_Admins); m_pCacheFwd->Execute(&result); g_Players.RecheckAnyAdmins(); } } } const char *AdminCache::GetAdminName(AdminId id) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return NULL; } return m_pStrings->GetString(pUser->nameidx); } bool AdminCache::GetMethodIndex(const char *name, unsigned int *_index) { List::iterator iter; unsigned int index = 0; for (iter=m_AuthMethods.begin(); iter!=m_AuthMethods.end(); iter++,index++) { if ((*iter).name.compare(name) == 0) { *_index = index; return true; } } return false; } bool AdminCache::BindAdminIdentity(AdminId id, const char *auth, const char *ident) { if (ident[0] == '\0') { return false; } AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return false; } Trie *pTable; if (!sm_trie_retrieve(m_pAuthTables, auth, (void **)&pTable)) { return false; } /* If the id was a steam id strip off the STEAM_*: part */ if (strcmp(auth, "steam") == 0 && strncmp(ident, "STEAM_", 6) == 0) { ident += 8; } if (sm_trie_retrieve(pTable, ident, NULL)) { return false; } int i_ident = m_pStrings->AddString(ident); pUser = (AdminUser *)m_pMemory->GetAddress(id); pUser->auth.identidx = i_ident; GetMethodIndex(auth, &pUser->auth.index); return sm_trie_insert(pTable, ident, (void **)id); } AdminId AdminCache::FindAdminByIdentity(const char *auth, const char *identity) { Trie *pTable; if (!sm_trie_retrieve(m_pAuthTables, auth, (void **)&pTable)) { return INVALID_ADMIN_ID; } /* If the id was a steam id strip off the STEAM_*: part */ if (strcmp(auth, "steam") == 0 && strncmp(identity, "STEAM_", 6) == 0) { identity += 8; } void *object; if (!sm_trie_retrieve(pTable, identity, &object)) { return INVALID_ADMIN_ID; } return (AdminId)object; } void AdminCache::SetAdminFlag(AdminId id, AdminFlag flag, bool enabled) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return; } if (flag < Admin_Reservation || flag >= AdminFlags_TOTAL) { return; } FlagBits bits = (1<<(FlagBits)flag); if (enabled) { pUser->flags |= bits; pUser->eflags |= bits; } else { pUser->flags &= ~bits; pUser->eflags &= ~bits; } pUser->serialchange++; } bool AdminCache::GetAdminFlag(AdminId id, AdminFlag flag, AccessMode mode) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return false; } if (flag < Admin_Reservation || flag >= AdminFlags_TOTAL) { return false; } FlagBits bit = (1<<(FlagBits)flag); if (mode == Access_Real) { return ((pUser->flags & bit) == bit); } else if (mode == Access_Effective) { bool has_bit = ((pUser->eflags & bit) == bit); if (!has_bit && flag != Admin_Root && ((pUser->eflags & ADMFLAG_ROOT) == ADMFLAG_ROOT)) { has_bit = true; } return has_bit; } return false; } FlagBits AdminCache::GetAdminFlags(AdminId id, AccessMode mode) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return 0; } if (mode == Access_Real) { return pUser->flags; } else if (mode == Access_Effective) { return pUser->eflags; } return 0; } void AdminCache::SetAdminFlags(AdminId id, AccessMode mode, FlagBits bits) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return; } if (mode == Access_Real) { pUser->flags = bits; pUser->eflags = bits; } else if (mode == Access_Effective) { pUser->eflags = bits; } pUser->serialchange++; } bool AdminCache::AdminInheritGroup(AdminId id, GroupId gid) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return false; } AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(gid); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return false; } /* First check for duplicates */ if (pUser->grp_count != 0) { int *temp_table = (int *)m_pMemory->GetAddress(pUser->grp_table); for (unsigned int i=0; igrp_count; i++) { if (temp_table[i] == gid) { return false; } } } int *table; if (pUser->grp_count + 1 > pUser->grp_size) { int new_size = 0; int tblidx; if (pUser->grp_size == 0) { new_size = 2; } else { new_size = pUser->grp_size * 2; } /* Create and refresh pointers */ tblidx = m_pMemory->CreateMem(new_size * sizeof(int), (void **)&table); pUser = (AdminUser *)m_pMemory->GetAddress(id); pGroup = (AdminGroup *)m_pMemory->GetAddress(gid); /* Copy old data if necessary */ if (pUser->grp_table != -1) { int *old_table = (int *)m_pMemory->GetAddress(pUser->grp_table); memcpy(table, old_table, sizeof(int) * pUser->grp_count); } pUser->grp_table = tblidx; pUser->grp_size = new_size; } else { table = (int *)m_pMemory->GetAddress(pUser->grp_table); } table[pUser->grp_count] = gid; pUser->grp_count++; /* Compute new effective permissions */ pUser->eflags |= pGroup->addflags; if (pGroup->immunity_level > pUser->immunity_level) { pUser->immunity_level = pGroup->immunity_level; } pUser->serialchange++; return true; } bool AdminCache::IsValidAdmin(AdminId id) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); return (pUser != NULL && pUser->magic == USR_MAGIC_SET); } unsigned int AdminCache::GetAdminGroupCount(AdminId id) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return 0; } return pUser->grp_count; } GroupId AdminCache::GetAdminGroup(AdminId id, unsigned int index, const char **name) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET || index >= pUser->grp_count) { return INVALID_GROUP_ID; } int *table = (int *)m_pMemory->GetAddress(pUser->grp_table); GroupId gid = table[index]; if (name) { *name = GetGroupName(gid); } return gid; } const char *AdminCache::GetAdminPassword(AdminId id) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return NULL; } return m_pStrings->GetString(pUser->password); } void AdminCache::SetAdminPassword(AdminId id, const char *password) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return; } if (password[0] == '\0') { pUser->password = -1; return; } int i_password = m_pStrings->AddString(password); pUser = (AdminUser *)m_pMemory->GetAddress(id); pUser->password = i_password; } unsigned int AdminCache::FlagBitsToBitArray(FlagBits bits, bool array[], unsigned int maxSize) { unsigned int i; for (i=0; iGetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return false; } return ((pUser->eflags & bits) == bits); } bool AdminCache::CanAdminTarget(AdminId id, AdminId target) { /** * Zeroth, if the targeting AdminId is INVALID_ADMIN_ID, targeting fails. * First, if the targeted AdminId is INVALID_ADMIN_ID, targeting succeeds. */ if (id == INVALID_ADMIN_ID) { return false; } if (target == INVALID_ADMIN_ID) { return true; } if (id == target) { return true; } AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return false; } AdminUser *pTarget = (AdminUser *)m_pMemory->GetAddress(target); if (!pTarget || pTarget->magic != USR_MAGIC_SET) { return false; } /** * Second, if the targeting admin is root, targeting succeeds. */ if (pUser->eflags & ADMFLAG_ROOT) { return true; } /** Fourth, if the targeted admin is immune from targeting admin. */ int mode = sm_immunity_mode.GetInt(); switch (mode) { case 1: { if (pTarget->immunity_level > pUser->immunity_level) { return false; } break; } case 3: { /* If neither has any immunity, let this pass. */ if (!pUser->immunity_level && !pTarget->immunity_level) { return true; } /* Don't break, go to the next case. */ } case 2: { if (pTarget->immunity_level >= pUser->immunity_level) { return false; } break; } } /** * Fifth, if the targeted admin has specific immunity from the * targeting admin via group immunities, targeting fails. */ //:TODO: speed this up... maybe with trie hacks. //idea is to insert %d.%d in the trie after computing this and use it as a cache lookup. //problem is the trie cannot delete prefixes, so we'd have a problem with invalidations. if (pTarget->grp_count > 0 && pUser->grp_count > 0) { int *grp_table = (int *)m_pMemory->GetAddress(pTarget->grp_table); int *src_table = (int *)m_pMemory->GetAddress(pUser->grp_table); GroupId id, other; unsigned int num; for (unsigned int i=0; igrp_count; i++) { id = grp_table[i]; num = GetGroupImmunityCount(id); for (unsigned int j=0; jgrp_count; k++) { if (other == src_table[k]) { return false; } } } } } return true; } bool AdminCache::FindFlag(char c, AdminFlag *pAdmFlag) { if (c < 'a' || c > 'z' || !g_FlagSet[(unsigned)c - (unsigned)'a']) { return false; } if (pAdmFlag) { *pAdmFlag = g_FlagLetters[(unsigned)c - (unsigned)'a']; } return true; } bool AdminCache::FindFlagChar(AdminFlag flag, char *c) { if (!g_FlagSet[flag]) { return false; } if (c) { *c = g_ReverseFlags[flag]; } return true; } FlagBits AdminCache::ReadFlagString(const char *flags, const char **end) { FlagBits bits = 0; while (flags && (*flags != '\0')) { AdminFlag flag; if (!FindFlag(*flags, &flag)) { break; } bits |= FlagArrayToBits(&flag, 1); flags++; } if (end) { *end = flags; } return bits; } unsigned int AdminCache::GetAdminSerialChange(AdminId id) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return 0; } return pUser->serialchange; } bool AdminCache::CanAdminUseCommand(int client, const char *cmd) { FlagBits bits; OverrideType otype = Override_Command; if (cmd[0] == '@') { otype = Override_CommandGroup; cmd++; } if (!g_ConCmds.LookForCommandAdminFlags(cmd, &bits)) { if (!GetCommandOverride(cmd, otype, &bits)) { bits = 0; } } return g_ConCmds.CheckCommandAccess(client, cmd, bits); } unsigned int AdminCache::SetGroupImmunityLevel(GroupId gid, unsigned int level) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(gid); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return 0; } unsigned int old_level = pGroup->immunity_level; pGroup->immunity_level = level; return old_level; } unsigned int AdminCache::GetGroupImmunityLevel(GroupId gid) { AdminGroup *pGroup = (AdminGroup *)m_pMemory->GetAddress(gid); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return 0; } return pGroup->immunity_level; } unsigned int AdminCache::SetAdminImmunityLevel(AdminId id, unsigned int level) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return 0; } unsigned int old_level = pUser->immunity_level; pUser->immunity_level = level; return old_level; } unsigned int AdminCache::GetAdminImmunityLevel(AdminId id) { AdminUser *pUser = (AdminUser *)m_pMemory->GetAddress(id); if (!pUser || pUser->magic != USR_MAGIC_SET) { return 0; } return pUser->immunity_level; } bool AdminCache::CheckAccess(int client, const char *cmd, FlagBits flags, bool override_only) { if (client == 0) { return true; } /* Auto-detect a command if we can */ FlagBits bits = flags; bool found_command = false; if (!override_only) { found_command = g_ConCmds.LookForCommandAdminFlags(cmd, &bits); } if (!found_command) { GetCommandOverride(cmd, Override_Command, &bits); } return g_ConCmds.CheckCommandAccess(client, cmd, bits) ? 1 : 0; } void iterator_glob_basic_override(Trie *pTrie, const char *key, void **value, void *data) { FILE *fp; int flags; char flagstr[64]; fp = (FILE *)data; flags = (int)*value; g_Admins.FillFlagString(flags, flagstr, sizeof(flagstr)); fprintf(fp, "\t\"%s\"\t\t\"%s\"\n", key, flagstr); } void iterator_glob_grp_override(Trie *pTrie, const char *key, void **value, void *data) { FILE *fp; int flags; char flagstr[64]; fp = (FILE *)data; flags = (int)*value; g_Admins.FillFlagString(flags, flagstr, sizeof(flagstr)); fprintf(fp, "\t\"@%s\"\t\t\"%s\"\n", key, flagstr); } void iterator_group_basic_override(Trie *pTrie, const char *key, void **value, void *data) { FILE *fp; int flags; char flagstr[64]; fp = (FILE *)data; flags = (int)*value; g_Admins.FillFlagString(flags, flagstr, sizeof(flagstr)); fprintf(fp, "\t\t\t\"%s\"\t\t\"%s\"\n", key, flagstr); } void iterator_group_grp_override(Trie *pTrie, const char *key, void **value, void *data) { FILE *fp; int flags; char flagstr[64]; fp = (FILE *)data; flags = (int)*value; g_Admins.FillFlagString(flags, flagstr, sizeof(flagstr)); fprintf(fp, "\t\t\t\"@%s\"\t\t\"%s\"\n", key, flagstr); } void AdminCache::DumpCache(FILE *fp) { int *itable; AdminId aid; GroupId gid; char flagstr[64]; unsigned int num; AdminUser *pAdmin; AdminGroup *pGroup; char name_buffer[512]; fprintf(fp, "\"Groups\"\n{\n"); num = 0; gid = m_FirstGroup; while (gid != INVALID_GROUP_ID && (pGroup = GetGroup(gid)) != NULL) { num++; FillFlagString(pGroup->addflags, flagstr, sizeof(flagstr)); fprintf(fp, "\t/* num = %d, gid = 0x%X */\n", num, gid); fprintf(fp, "\t\"%s\"\n\t{\n", GetString(pGroup->nameidx)); fprintf(fp, "\t\t\"flags\"\t\t\t\"%s\"\n", flagstr); fprintf(fp, "\t\t\"immunity\"\t\t\"%d\"\n", pGroup->immunity_level); if (pGroup->immune_table != -1 && (itable = (int *)m_pMemory->GetAddress(pGroup->immune_table)) != NULL) { AdminGroup *pAltGroup; const char *gname, *mod; for (int i = 1; i <= itable[0]; i++) { if ((pAltGroup = GetGroup(itable[i])) == NULL) { /* Assume the rest of the table is corrupt */ break; } gname = GetString(pAltGroup->nameidx); if (atoi(gname) != 0) { mod = "@"; } else { mod = ""; } fprintf(fp, "\t\t\"immunity\"\t\t\"%s%s\"\n", mod, gname); } } fprintf(fp, "\n\t\t\"Overrides\"\n\t\t{\n"); if (pGroup->pCmdGrpTable != NULL) { sm_trie_bad_iterator(pGroup->pCmdGrpTable, name_buffer, sizeof(name_buffer), iterator_group_grp_override, fp); } if (pGroup->pCmdTable != NULL) { sm_trie_bad_iterator(pGroup->pCmdTable, name_buffer, sizeof(name_buffer), iterator_group_basic_override, fp); } fprintf(fp, "\t\t}\n"); fprintf(fp, "\t}\n"); if ((gid = pGroup->next_grp) != INVALID_GROUP_ID) { fprintf(fp, "\n"); } } fprintf(fp, "}\n\n"); fprintf(fp, "\"Admins\"\n{\n"); num = 0; aid = m_FirstUser; while (aid != INVALID_ADMIN_ID && (pAdmin = GetUser(aid)) != NULL) { num++; FillFlagString(pAdmin->flags, flagstr, sizeof(flagstr)); fprintf(fp, "\t/* num = %d, aid = 0x%X, serialno = 0x%X*/\n", num, aid, pAdmin->serialchange); if (pAdmin->nameidx != -1) { fprintf(fp, "\t\"%s\"\n\t{\n", GetString(pAdmin->nameidx)); } else { fprintf(fp, "\t\"\"\n\t{\n"); } if (pAdmin->auth.identidx != -1) { fprintf(fp, "\t\t\"auth\"\t\t\t\"%s\"\n", GetMethodName(pAdmin->auth.index)); fprintf(fp, "\t\t\"identity\"\t\t\"%s\"\n", GetString(pAdmin->auth.identidx)); } if (pAdmin->password != -1) { fprintf(fp, "\t\t\"password\"\t\t\"%s\"\n", GetString(pAdmin->password)); } fprintf(fp, "\t\t\"flags\"\t\t\t\"%s\"\n", flagstr); fprintf(fp, "\t\t\"immunity\"\t\t\"%d\"\n", pAdmin->immunity_level); if (pAdmin->grp_count != 0 && pAdmin->grp_table != -1 && (itable = (int *)m_pMemory->GetAddress(pAdmin->grp_table)) != NULL) { unsigned int i; for (i = 0; i < pAdmin->grp_count; i++) { if ((pGroup = GetGroup(itable[i])) == NULL) { /* Assume the rest of the table is corrupt */ break; } fprintf(fp, "\t\t\"group\"\t\t\t\"%s\"\n", GetString(pGroup->nameidx)); } } fprintf(fp, "\t}\n"); if ((aid = pAdmin->next_user) != INVALID_ADMIN_ID) { fprintf(fp, "\n"); } } fprintf(fp, "}\n\n"); fprintf(fp, "\"Overrides\"\n{\n"); if (m_pCmdGrpOverrides != NULL) { sm_trie_bad_iterator(m_pCmdGrpOverrides, name_buffer, sizeof(name_buffer), iterator_glob_grp_override, fp); } if (m_pCmdOverrides != NULL) { sm_trie_bad_iterator(m_pCmdOverrides, name_buffer, sizeof(name_buffer), iterator_glob_basic_override, fp); } fprintf(fp, "}\n"); } AdminGroup *AdminCache::GetGroup(GroupId gid) { AdminGroup *pGroup; pGroup = (AdminGroup *)m_pMemory->GetAddress(gid); if (!pGroup || pGroup->magic != GRP_MAGIC_SET) { return NULL; } return pGroup; } AdminUser *AdminCache::GetUser(AdminId aid) { AdminUser *pAdmin; pAdmin = (AdminUser *)m_pMemory->GetAddress(aid); if (!pAdmin || pAdmin->magic != USR_MAGIC_SET) { return NULL; } return pAdmin; } const char *AdminCache::GetMethodName(unsigned int index) { List::iterator iter; for (iter=m_AuthMethods.begin(); iter!=m_AuthMethods.end(); iter++) { if (index-- == 0) { return (*iter).name.c_str(); } } return NULL; } const char *AdminCache::GetString(int idx) { return m_pStrings->GetString(idx); } size_t AdminCache::FillFlagString(FlagBits bits, char *buffer, size_t maxlen) { size_t pos; unsigned int i, num_flags; AdminFlag flags[AdminFlags_TOTAL]; num_flags = FlagBitsToArray(bits, flags, AdminFlags_TOTAL); pos = 0; for (i = 0; pos < maxlen && i < num_flags; i++) { if (FindFlagChar(flags[i], &buffer[pos])) { pos++; } } buffer[pos] = '\0'; return pos; } CON_COMMAND(sm_dump_admcache, "Dumps the admin cache for debugging") { FILE *fp; char buffer[PLATFORM_MAX_PATH]; g_SourceMod.BuildPath(Path_SM, buffer, sizeof(buffer), "data/admin_cache_dump.txt"); if ((fp = fopen(buffer, "wt")) == NULL) { g_RootMenu.ConsolePrint("Could not open file for writing: %s", buffer); return; } g_Admins.DumpCache(fp); g_RootMenu.ConsolePrint("Admin cache dumped to: %s", buffer); fclose(fp); }