mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-25 07:58:33 +01:00
0f33228155
the case where one of several guests wants to rematch is a hard problem for later.) Requires passing old-style relayIDs (connname plus device index) when devIDs aren't available, which they may not always be.
493 lines
15 KiB
C
493 lines
15 KiB
C
/* -*- compile-command: "make MEMDEBUG=TRUE -j3"; -*- */
|
|
/*
|
|
* Copyright 2000-2013 by Eric House (xwords@eehouse.org). 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
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include "gamesdb.h"
|
|
#include "main.h"
|
|
|
|
static void getColumnText( sqlite3_stmt *ppStmt, int iCol, XP_UCHAR* buf,
|
|
int len );
|
|
#ifdef DEBUG
|
|
static char* sqliteErr2str( int err );
|
|
static void assertPrintResult( sqlite3* pDb, int result, int expect );
|
|
#endif
|
|
|
|
sqlite3*
|
|
openGamesDB( const char* dbName )
|
|
{
|
|
int result = sqlite3_initialize();
|
|
XP_ASSERT( SQLITE_OK == result );
|
|
|
|
sqlite3* pDb = NULL;
|
|
result = sqlite3_open( dbName, &pDb );
|
|
XP_ASSERT( SQLITE_OK == result );
|
|
|
|
const char* createGamesStr =
|
|
"CREATE TABLE games ( "
|
|
"rowid INTEGER PRIMARY KEY AUTOINCREMENT"
|
|
",game BLOB"
|
|
",inviteInfo BLOB"
|
|
",room VARCHAR(32)"
|
|
",connvia VARCHAR(32)"
|
|
",ended INT(1)"
|
|
",turn INT(2)"
|
|
",nmoves INT"
|
|
",seed INT"
|
|
",gameid INT"
|
|
",ntotal INT(2)"
|
|
",nmissing INT(2)"
|
|
")";
|
|
result = sqlite3_exec( pDb, createGamesStr, NULL, NULL, NULL );
|
|
|
|
const char* createValuesStr =
|
|
"CREATE TABLE pairs ( key TEXT UNIQUE,value TEXT )";
|
|
result = sqlite3_exec( pDb, createValuesStr, NULL, NULL, NULL );
|
|
XP_LOGF( "sqlite3_exec=>%d", result );
|
|
XP_USE( result );
|
|
|
|
return pDb;
|
|
}
|
|
|
|
void
|
|
closeGamesDB( sqlite3* pDb )
|
|
{
|
|
sqlite3_close( pDb );
|
|
XP_LOGF( "%s finished", __func__ );
|
|
}
|
|
|
|
static sqlite3_int64
|
|
writeBlobColumn( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 curRow,
|
|
const char* column )
|
|
{
|
|
int result;
|
|
/* size includes stream version as header */
|
|
XP_U16 len = stream_getSize( stream );
|
|
char buf[256];
|
|
char* query;
|
|
|
|
sqlite3_stmt* stmt = NULL;
|
|
XP_Bool newGame = -1 == curRow;
|
|
if ( newGame ) { /* new row; need to insert blob first */
|
|
const char* fmt = "INSERT INTO games (%s) VALUES (?)";
|
|
snprintf( buf, sizeof(buf), fmt, column );
|
|
query = buf;
|
|
} else {
|
|
const char* fmt = "UPDATE games SET %s=? where rowid=%lld";
|
|
snprintf( buf, sizeof(buf), fmt, column, curRow );
|
|
query = buf;
|
|
}
|
|
|
|
result = sqlite3_prepare_v2( pDb, query, -1, &stmt, NULL );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
result = sqlite3_bind_zeroblob( stmt, 1 /*col 0 ??*/, sizeof(XP_U16) + len );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
result = sqlite3_step( stmt );
|
|
if ( SQLITE_DONE != result ) {
|
|
XP_LOGF( "%s: sqlite3_step => %s", __func__, sqliteErr2str( result ) );
|
|
XP_ASSERT(0);
|
|
}
|
|
XP_USE( result );
|
|
|
|
if ( newGame ) { /* new row; need to insert blob first */
|
|
curRow = sqlite3_last_insert_rowid( pDb );
|
|
XP_LOGF( "%s: new rowid: %lld", __func__, curRow );
|
|
}
|
|
|
|
sqlite3_blob* blob;
|
|
result = sqlite3_blob_open( pDb, "main", "games", column,
|
|
curRow, 1 /*flags: writeable*/, &blob );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
XP_U16 strVersion = stream_getVersion( stream );
|
|
XP_ASSERT( strVersion <= CUR_STREAM_VERS );
|
|
result = sqlite3_blob_write( blob, &strVersion, sizeof(strVersion), 0 );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
const XP_U8* ptr = stream_getPtr( stream );
|
|
result = sqlite3_blob_write( blob, ptr, len, sizeof(strVersion) );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
result = sqlite3_blob_close( blob );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
if ( !!stmt ) {
|
|
sqlite3_finalize( stmt );
|
|
}
|
|
|
|
return curRow;
|
|
}
|
|
|
|
sqlite3_int64
|
|
writeNewGameToDB( XWStreamCtxt* stream, sqlite3* pDb )
|
|
{
|
|
sqlite3_int64 newRow = writeBlobColumn( stream, pDb, -1, "game" );
|
|
return newRow;
|
|
}
|
|
|
|
void
|
|
writeToDB( XWStreamCtxt* stream, void* closure )
|
|
{
|
|
CommonGlobals* cGlobals = (CommonGlobals*)closure;
|
|
sqlite3_int64 selRow = cGlobals->selRow;
|
|
sqlite3* pDb = cGlobals->pDb;
|
|
|
|
XP_Bool newGame = -1 == selRow;
|
|
selRow = writeBlobColumn( stream, pDb, selRow, "game" );
|
|
|
|
if ( newGame ) { /* new row; need to insert blob first */
|
|
cGlobals->selRow = selRow;
|
|
}
|
|
|
|
(*cGlobals->onSave)( cGlobals->onSaveClosure, selRow, newGame );
|
|
}
|
|
|
|
void
|
|
summarize( CommonGlobals* cGlobals )
|
|
{
|
|
XP_S16 nMoves = model_getNMoves( cGlobals->game.model );
|
|
XP_Bool gameOver = server_getGameIsOver( cGlobals->game.server );
|
|
XP_S16 turn = server_getCurrentTurn( cGlobals->game.server );
|
|
XP_U16 seed = 0;
|
|
XP_S16 nMissing = 0;
|
|
XP_U16 nTotal = cGlobals->gi->nPlayers;
|
|
XP_U32 gameID = cGlobals->gi->gameID;
|
|
XP_ASSERT( 0 != gameID );
|
|
CommsAddrRec addr = {0};
|
|
gchar* room = "";
|
|
|
|
// gchar* connvia = "local";
|
|
gchar connvia[128] = {0};
|
|
|
|
if ( !!cGlobals->game.comms ) {
|
|
nMissing = server_getMissingPlayers( cGlobals->game.server );
|
|
comms_getAddr( cGlobals->game.comms, &addr );
|
|
CommsConnType typ;
|
|
for ( XP_U32 st = 0; addr_iter( &addr, &typ, &st ); ) {
|
|
if ( !!connvia[0] ) {
|
|
strcat( connvia, "+" );
|
|
}
|
|
switch( typ) {
|
|
case COMMS_CONN_RELAY:
|
|
room = addr.u.ip_relay.invite;
|
|
strcat( connvia, "Relay" );
|
|
break;
|
|
case COMMS_CONN_SMS:
|
|
strcat( connvia, "SMS" );
|
|
break;
|
|
case COMMS_CONN_BT:
|
|
strcat( connvia, "BT" );
|
|
break;
|
|
case COMMS_CONN_IP_DIRECT:
|
|
strcat( connvia, "IP" );
|
|
break;
|
|
default:
|
|
XP_ASSERT(0);
|
|
break;
|
|
}
|
|
}
|
|
seed = comms_getChannelSeed( cGlobals->game.comms );
|
|
} else {
|
|
strcat( connvia, "local" );
|
|
}
|
|
|
|
const char* fmt = "UPDATE games "
|
|
" SET room='%s', ended=%d, turn=%d, ntotal=%d, nmissing=%d, nmoves=%d, seed=%d, gameid=%d, connvia='%s'"
|
|
" WHERE rowid=%lld";
|
|
XP_UCHAR buf[256];
|
|
snprintf( buf, sizeof(buf), fmt, room, gameOver?1:0, turn, nTotal, nMissing,
|
|
nMoves, seed, gameID, connvia, cGlobals->selRow );
|
|
XP_LOGF( "query: %s", buf );
|
|
sqlite3_stmt* stmt = NULL;
|
|
int result = sqlite3_prepare_v2( cGlobals->pDb, buf, -1, &stmt, NULL );
|
|
assertPrintResult( cGlobals->pDb, result, SQLITE_OK );
|
|
result = sqlite3_step( stmt );
|
|
if ( SQLITE_DONE != result ) {
|
|
XP_LOGF( "sqlite3_step=>%s", sqliteErr2str( result ) );
|
|
XP_ASSERT( 0 );
|
|
}
|
|
sqlite3_finalize( stmt );
|
|
XP_USE( result );
|
|
}
|
|
|
|
GSList*
|
|
listGames( sqlite3* pDb )
|
|
{
|
|
GSList* list = NULL;
|
|
|
|
sqlite3_stmt *ppStmt;
|
|
int result = sqlite3_prepare_v2( pDb,
|
|
"SELECT rowid FROM games ORDER BY rowid",
|
|
-1, &ppStmt, NULL );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
XP_USE( result );
|
|
while ( NULL != ppStmt ) {
|
|
switch( sqlite3_step( ppStmt ) ) {
|
|
case SQLITE_ROW: /* have data */
|
|
{
|
|
sqlite3_int64* data = g_malloc( sizeof( *data ) );
|
|
*data = sqlite3_column_int64( ppStmt, 0 );
|
|
XP_LOGF( "%s: got a row; id=%lld", __func__, *data );
|
|
list = g_slist_append( list, data );
|
|
}
|
|
break;
|
|
case SQLITE_DONE:
|
|
sqlite3_finalize( ppStmt );
|
|
ppStmt = NULL;
|
|
break;
|
|
default:
|
|
XP_ASSERT( 0 );
|
|
break;
|
|
}
|
|
}
|
|
return list;
|
|
}
|
|
|
|
XP_Bool
|
|
getGameInfo( sqlite3* pDb, sqlite3_int64 rowid, GameInfo* gib )
|
|
{
|
|
XP_Bool success = XP_FALSE;
|
|
const char* fmt = "SELECT room, ended, turn, nmoves, ntotal, nmissing, "
|
|
"seed, connvia, gameid "
|
|
"FROM games WHERE rowid = %lld";
|
|
XP_UCHAR query[256];
|
|
snprintf( query, sizeof(query), fmt, rowid );
|
|
|
|
sqlite3_stmt* ppStmt;
|
|
int result = sqlite3_prepare_v2( pDb, query, -1, &ppStmt, NULL );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
result = sqlite3_step( ppStmt );
|
|
if ( SQLITE_ROW == result ) {
|
|
success = XP_TRUE;
|
|
getColumnText( ppStmt, 0, gib->room, sizeof(gib->room) );
|
|
gib->gameOver = 1 == sqlite3_column_int( ppStmt, 1 );
|
|
gib->turn = sqlite3_column_int( ppStmt, 2 );
|
|
gib->nMoves = sqlite3_column_int( ppStmt, 3 );
|
|
gib->nTotal = sqlite3_column_int( ppStmt, 4 );
|
|
gib->nMissing = sqlite3_column_int( ppStmt, 5 );
|
|
gib->seed = sqlite3_column_int( ppStmt, 6 );
|
|
getColumnText( ppStmt, 7, gib->conn, sizeof(gib->conn) );
|
|
gib->gameID = sqlite3_column_int( ppStmt, 8 );
|
|
snprintf( gib->name, sizeof(gib->name), "Game %lld", rowid );
|
|
}
|
|
sqlite3_finalize( ppStmt );
|
|
return success;
|
|
}
|
|
|
|
void
|
|
getRowsForGameID( sqlite3* pDb, XP_U32 gameID, sqlite3_int64* rowids,
|
|
int* nRowIDs )
|
|
{
|
|
int maxRowIDs = *nRowIDs;
|
|
*nRowIDs = 0;
|
|
|
|
char buf[256];
|
|
snprintf( buf, sizeof(buf), "SELECT rowid from games WHERE gameid = %d "
|
|
"LIMIT %d", gameID, maxRowIDs );
|
|
sqlite3_stmt *ppStmt;
|
|
int result = sqlite3_prepare_v2( pDb, buf, -1, &ppStmt, NULL );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
int ii;
|
|
for ( ii = 0; ii < maxRowIDs; ++ii ) {
|
|
result = sqlite3_step( ppStmt );
|
|
if ( SQLITE_ROW != result ) {
|
|
break;
|
|
}
|
|
rowids[ii] = sqlite3_column_int64( ppStmt, 0 );
|
|
++*nRowIDs;
|
|
}
|
|
sqlite3_finalize( ppStmt );
|
|
}
|
|
|
|
static XP_Bool
|
|
loadBlobColumn( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid,
|
|
const char* column )
|
|
{
|
|
char buf[256];
|
|
snprintf( buf, sizeof(buf), "SELECT %s from games WHERE rowid = %lld",
|
|
column, rowid );
|
|
|
|
sqlite3_stmt *ppStmt;
|
|
int result = sqlite3_prepare_v2( pDb, buf, -1, &ppStmt, NULL );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
result = sqlite3_step( ppStmt );
|
|
XP_Bool success = SQLITE_ROW == result;
|
|
if ( success ) {
|
|
const void* ptr = sqlite3_column_blob( ppStmt, 0 );
|
|
int size = sqlite3_column_bytes( ppStmt, 0 );
|
|
success = 0 < size;
|
|
if ( success ) {
|
|
XP_U16 strVersion;
|
|
XP_MEMCPY( &strVersion, ptr, sizeof(strVersion) );
|
|
XP_ASSERT( strVersion <= CUR_STREAM_VERS );
|
|
stream_setVersion( stream, strVersion );
|
|
XP_ASSERT( size >= sizeof(strVersion) );
|
|
stream_putBytes( stream, ptr + sizeof(strVersion),
|
|
size - sizeof(strVersion) );
|
|
}
|
|
}
|
|
sqlite3_finalize( ppStmt );
|
|
return success;
|
|
}
|
|
|
|
XP_Bool
|
|
loadGame( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid )
|
|
{
|
|
return loadBlobColumn( stream, pDb, rowid, "game" );
|
|
}
|
|
|
|
void
|
|
saveInviteAddrs( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid )
|
|
{
|
|
sqlite3_int64 row = writeBlobColumn( stream, pDb, rowid, "inviteInfo" );
|
|
assert( row == rowid );
|
|
}
|
|
|
|
XP_Bool
|
|
loadInviteAddrs( XWStreamCtxt* stream, sqlite3* pDb, sqlite3_int64 rowid )
|
|
{
|
|
return loadBlobColumn( stream, pDb, rowid, "inviteInfo" );
|
|
}
|
|
|
|
void
|
|
deleteGame( sqlite3* pDb, sqlite3_int64 rowid )
|
|
{
|
|
char query[256];
|
|
snprintf( query, sizeof(query), "DELETE FROM games WHERE rowid = %lld", rowid );
|
|
sqlite3_stmt* ppStmt;
|
|
int result = sqlite3_prepare_v2( pDb, query, -1, &ppStmt, NULL );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
result = sqlite3_step( ppStmt );
|
|
assertPrintResult( pDb, result, SQLITE_DONE );
|
|
XP_USE( result );
|
|
sqlite3_finalize( ppStmt );
|
|
}
|
|
|
|
void
|
|
db_store( sqlite3* pDb, const gchar* key, const gchar* value )
|
|
{
|
|
char buf[256];
|
|
snprintf( buf, sizeof(buf),
|
|
"INSERT OR REPLACE INTO pairs (key, value) VALUES ('%s', '%s')",
|
|
key, value );
|
|
sqlite3_stmt *ppStmt;
|
|
int result = sqlite3_prepare_v2( pDb, buf, -1, &ppStmt, NULL );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
result = sqlite3_step( ppStmt );
|
|
assertPrintResult( pDb, result, SQLITE_DONE );
|
|
XP_USE( result );
|
|
sqlite3_finalize( ppStmt );
|
|
}
|
|
|
|
XP_Bool
|
|
db_fetch( sqlite3* pDb, const gchar* key, gchar* buf, gint buflen )
|
|
{
|
|
char query[256];
|
|
snprintf( query, sizeof(query),
|
|
"SELECT value from pairs where key = '%s'", key );
|
|
sqlite3_stmt *ppStmt;
|
|
int result = sqlite3_prepare_v2( pDb, query, -1, &ppStmt, NULL );
|
|
XP_Bool found = SQLITE_OK == result;
|
|
if ( found ) {
|
|
result = sqlite3_step( ppStmt );
|
|
found = SQLITE_ROW == result;
|
|
if ( found ) {
|
|
getColumnText( ppStmt, 0, buf, buflen );
|
|
} else {
|
|
buf[0] = '\0';
|
|
}
|
|
}
|
|
sqlite3_finalize( ppStmt );
|
|
return found;
|
|
}
|
|
|
|
void
|
|
db_remove( sqlite3* pDb, const gchar* key )
|
|
{
|
|
char query[256];
|
|
snprintf( query, sizeof(query), "DELETE FROM pairs WHERE key = '%s'", key );
|
|
sqlite3_stmt *ppStmt;
|
|
int result = sqlite3_prepare_v2( pDb, query, -1, &ppStmt, NULL );
|
|
assertPrintResult( pDb, result, SQLITE_OK );
|
|
result = sqlite3_step( ppStmt );
|
|
assertPrintResult( pDb, result, SQLITE_DONE );
|
|
XP_USE( result );
|
|
sqlite3_finalize( ppStmt );
|
|
}
|
|
|
|
static void
|
|
getColumnText( sqlite3_stmt *ppStmt, int iCol, XP_UCHAR* buf,
|
|
int XP_UNUSED_DBG(len) )
|
|
{
|
|
const unsigned char* txt = sqlite3_column_text( ppStmt, iCol );
|
|
int needLen = sqlite3_column_bytes( ppStmt, iCol );
|
|
XP_ASSERT( needLen < len );
|
|
XP_MEMCPY( buf, txt, needLen );
|
|
buf[needLen] = '\0';
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
# define CASESTR(c) case c: return #c
|
|
static char*
|
|
sqliteErr2str( int err )
|
|
{
|
|
switch( err ) {
|
|
CASESTR( SQLITE_OK );
|
|
CASESTR( SQLITE_ERROR );
|
|
CASESTR( SQLITE_INTERNAL );
|
|
CASESTR( SQLITE_PERM );
|
|
CASESTR( SQLITE_ABORT );
|
|
CASESTR( SQLITE_BUSY );
|
|
CASESTR( SQLITE_LOCKED );
|
|
CASESTR( SQLITE_NOMEM );
|
|
CASESTR( SQLITE_READONLY );
|
|
CASESTR( SQLITE_INTERRUPT );
|
|
CASESTR( SQLITE_IOERR );
|
|
CASESTR( SQLITE_CORRUPT );
|
|
CASESTR( SQLITE_NOTFOUND );
|
|
CASESTR( SQLITE_FULL );
|
|
CASESTR( SQLITE_CANTOPEN );
|
|
CASESTR( SQLITE_PROTOCOL );
|
|
CASESTR( SQLITE_EMPTY );
|
|
CASESTR( SQLITE_SCHEMA );
|
|
CASESTR( SQLITE_TOOBIG );
|
|
CASESTR( SQLITE_CONSTRAINT );
|
|
CASESTR( SQLITE_MISMATCH );
|
|
CASESTR( SQLITE_MISUSE );
|
|
CASESTR( SQLITE_NOLFS );
|
|
CASESTR( SQLITE_AUTH );
|
|
CASESTR( SQLITE_FORMAT );
|
|
CASESTR( SQLITE_RANGE );
|
|
CASESTR( SQLITE_NOTADB );
|
|
CASESTR( SQLITE_NOTICE );
|
|
CASESTR( SQLITE_WARNING );
|
|
CASESTR( SQLITE_ROW );
|
|
CASESTR( SQLITE_DONE );
|
|
}
|
|
return "<unknown>";
|
|
}
|
|
|
|
static void
|
|
assertPrintResult( sqlite3* pDb, int result, int expect )
|
|
{
|
|
int code = sqlite3_errcode( pDb );
|
|
XP_ASSERT( code == result ); /* do I need to pass it? */
|
|
if ( code != expect ) {
|
|
const char* msg = sqlite3_errmsg( pDb );
|
|
XP_LOGF( "sqlite3 error: %s", msg );
|
|
XP_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
#endif
|