175897234Sdrh /* 2b19a2bc6Sdrh ** 2001 September 15 375897234Sdrh ** 4b19a2bc6Sdrh ** The author disclaims copyright to this source code. In place of 5b19a2bc6Sdrh ** a legal notice, here is a blessing: 675897234Sdrh ** 7b19a2bc6Sdrh ** May you do good and not evil. 8b19a2bc6Sdrh ** May you find forgiveness for yourself and forgive others. 9b19a2bc6Sdrh ** May you share freely, never taking more than you give. 1075897234Sdrh ** 1175897234Sdrh ************************************************************************* 12bd08af48Sdrh ** A TCL Interface to SQLite. Append this file to sqlite3.c and 13bd08af48Sdrh ** compile the whole thing to build a TCL-enabled version of SQLite. 1475897234Sdrh ** 15*404ca075Sdanielk1977 ** $Id: tclsqlite.c,v 1.238 2009/03/16 13:19:36 danielk1977 Exp $ 1675897234Sdrh */ 17bd08af48Sdrh #include "tcl.h" 18b4e9af9fSdanielk1977 #include <errno.h> 196d31316cSdrh 20bd08af48Sdrh /* 21bd08af48Sdrh ** Some additional include files are needed if this file is not 22bd08af48Sdrh ** appended to the amalgamation. 23bd08af48Sdrh */ 24bd08af48Sdrh #ifndef SQLITE_AMALGAMATION 2506b2718aSdrh # include "sqliteInt.h" 2675897234Sdrh # include <stdlib.h> 2775897234Sdrh # include <string.h> 28ce927065Sdrh # include <assert.h> 29d1e4733dSdrh # include <ctype.h> 30bd08af48Sdrh #endif 3175897234Sdrh 32ad6e1370Sdrh /* 33ad6e1370Sdrh * Windows needs to know which symbols to export. Unix does not. 34ad6e1370Sdrh * BUILD_sqlite should be undefined for Unix. 35ad6e1370Sdrh */ 36ad6e1370Sdrh #ifdef BUILD_sqlite 37ad6e1370Sdrh #undef TCL_STORAGE_CLASS 38ad6e1370Sdrh #define TCL_STORAGE_CLASS DLLEXPORT 39ad6e1370Sdrh #endif /* BUILD_sqlite */ 4029bc4615Sdrh 41a21c6b6fSdanielk1977 #define NUM_PREPARED_STMTS 10 42fb7e7651Sdrh #define MAX_PREPARED_STMTS 100 43fb7e7651Sdrh 4475897234Sdrh /* 4598808babSdrh ** If TCL uses UTF-8 and SQLite is configured to use iso8859, then we 4698808babSdrh ** have to do a translation when going between the two. Set the 4798808babSdrh ** UTF_TRANSLATION_NEEDED macro to indicate that we need to do 4898808babSdrh ** this translation. 4998808babSdrh */ 5098808babSdrh #if defined(TCL_UTF_MAX) && !defined(SQLITE_UTF8) 5198808babSdrh # define UTF_TRANSLATION_NEEDED 1 5298808babSdrh #endif 5398808babSdrh 5498808babSdrh /* 55cabb0819Sdrh ** New SQL functions can be created as TCL scripts. Each such function 56cabb0819Sdrh ** is described by an instance of the following structure. 57cabb0819Sdrh */ 58cabb0819Sdrh typedef struct SqlFunc SqlFunc; 59cabb0819Sdrh struct SqlFunc { 60cabb0819Sdrh Tcl_Interp *interp; /* The TCL interpret to execute the function */ 61d1e4733dSdrh Tcl_Obj *pScript; /* The Tcl_Obj representation of the script */ 62d1e4733dSdrh int useEvalObjv; /* True if it is safe to use Tcl_EvalObjv */ 63d1e4733dSdrh char *zName; /* Name of this function */ 64cabb0819Sdrh SqlFunc *pNext; /* Next function on the list of them all */ 65cabb0819Sdrh }; 66cabb0819Sdrh 67cabb0819Sdrh /* 680202b29eSdanielk1977 ** New collation sequences function can be created as TCL scripts. Each such 690202b29eSdanielk1977 ** function is described by an instance of the following structure. 700202b29eSdanielk1977 */ 710202b29eSdanielk1977 typedef struct SqlCollate SqlCollate; 720202b29eSdanielk1977 struct SqlCollate { 730202b29eSdanielk1977 Tcl_Interp *interp; /* The TCL interpret to execute the function */ 740202b29eSdanielk1977 char *zScript; /* The script to be run */ 750202b29eSdanielk1977 SqlCollate *pNext; /* Next function on the list of them all */ 760202b29eSdanielk1977 }; 770202b29eSdanielk1977 780202b29eSdanielk1977 /* 79fb7e7651Sdrh ** Prepared statements are cached for faster execution. Each prepared 80fb7e7651Sdrh ** statement is described by an instance of the following structure. 81fb7e7651Sdrh */ 82fb7e7651Sdrh typedef struct SqlPreparedStmt SqlPreparedStmt; 83fb7e7651Sdrh struct SqlPreparedStmt { 84fb7e7651Sdrh SqlPreparedStmt *pNext; /* Next in linked list */ 85fb7e7651Sdrh SqlPreparedStmt *pPrev; /* Previous on the list */ 86fb7e7651Sdrh sqlite3_stmt *pStmt; /* The prepared statement */ 87fb7e7651Sdrh int nSql; /* chars in zSql[] */ 88d0e2a854Sdanielk1977 const char *zSql; /* Text of the SQL statement */ 89fb7e7651Sdrh }; 90fb7e7651Sdrh 91d0441796Sdanielk1977 typedef struct IncrblobChannel IncrblobChannel; 92d0441796Sdanielk1977 93fb7e7651Sdrh /* 94bec3f402Sdrh ** There is one instance of this structure for each SQLite database 95bec3f402Sdrh ** that has been opened by the SQLite TCL interface. 96bec3f402Sdrh */ 97bec3f402Sdrh typedef struct SqliteDb SqliteDb; 98bec3f402Sdrh struct SqliteDb { 99dddca286Sdrh sqlite3 *db; /* The "real" database structure. MUST BE FIRST */ 100bec3f402Sdrh Tcl_Interp *interp; /* The interpreter used for this database */ 1016d31316cSdrh char *zBusy; /* The busy callback routine */ 102aa940eacSdrh char *zCommit; /* The commit hook callback routine */ 103b5a20d3cSdrh char *zTrace; /* The trace callback routine */ 10419e2d37fSdrh char *zProfile; /* The profile callback routine */ 105348bb5d6Sdanielk1977 char *zProgress; /* The progress callback routine */ 106e22a334bSdrh char *zAuth; /* The authorization callback routine */ 1071f1549f8Sdrh int disableAuth; /* Disable the authorizer if it exists */ 10855c45f2eSdanielk1977 char *zNull; /* Text to substitute for an SQL NULL value */ 109cabb0819Sdrh SqlFunc *pFunc; /* List of SQL functions */ 11094eb6a14Sdanielk1977 Tcl_Obj *pUpdateHook; /* Update hook script (if any) */ 11171fd80bfSdanielk1977 Tcl_Obj *pRollbackHook; /* Rollback hook script (if any) */ 112*404ca075Sdanielk1977 Tcl_Obj *pUnlockNotify; /* Unlock notify script (if any) */ 1130202b29eSdanielk1977 SqlCollate *pCollate; /* List of SQL collation functions */ 1146f8a503dSdanielk1977 int rc; /* Return code of most recent sqlite3_exec() */ 1157cedc8d4Sdanielk1977 Tcl_Obj *pCollateNeeded; /* Collation needed script */ 116fb7e7651Sdrh SqlPreparedStmt *stmtList; /* List of prepared statements*/ 117fb7e7651Sdrh SqlPreparedStmt *stmtLast; /* Last statement in the list */ 118fb7e7651Sdrh int maxStmt; /* The next maximum number of stmtList */ 119fb7e7651Sdrh int nStmt; /* Number of statements in stmtList */ 120d0441796Sdanielk1977 IncrblobChannel *pIncrblob;/* Linked list of open incrblob channels */ 121d1d38488Sdrh int nStep, nSort; /* Statistics for most recent operation */ 122cd38d520Sdanielk1977 int nTransaction; /* Number of nested [transaction] methods */ 12398808babSdrh }; 124297ecf14Sdrh 125b4e9af9fSdanielk1977 struct IncrblobChannel { 126d0441796Sdanielk1977 sqlite3_blob *pBlob; /* sqlite3 blob handle */ 127dcbb5d3fSdanielk1977 SqliteDb *pDb; /* Associated database connection */ 128b4e9af9fSdanielk1977 int iSeek; /* Current seek offset */ 129d0441796Sdanielk1977 Tcl_Channel channel; /* Channel identifier */ 130d0441796Sdanielk1977 IncrblobChannel *pNext; /* Linked list of all open incrblob channels */ 131d0441796Sdanielk1977 IncrblobChannel *pPrev; /* Linked list of all open incrblob channels */ 132b4e9af9fSdanielk1977 }; 133b4e9af9fSdanielk1977 134ea678832Sdrh /* 135ea678832Sdrh ** Compute a string length that is limited to what can be stored in 136ea678832Sdrh ** lower 30 bits of a 32-bit signed integer. 137ea678832Sdrh */ 1384f21c4afSdrh static int strlen30(const char *z){ 139ea678832Sdrh const char *z2 = z; 140ea678832Sdrh while( *z2 ){ z2++; } 141ea678832Sdrh return 0x3fffffff & (int)(z2 - z); 142ea678832Sdrh } 143ea678832Sdrh 144ea678832Sdrh 14532a0d8bbSdanielk1977 #ifndef SQLITE_OMIT_INCRBLOB 146b4e9af9fSdanielk1977 /* 147d0441796Sdanielk1977 ** Close all incrblob channels opened using database connection pDb. 148d0441796Sdanielk1977 ** This is called when shutting down the database connection. 149d0441796Sdanielk1977 */ 150d0441796Sdanielk1977 static void closeIncrblobChannels(SqliteDb *pDb){ 151d0441796Sdanielk1977 IncrblobChannel *p; 152d0441796Sdanielk1977 IncrblobChannel *pNext; 153d0441796Sdanielk1977 154d0441796Sdanielk1977 for(p=pDb->pIncrblob; p; p=pNext){ 155d0441796Sdanielk1977 pNext = p->pNext; 156d0441796Sdanielk1977 157d0441796Sdanielk1977 /* Note: Calling unregister here call Tcl_Close on the incrblob channel, 158d0441796Sdanielk1977 ** which deletes the IncrblobChannel structure at *p. So do not 159d0441796Sdanielk1977 ** call Tcl_Free() here. 160d0441796Sdanielk1977 */ 161d0441796Sdanielk1977 Tcl_UnregisterChannel(pDb->interp, p->channel); 162d0441796Sdanielk1977 } 163d0441796Sdanielk1977 } 164d0441796Sdanielk1977 165d0441796Sdanielk1977 /* 166b4e9af9fSdanielk1977 ** Close an incremental blob channel. 167b4e9af9fSdanielk1977 */ 168b4e9af9fSdanielk1977 static int incrblobClose(ClientData instanceData, Tcl_Interp *interp){ 169b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 17092d4d7a9Sdanielk1977 int rc = sqlite3_blob_close(p->pBlob); 17192d4d7a9Sdanielk1977 sqlite3 *db = p->pDb->db; 172d0441796Sdanielk1977 173d0441796Sdanielk1977 /* Remove the channel from the SqliteDb.pIncrblob list. */ 174d0441796Sdanielk1977 if( p->pNext ){ 175d0441796Sdanielk1977 p->pNext->pPrev = p->pPrev; 176d0441796Sdanielk1977 } 177d0441796Sdanielk1977 if( p->pPrev ){ 178d0441796Sdanielk1977 p->pPrev->pNext = p->pNext; 179d0441796Sdanielk1977 } 180d0441796Sdanielk1977 if( p->pDb->pIncrblob==p ){ 181d0441796Sdanielk1977 p->pDb->pIncrblob = p->pNext; 182d0441796Sdanielk1977 } 183d0441796Sdanielk1977 18492d4d7a9Sdanielk1977 /* Free the IncrblobChannel structure */ 185b4e9af9fSdanielk1977 Tcl_Free((char *)p); 18692d4d7a9Sdanielk1977 18792d4d7a9Sdanielk1977 if( rc!=SQLITE_OK ){ 18892d4d7a9Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(db), TCL_VOLATILE); 18992d4d7a9Sdanielk1977 return TCL_ERROR; 19092d4d7a9Sdanielk1977 } 191b4e9af9fSdanielk1977 return TCL_OK; 192b4e9af9fSdanielk1977 } 193b4e9af9fSdanielk1977 194b4e9af9fSdanielk1977 /* 195b4e9af9fSdanielk1977 ** Read data from an incremental blob channel. 196b4e9af9fSdanielk1977 */ 197b4e9af9fSdanielk1977 static int incrblobInput( 198b4e9af9fSdanielk1977 ClientData instanceData, 199b4e9af9fSdanielk1977 char *buf, 200b4e9af9fSdanielk1977 int bufSize, 201b4e9af9fSdanielk1977 int *errorCodePtr 202b4e9af9fSdanielk1977 ){ 203b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 204b4e9af9fSdanielk1977 int nRead = bufSize; /* Number of bytes to read */ 205b4e9af9fSdanielk1977 int nBlob; /* Total size of the blob */ 206b4e9af9fSdanielk1977 int rc; /* sqlite error code */ 207b4e9af9fSdanielk1977 208b4e9af9fSdanielk1977 nBlob = sqlite3_blob_bytes(p->pBlob); 209b4e9af9fSdanielk1977 if( (p->iSeek+nRead)>nBlob ){ 210b4e9af9fSdanielk1977 nRead = nBlob-p->iSeek; 211b4e9af9fSdanielk1977 } 212b4e9af9fSdanielk1977 if( nRead<=0 ){ 213b4e9af9fSdanielk1977 return 0; 214b4e9af9fSdanielk1977 } 215b4e9af9fSdanielk1977 216b4e9af9fSdanielk1977 rc = sqlite3_blob_read(p->pBlob, (void *)buf, nRead, p->iSeek); 217b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 218b4e9af9fSdanielk1977 *errorCodePtr = rc; 219b4e9af9fSdanielk1977 return -1; 220b4e9af9fSdanielk1977 } 221b4e9af9fSdanielk1977 222b4e9af9fSdanielk1977 p->iSeek += nRead; 223b4e9af9fSdanielk1977 return nRead; 224b4e9af9fSdanielk1977 } 225b4e9af9fSdanielk1977 226d0441796Sdanielk1977 /* 227d0441796Sdanielk1977 ** Write data to an incremental blob channel. 228d0441796Sdanielk1977 */ 229b4e9af9fSdanielk1977 static int incrblobOutput( 230b4e9af9fSdanielk1977 ClientData instanceData, 231b4e9af9fSdanielk1977 CONST char *buf, 232b4e9af9fSdanielk1977 int toWrite, 233b4e9af9fSdanielk1977 int *errorCodePtr 234b4e9af9fSdanielk1977 ){ 235b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 236b4e9af9fSdanielk1977 int nWrite = toWrite; /* Number of bytes to write */ 237b4e9af9fSdanielk1977 int nBlob; /* Total size of the blob */ 238b4e9af9fSdanielk1977 int rc; /* sqlite error code */ 239b4e9af9fSdanielk1977 240b4e9af9fSdanielk1977 nBlob = sqlite3_blob_bytes(p->pBlob); 241b4e9af9fSdanielk1977 if( (p->iSeek+nWrite)>nBlob ){ 242b4e9af9fSdanielk1977 *errorCodePtr = EINVAL; 243b4e9af9fSdanielk1977 return -1; 244b4e9af9fSdanielk1977 } 245b4e9af9fSdanielk1977 if( nWrite<=0 ){ 246b4e9af9fSdanielk1977 return 0; 247b4e9af9fSdanielk1977 } 248b4e9af9fSdanielk1977 249b4e9af9fSdanielk1977 rc = sqlite3_blob_write(p->pBlob, (void *)buf, nWrite, p->iSeek); 250b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 251b4e9af9fSdanielk1977 *errorCodePtr = EIO; 252b4e9af9fSdanielk1977 return -1; 253b4e9af9fSdanielk1977 } 254b4e9af9fSdanielk1977 255b4e9af9fSdanielk1977 p->iSeek += nWrite; 256b4e9af9fSdanielk1977 return nWrite; 257b4e9af9fSdanielk1977 } 258b4e9af9fSdanielk1977 259b4e9af9fSdanielk1977 /* 260b4e9af9fSdanielk1977 ** Seek an incremental blob channel. 261b4e9af9fSdanielk1977 */ 262b4e9af9fSdanielk1977 static int incrblobSeek( 263b4e9af9fSdanielk1977 ClientData instanceData, 264b4e9af9fSdanielk1977 long offset, 265b4e9af9fSdanielk1977 int seekMode, 266b4e9af9fSdanielk1977 int *errorCodePtr 267b4e9af9fSdanielk1977 ){ 268b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 269b4e9af9fSdanielk1977 270b4e9af9fSdanielk1977 switch( seekMode ){ 271b4e9af9fSdanielk1977 case SEEK_SET: 272b4e9af9fSdanielk1977 p->iSeek = offset; 273b4e9af9fSdanielk1977 break; 274b4e9af9fSdanielk1977 case SEEK_CUR: 275b4e9af9fSdanielk1977 p->iSeek += offset; 276b4e9af9fSdanielk1977 break; 277b4e9af9fSdanielk1977 case SEEK_END: 278b4e9af9fSdanielk1977 p->iSeek = sqlite3_blob_bytes(p->pBlob) + offset; 279b4e9af9fSdanielk1977 break; 280b4e9af9fSdanielk1977 281b4e9af9fSdanielk1977 default: assert(!"Bad seekMode"); 282b4e9af9fSdanielk1977 } 283b4e9af9fSdanielk1977 284b4e9af9fSdanielk1977 return p->iSeek; 285b4e9af9fSdanielk1977 } 286b4e9af9fSdanielk1977 287b4e9af9fSdanielk1977 288b4e9af9fSdanielk1977 static void incrblobWatch(ClientData instanceData, int mode){ 289b4e9af9fSdanielk1977 /* NO-OP */ 290b4e9af9fSdanielk1977 } 291b4e9af9fSdanielk1977 static int incrblobHandle(ClientData instanceData, int dir, ClientData *hPtr){ 292b4e9af9fSdanielk1977 return TCL_ERROR; 293b4e9af9fSdanielk1977 } 294b4e9af9fSdanielk1977 295b4e9af9fSdanielk1977 static Tcl_ChannelType IncrblobChannelType = { 296b4e9af9fSdanielk1977 "incrblob", /* typeName */ 297b4e9af9fSdanielk1977 TCL_CHANNEL_VERSION_2, /* version */ 298b4e9af9fSdanielk1977 incrblobClose, /* closeProc */ 299b4e9af9fSdanielk1977 incrblobInput, /* inputProc */ 300b4e9af9fSdanielk1977 incrblobOutput, /* outputProc */ 301b4e9af9fSdanielk1977 incrblobSeek, /* seekProc */ 302b4e9af9fSdanielk1977 0, /* setOptionProc */ 303b4e9af9fSdanielk1977 0, /* getOptionProc */ 304b4e9af9fSdanielk1977 incrblobWatch, /* watchProc (this is a no-op) */ 305b4e9af9fSdanielk1977 incrblobHandle, /* getHandleProc (always returns error) */ 306b4e9af9fSdanielk1977 0, /* close2Proc */ 307b4e9af9fSdanielk1977 0, /* blockModeProc */ 308b4e9af9fSdanielk1977 0, /* flushProc */ 309b4e9af9fSdanielk1977 0, /* handlerProc */ 310b4e9af9fSdanielk1977 0, /* wideSeekProc */ 311b4e9af9fSdanielk1977 }; 312b4e9af9fSdanielk1977 313b4e9af9fSdanielk1977 /* 314b4e9af9fSdanielk1977 ** Create a new incrblob channel. 315b4e9af9fSdanielk1977 */ 316b4e9af9fSdanielk1977 static int createIncrblobChannel( 317b4e9af9fSdanielk1977 Tcl_Interp *interp, 318b4e9af9fSdanielk1977 SqliteDb *pDb, 319b4e9af9fSdanielk1977 const char *zDb, 320b4e9af9fSdanielk1977 const char *zTable, 321b4e9af9fSdanielk1977 const char *zColumn, 3228cbadb02Sdanielk1977 sqlite_int64 iRow, 3238cbadb02Sdanielk1977 int isReadonly 324b4e9af9fSdanielk1977 ){ 325b4e9af9fSdanielk1977 IncrblobChannel *p; 3268cbadb02Sdanielk1977 sqlite3 *db = pDb->db; 327b4e9af9fSdanielk1977 sqlite3_blob *pBlob; 328b4e9af9fSdanielk1977 int rc; 3298cbadb02Sdanielk1977 int flags = TCL_READABLE|(isReadonly ? 0 : TCL_WRITABLE); 330b4e9af9fSdanielk1977 331b4e9af9fSdanielk1977 /* This variable is used to name the channels: "incrblob_[incr count]" */ 332b4e9af9fSdanielk1977 static int count = 0; 333b4e9af9fSdanielk1977 char zChannel[64]; 334b4e9af9fSdanielk1977 3358cbadb02Sdanielk1977 rc = sqlite3_blob_open(db, zDb, zTable, zColumn, iRow, !isReadonly, &pBlob); 336b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 337b4e9af9fSdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 338b4e9af9fSdanielk1977 return TCL_ERROR; 339b4e9af9fSdanielk1977 } 340b4e9af9fSdanielk1977 341b4e9af9fSdanielk1977 p = (IncrblobChannel *)Tcl_Alloc(sizeof(IncrblobChannel)); 342b4e9af9fSdanielk1977 p->iSeek = 0; 343b4e9af9fSdanielk1977 p->pBlob = pBlob; 344b4e9af9fSdanielk1977 3455bb3eb9bSdrh sqlite3_snprintf(sizeof(zChannel), zChannel, "incrblob_%d", ++count); 346d0441796Sdanielk1977 p->channel = Tcl_CreateChannel(&IncrblobChannelType, zChannel, p, flags); 347d0441796Sdanielk1977 Tcl_RegisterChannel(interp, p->channel); 348b4e9af9fSdanielk1977 349d0441796Sdanielk1977 /* Link the new channel into the SqliteDb.pIncrblob list. */ 350d0441796Sdanielk1977 p->pNext = pDb->pIncrblob; 351d0441796Sdanielk1977 p->pPrev = 0; 352d0441796Sdanielk1977 if( p->pNext ){ 353d0441796Sdanielk1977 p->pNext->pPrev = p; 354d0441796Sdanielk1977 } 355d0441796Sdanielk1977 pDb->pIncrblob = p; 356d0441796Sdanielk1977 p->pDb = pDb; 357d0441796Sdanielk1977 358d0441796Sdanielk1977 Tcl_SetResult(interp, (char *)Tcl_GetChannelName(p->channel), TCL_VOLATILE); 359b4e9af9fSdanielk1977 return TCL_OK; 360b4e9af9fSdanielk1977 } 36132a0d8bbSdanielk1977 #else /* else clause for "#ifndef SQLITE_OMIT_INCRBLOB" */ 36232a0d8bbSdanielk1977 #define closeIncrblobChannels(pDb) 36332a0d8bbSdanielk1977 #endif 364b4e9af9fSdanielk1977 3656d31316cSdrh /* 366d1e4733dSdrh ** Look at the script prefix in pCmd. We will be executing this script 367d1e4733dSdrh ** after first appending one or more arguments. This routine analyzes 368d1e4733dSdrh ** the script to see if it is safe to use Tcl_EvalObjv() on the script 369d1e4733dSdrh ** rather than the more general Tcl_EvalEx(). Tcl_EvalObjv() is much 370d1e4733dSdrh ** faster. 371d1e4733dSdrh ** 372d1e4733dSdrh ** Scripts that are safe to use with Tcl_EvalObjv() consists of a 373d1e4733dSdrh ** command name followed by zero or more arguments with no [...] or $ 374d1e4733dSdrh ** or {...} or ; to be seen anywhere. Most callback scripts consist 375d1e4733dSdrh ** of just a single procedure name and they meet this requirement. 376d1e4733dSdrh */ 377d1e4733dSdrh static int safeToUseEvalObjv(Tcl_Interp *interp, Tcl_Obj *pCmd){ 378d1e4733dSdrh /* We could try to do something with Tcl_Parse(). But we will instead 379d1e4733dSdrh ** just do a search for forbidden characters. If any of the forbidden 380d1e4733dSdrh ** characters appear in pCmd, we will report the string as unsafe. 381d1e4733dSdrh */ 382d1e4733dSdrh const char *z; 383d1e4733dSdrh int n; 384d1e4733dSdrh z = Tcl_GetStringFromObj(pCmd, &n); 385d1e4733dSdrh while( n-- > 0 ){ 386d1e4733dSdrh int c = *(z++); 387d1e4733dSdrh if( c=='$' || c=='[' || c==';' ) return 0; 388d1e4733dSdrh } 389d1e4733dSdrh return 1; 390d1e4733dSdrh } 391d1e4733dSdrh 392d1e4733dSdrh /* 393d1e4733dSdrh ** Find an SqlFunc structure with the given name. Or create a new 394d1e4733dSdrh ** one if an existing one cannot be found. Return a pointer to the 395d1e4733dSdrh ** structure. 396d1e4733dSdrh */ 397d1e4733dSdrh static SqlFunc *findSqlFunc(SqliteDb *pDb, const char *zName){ 398d1e4733dSdrh SqlFunc *p, *pNew; 399d1e4733dSdrh int i; 4004f21c4afSdrh pNew = (SqlFunc*)Tcl_Alloc( sizeof(*pNew) + strlen30(zName) + 1 ); 401d1e4733dSdrh pNew->zName = (char*)&pNew[1]; 402d1e4733dSdrh for(i=0; zName[i]; i++){ pNew->zName[i] = tolower(zName[i]); } 403d1e4733dSdrh pNew->zName[i] = 0; 404d1e4733dSdrh for(p=pDb->pFunc; p; p=p->pNext){ 405d1e4733dSdrh if( strcmp(p->zName, pNew->zName)==0 ){ 406d1e4733dSdrh Tcl_Free((char*)pNew); 407d1e4733dSdrh return p; 408d1e4733dSdrh } 409d1e4733dSdrh } 410d1e4733dSdrh pNew->interp = pDb->interp; 411d1e4733dSdrh pNew->pScript = 0; 412d1e4733dSdrh pNew->pNext = pDb->pFunc; 413d1e4733dSdrh pDb->pFunc = pNew; 414d1e4733dSdrh return pNew; 415d1e4733dSdrh } 416d1e4733dSdrh 417d1e4733dSdrh /* 418fb7e7651Sdrh ** Finalize and free a list of prepared statements 419fb7e7651Sdrh */ 420fb7e7651Sdrh static void flushStmtCache( SqliteDb *pDb ){ 421fb7e7651Sdrh SqlPreparedStmt *pPreStmt; 422fb7e7651Sdrh 423fb7e7651Sdrh while( pDb->stmtList ){ 424fb7e7651Sdrh sqlite3_finalize( pDb->stmtList->pStmt ); 425fb7e7651Sdrh pPreStmt = pDb->stmtList; 426fb7e7651Sdrh pDb->stmtList = pDb->stmtList->pNext; 427fb7e7651Sdrh Tcl_Free( (char*)pPreStmt ); 428fb7e7651Sdrh } 429fb7e7651Sdrh pDb->nStmt = 0; 430fb7e7651Sdrh pDb->stmtLast = 0; 431fb7e7651Sdrh } 432fb7e7651Sdrh 433fb7e7651Sdrh /* 434895d7472Sdrh ** TCL calls this procedure when an sqlite3 database command is 435895d7472Sdrh ** deleted. 43675897234Sdrh */ 43775897234Sdrh static void DbDeleteCmd(void *db){ 438bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)db; 439fb7e7651Sdrh flushStmtCache(pDb); 440d0441796Sdanielk1977 closeIncrblobChannels(pDb); 4416f8a503dSdanielk1977 sqlite3_close(pDb->db); 442cabb0819Sdrh while( pDb->pFunc ){ 443cabb0819Sdrh SqlFunc *pFunc = pDb->pFunc; 444cabb0819Sdrh pDb->pFunc = pFunc->pNext; 445d1e4733dSdrh Tcl_DecrRefCount(pFunc->pScript); 446cabb0819Sdrh Tcl_Free((char*)pFunc); 447cabb0819Sdrh } 4480202b29eSdanielk1977 while( pDb->pCollate ){ 4490202b29eSdanielk1977 SqlCollate *pCollate = pDb->pCollate; 4500202b29eSdanielk1977 pDb->pCollate = pCollate->pNext; 4510202b29eSdanielk1977 Tcl_Free((char*)pCollate); 4520202b29eSdanielk1977 } 453bec3f402Sdrh if( pDb->zBusy ){ 454bec3f402Sdrh Tcl_Free(pDb->zBusy); 455bec3f402Sdrh } 456b5a20d3cSdrh if( pDb->zTrace ){ 457b5a20d3cSdrh Tcl_Free(pDb->zTrace); 4580d1a643aSdrh } 45919e2d37fSdrh if( pDb->zProfile ){ 46019e2d37fSdrh Tcl_Free(pDb->zProfile); 46119e2d37fSdrh } 462e22a334bSdrh if( pDb->zAuth ){ 463e22a334bSdrh Tcl_Free(pDb->zAuth); 464e22a334bSdrh } 46555c45f2eSdanielk1977 if( pDb->zNull ){ 46655c45f2eSdanielk1977 Tcl_Free(pDb->zNull); 46755c45f2eSdanielk1977 } 46894eb6a14Sdanielk1977 if( pDb->pUpdateHook ){ 46994eb6a14Sdanielk1977 Tcl_DecrRefCount(pDb->pUpdateHook); 47094eb6a14Sdanielk1977 } 47171fd80bfSdanielk1977 if( pDb->pRollbackHook ){ 47271fd80bfSdanielk1977 Tcl_DecrRefCount(pDb->pRollbackHook); 47371fd80bfSdanielk1977 } 47494eb6a14Sdanielk1977 if( pDb->pCollateNeeded ){ 47594eb6a14Sdanielk1977 Tcl_DecrRefCount(pDb->pCollateNeeded); 47694eb6a14Sdanielk1977 } 477bec3f402Sdrh Tcl_Free((char*)pDb); 478bec3f402Sdrh } 479bec3f402Sdrh 480bec3f402Sdrh /* 481bec3f402Sdrh ** This routine is called when a database file is locked while trying 482bec3f402Sdrh ** to execute SQL. 483bec3f402Sdrh */ 4842a764eb0Sdanielk1977 static int DbBusyHandler(void *cd, int nTries){ 485bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)cd; 486bec3f402Sdrh int rc; 487bec3f402Sdrh char zVal[30]; 488bec3f402Sdrh 4895bb3eb9bSdrh sqlite3_snprintf(sizeof(zVal), zVal, "%d", nTries); 490d1e4733dSdrh rc = Tcl_VarEval(pDb->interp, pDb->zBusy, " ", zVal, (char*)0); 491bec3f402Sdrh if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 492bec3f402Sdrh return 0; 493bec3f402Sdrh } 494bec3f402Sdrh return 1; 49575897234Sdrh } 49675897234Sdrh 49726e4a8b1Sdrh #ifndef SQLITE_OMIT_PROGRESS_CALLBACK 49875897234Sdrh /* 499348bb5d6Sdanielk1977 ** This routine is invoked as the 'progress callback' for the database. 500348bb5d6Sdanielk1977 */ 501348bb5d6Sdanielk1977 static int DbProgressHandler(void *cd){ 502348bb5d6Sdanielk1977 SqliteDb *pDb = (SqliteDb*)cd; 503348bb5d6Sdanielk1977 int rc; 504348bb5d6Sdanielk1977 505348bb5d6Sdanielk1977 assert( pDb->zProgress ); 506348bb5d6Sdanielk1977 rc = Tcl_Eval(pDb->interp, pDb->zProgress); 507348bb5d6Sdanielk1977 if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 508348bb5d6Sdanielk1977 return 1; 509348bb5d6Sdanielk1977 } 510348bb5d6Sdanielk1977 return 0; 511348bb5d6Sdanielk1977 } 51226e4a8b1Sdrh #endif 513348bb5d6Sdanielk1977 514d1167393Sdrh #ifndef SQLITE_OMIT_TRACE 515348bb5d6Sdanielk1977 /* 516b5a20d3cSdrh ** This routine is called by the SQLite trace handler whenever a new 517b5a20d3cSdrh ** block of SQL is executed. The TCL script in pDb->zTrace is executed. 5180d1a643aSdrh */ 519b5a20d3cSdrh static void DbTraceHandler(void *cd, const char *zSql){ 5200d1a643aSdrh SqliteDb *pDb = (SqliteDb*)cd; 521b5a20d3cSdrh Tcl_DString str; 5220d1a643aSdrh 523b5a20d3cSdrh Tcl_DStringInit(&str); 524b5a20d3cSdrh Tcl_DStringAppend(&str, pDb->zTrace, -1); 525b5a20d3cSdrh Tcl_DStringAppendElement(&str, zSql); 526b5a20d3cSdrh Tcl_Eval(pDb->interp, Tcl_DStringValue(&str)); 527b5a20d3cSdrh Tcl_DStringFree(&str); 528b5a20d3cSdrh Tcl_ResetResult(pDb->interp); 5290d1a643aSdrh } 530d1167393Sdrh #endif 5310d1a643aSdrh 532d1167393Sdrh #ifndef SQLITE_OMIT_TRACE 5330d1a643aSdrh /* 53419e2d37fSdrh ** This routine is called by the SQLite profile handler after a statement 53519e2d37fSdrh ** SQL has executed. The TCL script in pDb->zProfile is evaluated. 53619e2d37fSdrh */ 53719e2d37fSdrh static void DbProfileHandler(void *cd, const char *zSql, sqlite_uint64 tm){ 53819e2d37fSdrh SqliteDb *pDb = (SqliteDb*)cd; 53919e2d37fSdrh Tcl_DString str; 54019e2d37fSdrh char zTm[100]; 54119e2d37fSdrh 54219e2d37fSdrh sqlite3_snprintf(sizeof(zTm)-1, zTm, "%lld", tm); 54319e2d37fSdrh Tcl_DStringInit(&str); 54419e2d37fSdrh Tcl_DStringAppend(&str, pDb->zProfile, -1); 54519e2d37fSdrh Tcl_DStringAppendElement(&str, zSql); 54619e2d37fSdrh Tcl_DStringAppendElement(&str, zTm); 54719e2d37fSdrh Tcl_Eval(pDb->interp, Tcl_DStringValue(&str)); 54819e2d37fSdrh Tcl_DStringFree(&str); 54919e2d37fSdrh Tcl_ResetResult(pDb->interp); 55019e2d37fSdrh } 551d1167393Sdrh #endif 55219e2d37fSdrh 55319e2d37fSdrh /* 554aa940eacSdrh ** This routine is called when a transaction is committed. The 555aa940eacSdrh ** TCL script in pDb->zCommit is executed. If it returns non-zero or 556aa940eacSdrh ** if it throws an exception, the transaction is rolled back instead 557aa940eacSdrh ** of being committed. 558aa940eacSdrh */ 559aa940eacSdrh static int DbCommitHandler(void *cd){ 560aa940eacSdrh SqliteDb *pDb = (SqliteDb*)cd; 561aa940eacSdrh int rc; 562aa940eacSdrh 563aa940eacSdrh rc = Tcl_Eval(pDb->interp, pDb->zCommit); 564aa940eacSdrh if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 565aa940eacSdrh return 1; 566aa940eacSdrh } 567aa940eacSdrh return 0; 568aa940eacSdrh } 569aa940eacSdrh 57071fd80bfSdanielk1977 static void DbRollbackHandler(void *clientData){ 57171fd80bfSdanielk1977 SqliteDb *pDb = (SqliteDb*)clientData; 57271fd80bfSdanielk1977 assert(pDb->pRollbackHook); 57371fd80bfSdanielk1977 if( TCL_OK!=Tcl_EvalObjEx(pDb->interp, pDb->pRollbackHook, 0) ){ 57471fd80bfSdanielk1977 Tcl_BackgroundError(pDb->interp); 57571fd80bfSdanielk1977 } 57671fd80bfSdanielk1977 } 57771fd80bfSdanielk1977 578*404ca075Sdanielk1977 #ifdef SQLITE_TEST 579*404ca075Sdanielk1977 static void setTestUnlockNotifyVars(Tcl_Interp *interp, int iArg, int nArg){ 580*404ca075Sdanielk1977 char zBuf[64]; 581*404ca075Sdanielk1977 sprintf(zBuf, "%d", iArg); 582*404ca075Sdanielk1977 Tcl_SetVar(interp, "sqlite_unlock_notify_arg", zBuf, TCL_GLOBAL_ONLY); 583*404ca075Sdanielk1977 sprintf(zBuf, "%d", nArg); 584*404ca075Sdanielk1977 Tcl_SetVar(interp, "sqlite_unlock_notify_argcount", zBuf, TCL_GLOBAL_ONLY); 585*404ca075Sdanielk1977 } 586*404ca075Sdanielk1977 #else 587*404ca075Sdanielk1977 #define setTestUnlockNotifyVars(x,y,z) 588*404ca075Sdanielk1977 #endif 589*404ca075Sdanielk1977 590*404ca075Sdanielk1977 static void DbUnlockNotify(void **apArg, int nArg){ 591*404ca075Sdanielk1977 int i; 592*404ca075Sdanielk1977 for(i=0; i<nArg; i++){ 593*404ca075Sdanielk1977 const int flags = (TCL_EVAL_GLOBAL|TCL_EVAL_DIRECT); 594*404ca075Sdanielk1977 SqliteDb *pDb = (SqliteDb *)apArg[i]; 595*404ca075Sdanielk1977 setTestUnlockNotifyVars(pDb->interp, i, nArg); 596*404ca075Sdanielk1977 assert( pDb->pUnlockNotify); 597*404ca075Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pDb->pUnlockNotify, flags); 598*404ca075Sdanielk1977 Tcl_DecrRefCount(pDb->pUnlockNotify); 599*404ca075Sdanielk1977 pDb->pUnlockNotify = 0; 600*404ca075Sdanielk1977 } 601*404ca075Sdanielk1977 } 602*404ca075Sdanielk1977 60394eb6a14Sdanielk1977 static void DbUpdateHandler( 60494eb6a14Sdanielk1977 void *p, 60594eb6a14Sdanielk1977 int op, 60694eb6a14Sdanielk1977 const char *zDb, 60794eb6a14Sdanielk1977 const char *zTbl, 60894eb6a14Sdanielk1977 sqlite_int64 rowid 60994eb6a14Sdanielk1977 ){ 61094eb6a14Sdanielk1977 SqliteDb *pDb = (SqliteDb *)p; 61194eb6a14Sdanielk1977 Tcl_Obj *pCmd; 61294eb6a14Sdanielk1977 61394eb6a14Sdanielk1977 assert( pDb->pUpdateHook ); 61494eb6a14Sdanielk1977 assert( op==SQLITE_INSERT || op==SQLITE_UPDATE || op==SQLITE_DELETE ); 61594eb6a14Sdanielk1977 61694eb6a14Sdanielk1977 pCmd = Tcl_DuplicateObj(pDb->pUpdateHook); 61794eb6a14Sdanielk1977 Tcl_IncrRefCount(pCmd); 61894eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj( 61994eb6a14Sdanielk1977 ( (op==SQLITE_INSERT)?"INSERT":(op==SQLITE_UPDATE)?"UPDATE":"DELETE"), -1)); 62094eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj(zDb, -1)); 62194eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj(zTbl, -1)); 62294eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewWideIntObj(rowid)); 62394eb6a14Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pCmd, TCL_EVAL_DIRECT); 62494eb6a14Sdanielk1977 } 62594eb6a14Sdanielk1977 6267cedc8d4Sdanielk1977 static void tclCollateNeeded( 6277cedc8d4Sdanielk1977 void *pCtx, 6289bb575fdSdrh sqlite3 *db, 6297cedc8d4Sdanielk1977 int enc, 6307cedc8d4Sdanielk1977 const char *zName 6317cedc8d4Sdanielk1977 ){ 6327cedc8d4Sdanielk1977 SqliteDb *pDb = (SqliteDb *)pCtx; 6337cedc8d4Sdanielk1977 Tcl_Obj *pScript = Tcl_DuplicateObj(pDb->pCollateNeeded); 6347cedc8d4Sdanielk1977 Tcl_IncrRefCount(pScript); 6357cedc8d4Sdanielk1977 Tcl_ListObjAppendElement(0, pScript, Tcl_NewStringObj(zName, -1)); 6367cedc8d4Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pScript, 0); 6377cedc8d4Sdanielk1977 Tcl_DecrRefCount(pScript); 6387cedc8d4Sdanielk1977 } 6397cedc8d4Sdanielk1977 640aa940eacSdrh /* 6410202b29eSdanielk1977 ** This routine is called to evaluate an SQL collation function implemented 6420202b29eSdanielk1977 ** using TCL script. 6430202b29eSdanielk1977 */ 6440202b29eSdanielk1977 static int tclSqlCollate( 6450202b29eSdanielk1977 void *pCtx, 6460202b29eSdanielk1977 int nA, 6470202b29eSdanielk1977 const void *zA, 6480202b29eSdanielk1977 int nB, 6490202b29eSdanielk1977 const void *zB 6500202b29eSdanielk1977 ){ 6510202b29eSdanielk1977 SqlCollate *p = (SqlCollate *)pCtx; 6520202b29eSdanielk1977 Tcl_Obj *pCmd; 6530202b29eSdanielk1977 6540202b29eSdanielk1977 pCmd = Tcl_NewStringObj(p->zScript, -1); 6550202b29eSdanielk1977 Tcl_IncrRefCount(pCmd); 6560202b29eSdanielk1977 Tcl_ListObjAppendElement(p->interp, pCmd, Tcl_NewStringObj(zA, nA)); 6570202b29eSdanielk1977 Tcl_ListObjAppendElement(p->interp, pCmd, Tcl_NewStringObj(zB, nB)); 658d1e4733dSdrh Tcl_EvalObjEx(p->interp, pCmd, TCL_EVAL_DIRECT); 6590202b29eSdanielk1977 Tcl_DecrRefCount(pCmd); 6600202b29eSdanielk1977 return (atoi(Tcl_GetStringResult(p->interp))); 6610202b29eSdanielk1977 } 6620202b29eSdanielk1977 6630202b29eSdanielk1977 /* 664cabb0819Sdrh ** This routine is called to evaluate an SQL function implemented 665cabb0819Sdrh ** using TCL script. 666cabb0819Sdrh */ 6670ae8b831Sdanielk1977 static void tclSqlFunc(sqlite3_context *context, int argc, sqlite3_value**argv){ 6686f8a503dSdanielk1977 SqlFunc *p = sqlite3_user_data(context); 669d1e4733dSdrh Tcl_Obj *pCmd; 670cabb0819Sdrh int i; 671cabb0819Sdrh int rc; 672cabb0819Sdrh 673d1e4733dSdrh if( argc==0 ){ 674d1e4733dSdrh /* If there are no arguments to the function, call Tcl_EvalObjEx on the 675d1e4733dSdrh ** script object directly. This allows the TCL compiler to generate 676d1e4733dSdrh ** bytecode for the command on the first invocation and thus make 677d1e4733dSdrh ** subsequent invocations much faster. */ 678d1e4733dSdrh pCmd = p->pScript; 679d1e4733dSdrh Tcl_IncrRefCount(pCmd); 680d1e4733dSdrh rc = Tcl_EvalObjEx(p->interp, pCmd, 0); 681d1e4733dSdrh Tcl_DecrRefCount(pCmd); 68251ad0ecdSdanielk1977 }else{ 683d1e4733dSdrh /* If there are arguments to the function, make a shallow copy of the 684d1e4733dSdrh ** script object, lappend the arguments, then evaluate the copy. 685d1e4733dSdrh ** 686d1e4733dSdrh ** By "shallow" copy, we mean a only the outer list Tcl_Obj is duplicated. 687d1e4733dSdrh ** The new Tcl_Obj contains pointers to the original list elements. 688d1e4733dSdrh ** That way, when Tcl_EvalObjv() is run and shimmers the first element 689d1e4733dSdrh ** of the list to tclCmdNameType, that alternate representation will 690d1e4733dSdrh ** be preserved and reused on the next invocation. 691d1e4733dSdrh */ 692d1e4733dSdrh Tcl_Obj **aArg; 693d1e4733dSdrh int nArg; 694d1e4733dSdrh if( Tcl_ListObjGetElements(p->interp, p->pScript, &nArg, &aArg) ){ 695d1e4733dSdrh sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 696d1e4733dSdrh return; 697d1e4733dSdrh } 698d1e4733dSdrh pCmd = Tcl_NewListObj(nArg, aArg); 699d1e4733dSdrh Tcl_IncrRefCount(pCmd); 700d1e4733dSdrh for(i=0; i<argc; i++){ 701d1e4733dSdrh sqlite3_value *pIn = argv[i]; 702d1e4733dSdrh Tcl_Obj *pVal; 703d1e4733dSdrh 704d1e4733dSdrh /* Set pVal to contain the i'th column of this row. */ 705d1e4733dSdrh switch( sqlite3_value_type(pIn) ){ 706d1e4733dSdrh case SQLITE_BLOB: { 707d1e4733dSdrh int bytes = sqlite3_value_bytes(pIn); 708d1e4733dSdrh pVal = Tcl_NewByteArrayObj(sqlite3_value_blob(pIn), bytes); 709d1e4733dSdrh break; 710d1e4733dSdrh } 711d1e4733dSdrh case SQLITE_INTEGER: { 712d1e4733dSdrh sqlite_int64 v = sqlite3_value_int64(pIn); 713d1e4733dSdrh if( v>=-2147483647 && v<=2147483647 ){ 714d1e4733dSdrh pVal = Tcl_NewIntObj(v); 715d1e4733dSdrh }else{ 716d1e4733dSdrh pVal = Tcl_NewWideIntObj(v); 717d1e4733dSdrh } 718d1e4733dSdrh break; 719d1e4733dSdrh } 720d1e4733dSdrh case SQLITE_FLOAT: { 721d1e4733dSdrh double r = sqlite3_value_double(pIn); 722d1e4733dSdrh pVal = Tcl_NewDoubleObj(r); 723d1e4733dSdrh break; 724d1e4733dSdrh } 725d1e4733dSdrh case SQLITE_NULL: { 726d1e4733dSdrh pVal = Tcl_NewStringObj("", 0); 727d1e4733dSdrh break; 728d1e4733dSdrh } 729d1e4733dSdrh default: { 730d1e4733dSdrh int bytes = sqlite3_value_bytes(pIn); 73100fd957bSdanielk1977 pVal = Tcl_NewStringObj((char *)sqlite3_value_text(pIn), bytes); 732d1e4733dSdrh break; 73351ad0ecdSdanielk1977 } 734cabb0819Sdrh } 735d1e4733dSdrh rc = Tcl_ListObjAppendElement(p->interp, pCmd, pVal); 736d1e4733dSdrh if( rc ){ 737d1e4733dSdrh Tcl_DecrRefCount(pCmd); 738d1e4733dSdrh sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 739d1e4733dSdrh return; 740d1e4733dSdrh } 741d1e4733dSdrh } 742d1e4733dSdrh if( !p->useEvalObjv ){ 743d1e4733dSdrh /* Tcl_EvalObjEx() will automatically call Tcl_EvalObjv() if pCmd 744d1e4733dSdrh ** is a list without a string representation. To prevent this from 745d1e4733dSdrh ** happening, make sure pCmd has a valid string representation */ 746d1e4733dSdrh Tcl_GetString(pCmd); 747d1e4733dSdrh } 748d1e4733dSdrh rc = Tcl_EvalObjEx(p->interp, pCmd, TCL_EVAL_DIRECT); 749d1e4733dSdrh Tcl_DecrRefCount(pCmd); 750d1e4733dSdrh } 751562e8d3cSdanielk1977 752c7f269d5Sdrh if( rc && rc!=TCL_RETURN ){ 7537e18c259Sdanielk1977 sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 754cabb0819Sdrh }else{ 755c7f269d5Sdrh Tcl_Obj *pVar = Tcl_GetObjResult(p->interp); 756c7f269d5Sdrh int n; 757c7f269d5Sdrh u8 *data; 758c7f269d5Sdrh char *zType = pVar->typePtr ? pVar->typePtr->name : ""; 759c7f269d5Sdrh char c = zType[0]; 760df0bddaeSdrh if( c=='b' && strcmp(zType,"bytearray")==0 && pVar->bytes==0 ){ 761d1e4733dSdrh /* Only return a BLOB type if the Tcl variable is a bytearray and 762df0bddaeSdrh ** has no string representation. */ 763c7f269d5Sdrh data = Tcl_GetByteArrayFromObj(pVar, &n); 764c7f269d5Sdrh sqlite3_result_blob(context, data, n, SQLITE_TRANSIENT); 765985e0c63Sdrh }else if( c=='b' && strcmp(zType,"boolean")==0 ){ 766c7f269d5Sdrh Tcl_GetIntFromObj(0, pVar, &n); 767c7f269d5Sdrh sqlite3_result_int(context, n); 768c7f269d5Sdrh }else if( c=='d' && strcmp(zType,"double")==0 ){ 769c7f269d5Sdrh double r; 770c7f269d5Sdrh Tcl_GetDoubleFromObj(0, pVar, &r); 771c7f269d5Sdrh sqlite3_result_double(context, r); 772985e0c63Sdrh }else if( (c=='w' && strcmp(zType,"wideInt")==0) || 773985e0c63Sdrh (c=='i' && strcmp(zType,"int")==0) ){ 774df0bddaeSdrh Tcl_WideInt v; 775df0bddaeSdrh Tcl_GetWideIntFromObj(0, pVar, &v); 776df0bddaeSdrh sqlite3_result_int64(context, v); 777c7f269d5Sdrh }else{ 77800fd957bSdanielk1977 data = (unsigned char *)Tcl_GetStringFromObj(pVar, &n); 77900fd957bSdanielk1977 sqlite3_result_text(context, (char *)data, n, SQLITE_TRANSIENT); 780c7f269d5Sdrh } 781cabb0819Sdrh } 782cabb0819Sdrh } 783895d7472Sdrh 784e22a334bSdrh #ifndef SQLITE_OMIT_AUTHORIZATION 785e22a334bSdrh /* 786e22a334bSdrh ** This is the authentication function. It appends the authentication 787e22a334bSdrh ** type code and the two arguments to zCmd[] then invokes the result 788e22a334bSdrh ** on the interpreter. The reply is examined to determine if the 789e22a334bSdrh ** authentication fails or succeeds. 790e22a334bSdrh */ 791e22a334bSdrh static int auth_callback( 792e22a334bSdrh void *pArg, 793e22a334bSdrh int code, 794e22a334bSdrh const char *zArg1, 795e22a334bSdrh const char *zArg2, 796e22a334bSdrh const char *zArg3, 797e22a334bSdrh const char *zArg4 798e22a334bSdrh ){ 799e22a334bSdrh char *zCode; 800e22a334bSdrh Tcl_DString str; 801e22a334bSdrh int rc; 802e22a334bSdrh const char *zReply; 803e22a334bSdrh SqliteDb *pDb = (SqliteDb*)pArg; 8041f1549f8Sdrh if( pDb->disableAuth ) return SQLITE_OK; 805e22a334bSdrh 806e22a334bSdrh switch( code ){ 807e22a334bSdrh case SQLITE_COPY : zCode="SQLITE_COPY"; break; 808e22a334bSdrh case SQLITE_CREATE_INDEX : zCode="SQLITE_CREATE_INDEX"; break; 809e22a334bSdrh case SQLITE_CREATE_TABLE : zCode="SQLITE_CREATE_TABLE"; break; 810e22a334bSdrh case SQLITE_CREATE_TEMP_INDEX : zCode="SQLITE_CREATE_TEMP_INDEX"; break; 811e22a334bSdrh case SQLITE_CREATE_TEMP_TABLE : zCode="SQLITE_CREATE_TEMP_TABLE"; break; 812e22a334bSdrh case SQLITE_CREATE_TEMP_TRIGGER: zCode="SQLITE_CREATE_TEMP_TRIGGER"; break; 813e22a334bSdrh case SQLITE_CREATE_TEMP_VIEW : zCode="SQLITE_CREATE_TEMP_VIEW"; break; 814e22a334bSdrh case SQLITE_CREATE_TRIGGER : zCode="SQLITE_CREATE_TRIGGER"; break; 815e22a334bSdrh case SQLITE_CREATE_VIEW : zCode="SQLITE_CREATE_VIEW"; break; 816e22a334bSdrh case SQLITE_DELETE : zCode="SQLITE_DELETE"; break; 817e22a334bSdrh case SQLITE_DROP_INDEX : zCode="SQLITE_DROP_INDEX"; break; 818e22a334bSdrh case SQLITE_DROP_TABLE : zCode="SQLITE_DROP_TABLE"; break; 819e22a334bSdrh case SQLITE_DROP_TEMP_INDEX : zCode="SQLITE_DROP_TEMP_INDEX"; break; 820e22a334bSdrh case SQLITE_DROP_TEMP_TABLE : zCode="SQLITE_DROP_TEMP_TABLE"; break; 821e22a334bSdrh case SQLITE_DROP_TEMP_TRIGGER : zCode="SQLITE_DROP_TEMP_TRIGGER"; break; 822e22a334bSdrh case SQLITE_DROP_TEMP_VIEW : zCode="SQLITE_DROP_TEMP_VIEW"; break; 823e22a334bSdrh case SQLITE_DROP_TRIGGER : zCode="SQLITE_DROP_TRIGGER"; break; 824e22a334bSdrh case SQLITE_DROP_VIEW : zCode="SQLITE_DROP_VIEW"; break; 825e22a334bSdrh case SQLITE_INSERT : zCode="SQLITE_INSERT"; break; 826e22a334bSdrh case SQLITE_PRAGMA : zCode="SQLITE_PRAGMA"; break; 827e22a334bSdrh case SQLITE_READ : zCode="SQLITE_READ"; break; 828e22a334bSdrh case SQLITE_SELECT : zCode="SQLITE_SELECT"; break; 829e22a334bSdrh case SQLITE_TRANSACTION : zCode="SQLITE_TRANSACTION"; break; 830e22a334bSdrh case SQLITE_UPDATE : zCode="SQLITE_UPDATE"; break; 83181e293b4Sdrh case SQLITE_ATTACH : zCode="SQLITE_ATTACH"; break; 83281e293b4Sdrh case SQLITE_DETACH : zCode="SQLITE_DETACH"; break; 8331c8c23ccSdanielk1977 case SQLITE_ALTER_TABLE : zCode="SQLITE_ALTER_TABLE"; break; 8341d54df88Sdanielk1977 case SQLITE_REINDEX : zCode="SQLITE_REINDEX"; break; 835e6e04969Sdrh case SQLITE_ANALYZE : zCode="SQLITE_ANALYZE"; break; 836f1a381e7Sdanielk1977 case SQLITE_CREATE_VTABLE : zCode="SQLITE_CREATE_VTABLE"; break; 837f1a381e7Sdanielk1977 case SQLITE_DROP_VTABLE : zCode="SQLITE_DROP_VTABLE"; break; 8385169bbc6Sdrh case SQLITE_FUNCTION : zCode="SQLITE_FUNCTION"; break; 839ab9b703fSdanielk1977 case SQLITE_SAVEPOINT : zCode="SQLITE_SAVEPOINT"; break; 840e22a334bSdrh default : zCode="????"; break; 841e22a334bSdrh } 842e22a334bSdrh Tcl_DStringInit(&str); 843e22a334bSdrh Tcl_DStringAppend(&str, pDb->zAuth, -1); 844e22a334bSdrh Tcl_DStringAppendElement(&str, zCode); 845e22a334bSdrh Tcl_DStringAppendElement(&str, zArg1 ? zArg1 : ""); 846e22a334bSdrh Tcl_DStringAppendElement(&str, zArg2 ? zArg2 : ""); 847e22a334bSdrh Tcl_DStringAppendElement(&str, zArg3 ? zArg3 : ""); 848e22a334bSdrh Tcl_DStringAppendElement(&str, zArg4 ? zArg4 : ""); 849e22a334bSdrh rc = Tcl_GlobalEval(pDb->interp, Tcl_DStringValue(&str)); 850e22a334bSdrh Tcl_DStringFree(&str); 851e22a334bSdrh zReply = Tcl_GetStringResult(pDb->interp); 852e22a334bSdrh if( strcmp(zReply,"SQLITE_OK")==0 ){ 853e22a334bSdrh rc = SQLITE_OK; 854e22a334bSdrh }else if( strcmp(zReply,"SQLITE_DENY")==0 ){ 855e22a334bSdrh rc = SQLITE_DENY; 856e22a334bSdrh }else if( strcmp(zReply,"SQLITE_IGNORE")==0 ){ 857e22a334bSdrh rc = SQLITE_IGNORE; 858e22a334bSdrh }else{ 859e22a334bSdrh rc = 999; 860e22a334bSdrh } 861e22a334bSdrh return rc; 862e22a334bSdrh } 863e22a334bSdrh #endif /* SQLITE_OMIT_AUTHORIZATION */ 864cabb0819Sdrh 865cabb0819Sdrh /* 866ef2cb63eSdanielk1977 ** zText is a pointer to text obtained via an sqlite3_result_text() 867ef2cb63eSdanielk1977 ** or similar interface. This routine returns a Tcl string object, 868ef2cb63eSdanielk1977 ** reference count set to 0, containing the text. If a translation 869ef2cb63eSdanielk1977 ** between iso8859 and UTF-8 is required, it is preformed. 870ef2cb63eSdanielk1977 */ 871ef2cb63eSdanielk1977 static Tcl_Obj *dbTextToObj(char const *zText){ 872ef2cb63eSdanielk1977 Tcl_Obj *pVal; 873ef2cb63eSdanielk1977 #ifdef UTF_TRANSLATION_NEEDED 874ef2cb63eSdanielk1977 Tcl_DString dCol; 875ef2cb63eSdanielk1977 Tcl_DStringInit(&dCol); 876ef2cb63eSdanielk1977 Tcl_ExternalToUtfDString(NULL, zText, -1, &dCol); 877ef2cb63eSdanielk1977 pVal = Tcl_NewStringObj(Tcl_DStringValue(&dCol), -1); 878ef2cb63eSdanielk1977 Tcl_DStringFree(&dCol); 879ef2cb63eSdanielk1977 #else 880ef2cb63eSdanielk1977 pVal = Tcl_NewStringObj(zText, -1); 881ef2cb63eSdanielk1977 #endif 882ef2cb63eSdanielk1977 return pVal; 883ef2cb63eSdanielk1977 } 884ef2cb63eSdanielk1977 885ef2cb63eSdanielk1977 /* 8861067fe11Stpoindex ** This routine reads a line of text from FILE in, stores 8871067fe11Stpoindex ** the text in memory obtained from malloc() and returns a pointer 8881067fe11Stpoindex ** to the text. NULL is returned at end of file, or if malloc() 8891067fe11Stpoindex ** fails. 8901067fe11Stpoindex ** 8911067fe11Stpoindex ** The interface is like "readline" but no command-line editing 8921067fe11Stpoindex ** is done. 8931067fe11Stpoindex ** 8941067fe11Stpoindex ** copied from shell.c from '.import' command 8951067fe11Stpoindex */ 8961067fe11Stpoindex static char *local_getline(char *zPrompt, FILE *in){ 8971067fe11Stpoindex char *zLine; 8981067fe11Stpoindex int nLine; 8991067fe11Stpoindex int n; 9001067fe11Stpoindex int eol; 9011067fe11Stpoindex 9021067fe11Stpoindex nLine = 100; 9031067fe11Stpoindex zLine = malloc( nLine ); 9041067fe11Stpoindex if( zLine==0 ) return 0; 9051067fe11Stpoindex n = 0; 9061067fe11Stpoindex eol = 0; 9071067fe11Stpoindex while( !eol ){ 9081067fe11Stpoindex if( n+100>nLine ){ 9091067fe11Stpoindex nLine = nLine*2 + 100; 9101067fe11Stpoindex zLine = realloc(zLine, nLine); 9111067fe11Stpoindex if( zLine==0 ) return 0; 9121067fe11Stpoindex } 9131067fe11Stpoindex if( fgets(&zLine[n], nLine - n, in)==0 ){ 9141067fe11Stpoindex if( n==0 ){ 9151067fe11Stpoindex free(zLine); 9161067fe11Stpoindex return 0; 9171067fe11Stpoindex } 9181067fe11Stpoindex zLine[n] = 0; 9191067fe11Stpoindex eol = 1; 9201067fe11Stpoindex break; 9211067fe11Stpoindex } 9221067fe11Stpoindex while( zLine[n] ){ n++; } 9231067fe11Stpoindex if( n>0 && zLine[n-1]=='\n' ){ 9241067fe11Stpoindex n--; 9251067fe11Stpoindex zLine[n] = 0; 9261067fe11Stpoindex eol = 1; 9271067fe11Stpoindex } 9281067fe11Stpoindex } 9291067fe11Stpoindex zLine = realloc( zLine, n+1 ); 9301067fe11Stpoindex return zLine; 9311067fe11Stpoindex } 9321067fe11Stpoindex 9338e556520Sdanielk1977 9348e556520Sdanielk1977 /* 9358e556520Sdanielk1977 ** Figure out the column names for the data returned by the statement 9368e556520Sdanielk1977 ** passed as the second argument. 9378e556520Sdanielk1977 ** 9388e556520Sdanielk1977 ** If parameter papColName is not NULL, then *papColName is set to point 9398e556520Sdanielk1977 ** at an array allocated using Tcl_Alloc(). It is the callers responsibility 9408e556520Sdanielk1977 ** to free this array using Tcl_Free(), and to decrement the reference 9418e556520Sdanielk1977 ** count of each Tcl_Obj* member of the array. 9428e556520Sdanielk1977 ** 9438e556520Sdanielk1977 ** The return value of this function is the number of columns of data 9448e556520Sdanielk1977 ** returned by pStmt (and hence the size of the *papColName array). 9458e556520Sdanielk1977 ** 9468e556520Sdanielk1977 ** If pArray is not NULL, then it contains the name of a Tcl array 9478e556520Sdanielk1977 ** variable. The "*" member of this array is set to a list containing 9488e556520Sdanielk1977 ** the names of the columns returned by the statement, in order from 9498e556520Sdanielk1977 ** left to right. e.g. if the names of the returned columns are a, b and 9508e556520Sdanielk1977 ** c, it does the equivalent of the tcl command: 9518e556520Sdanielk1977 ** 9528e556520Sdanielk1977 ** set ${pArray}(*) {a b c} 9538e556520Sdanielk1977 */ 9548e556520Sdanielk1977 static int 9558e556520Sdanielk1977 computeColumnNames( 9568e556520Sdanielk1977 Tcl_Interp *interp, 9578e556520Sdanielk1977 sqlite3_stmt *pStmt, /* SQL statement */ 9588e556520Sdanielk1977 Tcl_Obj ***papColName, /* OUT: Array of column names */ 9598e556520Sdanielk1977 Tcl_Obj *pArray /* Name of array variable (may be null) */ 9608e556520Sdanielk1977 ){ 9618e556520Sdanielk1977 int nCol; 9628e556520Sdanielk1977 9638e556520Sdanielk1977 /* Compute column names */ 9648e556520Sdanielk1977 nCol = sqlite3_column_count(pStmt); 9658e556520Sdanielk1977 if( papColName ){ 9668e556520Sdanielk1977 int i; 9678e556520Sdanielk1977 Tcl_Obj **apColName = (Tcl_Obj**)Tcl_Alloc( sizeof(Tcl_Obj*)*nCol ); 9688e556520Sdanielk1977 for(i=0; i<nCol; i++){ 9698e556520Sdanielk1977 apColName[i] = dbTextToObj(sqlite3_column_name(pStmt,i)); 9708e556520Sdanielk1977 Tcl_IncrRefCount(apColName[i]); 9718e556520Sdanielk1977 } 9728e556520Sdanielk1977 9738e556520Sdanielk1977 /* If results are being stored in an array variable, then create 9748e556520Sdanielk1977 ** the array(*) entry for that array 9758e556520Sdanielk1977 */ 9768e556520Sdanielk1977 if( pArray ){ 9778e556520Sdanielk1977 Tcl_Obj *pColList = Tcl_NewObj(); 9788e556520Sdanielk1977 Tcl_Obj *pStar = Tcl_NewStringObj("*", -1); 9798e556520Sdanielk1977 Tcl_IncrRefCount(pColList); 9808e556520Sdanielk1977 for(i=0; i<nCol; i++){ 9818e556520Sdanielk1977 Tcl_ListObjAppendElement(interp, pColList, apColName[i]); 9828e556520Sdanielk1977 } 9838e556520Sdanielk1977 Tcl_IncrRefCount(pStar); 9848e556520Sdanielk1977 Tcl_ObjSetVar2(interp, pArray, pStar, pColList,0); 9858e556520Sdanielk1977 Tcl_DecrRefCount(pColList); 9868e556520Sdanielk1977 Tcl_DecrRefCount(pStar); 9878e556520Sdanielk1977 } 9888e556520Sdanielk1977 *papColName = apColName; 9898e556520Sdanielk1977 } 9908e556520Sdanielk1977 9918e556520Sdanielk1977 return nCol; 9928e556520Sdanielk1977 } 9938e556520Sdanielk1977 9941067fe11Stpoindex /* 99575897234Sdrh ** The "sqlite" command below creates a new Tcl command for each 99675897234Sdrh ** connection it opens to an SQLite database. This routine is invoked 99775897234Sdrh ** whenever one of those connection-specific commands is executed 99875897234Sdrh ** in Tcl. For example, if you run Tcl code like this: 99975897234Sdrh ** 10009bb575fdSdrh ** sqlite3 db1 "my_database" 100175897234Sdrh ** db1 close 100275897234Sdrh ** 100375897234Sdrh ** The first command opens a connection to the "my_database" database 100475897234Sdrh ** and calls that connection "db1". The second command causes this 100575897234Sdrh ** subroutine to be invoked. 100675897234Sdrh */ 10076d31316cSdrh static int DbObjCmd(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){ 1008bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)cd; 10096d31316cSdrh int choice; 101022fbcb8dSdrh int rc = TCL_OK; 10110de8c112Sdrh static const char *DB_strs[] = { 1012dc2c4915Sdrh "authorizer", "backup", "busy", 1013dc2c4915Sdrh "cache", "changes", "close", 1014dc2c4915Sdrh "collate", "collation_needed", "commit_hook", 1015dc2c4915Sdrh "complete", "copy", "enable_load_extension", 1016dc2c4915Sdrh "errorcode", "eval", "exists", 1017dc2c4915Sdrh "function", "incrblob", "interrupt", 1018dc2c4915Sdrh "last_insert_rowid", "nullvalue", "onecolumn", 1019dc2c4915Sdrh "profile", "progress", "rekey", 1020dc2c4915Sdrh "restore", "rollback_hook", "status", 1021dc2c4915Sdrh "timeout", "total_changes", "trace", 1022*404ca075Sdanielk1977 "transaction", "unlock_notify", "update_hook", 1023*404ca075Sdanielk1977 "version", 0 10246d31316cSdrh }; 1025411995dcSdrh enum DB_enum { 1026dc2c4915Sdrh DB_AUTHORIZER, DB_BACKUP, DB_BUSY, 1027dc2c4915Sdrh DB_CACHE, DB_CHANGES, DB_CLOSE, 1028dc2c4915Sdrh DB_COLLATE, DB_COLLATION_NEEDED, DB_COMMIT_HOOK, 1029dc2c4915Sdrh DB_COMPLETE, DB_COPY, DB_ENABLE_LOAD_EXTENSION, 1030dc2c4915Sdrh DB_ERRORCODE, DB_EVAL, DB_EXISTS, 1031dc2c4915Sdrh DB_FUNCTION, DB_INCRBLOB, DB_INTERRUPT, 1032dc2c4915Sdrh DB_LAST_INSERT_ROWID, DB_NULLVALUE, DB_ONECOLUMN, 1033dc2c4915Sdrh DB_PROFILE, DB_PROGRESS, DB_REKEY, 1034dc2c4915Sdrh DB_RESTORE, DB_ROLLBACK_HOOK, DB_STATUS, 1035dc2c4915Sdrh DB_TIMEOUT, DB_TOTAL_CHANGES, DB_TRACE, 1036*404ca075Sdanielk1977 DB_TRANSACTION, DB_UNLOCK_NOTIFY, DB_UPDATE_HOOK, 1037*404ca075Sdanielk1977 DB_VERSION, 10386d31316cSdrh }; 10391067fe11Stpoindex /* don't leave trailing commas on DB_enum, it confuses the AIX xlc compiler */ 10406d31316cSdrh 10416d31316cSdrh if( objc<2 ){ 10426d31316cSdrh Tcl_WrongNumArgs(interp, 1, objv, "SUBCOMMAND ..."); 104375897234Sdrh return TCL_ERROR; 104475897234Sdrh } 1045411995dcSdrh if( Tcl_GetIndexFromObj(interp, objv[1], DB_strs, "option", 0, &choice) ){ 10466d31316cSdrh return TCL_ERROR; 10476d31316cSdrh } 10486d31316cSdrh 1049411995dcSdrh switch( (enum DB_enum)choice ){ 105075897234Sdrh 1051e22a334bSdrh /* $db authorizer ?CALLBACK? 1052e22a334bSdrh ** 1053e22a334bSdrh ** Invoke the given callback to authorize each SQL operation as it is 1054e22a334bSdrh ** compiled. 5 arguments are appended to the callback before it is 1055e22a334bSdrh ** invoked: 1056e22a334bSdrh ** 1057e22a334bSdrh ** (1) The authorization type (ex: SQLITE_CREATE_TABLE, SQLITE_INSERT, ...) 1058e22a334bSdrh ** (2) First descriptive name (depends on authorization type) 1059e22a334bSdrh ** (3) Second descriptive name 1060e22a334bSdrh ** (4) Name of the database (ex: "main", "temp") 1061e22a334bSdrh ** (5) Name of trigger that is doing the access 1062e22a334bSdrh ** 1063e22a334bSdrh ** The callback should return on of the following strings: SQLITE_OK, 1064e22a334bSdrh ** SQLITE_IGNORE, or SQLITE_DENY. Any other return value is an error. 1065e22a334bSdrh ** 1066e22a334bSdrh ** If this method is invoked with no arguments, the current authorization 1067e22a334bSdrh ** callback string is returned. 1068e22a334bSdrh */ 1069e22a334bSdrh case DB_AUTHORIZER: { 10701211de37Sdrh #ifdef SQLITE_OMIT_AUTHORIZATION 10711211de37Sdrh Tcl_AppendResult(interp, "authorization not available in this build", 0); 10721211de37Sdrh return TCL_ERROR; 10731211de37Sdrh #else 1074e22a334bSdrh if( objc>3 ){ 1075e22a334bSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 10760f14e2ebSdrh return TCL_ERROR; 1077e22a334bSdrh }else if( objc==2 ){ 1078b5a20d3cSdrh if( pDb->zAuth ){ 1079e22a334bSdrh Tcl_AppendResult(interp, pDb->zAuth, 0); 1080e22a334bSdrh } 1081e22a334bSdrh }else{ 1082e22a334bSdrh char *zAuth; 1083e22a334bSdrh int len; 1084e22a334bSdrh if( pDb->zAuth ){ 1085e22a334bSdrh Tcl_Free(pDb->zAuth); 1086e22a334bSdrh } 1087e22a334bSdrh zAuth = Tcl_GetStringFromObj(objv[2], &len); 1088e22a334bSdrh if( zAuth && len>0 ){ 1089e22a334bSdrh pDb->zAuth = Tcl_Alloc( len + 1 ); 10905bb3eb9bSdrh memcpy(pDb->zAuth, zAuth, len+1); 1091e22a334bSdrh }else{ 1092e22a334bSdrh pDb->zAuth = 0; 1093e22a334bSdrh } 1094e22a334bSdrh if( pDb->zAuth ){ 1095e22a334bSdrh pDb->interp = interp; 10966f8a503dSdanielk1977 sqlite3_set_authorizer(pDb->db, auth_callback, pDb); 1097e22a334bSdrh }else{ 10986f8a503dSdanielk1977 sqlite3_set_authorizer(pDb->db, 0, 0); 1099e22a334bSdrh } 1100e22a334bSdrh } 11011211de37Sdrh #endif 1102e22a334bSdrh break; 1103e22a334bSdrh } 1104e22a334bSdrh 1105dc2c4915Sdrh /* $db backup ?DATABASE? FILENAME 1106dc2c4915Sdrh ** 1107dc2c4915Sdrh ** Open or create a database file named FILENAME. Transfer the 1108dc2c4915Sdrh ** content of local database DATABASE (default: "main") into the 1109dc2c4915Sdrh ** FILENAME database. 1110dc2c4915Sdrh */ 1111dc2c4915Sdrh case DB_BACKUP: { 1112dc2c4915Sdrh const char *zDestFile; 1113dc2c4915Sdrh const char *zSrcDb; 1114dc2c4915Sdrh sqlite3 *pDest; 1115dc2c4915Sdrh sqlite3_backup *pBackup; 1116dc2c4915Sdrh 1117dc2c4915Sdrh if( objc==3 ){ 1118dc2c4915Sdrh zSrcDb = "main"; 1119dc2c4915Sdrh zDestFile = Tcl_GetString(objv[2]); 1120dc2c4915Sdrh }else if( objc==4 ){ 1121dc2c4915Sdrh zSrcDb = Tcl_GetString(objv[2]); 1122dc2c4915Sdrh zDestFile = Tcl_GetString(objv[3]); 1123dc2c4915Sdrh }else{ 1124dc2c4915Sdrh Tcl_WrongNumArgs(interp, 2, objv, "?DATABASE? FILENAME"); 1125dc2c4915Sdrh return TCL_ERROR; 1126dc2c4915Sdrh } 1127dc2c4915Sdrh rc = sqlite3_open(zDestFile, &pDest); 1128dc2c4915Sdrh if( rc!=SQLITE_OK ){ 1129dc2c4915Sdrh Tcl_AppendResult(interp, "cannot open target database: ", 1130dc2c4915Sdrh sqlite3_errmsg(pDest), (char*)0); 1131dc2c4915Sdrh sqlite3_close(pDest); 1132dc2c4915Sdrh return TCL_ERROR; 1133dc2c4915Sdrh } 1134dc2c4915Sdrh pBackup = sqlite3_backup_init(pDest, "main", pDb->db, zSrcDb); 1135dc2c4915Sdrh if( pBackup==0 ){ 1136dc2c4915Sdrh Tcl_AppendResult(interp, "backup failed: ", 1137dc2c4915Sdrh sqlite3_errmsg(pDest), (char*)0); 1138dc2c4915Sdrh sqlite3_close(pDest); 1139dc2c4915Sdrh return TCL_ERROR; 1140dc2c4915Sdrh } 1141dc2c4915Sdrh while( (rc = sqlite3_backup_step(pBackup,100))==SQLITE_OK ){} 1142dc2c4915Sdrh sqlite3_backup_finish(pBackup); 1143dc2c4915Sdrh if( rc==SQLITE_DONE ){ 1144dc2c4915Sdrh rc = TCL_OK; 1145dc2c4915Sdrh }else{ 1146dc2c4915Sdrh Tcl_AppendResult(interp, "backup failed: ", 1147dc2c4915Sdrh sqlite3_errmsg(pDest), (char*)0); 1148dc2c4915Sdrh rc = TCL_ERROR; 1149dc2c4915Sdrh } 1150dc2c4915Sdrh sqlite3_close(pDest); 1151dc2c4915Sdrh break; 1152dc2c4915Sdrh } 1153dc2c4915Sdrh 1154bec3f402Sdrh /* $db busy ?CALLBACK? 1155bec3f402Sdrh ** 1156bec3f402Sdrh ** Invoke the given callback if an SQL statement attempts to open 1157bec3f402Sdrh ** a locked database file. 1158bec3f402Sdrh */ 11596d31316cSdrh case DB_BUSY: { 11606d31316cSdrh if( objc>3 ){ 11616d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "CALLBACK"); 1162bec3f402Sdrh return TCL_ERROR; 11636d31316cSdrh }else if( objc==2 ){ 1164bec3f402Sdrh if( pDb->zBusy ){ 1165bec3f402Sdrh Tcl_AppendResult(interp, pDb->zBusy, 0); 1166bec3f402Sdrh } 1167bec3f402Sdrh }else{ 11686d31316cSdrh char *zBusy; 11696d31316cSdrh int len; 1170bec3f402Sdrh if( pDb->zBusy ){ 1171bec3f402Sdrh Tcl_Free(pDb->zBusy); 11726d31316cSdrh } 11736d31316cSdrh zBusy = Tcl_GetStringFromObj(objv[2], &len); 11746d31316cSdrh if( zBusy && len>0 ){ 11756d31316cSdrh pDb->zBusy = Tcl_Alloc( len + 1 ); 11765bb3eb9bSdrh memcpy(pDb->zBusy, zBusy, len+1); 11776d31316cSdrh }else{ 1178bec3f402Sdrh pDb->zBusy = 0; 1179bec3f402Sdrh } 1180bec3f402Sdrh if( pDb->zBusy ){ 1181bec3f402Sdrh pDb->interp = interp; 11826f8a503dSdanielk1977 sqlite3_busy_handler(pDb->db, DbBusyHandler, pDb); 11836d31316cSdrh }else{ 11846f8a503dSdanielk1977 sqlite3_busy_handler(pDb->db, 0, 0); 1185bec3f402Sdrh } 1186bec3f402Sdrh } 11876d31316cSdrh break; 11886d31316cSdrh } 1189bec3f402Sdrh 1190fb7e7651Sdrh /* $db cache flush 1191fb7e7651Sdrh ** $db cache size n 1192fb7e7651Sdrh ** 1193fb7e7651Sdrh ** Flush the prepared statement cache, or set the maximum number of 1194fb7e7651Sdrh ** cached statements. 1195fb7e7651Sdrh */ 1196fb7e7651Sdrh case DB_CACHE: { 1197fb7e7651Sdrh char *subCmd; 1198fb7e7651Sdrh int n; 1199fb7e7651Sdrh 1200fb7e7651Sdrh if( objc<=2 ){ 1201fb7e7651Sdrh Tcl_WrongNumArgs(interp, 1, objv, "cache option ?arg?"); 1202fb7e7651Sdrh return TCL_ERROR; 1203fb7e7651Sdrh } 1204fb7e7651Sdrh subCmd = Tcl_GetStringFromObj( objv[2], 0 ); 1205fb7e7651Sdrh if( *subCmd=='f' && strcmp(subCmd,"flush")==0 ){ 1206fb7e7651Sdrh if( objc!=3 ){ 1207fb7e7651Sdrh Tcl_WrongNumArgs(interp, 2, objv, "flush"); 1208fb7e7651Sdrh return TCL_ERROR; 1209fb7e7651Sdrh }else{ 1210fb7e7651Sdrh flushStmtCache( pDb ); 1211fb7e7651Sdrh } 1212fb7e7651Sdrh }else if( *subCmd=='s' && strcmp(subCmd,"size")==0 ){ 1213fb7e7651Sdrh if( objc!=4 ){ 1214fb7e7651Sdrh Tcl_WrongNumArgs(interp, 2, objv, "size n"); 1215fb7e7651Sdrh return TCL_ERROR; 1216fb7e7651Sdrh }else{ 1217fb7e7651Sdrh if( TCL_ERROR==Tcl_GetIntFromObj(interp, objv[3], &n) ){ 1218fb7e7651Sdrh Tcl_AppendResult( interp, "cannot convert \"", 1219fb7e7651Sdrh Tcl_GetStringFromObj(objv[3],0), "\" to integer", 0); 1220fb7e7651Sdrh return TCL_ERROR; 1221fb7e7651Sdrh }else{ 1222fb7e7651Sdrh if( n<0 ){ 1223fb7e7651Sdrh flushStmtCache( pDb ); 1224fb7e7651Sdrh n = 0; 1225fb7e7651Sdrh }else if( n>MAX_PREPARED_STMTS ){ 1226fb7e7651Sdrh n = MAX_PREPARED_STMTS; 1227fb7e7651Sdrh } 1228fb7e7651Sdrh pDb->maxStmt = n; 1229fb7e7651Sdrh } 1230fb7e7651Sdrh } 1231fb7e7651Sdrh }else{ 1232fb7e7651Sdrh Tcl_AppendResult( interp, "bad option \"", 1233191fadcfSdanielk1977 Tcl_GetStringFromObj(objv[2],0), "\": must be flush or size", 0); 1234fb7e7651Sdrh return TCL_ERROR; 1235fb7e7651Sdrh } 1236fb7e7651Sdrh break; 1237fb7e7651Sdrh } 1238fb7e7651Sdrh 1239b28af71aSdanielk1977 /* $db changes 1240c8d30ac1Sdrh ** 1241c8d30ac1Sdrh ** Return the number of rows that were modified, inserted, or deleted by 1242b28af71aSdanielk1977 ** the most recent INSERT, UPDATE or DELETE statement, not including 1243b28af71aSdanielk1977 ** any changes made by trigger programs. 1244c8d30ac1Sdrh */ 1245c8d30ac1Sdrh case DB_CHANGES: { 1246c8d30ac1Sdrh Tcl_Obj *pResult; 1247c8d30ac1Sdrh if( objc!=2 ){ 1248c8d30ac1Sdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 1249c8d30ac1Sdrh return TCL_ERROR; 1250c8d30ac1Sdrh } 1251c8d30ac1Sdrh pResult = Tcl_GetObjResult(interp); 1252b28af71aSdanielk1977 Tcl_SetIntObj(pResult, sqlite3_changes(pDb->db)); 1253f146a776Srdc break; 1254f146a776Srdc } 1255f146a776Srdc 125675897234Sdrh /* $db close 125775897234Sdrh ** 125875897234Sdrh ** Shutdown the database 125975897234Sdrh */ 12606d31316cSdrh case DB_CLOSE: { 12616d31316cSdrh Tcl_DeleteCommand(interp, Tcl_GetStringFromObj(objv[0], 0)); 12626d31316cSdrh break; 12636d31316cSdrh } 126475897234Sdrh 12650f14e2ebSdrh /* 12660f14e2ebSdrh ** $db collate NAME SCRIPT 12670f14e2ebSdrh ** 12680f14e2ebSdrh ** Create a new SQL collation function called NAME. Whenever 12690f14e2ebSdrh ** that function is called, invoke SCRIPT to evaluate the function. 12700f14e2ebSdrh */ 12710f14e2ebSdrh case DB_COLLATE: { 12720f14e2ebSdrh SqlCollate *pCollate; 12730f14e2ebSdrh char *zName; 12740f14e2ebSdrh char *zScript; 12750f14e2ebSdrh int nScript; 12760f14e2ebSdrh if( objc!=4 ){ 12770f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, "NAME SCRIPT"); 12780f14e2ebSdrh return TCL_ERROR; 12790f14e2ebSdrh } 12800f14e2ebSdrh zName = Tcl_GetStringFromObj(objv[2], 0); 12810f14e2ebSdrh zScript = Tcl_GetStringFromObj(objv[3], &nScript); 12820f14e2ebSdrh pCollate = (SqlCollate*)Tcl_Alloc( sizeof(*pCollate) + nScript + 1 ); 12830f14e2ebSdrh if( pCollate==0 ) return TCL_ERROR; 12840f14e2ebSdrh pCollate->interp = interp; 12850f14e2ebSdrh pCollate->pNext = pDb->pCollate; 12860f14e2ebSdrh pCollate->zScript = (char*)&pCollate[1]; 12870f14e2ebSdrh pDb->pCollate = pCollate; 12885bb3eb9bSdrh memcpy(pCollate->zScript, zScript, nScript+1); 12890f14e2ebSdrh if( sqlite3_create_collation(pDb->db, zName, SQLITE_UTF8, 12900f14e2ebSdrh pCollate, tclSqlCollate) ){ 12919636c4e1Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 12920f14e2ebSdrh return TCL_ERROR; 12930f14e2ebSdrh } 12940f14e2ebSdrh break; 12950f14e2ebSdrh } 12960f14e2ebSdrh 12970f14e2ebSdrh /* 12980f14e2ebSdrh ** $db collation_needed SCRIPT 12990f14e2ebSdrh ** 13000f14e2ebSdrh ** Create a new SQL collation function called NAME. Whenever 13010f14e2ebSdrh ** that function is called, invoke SCRIPT to evaluate the function. 13020f14e2ebSdrh */ 13030f14e2ebSdrh case DB_COLLATION_NEEDED: { 13040f14e2ebSdrh if( objc!=3 ){ 13050f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, "SCRIPT"); 13060f14e2ebSdrh return TCL_ERROR; 13070f14e2ebSdrh } 13080f14e2ebSdrh if( pDb->pCollateNeeded ){ 13090f14e2ebSdrh Tcl_DecrRefCount(pDb->pCollateNeeded); 13100f14e2ebSdrh } 13110f14e2ebSdrh pDb->pCollateNeeded = Tcl_DuplicateObj(objv[2]); 13120f14e2ebSdrh Tcl_IncrRefCount(pDb->pCollateNeeded); 13130f14e2ebSdrh sqlite3_collation_needed(pDb->db, pDb, tclCollateNeeded); 13140f14e2ebSdrh break; 13150f14e2ebSdrh } 13160f14e2ebSdrh 131719e2d37fSdrh /* $db commit_hook ?CALLBACK? 131819e2d37fSdrh ** 131919e2d37fSdrh ** Invoke the given callback just before committing every SQL transaction. 132019e2d37fSdrh ** If the callback throws an exception or returns non-zero, then the 132119e2d37fSdrh ** transaction is aborted. If CALLBACK is an empty string, the callback 132219e2d37fSdrh ** is disabled. 132319e2d37fSdrh */ 132419e2d37fSdrh case DB_COMMIT_HOOK: { 132519e2d37fSdrh if( objc>3 ){ 132619e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 132719e2d37fSdrh return TCL_ERROR; 132819e2d37fSdrh }else if( objc==2 ){ 132919e2d37fSdrh if( pDb->zCommit ){ 133019e2d37fSdrh Tcl_AppendResult(interp, pDb->zCommit, 0); 133119e2d37fSdrh } 133219e2d37fSdrh }else{ 133319e2d37fSdrh char *zCommit; 133419e2d37fSdrh int len; 133519e2d37fSdrh if( pDb->zCommit ){ 133619e2d37fSdrh Tcl_Free(pDb->zCommit); 133719e2d37fSdrh } 133819e2d37fSdrh zCommit = Tcl_GetStringFromObj(objv[2], &len); 133919e2d37fSdrh if( zCommit && len>0 ){ 134019e2d37fSdrh pDb->zCommit = Tcl_Alloc( len + 1 ); 13415bb3eb9bSdrh memcpy(pDb->zCommit, zCommit, len+1); 134219e2d37fSdrh }else{ 134319e2d37fSdrh pDb->zCommit = 0; 134419e2d37fSdrh } 134519e2d37fSdrh if( pDb->zCommit ){ 134619e2d37fSdrh pDb->interp = interp; 134719e2d37fSdrh sqlite3_commit_hook(pDb->db, DbCommitHandler, pDb); 134819e2d37fSdrh }else{ 134919e2d37fSdrh sqlite3_commit_hook(pDb->db, 0, 0); 135019e2d37fSdrh } 135119e2d37fSdrh } 135219e2d37fSdrh break; 135319e2d37fSdrh } 135419e2d37fSdrh 135575897234Sdrh /* $db complete SQL 135675897234Sdrh ** 135775897234Sdrh ** Return TRUE if SQL is a complete SQL statement. Return FALSE if 135875897234Sdrh ** additional lines of input are needed. This is similar to the 135975897234Sdrh ** built-in "info complete" command of Tcl. 136075897234Sdrh */ 13616d31316cSdrh case DB_COMPLETE: { 1362ccae6026Sdrh #ifndef SQLITE_OMIT_COMPLETE 13636d31316cSdrh Tcl_Obj *pResult; 13646d31316cSdrh int isComplete; 13656d31316cSdrh if( objc!=3 ){ 13666d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL"); 136775897234Sdrh return TCL_ERROR; 136875897234Sdrh } 13696f8a503dSdanielk1977 isComplete = sqlite3_complete( Tcl_GetStringFromObj(objv[2], 0) ); 13706d31316cSdrh pResult = Tcl_GetObjResult(interp); 13716d31316cSdrh Tcl_SetBooleanObj(pResult, isComplete); 1372ccae6026Sdrh #endif 13736d31316cSdrh break; 13746d31316cSdrh } 137575897234Sdrh 137619e2d37fSdrh /* $db copy conflict-algorithm table filename ?SEPARATOR? ?NULLINDICATOR? 137719e2d37fSdrh ** 137819e2d37fSdrh ** Copy data into table from filename, optionally using SEPARATOR 137919e2d37fSdrh ** as column separators. If a column contains a null string, or the 138019e2d37fSdrh ** value of NULLINDICATOR, a NULL is inserted for the column. 138119e2d37fSdrh ** conflict-algorithm is one of the sqlite conflict algorithms: 138219e2d37fSdrh ** rollback, abort, fail, ignore, replace 138319e2d37fSdrh ** On success, return the number of lines processed, not necessarily same 138419e2d37fSdrh ** as 'db changes' due to conflict-algorithm selected. 138519e2d37fSdrh ** 138619e2d37fSdrh ** This code is basically an implementation/enhancement of 138719e2d37fSdrh ** the sqlite3 shell.c ".import" command. 138819e2d37fSdrh ** 138919e2d37fSdrh ** This command usage is equivalent to the sqlite2.x COPY statement, 139019e2d37fSdrh ** which imports file data into a table using the PostgreSQL COPY file format: 139119e2d37fSdrh ** $db copy $conflit_algo $table_name $filename \t \\N 139219e2d37fSdrh */ 139319e2d37fSdrh case DB_COPY: { 139419e2d37fSdrh char *zTable; /* Insert data into this table */ 139519e2d37fSdrh char *zFile; /* The file from which to extract data */ 139619e2d37fSdrh char *zConflict; /* The conflict algorithm to use */ 139719e2d37fSdrh sqlite3_stmt *pStmt; /* A statement */ 139819e2d37fSdrh int nCol; /* Number of columns in the table */ 139919e2d37fSdrh int nByte; /* Number of bytes in an SQL string */ 140019e2d37fSdrh int i, j; /* Loop counters */ 140119e2d37fSdrh int nSep; /* Number of bytes in zSep[] */ 140219e2d37fSdrh int nNull; /* Number of bytes in zNull[] */ 140319e2d37fSdrh char *zSql; /* An SQL statement */ 140419e2d37fSdrh char *zLine; /* A single line of input from the file */ 140519e2d37fSdrh char **azCol; /* zLine[] broken up into columns */ 140619e2d37fSdrh char *zCommit; /* How to commit changes */ 140719e2d37fSdrh FILE *in; /* The input file */ 140819e2d37fSdrh int lineno = 0; /* Line number of input file */ 140919e2d37fSdrh char zLineNum[80]; /* Line number print buffer */ 141019e2d37fSdrh Tcl_Obj *pResult; /* interp result */ 141119e2d37fSdrh 141219e2d37fSdrh char *zSep; 141319e2d37fSdrh char *zNull; 141419e2d37fSdrh if( objc<5 || objc>7 ){ 141519e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, 141619e2d37fSdrh "CONFLICT-ALGORITHM TABLE FILENAME ?SEPARATOR? ?NULLINDICATOR?"); 141719e2d37fSdrh return TCL_ERROR; 141819e2d37fSdrh } 141919e2d37fSdrh if( objc>=6 ){ 142019e2d37fSdrh zSep = Tcl_GetStringFromObj(objv[5], 0); 142119e2d37fSdrh }else{ 142219e2d37fSdrh zSep = "\t"; 142319e2d37fSdrh } 142419e2d37fSdrh if( objc>=7 ){ 142519e2d37fSdrh zNull = Tcl_GetStringFromObj(objv[6], 0); 142619e2d37fSdrh }else{ 142719e2d37fSdrh zNull = ""; 142819e2d37fSdrh } 142919e2d37fSdrh zConflict = Tcl_GetStringFromObj(objv[2], 0); 143019e2d37fSdrh zTable = Tcl_GetStringFromObj(objv[3], 0); 143119e2d37fSdrh zFile = Tcl_GetStringFromObj(objv[4], 0); 14324f21c4afSdrh nSep = strlen30(zSep); 14334f21c4afSdrh nNull = strlen30(zNull); 143419e2d37fSdrh if( nSep==0 ){ 143519e2d37fSdrh Tcl_AppendResult(interp,"Error: non-null separator required for copy",0); 143619e2d37fSdrh return TCL_ERROR; 143719e2d37fSdrh } 14383e59c012Sdrh if(strcmp(zConflict, "rollback") != 0 && 14393e59c012Sdrh strcmp(zConflict, "abort" ) != 0 && 14403e59c012Sdrh strcmp(zConflict, "fail" ) != 0 && 14413e59c012Sdrh strcmp(zConflict, "ignore" ) != 0 && 14423e59c012Sdrh strcmp(zConflict, "replace" ) != 0 ) { 144319e2d37fSdrh Tcl_AppendResult(interp, "Error: \"", zConflict, 144419e2d37fSdrh "\", conflict-algorithm must be one of: rollback, " 144519e2d37fSdrh "abort, fail, ignore, or replace", 0); 144619e2d37fSdrh return TCL_ERROR; 144719e2d37fSdrh } 144819e2d37fSdrh zSql = sqlite3_mprintf("SELECT * FROM '%q'", zTable); 144919e2d37fSdrh if( zSql==0 ){ 145019e2d37fSdrh Tcl_AppendResult(interp, "Error: no such table: ", zTable, 0); 145119e2d37fSdrh return TCL_ERROR; 145219e2d37fSdrh } 14534f21c4afSdrh nByte = strlen30(zSql); 14543e701a18Sdrh rc = sqlite3_prepare(pDb->db, zSql, -1, &pStmt, 0); 145519e2d37fSdrh sqlite3_free(zSql); 145619e2d37fSdrh if( rc ){ 145719e2d37fSdrh Tcl_AppendResult(interp, "Error: ", sqlite3_errmsg(pDb->db), 0); 145819e2d37fSdrh nCol = 0; 145919e2d37fSdrh }else{ 146019e2d37fSdrh nCol = sqlite3_column_count(pStmt); 146119e2d37fSdrh } 146219e2d37fSdrh sqlite3_finalize(pStmt); 146319e2d37fSdrh if( nCol==0 ) { 146419e2d37fSdrh return TCL_ERROR; 146519e2d37fSdrh } 146619e2d37fSdrh zSql = malloc( nByte + 50 + nCol*2 ); 146719e2d37fSdrh if( zSql==0 ) { 146819e2d37fSdrh Tcl_AppendResult(interp, "Error: can't malloc()", 0); 146919e2d37fSdrh return TCL_ERROR; 147019e2d37fSdrh } 147119e2d37fSdrh sqlite3_snprintf(nByte+50, zSql, "INSERT OR %q INTO '%q' VALUES(?", 147219e2d37fSdrh zConflict, zTable); 14734f21c4afSdrh j = strlen30(zSql); 147419e2d37fSdrh for(i=1; i<nCol; i++){ 147519e2d37fSdrh zSql[j++] = ','; 147619e2d37fSdrh zSql[j++] = '?'; 147719e2d37fSdrh } 147819e2d37fSdrh zSql[j++] = ')'; 147919e2d37fSdrh zSql[j] = 0; 14803e701a18Sdrh rc = sqlite3_prepare(pDb->db, zSql, -1, &pStmt, 0); 148119e2d37fSdrh free(zSql); 148219e2d37fSdrh if( rc ){ 148319e2d37fSdrh Tcl_AppendResult(interp, "Error: ", sqlite3_errmsg(pDb->db), 0); 148419e2d37fSdrh sqlite3_finalize(pStmt); 148519e2d37fSdrh return TCL_ERROR; 148619e2d37fSdrh } 148719e2d37fSdrh in = fopen(zFile, "rb"); 148819e2d37fSdrh if( in==0 ){ 148919e2d37fSdrh Tcl_AppendResult(interp, "Error: cannot open file: ", zFile, NULL); 149019e2d37fSdrh sqlite3_finalize(pStmt); 149119e2d37fSdrh return TCL_ERROR; 149219e2d37fSdrh } 149319e2d37fSdrh azCol = malloc( sizeof(azCol[0])*(nCol+1) ); 149419e2d37fSdrh if( azCol==0 ) { 149519e2d37fSdrh Tcl_AppendResult(interp, "Error: can't malloc()", 0); 149643617e9aSdrh fclose(in); 149719e2d37fSdrh return TCL_ERROR; 149819e2d37fSdrh } 14993752785fSdrh (void)sqlite3_exec(pDb->db, "BEGIN", 0, 0, 0); 150019e2d37fSdrh zCommit = "COMMIT"; 150119e2d37fSdrh while( (zLine = local_getline(0, in))!=0 ){ 150219e2d37fSdrh char *z; 150319e2d37fSdrh i = 0; 150419e2d37fSdrh lineno++; 150519e2d37fSdrh azCol[0] = zLine; 150619e2d37fSdrh for(i=0, z=zLine; *z; z++){ 150719e2d37fSdrh if( *z==zSep[0] && strncmp(z, zSep, nSep)==0 ){ 150819e2d37fSdrh *z = 0; 150919e2d37fSdrh i++; 151019e2d37fSdrh if( i<nCol ){ 151119e2d37fSdrh azCol[i] = &z[nSep]; 151219e2d37fSdrh z += nSep-1; 151319e2d37fSdrh } 151419e2d37fSdrh } 151519e2d37fSdrh } 151619e2d37fSdrh if( i+1!=nCol ){ 151719e2d37fSdrh char *zErr; 15184f21c4afSdrh int nErr = strlen30(zFile) + 200; 15195bb3eb9bSdrh zErr = malloc(nErr); 1520c1f4494eSdrh if( zErr ){ 15215bb3eb9bSdrh sqlite3_snprintf(nErr, zErr, 1522955de52cSdanielk1977 "Error: %s line %d: expected %d columns of data but found %d", 152319e2d37fSdrh zFile, lineno, nCol, i+1); 152419e2d37fSdrh Tcl_AppendResult(interp, zErr, 0); 152519e2d37fSdrh free(zErr); 1526c1f4494eSdrh } 152719e2d37fSdrh zCommit = "ROLLBACK"; 152819e2d37fSdrh break; 152919e2d37fSdrh } 153019e2d37fSdrh for(i=0; i<nCol; i++){ 153119e2d37fSdrh /* check for null data, if so, bind as null */ 1532ea678832Sdrh if( (nNull>0 && strcmp(azCol[i], zNull)==0) 15334f21c4afSdrh || strlen30(azCol[i])==0 1534ea678832Sdrh ){ 153519e2d37fSdrh sqlite3_bind_null(pStmt, i+1); 153619e2d37fSdrh }else{ 153719e2d37fSdrh sqlite3_bind_text(pStmt, i+1, azCol[i], -1, SQLITE_STATIC); 153819e2d37fSdrh } 153919e2d37fSdrh } 154019e2d37fSdrh sqlite3_step(pStmt); 154119e2d37fSdrh rc = sqlite3_reset(pStmt); 154219e2d37fSdrh free(zLine); 154319e2d37fSdrh if( rc!=SQLITE_OK ){ 154419e2d37fSdrh Tcl_AppendResult(interp,"Error: ", sqlite3_errmsg(pDb->db), 0); 154519e2d37fSdrh zCommit = "ROLLBACK"; 154619e2d37fSdrh break; 154719e2d37fSdrh } 154819e2d37fSdrh } 154919e2d37fSdrh free(azCol); 155019e2d37fSdrh fclose(in); 155119e2d37fSdrh sqlite3_finalize(pStmt); 15523752785fSdrh (void)sqlite3_exec(pDb->db, zCommit, 0, 0, 0); 155319e2d37fSdrh 155419e2d37fSdrh if( zCommit[0] == 'C' ){ 155519e2d37fSdrh /* success, set result as number of lines processed */ 155619e2d37fSdrh pResult = Tcl_GetObjResult(interp); 155719e2d37fSdrh Tcl_SetIntObj(pResult, lineno); 155819e2d37fSdrh rc = TCL_OK; 155919e2d37fSdrh }else{ 156019e2d37fSdrh /* failure, append lineno where failed */ 15615bb3eb9bSdrh sqlite3_snprintf(sizeof(zLineNum), zLineNum,"%d",lineno); 156219e2d37fSdrh Tcl_AppendResult(interp,", failed while processing line: ",zLineNum,0); 156319e2d37fSdrh rc = TCL_ERROR; 156419e2d37fSdrh } 156519e2d37fSdrh break; 156619e2d37fSdrh } 156719e2d37fSdrh 156875897234Sdrh /* 15694144905bSdrh ** $db enable_load_extension BOOLEAN 15704144905bSdrh ** 15714144905bSdrh ** Turn the extension loading feature on or off. It if off by 15724144905bSdrh ** default. 15734144905bSdrh */ 15744144905bSdrh case DB_ENABLE_LOAD_EXTENSION: { 1575f533acc0Sdrh #ifndef SQLITE_OMIT_LOAD_EXTENSION 15764144905bSdrh int onoff; 15774144905bSdrh if( objc!=3 ){ 15784144905bSdrh Tcl_WrongNumArgs(interp, 2, objv, "BOOLEAN"); 15794144905bSdrh return TCL_ERROR; 15804144905bSdrh } 15814144905bSdrh if( Tcl_GetBooleanFromObj(interp, objv[2], &onoff) ){ 15824144905bSdrh return TCL_ERROR; 15834144905bSdrh } 15844144905bSdrh sqlite3_enable_load_extension(pDb->db, onoff); 15854144905bSdrh break; 1586f533acc0Sdrh #else 1587f533acc0Sdrh Tcl_AppendResult(interp, "extension loading is turned off at compile-time", 1588f533acc0Sdrh 0); 1589f533acc0Sdrh return TCL_ERROR; 1590f533acc0Sdrh #endif 15914144905bSdrh } 15924144905bSdrh 15934144905bSdrh /* 1594dcd997eaSdrh ** $db errorcode 1595dcd997eaSdrh ** 1596dcd997eaSdrh ** Return the numeric error code that was returned by the most recent 15976f8a503dSdanielk1977 ** call to sqlite3_exec(). 1598dcd997eaSdrh */ 1599dcd997eaSdrh case DB_ERRORCODE: { 1600f3ce83f5Sdanielk1977 Tcl_SetObjResult(interp, Tcl_NewIntObj(sqlite3_errcode(pDb->db))); 1601dcd997eaSdrh break; 1602dcd997eaSdrh } 1603dcd997eaSdrh 1604dcd997eaSdrh /* 1605895d7472Sdrh ** $db eval $sql ?array? ?{ ...code... }? 16061807ce37Sdrh ** $db onecolumn $sql 160775897234Sdrh ** 160875897234Sdrh ** The SQL statement in $sql is evaluated. For each row, the values are 1609bec3f402Sdrh ** placed in elements of the array named "array" and ...code... is executed. 161075897234Sdrh ** If "array" and "code" are omitted, then no callback is every invoked. 161175897234Sdrh ** If "array" is an empty string, then the values are placed in variables 161275897234Sdrh ** that have the same name as the fields extracted by the query. 16131807ce37Sdrh ** 16141807ce37Sdrh ** The onecolumn method is the equivalent of: 16151807ce37Sdrh ** lindex [$db eval $sql] 0 161675897234Sdrh */ 16171807ce37Sdrh case DB_ONECOLUMN: 161897f2ebc1Sdrh case DB_EVAL: 161997f2ebc1Sdrh case DB_EXISTS: { 16209d74b4c5Sdrh char const *zSql; /* Next SQL statement to execute */ 16219d74b4c5Sdrh char const *zLeft; /* What is left after first stmt in zSql */ 16229d74b4c5Sdrh sqlite3_stmt *pStmt; /* Compiled SQL statment */ 162392febd92Sdrh Tcl_Obj *pArray; /* Name of array into which results are written */ 162492febd92Sdrh Tcl_Obj *pScript; /* Script to run for each result set */ 16251d895039Sdrh Tcl_Obj **apParm; /* Parameters that need a Tcl_DecrRefCount() */ 16261d895039Sdrh int nParm; /* Number of entries used in apParm[] */ 16271d895039Sdrh Tcl_Obj *aParm[10]; /* Static space for apParm[] in the common case */ 16281807ce37Sdrh Tcl_Obj *pRet; /* Value to be returned */ 1629fb7e7651Sdrh SqlPreparedStmt *pPreStmt; /* Pointer to a prepared statement */ 1630fb7e7651Sdrh int rc2; 1631ef2cb63eSdanielk1977 163297f2ebc1Sdrh if( choice==DB_EVAL ){ 163392febd92Sdrh if( objc<3 || objc>5 ){ 1634895d7472Sdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL ?ARRAY-NAME? ?SCRIPT?"); 163530ccda10Sdanielk1977 return TCL_ERROR; 163630ccda10Sdanielk1977 } 16371807ce37Sdrh pRet = Tcl_NewObj(); 16381807ce37Sdrh Tcl_IncrRefCount(pRet); 163997f2ebc1Sdrh }else{ 164097f2ebc1Sdrh if( objc!=3 ){ 164197f2ebc1Sdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL"); 164297f2ebc1Sdrh return TCL_ERROR; 164397f2ebc1Sdrh } 164497f2ebc1Sdrh if( choice==DB_EXISTS ){ 1645446a9b82Sdrh pRet = Tcl_NewBooleanObj(0); 1646446a9b82Sdrh Tcl_IncrRefCount(pRet); 1647446a9b82Sdrh }else{ 1648446a9b82Sdrh pRet = 0; 164997f2ebc1Sdrh } 16501807ce37Sdrh } 165192febd92Sdrh if( objc==3 ){ 165292febd92Sdrh pArray = pScript = 0; 165392febd92Sdrh }else if( objc==4 ){ 165492febd92Sdrh pArray = 0; 165592febd92Sdrh pScript = objv[3]; 165692febd92Sdrh }else{ 165792febd92Sdrh pArray = objv[3]; 165892febd92Sdrh if( Tcl_GetString(pArray)[0]==0 ) pArray = 0; 165992febd92Sdrh pScript = objv[4]; 166092febd92Sdrh } 166130ccda10Sdanielk1977 16621d895039Sdrh Tcl_IncrRefCount(objv[2]); 166330ccda10Sdanielk1977 zSql = Tcl_GetStringFromObj(objv[2], 0); 166490b6bb19Sdrh while( rc==TCL_OK && zSql[0] ){ 166592febd92Sdrh int i; /* Loop counter */ 1666fb7e7651Sdrh int nVar; /* Number of bind parameters in the pStmt */ 16678e556520Sdanielk1977 int nCol = -1; /* Number of columns in the result set */ 166892febd92Sdrh Tcl_Obj **apColName = 0; /* Array of column names */ 1669fb7e7651Sdrh int len; /* String length of zSql */ 167030ccda10Sdanielk1977 1671fb7e7651Sdrh /* Try to find a SQL statement that has already been compiled and 1672fb7e7651Sdrh ** which matches the next sequence of SQL. 1673fb7e7651Sdrh */ 1674fb7e7651Sdrh pStmt = 0; 16754f21c4afSdrh len = strlen30(zSql); 16768e556520Sdanielk1977 for(pPreStmt = pDb->stmtList; pPreStmt; pPreStmt=pPreStmt->pNext){ 1677fb7e7651Sdrh int n = pPreStmt->nSql; 1678fb7e7651Sdrh if( len>=n 1679fb7e7651Sdrh && memcmp(pPreStmt->zSql, zSql, n)==0 1680fb7e7651Sdrh && (zSql[n]==0 || zSql[n-1]==';') 1681fb7e7651Sdrh ){ 1682fb7e7651Sdrh pStmt = pPreStmt->pStmt; 1683fb7e7651Sdrh zLeft = &zSql[pPreStmt->nSql]; 1684fb7e7651Sdrh 1685fb7e7651Sdrh /* When a prepared statement is found, unlink it from the 1686fb7e7651Sdrh ** cache list. It will later be added back to the beginning 1687fb7e7651Sdrh ** of the cache list in order to implement LRU replacement. 1688fb7e7651Sdrh */ 1689fb7e7651Sdrh if( pPreStmt->pPrev ){ 1690fb7e7651Sdrh pPreStmt->pPrev->pNext = pPreStmt->pNext; 1691fb7e7651Sdrh }else{ 1692fb7e7651Sdrh pDb->stmtList = pPreStmt->pNext; 1693fb7e7651Sdrh } 1694fb7e7651Sdrh if( pPreStmt->pNext ){ 1695fb7e7651Sdrh pPreStmt->pNext->pPrev = pPreStmt->pPrev; 1696fb7e7651Sdrh }else{ 1697fb7e7651Sdrh pDb->stmtLast = pPreStmt->pPrev; 1698fb7e7651Sdrh } 1699fb7e7651Sdrh pDb->nStmt--; 1700fb7e7651Sdrh break; 1701fb7e7651Sdrh } 1702fb7e7651Sdrh } 1703fb7e7651Sdrh 1704fb7e7651Sdrh /* If no prepared statement was found. Compile the SQL text 1705fb7e7651Sdrh */ 1706fb7e7651Sdrh if( pStmt==0 ){ 17078e556520Sdanielk1977 if( SQLITE_OK!=sqlite3_prepare_v2(pDb->db, zSql, -1, &pStmt, &zLeft) ){ 1708ef2cb63eSdanielk1977 Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 170930ccda10Sdanielk1977 rc = TCL_ERROR; 171030ccda10Sdanielk1977 break; 171130ccda10Sdanielk1977 } 171292febd92Sdrh if( pStmt==0 ){ 171392febd92Sdrh if( SQLITE_OK!=sqlite3_errcode(pDb->db) ){ 1714fb7e7651Sdrh /* A compile-time error in the statement 1715fb7e7651Sdrh */ 171692febd92Sdrh Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 171792febd92Sdrh rc = TCL_ERROR; 171892febd92Sdrh break; 171992febd92Sdrh }else{ 1720fb7e7651Sdrh /* The statement was a no-op. Continue to the next statement 1721fb7e7651Sdrh ** in the SQL string. 1722fb7e7651Sdrh */ 172392febd92Sdrh zSql = zLeft; 172492febd92Sdrh continue; 172592febd92Sdrh } 172692febd92Sdrh } 1727fb7e7651Sdrh assert( pPreStmt==0 ); 1728fb7e7651Sdrh } 172930ccda10Sdanielk1977 1730fb7e7651Sdrh /* Bind values to parameters that begin with $ or : 1731fb7e7651Sdrh */ 173292febd92Sdrh nVar = sqlite3_bind_parameter_count(pStmt); 17331d895039Sdrh nParm = 0; 17341d895039Sdrh if( nVar>sizeof(aParm)/sizeof(aParm[0]) ){ 17351d895039Sdrh apParm = (Tcl_Obj**)Tcl_Alloc(nVar*sizeof(apParm[0])); 17361d895039Sdrh }else{ 17371d895039Sdrh apParm = aParm; 17381d895039Sdrh } 173992febd92Sdrh for(i=1; i<=nVar; i++){ 174092febd92Sdrh const char *zVar = sqlite3_bind_parameter_name(pStmt, i); 17414f5e80f9Sdrh if( zVar!=0 && (zVar[0]=='$' || zVar[0]==':' || zVar[0]=='@') ){ 174292febd92Sdrh Tcl_Obj *pVar = Tcl_GetVar2Ex(interp, &zVar[1], 0, 0); 174392febd92Sdrh if( pVar ){ 174492febd92Sdrh int n; 174592febd92Sdrh u8 *data; 174692febd92Sdrh char *zType = pVar->typePtr ? pVar->typePtr->name : ""; 174792febd92Sdrh char c = zType[0]; 17481c747819Sdrh if( zVar[0]=='@' || 17491c747819Sdrh (c=='b' && strcmp(zType,"bytearray")==0 && pVar->bytes==0) ){ 17501c747819Sdrh /* Load a BLOB type if the Tcl variable is a bytearray and 17511c747819Sdrh ** it has no string representation or the host 17524f5e80f9Sdrh ** parameter name begins with "@". */ 175392febd92Sdrh data = Tcl_GetByteArrayFromObj(pVar, &n); 175492febd92Sdrh sqlite3_bind_blob(pStmt, i, data, n, SQLITE_STATIC); 17551d895039Sdrh Tcl_IncrRefCount(pVar); 17561d895039Sdrh apParm[nParm++] = pVar; 1757985e0c63Sdrh }else if( c=='b' && strcmp(zType,"boolean")==0 ){ 175892febd92Sdrh Tcl_GetIntFromObj(interp, pVar, &n); 175992febd92Sdrh sqlite3_bind_int(pStmt, i, n); 176092febd92Sdrh }else if( c=='d' && strcmp(zType,"double")==0 ){ 176192febd92Sdrh double r; 176292febd92Sdrh Tcl_GetDoubleFromObj(interp, pVar, &r); 176392febd92Sdrh sqlite3_bind_double(pStmt, i, r); 1764985e0c63Sdrh }else if( (c=='w' && strcmp(zType,"wideInt")==0) || 1765985e0c63Sdrh (c=='i' && strcmp(zType,"int")==0) ){ 1766df0bddaeSdrh Tcl_WideInt v; 1767df0bddaeSdrh Tcl_GetWideIntFromObj(interp, pVar, &v); 1768df0bddaeSdrh sqlite3_bind_int64(pStmt, i, v); 176992febd92Sdrh }else{ 177000fd957bSdanielk1977 data = (unsigned char *)Tcl_GetStringFromObj(pVar, &n); 1771b08c2a72Sdrh sqlite3_bind_text(pStmt, i, (char *)data, n, SQLITE_STATIC); 17721d895039Sdrh Tcl_IncrRefCount(pVar); 17731d895039Sdrh apParm[nParm++] = pVar; 177492febd92Sdrh } 1775fb7e7651Sdrh }else{ 1776fb7e7651Sdrh sqlite3_bind_null( pStmt, i ); 177792febd92Sdrh } 177892febd92Sdrh } 177992febd92Sdrh } 178092febd92Sdrh 178192febd92Sdrh /* Execute the SQL 178292febd92Sdrh */ 178390b6bb19Sdrh while( rc==TCL_OK && pStmt && SQLITE_ROW==sqlite3_step(pStmt) ){ 17848e556520Sdanielk1977 17858e556520Sdanielk1977 /* Compute column names. This must be done after the first successful 17868e556520Sdanielk1977 ** call to sqlite3_step(), in case the query is recompiled and the 17878e556520Sdanielk1977 ** number or names of the returned columns changes. 17888e556520Sdanielk1977 */ 17898e556520Sdanielk1977 assert(!pArray||pScript); 17908e556520Sdanielk1977 if (nCol < 0) { 17918e556520Sdanielk1977 Tcl_Obj ***ap = (pScript?&apColName:0); 17928e556520Sdanielk1977 nCol = computeColumnNames(interp, pStmt, ap, pArray); 17938e556520Sdanielk1977 } 17948e556520Sdanielk1977 179592febd92Sdrh for(i=0; i<nCol; i++){ 179630ccda10Sdanielk1977 Tcl_Obj *pVal; 179730ccda10Sdanielk1977 179830ccda10Sdanielk1977 /* Set pVal to contain the i'th column of this row. */ 179992febd92Sdrh switch( sqlite3_column_type(pStmt, i) ){ 180092febd92Sdrh case SQLITE_BLOB: { 18013fd0a736Sdanielk1977 int bytes = sqlite3_column_bytes(pStmt, i); 1802eee4c8caSdrh const char *zBlob = sqlite3_column_blob(pStmt, i); 1803a7a8e14bSdanielk1977 if( !zBlob ) bytes = 0; 1804eee4c8caSdrh pVal = Tcl_NewByteArrayObj((u8*)zBlob, bytes); 180592febd92Sdrh break; 180692febd92Sdrh } 180792febd92Sdrh case SQLITE_INTEGER: { 180892febd92Sdrh sqlite_int64 v = sqlite3_column_int64(pStmt, i); 180992febd92Sdrh if( v>=-2147483647 && v<=2147483647 ){ 181092febd92Sdrh pVal = Tcl_NewIntObj(v); 181192febd92Sdrh }else{ 181292febd92Sdrh pVal = Tcl_NewWideIntObj(v); 181392febd92Sdrh } 181492febd92Sdrh break; 181592febd92Sdrh } 181692febd92Sdrh case SQLITE_FLOAT: { 181792febd92Sdrh double r = sqlite3_column_double(pStmt, i); 181892febd92Sdrh pVal = Tcl_NewDoubleObj(r); 181992febd92Sdrh break; 182092febd92Sdrh } 182155c45f2eSdanielk1977 case SQLITE_NULL: { 182255c45f2eSdanielk1977 pVal = dbTextToObj(pDb->zNull); 182355c45f2eSdanielk1977 break; 182455c45f2eSdanielk1977 } 182592febd92Sdrh default: { 182600fd957bSdanielk1977 pVal = dbTextToObj((char *)sqlite3_column_text(pStmt, i)); 182792febd92Sdrh break; 182892febd92Sdrh } 182930ccda10Sdanielk1977 } 183030ccda10Sdanielk1977 183192febd92Sdrh if( pScript ){ 183292febd92Sdrh if( pArray==0 ){ 183392febd92Sdrh Tcl_ObjSetVar2(interp, apColName[i], 0, pVal, 0); 183430ccda10Sdanielk1977 }else{ 183592febd92Sdrh Tcl_ObjSetVar2(interp, pArray, apColName[i], pVal, 0); 183630ccda10Sdanielk1977 } 18371807ce37Sdrh }else if( choice==DB_ONECOLUMN ){ 183897f2ebc1Sdrh assert( pRet==0 ); 18391807ce37Sdrh if( pRet==0 ){ 18401807ce37Sdrh pRet = pVal; 18411807ce37Sdrh Tcl_IncrRefCount(pRet); 18421807ce37Sdrh } 184390b6bb19Sdrh rc = TCL_BREAK; 184497f2ebc1Sdrh i = nCol; 184597f2ebc1Sdrh }else if( choice==DB_EXISTS ){ 1846446a9b82Sdrh Tcl_DecrRefCount(pRet); 1847446a9b82Sdrh pRet = Tcl_NewBooleanObj(1); 1848446a9b82Sdrh Tcl_IncrRefCount(pRet); 184997f2ebc1Sdrh rc = TCL_BREAK; 185097f2ebc1Sdrh i = nCol; 185130ccda10Sdanielk1977 }else{ 185230ccda10Sdanielk1977 Tcl_ListObjAppendElement(interp, pRet, pVal); 185330ccda10Sdanielk1977 } 185430ccda10Sdanielk1977 } 185530ccda10Sdanielk1977 185692febd92Sdrh if( pScript ){ 1857d1d38488Sdrh pDb->nStep = sqlite3_stmt_status(pStmt, 1858d1d38488Sdrh SQLITE_STMTSTATUS_FULLSCAN_STEP, 0); 1859d1d38488Sdrh pDb->nSort = sqlite3_stmt_status(pStmt, 1860d1d38488Sdrh SQLITE_STMTSTATUS_SORT, 0); 186192febd92Sdrh rc = Tcl_EvalObjEx(interp, pScript, 0); 186290b6bb19Sdrh if( rc==TCL_CONTINUE ){ 186390b6bb19Sdrh rc = TCL_OK; 186430ccda10Sdanielk1977 } 186530ccda10Sdanielk1977 } 186690b6bb19Sdrh } 186790b6bb19Sdrh if( rc==TCL_BREAK ){ 186890b6bb19Sdrh rc = TCL_OK; 186990b6bb19Sdrh } 187030ccda10Sdanielk1977 187192febd92Sdrh /* Free the column name objects */ 187292febd92Sdrh if( pScript ){ 18738e556520Sdanielk1977 /* If the query returned no rows, but an array variable was 18748e556520Sdanielk1977 ** specified, call computeColumnNames() now to populate the 18758e556520Sdanielk1977 ** arrayname(*) variable. 18768e556520Sdanielk1977 */ 18778e556520Sdanielk1977 if (pArray && nCol < 0) { 18788e556520Sdanielk1977 Tcl_Obj ***ap = (pScript?&apColName:0); 18798e556520Sdanielk1977 nCol = computeColumnNames(interp, pStmt, ap, pArray); 18808e556520Sdanielk1977 } 188192febd92Sdrh for(i=0; i<nCol; i++){ 188292febd92Sdrh Tcl_DecrRefCount(apColName[i]); 188392febd92Sdrh } 188492febd92Sdrh Tcl_Free((char*)apColName); 188592febd92Sdrh } 188692febd92Sdrh 18871d895039Sdrh /* Free the bound string and blob parameters */ 18881d895039Sdrh for(i=0; i<nParm; i++){ 18891d895039Sdrh Tcl_DecrRefCount(apParm[i]); 18901d895039Sdrh } 18911d895039Sdrh if( apParm!=aParm ){ 18921d895039Sdrh Tcl_Free((char*)apParm); 18931d895039Sdrh } 18941d895039Sdrh 1895fb7e7651Sdrh /* Reset the statement. If the result code is SQLITE_SCHEMA, then 1896fb7e7651Sdrh ** flush the statement cache and try the statement again. 189792febd92Sdrh */ 1898fb7e7651Sdrh rc2 = sqlite3_reset(pStmt); 1899d1d38488Sdrh pDb->nStep = sqlite3_stmt_status(pStmt, 1900d1d38488Sdrh SQLITE_STMTSTATUS_FULLSCAN_STEP, 1); 1901d1d38488Sdrh pDb->nSort = sqlite3_stmt_status(pStmt, 1902d1d38488Sdrh SQLITE_STMTSTATUS_SORT, 1); 19038e556520Sdanielk1977 if( SQLITE_OK!=rc2 ){ 1904fb7e7651Sdrh /* If a run-time error occurs, report the error and stop reading 1905fb7e7651Sdrh ** the SQL 1906fb7e7651Sdrh */ 1907ef2cb63eSdanielk1977 Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 1908fb7e7651Sdrh sqlite3_finalize(pStmt); 190930ccda10Sdanielk1977 rc = TCL_ERROR; 1910fb7e7651Sdrh if( pPreStmt ) Tcl_Free((char*)pPreStmt); 191130ccda10Sdanielk1977 break; 1912fb7e7651Sdrh }else if( pDb->maxStmt<=0 ){ 1913fb7e7651Sdrh /* If the cache is turned off, deallocated the statement */ 1914fb7e7651Sdrh if( pPreStmt ) Tcl_Free((char*)pPreStmt); 1915fb7e7651Sdrh sqlite3_finalize(pStmt); 1916fb7e7651Sdrh }else{ 1917fb7e7651Sdrh /* Everything worked and the cache is operational. 1918fb7e7651Sdrh ** Create a new SqlPreparedStmt structure if we need one. 1919fb7e7651Sdrh ** (If we already have one we can just reuse it.) 1920fb7e7651Sdrh */ 1921fb7e7651Sdrh if( pPreStmt==0 ){ 1922fb7e7651Sdrh len = zLeft - zSql; 1923d0e2a854Sdanielk1977 pPreStmt = (SqlPreparedStmt*)Tcl_Alloc( sizeof(*pPreStmt) ); 1924fb7e7651Sdrh if( pPreStmt==0 ) return TCL_ERROR; 1925fb7e7651Sdrh pPreStmt->pStmt = pStmt; 1926fb7e7651Sdrh pPreStmt->nSql = len; 1927d0e2a854Sdanielk1977 pPreStmt->zSql = sqlite3_sql(pStmt); 19284f21c4afSdrh assert( strlen30(pPreStmt->zSql)==len ); 1929d0e2a854Sdanielk1977 assert( 0==memcmp(pPreStmt->zSql, zSql, len) ); 193030ccda10Sdanielk1977 } 193130ccda10Sdanielk1977 1932fb7e7651Sdrh /* Add the prepared statement to the beginning of the cache list 1933fb7e7651Sdrh */ 1934fb7e7651Sdrh pPreStmt->pNext = pDb->stmtList; 1935fb7e7651Sdrh pPreStmt->pPrev = 0; 1936fb7e7651Sdrh if( pDb->stmtList ){ 1937fb7e7651Sdrh pDb->stmtList->pPrev = pPreStmt; 1938fb7e7651Sdrh } 1939fb7e7651Sdrh pDb->stmtList = pPreStmt; 1940fb7e7651Sdrh if( pDb->stmtLast==0 ){ 1941fb7e7651Sdrh assert( pDb->nStmt==0 ); 1942fb7e7651Sdrh pDb->stmtLast = pPreStmt; 1943fb7e7651Sdrh }else{ 1944fb7e7651Sdrh assert( pDb->nStmt>0 ); 1945fb7e7651Sdrh } 1946fb7e7651Sdrh pDb->nStmt++; 1947fb7e7651Sdrh 1948fb7e7651Sdrh /* If we have too many statement in cache, remove the surplus from the 1949fb7e7651Sdrh ** end of the cache list. 1950fb7e7651Sdrh */ 1951fb7e7651Sdrh while( pDb->nStmt>pDb->maxStmt ){ 1952fb7e7651Sdrh sqlite3_finalize(pDb->stmtLast->pStmt); 1953fb7e7651Sdrh pDb->stmtLast = pDb->stmtLast->pPrev; 1954fb7e7651Sdrh Tcl_Free((char*)pDb->stmtLast->pNext); 1955fb7e7651Sdrh pDb->stmtLast->pNext = 0; 1956fb7e7651Sdrh pDb->nStmt--; 1957fb7e7651Sdrh } 1958fb7e7651Sdrh } 1959fb7e7651Sdrh 1960fb7e7651Sdrh /* Proceed to the next statement */ 196130ccda10Sdanielk1977 zSql = zLeft; 196230ccda10Sdanielk1977 } 19631d895039Sdrh Tcl_DecrRefCount(objv[2]); 196430ccda10Sdanielk1977 19651807ce37Sdrh if( pRet ){ 1966ef2cb63eSdanielk1977 if( rc==TCL_OK ){ 196730ccda10Sdanielk1977 Tcl_SetObjResult(interp, pRet); 196830ccda10Sdanielk1977 } 1969ef2cb63eSdanielk1977 Tcl_DecrRefCount(pRet); 1970050be329Sdrh }else if( rc==TCL_OK ){ 1971050be329Sdrh Tcl_ResetResult(interp); 19721807ce37Sdrh } 197330ccda10Sdanielk1977 break; 197430ccda10Sdanielk1977 } 1975bec3f402Sdrh 1976bec3f402Sdrh /* 1977e3602be8Sdrh ** $db function NAME [-argcount N] SCRIPT 1978cabb0819Sdrh ** 1979cabb0819Sdrh ** Create a new SQL function called NAME. Whenever that function is 1980cabb0819Sdrh ** called, invoke SCRIPT to evaluate the function. 1981cabb0819Sdrh */ 1982cabb0819Sdrh case DB_FUNCTION: { 1983cabb0819Sdrh SqlFunc *pFunc; 1984d1e4733dSdrh Tcl_Obj *pScript; 1985cabb0819Sdrh char *zName; 1986e3602be8Sdrh int nArg = -1; 1987e3602be8Sdrh if( objc==6 ){ 1988e3602be8Sdrh const char *z = Tcl_GetString(objv[3]); 19894f21c4afSdrh int n = strlen30(z); 1990e3602be8Sdrh if( n>2 && strncmp(z, "-argcount",n)==0 ){ 1991e3602be8Sdrh if( Tcl_GetIntFromObj(interp, objv[4], &nArg) ) return TCL_ERROR; 1992e3602be8Sdrh if( nArg<0 ){ 1993e3602be8Sdrh Tcl_AppendResult(interp, "number of arguments must be non-negative", 1994e3602be8Sdrh (char*)0); 1995cabb0819Sdrh return TCL_ERROR; 1996cabb0819Sdrh } 1997e3602be8Sdrh } 1998e3602be8Sdrh pScript = objv[5]; 1999e3602be8Sdrh }else if( objc!=4 ){ 2000e3602be8Sdrh Tcl_WrongNumArgs(interp, 2, objv, "NAME [-argcount N] SCRIPT"); 2001e3602be8Sdrh return TCL_ERROR; 2002e3602be8Sdrh }else{ 2003d1e4733dSdrh pScript = objv[3]; 2004e3602be8Sdrh } 2005e3602be8Sdrh zName = Tcl_GetStringFromObj(objv[2], 0); 2006d1e4733dSdrh pFunc = findSqlFunc(pDb, zName); 2007cabb0819Sdrh if( pFunc==0 ) return TCL_ERROR; 2008d1e4733dSdrh if( pFunc->pScript ){ 2009d1e4733dSdrh Tcl_DecrRefCount(pFunc->pScript); 2010d1e4733dSdrh } 2011d1e4733dSdrh pFunc->pScript = pScript; 2012d1e4733dSdrh Tcl_IncrRefCount(pScript); 2013d1e4733dSdrh pFunc->useEvalObjv = safeToUseEvalObjv(interp, pScript); 2014e3602be8Sdrh rc = sqlite3_create_function(pDb->db, zName, nArg, SQLITE_UTF8, 2015d8123366Sdanielk1977 pFunc, tclSqlFunc, 0, 0); 2016fb7e7651Sdrh if( rc!=SQLITE_OK ){ 2017fb7e7651Sdrh rc = TCL_ERROR; 20189636c4e1Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 2019fb7e7651Sdrh } 2020cabb0819Sdrh break; 2021cabb0819Sdrh } 2022cabb0819Sdrh 2023cabb0819Sdrh /* 20248cbadb02Sdanielk1977 ** $db incrblob ?-readonly? ?DB? TABLE COLUMN ROWID 2025b4e9af9fSdanielk1977 */ 2026b4e9af9fSdanielk1977 case DB_INCRBLOB: { 202732a0d8bbSdanielk1977 #ifdef SQLITE_OMIT_INCRBLOB 202832a0d8bbSdanielk1977 Tcl_AppendResult(interp, "incrblob not available in this build", 0); 202932a0d8bbSdanielk1977 return TCL_ERROR; 203032a0d8bbSdanielk1977 #else 20318cbadb02Sdanielk1977 int isReadonly = 0; 2032b4e9af9fSdanielk1977 const char *zDb = "main"; 2033b4e9af9fSdanielk1977 const char *zTable; 2034b4e9af9fSdanielk1977 const char *zColumn; 2035b4e9af9fSdanielk1977 sqlite_int64 iRow; 2036b4e9af9fSdanielk1977 20378cbadb02Sdanielk1977 /* Check for the -readonly option */ 20388cbadb02Sdanielk1977 if( objc>3 && strcmp(Tcl_GetString(objv[2]), "-readonly")==0 ){ 20398cbadb02Sdanielk1977 isReadonly = 1; 20408cbadb02Sdanielk1977 } 20418cbadb02Sdanielk1977 20428cbadb02Sdanielk1977 if( objc!=(5+isReadonly) && objc!=(6+isReadonly) ){ 20438cbadb02Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?-readonly? ?DB? TABLE COLUMN ROWID"); 2044b4e9af9fSdanielk1977 return TCL_ERROR; 2045b4e9af9fSdanielk1977 } 2046b4e9af9fSdanielk1977 20478cbadb02Sdanielk1977 if( objc==(6+isReadonly) ){ 2048b4e9af9fSdanielk1977 zDb = Tcl_GetString(objv[2]); 2049b4e9af9fSdanielk1977 } 2050b4e9af9fSdanielk1977 zTable = Tcl_GetString(objv[objc-3]); 2051b4e9af9fSdanielk1977 zColumn = Tcl_GetString(objv[objc-2]); 2052b4e9af9fSdanielk1977 rc = Tcl_GetWideIntFromObj(interp, objv[objc-1], &iRow); 2053b4e9af9fSdanielk1977 2054b4e9af9fSdanielk1977 if( rc==TCL_OK ){ 20558cbadb02Sdanielk1977 rc = createIncrblobChannel( 20568cbadb02Sdanielk1977 interp, pDb, zDb, zTable, zColumn, iRow, isReadonly 20578cbadb02Sdanielk1977 ); 2058b4e9af9fSdanielk1977 } 205932a0d8bbSdanielk1977 #endif 2060b4e9af9fSdanielk1977 break; 2061b4e9af9fSdanielk1977 } 2062b4e9af9fSdanielk1977 2063b4e9af9fSdanielk1977 /* 2064f11bded5Sdrh ** $db interrupt 2065f11bded5Sdrh ** 2066f11bded5Sdrh ** Interrupt the execution of the inner-most SQL interpreter. This 2067f11bded5Sdrh ** causes the SQL statement to return an error of SQLITE_INTERRUPT. 2068f11bded5Sdrh */ 2069f11bded5Sdrh case DB_INTERRUPT: { 2070f11bded5Sdrh sqlite3_interrupt(pDb->db); 2071f11bded5Sdrh break; 2072f11bded5Sdrh } 2073f11bded5Sdrh 2074f11bded5Sdrh /* 207519e2d37fSdrh ** $db nullvalue ?STRING? 207619e2d37fSdrh ** 207719e2d37fSdrh ** Change text used when a NULL comes back from the database. If ?STRING? 207819e2d37fSdrh ** is not present, then the current string used for NULL is returned. 207919e2d37fSdrh ** If STRING is present, then STRING is returned. 208019e2d37fSdrh ** 208119e2d37fSdrh */ 208219e2d37fSdrh case DB_NULLVALUE: { 208319e2d37fSdrh if( objc!=2 && objc!=3 ){ 208419e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "NULLVALUE"); 208519e2d37fSdrh return TCL_ERROR; 208619e2d37fSdrh } 208719e2d37fSdrh if( objc==3 ){ 208819e2d37fSdrh int len; 208919e2d37fSdrh char *zNull = Tcl_GetStringFromObj(objv[2], &len); 209019e2d37fSdrh if( pDb->zNull ){ 209119e2d37fSdrh Tcl_Free(pDb->zNull); 209219e2d37fSdrh } 209319e2d37fSdrh if( zNull && len>0 ){ 209419e2d37fSdrh pDb->zNull = Tcl_Alloc( len + 1 ); 209519e2d37fSdrh strncpy(pDb->zNull, zNull, len); 209619e2d37fSdrh pDb->zNull[len] = '\0'; 209719e2d37fSdrh }else{ 209819e2d37fSdrh pDb->zNull = 0; 209919e2d37fSdrh } 210019e2d37fSdrh } 210119e2d37fSdrh Tcl_SetObjResult(interp, dbTextToObj(pDb->zNull)); 210219e2d37fSdrh break; 210319e2d37fSdrh } 210419e2d37fSdrh 210519e2d37fSdrh /* 2106af9ff33aSdrh ** $db last_insert_rowid 2107af9ff33aSdrh ** 2108af9ff33aSdrh ** Return an integer which is the ROWID for the most recent insert. 2109af9ff33aSdrh */ 2110af9ff33aSdrh case DB_LAST_INSERT_ROWID: { 2111af9ff33aSdrh Tcl_Obj *pResult; 2112f7e678d6Sdrh Tcl_WideInt rowid; 2113af9ff33aSdrh if( objc!=2 ){ 2114af9ff33aSdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 2115af9ff33aSdrh return TCL_ERROR; 2116af9ff33aSdrh } 21176f8a503dSdanielk1977 rowid = sqlite3_last_insert_rowid(pDb->db); 2118af9ff33aSdrh pResult = Tcl_GetObjResult(interp); 2119f7e678d6Sdrh Tcl_SetWideIntObj(pResult, rowid); 2120af9ff33aSdrh break; 2121af9ff33aSdrh } 2122af9ff33aSdrh 2123af9ff33aSdrh /* 21241807ce37Sdrh ** The DB_ONECOLUMN method is implemented together with DB_EVAL. 21255d9d7576Sdrh */ 21261807ce37Sdrh 21271807ce37Sdrh /* $db progress ?N CALLBACK? 21281807ce37Sdrh ** 21291807ce37Sdrh ** Invoke the given callback every N virtual machine opcodes while executing 21301807ce37Sdrh ** queries. 21311807ce37Sdrh */ 21321807ce37Sdrh case DB_PROGRESS: { 21331807ce37Sdrh if( objc==2 ){ 21341807ce37Sdrh if( pDb->zProgress ){ 21351807ce37Sdrh Tcl_AppendResult(interp, pDb->zProgress, 0); 21365d9d7576Sdrh } 21371807ce37Sdrh }else if( objc==4 ){ 21381807ce37Sdrh char *zProgress; 21391807ce37Sdrh int len; 21401807ce37Sdrh int N; 21411807ce37Sdrh if( TCL_OK!=Tcl_GetIntFromObj(interp, objv[2], &N) ){ 21421807ce37Sdrh return TCL_ERROR; 21431807ce37Sdrh }; 21441807ce37Sdrh if( pDb->zProgress ){ 21451807ce37Sdrh Tcl_Free(pDb->zProgress); 21461807ce37Sdrh } 21471807ce37Sdrh zProgress = Tcl_GetStringFromObj(objv[3], &len); 21481807ce37Sdrh if( zProgress && len>0 ){ 21491807ce37Sdrh pDb->zProgress = Tcl_Alloc( len + 1 ); 21505bb3eb9bSdrh memcpy(pDb->zProgress, zProgress, len+1); 21511807ce37Sdrh }else{ 21521807ce37Sdrh pDb->zProgress = 0; 21531807ce37Sdrh } 21541807ce37Sdrh #ifndef SQLITE_OMIT_PROGRESS_CALLBACK 21551807ce37Sdrh if( pDb->zProgress ){ 21561807ce37Sdrh pDb->interp = interp; 21571807ce37Sdrh sqlite3_progress_handler(pDb->db, N, DbProgressHandler, pDb); 21581807ce37Sdrh }else{ 21591807ce37Sdrh sqlite3_progress_handler(pDb->db, 0, 0, 0); 21601807ce37Sdrh } 21611807ce37Sdrh #endif 21621807ce37Sdrh }else{ 21631807ce37Sdrh Tcl_WrongNumArgs(interp, 2, objv, "N CALLBACK"); 21641807ce37Sdrh return TCL_ERROR; 21655d9d7576Sdrh } 21665d9d7576Sdrh break; 21675d9d7576Sdrh } 21685d9d7576Sdrh 216919e2d37fSdrh /* $db profile ?CALLBACK? 217019e2d37fSdrh ** 217119e2d37fSdrh ** Make arrangements to invoke the CALLBACK routine after each SQL statement 217219e2d37fSdrh ** that has run. The text of the SQL and the amount of elapse time are 217319e2d37fSdrh ** appended to CALLBACK before the script is run. 217419e2d37fSdrh */ 217519e2d37fSdrh case DB_PROFILE: { 217619e2d37fSdrh if( objc>3 ){ 217719e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 217819e2d37fSdrh return TCL_ERROR; 217919e2d37fSdrh }else if( objc==2 ){ 218019e2d37fSdrh if( pDb->zProfile ){ 218119e2d37fSdrh Tcl_AppendResult(interp, pDb->zProfile, 0); 218219e2d37fSdrh } 218319e2d37fSdrh }else{ 218419e2d37fSdrh char *zProfile; 218519e2d37fSdrh int len; 218619e2d37fSdrh if( pDb->zProfile ){ 218719e2d37fSdrh Tcl_Free(pDb->zProfile); 218819e2d37fSdrh } 218919e2d37fSdrh zProfile = Tcl_GetStringFromObj(objv[2], &len); 219019e2d37fSdrh if( zProfile && len>0 ){ 219119e2d37fSdrh pDb->zProfile = Tcl_Alloc( len + 1 ); 21925bb3eb9bSdrh memcpy(pDb->zProfile, zProfile, len+1); 219319e2d37fSdrh }else{ 219419e2d37fSdrh pDb->zProfile = 0; 219519e2d37fSdrh } 219619e2d37fSdrh #ifndef SQLITE_OMIT_TRACE 219719e2d37fSdrh if( pDb->zProfile ){ 219819e2d37fSdrh pDb->interp = interp; 219919e2d37fSdrh sqlite3_profile(pDb->db, DbProfileHandler, pDb); 220019e2d37fSdrh }else{ 220119e2d37fSdrh sqlite3_profile(pDb->db, 0, 0); 220219e2d37fSdrh } 220319e2d37fSdrh #endif 220419e2d37fSdrh } 220519e2d37fSdrh break; 220619e2d37fSdrh } 220719e2d37fSdrh 22085d9d7576Sdrh /* 220922fbcb8dSdrh ** $db rekey KEY 221022fbcb8dSdrh ** 221122fbcb8dSdrh ** Change the encryption key on the currently open database. 221222fbcb8dSdrh */ 221322fbcb8dSdrh case DB_REKEY: { 221422fbcb8dSdrh int nKey; 221522fbcb8dSdrh void *pKey; 221622fbcb8dSdrh if( objc!=3 ){ 221722fbcb8dSdrh Tcl_WrongNumArgs(interp, 2, objv, "KEY"); 221822fbcb8dSdrh return TCL_ERROR; 221922fbcb8dSdrh } 222022fbcb8dSdrh pKey = Tcl_GetByteArrayFromObj(objv[2], &nKey); 22219eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 22222011d5f5Sdrh rc = sqlite3_rekey(pDb->db, pKey, nKey); 222322fbcb8dSdrh if( rc ){ 2224f20b21c8Sdanielk1977 Tcl_AppendResult(interp, sqlite3ErrStr(rc), 0); 222522fbcb8dSdrh rc = TCL_ERROR; 222622fbcb8dSdrh } 222722fbcb8dSdrh #endif 222822fbcb8dSdrh break; 222922fbcb8dSdrh } 223022fbcb8dSdrh 2231dc2c4915Sdrh /* $db restore ?DATABASE? FILENAME 2232dc2c4915Sdrh ** 2233dc2c4915Sdrh ** Open a database file named FILENAME. Transfer the content 2234dc2c4915Sdrh ** of FILENAME into the local database DATABASE (default: "main"). 2235dc2c4915Sdrh */ 2236dc2c4915Sdrh case DB_RESTORE: { 2237dc2c4915Sdrh const char *zSrcFile; 2238dc2c4915Sdrh const char *zDestDb; 2239dc2c4915Sdrh sqlite3 *pSrc; 2240dc2c4915Sdrh sqlite3_backup *pBackup; 2241dc2c4915Sdrh int nTimeout = 0; 2242dc2c4915Sdrh 2243dc2c4915Sdrh if( objc==3 ){ 2244dc2c4915Sdrh zDestDb = "main"; 2245dc2c4915Sdrh zSrcFile = Tcl_GetString(objv[2]); 2246dc2c4915Sdrh }else if( objc==4 ){ 2247dc2c4915Sdrh zDestDb = Tcl_GetString(objv[2]); 2248dc2c4915Sdrh zSrcFile = Tcl_GetString(objv[3]); 2249dc2c4915Sdrh }else{ 2250dc2c4915Sdrh Tcl_WrongNumArgs(interp, 2, objv, "?DATABASE? FILENAME"); 2251dc2c4915Sdrh return TCL_ERROR; 2252dc2c4915Sdrh } 2253dc2c4915Sdrh rc = sqlite3_open_v2(zSrcFile, &pSrc, SQLITE_OPEN_READONLY, 0); 2254dc2c4915Sdrh if( rc!=SQLITE_OK ){ 2255dc2c4915Sdrh Tcl_AppendResult(interp, "cannot open source database: ", 2256dc2c4915Sdrh sqlite3_errmsg(pSrc), (char*)0); 2257dc2c4915Sdrh sqlite3_close(pSrc); 2258dc2c4915Sdrh return TCL_ERROR; 2259dc2c4915Sdrh } 2260dc2c4915Sdrh pBackup = sqlite3_backup_init(pDb->db, zDestDb, pSrc, "main"); 2261dc2c4915Sdrh if( pBackup==0 ){ 2262dc2c4915Sdrh Tcl_AppendResult(interp, "restore failed: ", 2263dc2c4915Sdrh sqlite3_errmsg(pDb->db), (char*)0); 2264dc2c4915Sdrh sqlite3_close(pSrc); 2265dc2c4915Sdrh return TCL_ERROR; 2266dc2c4915Sdrh } 2267dc2c4915Sdrh while( (rc = sqlite3_backup_step(pBackup,100))==SQLITE_OK 2268dc2c4915Sdrh || rc==SQLITE_BUSY ){ 2269dc2c4915Sdrh if( rc==SQLITE_BUSY ){ 2270dc2c4915Sdrh if( nTimeout++ >= 3 ) break; 2271dc2c4915Sdrh sqlite3_sleep(100); 2272dc2c4915Sdrh } 2273dc2c4915Sdrh } 2274dc2c4915Sdrh sqlite3_backup_finish(pBackup); 2275dc2c4915Sdrh if( rc==SQLITE_DONE ){ 2276dc2c4915Sdrh rc = TCL_OK; 2277dc2c4915Sdrh }else if( rc==SQLITE_BUSY || rc==SQLITE_LOCKED ){ 2278dc2c4915Sdrh Tcl_AppendResult(interp, "restore failed: source database busy", 2279dc2c4915Sdrh (char*)0); 2280dc2c4915Sdrh rc = TCL_ERROR; 2281dc2c4915Sdrh }else{ 2282dc2c4915Sdrh Tcl_AppendResult(interp, "restore failed: ", 2283dc2c4915Sdrh sqlite3_errmsg(pDb->db), (char*)0); 2284dc2c4915Sdrh rc = TCL_ERROR; 2285dc2c4915Sdrh } 2286dc2c4915Sdrh sqlite3_close(pSrc); 2287dc2c4915Sdrh break; 2288dc2c4915Sdrh } 2289dc2c4915Sdrh 229022fbcb8dSdrh /* 2291d1d38488Sdrh ** $db status (step|sort) 2292d1d38488Sdrh ** 2293d1d38488Sdrh ** Display SQLITE_STMTSTATUS_FULLSCAN_STEP or 2294d1d38488Sdrh ** SQLITE_STMTSTATUS_SORT for the most recent eval. 2295d1d38488Sdrh */ 2296d1d38488Sdrh case DB_STATUS: { 2297d1d38488Sdrh int v; 2298d1d38488Sdrh const char *zOp; 2299d1d38488Sdrh if( objc!=3 ){ 2300d1d38488Sdrh Tcl_WrongNumArgs(interp, 2, objv, "(step|sort)"); 2301d1d38488Sdrh return TCL_ERROR; 2302d1d38488Sdrh } 2303d1d38488Sdrh zOp = Tcl_GetString(objv[2]); 2304d1d38488Sdrh if( strcmp(zOp, "step")==0 ){ 2305d1d38488Sdrh v = pDb->nStep; 2306d1d38488Sdrh }else if( strcmp(zOp, "sort")==0 ){ 2307d1d38488Sdrh v = pDb->nSort; 2308d1d38488Sdrh }else{ 2309d1d38488Sdrh Tcl_AppendResult(interp, "bad argument: should be step or sort", 2310d1d38488Sdrh (char*)0); 2311d1d38488Sdrh return TCL_ERROR; 2312d1d38488Sdrh } 2313d1d38488Sdrh Tcl_SetObjResult(interp, Tcl_NewIntObj(v)); 2314d1d38488Sdrh break; 2315d1d38488Sdrh } 2316d1d38488Sdrh 2317d1d38488Sdrh /* 2318bec3f402Sdrh ** $db timeout MILLESECONDS 2319bec3f402Sdrh ** 2320bec3f402Sdrh ** Delay for the number of milliseconds specified when a file is locked. 2321bec3f402Sdrh */ 23226d31316cSdrh case DB_TIMEOUT: { 2323bec3f402Sdrh int ms; 23246d31316cSdrh if( objc!=3 ){ 23256d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "MILLISECONDS"); 2326bec3f402Sdrh return TCL_ERROR; 232775897234Sdrh } 23286d31316cSdrh if( Tcl_GetIntFromObj(interp, objv[2], &ms) ) return TCL_ERROR; 23296f8a503dSdanielk1977 sqlite3_busy_timeout(pDb->db, ms); 23306d31316cSdrh break; 233175897234Sdrh } 2332b5a20d3cSdrh 23330f14e2ebSdrh /* 23340f14e2ebSdrh ** $db total_changes 23350f14e2ebSdrh ** 23360f14e2ebSdrh ** Return the number of rows that were modified, inserted, or deleted 23370f14e2ebSdrh ** since the database handle was created. 23380f14e2ebSdrh */ 23390f14e2ebSdrh case DB_TOTAL_CHANGES: { 23400f14e2ebSdrh Tcl_Obj *pResult; 23410f14e2ebSdrh if( objc!=2 ){ 23420f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 23430f14e2ebSdrh return TCL_ERROR; 23440f14e2ebSdrh } 23450f14e2ebSdrh pResult = Tcl_GetObjResult(interp); 23460f14e2ebSdrh Tcl_SetIntObj(pResult, sqlite3_total_changes(pDb->db)); 23470f14e2ebSdrh break; 23480f14e2ebSdrh } 23490f14e2ebSdrh 2350b5a20d3cSdrh /* $db trace ?CALLBACK? 2351b5a20d3cSdrh ** 2352b5a20d3cSdrh ** Make arrangements to invoke the CALLBACK routine for each SQL statement 2353b5a20d3cSdrh ** that is executed. The text of the SQL is appended to CALLBACK before 2354b5a20d3cSdrh ** it is executed. 2355b5a20d3cSdrh */ 2356b5a20d3cSdrh case DB_TRACE: { 2357b5a20d3cSdrh if( objc>3 ){ 2358b5a20d3cSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 2359b97759edSdrh return TCL_ERROR; 2360b5a20d3cSdrh }else if( objc==2 ){ 2361b5a20d3cSdrh if( pDb->zTrace ){ 2362b5a20d3cSdrh Tcl_AppendResult(interp, pDb->zTrace, 0); 2363b5a20d3cSdrh } 2364b5a20d3cSdrh }else{ 2365b5a20d3cSdrh char *zTrace; 2366b5a20d3cSdrh int len; 2367b5a20d3cSdrh if( pDb->zTrace ){ 2368b5a20d3cSdrh Tcl_Free(pDb->zTrace); 2369b5a20d3cSdrh } 2370b5a20d3cSdrh zTrace = Tcl_GetStringFromObj(objv[2], &len); 2371b5a20d3cSdrh if( zTrace && len>0 ){ 2372b5a20d3cSdrh pDb->zTrace = Tcl_Alloc( len + 1 ); 23735bb3eb9bSdrh memcpy(pDb->zTrace, zTrace, len+1); 2374b5a20d3cSdrh }else{ 2375b5a20d3cSdrh pDb->zTrace = 0; 2376b5a20d3cSdrh } 237719e2d37fSdrh #ifndef SQLITE_OMIT_TRACE 2378b5a20d3cSdrh if( pDb->zTrace ){ 2379b5a20d3cSdrh pDb->interp = interp; 23806f8a503dSdanielk1977 sqlite3_trace(pDb->db, DbTraceHandler, pDb); 2381b5a20d3cSdrh }else{ 23826f8a503dSdanielk1977 sqlite3_trace(pDb->db, 0, 0); 2383b5a20d3cSdrh } 238419e2d37fSdrh #endif 2385b5a20d3cSdrh } 2386b5a20d3cSdrh break; 2387b5a20d3cSdrh } 2388b5a20d3cSdrh 23893d21423cSdrh /* $db transaction [-deferred|-immediate|-exclusive] SCRIPT 23903d21423cSdrh ** 23913d21423cSdrh ** Start a new transaction (if we are not already in the midst of a 23923d21423cSdrh ** transaction) and execute the TCL script SCRIPT. After SCRIPT 23933d21423cSdrh ** completes, either commit the transaction or roll it back if SCRIPT 23943d21423cSdrh ** throws an exception. Or if no new transation was started, do nothing. 23953d21423cSdrh ** pass the exception on up the stack. 23963d21423cSdrh ** 23973d21423cSdrh ** This command was inspired by Dave Thomas's talk on Ruby at the 23983d21423cSdrh ** 2005 O'Reilly Open Source Convention (OSCON). 23993d21423cSdrh */ 24003d21423cSdrh case DB_TRANSACTION: { 24013d21423cSdrh Tcl_Obj *pScript; 2402cd38d520Sdanielk1977 const char *zBegin = "SAVEPOINT _tcl_transaction"; 2403cd38d520Sdanielk1977 const char *zEnd; 24043d21423cSdrh if( objc!=3 && objc!=4 ){ 24053d21423cSdrh Tcl_WrongNumArgs(interp, 2, objv, "[TYPE] SCRIPT"); 24063d21423cSdrh return TCL_ERROR; 24073d21423cSdrh } 2408cd38d520Sdanielk1977 2409cd38d520Sdanielk1977 if( pDb->nTransaction ){ 2410cd38d520Sdanielk1977 zBegin = "SAVEPOINT _tcl_transaction"; 2411cd38d520Sdanielk1977 }else if( pDb->nTransaction==0 && objc==4 ){ 24123d21423cSdrh static const char *TTYPE_strs[] = { 2413ce604012Sdrh "deferred", "exclusive", "immediate", 0 24143d21423cSdrh }; 24153d21423cSdrh enum TTYPE_enum { 24163d21423cSdrh TTYPE_DEFERRED, TTYPE_EXCLUSIVE, TTYPE_IMMEDIATE 24173d21423cSdrh }; 24183d21423cSdrh int ttype; 2419b5555e7eSdrh if( Tcl_GetIndexFromObj(interp, objv[2], TTYPE_strs, "transaction type", 24203d21423cSdrh 0, &ttype) ){ 24213d21423cSdrh return TCL_ERROR; 24223d21423cSdrh } 24233d21423cSdrh switch( (enum TTYPE_enum)ttype ){ 24243d21423cSdrh case TTYPE_DEFERRED: /* no-op */; break; 24253d21423cSdrh case TTYPE_EXCLUSIVE: zBegin = "BEGIN EXCLUSIVE"; break; 24263d21423cSdrh case TTYPE_IMMEDIATE: zBegin = "BEGIN IMMEDIATE"; break; 24273d21423cSdrh } 24283d21423cSdrh } 2429cd38d520Sdanielk1977 pScript = objv[objc-1]; 2430cd38d520Sdanielk1977 24311f1549f8Sdrh pDb->disableAuth++; 2432cd38d520Sdanielk1977 rc = sqlite3_exec(pDb->db, zBegin, 0, 0, 0); 24331f1549f8Sdrh pDb->disableAuth--; 2434cd38d520Sdanielk1977 if( rc!=SQLITE_OK ){ 2435cd38d520Sdanielk1977 Tcl_AppendResult(interp, sqlite3_errmsg(pDb->db), 0); 2436cd38d520Sdanielk1977 return TCL_ERROR; 24373d21423cSdrh } 2438cd38d520Sdanielk1977 2439cd38d520Sdanielk1977 pDb->nTransaction++; 24403d21423cSdrh rc = Tcl_EvalObjEx(interp, pScript, 0); 2441cd38d520Sdanielk1977 pDb->nTransaction--; 2442cd38d520Sdanielk1977 2443cd38d520Sdanielk1977 if( rc!=TCL_ERROR ){ 2444cd38d520Sdanielk1977 if( pDb->nTransaction ){ 2445cd38d520Sdanielk1977 zEnd = "RELEASE _tcl_transaction"; 24463d21423cSdrh }else{ 24473d21423cSdrh zEnd = "COMMIT"; 24483d21423cSdrh } 2449cd38d520Sdanielk1977 }else{ 2450cd38d520Sdanielk1977 if( pDb->nTransaction ){ 2451cd38d520Sdanielk1977 zEnd = "ROLLBACK TO _tcl_transaction ; RELEASE _tcl_transaction"; 2452cd38d520Sdanielk1977 }else{ 2453cd38d520Sdanielk1977 zEnd = "ROLLBACK"; 2454cd38d520Sdanielk1977 } 2455cd38d520Sdanielk1977 } 2456cd38d520Sdanielk1977 24571f1549f8Sdrh pDb->disableAuth++; 2458109b4350Sdrh if( sqlite3_exec(pDb->db, zEnd, 0, 0, 0) ){ 2459cd38d520Sdanielk1977 /* This is a tricky scenario to handle. The most likely cause of an 2460cd38d520Sdanielk1977 ** error is that the exec() above was an attempt to commit the 2461cd38d520Sdanielk1977 ** top-level transaction that returned SQLITE_BUSY. Or, less likely, 2462cd38d520Sdanielk1977 ** that an IO-error has occured. In either case, throw a Tcl exception 2463cd38d520Sdanielk1977 ** and try to rollback the transaction. 2464cd38d520Sdanielk1977 ** 2465cd38d520Sdanielk1977 ** But it could also be that the user executed one or more BEGIN, 2466cd38d520Sdanielk1977 ** COMMIT, SAVEPOINT, RELEASE or ROLLBACK commands that are confusing 2467cd38d520Sdanielk1977 ** this method's logic. Not clear how this would be best handled. 2468cd38d520Sdanielk1977 */ 2469cd38d520Sdanielk1977 if( rc!=TCL_ERROR ){ 2470cd38d520Sdanielk1977 Tcl_AppendResult(interp, sqlite3_errmsg(pDb->db), 0); 2471cd38d520Sdanielk1977 rc = TCL_ERROR; 2472cd38d520Sdanielk1977 } 2473109b4350Sdrh sqlite3_exec(pDb->db, "ROLLBACK", 0, 0, 0); 2474109b4350Sdrh } 24751f1549f8Sdrh pDb->disableAuth--; 2476cd38d520Sdanielk1977 24773d21423cSdrh break; 24783d21423cSdrh } 24793d21423cSdrh 248094eb6a14Sdanielk1977 /* 2481*404ca075Sdanielk1977 ** $db unlock_notify ?script? 2482*404ca075Sdanielk1977 */ 2483*404ca075Sdanielk1977 case DB_UNLOCK_NOTIFY: { 2484*404ca075Sdanielk1977 #ifndef SQLITE_ENABLE_UNLOCK_NOTIFY 2485*404ca075Sdanielk1977 Tcl_AppendResult(interp, "unlock_notify not available in this build", 0); 2486*404ca075Sdanielk1977 rc = TCL_ERROR; 2487*404ca075Sdanielk1977 #else 2488*404ca075Sdanielk1977 if( objc!=2 && objc!=3 ){ 2489*404ca075Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?SCRIPT?"); 2490*404ca075Sdanielk1977 rc = TCL_ERROR; 2491*404ca075Sdanielk1977 }else{ 2492*404ca075Sdanielk1977 void (*xNotify)(void **, int) = 0; 2493*404ca075Sdanielk1977 void *pNotifyArg = 0; 2494*404ca075Sdanielk1977 2495*404ca075Sdanielk1977 if( pDb->pUnlockNotify ){ 2496*404ca075Sdanielk1977 Tcl_DecrRefCount(pDb->pUnlockNotify); 2497*404ca075Sdanielk1977 pDb->pUnlockNotify = 0; 2498*404ca075Sdanielk1977 } 2499*404ca075Sdanielk1977 2500*404ca075Sdanielk1977 if( objc==3 ){ 2501*404ca075Sdanielk1977 xNotify = DbUnlockNotify; 2502*404ca075Sdanielk1977 pNotifyArg = (void *)pDb; 2503*404ca075Sdanielk1977 pDb->pUnlockNotify = objv[2]; 2504*404ca075Sdanielk1977 Tcl_IncrRefCount(pDb->pUnlockNotify); 2505*404ca075Sdanielk1977 } 2506*404ca075Sdanielk1977 2507*404ca075Sdanielk1977 if( sqlite3_unlock_notify(pDb->db, xNotify, pNotifyArg) ){ 2508*404ca075Sdanielk1977 Tcl_AppendResult(interp, sqlite3_errmsg(pDb->db), 0); 2509*404ca075Sdanielk1977 rc = TCL_ERROR; 2510*404ca075Sdanielk1977 } 2511*404ca075Sdanielk1977 } 2512*404ca075Sdanielk1977 #endif 2513*404ca075Sdanielk1977 break; 2514*404ca075Sdanielk1977 } 2515*404ca075Sdanielk1977 2516*404ca075Sdanielk1977 /* 251794eb6a14Sdanielk1977 ** $db update_hook ?script? 251871fd80bfSdanielk1977 ** $db rollback_hook ?script? 251994eb6a14Sdanielk1977 */ 252071fd80bfSdanielk1977 case DB_UPDATE_HOOK: 252171fd80bfSdanielk1977 case DB_ROLLBACK_HOOK: { 252271fd80bfSdanielk1977 252371fd80bfSdanielk1977 /* set ppHook to point at pUpdateHook or pRollbackHook, depending on 252471fd80bfSdanielk1977 ** whether [$db update_hook] or [$db rollback_hook] was invoked. 252571fd80bfSdanielk1977 */ 252671fd80bfSdanielk1977 Tcl_Obj **ppHook; 252771fd80bfSdanielk1977 if( choice==DB_UPDATE_HOOK ){ 252871fd80bfSdanielk1977 ppHook = &pDb->pUpdateHook; 252971fd80bfSdanielk1977 }else{ 253071fd80bfSdanielk1977 ppHook = &pDb->pRollbackHook; 253171fd80bfSdanielk1977 } 253271fd80bfSdanielk1977 253394eb6a14Sdanielk1977 if( objc!=2 && objc!=3 ){ 253494eb6a14Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?SCRIPT?"); 253594eb6a14Sdanielk1977 return TCL_ERROR; 253694eb6a14Sdanielk1977 } 253771fd80bfSdanielk1977 if( *ppHook ){ 253871fd80bfSdanielk1977 Tcl_SetObjResult(interp, *ppHook); 253994eb6a14Sdanielk1977 if( objc==3 ){ 254071fd80bfSdanielk1977 Tcl_DecrRefCount(*ppHook); 254171fd80bfSdanielk1977 *ppHook = 0; 254294eb6a14Sdanielk1977 } 254394eb6a14Sdanielk1977 } 254494eb6a14Sdanielk1977 if( objc==3 ){ 254571fd80bfSdanielk1977 assert( !(*ppHook) ); 254694eb6a14Sdanielk1977 if( Tcl_GetCharLength(objv[2])>0 ){ 254771fd80bfSdanielk1977 *ppHook = objv[2]; 254871fd80bfSdanielk1977 Tcl_IncrRefCount(*ppHook); 254994eb6a14Sdanielk1977 } 255094eb6a14Sdanielk1977 } 255171fd80bfSdanielk1977 255271fd80bfSdanielk1977 sqlite3_update_hook(pDb->db, (pDb->pUpdateHook?DbUpdateHandler:0), pDb); 255371fd80bfSdanielk1977 sqlite3_rollback_hook(pDb->db,(pDb->pRollbackHook?DbRollbackHandler:0),pDb); 255471fd80bfSdanielk1977 255594eb6a14Sdanielk1977 break; 255694eb6a14Sdanielk1977 } 255794eb6a14Sdanielk1977 25584397de57Sdanielk1977 /* $db version 25594397de57Sdanielk1977 ** 25604397de57Sdanielk1977 ** Return the version string for this database. 25614397de57Sdanielk1977 */ 25624397de57Sdanielk1977 case DB_VERSION: { 25634397de57Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_libversion(), TCL_STATIC); 25644397de57Sdanielk1977 break; 25654397de57Sdanielk1977 } 25664397de57Sdanielk1977 25671067fe11Stpoindex 25686d31316cSdrh } /* End of the SWITCH statement */ 256922fbcb8dSdrh return rc; 257075897234Sdrh } 257175897234Sdrh 257275897234Sdrh /* 25733570ad93Sdrh ** sqlite3 DBNAME FILENAME ?-vfs VFSNAME? ?-key KEY? ?-readonly BOOLEAN? 25749a6284c1Sdanielk1977 ** ?-create BOOLEAN? ?-nomutex BOOLEAN? 257575897234Sdrh ** 257675897234Sdrh ** This is the main Tcl command. When the "sqlite" Tcl command is 257775897234Sdrh ** invoked, this routine runs to process that command. 257875897234Sdrh ** 257975897234Sdrh ** The first argument, DBNAME, is an arbitrary name for a new 258075897234Sdrh ** database connection. This command creates a new command named 258175897234Sdrh ** DBNAME that is used to control that connection. The database 258275897234Sdrh ** connection is deleted when the DBNAME command is deleted. 258375897234Sdrh ** 25843570ad93Sdrh ** The second argument is the name of the database file. 2585fbc3eab8Sdrh ** 258675897234Sdrh */ 258722fbcb8dSdrh static int DbMain(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){ 2588bec3f402Sdrh SqliteDb *p; 258922fbcb8dSdrh void *pKey = 0; 259022fbcb8dSdrh int nKey = 0; 259122fbcb8dSdrh const char *zArg; 259275897234Sdrh char *zErrMsg; 25933570ad93Sdrh int i; 259422fbcb8dSdrh const char *zFile; 25953570ad93Sdrh const char *zVfs = 0; 2596cc91e7f8Sdrh int flags = SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE | SQLITE_OPEN_NOMUTEX; 2597882e8e4dSdrh Tcl_DString translatedFilename; 259822fbcb8dSdrh if( objc==2 ){ 259922fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[1], 0); 260022fbcb8dSdrh if( strcmp(zArg,"-version")==0 ){ 26016f8a503dSdanielk1977 Tcl_AppendResult(interp,sqlite3_version,0); 2602647cb0e1Sdrh return TCL_OK; 2603647cb0e1Sdrh } 26049eb9e26bSdrh if( strcmp(zArg,"-has-codec")==0 ){ 26059eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 260622fbcb8dSdrh Tcl_AppendResult(interp,"1",0); 260722fbcb8dSdrh #else 260822fbcb8dSdrh Tcl_AppendResult(interp,"0",0); 260922fbcb8dSdrh #endif 261022fbcb8dSdrh return TCL_OK; 261122fbcb8dSdrh } 2612fbc3eab8Sdrh } 26133570ad93Sdrh for(i=3; i+1<objc; i+=2){ 26143570ad93Sdrh zArg = Tcl_GetString(objv[i]); 261522fbcb8dSdrh if( strcmp(zArg,"-key")==0 ){ 26163570ad93Sdrh pKey = Tcl_GetByteArrayFromObj(objv[i+1], &nKey); 26173570ad93Sdrh }else if( strcmp(zArg, "-vfs")==0 ){ 26183570ad93Sdrh i++; 26193570ad93Sdrh zVfs = Tcl_GetString(objv[i]); 26203570ad93Sdrh }else if( strcmp(zArg, "-readonly")==0 ){ 26213570ad93Sdrh int b; 26223570ad93Sdrh if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 26233570ad93Sdrh if( b ){ 262433f4e02aSdrh flags &= ~(SQLITE_OPEN_READWRITE|SQLITE_OPEN_CREATE); 26253570ad93Sdrh flags |= SQLITE_OPEN_READONLY; 26263570ad93Sdrh }else{ 26273570ad93Sdrh flags &= ~SQLITE_OPEN_READONLY; 26283570ad93Sdrh flags |= SQLITE_OPEN_READWRITE; 26293570ad93Sdrh } 26303570ad93Sdrh }else if( strcmp(zArg, "-create")==0 ){ 26313570ad93Sdrh int b; 26323570ad93Sdrh if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 263333f4e02aSdrh if( b && (flags & SQLITE_OPEN_READONLY)==0 ){ 26343570ad93Sdrh flags |= SQLITE_OPEN_CREATE; 26353570ad93Sdrh }else{ 26363570ad93Sdrh flags &= ~SQLITE_OPEN_CREATE; 26373570ad93Sdrh } 26389a6284c1Sdanielk1977 }else if( strcmp(zArg, "-nomutex")==0 ){ 26399a6284c1Sdanielk1977 int b; 26409a6284c1Sdanielk1977 if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 26419a6284c1Sdanielk1977 if( b ){ 26429a6284c1Sdanielk1977 flags |= SQLITE_OPEN_NOMUTEX; 2643039963adSdrh flags &= ~SQLITE_OPEN_FULLMUTEX; 26449a6284c1Sdanielk1977 }else{ 26459a6284c1Sdanielk1977 flags &= ~SQLITE_OPEN_NOMUTEX; 26469a6284c1Sdanielk1977 } 2647039963adSdrh }else if( strcmp(zArg, "-fullmutex")==0 ){ 2648039963adSdrh int b; 2649039963adSdrh if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 2650039963adSdrh if( b ){ 2651039963adSdrh flags |= SQLITE_OPEN_FULLMUTEX; 2652039963adSdrh flags &= ~SQLITE_OPEN_NOMUTEX; 2653039963adSdrh }else{ 2654039963adSdrh flags &= ~SQLITE_OPEN_FULLMUTEX; 2655039963adSdrh } 26563570ad93Sdrh }else{ 26573570ad93Sdrh Tcl_AppendResult(interp, "unknown option: ", zArg, (char*)0); 26583570ad93Sdrh return TCL_ERROR; 265922fbcb8dSdrh } 266022fbcb8dSdrh } 26613570ad93Sdrh if( objc<3 || (objc&1)!=1 ){ 266222fbcb8dSdrh Tcl_WrongNumArgs(interp, 1, objv, 26633570ad93Sdrh "HANDLE FILENAME ?-vfs VFSNAME? ?-readonly BOOLEAN? ?-create BOOLEAN?" 2664039963adSdrh " ?-nomutex BOOLEAN? ?-fullmutex BOOLEAN?" 26659eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 26663570ad93Sdrh " ?-key CODECKEY?" 266722fbcb8dSdrh #endif 266822fbcb8dSdrh ); 266975897234Sdrh return TCL_ERROR; 267075897234Sdrh } 267175897234Sdrh zErrMsg = 0; 26724cdc9e84Sdrh p = (SqliteDb*)Tcl_Alloc( sizeof(*p) ); 267375897234Sdrh if( p==0 ){ 2674bec3f402Sdrh Tcl_SetResult(interp, "malloc failed", TCL_STATIC); 2675bec3f402Sdrh return TCL_ERROR; 2676bec3f402Sdrh } 2677bec3f402Sdrh memset(p, 0, sizeof(*p)); 267822fbcb8dSdrh zFile = Tcl_GetStringFromObj(objv[2], 0); 2679882e8e4dSdrh zFile = Tcl_TranslateFileName(interp, zFile, &translatedFilename); 26803570ad93Sdrh sqlite3_open_v2(zFile, &p->db, flags, zVfs); 2681882e8e4dSdrh Tcl_DStringFree(&translatedFilename); 268280290863Sdanielk1977 if( SQLITE_OK!=sqlite3_errcode(p->db) ){ 26839404d50eSdrh zErrMsg = sqlite3_mprintf("%s", sqlite3_errmsg(p->db)); 268480290863Sdanielk1977 sqlite3_close(p->db); 268580290863Sdanielk1977 p->db = 0; 268680290863Sdanielk1977 } 26872011d5f5Sdrh #ifdef SQLITE_HAS_CODEC 2688f3a65f7eSdrh if( p->db ){ 26892011d5f5Sdrh sqlite3_key(p->db, pKey, nKey); 2690f3a65f7eSdrh } 2691eb8ed70dSdrh #endif 2692bec3f402Sdrh if( p->db==0 ){ 269375897234Sdrh Tcl_SetResult(interp, zErrMsg, TCL_VOLATILE); 2694bec3f402Sdrh Tcl_Free((char*)p); 26959404d50eSdrh sqlite3_free(zErrMsg); 269675897234Sdrh return TCL_ERROR; 269775897234Sdrh } 2698fb7e7651Sdrh p->maxStmt = NUM_PREPARED_STMTS; 26995169bbc6Sdrh p->interp = interp; 270022fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[1], 0); 270122fbcb8dSdrh Tcl_CreateObjCommand(interp, zArg, DbObjCmd, (char*)p, DbDeleteCmd); 270275897234Sdrh return TCL_OK; 270375897234Sdrh } 270475897234Sdrh 270575897234Sdrh /* 270690ca9753Sdrh ** Provide a dummy Tcl_InitStubs if we are using this as a static 270790ca9753Sdrh ** library. 270890ca9753Sdrh */ 270990ca9753Sdrh #ifndef USE_TCL_STUBS 271090ca9753Sdrh # undef Tcl_InitStubs 271190ca9753Sdrh # define Tcl_InitStubs(a,b,c) 271290ca9753Sdrh #endif 271390ca9753Sdrh 271490ca9753Sdrh /* 271529bc4615Sdrh ** Make sure we have a PACKAGE_VERSION macro defined. This will be 271629bc4615Sdrh ** defined automatically by the TEA makefile. But other makefiles 271729bc4615Sdrh ** do not define it. 271829bc4615Sdrh */ 271929bc4615Sdrh #ifndef PACKAGE_VERSION 272029bc4615Sdrh # define PACKAGE_VERSION SQLITE_VERSION 272129bc4615Sdrh #endif 272229bc4615Sdrh 272329bc4615Sdrh /* 272475897234Sdrh ** Initialize this module. 272575897234Sdrh ** 272675897234Sdrh ** This Tcl module contains only a single new Tcl command named "sqlite". 272775897234Sdrh ** (Hence there is no namespace. There is no point in using a namespace 272875897234Sdrh ** if the extension only supplies one new name!) The "sqlite" command is 272975897234Sdrh ** used to open a new SQLite database. See the DbMain() routine above 273075897234Sdrh ** for additional information. 273175897234Sdrh */ 2732ad6e1370Sdrh EXTERN int Sqlite3_Init(Tcl_Interp *interp){ 273392febd92Sdrh Tcl_InitStubs(interp, "8.4", 0); 2734ef4ac8f9Sdrh Tcl_CreateObjCommand(interp, "sqlite3", (Tcl_ObjCmdProc*)DbMain, 0, 0); 273529bc4615Sdrh Tcl_PkgProvide(interp, "sqlite3", PACKAGE_VERSION); 273649766d6cSdrh Tcl_CreateObjCommand(interp, "sqlite", (Tcl_ObjCmdProc*)DbMain, 0, 0); 273729bc4615Sdrh Tcl_PkgProvide(interp, "sqlite", PACKAGE_VERSION); 273890ca9753Sdrh return TCL_OK; 273990ca9753Sdrh } 2740ad6e1370Sdrh EXTERN int Tclsqlite3_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2741ad6e1370Sdrh EXTERN int Sqlite3_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2742ad6e1370Sdrh EXTERN int Tclsqlite3_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2743e2c3a659Sdrh EXTERN int Sqlite3_Unload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2744e2c3a659Sdrh EXTERN int Tclsqlite3_Unload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2745e2c3a659Sdrh EXTERN int Sqlite3_SafeUnload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2746e2c3a659Sdrh EXTERN int Tclsqlite3_SafeUnload(Tcl_Interp *interp, int flags){ return TCL_OK;} 2747e2c3a659Sdrh 274849766d6cSdrh 274949766d6cSdrh #ifndef SQLITE_3_SUFFIX_ONLY 2750ad6e1370Sdrh EXTERN int Sqlite_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2751ad6e1370Sdrh EXTERN int Tclsqlite_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2752ad6e1370Sdrh EXTERN int Sqlite_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2753ad6e1370Sdrh EXTERN int Tclsqlite_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2754e2c3a659Sdrh EXTERN int Sqlite_Unload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2755e2c3a659Sdrh EXTERN int Tclsqlite_Unload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2756e2c3a659Sdrh EXTERN int Sqlite_SafeUnload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2757e2c3a659Sdrh EXTERN int Tclsqlite_SafeUnload(Tcl_Interp *interp, int flags){ return TCL_OK;} 275849766d6cSdrh #endif 275975897234Sdrh 27603e27c026Sdrh #ifdef TCLSH 27613e27c026Sdrh /***************************************************************************** 27623e27c026Sdrh ** The code that follows is used to build standalone TCL interpreters 27633570ad93Sdrh ** that are statically linked with SQLite. 276475897234Sdrh */ 2765348784efSdrh 2766348784efSdrh /* 27673e27c026Sdrh ** If the macro TCLSH is one, then put in code this for the 27683e27c026Sdrh ** "main" routine that will initialize Tcl and take input from 27693570ad93Sdrh ** standard input, or if a file is named on the command line 27703570ad93Sdrh ** the TCL interpreter reads and evaluates that file. 2771348784efSdrh */ 27723e27c026Sdrh #if TCLSH==1 2773348784efSdrh static char zMainloop[] = 2774348784efSdrh "set line {}\n" 2775348784efSdrh "while {![eof stdin]} {\n" 2776348784efSdrh "if {$line!=\"\"} {\n" 2777348784efSdrh "puts -nonewline \"> \"\n" 2778348784efSdrh "} else {\n" 2779348784efSdrh "puts -nonewline \"% \"\n" 2780348784efSdrh "}\n" 2781348784efSdrh "flush stdout\n" 2782348784efSdrh "append line [gets stdin]\n" 2783348784efSdrh "if {[info complete $line]} {\n" 2784348784efSdrh "if {[catch {uplevel #0 $line} result]} {\n" 2785348784efSdrh "puts stderr \"Error: $result\"\n" 2786348784efSdrh "} elseif {$result!=\"\"} {\n" 2787348784efSdrh "puts $result\n" 2788348784efSdrh "}\n" 2789348784efSdrh "set line {}\n" 2790348784efSdrh "} else {\n" 2791348784efSdrh "append line \\n\n" 2792348784efSdrh "}\n" 2793348784efSdrh "}\n" 2794348784efSdrh ; 27953e27c026Sdrh #endif 27963e27c026Sdrh 27973e27c026Sdrh /* 27983e27c026Sdrh ** If the macro TCLSH is two, then get the main loop code out of 27993e27c026Sdrh ** the separate file "spaceanal_tcl.h". 28003e27c026Sdrh */ 28013e27c026Sdrh #if TCLSH==2 28023e27c026Sdrh static char zMainloop[] = 28033e27c026Sdrh #include "spaceanal_tcl.h" 28043e27c026Sdrh ; 28053e27c026Sdrh #endif 2806348784efSdrh 2807348784efSdrh #define TCLSH_MAIN main /* Needed to fake out mktclapp */ 2808348784efSdrh int TCLSH_MAIN(int argc, char **argv){ 2809348784efSdrh Tcl_Interp *interp; 28100a549071Sdanielk1977 28110a549071Sdanielk1977 /* Call sqlite3_shutdown() once before doing anything else. This is to 28120a549071Sdanielk1977 ** test that sqlite3_shutdown() can be safely called by a process before 28130a549071Sdanielk1977 ** sqlite3_initialize() is. */ 28140a549071Sdanielk1977 sqlite3_shutdown(); 28150a549071Sdanielk1977 2816297ecf14Sdrh Tcl_FindExecutable(argv[0]); 2817348784efSdrh interp = Tcl_CreateInterp(); 281838f8271fSdrh Sqlite3_Init(interp); 2819d9b0257aSdrh #ifdef SQLITE_TEST 2820d1bf3512Sdrh { 28212f999a67Sdrh extern int Md5_Init(Tcl_Interp*); 28222f999a67Sdrh extern int Sqliteconfig_Init(Tcl_Interp*); 2823d1bf3512Sdrh extern int Sqlitetest1_Init(Tcl_Interp*); 28245c4d9703Sdrh extern int Sqlitetest2_Init(Tcl_Interp*); 28255c4d9703Sdrh extern int Sqlitetest3_Init(Tcl_Interp*); 2826a6064dcfSdrh extern int Sqlitetest4_Init(Tcl_Interp*); 2827998b56c3Sdanielk1977 extern int Sqlitetest5_Init(Tcl_Interp*); 28289c06c953Sdrh extern int Sqlitetest6_Init(Tcl_Interp*); 282929c636bcSdrh extern int Sqlitetest7_Init(Tcl_Interp*); 2830b9bb7c18Sdrh extern int Sqlitetest8_Init(Tcl_Interp*); 2831a713f2c3Sdanielk1977 extern int Sqlitetest9_Init(Tcl_Interp*); 28322366940dSdrh extern int Sqlitetestasync_Init(Tcl_Interp*); 28331409be69Sdrh extern int Sqlitetest_autoext_Init(Tcl_Interp*); 2834984bfaa4Sdrh extern int Sqlitetest_func_Init(Tcl_Interp*); 283515926590Sdrh extern int Sqlitetest_hexio_Init(Tcl_Interp*); 28362f999a67Sdrh extern int Sqlitetest_malloc_Init(Tcl_Interp*); 28371a9ed0b2Sdanielk1977 extern int Sqlitetest_mutex_Init(Tcl_Interp*); 28382f999a67Sdrh extern int Sqlitetestschema_Init(Tcl_Interp*); 28392f999a67Sdrh extern int Sqlitetestsse_Init(Tcl_Interp*); 28402f999a67Sdrh extern int Sqlitetesttclvar_Init(Tcl_Interp*); 284144918fa0Sdanielk1977 extern int SqlitetestThread_Init(Tcl_Interp*); 2842a15db353Sdanielk1977 extern int SqlitetestOnefile_Init(); 28435d1f5aa6Sdanielk1977 extern int SqlitetestOsinst_Init(Tcl_Interp*); 28440410302eSdanielk1977 extern int Sqlitetestbackup_Init(Tcl_Interp*); 28452e66f0b9Sdrh 28462f999a67Sdrh Md5_Init(interp); 28472f999a67Sdrh Sqliteconfig_Init(interp); 28486490bebdSdanielk1977 Sqlitetest1_Init(interp); 28495c4d9703Sdrh Sqlitetest2_Init(interp); 2850de647130Sdrh Sqlitetest3_Init(interp); 2851fc57d7bfSdanielk1977 Sqlitetest4_Init(interp); 2852998b56c3Sdanielk1977 Sqlitetest5_Init(interp); 28539c06c953Sdrh Sqlitetest6_Init(interp); 285429c636bcSdrh Sqlitetest7_Init(interp); 2855b9bb7c18Sdrh Sqlitetest8_Init(interp); 2856a713f2c3Sdanielk1977 Sqlitetest9_Init(interp); 28572366940dSdrh Sqlitetestasync_Init(interp); 28581409be69Sdrh Sqlitetest_autoext_Init(interp); 2859984bfaa4Sdrh Sqlitetest_func_Init(interp); 286015926590Sdrh Sqlitetest_hexio_Init(interp); 28612f999a67Sdrh Sqlitetest_malloc_Init(interp); 28621a9ed0b2Sdanielk1977 Sqlitetest_mutex_Init(interp); 28632f999a67Sdrh Sqlitetestschema_Init(interp); 28642f999a67Sdrh Sqlitetesttclvar_Init(interp); 286544918fa0Sdanielk1977 SqlitetestThread_Init(interp); 2866a15db353Sdanielk1977 SqlitetestOnefile_Init(interp); 28675d1f5aa6Sdanielk1977 SqlitetestOsinst_Init(interp); 28680410302eSdanielk1977 Sqlitetestbackup_Init(interp); 2869a15db353Sdanielk1977 287089dec819Sdrh #ifdef SQLITE_SSE 28712e66f0b9Sdrh Sqlitetestsse_Init(interp); 28722e66f0b9Sdrh #endif 2873d1bf3512Sdrh } 2874d1bf3512Sdrh #endif 28753e27c026Sdrh if( argc>=2 || TCLSH==2 ){ 2876348784efSdrh int i; 2877ad42c3a3Sshess char zArgc[32]; 2878ad42c3a3Sshess sqlite3_snprintf(sizeof(zArgc), zArgc, "%d", argc-(3-TCLSH)); 2879ad42c3a3Sshess Tcl_SetVar(interp,"argc", zArgc, TCL_GLOBAL_ONLY); 2880348784efSdrh Tcl_SetVar(interp,"argv0",argv[1],TCL_GLOBAL_ONLY); 2881348784efSdrh Tcl_SetVar(interp,"argv", "", TCL_GLOBAL_ONLY); 288261212b69Sdrh for(i=3-TCLSH; i<argc; i++){ 2883348784efSdrh Tcl_SetVar(interp, "argv", argv[i], 2884348784efSdrh TCL_GLOBAL_ONLY | TCL_LIST_ELEMENT | TCL_APPEND_VALUE); 2885348784efSdrh } 28863e27c026Sdrh if( TCLSH==1 && Tcl_EvalFile(interp, argv[1])!=TCL_OK ){ 28870de8c112Sdrh const char *zInfo = Tcl_GetVar(interp, "errorInfo", TCL_GLOBAL_ONLY); 2888a81c64a2Sdrh if( zInfo==0 ) zInfo = Tcl_GetStringResult(interp); 2889c61053b7Sdrh fprintf(stderr,"%s: %s\n", *argv, zInfo); 2890348784efSdrh return 1; 2891348784efSdrh } 28923e27c026Sdrh } 28933e27c026Sdrh if( argc<=1 || TCLSH==2 ){ 2894348784efSdrh Tcl_GlobalEval(interp, zMainloop); 2895348784efSdrh } 2896348784efSdrh return 0; 2897348784efSdrh } 2898348784efSdrh #endif /* TCLSH */ 2899