490 lines
12 KiB
C++
490 lines
12 KiB
C++
/**
|
|
* vim: set ts=4 sw=4 tw=99 noet:
|
|
* =============================================================================
|
|
* SourceMod Client Preferences Extension
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
*
|
|
* 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 <http://www.sourcemod.net/license.php>.
|
|
*
|
|
* Version: $Id$
|
|
*/
|
|
|
|
#include <sourcemod_version.h>
|
|
#include "extension.h"
|
|
|
|
using namespace ke;
|
|
|
|
/**
|
|
* @file extension.cpp
|
|
* @brief Implement extension code here.
|
|
*/
|
|
|
|
ClientPrefs g_ClientPrefs; /**< Global singleton for extension's main interface */
|
|
|
|
SMEXT_LINK(&g_ClientPrefs);
|
|
|
|
HandleType_t g_CookieType = 0;
|
|
CookieTypeHandler g_CookieTypeHandler;
|
|
|
|
HandleType_t g_CookieIterator = 0;
|
|
CookieIteratorHandler g_CookieIteratorHandler;
|
|
DbDriver g_DriverType;
|
|
|
|
bool ClientPrefs::SDK_OnLoad(char *error, size_t maxlength, bool late)
|
|
{
|
|
DBInfo = dbi->FindDatabaseConf("clientprefs");
|
|
|
|
if (DBInfo == NULL)
|
|
{
|
|
DBInfo = dbi->FindDatabaseConf("default");
|
|
|
|
if (DBInfo == NULL)
|
|
{
|
|
DBInfo = dbi->FindDatabaseConf("storage-local");
|
|
}
|
|
}
|
|
|
|
if (DBInfo == NULL)
|
|
{
|
|
snprintf(error, maxlength, "Could not find any suitable database configs");
|
|
return false;
|
|
}
|
|
|
|
if (DBInfo->driver && DBInfo->driver[0] != '\0')
|
|
{
|
|
Driver = dbi->FindOrLoadDriver(DBInfo->driver);
|
|
}
|
|
else
|
|
{
|
|
Driver = dbi->GetDefaultDriver();
|
|
}
|
|
|
|
if (Driver == NULL)
|
|
{
|
|
snprintf(error, maxlength, "Could not load DB Driver \"%s\"", DBInfo->driver);
|
|
return false;
|
|
}
|
|
|
|
databaseLoading = true;
|
|
TQueryOp *op = new TQueryOp(Query_Connect, 0);
|
|
dbi->AddToThreadQueue(op, PrioQueue_High);
|
|
|
|
dbi->AddDependency(myself, Driver);
|
|
|
|
sharesys->AddNatives(myself, g_ClientPrefNatives);
|
|
sharesys->RegisterLibrary(myself, "clientprefs");
|
|
identity = sharesys->CreateIdentity(sharesys->CreateIdentType("ClientPrefs"), this);
|
|
g_CookieManager.cookieDataLoadedForward = forwards->CreateForward("OnClientCookiesCached", ET_Ignore, 1, NULL, Param_Cell);
|
|
|
|
g_CookieType = handlesys->CreateType("Cookie",
|
|
&g_CookieTypeHandler,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
myself->GetIdentity(),
|
|
NULL);
|
|
|
|
g_CookieIterator = handlesys->CreateType("CookieIterator",
|
|
&g_CookieIteratorHandler,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
myself->GetIdentity(),
|
|
NULL);
|
|
|
|
IMenuStyle *style = menus->GetDefaultStyle();
|
|
g_CookieManager.clientMenu = style->CreateMenu(&g_Handler, identity);
|
|
g_CookieManager.clientMenu->SetDefaultTitle("Client Settings:");
|
|
|
|
plsys->AddPluginsListener(&g_CookieManager);
|
|
|
|
phrases = translator->CreatePhraseCollection();
|
|
phrases->AddPhraseFile("clientprefs.phrases");
|
|
phrases->AddPhraseFile("common.phrases");
|
|
|
|
if (late)
|
|
{
|
|
this->CatchLateLoadClients();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void ClientPrefs::SDK_OnAllLoaded()
|
|
{
|
|
playerhelpers->AddClientListener(&g_CookieManager);
|
|
}
|
|
|
|
bool ClientPrefs::QueryInterfaceDrop(SMInterface *pInterface)
|
|
{
|
|
if ((void *)pInterface == (void *)(Database->GetDriver()))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void ClientPrefs::NotifyInterfaceDrop(SMInterface *pInterface)
|
|
{
|
|
if (Database && (void *)pInterface == (void *)(Database->GetDriver()))
|
|
Database = NULL;
|
|
}
|
|
|
|
void ClientPrefs::SDK_OnDependenciesDropped()
|
|
{
|
|
// At this point, we're guaranteed that DBI has flushed the worker thread
|
|
// for us, so no cookies should have outstanding queries.
|
|
g_CookieManager.Unload();
|
|
|
|
handlesys->RemoveType(g_CookieType, myself->GetIdentity());
|
|
handlesys->RemoveType(g_CookieIterator, myself->GetIdentity());
|
|
|
|
Database = NULL;
|
|
|
|
if (g_CookieManager.cookieDataLoadedForward != NULL)
|
|
{
|
|
forwards->ReleaseForward(g_CookieManager.cookieDataLoadedForward);
|
|
g_CookieManager.cookieDataLoadedForward = NULL;
|
|
}
|
|
|
|
if (g_CookieManager.clientMenu != NULL)
|
|
{
|
|
Handle_t menuHandle = g_CookieManager.clientMenu->GetHandle();
|
|
|
|
if (menuHandle != BAD_HANDLE)
|
|
{
|
|
HandleSecurity sec = HandleSecurity(identity, identity);
|
|
HandleError err = handlesys->FreeHandle(menuHandle, &sec);
|
|
if (HandleError_None != err)
|
|
{
|
|
g_pSM->LogError(myself, "Error %d when attempting to free client menu handle", err);
|
|
}
|
|
}
|
|
|
|
g_CookieManager.clientMenu = NULL;
|
|
}
|
|
|
|
if (phrases != NULL)
|
|
{
|
|
phrases->Destroy();
|
|
phrases = NULL;
|
|
}
|
|
|
|
plsys->RemovePluginsListener(&g_CookieManager);
|
|
playerhelpers->RemoveClientListener(&g_CookieManager);
|
|
}
|
|
|
|
void ClientPrefs::OnCoreMapStart(edict_t *pEdictList, int edictCount, int clientMax)
|
|
{
|
|
this->AttemptReconnection();
|
|
}
|
|
|
|
void ClientPrefs::AttemptReconnection()
|
|
{
|
|
if (Database || databaseLoading)
|
|
return; /* We're already loading, or have loaded. */
|
|
|
|
g_pSM->LogMessage(myself, "Attempting to reconnect to database...");
|
|
databaseLoading = true;
|
|
|
|
TQueryOp *op = new TQueryOp(Query_Connect, 0);
|
|
dbi->AddToThreadQueue(op, PrioQueue_High);
|
|
|
|
this->CatchLateLoadClients(); /* DB reconnection, we should check if we missed anyone... */
|
|
}
|
|
|
|
void ClientPrefs::DatabaseConnect()
|
|
{
|
|
char error[256];
|
|
int errCode = 0;
|
|
|
|
Database = AdoptRef(Driver->Connect(DBInfo, true, error, sizeof(error)));
|
|
|
|
if (!Database)
|
|
{
|
|
g_pSM->LogError(myself, error);
|
|
databaseLoading = false;
|
|
return;
|
|
}
|
|
|
|
const char *identifier = Driver->GetIdentifier();
|
|
|
|
if (strcmp(identifier, "sqlite") == 0)
|
|
{
|
|
g_DriverType = Driver_SQLite;
|
|
|
|
if (!Database->DoSimpleQuery(
|
|
"CREATE TABLE IF NOT EXISTS sm_cookies \
|
|
( \
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT, \
|
|
name varchar(30) NOT NULL UNIQUE, \
|
|
description varchar(255), \
|
|
access INTEGER \
|
|
)"))
|
|
{
|
|
g_pSM->LogMessage(myself, "Failed to CreateTable sm_cookies: %s", Database->GetError());
|
|
goto fatal_fail;
|
|
}
|
|
|
|
if (!Database->DoSimpleQuery(
|
|
"CREATE TABLE IF NOT EXISTS sm_cookie_cache \
|
|
( \
|
|
player varchar(65) NOT NULL, \
|
|
cookie_id int(10) NOT NULL, \
|
|
value varchar(100), \
|
|
timestamp int, \
|
|
PRIMARY KEY (player, cookie_id) \
|
|
)"))
|
|
{
|
|
g_pSM->LogMessage(myself, "Failed to CreateTable sm_cookie_cache: %s", Database->GetError());
|
|
goto fatal_fail;
|
|
}
|
|
}
|
|
else if (strcmp(identifier, "mysql") == 0)
|
|
{
|
|
g_DriverType = Driver_MySQL;
|
|
|
|
if (!Database->DoSimpleQuery(
|
|
"CREATE TABLE IF NOT EXISTS sm_cookies \
|
|
( \
|
|
id INTEGER unsigned NOT NULL auto_increment, \
|
|
name varchar(30) NOT NULL UNIQUE, \
|
|
description varchar(255), \
|
|
access INTEGER, \
|
|
PRIMARY KEY (id) \
|
|
)"))
|
|
{
|
|
g_pSM->LogMessage(myself, "Failed to CreateTable sm_cookies: %s", Database->GetError());
|
|
goto fatal_fail;
|
|
}
|
|
|
|
if (!Database->DoSimpleQuery(
|
|
"CREATE TABLE IF NOT EXISTS sm_cookie_cache \
|
|
( \
|
|
player varchar(65) NOT NULL, \
|
|
cookie_id int(10) NOT NULL, \
|
|
value varchar(100), \
|
|
timestamp int NOT NULL, \
|
|
PRIMARY KEY (player, cookie_id) \
|
|
)"))
|
|
{
|
|
g_pSM->LogMessage(myself, "Failed to CreateTable sm_cookie_cache: %s", Database->GetError());
|
|
goto fatal_fail;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
g_pSM->LogError(myself, "Unsupported driver \"%s\"", identifier);
|
|
goto fatal_fail;
|
|
}
|
|
|
|
databaseLoading = false;
|
|
|
|
// Need a new scope because of the goto above.
|
|
{
|
|
AutoLock lock(&queryLock);
|
|
this->ProcessQueryCache();
|
|
}
|
|
return;
|
|
|
|
fatal_fail:
|
|
Database = NULL;
|
|
databaseLoading = false;
|
|
}
|
|
|
|
bool ClientPrefs::AddQueryToQueue(TQueryOp *query)
|
|
{
|
|
{
|
|
AutoLock lock(&queryLock);
|
|
if (!Database)
|
|
{
|
|
cachedQueries.append(query);
|
|
return false;
|
|
}
|
|
|
|
if (!cachedQueries.empty())
|
|
this->ProcessQueryCache();
|
|
}
|
|
|
|
query->SetDatabase(Database);
|
|
dbi->AddToThreadQueue(query, PrioQueue_Normal);
|
|
return true;
|
|
}
|
|
|
|
void ClientPrefs::ProcessQueryCache()
|
|
{
|
|
queryLock.AssertCurrentThreadOwns();
|
|
|
|
if (!Database)
|
|
return;
|
|
|
|
for (size_t iter = 0; iter < cachedQueries.length(); ++iter)
|
|
{
|
|
TQueryOp *op = cachedQueries[iter];
|
|
op->SetDatabase(Database);
|
|
dbi->AddToThreadQueue(op, PrioQueue_Normal);
|
|
}
|
|
|
|
cachedQueries.clear();
|
|
}
|
|
|
|
const char *GetPlayerCompatAuthId(IGamePlayer *pPlayer)
|
|
{
|
|
/* For legacy reasons, OnClientAuthorized gives the Steam2 id here if using Steam auth */
|
|
const char *steamId = pPlayer->GetSteam2Id();
|
|
return steamId ? steamId : pPlayer->GetAuthString();
|
|
}
|
|
|
|
void ClientPrefs::CatchLateLoadClients()
|
|
{
|
|
IGamePlayer *pPlayer;
|
|
for (int i = playerhelpers->GetMaxClients()+1; --i > 0;)
|
|
{
|
|
if (g_CookieManager.AreClientCookiesPending(i) || g_CookieManager.AreClientCookiesCached(i))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
pPlayer = playerhelpers->GetGamePlayer(i);
|
|
|
|
if (!pPlayer || !pPlayer->IsAuthorized())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
g_CookieManager.OnClientAuthorized(i, GetPlayerCompatAuthId(pPlayer));
|
|
}
|
|
}
|
|
|
|
void ClientPrefs::ClearQueryCache(int serial)
|
|
{
|
|
AutoLock lock(&queryLock);
|
|
for (size_t iter = 0; iter < cachedQueries.length(); ++iter)
|
|
{
|
|
TQueryOp *op = cachedQueries[iter];
|
|
if (op && op->PullQueryType() == Query_SelectData && op->PullQuerySerial() == serial)
|
|
{
|
|
op->Destroy();
|
|
cachedQueries.remove(iter--);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool Translate(char *buffer,
|
|
size_t maxlength,
|
|
const char *format,
|
|
unsigned int numparams,
|
|
size_t *pOutLength,
|
|
...)
|
|
{
|
|
va_list ap;
|
|
unsigned int i;
|
|
const char *fail_phrase;
|
|
void *params[MAX_TRANSLATE_PARAMS];
|
|
|
|
if (numparams > MAX_TRANSLATE_PARAMS)
|
|
{
|
|
assert(false);
|
|
return false;
|
|
}
|
|
|
|
va_start(ap, pOutLength);
|
|
for (i = 0; i < numparams; i++)
|
|
{
|
|
params[i] = va_arg(ap, void *);
|
|
}
|
|
va_end(ap);
|
|
|
|
if (!g_ClientPrefs.phrases->FormatString(buffer,
|
|
maxlength,
|
|
format,
|
|
params,
|
|
numparams,
|
|
pOutLength,
|
|
&fail_phrase))
|
|
{
|
|
if (fail_phrase != NULL)
|
|
{
|
|
g_pSM->LogError(myself, "[SM] Could not find core phrase: %s", fail_phrase);
|
|
}
|
|
else
|
|
{
|
|
g_pSM->LogError(myself, "[SM] Unknown fatal error while translating a core phrase.");
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
char * UTIL_strncpy(char * destination, const char * source, size_t num)
|
|
{
|
|
if (source == NULL)
|
|
{
|
|
destination[0] = '\0';
|
|
return destination;
|
|
}
|
|
|
|
size_t req = strlen(source);
|
|
if (!req)
|
|
{
|
|
destination[0] = '\0';
|
|
return destination;
|
|
}
|
|
else if (req >= num)
|
|
{
|
|
req = num-1;
|
|
}
|
|
|
|
strncpy(destination, source, req);
|
|
destination[req] = '\0';
|
|
return destination;
|
|
}
|
|
|
|
IdentityToken_t *ClientPrefs::GetIdentity() const
|
|
{
|
|
return identity;
|
|
}
|
|
|
|
const char *ClientPrefs::GetExtensionVerString()
|
|
{
|
|
return SOURCEMOD_VERSION;
|
|
}
|
|
|
|
const char *ClientPrefs::GetExtensionDateString()
|
|
{
|
|
return SOURCEMOD_BUILD_TIME;
|
|
}
|
|
|
|
ClientPrefs::ClientPrefs()
|
|
{
|
|
Driver = NULL;
|
|
databaseLoading = false;
|
|
phrases = NULL;
|
|
DBInfo = NULL;
|
|
|
|
identity = NULL;
|
|
}
|