/** * vim: set ts=4 : * ============================================================================= * 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 . * * 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 "extension.h" /** * @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; static const DatabaseInfo *storage_local = NULL; bool ClientPrefs::SDK_OnLoad(char *error, size_t maxlength, bool late) { queryMutex = threader->MakeMutex(); cookieMutex = threader->MakeMutex(); DBInfo = dbi->FindDatabaseConf("clientprefs"); if (DBInfo == NULL) { DBInfo = dbi->FindDatabaseConf("default"); if (DBInfo == NULL || (strcmp(DBInfo->host, "localhost") == 0 && strcmp(DBInfo->database, "sourcemod") == 0 && strcmp(DBInfo->user, "root") == 0 && strcmp(DBInfo->pass, "") == 0 && strcmp(DBInfo->driver, "") == 0)) { storage_local = dbi->FindDatabaseConf("storage-local"); if (DBInfo == NULL) { DBInfo = storage_local; } } if (DBInfo == NULL) { snprintf(error, maxlength, "Could not find \"clientprefs\" or \"default\" database configs"); return false; } } if (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; } Database = NULL; 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) { int maxclients = playerhelpers->GetMaxClients(); for (int i = 1; i <= maxclients; i++) { IGamePlayer *pPlayer = playerhelpers->GetGamePlayer(i); if (!pPlayer || !pPlayer->IsAuthorized()) { continue; } g_CookieManager.OnClientAuthorized(i, pPlayer->GetAuthString()); } } 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 != NULL && (void *)pInterface == (void *)(Database->GetDriver())) { Database->Close(); Database = NULL; } } void ClientPrefs::SDK_OnUnload() { handlesys->RemoveType(g_CookieType, myself->GetIdentity()); handlesys->RemoveType(g_CookieIterator, myself->GetIdentity()); g_CookieManager.Unload(); if (Database != NULL) { Database->Close(); } forwards->ReleaseForward(g_CookieManager.cookieDataLoadedForward); HandleSecurity sec = HandleSecurity(identity, identity); HandleError err = handlesys->FreeHandle(g_CookieManager.clientMenu->GetHandle(), &sec); if (HandleError_None != err) { g_pSM->LogError(myself, "Error %d when attempting to free client menu handle", err); } phrases->Destroy(); sharesys->DestroyIdentity( identity ); plsys->RemovePluginsListener(&g_CookieManager); playerhelpers->RemoveClientListener(&g_CookieManager); queryMutex->DestroyThis(); cookieMutex->DestroyThis(); } void ClientPrefs::OnCoreMapStart(edict_t *pEdictList, int edictCount, int clientMax) { if (Database == NULL && !databaseLoading) { g_pSM->LogMessage(myself, "Attempting to reconnect to database..."); databaseLoading = true; TQueryOp *op = new TQueryOp(Query_Connect, 0); dbi->AddToThreadQueue(op, PrioQueue_High); } } void ClientPrefs::DatabaseConnect() { char error[256]; int errCode = 0; Database = Driver->Connect(DBInfo, true, error, sizeof(error)); if (Database == NULL && DBInfo != storage_local && storage_local != NULL) { DBInfo = storage_local; Database = Driver->Connect(DBInfo, true, error, sizeof(error)); } if (Database == NULL) { g_pSM->LogError(myself, error); databaseLoading = false; ProcessQueryCache(); 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; ProcessQueryCache(); return; fatal_fail: Database->Close(); Database = NULL; databaseLoading = false; ProcessQueryCache(); } bool ClientPrefs::AddQueryToQueue( TQueryOp *query ) { queryMutex->Lock(); if (Database == NULL && databaseLoading) { cachedQueries.push_back(query); queryMutex->Unlock(); return true; } queryMutex->Unlock(); if (Database) { query->SetDatabase(Database); dbi->AddToThreadQueue(query, PrioQueue_Normal); return true; } query->Destroy(); /* If Database is NULL and we're not in the loading phase it must have failed - Can't do much */ return false; } void ClientPrefs::ProcessQueryCache() { SourceHook::List::iterator iter; queryMutex->Lock(); iter = cachedQueries.begin(); while (iter != cachedQueries.end()) { TQueryOp *op = *iter; if (Database != NULL) { op->SetDatabase(Database); dbi->AddToThreadQueue(op, PrioQueue_Normal); } else { op->Destroy(); } iter++; } cachedQueries.clear(); queryMutex->Unlock(); } size_t IsAuthIdConnected(char *authID) { IGamePlayer *player; int maxPlayers = playerhelpers->GetMaxClients(); for (int playerIndex = 1; playerIndex <= maxPlayers; playerIndex++) { player = playerhelpers->GetGamePlayer(playerIndex); if (!player || !player->IsConnected()) { continue; } const char *authString = player->GetAuthString(); if (!authString || authString[0] == '\0') { continue; } if (strcmp(authString, authID) == 0) { return playerIndex; } } return 0; } size_t UTIL_Format(char *buffer, size_t maxlength, const char *fmt, ...) { va_list ap; va_start(ap, fmt); size_t len = vsnprintf(buffer, maxlength, fmt, ap); va_end(ap); if (len >= maxlength) { buffer[maxlength - 1] = '\0'; return (maxlength - 1); } else { return len; } } 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; } IdentityToken_t *ClientPrefs::GetIdentity() const { return identity; } const char *ClientPrefs::GetExtensionVerString() { return SM_VERSION_STRING; } const char *ClientPrefs::GetExtensionDateString() { return SM_BUILD_TIMESTAMP; }