/** * vim: set ts=4 : * =============================================================== * SourceMod (C)2004-2007 AlliedModders LLC. All rights reserved. * =============================================================== * * This file is not open source and may not be copied without explicit * written permission of AlliedModders LLC. This file may not be redistributed * in whole or significant part. * For information, see LICENSE.txt or http://www.sourcemod.net/license.php * * Version: $Id$ */ #include "Database.h" #include "HandleSys.h" #include "ShareSys.h" #include "sourcemod.h" #include "sm_stringutil.h" #include "TextParsers.h" #include "Logger.h" #include "ExtensionSys.h" #include #define DBPARSE_LEVEL_NONE 0 #define DBPARSE_LEVEL_MAIN 1 #define DBPARSE_LEVEL_DATABASE 2 DBManager g_DBMan; DBManager::DBManager() : m_StrTab(512), m_ParseLevel(0), m_ParseState(0), m_pDefault(NULL) { } void DBManager::OnSourceModAllInitialized() { HandleAccess sec; g_HandleSys.InitAccessDefaults(NULL, &sec); sec.access[HandleAccess_Delete] |= HANDLE_RESTRICT_IDENTITY; sec.access[HandleAccess_Clone] |= HANDLE_RESTRICT_IDENTITY; m_DriverType = g_HandleSys.CreateType("IDriver", this, 0, NULL, &sec, g_pCoreIdent, NULL); m_DatabaseType = g_HandleSys.CreateType("IDatabase", this, 0, NULL, NULL, g_pCoreIdent, NULL); g_ShareSys.AddInterface(NULL, this); g_SourceMod.BuildPath(Path_SM, m_Filename, sizeof(m_Filename), "configs/databases.cfg"); } void DBManager::OnSourceModLevelChange(const char *mapName) { SMCParseError err; unsigned int line = 0; if ((err = g_TextParser.ParseFile_SMC(m_Filename, this, &line, NULL)) != SMCParse_Okay) { g_Logger.LogError("[SM] Detected parse error(s) in file \"%s\"", m_Filename); if (err != SMCParse_Custom) { const char *txt = g_TextParser.GetSMCErrorString(err); g_Logger.LogError("[SM] Line %d: %s", line, txt); } } } void DBManager::OnSourceModShutdown() { g_HandleSys.RemoveType(m_DatabaseType, g_pCoreIdent); g_HandleSys.RemoveType(m_DriverType, g_pCoreIdent); } unsigned int DBManager::GetInterfaceVersion() { return SMINTERFACE_DBI_VERSION; } const char *DBManager::GetInterfaceName() { return SMINTERFACE_DBI_NAME; } void DBManager::OnHandleDestroy(HandleType_t type, void *object) { if (type == m_DriverType) { /* Ignore */ return; } if (g_HandleSys.TypeCheck(type, m_DatabaseType)) { IDatabase *pdb = (IDatabase *)object; pdb->Close(); } } void DBManager::ReadSMC_ParseStart() { m_confs.clear(); m_ParseLevel = 0; m_ParseState = DBPARSE_LEVEL_NONE; m_StrTab.Reset(); m_DefDriver.clear(); } ConfDbInfo s_CurInfo; SMCParseResult DBManager::ReadSMC_NewSection(const char *name, bool opt_quotes) { if (m_ParseLevel) { m_ParseLevel++; return SMCParse_Continue; } if (m_ParseState == DBPARSE_LEVEL_NONE) { if (strcmp(name, "Databases") == 0) { m_ParseState = DBPARSE_LEVEL_MAIN; } else { m_ParseLevel++; } } else if (m_ParseState == DBPARSE_LEVEL_MAIN) { s_CurInfo = ConfDbInfo(); s_CurInfo.name = m_StrTab.AddString(name); m_ParseState = DBPARSE_LEVEL_DATABASE; } else if (m_ParseState == DBPARSE_LEVEL_DATABASE) { m_ParseLevel++; } return SMCParse_Continue; } SMCParseResult DBManager::ReadSMC_KeyValue(const char *key, const char *value, bool key_quotes, bool value_quotes) { if (m_ParseLevel) { return SMCParse_Continue; } if (m_ParseState == DBPARSE_LEVEL_MAIN) { if (strcmp(key, "driver_default") == 0) { m_DefDriver.assign(value); } } else if (m_ParseState == DBPARSE_LEVEL_DATABASE) { if (strcmp(key, "driver") == 0) { if (strcmp(value, "default") != 0) { s_CurInfo.driver = m_StrTab.AddString(value); } } else if (strcmp(key, "database") == 0) { s_CurInfo.database = m_StrTab.AddString(value); } else if (strcmp(key, "host") == 0) { s_CurInfo.host = m_StrTab.AddString(value); } else if (strcmp(key, "user") == 0) { s_CurInfo.user = m_StrTab.AddString(value); } else if (strcmp(key, "pass") == 0) { s_CurInfo.pass = m_StrTab.AddString(value); } else if (strcmp(key, "timeout") == 0) { s_CurInfo.info.maxTimeout = atoi(value); } else if (strcmp(key, "port") == 0) { s_CurInfo.info.port = atoi(value); } } return SMCParse_Continue; } #define ASSIGN_VAR(var) \ if (s_CurInfo.var == -1) { \ s_CurInfo.info.var = ""; \ } else { \ s_CurInfo.info.var = m_StrTab.GetString(s_CurInfo.var); \ } SMCParseResult DBManager::ReadSMC_LeavingSection() { if (m_ParseLevel) { m_ParseLevel--; return SMCParse_Continue; } if (m_ParseState == DBPARSE_LEVEL_DATABASE) { /* Set all of the info members to either a blank string * or the string pointer from the string table. */ ASSIGN_VAR(driver); ASSIGN_VAR(database); ASSIGN_VAR(host); ASSIGN_VAR(user); ASSIGN_VAR(pass); /* Save it.. */ m_confs.push_back(s_CurInfo); /* Go up one level */ m_ParseState = DBPARSE_LEVEL_MAIN; } else if (m_ParseState == DBPARSE_LEVEL_MAIN) { m_ParseState = DBPARSE_LEVEL_NONE; return SMCParse_Halt; } return SMCParse_Continue; } #undef ASSIGN_VAR void DBManager::ReadSMC_ParseEnd(bool halted, bool failed) { } bool DBManager::Connect(const char *name, IDBDriver **pdr, IDatabase **pdb, bool persistent, char *error, size_t maxlength) { ConfDbInfo *pInfo = GetDatabaseConf(name); if (!pInfo) { if (pdr) { *pdr = NULL; } *pdb = NULL; UTIL_Format(error, maxlength, "Configuration \"%s\" not found", name); return false; } const char *dname = pInfo->info.driver; if (!pInfo->realDriver) { /* Try to assign a real driver pointer */ if (pInfo->info.driver[0] == '\0') { if (!m_pDefault && m_DefDriver.size() > 0) { m_pDefault = FindOrLoadDriver(m_DefDriver.c_str()); } dname = m_DefDriver.size() ? m_DefDriver.c_str() : "default"; pInfo->realDriver = m_pDefault; } else { pInfo->realDriver = FindOrLoadDriver(pInfo->info.driver); } } if (pInfo->realDriver) { if (pdr) { *pdr = pInfo->realDriver; } *pdb = pInfo->realDriver->Connect(&pInfo->info, persistent, error, maxlength); return (*pdb != NULL); } if (pdr) { *pdr = NULL; } *pdb = NULL; UTIL_Format(error, maxlength, "Driver \"%s\" not found", dname); return false; } void DBManager::AddDriver(IDBDriver *pDriver) { m_drivers.push_back(pDriver); } void DBManager::RemoveDriver(IDBDriver *pDriver) { for (size_t i=0; i::iterator iter; for (iter=m_confs.begin(); iter!=m_confs.end(); iter++) { ConfDbInfo &db = (*iter); if (db.realDriver == pDriver) { db.realDriver = NULL; } } } IDBDriver *DBManager::GetDefaultDriver() { if (!m_pDefault && m_DefDriver.size() > 0) { m_pDefault = FindOrLoadDriver(m_DefDriver.c_str()); } return m_pDefault; } Handle_t DBManager::CreateHandle(DBHandleType dtype, void *ptr, IdentityToken_t *pToken) { HandleType_t type = 0; if (dtype == DBHandle_Driver) { type = m_DriverType; } else if (dtype == DBHandle_Database) { type = m_DatabaseType; } else { return BAD_HANDLE; } return g_HandleSys.CreateHandle(type, ptr, pToken, g_pCoreIdent, NULL); } HandleError DBManager::ReadHandle(Handle_t hndl, DBHandleType dtype, void **ptr) { HandleType_t type = 0; if (dtype == DBHandle_Driver) { type = m_DriverType; } else if (dtype == DBHandle_Database) { type = m_DatabaseType; } else { return HandleError_Type; } HandleSecurity sec(NULL, g_pCoreIdent); return g_HandleSys.ReadHandle(hndl, type, &sec, ptr); } HandleError DBManager::ReleaseHandle(Handle_t hndl, DBHandleType type, IdentityToken_t *token) { HandleSecurity sec(token, g_pCoreIdent); return g_HandleSys.FreeHandle(hndl, &sec); } unsigned int DBManager::GetDriverCount() { return (unsigned int)m_drivers.size(); } IDBDriver *DBManager::GetDriver(unsigned int index) { if (index >= m_drivers.size()) { return NULL; } return m_drivers[index]; } const DatabaseInfo *DBManager::FindDatabaseConf(const char *name) { ConfDbInfo *info = GetDatabaseConf(name); if (!info) { return NULL; } return &info->info; } ConfDbInfo *DBManager::GetDatabaseConf(const char *name) { List::iterator iter; for (iter=m_confs.begin(); iter!=m_confs.end(); iter++) { ConfDbInfo &conf = (*iter); if (strcmp(m_StrTab.GetString(conf.name), name) == 0) { return &conf; } } return NULL; } IDBDriver *DBManager::FindOrLoadDriver(const char *name) { size_t last_size = m_drivers.size(); for (size_t i=0; iGetIdentifier(), name) == 0) { return m_drivers[i]; } } char filename[PLATFORM_MAX_PATH]; UTIL_Format(filename, sizeof(filename), "dbi.%s.ext", name); IExtension *pExt = g_Extensions.LoadAutoExtension(filename); if (!pExt || !pExt->IsLoaded() || m_drivers.size() <= last_size) { return NULL; } /* last_size is now gauranteed to be a valid index. * The identifier must match the name. */ if (strcmp(m_drivers[last_size]->GetIdentifier(), name) == 0) { return m_drivers[last_size]; } return NULL; }