eb54cbb4f5
added a basic test plugin --HG-- extra : convert_revision : svn%3A39bc706e-5318-0410-9160-8a85361fbb7c/trunk%40875
420 lines
9.4 KiB
C++
420 lines
9.4 KiB
C++
/**
|
|
* 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 <stdlib.h>
|
|
|
|
#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<m_drivers.size(); i++)
|
|
{
|
|
if (m_drivers[i] == pDriver)
|
|
{
|
|
m_drivers.erase(m_drivers.iterAt(i));
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Make sure NOTHING references this! */
|
|
List<ConfDbInfo>::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<ConfDbInfo>::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; i<last_size; i++)
|
|
{
|
|
if (strcmp(m_drivers[i]->GetIdentifier(), 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;
|
|
}
|