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. 1457a0227fSdrh ** 1557a0227fSdrh ** Compile-time options: 1657a0227fSdrh ** 1757a0227fSdrh ** -DTCLSH=1 Add a "main()" routine that works as a tclsh. 1857a0227fSdrh ** 1957a0227fSdrh ** -DSQLITE_TCLMD5 When used in conjuction with -DTCLSH=1, add 2057a0227fSdrh ** four new commands to the TCL interpreter for 2157a0227fSdrh ** generating MD5 checksums: md5, md5file, 2257a0227fSdrh ** md5-10x8, and md5file-10x8. 2357a0227fSdrh ** 2457a0227fSdrh ** -DSQLITE_TEST When used in conjuction with -DTCLSH=1, add 2557a0227fSdrh ** hundreds of new commands used for testing 2657a0227fSdrh ** SQLite. This option implies -DSQLITE_TCLMD5. 2775897234Sdrh */ 28bd08af48Sdrh #include "tcl.h" 29b4e9af9fSdanielk1977 #include <errno.h> 306d31316cSdrh 31bd08af48Sdrh /* 32bd08af48Sdrh ** Some additional include files are needed if this file is not 33bd08af48Sdrh ** appended to the amalgamation. 34bd08af48Sdrh */ 35bd08af48Sdrh #ifndef SQLITE_AMALGAMATION 3665e8c82eSdrh # include "sqlite3.h" 3775897234Sdrh # include <stdlib.h> 3875897234Sdrh # include <string.h> 39ce927065Sdrh # include <assert.h> 4065e8c82eSdrh typedef unsigned char u8; 41bd08af48Sdrh #endif 42eb206381Sdrh #include <ctype.h> 4375897234Sdrh 44ad6e1370Sdrh /* 45ad6e1370Sdrh * Windows needs to know which symbols to export. Unix does not. 46ad6e1370Sdrh * BUILD_sqlite should be undefined for Unix. 47ad6e1370Sdrh */ 48ad6e1370Sdrh #ifdef BUILD_sqlite 49ad6e1370Sdrh #undef TCL_STORAGE_CLASS 50ad6e1370Sdrh #define TCL_STORAGE_CLASS DLLEXPORT 51ad6e1370Sdrh #endif /* BUILD_sqlite */ 5229bc4615Sdrh 53a21c6b6fSdanielk1977 #define NUM_PREPARED_STMTS 10 54fb7e7651Sdrh #define MAX_PREPARED_STMTS 100 55fb7e7651Sdrh 5675897234Sdrh /* 5798808babSdrh ** If TCL uses UTF-8 and SQLite is configured to use iso8859, then we 5898808babSdrh ** have to do a translation when going between the two. Set the 5998808babSdrh ** UTF_TRANSLATION_NEEDED macro to indicate that we need to do 6098808babSdrh ** this translation. 6198808babSdrh */ 6298808babSdrh #if defined(TCL_UTF_MAX) && !defined(SQLITE_UTF8) 6398808babSdrh # define UTF_TRANSLATION_NEEDED 1 6498808babSdrh #endif 6598808babSdrh 6698808babSdrh /* 67cabb0819Sdrh ** New SQL functions can be created as TCL scripts. Each such function 68cabb0819Sdrh ** is described by an instance of the following structure. 69cabb0819Sdrh */ 70cabb0819Sdrh typedef struct SqlFunc SqlFunc; 71cabb0819Sdrh struct SqlFunc { 72cabb0819Sdrh Tcl_Interp *interp; /* The TCL interpret to execute the function */ 73d1e4733dSdrh Tcl_Obj *pScript; /* The Tcl_Obj representation of the script */ 74d1e4733dSdrh int useEvalObjv; /* True if it is safe to use Tcl_EvalObjv */ 75d1e4733dSdrh char *zName; /* Name of this function */ 76cabb0819Sdrh SqlFunc *pNext; /* Next function on the list of them all */ 77cabb0819Sdrh }; 78cabb0819Sdrh 79cabb0819Sdrh /* 800202b29eSdanielk1977 ** New collation sequences function can be created as TCL scripts. Each such 810202b29eSdanielk1977 ** function is described by an instance of the following structure. 820202b29eSdanielk1977 */ 830202b29eSdanielk1977 typedef struct SqlCollate SqlCollate; 840202b29eSdanielk1977 struct SqlCollate { 850202b29eSdanielk1977 Tcl_Interp *interp; /* The TCL interpret to execute the function */ 860202b29eSdanielk1977 char *zScript; /* The script to be run */ 870202b29eSdanielk1977 SqlCollate *pNext; /* Next function on the list of them all */ 880202b29eSdanielk1977 }; 890202b29eSdanielk1977 900202b29eSdanielk1977 /* 91fb7e7651Sdrh ** Prepared statements are cached for faster execution. Each prepared 92fb7e7651Sdrh ** statement is described by an instance of the following structure. 93fb7e7651Sdrh */ 94fb7e7651Sdrh typedef struct SqlPreparedStmt SqlPreparedStmt; 95fb7e7651Sdrh struct SqlPreparedStmt { 96fb7e7651Sdrh SqlPreparedStmt *pNext; /* Next in linked list */ 97fb7e7651Sdrh SqlPreparedStmt *pPrev; /* Previous on the list */ 98fb7e7651Sdrh sqlite3_stmt *pStmt; /* The prepared statement */ 99fb7e7651Sdrh int nSql; /* chars in zSql[] */ 100d0e2a854Sdanielk1977 const char *zSql; /* Text of the SQL statement */ 1014a4c11aaSdan int nParm; /* Size of apParm array */ 1024a4c11aaSdan Tcl_Obj **apParm; /* Array of referenced object pointers */ 103fb7e7651Sdrh }; 104fb7e7651Sdrh 105d0441796Sdanielk1977 typedef struct IncrblobChannel IncrblobChannel; 106d0441796Sdanielk1977 107fb7e7651Sdrh /* 108bec3f402Sdrh ** There is one instance of this structure for each SQLite database 109bec3f402Sdrh ** that has been opened by the SQLite TCL interface. 110bec3f402Sdrh */ 111bec3f402Sdrh typedef struct SqliteDb SqliteDb; 112bec3f402Sdrh struct SqliteDb { 113dddca286Sdrh sqlite3 *db; /* The "real" database structure. MUST BE FIRST */ 114bec3f402Sdrh Tcl_Interp *interp; /* The interpreter used for this database */ 1156d31316cSdrh char *zBusy; /* The busy callback routine */ 116aa940eacSdrh char *zCommit; /* The commit hook callback routine */ 117b5a20d3cSdrh char *zTrace; /* The trace callback routine */ 11819e2d37fSdrh char *zProfile; /* The profile callback routine */ 119348bb5d6Sdanielk1977 char *zProgress; /* The progress callback routine */ 120e22a334bSdrh char *zAuth; /* The authorization callback routine */ 1211f1549f8Sdrh int disableAuth; /* Disable the authorizer if it exists */ 12255c45f2eSdanielk1977 char *zNull; /* Text to substitute for an SQL NULL value */ 123cabb0819Sdrh SqlFunc *pFunc; /* List of SQL functions */ 12494eb6a14Sdanielk1977 Tcl_Obj *pUpdateHook; /* Update hook script (if any) */ 12571fd80bfSdanielk1977 Tcl_Obj *pRollbackHook; /* Rollback hook script (if any) */ 126404ca075Sdanielk1977 Tcl_Obj *pUnlockNotify; /* Unlock notify script (if any) */ 1270202b29eSdanielk1977 SqlCollate *pCollate; /* List of SQL collation functions */ 1286f8a503dSdanielk1977 int rc; /* Return code of most recent sqlite3_exec() */ 1297cedc8d4Sdanielk1977 Tcl_Obj *pCollateNeeded; /* Collation needed script */ 130fb7e7651Sdrh SqlPreparedStmt *stmtList; /* List of prepared statements*/ 131fb7e7651Sdrh SqlPreparedStmt *stmtLast; /* Last statement in the list */ 132fb7e7651Sdrh int maxStmt; /* The next maximum number of stmtList */ 133fb7e7651Sdrh int nStmt; /* Number of statements in stmtList */ 134d0441796Sdanielk1977 IncrblobChannel *pIncrblob;/* Linked list of open incrblob channels */ 135d1d38488Sdrh int nStep, nSort; /* Statistics for most recent operation */ 136cd38d520Sdanielk1977 int nTransaction; /* Number of nested [transaction] methods */ 13798808babSdrh }; 138297ecf14Sdrh 139b4e9af9fSdanielk1977 struct IncrblobChannel { 140d0441796Sdanielk1977 sqlite3_blob *pBlob; /* sqlite3 blob handle */ 141dcbb5d3fSdanielk1977 SqliteDb *pDb; /* Associated database connection */ 142b4e9af9fSdanielk1977 int iSeek; /* Current seek offset */ 143d0441796Sdanielk1977 Tcl_Channel channel; /* Channel identifier */ 144d0441796Sdanielk1977 IncrblobChannel *pNext; /* Linked list of all open incrblob channels */ 145d0441796Sdanielk1977 IncrblobChannel *pPrev; /* Linked list of all open incrblob channels */ 146b4e9af9fSdanielk1977 }; 147b4e9af9fSdanielk1977 148ea678832Sdrh /* 149ea678832Sdrh ** Compute a string length that is limited to what can be stored in 150ea678832Sdrh ** lower 30 bits of a 32-bit signed integer. 151ea678832Sdrh */ 1524f21c4afSdrh static int strlen30(const char *z){ 153ea678832Sdrh const char *z2 = z; 154ea678832Sdrh while( *z2 ){ z2++; } 155ea678832Sdrh return 0x3fffffff & (int)(z2 - z); 156ea678832Sdrh } 157ea678832Sdrh 158ea678832Sdrh 15932a0d8bbSdanielk1977 #ifndef SQLITE_OMIT_INCRBLOB 160b4e9af9fSdanielk1977 /* 161d0441796Sdanielk1977 ** Close all incrblob channels opened using database connection pDb. 162d0441796Sdanielk1977 ** This is called when shutting down the database connection. 163d0441796Sdanielk1977 */ 164d0441796Sdanielk1977 static void closeIncrblobChannels(SqliteDb *pDb){ 165d0441796Sdanielk1977 IncrblobChannel *p; 166d0441796Sdanielk1977 IncrblobChannel *pNext; 167d0441796Sdanielk1977 168d0441796Sdanielk1977 for(p=pDb->pIncrblob; p; p=pNext){ 169d0441796Sdanielk1977 pNext = p->pNext; 170d0441796Sdanielk1977 171d0441796Sdanielk1977 /* Note: Calling unregister here call Tcl_Close on the incrblob channel, 172d0441796Sdanielk1977 ** which deletes the IncrblobChannel structure at *p. So do not 173d0441796Sdanielk1977 ** call Tcl_Free() here. 174d0441796Sdanielk1977 */ 175d0441796Sdanielk1977 Tcl_UnregisterChannel(pDb->interp, p->channel); 176d0441796Sdanielk1977 } 177d0441796Sdanielk1977 } 178d0441796Sdanielk1977 179d0441796Sdanielk1977 /* 180b4e9af9fSdanielk1977 ** Close an incremental blob channel. 181b4e9af9fSdanielk1977 */ 182b4e9af9fSdanielk1977 static int incrblobClose(ClientData instanceData, Tcl_Interp *interp){ 183b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 18492d4d7a9Sdanielk1977 int rc = sqlite3_blob_close(p->pBlob); 18592d4d7a9Sdanielk1977 sqlite3 *db = p->pDb->db; 186d0441796Sdanielk1977 187d0441796Sdanielk1977 /* Remove the channel from the SqliteDb.pIncrblob list. */ 188d0441796Sdanielk1977 if( p->pNext ){ 189d0441796Sdanielk1977 p->pNext->pPrev = p->pPrev; 190d0441796Sdanielk1977 } 191d0441796Sdanielk1977 if( p->pPrev ){ 192d0441796Sdanielk1977 p->pPrev->pNext = p->pNext; 193d0441796Sdanielk1977 } 194d0441796Sdanielk1977 if( p->pDb->pIncrblob==p ){ 195d0441796Sdanielk1977 p->pDb->pIncrblob = p->pNext; 196d0441796Sdanielk1977 } 197d0441796Sdanielk1977 19892d4d7a9Sdanielk1977 /* Free the IncrblobChannel structure */ 199b4e9af9fSdanielk1977 Tcl_Free((char *)p); 20092d4d7a9Sdanielk1977 20192d4d7a9Sdanielk1977 if( rc!=SQLITE_OK ){ 20292d4d7a9Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(db), TCL_VOLATILE); 20392d4d7a9Sdanielk1977 return TCL_ERROR; 20492d4d7a9Sdanielk1977 } 205b4e9af9fSdanielk1977 return TCL_OK; 206b4e9af9fSdanielk1977 } 207b4e9af9fSdanielk1977 208b4e9af9fSdanielk1977 /* 209b4e9af9fSdanielk1977 ** Read data from an incremental blob channel. 210b4e9af9fSdanielk1977 */ 211b4e9af9fSdanielk1977 static int incrblobInput( 212b4e9af9fSdanielk1977 ClientData instanceData, 213b4e9af9fSdanielk1977 char *buf, 214b4e9af9fSdanielk1977 int bufSize, 215b4e9af9fSdanielk1977 int *errorCodePtr 216b4e9af9fSdanielk1977 ){ 217b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 218b4e9af9fSdanielk1977 int nRead = bufSize; /* Number of bytes to read */ 219b4e9af9fSdanielk1977 int nBlob; /* Total size of the blob */ 220b4e9af9fSdanielk1977 int rc; /* sqlite error code */ 221b4e9af9fSdanielk1977 222b4e9af9fSdanielk1977 nBlob = sqlite3_blob_bytes(p->pBlob); 223b4e9af9fSdanielk1977 if( (p->iSeek+nRead)>nBlob ){ 224b4e9af9fSdanielk1977 nRead = nBlob-p->iSeek; 225b4e9af9fSdanielk1977 } 226b4e9af9fSdanielk1977 if( nRead<=0 ){ 227b4e9af9fSdanielk1977 return 0; 228b4e9af9fSdanielk1977 } 229b4e9af9fSdanielk1977 230b4e9af9fSdanielk1977 rc = sqlite3_blob_read(p->pBlob, (void *)buf, nRead, p->iSeek); 231b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 232b4e9af9fSdanielk1977 *errorCodePtr = rc; 233b4e9af9fSdanielk1977 return -1; 234b4e9af9fSdanielk1977 } 235b4e9af9fSdanielk1977 236b4e9af9fSdanielk1977 p->iSeek += nRead; 237b4e9af9fSdanielk1977 return nRead; 238b4e9af9fSdanielk1977 } 239b4e9af9fSdanielk1977 240d0441796Sdanielk1977 /* 241d0441796Sdanielk1977 ** Write data to an incremental blob channel. 242d0441796Sdanielk1977 */ 243b4e9af9fSdanielk1977 static int incrblobOutput( 244b4e9af9fSdanielk1977 ClientData instanceData, 245b4e9af9fSdanielk1977 CONST char *buf, 246b4e9af9fSdanielk1977 int toWrite, 247b4e9af9fSdanielk1977 int *errorCodePtr 248b4e9af9fSdanielk1977 ){ 249b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 250b4e9af9fSdanielk1977 int nWrite = toWrite; /* Number of bytes to write */ 251b4e9af9fSdanielk1977 int nBlob; /* Total size of the blob */ 252b4e9af9fSdanielk1977 int rc; /* sqlite error code */ 253b4e9af9fSdanielk1977 254b4e9af9fSdanielk1977 nBlob = sqlite3_blob_bytes(p->pBlob); 255b4e9af9fSdanielk1977 if( (p->iSeek+nWrite)>nBlob ){ 256b4e9af9fSdanielk1977 *errorCodePtr = EINVAL; 257b4e9af9fSdanielk1977 return -1; 258b4e9af9fSdanielk1977 } 259b4e9af9fSdanielk1977 if( nWrite<=0 ){ 260b4e9af9fSdanielk1977 return 0; 261b4e9af9fSdanielk1977 } 262b4e9af9fSdanielk1977 263b4e9af9fSdanielk1977 rc = sqlite3_blob_write(p->pBlob, (void *)buf, nWrite, p->iSeek); 264b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 265b4e9af9fSdanielk1977 *errorCodePtr = EIO; 266b4e9af9fSdanielk1977 return -1; 267b4e9af9fSdanielk1977 } 268b4e9af9fSdanielk1977 269b4e9af9fSdanielk1977 p->iSeek += nWrite; 270b4e9af9fSdanielk1977 return nWrite; 271b4e9af9fSdanielk1977 } 272b4e9af9fSdanielk1977 273b4e9af9fSdanielk1977 /* 274b4e9af9fSdanielk1977 ** Seek an incremental blob channel. 275b4e9af9fSdanielk1977 */ 276b4e9af9fSdanielk1977 static int incrblobSeek( 277b4e9af9fSdanielk1977 ClientData instanceData, 278b4e9af9fSdanielk1977 long offset, 279b4e9af9fSdanielk1977 int seekMode, 280b4e9af9fSdanielk1977 int *errorCodePtr 281b4e9af9fSdanielk1977 ){ 282b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 283b4e9af9fSdanielk1977 284b4e9af9fSdanielk1977 switch( seekMode ){ 285b4e9af9fSdanielk1977 case SEEK_SET: 286b4e9af9fSdanielk1977 p->iSeek = offset; 287b4e9af9fSdanielk1977 break; 288b4e9af9fSdanielk1977 case SEEK_CUR: 289b4e9af9fSdanielk1977 p->iSeek += offset; 290b4e9af9fSdanielk1977 break; 291b4e9af9fSdanielk1977 case SEEK_END: 292b4e9af9fSdanielk1977 p->iSeek = sqlite3_blob_bytes(p->pBlob) + offset; 293b4e9af9fSdanielk1977 break; 294b4e9af9fSdanielk1977 295b4e9af9fSdanielk1977 default: assert(!"Bad seekMode"); 296b4e9af9fSdanielk1977 } 297b4e9af9fSdanielk1977 298b4e9af9fSdanielk1977 return p->iSeek; 299b4e9af9fSdanielk1977 } 300b4e9af9fSdanielk1977 301b4e9af9fSdanielk1977 302b4e9af9fSdanielk1977 static void incrblobWatch(ClientData instanceData, int mode){ 303b4e9af9fSdanielk1977 /* NO-OP */ 304b4e9af9fSdanielk1977 } 305b4e9af9fSdanielk1977 static int incrblobHandle(ClientData instanceData, int dir, ClientData *hPtr){ 306b4e9af9fSdanielk1977 return TCL_ERROR; 307b4e9af9fSdanielk1977 } 308b4e9af9fSdanielk1977 309b4e9af9fSdanielk1977 static Tcl_ChannelType IncrblobChannelType = { 310b4e9af9fSdanielk1977 "incrblob", /* typeName */ 311b4e9af9fSdanielk1977 TCL_CHANNEL_VERSION_2, /* version */ 312b4e9af9fSdanielk1977 incrblobClose, /* closeProc */ 313b4e9af9fSdanielk1977 incrblobInput, /* inputProc */ 314b4e9af9fSdanielk1977 incrblobOutput, /* outputProc */ 315b4e9af9fSdanielk1977 incrblobSeek, /* seekProc */ 316b4e9af9fSdanielk1977 0, /* setOptionProc */ 317b4e9af9fSdanielk1977 0, /* getOptionProc */ 318b4e9af9fSdanielk1977 incrblobWatch, /* watchProc (this is a no-op) */ 319b4e9af9fSdanielk1977 incrblobHandle, /* getHandleProc (always returns error) */ 320b4e9af9fSdanielk1977 0, /* close2Proc */ 321b4e9af9fSdanielk1977 0, /* blockModeProc */ 322b4e9af9fSdanielk1977 0, /* flushProc */ 323b4e9af9fSdanielk1977 0, /* handlerProc */ 324b4e9af9fSdanielk1977 0, /* wideSeekProc */ 325b4e9af9fSdanielk1977 }; 326b4e9af9fSdanielk1977 327b4e9af9fSdanielk1977 /* 328b4e9af9fSdanielk1977 ** Create a new incrblob channel. 329b4e9af9fSdanielk1977 */ 330b4e9af9fSdanielk1977 static int createIncrblobChannel( 331b4e9af9fSdanielk1977 Tcl_Interp *interp, 332b4e9af9fSdanielk1977 SqliteDb *pDb, 333b4e9af9fSdanielk1977 const char *zDb, 334b4e9af9fSdanielk1977 const char *zTable, 335b4e9af9fSdanielk1977 const char *zColumn, 3368cbadb02Sdanielk1977 sqlite_int64 iRow, 3378cbadb02Sdanielk1977 int isReadonly 338b4e9af9fSdanielk1977 ){ 339b4e9af9fSdanielk1977 IncrblobChannel *p; 3408cbadb02Sdanielk1977 sqlite3 *db = pDb->db; 341b4e9af9fSdanielk1977 sqlite3_blob *pBlob; 342b4e9af9fSdanielk1977 int rc; 3438cbadb02Sdanielk1977 int flags = TCL_READABLE|(isReadonly ? 0 : TCL_WRITABLE); 344b4e9af9fSdanielk1977 345b4e9af9fSdanielk1977 /* This variable is used to name the channels: "incrblob_[incr count]" */ 346b4e9af9fSdanielk1977 static int count = 0; 347b4e9af9fSdanielk1977 char zChannel[64]; 348b4e9af9fSdanielk1977 3498cbadb02Sdanielk1977 rc = sqlite3_blob_open(db, zDb, zTable, zColumn, iRow, !isReadonly, &pBlob); 350b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 351b4e9af9fSdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 352b4e9af9fSdanielk1977 return TCL_ERROR; 353b4e9af9fSdanielk1977 } 354b4e9af9fSdanielk1977 355b4e9af9fSdanielk1977 p = (IncrblobChannel *)Tcl_Alloc(sizeof(IncrblobChannel)); 356b4e9af9fSdanielk1977 p->iSeek = 0; 357b4e9af9fSdanielk1977 p->pBlob = pBlob; 358b4e9af9fSdanielk1977 3595bb3eb9bSdrh sqlite3_snprintf(sizeof(zChannel), zChannel, "incrblob_%d", ++count); 360d0441796Sdanielk1977 p->channel = Tcl_CreateChannel(&IncrblobChannelType, zChannel, p, flags); 361d0441796Sdanielk1977 Tcl_RegisterChannel(interp, p->channel); 362b4e9af9fSdanielk1977 363d0441796Sdanielk1977 /* Link the new channel into the SqliteDb.pIncrblob list. */ 364d0441796Sdanielk1977 p->pNext = pDb->pIncrblob; 365d0441796Sdanielk1977 p->pPrev = 0; 366d0441796Sdanielk1977 if( p->pNext ){ 367d0441796Sdanielk1977 p->pNext->pPrev = p; 368d0441796Sdanielk1977 } 369d0441796Sdanielk1977 pDb->pIncrblob = p; 370d0441796Sdanielk1977 p->pDb = pDb; 371d0441796Sdanielk1977 372d0441796Sdanielk1977 Tcl_SetResult(interp, (char *)Tcl_GetChannelName(p->channel), TCL_VOLATILE); 373b4e9af9fSdanielk1977 return TCL_OK; 374b4e9af9fSdanielk1977 } 37532a0d8bbSdanielk1977 #else /* else clause for "#ifndef SQLITE_OMIT_INCRBLOB" */ 37632a0d8bbSdanielk1977 #define closeIncrblobChannels(pDb) 37732a0d8bbSdanielk1977 #endif 378b4e9af9fSdanielk1977 3796d31316cSdrh /* 380d1e4733dSdrh ** Look at the script prefix in pCmd. We will be executing this script 381d1e4733dSdrh ** after first appending one or more arguments. This routine analyzes 382d1e4733dSdrh ** the script to see if it is safe to use Tcl_EvalObjv() on the script 383d1e4733dSdrh ** rather than the more general Tcl_EvalEx(). Tcl_EvalObjv() is much 384d1e4733dSdrh ** faster. 385d1e4733dSdrh ** 386d1e4733dSdrh ** Scripts that are safe to use with Tcl_EvalObjv() consists of a 387d1e4733dSdrh ** command name followed by zero or more arguments with no [...] or $ 388d1e4733dSdrh ** or {...} or ; to be seen anywhere. Most callback scripts consist 389d1e4733dSdrh ** of just a single procedure name and they meet this requirement. 390d1e4733dSdrh */ 391d1e4733dSdrh static int safeToUseEvalObjv(Tcl_Interp *interp, Tcl_Obj *pCmd){ 392d1e4733dSdrh /* We could try to do something with Tcl_Parse(). But we will instead 393d1e4733dSdrh ** just do a search for forbidden characters. If any of the forbidden 394d1e4733dSdrh ** characters appear in pCmd, we will report the string as unsafe. 395d1e4733dSdrh */ 396d1e4733dSdrh const char *z; 397d1e4733dSdrh int n; 398d1e4733dSdrh z = Tcl_GetStringFromObj(pCmd, &n); 399d1e4733dSdrh while( n-- > 0 ){ 400d1e4733dSdrh int c = *(z++); 401d1e4733dSdrh if( c=='$' || c=='[' || c==';' ) return 0; 402d1e4733dSdrh } 403d1e4733dSdrh return 1; 404d1e4733dSdrh } 405d1e4733dSdrh 406d1e4733dSdrh /* 407d1e4733dSdrh ** Find an SqlFunc structure with the given name. Or create a new 408d1e4733dSdrh ** one if an existing one cannot be found. Return a pointer to the 409d1e4733dSdrh ** structure. 410d1e4733dSdrh */ 411d1e4733dSdrh static SqlFunc *findSqlFunc(SqliteDb *pDb, const char *zName){ 412d1e4733dSdrh SqlFunc *p, *pNew; 413d1e4733dSdrh int i; 4144f21c4afSdrh pNew = (SqlFunc*)Tcl_Alloc( sizeof(*pNew) + strlen30(zName) + 1 ); 415d1e4733dSdrh pNew->zName = (char*)&pNew[1]; 416d1e4733dSdrh for(i=0; zName[i]; i++){ pNew->zName[i] = tolower(zName[i]); } 417d1e4733dSdrh pNew->zName[i] = 0; 418d1e4733dSdrh for(p=pDb->pFunc; p; p=p->pNext){ 419d1e4733dSdrh if( strcmp(p->zName, pNew->zName)==0 ){ 420d1e4733dSdrh Tcl_Free((char*)pNew); 421d1e4733dSdrh return p; 422d1e4733dSdrh } 423d1e4733dSdrh } 424d1e4733dSdrh pNew->interp = pDb->interp; 425d1e4733dSdrh pNew->pScript = 0; 426d1e4733dSdrh pNew->pNext = pDb->pFunc; 427d1e4733dSdrh pDb->pFunc = pNew; 428d1e4733dSdrh return pNew; 429d1e4733dSdrh } 430d1e4733dSdrh 431d1e4733dSdrh /* 432fb7e7651Sdrh ** Finalize and free a list of prepared statements 433fb7e7651Sdrh */ 434fb7e7651Sdrh static void flushStmtCache( SqliteDb *pDb ){ 435fb7e7651Sdrh SqlPreparedStmt *pPreStmt; 436fb7e7651Sdrh 437fb7e7651Sdrh while( pDb->stmtList ){ 438fb7e7651Sdrh sqlite3_finalize( pDb->stmtList->pStmt ); 439fb7e7651Sdrh pPreStmt = pDb->stmtList; 440fb7e7651Sdrh pDb->stmtList = pDb->stmtList->pNext; 441fb7e7651Sdrh Tcl_Free( (char*)pPreStmt ); 442fb7e7651Sdrh } 443fb7e7651Sdrh pDb->nStmt = 0; 444fb7e7651Sdrh pDb->stmtLast = 0; 445fb7e7651Sdrh } 446fb7e7651Sdrh 447fb7e7651Sdrh /* 448895d7472Sdrh ** TCL calls this procedure when an sqlite3 database command is 449895d7472Sdrh ** deleted. 45075897234Sdrh */ 45175897234Sdrh static void DbDeleteCmd(void *db){ 452bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)db; 453fb7e7651Sdrh flushStmtCache(pDb); 454d0441796Sdanielk1977 closeIncrblobChannels(pDb); 4556f8a503dSdanielk1977 sqlite3_close(pDb->db); 456cabb0819Sdrh while( pDb->pFunc ){ 457cabb0819Sdrh SqlFunc *pFunc = pDb->pFunc; 458cabb0819Sdrh pDb->pFunc = pFunc->pNext; 459d1e4733dSdrh Tcl_DecrRefCount(pFunc->pScript); 460cabb0819Sdrh Tcl_Free((char*)pFunc); 461cabb0819Sdrh } 4620202b29eSdanielk1977 while( pDb->pCollate ){ 4630202b29eSdanielk1977 SqlCollate *pCollate = pDb->pCollate; 4640202b29eSdanielk1977 pDb->pCollate = pCollate->pNext; 4650202b29eSdanielk1977 Tcl_Free((char*)pCollate); 4660202b29eSdanielk1977 } 467bec3f402Sdrh if( pDb->zBusy ){ 468bec3f402Sdrh Tcl_Free(pDb->zBusy); 469bec3f402Sdrh } 470b5a20d3cSdrh if( pDb->zTrace ){ 471b5a20d3cSdrh Tcl_Free(pDb->zTrace); 4720d1a643aSdrh } 47319e2d37fSdrh if( pDb->zProfile ){ 47419e2d37fSdrh Tcl_Free(pDb->zProfile); 47519e2d37fSdrh } 476e22a334bSdrh if( pDb->zAuth ){ 477e22a334bSdrh Tcl_Free(pDb->zAuth); 478e22a334bSdrh } 47955c45f2eSdanielk1977 if( pDb->zNull ){ 48055c45f2eSdanielk1977 Tcl_Free(pDb->zNull); 48155c45f2eSdanielk1977 } 48294eb6a14Sdanielk1977 if( pDb->pUpdateHook ){ 48394eb6a14Sdanielk1977 Tcl_DecrRefCount(pDb->pUpdateHook); 48494eb6a14Sdanielk1977 } 48571fd80bfSdanielk1977 if( pDb->pRollbackHook ){ 48671fd80bfSdanielk1977 Tcl_DecrRefCount(pDb->pRollbackHook); 48771fd80bfSdanielk1977 } 48894eb6a14Sdanielk1977 if( pDb->pCollateNeeded ){ 48994eb6a14Sdanielk1977 Tcl_DecrRefCount(pDb->pCollateNeeded); 49094eb6a14Sdanielk1977 } 491bec3f402Sdrh Tcl_Free((char*)pDb); 492bec3f402Sdrh } 493bec3f402Sdrh 494bec3f402Sdrh /* 495bec3f402Sdrh ** This routine is called when a database file is locked while trying 496bec3f402Sdrh ** to execute SQL. 497bec3f402Sdrh */ 4982a764eb0Sdanielk1977 static int DbBusyHandler(void *cd, int nTries){ 499bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)cd; 500bec3f402Sdrh int rc; 501bec3f402Sdrh char zVal[30]; 502bec3f402Sdrh 5035bb3eb9bSdrh sqlite3_snprintf(sizeof(zVal), zVal, "%d", nTries); 504d1e4733dSdrh rc = Tcl_VarEval(pDb->interp, pDb->zBusy, " ", zVal, (char*)0); 505bec3f402Sdrh if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 506bec3f402Sdrh return 0; 507bec3f402Sdrh } 508bec3f402Sdrh return 1; 50975897234Sdrh } 51075897234Sdrh 51126e4a8b1Sdrh #ifndef SQLITE_OMIT_PROGRESS_CALLBACK 51275897234Sdrh /* 513348bb5d6Sdanielk1977 ** This routine is invoked as the 'progress callback' for the database. 514348bb5d6Sdanielk1977 */ 515348bb5d6Sdanielk1977 static int DbProgressHandler(void *cd){ 516348bb5d6Sdanielk1977 SqliteDb *pDb = (SqliteDb*)cd; 517348bb5d6Sdanielk1977 int rc; 518348bb5d6Sdanielk1977 519348bb5d6Sdanielk1977 assert( pDb->zProgress ); 520348bb5d6Sdanielk1977 rc = Tcl_Eval(pDb->interp, pDb->zProgress); 521348bb5d6Sdanielk1977 if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 522348bb5d6Sdanielk1977 return 1; 523348bb5d6Sdanielk1977 } 524348bb5d6Sdanielk1977 return 0; 525348bb5d6Sdanielk1977 } 52626e4a8b1Sdrh #endif 527348bb5d6Sdanielk1977 528d1167393Sdrh #ifndef SQLITE_OMIT_TRACE 529348bb5d6Sdanielk1977 /* 530b5a20d3cSdrh ** This routine is called by the SQLite trace handler whenever a new 531b5a20d3cSdrh ** block of SQL is executed. The TCL script in pDb->zTrace is executed. 5320d1a643aSdrh */ 533b5a20d3cSdrh static void DbTraceHandler(void *cd, const char *zSql){ 5340d1a643aSdrh SqliteDb *pDb = (SqliteDb*)cd; 535b5a20d3cSdrh Tcl_DString str; 5360d1a643aSdrh 537b5a20d3cSdrh Tcl_DStringInit(&str); 538b5a20d3cSdrh Tcl_DStringAppend(&str, pDb->zTrace, -1); 539b5a20d3cSdrh Tcl_DStringAppendElement(&str, zSql); 540b5a20d3cSdrh Tcl_Eval(pDb->interp, Tcl_DStringValue(&str)); 541b5a20d3cSdrh Tcl_DStringFree(&str); 542b5a20d3cSdrh Tcl_ResetResult(pDb->interp); 5430d1a643aSdrh } 544d1167393Sdrh #endif 5450d1a643aSdrh 546d1167393Sdrh #ifndef SQLITE_OMIT_TRACE 5470d1a643aSdrh /* 54819e2d37fSdrh ** This routine is called by the SQLite profile handler after a statement 54919e2d37fSdrh ** SQL has executed. The TCL script in pDb->zProfile is evaluated. 55019e2d37fSdrh */ 55119e2d37fSdrh static void DbProfileHandler(void *cd, const char *zSql, sqlite_uint64 tm){ 55219e2d37fSdrh SqliteDb *pDb = (SqliteDb*)cd; 55319e2d37fSdrh Tcl_DString str; 55419e2d37fSdrh char zTm[100]; 55519e2d37fSdrh 55619e2d37fSdrh sqlite3_snprintf(sizeof(zTm)-1, zTm, "%lld", tm); 55719e2d37fSdrh Tcl_DStringInit(&str); 55819e2d37fSdrh Tcl_DStringAppend(&str, pDb->zProfile, -1); 55919e2d37fSdrh Tcl_DStringAppendElement(&str, zSql); 56019e2d37fSdrh Tcl_DStringAppendElement(&str, zTm); 56119e2d37fSdrh Tcl_Eval(pDb->interp, Tcl_DStringValue(&str)); 56219e2d37fSdrh Tcl_DStringFree(&str); 56319e2d37fSdrh Tcl_ResetResult(pDb->interp); 56419e2d37fSdrh } 565d1167393Sdrh #endif 56619e2d37fSdrh 56719e2d37fSdrh /* 568aa940eacSdrh ** This routine is called when a transaction is committed. The 569aa940eacSdrh ** TCL script in pDb->zCommit is executed. If it returns non-zero or 570aa940eacSdrh ** if it throws an exception, the transaction is rolled back instead 571aa940eacSdrh ** of being committed. 572aa940eacSdrh */ 573aa940eacSdrh static int DbCommitHandler(void *cd){ 574aa940eacSdrh SqliteDb *pDb = (SqliteDb*)cd; 575aa940eacSdrh int rc; 576aa940eacSdrh 577aa940eacSdrh rc = Tcl_Eval(pDb->interp, pDb->zCommit); 578aa940eacSdrh if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 579aa940eacSdrh return 1; 580aa940eacSdrh } 581aa940eacSdrh return 0; 582aa940eacSdrh } 583aa940eacSdrh 58471fd80bfSdanielk1977 static void DbRollbackHandler(void *clientData){ 58571fd80bfSdanielk1977 SqliteDb *pDb = (SqliteDb*)clientData; 58671fd80bfSdanielk1977 assert(pDb->pRollbackHook); 58771fd80bfSdanielk1977 if( TCL_OK!=Tcl_EvalObjEx(pDb->interp, pDb->pRollbackHook, 0) ){ 58871fd80bfSdanielk1977 Tcl_BackgroundError(pDb->interp); 58971fd80bfSdanielk1977 } 59071fd80bfSdanielk1977 } 59171fd80bfSdanielk1977 592bcf4f484Sdrh #if defined(SQLITE_TEST) && defined(SQLITE_ENABLE_UNLOCK_NOTIFY) 593404ca075Sdanielk1977 static void setTestUnlockNotifyVars(Tcl_Interp *interp, int iArg, int nArg){ 594404ca075Sdanielk1977 char zBuf[64]; 595404ca075Sdanielk1977 sprintf(zBuf, "%d", iArg); 596404ca075Sdanielk1977 Tcl_SetVar(interp, "sqlite_unlock_notify_arg", zBuf, TCL_GLOBAL_ONLY); 597404ca075Sdanielk1977 sprintf(zBuf, "%d", nArg); 598404ca075Sdanielk1977 Tcl_SetVar(interp, "sqlite_unlock_notify_argcount", zBuf, TCL_GLOBAL_ONLY); 599404ca075Sdanielk1977 } 600404ca075Sdanielk1977 #else 601404ca075Sdanielk1977 # define setTestUnlockNotifyVars(x,y,z) 602404ca075Sdanielk1977 #endif 603404ca075Sdanielk1977 60469910da9Sdrh #ifdef SQLITE_ENABLE_UNLOCK_NOTIFY 605404ca075Sdanielk1977 static void DbUnlockNotify(void **apArg, int nArg){ 606404ca075Sdanielk1977 int i; 607404ca075Sdanielk1977 for(i=0; i<nArg; i++){ 608404ca075Sdanielk1977 const int flags = (TCL_EVAL_GLOBAL|TCL_EVAL_DIRECT); 609404ca075Sdanielk1977 SqliteDb *pDb = (SqliteDb *)apArg[i]; 610404ca075Sdanielk1977 setTestUnlockNotifyVars(pDb->interp, i, nArg); 611404ca075Sdanielk1977 assert( pDb->pUnlockNotify); 612404ca075Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pDb->pUnlockNotify, flags); 613404ca075Sdanielk1977 Tcl_DecrRefCount(pDb->pUnlockNotify); 614404ca075Sdanielk1977 pDb->pUnlockNotify = 0; 615404ca075Sdanielk1977 } 616404ca075Sdanielk1977 } 61769910da9Sdrh #endif 618404ca075Sdanielk1977 61994eb6a14Sdanielk1977 static void DbUpdateHandler( 62094eb6a14Sdanielk1977 void *p, 62194eb6a14Sdanielk1977 int op, 62294eb6a14Sdanielk1977 const char *zDb, 62394eb6a14Sdanielk1977 const char *zTbl, 62494eb6a14Sdanielk1977 sqlite_int64 rowid 62594eb6a14Sdanielk1977 ){ 62694eb6a14Sdanielk1977 SqliteDb *pDb = (SqliteDb *)p; 62794eb6a14Sdanielk1977 Tcl_Obj *pCmd; 62894eb6a14Sdanielk1977 62994eb6a14Sdanielk1977 assert( pDb->pUpdateHook ); 63094eb6a14Sdanielk1977 assert( op==SQLITE_INSERT || op==SQLITE_UPDATE || op==SQLITE_DELETE ); 63194eb6a14Sdanielk1977 63294eb6a14Sdanielk1977 pCmd = Tcl_DuplicateObj(pDb->pUpdateHook); 63394eb6a14Sdanielk1977 Tcl_IncrRefCount(pCmd); 63494eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj( 63594eb6a14Sdanielk1977 ( (op==SQLITE_INSERT)?"INSERT":(op==SQLITE_UPDATE)?"UPDATE":"DELETE"), -1)); 63694eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj(zDb, -1)); 63794eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj(zTbl, -1)); 63894eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewWideIntObj(rowid)); 63994eb6a14Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pCmd, TCL_EVAL_DIRECT); 64094eb6a14Sdanielk1977 } 64194eb6a14Sdanielk1977 6427cedc8d4Sdanielk1977 static void tclCollateNeeded( 6437cedc8d4Sdanielk1977 void *pCtx, 6449bb575fdSdrh sqlite3 *db, 6457cedc8d4Sdanielk1977 int enc, 6467cedc8d4Sdanielk1977 const char *zName 6477cedc8d4Sdanielk1977 ){ 6487cedc8d4Sdanielk1977 SqliteDb *pDb = (SqliteDb *)pCtx; 6497cedc8d4Sdanielk1977 Tcl_Obj *pScript = Tcl_DuplicateObj(pDb->pCollateNeeded); 6507cedc8d4Sdanielk1977 Tcl_IncrRefCount(pScript); 6517cedc8d4Sdanielk1977 Tcl_ListObjAppendElement(0, pScript, Tcl_NewStringObj(zName, -1)); 6527cedc8d4Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pScript, 0); 6537cedc8d4Sdanielk1977 Tcl_DecrRefCount(pScript); 6547cedc8d4Sdanielk1977 } 6557cedc8d4Sdanielk1977 656aa940eacSdrh /* 6570202b29eSdanielk1977 ** This routine is called to evaluate an SQL collation function implemented 6580202b29eSdanielk1977 ** using TCL script. 6590202b29eSdanielk1977 */ 6600202b29eSdanielk1977 static int tclSqlCollate( 6610202b29eSdanielk1977 void *pCtx, 6620202b29eSdanielk1977 int nA, 6630202b29eSdanielk1977 const void *zA, 6640202b29eSdanielk1977 int nB, 6650202b29eSdanielk1977 const void *zB 6660202b29eSdanielk1977 ){ 6670202b29eSdanielk1977 SqlCollate *p = (SqlCollate *)pCtx; 6680202b29eSdanielk1977 Tcl_Obj *pCmd; 6690202b29eSdanielk1977 6700202b29eSdanielk1977 pCmd = Tcl_NewStringObj(p->zScript, -1); 6710202b29eSdanielk1977 Tcl_IncrRefCount(pCmd); 6720202b29eSdanielk1977 Tcl_ListObjAppendElement(p->interp, pCmd, Tcl_NewStringObj(zA, nA)); 6730202b29eSdanielk1977 Tcl_ListObjAppendElement(p->interp, pCmd, Tcl_NewStringObj(zB, nB)); 674d1e4733dSdrh Tcl_EvalObjEx(p->interp, pCmd, TCL_EVAL_DIRECT); 6750202b29eSdanielk1977 Tcl_DecrRefCount(pCmd); 6760202b29eSdanielk1977 return (atoi(Tcl_GetStringResult(p->interp))); 6770202b29eSdanielk1977 } 6780202b29eSdanielk1977 6790202b29eSdanielk1977 /* 680cabb0819Sdrh ** This routine is called to evaluate an SQL function implemented 681cabb0819Sdrh ** using TCL script. 682cabb0819Sdrh */ 6830ae8b831Sdanielk1977 static void tclSqlFunc(sqlite3_context *context, int argc, sqlite3_value**argv){ 6846f8a503dSdanielk1977 SqlFunc *p = sqlite3_user_data(context); 685d1e4733dSdrh Tcl_Obj *pCmd; 686cabb0819Sdrh int i; 687cabb0819Sdrh int rc; 688cabb0819Sdrh 689d1e4733dSdrh if( argc==0 ){ 690d1e4733dSdrh /* If there are no arguments to the function, call Tcl_EvalObjEx on the 691d1e4733dSdrh ** script object directly. This allows the TCL compiler to generate 692d1e4733dSdrh ** bytecode for the command on the first invocation and thus make 693d1e4733dSdrh ** subsequent invocations much faster. */ 694d1e4733dSdrh pCmd = p->pScript; 695d1e4733dSdrh Tcl_IncrRefCount(pCmd); 696d1e4733dSdrh rc = Tcl_EvalObjEx(p->interp, pCmd, 0); 697d1e4733dSdrh Tcl_DecrRefCount(pCmd); 69851ad0ecdSdanielk1977 }else{ 699d1e4733dSdrh /* If there are arguments to the function, make a shallow copy of the 700d1e4733dSdrh ** script object, lappend the arguments, then evaluate the copy. 701d1e4733dSdrh ** 702d1e4733dSdrh ** By "shallow" copy, we mean a only the outer list Tcl_Obj is duplicated. 703d1e4733dSdrh ** The new Tcl_Obj contains pointers to the original list elements. 704d1e4733dSdrh ** That way, when Tcl_EvalObjv() is run and shimmers the first element 705d1e4733dSdrh ** of the list to tclCmdNameType, that alternate representation will 706d1e4733dSdrh ** be preserved and reused on the next invocation. 707d1e4733dSdrh */ 708d1e4733dSdrh Tcl_Obj **aArg; 709d1e4733dSdrh int nArg; 710d1e4733dSdrh if( Tcl_ListObjGetElements(p->interp, p->pScript, &nArg, &aArg) ){ 711d1e4733dSdrh sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 712d1e4733dSdrh return; 713d1e4733dSdrh } 714d1e4733dSdrh pCmd = Tcl_NewListObj(nArg, aArg); 715d1e4733dSdrh Tcl_IncrRefCount(pCmd); 716d1e4733dSdrh for(i=0; i<argc; i++){ 717d1e4733dSdrh sqlite3_value *pIn = argv[i]; 718d1e4733dSdrh Tcl_Obj *pVal; 719d1e4733dSdrh 720d1e4733dSdrh /* Set pVal to contain the i'th column of this row. */ 721d1e4733dSdrh switch( sqlite3_value_type(pIn) ){ 722d1e4733dSdrh case SQLITE_BLOB: { 723d1e4733dSdrh int bytes = sqlite3_value_bytes(pIn); 724d1e4733dSdrh pVal = Tcl_NewByteArrayObj(sqlite3_value_blob(pIn), bytes); 725d1e4733dSdrh break; 726d1e4733dSdrh } 727d1e4733dSdrh case SQLITE_INTEGER: { 728d1e4733dSdrh sqlite_int64 v = sqlite3_value_int64(pIn); 729d1e4733dSdrh if( v>=-2147483647 && v<=2147483647 ){ 730d1e4733dSdrh pVal = Tcl_NewIntObj(v); 731d1e4733dSdrh }else{ 732d1e4733dSdrh pVal = Tcl_NewWideIntObj(v); 733d1e4733dSdrh } 734d1e4733dSdrh break; 735d1e4733dSdrh } 736d1e4733dSdrh case SQLITE_FLOAT: { 737d1e4733dSdrh double r = sqlite3_value_double(pIn); 738d1e4733dSdrh pVal = Tcl_NewDoubleObj(r); 739d1e4733dSdrh break; 740d1e4733dSdrh } 741d1e4733dSdrh case SQLITE_NULL: { 742d1e4733dSdrh pVal = Tcl_NewStringObj("", 0); 743d1e4733dSdrh break; 744d1e4733dSdrh } 745d1e4733dSdrh default: { 746d1e4733dSdrh int bytes = sqlite3_value_bytes(pIn); 74700fd957bSdanielk1977 pVal = Tcl_NewStringObj((char *)sqlite3_value_text(pIn), bytes); 748d1e4733dSdrh break; 74951ad0ecdSdanielk1977 } 750cabb0819Sdrh } 751d1e4733dSdrh rc = Tcl_ListObjAppendElement(p->interp, pCmd, pVal); 752d1e4733dSdrh if( rc ){ 753d1e4733dSdrh Tcl_DecrRefCount(pCmd); 754d1e4733dSdrh sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 755d1e4733dSdrh return; 756d1e4733dSdrh } 757d1e4733dSdrh } 758d1e4733dSdrh if( !p->useEvalObjv ){ 759d1e4733dSdrh /* Tcl_EvalObjEx() will automatically call Tcl_EvalObjv() if pCmd 760d1e4733dSdrh ** is a list without a string representation. To prevent this from 761d1e4733dSdrh ** happening, make sure pCmd has a valid string representation */ 762d1e4733dSdrh Tcl_GetString(pCmd); 763d1e4733dSdrh } 764d1e4733dSdrh rc = Tcl_EvalObjEx(p->interp, pCmd, TCL_EVAL_DIRECT); 765d1e4733dSdrh Tcl_DecrRefCount(pCmd); 766d1e4733dSdrh } 767562e8d3cSdanielk1977 768c7f269d5Sdrh if( rc && rc!=TCL_RETURN ){ 7697e18c259Sdanielk1977 sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 770cabb0819Sdrh }else{ 771c7f269d5Sdrh Tcl_Obj *pVar = Tcl_GetObjResult(p->interp); 772c7f269d5Sdrh int n; 773c7f269d5Sdrh u8 *data; 7744a4c11aaSdan const char *zType = (pVar->typePtr ? pVar->typePtr->name : ""); 775c7f269d5Sdrh char c = zType[0]; 776df0bddaeSdrh if( c=='b' && strcmp(zType,"bytearray")==0 && pVar->bytes==0 ){ 777d1e4733dSdrh /* Only return a BLOB type if the Tcl variable is a bytearray and 778df0bddaeSdrh ** has no string representation. */ 779c7f269d5Sdrh data = Tcl_GetByteArrayFromObj(pVar, &n); 780c7f269d5Sdrh sqlite3_result_blob(context, data, n, SQLITE_TRANSIENT); 781985e0c63Sdrh }else if( c=='b' && strcmp(zType,"boolean")==0 ){ 782c7f269d5Sdrh Tcl_GetIntFromObj(0, pVar, &n); 783c7f269d5Sdrh sqlite3_result_int(context, n); 784c7f269d5Sdrh }else if( c=='d' && strcmp(zType,"double")==0 ){ 785c7f269d5Sdrh double r; 786c7f269d5Sdrh Tcl_GetDoubleFromObj(0, pVar, &r); 787c7f269d5Sdrh sqlite3_result_double(context, r); 788985e0c63Sdrh }else if( (c=='w' && strcmp(zType,"wideInt")==0) || 789985e0c63Sdrh (c=='i' && strcmp(zType,"int")==0) ){ 790df0bddaeSdrh Tcl_WideInt v; 791df0bddaeSdrh Tcl_GetWideIntFromObj(0, pVar, &v); 792df0bddaeSdrh sqlite3_result_int64(context, v); 793c7f269d5Sdrh }else{ 79400fd957bSdanielk1977 data = (unsigned char *)Tcl_GetStringFromObj(pVar, &n); 79500fd957bSdanielk1977 sqlite3_result_text(context, (char *)data, n, SQLITE_TRANSIENT); 796c7f269d5Sdrh } 797cabb0819Sdrh } 798cabb0819Sdrh } 799895d7472Sdrh 800e22a334bSdrh #ifndef SQLITE_OMIT_AUTHORIZATION 801e22a334bSdrh /* 802e22a334bSdrh ** This is the authentication function. It appends the authentication 803e22a334bSdrh ** type code and the two arguments to zCmd[] then invokes the result 804e22a334bSdrh ** on the interpreter. The reply is examined to determine if the 805e22a334bSdrh ** authentication fails or succeeds. 806e22a334bSdrh */ 807e22a334bSdrh static int auth_callback( 808e22a334bSdrh void *pArg, 809e22a334bSdrh int code, 810e22a334bSdrh const char *zArg1, 811e22a334bSdrh const char *zArg2, 812e22a334bSdrh const char *zArg3, 813e22a334bSdrh const char *zArg4 814e22a334bSdrh ){ 815e22a334bSdrh char *zCode; 816e22a334bSdrh Tcl_DString str; 817e22a334bSdrh int rc; 818e22a334bSdrh const char *zReply; 819e22a334bSdrh SqliteDb *pDb = (SqliteDb*)pArg; 8201f1549f8Sdrh if( pDb->disableAuth ) return SQLITE_OK; 821e22a334bSdrh 822e22a334bSdrh switch( code ){ 823e22a334bSdrh case SQLITE_COPY : zCode="SQLITE_COPY"; break; 824e22a334bSdrh case SQLITE_CREATE_INDEX : zCode="SQLITE_CREATE_INDEX"; break; 825e22a334bSdrh case SQLITE_CREATE_TABLE : zCode="SQLITE_CREATE_TABLE"; break; 826e22a334bSdrh case SQLITE_CREATE_TEMP_INDEX : zCode="SQLITE_CREATE_TEMP_INDEX"; break; 827e22a334bSdrh case SQLITE_CREATE_TEMP_TABLE : zCode="SQLITE_CREATE_TEMP_TABLE"; break; 828e22a334bSdrh case SQLITE_CREATE_TEMP_TRIGGER: zCode="SQLITE_CREATE_TEMP_TRIGGER"; break; 829e22a334bSdrh case SQLITE_CREATE_TEMP_VIEW : zCode="SQLITE_CREATE_TEMP_VIEW"; break; 830e22a334bSdrh case SQLITE_CREATE_TRIGGER : zCode="SQLITE_CREATE_TRIGGER"; break; 831e22a334bSdrh case SQLITE_CREATE_VIEW : zCode="SQLITE_CREATE_VIEW"; break; 832e22a334bSdrh case SQLITE_DELETE : zCode="SQLITE_DELETE"; break; 833e22a334bSdrh case SQLITE_DROP_INDEX : zCode="SQLITE_DROP_INDEX"; break; 834e22a334bSdrh case SQLITE_DROP_TABLE : zCode="SQLITE_DROP_TABLE"; break; 835e22a334bSdrh case SQLITE_DROP_TEMP_INDEX : zCode="SQLITE_DROP_TEMP_INDEX"; break; 836e22a334bSdrh case SQLITE_DROP_TEMP_TABLE : zCode="SQLITE_DROP_TEMP_TABLE"; break; 837e22a334bSdrh case SQLITE_DROP_TEMP_TRIGGER : zCode="SQLITE_DROP_TEMP_TRIGGER"; break; 838e22a334bSdrh case SQLITE_DROP_TEMP_VIEW : zCode="SQLITE_DROP_TEMP_VIEW"; break; 839e22a334bSdrh case SQLITE_DROP_TRIGGER : zCode="SQLITE_DROP_TRIGGER"; break; 840e22a334bSdrh case SQLITE_DROP_VIEW : zCode="SQLITE_DROP_VIEW"; break; 841e22a334bSdrh case SQLITE_INSERT : zCode="SQLITE_INSERT"; break; 842e22a334bSdrh case SQLITE_PRAGMA : zCode="SQLITE_PRAGMA"; break; 843e22a334bSdrh case SQLITE_READ : zCode="SQLITE_READ"; break; 844e22a334bSdrh case SQLITE_SELECT : zCode="SQLITE_SELECT"; break; 845e22a334bSdrh case SQLITE_TRANSACTION : zCode="SQLITE_TRANSACTION"; break; 846e22a334bSdrh case SQLITE_UPDATE : zCode="SQLITE_UPDATE"; break; 84781e293b4Sdrh case SQLITE_ATTACH : zCode="SQLITE_ATTACH"; break; 84881e293b4Sdrh case SQLITE_DETACH : zCode="SQLITE_DETACH"; break; 8491c8c23ccSdanielk1977 case SQLITE_ALTER_TABLE : zCode="SQLITE_ALTER_TABLE"; break; 8501d54df88Sdanielk1977 case SQLITE_REINDEX : zCode="SQLITE_REINDEX"; break; 851e6e04969Sdrh case SQLITE_ANALYZE : zCode="SQLITE_ANALYZE"; break; 852f1a381e7Sdanielk1977 case SQLITE_CREATE_VTABLE : zCode="SQLITE_CREATE_VTABLE"; break; 853f1a381e7Sdanielk1977 case SQLITE_DROP_VTABLE : zCode="SQLITE_DROP_VTABLE"; break; 8545169bbc6Sdrh case SQLITE_FUNCTION : zCode="SQLITE_FUNCTION"; break; 855ab9b703fSdanielk1977 case SQLITE_SAVEPOINT : zCode="SQLITE_SAVEPOINT"; break; 856e22a334bSdrh default : zCode="????"; break; 857e22a334bSdrh } 858e22a334bSdrh Tcl_DStringInit(&str); 859e22a334bSdrh Tcl_DStringAppend(&str, pDb->zAuth, -1); 860e22a334bSdrh Tcl_DStringAppendElement(&str, zCode); 861e22a334bSdrh Tcl_DStringAppendElement(&str, zArg1 ? zArg1 : ""); 862e22a334bSdrh Tcl_DStringAppendElement(&str, zArg2 ? zArg2 : ""); 863e22a334bSdrh Tcl_DStringAppendElement(&str, zArg3 ? zArg3 : ""); 864e22a334bSdrh Tcl_DStringAppendElement(&str, zArg4 ? zArg4 : ""); 865e22a334bSdrh rc = Tcl_GlobalEval(pDb->interp, Tcl_DStringValue(&str)); 866e22a334bSdrh Tcl_DStringFree(&str); 867e22a334bSdrh zReply = Tcl_GetStringResult(pDb->interp); 868e22a334bSdrh if( strcmp(zReply,"SQLITE_OK")==0 ){ 869e22a334bSdrh rc = SQLITE_OK; 870e22a334bSdrh }else if( strcmp(zReply,"SQLITE_DENY")==0 ){ 871e22a334bSdrh rc = SQLITE_DENY; 872e22a334bSdrh }else if( strcmp(zReply,"SQLITE_IGNORE")==0 ){ 873e22a334bSdrh rc = SQLITE_IGNORE; 874e22a334bSdrh }else{ 875e22a334bSdrh rc = 999; 876e22a334bSdrh } 877e22a334bSdrh return rc; 878e22a334bSdrh } 879e22a334bSdrh #endif /* SQLITE_OMIT_AUTHORIZATION */ 880cabb0819Sdrh 881cabb0819Sdrh /* 882ef2cb63eSdanielk1977 ** zText is a pointer to text obtained via an sqlite3_result_text() 883ef2cb63eSdanielk1977 ** or similar interface. This routine returns a Tcl string object, 884ef2cb63eSdanielk1977 ** reference count set to 0, containing the text. If a translation 885ef2cb63eSdanielk1977 ** between iso8859 and UTF-8 is required, it is preformed. 886ef2cb63eSdanielk1977 */ 887ef2cb63eSdanielk1977 static Tcl_Obj *dbTextToObj(char const *zText){ 888ef2cb63eSdanielk1977 Tcl_Obj *pVal; 889ef2cb63eSdanielk1977 #ifdef UTF_TRANSLATION_NEEDED 890ef2cb63eSdanielk1977 Tcl_DString dCol; 891ef2cb63eSdanielk1977 Tcl_DStringInit(&dCol); 892ef2cb63eSdanielk1977 Tcl_ExternalToUtfDString(NULL, zText, -1, &dCol); 893ef2cb63eSdanielk1977 pVal = Tcl_NewStringObj(Tcl_DStringValue(&dCol), -1); 894ef2cb63eSdanielk1977 Tcl_DStringFree(&dCol); 895ef2cb63eSdanielk1977 #else 896ef2cb63eSdanielk1977 pVal = Tcl_NewStringObj(zText, -1); 897ef2cb63eSdanielk1977 #endif 898ef2cb63eSdanielk1977 return pVal; 899ef2cb63eSdanielk1977 } 900ef2cb63eSdanielk1977 901ef2cb63eSdanielk1977 /* 9021067fe11Stpoindex ** This routine reads a line of text from FILE in, stores 9031067fe11Stpoindex ** the text in memory obtained from malloc() and returns a pointer 9041067fe11Stpoindex ** to the text. NULL is returned at end of file, or if malloc() 9051067fe11Stpoindex ** fails. 9061067fe11Stpoindex ** 9071067fe11Stpoindex ** The interface is like "readline" but no command-line editing 9081067fe11Stpoindex ** is done. 9091067fe11Stpoindex ** 9101067fe11Stpoindex ** copied from shell.c from '.import' command 9111067fe11Stpoindex */ 9121067fe11Stpoindex static char *local_getline(char *zPrompt, FILE *in){ 9131067fe11Stpoindex char *zLine; 9141067fe11Stpoindex int nLine; 9151067fe11Stpoindex int n; 9161067fe11Stpoindex int eol; 9171067fe11Stpoindex 9181067fe11Stpoindex nLine = 100; 9191067fe11Stpoindex zLine = malloc( nLine ); 9201067fe11Stpoindex if( zLine==0 ) return 0; 9211067fe11Stpoindex n = 0; 9221067fe11Stpoindex eol = 0; 9231067fe11Stpoindex while( !eol ){ 9241067fe11Stpoindex if( n+100>nLine ){ 9251067fe11Stpoindex nLine = nLine*2 + 100; 9261067fe11Stpoindex zLine = realloc(zLine, nLine); 9271067fe11Stpoindex if( zLine==0 ) return 0; 9281067fe11Stpoindex } 9291067fe11Stpoindex if( fgets(&zLine[n], nLine - n, in)==0 ){ 9301067fe11Stpoindex if( n==0 ){ 9311067fe11Stpoindex free(zLine); 9321067fe11Stpoindex return 0; 9331067fe11Stpoindex } 9341067fe11Stpoindex zLine[n] = 0; 9351067fe11Stpoindex eol = 1; 9361067fe11Stpoindex break; 9371067fe11Stpoindex } 9381067fe11Stpoindex while( zLine[n] ){ n++; } 9391067fe11Stpoindex if( n>0 && zLine[n-1]=='\n' ){ 9401067fe11Stpoindex n--; 9411067fe11Stpoindex zLine[n] = 0; 9421067fe11Stpoindex eol = 1; 9431067fe11Stpoindex } 9441067fe11Stpoindex } 9451067fe11Stpoindex zLine = realloc( zLine, n+1 ); 9461067fe11Stpoindex return zLine; 9471067fe11Stpoindex } 9481067fe11Stpoindex 9498e556520Sdanielk1977 9508e556520Sdanielk1977 /* 9514a4c11aaSdan ** This function is part of the implementation of the command: 9528e556520Sdanielk1977 ** 9534a4c11aaSdan ** $db transaction [-deferred|-immediate|-exclusive] SCRIPT 9548e556520Sdanielk1977 ** 9554a4c11aaSdan ** It is invoked after evaluating the script SCRIPT to commit or rollback 9564a4c11aaSdan ** the transaction or savepoint opened by the [transaction] command. 9574a4c11aaSdan */ 9584a4c11aaSdan static int DbTransPostCmd( 9594a4c11aaSdan ClientData data[], /* data[0] is the Sqlite3Db* for $db */ 9604a4c11aaSdan Tcl_Interp *interp, /* Tcl interpreter */ 9614a4c11aaSdan int result /* Result of evaluating SCRIPT */ 9624a4c11aaSdan ){ 9634a4c11aaSdan static const char *azEnd[] = { 9644a4c11aaSdan "RELEASE _tcl_transaction", /* rc==TCL_ERROR, nTransaction!=0 */ 9654a4c11aaSdan "COMMIT", /* rc!=TCL_ERROR, nTransaction==0 */ 9664a4c11aaSdan "ROLLBACK TO _tcl_transaction ; RELEASE _tcl_transaction", 9674a4c11aaSdan "ROLLBACK" /* rc==TCL_ERROR, nTransaction==0 */ 9684a4c11aaSdan }; 9694a4c11aaSdan SqliteDb *pDb = (SqliteDb*)data[0]; 9704a4c11aaSdan int rc = result; 9714a4c11aaSdan const char *zEnd; 9724a4c11aaSdan 9734a4c11aaSdan pDb->nTransaction--; 9744a4c11aaSdan zEnd = azEnd[(rc==TCL_ERROR)*2 + (pDb->nTransaction==0)]; 9754a4c11aaSdan 9764a4c11aaSdan pDb->disableAuth++; 9774a4c11aaSdan if( sqlite3_exec(pDb->db, zEnd, 0, 0, 0) ){ 9784a4c11aaSdan /* This is a tricky scenario to handle. The most likely cause of an 9794a4c11aaSdan ** error is that the exec() above was an attempt to commit the 9804a4c11aaSdan ** top-level transaction that returned SQLITE_BUSY. Or, less likely, 9814a4c11aaSdan ** that an IO-error has occured. In either case, throw a Tcl exception 9824a4c11aaSdan ** and try to rollback the transaction. 9834a4c11aaSdan ** 9844a4c11aaSdan ** But it could also be that the user executed one or more BEGIN, 9854a4c11aaSdan ** COMMIT, SAVEPOINT, RELEASE or ROLLBACK commands that are confusing 9864a4c11aaSdan ** this method's logic. Not clear how this would be best handled. 9874a4c11aaSdan */ 9884a4c11aaSdan if( rc!=TCL_ERROR ){ 9894a4c11aaSdan Tcl_AppendResult(interp, sqlite3_errmsg(pDb->db), 0); 9904a4c11aaSdan rc = TCL_ERROR; 9914a4c11aaSdan } 9924a4c11aaSdan sqlite3_exec(pDb->db, "ROLLBACK", 0, 0, 0); 9934a4c11aaSdan } 9944a4c11aaSdan pDb->disableAuth--; 9954a4c11aaSdan 9964a4c11aaSdan return rc; 9974a4c11aaSdan } 9984a4c11aaSdan 9994a4c11aaSdan /* 10004a4c11aaSdan ** Search the cache for a prepared-statement object that implements the 10014a4c11aaSdan ** first SQL statement in the buffer pointed to by parameter zIn. If 10024a4c11aaSdan ** no such prepared-statement can be found, allocate and prepare a new 10034a4c11aaSdan ** one. In either case, bind the current values of the relevant Tcl 10044a4c11aaSdan ** variables to any $var, :var or @var variables in the statement. Before 10054a4c11aaSdan ** returning, set *ppPreStmt to point to the prepared-statement object. 10064a4c11aaSdan ** 10074a4c11aaSdan ** Output parameter *pzOut is set to point to the next SQL statement in 10084a4c11aaSdan ** buffer zIn, or to the '\0' byte at the end of zIn if there is no 10094a4c11aaSdan ** next statement. 10104a4c11aaSdan ** 10114a4c11aaSdan ** If successful, TCL_OK is returned. Otherwise, TCL_ERROR is returned 10124a4c11aaSdan ** and an error message loaded into interpreter pDb->interp. 10134a4c11aaSdan */ 10144a4c11aaSdan static int dbPrepareAndBind( 10154a4c11aaSdan SqliteDb *pDb, /* Database object */ 10164a4c11aaSdan char const *zIn, /* SQL to compile */ 10174a4c11aaSdan char const **pzOut, /* OUT: Pointer to next SQL statement */ 10184a4c11aaSdan SqlPreparedStmt **ppPreStmt /* OUT: Object used to cache statement */ 10194a4c11aaSdan ){ 10204a4c11aaSdan const char *zSql = zIn; /* Pointer to first SQL statement in zIn */ 10214a4c11aaSdan sqlite3_stmt *pStmt; /* Prepared statement object */ 10224a4c11aaSdan SqlPreparedStmt *pPreStmt; /* Pointer to cached statement */ 10234a4c11aaSdan int nSql; /* Length of zSql in bytes */ 10244a4c11aaSdan int nVar; /* Number of variables in statement */ 10254a4c11aaSdan int iParm = 0; /* Next free entry in apParm */ 10264a4c11aaSdan int i; 10274a4c11aaSdan Tcl_Interp *interp = pDb->interp; 10284a4c11aaSdan 10294a4c11aaSdan *ppPreStmt = 0; 10304a4c11aaSdan 10314a4c11aaSdan /* Trim spaces from the start of zSql and calculate the remaining length. */ 10324a4c11aaSdan while( isspace(zSql[0]) ){ zSql++; } 10334a4c11aaSdan nSql = strlen30(zSql); 10344a4c11aaSdan 10354a4c11aaSdan for(pPreStmt = pDb->stmtList; pPreStmt; pPreStmt=pPreStmt->pNext){ 10364a4c11aaSdan int n = pPreStmt->nSql; 10374a4c11aaSdan if( nSql>=n 10384a4c11aaSdan && memcmp(pPreStmt->zSql, zSql, n)==0 10394a4c11aaSdan && (zSql[n]==0 || zSql[n-1]==';') 10404a4c11aaSdan ){ 10414a4c11aaSdan pStmt = pPreStmt->pStmt; 10424a4c11aaSdan *pzOut = &zSql[pPreStmt->nSql]; 10434a4c11aaSdan 10444a4c11aaSdan /* When a prepared statement is found, unlink it from the 10454a4c11aaSdan ** cache list. It will later be added back to the beginning 10464a4c11aaSdan ** of the cache list in order to implement LRU replacement. 10474a4c11aaSdan */ 10484a4c11aaSdan if( pPreStmt->pPrev ){ 10494a4c11aaSdan pPreStmt->pPrev->pNext = pPreStmt->pNext; 10504a4c11aaSdan }else{ 10514a4c11aaSdan pDb->stmtList = pPreStmt->pNext; 10524a4c11aaSdan } 10534a4c11aaSdan if( pPreStmt->pNext ){ 10544a4c11aaSdan pPreStmt->pNext->pPrev = pPreStmt->pPrev; 10554a4c11aaSdan }else{ 10564a4c11aaSdan pDb->stmtLast = pPreStmt->pPrev; 10574a4c11aaSdan } 10584a4c11aaSdan pDb->nStmt--; 10594a4c11aaSdan nVar = sqlite3_bind_parameter_count(pStmt); 10604a4c11aaSdan break; 10614a4c11aaSdan } 10624a4c11aaSdan } 10634a4c11aaSdan 10644a4c11aaSdan /* If no prepared statement was found. Compile the SQL text. Also allocate 10654a4c11aaSdan ** a new SqlPreparedStmt structure. */ 10664a4c11aaSdan if( pPreStmt==0 ){ 10674a4c11aaSdan int nByte; 10684a4c11aaSdan 10694a4c11aaSdan if( SQLITE_OK!=sqlite3_prepare_v2(pDb->db, zSql, -1, &pStmt, pzOut) ){ 10704a4c11aaSdan Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 10714a4c11aaSdan return TCL_ERROR; 10724a4c11aaSdan } 10734a4c11aaSdan if( pStmt==0 ){ 10744a4c11aaSdan if( SQLITE_OK!=sqlite3_errcode(pDb->db) ){ 10754a4c11aaSdan /* A compile-time error in the statement. */ 10764a4c11aaSdan Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 10774a4c11aaSdan return TCL_ERROR; 10784a4c11aaSdan }else{ 10794a4c11aaSdan /* The statement was a no-op. Continue to the next statement 10804a4c11aaSdan ** in the SQL string. 10814a4c11aaSdan */ 10824a4c11aaSdan return TCL_OK; 10834a4c11aaSdan } 10844a4c11aaSdan } 10854a4c11aaSdan 10864a4c11aaSdan assert( pPreStmt==0 ); 10874a4c11aaSdan nVar = sqlite3_bind_parameter_count(pStmt); 10884a4c11aaSdan nByte = sizeof(SqlPreparedStmt) + nVar*sizeof(Tcl_Obj *); 10894a4c11aaSdan pPreStmt = (SqlPreparedStmt*)Tcl_Alloc(nByte); 10904a4c11aaSdan memset(pPreStmt, 0, nByte); 10914a4c11aaSdan 10924a4c11aaSdan pPreStmt->pStmt = pStmt; 10934a4c11aaSdan pPreStmt->nSql = (*pzOut - zSql); 10944a4c11aaSdan pPreStmt->zSql = sqlite3_sql(pStmt); 10954a4c11aaSdan pPreStmt->apParm = (Tcl_Obj **)&pPreStmt[1]; 10964a4c11aaSdan } 10974a4c11aaSdan assert( pPreStmt ); 10984a4c11aaSdan assert( strlen30(pPreStmt->zSql)==pPreStmt->nSql ); 10994a4c11aaSdan assert( 0==memcmp(pPreStmt->zSql, zSql, pPreStmt->nSql) ); 11004a4c11aaSdan 11014a4c11aaSdan /* Bind values to parameters that begin with $ or : */ 11024a4c11aaSdan for(i=1; i<=nVar; i++){ 11034a4c11aaSdan const char *zVar = sqlite3_bind_parameter_name(pStmt, i); 11044a4c11aaSdan if( zVar!=0 && (zVar[0]=='$' || zVar[0]==':' || zVar[0]=='@') ){ 11054a4c11aaSdan Tcl_Obj *pVar = Tcl_GetVar2Ex(interp, &zVar[1], 0, 0); 11064a4c11aaSdan if( pVar ){ 11074a4c11aaSdan int n; 11084a4c11aaSdan u8 *data; 11094a4c11aaSdan const char *zType = (pVar->typePtr ? pVar->typePtr->name : ""); 11104a4c11aaSdan char c = zType[0]; 11114a4c11aaSdan if( zVar[0]=='@' || 11124a4c11aaSdan (c=='b' && strcmp(zType,"bytearray")==0 && pVar->bytes==0) ){ 11134a4c11aaSdan /* Load a BLOB type if the Tcl variable is a bytearray and 11144a4c11aaSdan ** it has no string representation or the host 11154a4c11aaSdan ** parameter name begins with "@". */ 11164a4c11aaSdan data = Tcl_GetByteArrayFromObj(pVar, &n); 11174a4c11aaSdan sqlite3_bind_blob(pStmt, i, data, n, SQLITE_STATIC); 11184a4c11aaSdan Tcl_IncrRefCount(pVar); 11194a4c11aaSdan pPreStmt->apParm[iParm++] = pVar; 11204a4c11aaSdan }else if( c=='b' && strcmp(zType,"boolean")==0 ){ 11214a4c11aaSdan Tcl_GetIntFromObj(interp, pVar, &n); 11224a4c11aaSdan sqlite3_bind_int(pStmt, i, n); 11234a4c11aaSdan }else if( c=='d' && strcmp(zType,"double")==0 ){ 11244a4c11aaSdan double r; 11254a4c11aaSdan Tcl_GetDoubleFromObj(interp, pVar, &r); 11264a4c11aaSdan sqlite3_bind_double(pStmt, i, r); 11274a4c11aaSdan }else if( (c=='w' && strcmp(zType,"wideInt")==0) || 11284a4c11aaSdan (c=='i' && strcmp(zType,"int")==0) ){ 11294a4c11aaSdan Tcl_WideInt v; 11304a4c11aaSdan Tcl_GetWideIntFromObj(interp, pVar, &v); 11314a4c11aaSdan sqlite3_bind_int64(pStmt, i, v); 11324a4c11aaSdan }else{ 11334a4c11aaSdan data = (unsigned char *)Tcl_GetStringFromObj(pVar, &n); 11344a4c11aaSdan sqlite3_bind_text(pStmt, i, (char *)data, n, SQLITE_STATIC); 11354a4c11aaSdan Tcl_IncrRefCount(pVar); 11364a4c11aaSdan pPreStmt->apParm[iParm++] = pVar; 11374a4c11aaSdan } 11384a4c11aaSdan }else{ 11394a4c11aaSdan sqlite3_bind_null(pStmt, i); 11404a4c11aaSdan } 11414a4c11aaSdan } 11424a4c11aaSdan } 11434a4c11aaSdan pPreStmt->nParm = iParm; 11444a4c11aaSdan *ppPreStmt = pPreStmt; 1145937d0deaSdan 11464a4c11aaSdan return TCL_OK; 11474a4c11aaSdan } 11484a4c11aaSdan 11494a4c11aaSdan 11504a4c11aaSdan /* 11514a4c11aaSdan ** Release a statement reference obtained by calling dbPrepareAndBind(). 11524a4c11aaSdan ** There should be exactly one call to this function for each call to 11534a4c11aaSdan ** dbPrepareAndBind(). 11544a4c11aaSdan ** 11554a4c11aaSdan ** If the discard parameter is non-zero, then the statement is deleted 11564a4c11aaSdan ** immediately. Otherwise it is added to the LRU list and may be returned 11574a4c11aaSdan ** by a subsequent call to dbPrepareAndBind(). 11584a4c11aaSdan */ 11594a4c11aaSdan static void dbReleaseStmt( 11604a4c11aaSdan SqliteDb *pDb, /* Database handle */ 11614a4c11aaSdan SqlPreparedStmt *pPreStmt, /* Prepared statement handle to release */ 11624a4c11aaSdan int discard /* True to delete (not cache) the pPreStmt */ 11634a4c11aaSdan ){ 11644a4c11aaSdan int i; 11654a4c11aaSdan 11664a4c11aaSdan /* Free the bound string and blob parameters */ 11674a4c11aaSdan for(i=0; i<pPreStmt->nParm; i++){ 11684a4c11aaSdan Tcl_DecrRefCount(pPreStmt->apParm[i]); 11694a4c11aaSdan } 11704a4c11aaSdan pPreStmt->nParm = 0; 11714a4c11aaSdan 11724a4c11aaSdan if( pDb->maxStmt<=0 || discard ){ 11734a4c11aaSdan /* If the cache is turned off, deallocated the statement */ 11744a4c11aaSdan sqlite3_finalize(pPreStmt->pStmt); 11754a4c11aaSdan Tcl_Free((char *)pPreStmt); 11764a4c11aaSdan }else{ 11774a4c11aaSdan /* Add the prepared statement to the beginning of the cache list. */ 11784a4c11aaSdan pPreStmt->pNext = pDb->stmtList; 11794a4c11aaSdan pPreStmt->pPrev = 0; 11804a4c11aaSdan if( pDb->stmtList ){ 11814a4c11aaSdan pDb->stmtList->pPrev = pPreStmt; 11824a4c11aaSdan } 11834a4c11aaSdan pDb->stmtList = pPreStmt; 11844a4c11aaSdan if( pDb->stmtLast==0 ){ 11854a4c11aaSdan assert( pDb->nStmt==0 ); 11864a4c11aaSdan pDb->stmtLast = pPreStmt; 11874a4c11aaSdan }else{ 11884a4c11aaSdan assert( pDb->nStmt>0 ); 11894a4c11aaSdan } 11904a4c11aaSdan pDb->nStmt++; 11914a4c11aaSdan 11924a4c11aaSdan /* If we have too many statement in cache, remove the surplus from 11934a4c11aaSdan ** the end of the cache list. */ 11944a4c11aaSdan while( pDb->nStmt>pDb->maxStmt ){ 11954a4c11aaSdan sqlite3_finalize(pDb->stmtLast->pStmt); 11964a4c11aaSdan pDb->stmtLast = pDb->stmtLast->pPrev; 11974a4c11aaSdan Tcl_Free((char*)pDb->stmtLast->pNext); 11984a4c11aaSdan pDb->stmtLast->pNext = 0; 11994a4c11aaSdan pDb->nStmt--; 12004a4c11aaSdan } 12014a4c11aaSdan } 12024a4c11aaSdan } 12034a4c11aaSdan 12044a4c11aaSdan /* 12054a4c11aaSdan ** Structure used with dbEvalXXX() functions: 12064a4c11aaSdan ** 12074a4c11aaSdan ** dbEvalInit() 12084a4c11aaSdan ** dbEvalStep() 12094a4c11aaSdan ** dbEvalFinalize() 12104a4c11aaSdan ** dbEvalRowInfo() 12114a4c11aaSdan ** dbEvalColumnValue() 12124a4c11aaSdan */ 12134a4c11aaSdan typedef struct DbEvalContext DbEvalContext; 12144a4c11aaSdan struct DbEvalContext { 12154a4c11aaSdan SqliteDb *pDb; /* Database handle */ 12164a4c11aaSdan Tcl_Obj *pSql; /* Object holding string zSql */ 12174a4c11aaSdan const char *zSql; /* Remaining SQL to execute */ 12184a4c11aaSdan SqlPreparedStmt *pPreStmt; /* Current statement */ 12194a4c11aaSdan int nCol; /* Number of columns returned by pStmt */ 12204a4c11aaSdan Tcl_Obj *pArray; /* Name of array variable */ 12214a4c11aaSdan Tcl_Obj **apColName; /* Array of column names */ 12224a4c11aaSdan }; 12234a4c11aaSdan 12244a4c11aaSdan /* 12254a4c11aaSdan ** Release any cache of column names currently held as part of 12264a4c11aaSdan ** the DbEvalContext structure passed as the first argument. 12274a4c11aaSdan */ 12284a4c11aaSdan static void dbReleaseColumnNames(DbEvalContext *p){ 12294a4c11aaSdan if( p->apColName ){ 12304a4c11aaSdan int i; 12314a4c11aaSdan for(i=0; i<p->nCol; i++){ 12324a4c11aaSdan Tcl_DecrRefCount(p->apColName[i]); 12334a4c11aaSdan } 12344a4c11aaSdan Tcl_Free((char *)p->apColName); 12354a4c11aaSdan p->apColName = 0; 12364a4c11aaSdan } 12374a4c11aaSdan p->nCol = 0; 12384a4c11aaSdan } 12394a4c11aaSdan 12404a4c11aaSdan /* 12414a4c11aaSdan ** Initialize a DbEvalContext structure. 12428e556520Sdanielk1977 ** 12438e556520Sdanielk1977 ** If pArray is not NULL, then it contains the name of a Tcl array 12448e556520Sdanielk1977 ** variable. The "*" member of this array is set to a list containing 12454a4c11aaSdan ** the names of the columns returned by the statement as part of each 12464a4c11aaSdan ** call to dbEvalStep(), in order from left to right. e.g. if the names 12474a4c11aaSdan ** of the returned columns are a, b and c, it does the equivalent of the 12484a4c11aaSdan ** tcl command: 12498e556520Sdanielk1977 ** 12508e556520Sdanielk1977 ** set ${pArray}(*) {a b c} 12518e556520Sdanielk1977 */ 12524a4c11aaSdan static void dbEvalInit( 12534a4c11aaSdan DbEvalContext *p, /* Pointer to structure to initialize */ 12544a4c11aaSdan SqliteDb *pDb, /* Database handle */ 12554a4c11aaSdan Tcl_Obj *pSql, /* Object containing SQL script */ 12564a4c11aaSdan Tcl_Obj *pArray /* Name of Tcl array to set (*) element of */ 12578e556520Sdanielk1977 ){ 12584a4c11aaSdan memset(p, 0, sizeof(DbEvalContext)); 12594a4c11aaSdan p->pDb = pDb; 12604a4c11aaSdan p->zSql = Tcl_GetString(pSql); 12614a4c11aaSdan p->pSql = pSql; 12624a4c11aaSdan Tcl_IncrRefCount(pSql); 12634a4c11aaSdan if( pArray ){ 12644a4c11aaSdan p->pArray = pArray; 12654a4c11aaSdan Tcl_IncrRefCount(pArray); 12664a4c11aaSdan } 12674a4c11aaSdan } 12688e556520Sdanielk1977 12694a4c11aaSdan /* 12704a4c11aaSdan ** Obtain information about the row that the DbEvalContext passed as the 12714a4c11aaSdan ** first argument currently points to. 12724a4c11aaSdan */ 12734a4c11aaSdan static void dbEvalRowInfo( 12744a4c11aaSdan DbEvalContext *p, /* Evaluation context */ 12754a4c11aaSdan int *pnCol, /* OUT: Number of column names */ 12764a4c11aaSdan Tcl_Obj ***papColName /* OUT: Array of column names */ 12774a4c11aaSdan ){ 12788e556520Sdanielk1977 /* Compute column names */ 12794a4c11aaSdan if( 0==p->apColName ){ 12804a4c11aaSdan sqlite3_stmt *pStmt = p->pPreStmt->pStmt; 12814a4c11aaSdan int i; /* Iterator variable */ 12824a4c11aaSdan int nCol; /* Number of columns returned by pStmt */ 12834a4c11aaSdan Tcl_Obj **apColName = 0; /* Array of column names */ 12844a4c11aaSdan 12854a4c11aaSdan p->nCol = nCol = sqlite3_column_count(pStmt); 12864a4c11aaSdan if( nCol>0 && (papColName || p->pArray) ){ 12874a4c11aaSdan apColName = (Tcl_Obj**)Tcl_Alloc( sizeof(Tcl_Obj*)*nCol ); 12888e556520Sdanielk1977 for(i=0; i<nCol; i++){ 12898e556520Sdanielk1977 apColName[i] = dbTextToObj(sqlite3_column_name(pStmt,i)); 12908e556520Sdanielk1977 Tcl_IncrRefCount(apColName[i]); 12918e556520Sdanielk1977 } 12924a4c11aaSdan p->apColName = apColName; 12934a4c11aaSdan } 12948e556520Sdanielk1977 12958e556520Sdanielk1977 /* If results are being stored in an array variable, then create 12968e556520Sdanielk1977 ** the array(*) entry for that array 12978e556520Sdanielk1977 */ 12984a4c11aaSdan if( p->pArray ){ 12994a4c11aaSdan Tcl_Interp *interp = p->pDb->interp; 13008e556520Sdanielk1977 Tcl_Obj *pColList = Tcl_NewObj(); 13018e556520Sdanielk1977 Tcl_Obj *pStar = Tcl_NewStringObj("*", -1); 13024a4c11aaSdan 13038e556520Sdanielk1977 for(i=0; i<nCol; i++){ 13048e556520Sdanielk1977 Tcl_ListObjAppendElement(interp, pColList, apColName[i]); 13058e556520Sdanielk1977 } 13068e556520Sdanielk1977 Tcl_IncrRefCount(pStar); 13074a4c11aaSdan Tcl_ObjSetVar2(interp, p->pArray, pStar, pColList, 0); 13088e556520Sdanielk1977 Tcl_DecrRefCount(pStar); 13098e556520Sdanielk1977 } 13108e556520Sdanielk1977 } 13118e556520Sdanielk1977 13124a4c11aaSdan if( papColName ){ 13134a4c11aaSdan *papColName = p->apColName; 13144a4c11aaSdan } 13154a4c11aaSdan if( pnCol ){ 13164a4c11aaSdan *pnCol = p->nCol; 13174a4c11aaSdan } 13184a4c11aaSdan } 13194a4c11aaSdan 13204a4c11aaSdan /* 13214a4c11aaSdan ** Return one of TCL_OK, TCL_BREAK or TCL_ERROR. If TCL_ERROR is 13224a4c11aaSdan ** returned, then an error message is stored in the interpreter before 13234a4c11aaSdan ** returning. 13244a4c11aaSdan ** 13254a4c11aaSdan ** A return value of TCL_OK means there is a row of data available. The 13264a4c11aaSdan ** data may be accessed using dbEvalRowInfo() and dbEvalColumnValue(). This 13274a4c11aaSdan ** is analogous to a return of SQLITE_ROW from sqlite3_step(). If TCL_BREAK 13284a4c11aaSdan ** is returned, then the SQL script has finished executing and there are 13294a4c11aaSdan ** no further rows available. This is similar to SQLITE_DONE. 13304a4c11aaSdan */ 13314a4c11aaSdan static int dbEvalStep(DbEvalContext *p){ 13324a4c11aaSdan while( p->zSql[0] || p->pPreStmt ){ 13334a4c11aaSdan int rc; 13344a4c11aaSdan if( p->pPreStmt==0 ){ 13354a4c11aaSdan rc = dbPrepareAndBind(p->pDb, p->zSql, &p->zSql, &p->pPreStmt); 13364a4c11aaSdan if( rc!=TCL_OK ) return rc; 13374a4c11aaSdan }else{ 13384a4c11aaSdan int rcs; 13394a4c11aaSdan SqliteDb *pDb = p->pDb; 13404a4c11aaSdan SqlPreparedStmt *pPreStmt = p->pPreStmt; 13414a4c11aaSdan sqlite3_stmt *pStmt = pPreStmt->pStmt; 13424a4c11aaSdan 13434a4c11aaSdan rcs = sqlite3_step(pStmt); 13444a4c11aaSdan if( rcs==SQLITE_ROW ){ 13454a4c11aaSdan return TCL_OK; 13464a4c11aaSdan } 13474a4c11aaSdan if( p->pArray ){ 13484a4c11aaSdan dbEvalRowInfo(p, 0, 0); 13494a4c11aaSdan } 13504a4c11aaSdan rcs = sqlite3_reset(pStmt); 13514a4c11aaSdan 13524a4c11aaSdan pDb->nStep = sqlite3_stmt_status(pStmt,SQLITE_STMTSTATUS_FULLSCAN_STEP,1); 13534a4c11aaSdan pDb->nSort = sqlite3_stmt_status(pStmt,SQLITE_STMTSTATUS_SORT,1); 13544a4c11aaSdan dbReleaseColumnNames(p); 13554a4c11aaSdan p->pPreStmt = 0; 13564a4c11aaSdan 13574a4c11aaSdan if( rcs!=SQLITE_OK ){ 13584a4c11aaSdan /* If a run-time error occurs, report the error and stop reading 13594a4c11aaSdan ** the SQL. */ 13604a4c11aaSdan Tcl_SetObjResult(pDb->interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 13614a4c11aaSdan dbReleaseStmt(pDb, pPreStmt, 1); 13624a4c11aaSdan return TCL_ERROR; 13634a4c11aaSdan }else{ 13644a4c11aaSdan dbReleaseStmt(pDb, pPreStmt, 0); 13654a4c11aaSdan } 13664a4c11aaSdan } 13674a4c11aaSdan } 13684a4c11aaSdan 13694a4c11aaSdan /* Finished */ 13704a4c11aaSdan return TCL_BREAK; 13714a4c11aaSdan } 13724a4c11aaSdan 13734a4c11aaSdan /* 13744a4c11aaSdan ** Free all resources currently held by the DbEvalContext structure passed 13754a4c11aaSdan ** as the first argument. There should be exactly one call to this function 13764a4c11aaSdan ** for each call to dbEvalInit(). 13774a4c11aaSdan */ 13784a4c11aaSdan static void dbEvalFinalize(DbEvalContext *p){ 13794a4c11aaSdan if( p->pPreStmt ){ 13804a4c11aaSdan sqlite3_reset(p->pPreStmt->pStmt); 13814a4c11aaSdan dbReleaseStmt(p->pDb, p->pPreStmt, 0); 13824a4c11aaSdan p->pPreStmt = 0; 13834a4c11aaSdan } 13844a4c11aaSdan if( p->pArray ){ 13854a4c11aaSdan Tcl_DecrRefCount(p->pArray); 13864a4c11aaSdan p->pArray = 0; 13874a4c11aaSdan } 13884a4c11aaSdan Tcl_DecrRefCount(p->pSql); 13894a4c11aaSdan dbReleaseColumnNames(p); 13904a4c11aaSdan } 13914a4c11aaSdan 13924a4c11aaSdan /* 13934a4c11aaSdan ** Return a pointer to a Tcl_Obj structure with ref-count 0 that contains 13944a4c11aaSdan ** the value for the iCol'th column of the row currently pointed to by 13954a4c11aaSdan ** the DbEvalContext structure passed as the first argument. 13964a4c11aaSdan */ 13974a4c11aaSdan static Tcl_Obj *dbEvalColumnValue(DbEvalContext *p, int iCol){ 13984a4c11aaSdan sqlite3_stmt *pStmt = p->pPreStmt->pStmt; 13994a4c11aaSdan switch( sqlite3_column_type(pStmt, iCol) ){ 14004a4c11aaSdan case SQLITE_BLOB: { 14014a4c11aaSdan int bytes = sqlite3_column_bytes(pStmt, iCol); 14024a4c11aaSdan const char *zBlob = sqlite3_column_blob(pStmt, iCol); 14034a4c11aaSdan if( !zBlob ) bytes = 0; 14044a4c11aaSdan return Tcl_NewByteArrayObj((u8*)zBlob, bytes); 14054a4c11aaSdan } 14064a4c11aaSdan case SQLITE_INTEGER: { 14074a4c11aaSdan sqlite_int64 v = sqlite3_column_int64(pStmt, iCol); 14084a4c11aaSdan if( v>=-2147483647 && v<=2147483647 ){ 14094a4c11aaSdan return Tcl_NewIntObj(v); 14104a4c11aaSdan }else{ 14114a4c11aaSdan return Tcl_NewWideIntObj(v); 14124a4c11aaSdan } 14134a4c11aaSdan } 14144a4c11aaSdan case SQLITE_FLOAT: { 14154a4c11aaSdan return Tcl_NewDoubleObj(sqlite3_column_double(pStmt, iCol)); 14164a4c11aaSdan } 14174a4c11aaSdan case SQLITE_NULL: { 14184a4c11aaSdan return dbTextToObj(p->pDb->zNull); 14194a4c11aaSdan } 14204a4c11aaSdan } 14214a4c11aaSdan 14224a4c11aaSdan return dbTextToObj((char *)sqlite3_column_text(pStmt, iCol)); 14234a4c11aaSdan } 14244a4c11aaSdan 14254a4c11aaSdan /* 14264a4c11aaSdan ** If using Tcl version 8.6 or greater, use the NR functions to avoid 14274a4c11aaSdan ** recursive evalution of scripts by the [db eval] and [db trans] 14284a4c11aaSdan ** commands. Even if the headers used while compiling the extension 14294a4c11aaSdan ** are 8.6 or newer, the code still tests the Tcl version at runtime. 14304a4c11aaSdan ** This allows stubs-enabled builds to be used with older Tcl libraries. 14314a4c11aaSdan */ 14324a4c11aaSdan #if TCL_MAJOR_VERSION>8 || (TCL_MAJOR_VERSION==8 && TCL_MINOR_VERSION>=6) 1433a2c8a95bSdrh # define SQLITE_TCL_NRE 1 14344a4c11aaSdan static int DbUseNre(void){ 14354a4c11aaSdan int major, minor; 14364a4c11aaSdan Tcl_GetVersion(&major, &minor, 0, 0); 14374a4c11aaSdan return( (major==8 && minor>=6) || major>8 ); 14384a4c11aaSdan } 14394a4c11aaSdan #else 14404a4c11aaSdan /* 14414a4c11aaSdan ** Compiling using headers earlier than 8.6. In this case NR cannot be 14424a4c11aaSdan ** used, so DbUseNre() to always return zero. Add #defines for the other 14434a4c11aaSdan ** Tcl_NRxxx() functions to prevent them from causing compilation errors, 14444a4c11aaSdan ** even though the only invocations of them are within conditional blocks 14454a4c11aaSdan ** of the form: 14464a4c11aaSdan ** 14474a4c11aaSdan ** if( DbUseNre() ) { ... } 14484a4c11aaSdan */ 1449a2c8a95bSdrh # define SQLITE_TCL_NRE 0 14504a4c11aaSdan # define DbUseNre() 0 14514a4c11aaSdan # define Tcl_NRAddCallback(a,b,c,d,e,f) 0 14524a4c11aaSdan # define Tcl_NREvalObj(a,b,c) 0 14534a4c11aaSdan # define Tcl_NRCreateCommand(a,b,c,d,e,f) 0 14544a4c11aaSdan #endif 14554a4c11aaSdan 14564a4c11aaSdan /* 14574a4c11aaSdan ** This function is part of the implementation of the command: 14584a4c11aaSdan ** 14594a4c11aaSdan ** $db eval SQL ?ARRAYNAME? SCRIPT 14604a4c11aaSdan */ 14614a4c11aaSdan static int DbEvalNextCmd( 14624a4c11aaSdan ClientData data[], /* data[0] is the (DbEvalContext*) */ 14634a4c11aaSdan Tcl_Interp *interp, /* Tcl interpreter */ 14644a4c11aaSdan int result /* Result so far */ 14654a4c11aaSdan ){ 14664a4c11aaSdan int rc = result; /* Return code */ 14674a4c11aaSdan 14684a4c11aaSdan /* The first element of the data[] array is a pointer to a DbEvalContext 14694a4c11aaSdan ** structure allocated using Tcl_Alloc(). The second element of data[] 14704a4c11aaSdan ** is a pointer to a Tcl_Obj containing the script to run for each row 14714a4c11aaSdan ** returned by the queries encapsulated in data[0]. */ 14724a4c11aaSdan DbEvalContext *p = (DbEvalContext *)data[0]; 14734a4c11aaSdan Tcl_Obj *pScript = (Tcl_Obj *)data[1]; 14744a4c11aaSdan Tcl_Obj *pArray = p->pArray; 14754a4c11aaSdan 14764a4c11aaSdan while( (rc==TCL_OK || rc==TCL_CONTINUE) && TCL_OK==(rc = dbEvalStep(p)) ){ 14774a4c11aaSdan int i; 14784a4c11aaSdan int nCol; 14794a4c11aaSdan Tcl_Obj **apColName; 14804a4c11aaSdan dbEvalRowInfo(p, &nCol, &apColName); 14814a4c11aaSdan for(i=0; i<nCol; i++){ 14824a4c11aaSdan Tcl_Obj *pVal = dbEvalColumnValue(p, i); 14834a4c11aaSdan if( pArray==0 ){ 14844a4c11aaSdan Tcl_ObjSetVar2(interp, apColName[i], 0, pVal, 0); 14854a4c11aaSdan }else{ 14864a4c11aaSdan Tcl_ObjSetVar2(interp, pArray, apColName[i], pVal, 0); 14874a4c11aaSdan } 14884a4c11aaSdan } 14894a4c11aaSdan 14904a4c11aaSdan /* The required interpreter variables are now populated with the data 14914a4c11aaSdan ** from the current row. If using NRE, schedule callbacks to evaluate 14924a4c11aaSdan ** script pScript, then to invoke this function again to fetch the next 14934a4c11aaSdan ** row (or clean up if there is no next row or the script throws an 14944a4c11aaSdan ** exception). After scheduling the callbacks, return control to the 14954a4c11aaSdan ** caller. 14964a4c11aaSdan ** 14974a4c11aaSdan ** If not using NRE, evaluate pScript directly and continue with the 14984a4c11aaSdan ** next iteration of this while(...) loop. */ 14994a4c11aaSdan if( DbUseNre() ){ 15004a4c11aaSdan Tcl_NRAddCallback(interp, DbEvalNextCmd, (void*)p, (void*)pScript, 0, 0); 15014a4c11aaSdan return Tcl_NREvalObj(interp, pScript, 0); 15024a4c11aaSdan }else{ 15034a4c11aaSdan rc = Tcl_EvalObjEx(interp, pScript, 0); 15044a4c11aaSdan } 15054a4c11aaSdan } 15064a4c11aaSdan 15074a4c11aaSdan Tcl_DecrRefCount(pScript); 15084a4c11aaSdan dbEvalFinalize(p); 15094a4c11aaSdan Tcl_Free((char *)p); 15104a4c11aaSdan 15114a4c11aaSdan if( rc==TCL_OK || rc==TCL_BREAK ){ 15124a4c11aaSdan Tcl_ResetResult(interp); 15134a4c11aaSdan rc = TCL_OK; 15144a4c11aaSdan } 15154a4c11aaSdan return rc; 15168e556520Sdanielk1977 } 15178e556520Sdanielk1977 15181067fe11Stpoindex /* 151975897234Sdrh ** The "sqlite" command below creates a new Tcl command for each 152075897234Sdrh ** connection it opens to an SQLite database. This routine is invoked 152175897234Sdrh ** whenever one of those connection-specific commands is executed 152275897234Sdrh ** in Tcl. For example, if you run Tcl code like this: 152375897234Sdrh ** 15249bb575fdSdrh ** sqlite3 db1 "my_database" 152575897234Sdrh ** db1 close 152675897234Sdrh ** 152775897234Sdrh ** The first command opens a connection to the "my_database" database 152875897234Sdrh ** and calls that connection "db1". The second command causes this 152975897234Sdrh ** subroutine to be invoked. 153075897234Sdrh */ 15316d31316cSdrh static int DbObjCmd(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){ 1532bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)cd; 15336d31316cSdrh int choice; 153422fbcb8dSdrh int rc = TCL_OK; 15350de8c112Sdrh static const char *DB_strs[] = { 1536dc2c4915Sdrh "authorizer", "backup", "busy", 1537dc2c4915Sdrh "cache", "changes", "close", 1538dc2c4915Sdrh "collate", "collation_needed", "commit_hook", 1539dc2c4915Sdrh "complete", "copy", "enable_load_extension", 1540dc2c4915Sdrh "errorcode", "eval", "exists", 1541dc2c4915Sdrh "function", "incrblob", "interrupt", 1542dc2c4915Sdrh "last_insert_rowid", "nullvalue", "onecolumn", 1543dc2c4915Sdrh "profile", "progress", "rekey", 1544dc2c4915Sdrh "restore", "rollback_hook", "status", 1545dc2c4915Sdrh "timeout", "total_changes", "trace", 1546404ca075Sdanielk1977 "transaction", "unlock_notify", "update_hook", 1547404ca075Sdanielk1977 "version", 0 15486d31316cSdrh }; 1549411995dcSdrh enum DB_enum { 1550dc2c4915Sdrh DB_AUTHORIZER, DB_BACKUP, DB_BUSY, 1551dc2c4915Sdrh DB_CACHE, DB_CHANGES, DB_CLOSE, 1552dc2c4915Sdrh DB_COLLATE, DB_COLLATION_NEEDED, DB_COMMIT_HOOK, 1553dc2c4915Sdrh DB_COMPLETE, DB_COPY, DB_ENABLE_LOAD_EXTENSION, 1554dc2c4915Sdrh DB_ERRORCODE, DB_EVAL, DB_EXISTS, 1555dc2c4915Sdrh DB_FUNCTION, DB_INCRBLOB, DB_INTERRUPT, 1556dc2c4915Sdrh DB_LAST_INSERT_ROWID, DB_NULLVALUE, DB_ONECOLUMN, 1557dc2c4915Sdrh DB_PROFILE, DB_PROGRESS, DB_REKEY, 1558dc2c4915Sdrh DB_RESTORE, DB_ROLLBACK_HOOK, DB_STATUS, 1559dc2c4915Sdrh DB_TIMEOUT, DB_TOTAL_CHANGES, DB_TRACE, 1560404ca075Sdanielk1977 DB_TRANSACTION, DB_UNLOCK_NOTIFY, DB_UPDATE_HOOK, 1561404ca075Sdanielk1977 DB_VERSION, 15626d31316cSdrh }; 15631067fe11Stpoindex /* don't leave trailing commas on DB_enum, it confuses the AIX xlc compiler */ 15646d31316cSdrh 15656d31316cSdrh if( objc<2 ){ 15666d31316cSdrh Tcl_WrongNumArgs(interp, 1, objv, "SUBCOMMAND ..."); 156775897234Sdrh return TCL_ERROR; 156875897234Sdrh } 1569411995dcSdrh if( Tcl_GetIndexFromObj(interp, objv[1], DB_strs, "option", 0, &choice) ){ 15706d31316cSdrh return TCL_ERROR; 15716d31316cSdrh } 15726d31316cSdrh 1573411995dcSdrh switch( (enum DB_enum)choice ){ 157475897234Sdrh 1575e22a334bSdrh /* $db authorizer ?CALLBACK? 1576e22a334bSdrh ** 1577e22a334bSdrh ** Invoke the given callback to authorize each SQL operation as it is 1578e22a334bSdrh ** compiled. 5 arguments are appended to the callback before it is 1579e22a334bSdrh ** invoked: 1580e22a334bSdrh ** 1581e22a334bSdrh ** (1) The authorization type (ex: SQLITE_CREATE_TABLE, SQLITE_INSERT, ...) 1582e22a334bSdrh ** (2) First descriptive name (depends on authorization type) 1583e22a334bSdrh ** (3) Second descriptive name 1584e22a334bSdrh ** (4) Name of the database (ex: "main", "temp") 1585e22a334bSdrh ** (5) Name of trigger that is doing the access 1586e22a334bSdrh ** 1587e22a334bSdrh ** The callback should return on of the following strings: SQLITE_OK, 1588e22a334bSdrh ** SQLITE_IGNORE, or SQLITE_DENY. Any other return value is an error. 1589e22a334bSdrh ** 1590e22a334bSdrh ** If this method is invoked with no arguments, the current authorization 1591e22a334bSdrh ** callback string is returned. 1592e22a334bSdrh */ 1593e22a334bSdrh case DB_AUTHORIZER: { 15941211de37Sdrh #ifdef SQLITE_OMIT_AUTHORIZATION 15951211de37Sdrh Tcl_AppendResult(interp, "authorization not available in this build", 0); 15961211de37Sdrh return TCL_ERROR; 15971211de37Sdrh #else 1598e22a334bSdrh if( objc>3 ){ 1599e22a334bSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 16000f14e2ebSdrh return TCL_ERROR; 1601e22a334bSdrh }else if( objc==2 ){ 1602b5a20d3cSdrh if( pDb->zAuth ){ 1603e22a334bSdrh Tcl_AppendResult(interp, pDb->zAuth, 0); 1604e22a334bSdrh } 1605e22a334bSdrh }else{ 1606e22a334bSdrh char *zAuth; 1607e22a334bSdrh int len; 1608e22a334bSdrh if( pDb->zAuth ){ 1609e22a334bSdrh Tcl_Free(pDb->zAuth); 1610e22a334bSdrh } 1611e22a334bSdrh zAuth = Tcl_GetStringFromObj(objv[2], &len); 1612e22a334bSdrh if( zAuth && len>0 ){ 1613e22a334bSdrh pDb->zAuth = Tcl_Alloc( len + 1 ); 16145bb3eb9bSdrh memcpy(pDb->zAuth, zAuth, len+1); 1615e22a334bSdrh }else{ 1616e22a334bSdrh pDb->zAuth = 0; 1617e22a334bSdrh } 1618e22a334bSdrh if( pDb->zAuth ){ 1619e22a334bSdrh pDb->interp = interp; 16206f8a503dSdanielk1977 sqlite3_set_authorizer(pDb->db, auth_callback, pDb); 1621e22a334bSdrh }else{ 16226f8a503dSdanielk1977 sqlite3_set_authorizer(pDb->db, 0, 0); 1623e22a334bSdrh } 1624e22a334bSdrh } 16251211de37Sdrh #endif 1626e22a334bSdrh break; 1627e22a334bSdrh } 1628e22a334bSdrh 1629dc2c4915Sdrh /* $db backup ?DATABASE? FILENAME 1630dc2c4915Sdrh ** 1631dc2c4915Sdrh ** Open or create a database file named FILENAME. Transfer the 1632dc2c4915Sdrh ** content of local database DATABASE (default: "main") into the 1633dc2c4915Sdrh ** FILENAME database. 1634dc2c4915Sdrh */ 1635dc2c4915Sdrh case DB_BACKUP: { 1636dc2c4915Sdrh const char *zDestFile; 1637dc2c4915Sdrh const char *zSrcDb; 1638dc2c4915Sdrh sqlite3 *pDest; 1639dc2c4915Sdrh sqlite3_backup *pBackup; 1640dc2c4915Sdrh 1641dc2c4915Sdrh if( objc==3 ){ 1642dc2c4915Sdrh zSrcDb = "main"; 1643dc2c4915Sdrh zDestFile = Tcl_GetString(objv[2]); 1644dc2c4915Sdrh }else if( objc==4 ){ 1645dc2c4915Sdrh zSrcDb = Tcl_GetString(objv[2]); 1646dc2c4915Sdrh zDestFile = Tcl_GetString(objv[3]); 1647dc2c4915Sdrh }else{ 1648dc2c4915Sdrh Tcl_WrongNumArgs(interp, 2, objv, "?DATABASE? FILENAME"); 1649dc2c4915Sdrh return TCL_ERROR; 1650dc2c4915Sdrh } 1651dc2c4915Sdrh rc = sqlite3_open(zDestFile, &pDest); 1652dc2c4915Sdrh if( rc!=SQLITE_OK ){ 1653dc2c4915Sdrh Tcl_AppendResult(interp, "cannot open target database: ", 1654dc2c4915Sdrh sqlite3_errmsg(pDest), (char*)0); 1655dc2c4915Sdrh sqlite3_close(pDest); 1656dc2c4915Sdrh return TCL_ERROR; 1657dc2c4915Sdrh } 1658dc2c4915Sdrh pBackup = sqlite3_backup_init(pDest, "main", pDb->db, zSrcDb); 1659dc2c4915Sdrh if( pBackup==0 ){ 1660dc2c4915Sdrh Tcl_AppendResult(interp, "backup failed: ", 1661dc2c4915Sdrh sqlite3_errmsg(pDest), (char*)0); 1662dc2c4915Sdrh sqlite3_close(pDest); 1663dc2c4915Sdrh return TCL_ERROR; 1664dc2c4915Sdrh } 1665dc2c4915Sdrh while( (rc = sqlite3_backup_step(pBackup,100))==SQLITE_OK ){} 1666dc2c4915Sdrh sqlite3_backup_finish(pBackup); 1667dc2c4915Sdrh if( rc==SQLITE_DONE ){ 1668dc2c4915Sdrh rc = TCL_OK; 1669dc2c4915Sdrh }else{ 1670dc2c4915Sdrh Tcl_AppendResult(interp, "backup failed: ", 1671dc2c4915Sdrh sqlite3_errmsg(pDest), (char*)0); 1672dc2c4915Sdrh rc = TCL_ERROR; 1673dc2c4915Sdrh } 1674dc2c4915Sdrh sqlite3_close(pDest); 1675dc2c4915Sdrh break; 1676dc2c4915Sdrh } 1677dc2c4915Sdrh 1678bec3f402Sdrh /* $db busy ?CALLBACK? 1679bec3f402Sdrh ** 1680bec3f402Sdrh ** Invoke the given callback if an SQL statement attempts to open 1681bec3f402Sdrh ** a locked database file. 1682bec3f402Sdrh */ 16836d31316cSdrh case DB_BUSY: { 16846d31316cSdrh if( objc>3 ){ 16856d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "CALLBACK"); 1686bec3f402Sdrh return TCL_ERROR; 16876d31316cSdrh }else if( objc==2 ){ 1688bec3f402Sdrh if( pDb->zBusy ){ 1689bec3f402Sdrh Tcl_AppendResult(interp, pDb->zBusy, 0); 1690bec3f402Sdrh } 1691bec3f402Sdrh }else{ 16926d31316cSdrh char *zBusy; 16936d31316cSdrh int len; 1694bec3f402Sdrh if( pDb->zBusy ){ 1695bec3f402Sdrh Tcl_Free(pDb->zBusy); 16966d31316cSdrh } 16976d31316cSdrh zBusy = Tcl_GetStringFromObj(objv[2], &len); 16986d31316cSdrh if( zBusy && len>0 ){ 16996d31316cSdrh pDb->zBusy = Tcl_Alloc( len + 1 ); 17005bb3eb9bSdrh memcpy(pDb->zBusy, zBusy, len+1); 17016d31316cSdrh }else{ 1702bec3f402Sdrh pDb->zBusy = 0; 1703bec3f402Sdrh } 1704bec3f402Sdrh if( pDb->zBusy ){ 1705bec3f402Sdrh pDb->interp = interp; 17066f8a503dSdanielk1977 sqlite3_busy_handler(pDb->db, DbBusyHandler, pDb); 17076d31316cSdrh }else{ 17086f8a503dSdanielk1977 sqlite3_busy_handler(pDb->db, 0, 0); 1709bec3f402Sdrh } 1710bec3f402Sdrh } 17116d31316cSdrh break; 17126d31316cSdrh } 1713bec3f402Sdrh 1714fb7e7651Sdrh /* $db cache flush 1715fb7e7651Sdrh ** $db cache size n 1716fb7e7651Sdrh ** 1717fb7e7651Sdrh ** Flush the prepared statement cache, or set the maximum number of 1718fb7e7651Sdrh ** cached statements. 1719fb7e7651Sdrh */ 1720fb7e7651Sdrh case DB_CACHE: { 1721fb7e7651Sdrh char *subCmd; 1722fb7e7651Sdrh int n; 1723fb7e7651Sdrh 1724fb7e7651Sdrh if( objc<=2 ){ 1725fb7e7651Sdrh Tcl_WrongNumArgs(interp, 1, objv, "cache option ?arg?"); 1726fb7e7651Sdrh return TCL_ERROR; 1727fb7e7651Sdrh } 1728fb7e7651Sdrh subCmd = Tcl_GetStringFromObj( objv[2], 0 ); 1729fb7e7651Sdrh if( *subCmd=='f' && strcmp(subCmd,"flush")==0 ){ 1730fb7e7651Sdrh if( objc!=3 ){ 1731fb7e7651Sdrh Tcl_WrongNumArgs(interp, 2, objv, "flush"); 1732fb7e7651Sdrh return TCL_ERROR; 1733fb7e7651Sdrh }else{ 1734fb7e7651Sdrh flushStmtCache( pDb ); 1735fb7e7651Sdrh } 1736fb7e7651Sdrh }else if( *subCmd=='s' && strcmp(subCmd,"size")==0 ){ 1737fb7e7651Sdrh if( objc!=4 ){ 1738fb7e7651Sdrh Tcl_WrongNumArgs(interp, 2, objv, "size n"); 1739fb7e7651Sdrh return TCL_ERROR; 1740fb7e7651Sdrh }else{ 1741fb7e7651Sdrh if( TCL_ERROR==Tcl_GetIntFromObj(interp, objv[3], &n) ){ 1742fb7e7651Sdrh Tcl_AppendResult( interp, "cannot convert \"", 1743fb7e7651Sdrh Tcl_GetStringFromObj(objv[3],0), "\" to integer", 0); 1744fb7e7651Sdrh return TCL_ERROR; 1745fb7e7651Sdrh }else{ 1746fb7e7651Sdrh if( n<0 ){ 1747fb7e7651Sdrh flushStmtCache( pDb ); 1748fb7e7651Sdrh n = 0; 1749fb7e7651Sdrh }else if( n>MAX_PREPARED_STMTS ){ 1750fb7e7651Sdrh n = MAX_PREPARED_STMTS; 1751fb7e7651Sdrh } 1752fb7e7651Sdrh pDb->maxStmt = n; 1753fb7e7651Sdrh } 1754fb7e7651Sdrh } 1755fb7e7651Sdrh }else{ 1756fb7e7651Sdrh Tcl_AppendResult( interp, "bad option \"", 1757191fadcfSdanielk1977 Tcl_GetStringFromObj(objv[2],0), "\": must be flush or size", 0); 1758fb7e7651Sdrh return TCL_ERROR; 1759fb7e7651Sdrh } 1760fb7e7651Sdrh break; 1761fb7e7651Sdrh } 1762fb7e7651Sdrh 1763b28af71aSdanielk1977 /* $db changes 1764c8d30ac1Sdrh ** 1765c8d30ac1Sdrh ** Return the number of rows that were modified, inserted, or deleted by 1766b28af71aSdanielk1977 ** the most recent INSERT, UPDATE or DELETE statement, not including 1767b28af71aSdanielk1977 ** any changes made by trigger programs. 1768c8d30ac1Sdrh */ 1769c8d30ac1Sdrh case DB_CHANGES: { 1770c8d30ac1Sdrh Tcl_Obj *pResult; 1771c8d30ac1Sdrh if( objc!=2 ){ 1772c8d30ac1Sdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 1773c8d30ac1Sdrh return TCL_ERROR; 1774c8d30ac1Sdrh } 1775c8d30ac1Sdrh pResult = Tcl_GetObjResult(interp); 1776b28af71aSdanielk1977 Tcl_SetIntObj(pResult, sqlite3_changes(pDb->db)); 1777f146a776Srdc break; 1778f146a776Srdc } 1779f146a776Srdc 178075897234Sdrh /* $db close 178175897234Sdrh ** 178275897234Sdrh ** Shutdown the database 178375897234Sdrh */ 17846d31316cSdrh case DB_CLOSE: { 17856d31316cSdrh Tcl_DeleteCommand(interp, Tcl_GetStringFromObj(objv[0], 0)); 17866d31316cSdrh break; 17876d31316cSdrh } 178875897234Sdrh 17890f14e2ebSdrh /* 17900f14e2ebSdrh ** $db collate NAME SCRIPT 17910f14e2ebSdrh ** 17920f14e2ebSdrh ** Create a new SQL collation function called NAME. Whenever 17930f14e2ebSdrh ** that function is called, invoke SCRIPT to evaluate the function. 17940f14e2ebSdrh */ 17950f14e2ebSdrh case DB_COLLATE: { 17960f14e2ebSdrh SqlCollate *pCollate; 17970f14e2ebSdrh char *zName; 17980f14e2ebSdrh char *zScript; 17990f14e2ebSdrh int nScript; 18000f14e2ebSdrh if( objc!=4 ){ 18010f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, "NAME SCRIPT"); 18020f14e2ebSdrh return TCL_ERROR; 18030f14e2ebSdrh } 18040f14e2ebSdrh zName = Tcl_GetStringFromObj(objv[2], 0); 18050f14e2ebSdrh zScript = Tcl_GetStringFromObj(objv[3], &nScript); 18060f14e2ebSdrh pCollate = (SqlCollate*)Tcl_Alloc( sizeof(*pCollate) + nScript + 1 ); 18070f14e2ebSdrh if( pCollate==0 ) return TCL_ERROR; 18080f14e2ebSdrh pCollate->interp = interp; 18090f14e2ebSdrh pCollate->pNext = pDb->pCollate; 18100f14e2ebSdrh pCollate->zScript = (char*)&pCollate[1]; 18110f14e2ebSdrh pDb->pCollate = pCollate; 18125bb3eb9bSdrh memcpy(pCollate->zScript, zScript, nScript+1); 18130f14e2ebSdrh if( sqlite3_create_collation(pDb->db, zName, SQLITE_UTF8, 18140f14e2ebSdrh pCollate, tclSqlCollate) ){ 18159636c4e1Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 18160f14e2ebSdrh return TCL_ERROR; 18170f14e2ebSdrh } 18180f14e2ebSdrh break; 18190f14e2ebSdrh } 18200f14e2ebSdrh 18210f14e2ebSdrh /* 18220f14e2ebSdrh ** $db collation_needed SCRIPT 18230f14e2ebSdrh ** 18240f14e2ebSdrh ** Create a new SQL collation function called NAME. Whenever 18250f14e2ebSdrh ** that function is called, invoke SCRIPT to evaluate the function. 18260f14e2ebSdrh */ 18270f14e2ebSdrh case DB_COLLATION_NEEDED: { 18280f14e2ebSdrh if( objc!=3 ){ 18290f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, "SCRIPT"); 18300f14e2ebSdrh return TCL_ERROR; 18310f14e2ebSdrh } 18320f14e2ebSdrh if( pDb->pCollateNeeded ){ 18330f14e2ebSdrh Tcl_DecrRefCount(pDb->pCollateNeeded); 18340f14e2ebSdrh } 18350f14e2ebSdrh pDb->pCollateNeeded = Tcl_DuplicateObj(objv[2]); 18360f14e2ebSdrh Tcl_IncrRefCount(pDb->pCollateNeeded); 18370f14e2ebSdrh sqlite3_collation_needed(pDb->db, pDb, tclCollateNeeded); 18380f14e2ebSdrh break; 18390f14e2ebSdrh } 18400f14e2ebSdrh 184119e2d37fSdrh /* $db commit_hook ?CALLBACK? 184219e2d37fSdrh ** 184319e2d37fSdrh ** Invoke the given callback just before committing every SQL transaction. 184419e2d37fSdrh ** If the callback throws an exception or returns non-zero, then the 184519e2d37fSdrh ** transaction is aborted. If CALLBACK is an empty string, the callback 184619e2d37fSdrh ** is disabled. 184719e2d37fSdrh */ 184819e2d37fSdrh case DB_COMMIT_HOOK: { 184919e2d37fSdrh if( objc>3 ){ 185019e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 185119e2d37fSdrh return TCL_ERROR; 185219e2d37fSdrh }else if( objc==2 ){ 185319e2d37fSdrh if( pDb->zCommit ){ 185419e2d37fSdrh Tcl_AppendResult(interp, pDb->zCommit, 0); 185519e2d37fSdrh } 185619e2d37fSdrh }else{ 185719e2d37fSdrh char *zCommit; 185819e2d37fSdrh int len; 185919e2d37fSdrh if( pDb->zCommit ){ 186019e2d37fSdrh Tcl_Free(pDb->zCommit); 186119e2d37fSdrh } 186219e2d37fSdrh zCommit = Tcl_GetStringFromObj(objv[2], &len); 186319e2d37fSdrh if( zCommit && len>0 ){ 186419e2d37fSdrh pDb->zCommit = Tcl_Alloc( len + 1 ); 18655bb3eb9bSdrh memcpy(pDb->zCommit, zCommit, len+1); 186619e2d37fSdrh }else{ 186719e2d37fSdrh pDb->zCommit = 0; 186819e2d37fSdrh } 186919e2d37fSdrh if( pDb->zCommit ){ 187019e2d37fSdrh pDb->interp = interp; 187119e2d37fSdrh sqlite3_commit_hook(pDb->db, DbCommitHandler, pDb); 187219e2d37fSdrh }else{ 187319e2d37fSdrh sqlite3_commit_hook(pDb->db, 0, 0); 187419e2d37fSdrh } 187519e2d37fSdrh } 187619e2d37fSdrh break; 187719e2d37fSdrh } 187819e2d37fSdrh 187975897234Sdrh /* $db complete SQL 188075897234Sdrh ** 188175897234Sdrh ** Return TRUE if SQL is a complete SQL statement. Return FALSE if 188275897234Sdrh ** additional lines of input are needed. This is similar to the 188375897234Sdrh ** built-in "info complete" command of Tcl. 188475897234Sdrh */ 18856d31316cSdrh case DB_COMPLETE: { 1886ccae6026Sdrh #ifndef SQLITE_OMIT_COMPLETE 18876d31316cSdrh Tcl_Obj *pResult; 18886d31316cSdrh int isComplete; 18896d31316cSdrh if( objc!=3 ){ 18906d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL"); 189175897234Sdrh return TCL_ERROR; 189275897234Sdrh } 18936f8a503dSdanielk1977 isComplete = sqlite3_complete( Tcl_GetStringFromObj(objv[2], 0) ); 18946d31316cSdrh pResult = Tcl_GetObjResult(interp); 18956d31316cSdrh Tcl_SetBooleanObj(pResult, isComplete); 1896ccae6026Sdrh #endif 18976d31316cSdrh break; 18986d31316cSdrh } 189975897234Sdrh 190019e2d37fSdrh /* $db copy conflict-algorithm table filename ?SEPARATOR? ?NULLINDICATOR? 190119e2d37fSdrh ** 190219e2d37fSdrh ** Copy data into table from filename, optionally using SEPARATOR 190319e2d37fSdrh ** as column separators. If a column contains a null string, or the 190419e2d37fSdrh ** value of NULLINDICATOR, a NULL is inserted for the column. 190519e2d37fSdrh ** conflict-algorithm is one of the sqlite conflict algorithms: 190619e2d37fSdrh ** rollback, abort, fail, ignore, replace 190719e2d37fSdrh ** On success, return the number of lines processed, not necessarily same 190819e2d37fSdrh ** as 'db changes' due to conflict-algorithm selected. 190919e2d37fSdrh ** 191019e2d37fSdrh ** This code is basically an implementation/enhancement of 191119e2d37fSdrh ** the sqlite3 shell.c ".import" command. 191219e2d37fSdrh ** 191319e2d37fSdrh ** This command usage is equivalent to the sqlite2.x COPY statement, 191419e2d37fSdrh ** which imports file data into a table using the PostgreSQL COPY file format: 191519e2d37fSdrh ** $db copy $conflit_algo $table_name $filename \t \\N 191619e2d37fSdrh */ 191719e2d37fSdrh case DB_COPY: { 191819e2d37fSdrh char *zTable; /* Insert data into this table */ 191919e2d37fSdrh char *zFile; /* The file from which to extract data */ 192019e2d37fSdrh char *zConflict; /* The conflict algorithm to use */ 192119e2d37fSdrh sqlite3_stmt *pStmt; /* A statement */ 192219e2d37fSdrh int nCol; /* Number of columns in the table */ 192319e2d37fSdrh int nByte; /* Number of bytes in an SQL string */ 192419e2d37fSdrh int i, j; /* Loop counters */ 192519e2d37fSdrh int nSep; /* Number of bytes in zSep[] */ 192619e2d37fSdrh int nNull; /* Number of bytes in zNull[] */ 192719e2d37fSdrh char *zSql; /* An SQL statement */ 192819e2d37fSdrh char *zLine; /* A single line of input from the file */ 192919e2d37fSdrh char **azCol; /* zLine[] broken up into columns */ 193019e2d37fSdrh char *zCommit; /* How to commit changes */ 193119e2d37fSdrh FILE *in; /* The input file */ 193219e2d37fSdrh int lineno = 0; /* Line number of input file */ 193319e2d37fSdrh char zLineNum[80]; /* Line number print buffer */ 193419e2d37fSdrh Tcl_Obj *pResult; /* interp result */ 193519e2d37fSdrh 193619e2d37fSdrh char *zSep; 193719e2d37fSdrh char *zNull; 193819e2d37fSdrh if( objc<5 || objc>7 ){ 193919e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, 194019e2d37fSdrh "CONFLICT-ALGORITHM TABLE FILENAME ?SEPARATOR? ?NULLINDICATOR?"); 194119e2d37fSdrh return TCL_ERROR; 194219e2d37fSdrh } 194319e2d37fSdrh if( objc>=6 ){ 194419e2d37fSdrh zSep = Tcl_GetStringFromObj(objv[5], 0); 194519e2d37fSdrh }else{ 194619e2d37fSdrh zSep = "\t"; 194719e2d37fSdrh } 194819e2d37fSdrh if( objc>=7 ){ 194919e2d37fSdrh zNull = Tcl_GetStringFromObj(objv[6], 0); 195019e2d37fSdrh }else{ 195119e2d37fSdrh zNull = ""; 195219e2d37fSdrh } 195319e2d37fSdrh zConflict = Tcl_GetStringFromObj(objv[2], 0); 195419e2d37fSdrh zTable = Tcl_GetStringFromObj(objv[3], 0); 195519e2d37fSdrh zFile = Tcl_GetStringFromObj(objv[4], 0); 19564f21c4afSdrh nSep = strlen30(zSep); 19574f21c4afSdrh nNull = strlen30(zNull); 195819e2d37fSdrh if( nSep==0 ){ 195919e2d37fSdrh Tcl_AppendResult(interp,"Error: non-null separator required for copy",0); 196019e2d37fSdrh return TCL_ERROR; 196119e2d37fSdrh } 19623e59c012Sdrh if(strcmp(zConflict, "rollback") != 0 && 19633e59c012Sdrh strcmp(zConflict, "abort" ) != 0 && 19643e59c012Sdrh strcmp(zConflict, "fail" ) != 0 && 19653e59c012Sdrh strcmp(zConflict, "ignore" ) != 0 && 19663e59c012Sdrh strcmp(zConflict, "replace" ) != 0 ) { 196719e2d37fSdrh Tcl_AppendResult(interp, "Error: \"", zConflict, 196819e2d37fSdrh "\", conflict-algorithm must be one of: rollback, " 196919e2d37fSdrh "abort, fail, ignore, or replace", 0); 197019e2d37fSdrh return TCL_ERROR; 197119e2d37fSdrh } 197219e2d37fSdrh zSql = sqlite3_mprintf("SELECT * FROM '%q'", zTable); 197319e2d37fSdrh if( zSql==0 ){ 197419e2d37fSdrh Tcl_AppendResult(interp, "Error: no such table: ", zTable, 0); 197519e2d37fSdrh return TCL_ERROR; 197619e2d37fSdrh } 19774f21c4afSdrh nByte = strlen30(zSql); 19783e701a18Sdrh rc = sqlite3_prepare(pDb->db, zSql, -1, &pStmt, 0); 197919e2d37fSdrh sqlite3_free(zSql); 198019e2d37fSdrh if( rc ){ 198119e2d37fSdrh Tcl_AppendResult(interp, "Error: ", sqlite3_errmsg(pDb->db), 0); 198219e2d37fSdrh nCol = 0; 198319e2d37fSdrh }else{ 198419e2d37fSdrh nCol = sqlite3_column_count(pStmt); 198519e2d37fSdrh } 198619e2d37fSdrh sqlite3_finalize(pStmt); 198719e2d37fSdrh if( nCol==0 ) { 198819e2d37fSdrh return TCL_ERROR; 198919e2d37fSdrh } 199019e2d37fSdrh zSql = malloc( nByte + 50 + nCol*2 ); 199119e2d37fSdrh if( zSql==0 ) { 199219e2d37fSdrh Tcl_AppendResult(interp, "Error: can't malloc()", 0); 199319e2d37fSdrh return TCL_ERROR; 199419e2d37fSdrh } 199519e2d37fSdrh sqlite3_snprintf(nByte+50, zSql, "INSERT OR %q INTO '%q' VALUES(?", 199619e2d37fSdrh zConflict, zTable); 19974f21c4afSdrh j = strlen30(zSql); 199819e2d37fSdrh for(i=1; i<nCol; i++){ 199919e2d37fSdrh zSql[j++] = ','; 200019e2d37fSdrh zSql[j++] = '?'; 200119e2d37fSdrh } 200219e2d37fSdrh zSql[j++] = ')'; 200319e2d37fSdrh zSql[j] = 0; 20043e701a18Sdrh rc = sqlite3_prepare(pDb->db, zSql, -1, &pStmt, 0); 200519e2d37fSdrh free(zSql); 200619e2d37fSdrh if( rc ){ 200719e2d37fSdrh Tcl_AppendResult(interp, "Error: ", sqlite3_errmsg(pDb->db), 0); 200819e2d37fSdrh sqlite3_finalize(pStmt); 200919e2d37fSdrh return TCL_ERROR; 201019e2d37fSdrh } 201119e2d37fSdrh in = fopen(zFile, "rb"); 201219e2d37fSdrh if( in==0 ){ 201319e2d37fSdrh Tcl_AppendResult(interp, "Error: cannot open file: ", zFile, NULL); 201419e2d37fSdrh sqlite3_finalize(pStmt); 201519e2d37fSdrh return TCL_ERROR; 201619e2d37fSdrh } 201719e2d37fSdrh azCol = malloc( sizeof(azCol[0])*(nCol+1) ); 201819e2d37fSdrh if( azCol==0 ) { 201919e2d37fSdrh Tcl_AppendResult(interp, "Error: can't malloc()", 0); 202043617e9aSdrh fclose(in); 202119e2d37fSdrh return TCL_ERROR; 202219e2d37fSdrh } 20233752785fSdrh (void)sqlite3_exec(pDb->db, "BEGIN", 0, 0, 0); 202419e2d37fSdrh zCommit = "COMMIT"; 202519e2d37fSdrh while( (zLine = local_getline(0, in))!=0 ){ 202619e2d37fSdrh char *z; 202719e2d37fSdrh i = 0; 202819e2d37fSdrh lineno++; 202919e2d37fSdrh azCol[0] = zLine; 203019e2d37fSdrh for(i=0, z=zLine; *z; z++){ 203119e2d37fSdrh if( *z==zSep[0] && strncmp(z, zSep, nSep)==0 ){ 203219e2d37fSdrh *z = 0; 203319e2d37fSdrh i++; 203419e2d37fSdrh if( i<nCol ){ 203519e2d37fSdrh azCol[i] = &z[nSep]; 203619e2d37fSdrh z += nSep-1; 203719e2d37fSdrh } 203819e2d37fSdrh } 203919e2d37fSdrh } 204019e2d37fSdrh if( i+1!=nCol ){ 204119e2d37fSdrh char *zErr; 20424f21c4afSdrh int nErr = strlen30(zFile) + 200; 20435bb3eb9bSdrh zErr = malloc(nErr); 2044c1f4494eSdrh if( zErr ){ 20455bb3eb9bSdrh sqlite3_snprintf(nErr, zErr, 2046955de52cSdanielk1977 "Error: %s line %d: expected %d columns of data but found %d", 204719e2d37fSdrh zFile, lineno, nCol, i+1); 204819e2d37fSdrh Tcl_AppendResult(interp, zErr, 0); 204919e2d37fSdrh free(zErr); 2050c1f4494eSdrh } 205119e2d37fSdrh zCommit = "ROLLBACK"; 205219e2d37fSdrh break; 205319e2d37fSdrh } 205419e2d37fSdrh for(i=0; i<nCol; i++){ 205519e2d37fSdrh /* check for null data, if so, bind as null */ 2056ea678832Sdrh if( (nNull>0 && strcmp(azCol[i], zNull)==0) 20574f21c4afSdrh || strlen30(azCol[i])==0 2058ea678832Sdrh ){ 205919e2d37fSdrh sqlite3_bind_null(pStmt, i+1); 206019e2d37fSdrh }else{ 206119e2d37fSdrh sqlite3_bind_text(pStmt, i+1, azCol[i], -1, SQLITE_STATIC); 206219e2d37fSdrh } 206319e2d37fSdrh } 206419e2d37fSdrh sqlite3_step(pStmt); 206519e2d37fSdrh rc = sqlite3_reset(pStmt); 206619e2d37fSdrh free(zLine); 206719e2d37fSdrh if( rc!=SQLITE_OK ){ 206819e2d37fSdrh Tcl_AppendResult(interp,"Error: ", sqlite3_errmsg(pDb->db), 0); 206919e2d37fSdrh zCommit = "ROLLBACK"; 207019e2d37fSdrh break; 207119e2d37fSdrh } 207219e2d37fSdrh } 207319e2d37fSdrh free(azCol); 207419e2d37fSdrh fclose(in); 207519e2d37fSdrh sqlite3_finalize(pStmt); 20763752785fSdrh (void)sqlite3_exec(pDb->db, zCommit, 0, 0, 0); 207719e2d37fSdrh 207819e2d37fSdrh if( zCommit[0] == 'C' ){ 207919e2d37fSdrh /* success, set result as number of lines processed */ 208019e2d37fSdrh pResult = Tcl_GetObjResult(interp); 208119e2d37fSdrh Tcl_SetIntObj(pResult, lineno); 208219e2d37fSdrh rc = TCL_OK; 208319e2d37fSdrh }else{ 208419e2d37fSdrh /* failure, append lineno where failed */ 20855bb3eb9bSdrh sqlite3_snprintf(sizeof(zLineNum), zLineNum,"%d",lineno); 208619e2d37fSdrh Tcl_AppendResult(interp,", failed while processing line: ",zLineNum,0); 208719e2d37fSdrh rc = TCL_ERROR; 208819e2d37fSdrh } 208919e2d37fSdrh break; 209019e2d37fSdrh } 209119e2d37fSdrh 209275897234Sdrh /* 20934144905bSdrh ** $db enable_load_extension BOOLEAN 20944144905bSdrh ** 20954144905bSdrh ** Turn the extension loading feature on or off. It if off by 20964144905bSdrh ** default. 20974144905bSdrh */ 20984144905bSdrh case DB_ENABLE_LOAD_EXTENSION: { 2099f533acc0Sdrh #ifndef SQLITE_OMIT_LOAD_EXTENSION 21004144905bSdrh int onoff; 21014144905bSdrh if( objc!=3 ){ 21024144905bSdrh Tcl_WrongNumArgs(interp, 2, objv, "BOOLEAN"); 21034144905bSdrh return TCL_ERROR; 21044144905bSdrh } 21054144905bSdrh if( Tcl_GetBooleanFromObj(interp, objv[2], &onoff) ){ 21064144905bSdrh return TCL_ERROR; 21074144905bSdrh } 21084144905bSdrh sqlite3_enable_load_extension(pDb->db, onoff); 21094144905bSdrh break; 2110f533acc0Sdrh #else 2111f533acc0Sdrh Tcl_AppendResult(interp, "extension loading is turned off at compile-time", 2112f533acc0Sdrh 0); 2113f533acc0Sdrh return TCL_ERROR; 2114f533acc0Sdrh #endif 21154144905bSdrh } 21164144905bSdrh 21174144905bSdrh /* 2118dcd997eaSdrh ** $db errorcode 2119dcd997eaSdrh ** 2120dcd997eaSdrh ** Return the numeric error code that was returned by the most recent 21216f8a503dSdanielk1977 ** call to sqlite3_exec(). 2122dcd997eaSdrh */ 2123dcd997eaSdrh case DB_ERRORCODE: { 2124f3ce83f5Sdanielk1977 Tcl_SetObjResult(interp, Tcl_NewIntObj(sqlite3_errcode(pDb->db))); 2125dcd997eaSdrh break; 2126dcd997eaSdrh } 2127dcd997eaSdrh 2128dcd997eaSdrh /* 21294a4c11aaSdan ** $db exists $sql 21301807ce37Sdrh ** $db onecolumn $sql 213175897234Sdrh ** 21324a4c11aaSdan ** The onecolumn method is the equivalent of: 21334a4c11aaSdan ** lindex [$db eval $sql] 0 21344a4c11aaSdan */ 21354a4c11aaSdan case DB_EXISTS: 21364a4c11aaSdan case DB_ONECOLUMN: { 21374a4c11aaSdan DbEvalContext sEval; 21384a4c11aaSdan if( objc!=3 ){ 21394a4c11aaSdan Tcl_WrongNumArgs(interp, 2, objv, "SQL"); 21404a4c11aaSdan return TCL_ERROR; 21414a4c11aaSdan } 21424a4c11aaSdan 21434a4c11aaSdan dbEvalInit(&sEval, pDb, objv[2], 0); 21444a4c11aaSdan rc = dbEvalStep(&sEval); 21454a4c11aaSdan if( choice==DB_ONECOLUMN ){ 21464a4c11aaSdan if( rc==TCL_OK ){ 21474a4c11aaSdan Tcl_SetObjResult(interp, dbEvalColumnValue(&sEval, 0)); 21484a4c11aaSdan } 21494a4c11aaSdan }else if( rc==TCL_BREAK || rc==TCL_OK ){ 21504a4c11aaSdan Tcl_SetObjResult(interp, Tcl_NewBooleanObj(rc==TCL_OK)); 21514a4c11aaSdan } 21524a4c11aaSdan dbEvalFinalize(&sEval); 21534a4c11aaSdan 21544a4c11aaSdan if( rc==TCL_BREAK ){ 21554a4c11aaSdan rc = TCL_OK; 21564a4c11aaSdan } 21574a4c11aaSdan break; 21584a4c11aaSdan } 21594a4c11aaSdan 21604a4c11aaSdan /* 21614a4c11aaSdan ** $db eval $sql ?array? ?{ ...code... }? 21624a4c11aaSdan ** 216375897234Sdrh ** The SQL statement in $sql is evaluated. For each row, the values are 2164bec3f402Sdrh ** placed in elements of the array named "array" and ...code... is executed. 216575897234Sdrh ** If "array" and "code" are omitted, then no callback is every invoked. 216675897234Sdrh ** If "array" is an empty string, then the values are placed in variables 216775897234Sdrh ** that have the same name as the fields extracted by the query. 216875897234Sdrh */ 21694a4c11aaSdan case DB_EVAL: { 217092febd92Sdrh if( objc<3 || objc>5 ){ 2171895d7472Sdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL ?ARRAY-NAME? ?SCRIPT?"); 217230ccda10Sdanielk1977 return TCL_ERROR; 217330ccda10Sdanielk1977 } 21744a4c11aaSdan 217592febd92Sdrh if( objc==3 ){ 21764a4c11aaSdan DbEvalContext sEval; 21774a4c11aaSdan Tcl_Obj *pRet = Tcl_NewObj(); 21784a4c11aaSdan Tcl_IncrRefCount(pRet); 21794a4c11aaSdan dbEvalInit(&sEval, pDb, objv[2], 0); 21804a4c11aaSdan while( TCL_OK==(rc = dbEvalStep(&sEval)) ){ 21814a4c11aaSdan int i; 21824a4c11aaSdan int nCol; 21834a4c11aaSdan dbEvalRowInfo(&sEval, &nCol, 0); 218492febd92Sdrh for(i=0; i<nCol; i++){ 21854a4c11aaSdan Tcl_ListObjAppendElement(interp, pRet, dbEvalColumnValue(&sEval, i)); 218692febd92Sdrh } 218730ccda10Sdanielk1977 } 21884a4c11aaSdan dbEvalFinalize(&sEval); 218990b6bb19Sdrh if( rc==TCL_BREAK ){ 21904a4c11aaSdan Tcl_SetObjResult(interp, pRet); 219190b6bb19Sdrh rc = TCL_OK; 219290b6bb19Sdrh } 2193ef2cb63eSdanielk1977 Tcl_DecrRefCount(pRet); 21944a4c11aaSdan }else{ 21954a4c11aaSdan ClientData cd[2]; 21964a4c11aaSdan DbEvalContext *p; 21974a4c11aaSdan Tcl_Obj *pArray = 0; 21984a4c11aaSdan Tcl_Obj *pScript; 21994a4c11aaSdan 22004a4c11aaSdan if( objc==5 && *(char *)Tcl_GetString(objv[3]) ){ 22014a4c11aaSdan pArray = objv[3]; 22024a4c11aaSdan } 22034a4c11aaSdan pScript = objv[objc-1]; 22044a4c11aaSdan Tcl_IncrRefCount(pScript); 22054a4c11aaSdan 22064a4c11aaSdan p = (DbEvalContext *)Tcl_Alloc(sizeof(DbEvalContext)); 22074a4c11aaSdan dbEvalInit(p, pDb, objv[2], pArray); 22084a4c11aaSdan 22094a4c11aaSdan cd[0] = (void *)p; 22104a4c11aaSdan cd[1] = (void *)pScript; 22114a4c11aaSdan rc = DbEvalNextCmd(cd, interp, TCL_OK); 22121807ce37Sdrh } 221330ccda10Sdanielk1977 break; 221430ccda10Sdanielk1977 } 2215bec3f402Sdrh 2216bec3f402Sdrh /* 2217e3602be8Sdrh ** $db function NAME [-argcount N] SCRIPT 2218cabb0819Sdrh ** 2219cabb0819Sdrh ** Create a new SQL function called NAME. Whenever that function is 2220cabb0819Sdrh ** called, invoke SCRIPT to evaluate the function. 2221cabb0819Sdrh */ 2222cabb0819Sdrh case DB_FUNCTION: { 2223cabb0819Sdrh SqlFunc *pFunc; 2224d1e4733dSdrh Tcl_Obj *pScript; 2225cabb0819Sdrh char *zName; 2226e3602be8Sdrh int nArg = -1; 2227e3602be8Sdrh if( objc==6 ){ 2228e3602be8Sdrh const char *z = Tcl_GetString(objv[3]); 22294f21c4afSdrh int n = strlen30(z); 2230e3602be8Sdrh if( n>2 && strncmp(z, "-argcount",n)==0 ){ 2231e3602be8Sdrh if( Tcl_GetIntFromObj(interp, objv[4], &nArg) ) return TCL_ERROR; 2232e3602be8Sdrh if( nArg<0 ){ 2233e3602be8Sdrh Tcl_AppendResult(interp, "number of arguments must be non-negative", 2234e3602be8Sdrh (char*)0); 2235cabb0819Sdrh return TCL_ERROR; 2236cabb0819Sdrh } 2237e3602be8Sdrh } 2238e3602be8Sdrh pScript = objv[5]; 2239e3602be8Sdrh }else if( objc!=4 ){ 2240e3602be8Sdrh Tcl_WrongNumArgs(interp, 2, objv, "NAME [-argcount N] SCRIPT"); 2241e3602be8Sdrh return TCL_ERROR; 2242e3602be8Sdrh }else{ 2243d1e4733dSdrh pScript = objv[3]; 2244e3602be8Sdrh } 2245e3602be8Sdrh zName = Tcl_GetStringFromObj(objv[2], 0); 2246d1e4733dSdrh pFunc = findSqlFunc(pDb, zName); 2247cabb0819Sdrh if( pFunc==0 ) return TCL_ERROR; 2248d1e4733dSdrh if( pFunc->pScript ){ 2249d1e4733dSdrh Tcl_DecrRefCount(pFunc->pScript); 2250d1e4733dSdrh } 2251d1e4733dSdrh pFunc->pScript = pScript; 2252d1e4733dSdrh Tcl_IncrRefCount(pScript); 2253d1e4733dSdrh pFunc->useEvalObjv = safeToUseEvalObjv(interp, pScript); 2254e3602be8Sdrh rc = sqlite3_create_function(pDb->db, zName, nArg, SQLITE_UTF8, 2255d8123366Sdanielk1977 pFunc, tclSqlFunc, 0, 0); 2256fb7e7651Sdrh if( rc!=SQLITE_OK ){ 2257fb7e7651Sdrh rc = TCL_ERROR; 22589636c4e1Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 2259fb7e7651Sdrh } 2260cabb0819Sdrh break; 2261cabb0819Sdrh } 2262cabb0819Sdrh 2263cabb0819Sdrh /* 22648cbadb02Sdanielk1977 ** $db incrblob ?-readonly? ?DB? TABLE COLUMN ROWID 2265b4e9af9fSdanielk1977 */ 2266b4e9af9fSdanielk1977 case DB_INCRBLOB: { 226732a0d8bbSdanielk1977 #ifdef SQLITE_OMIT_INCRBLOB 226832a0d8bbSdanielk1977 Tcl_AppendResult(interp, "incrblob not available in this build", 0); 226932a0d8bbSdanielk1977 return TCL_ERROR; 227032a0d8bbSdanielk1977 #else 22718cbadb02Sdanielk1977 int isReadonly = 0; 2272b4e9af9fSdanielk1977 const char *zDb = "main"; 2273b4e9af9fSdanielk1977 const char *zTable; 2274b4e9af9fSdanielk1977 const char *zColumn; 2275b4e9af9fSdanielk1977 sqlite_int64 iRow; 2276b4e9af9fSdanielk1977 22778cbadb02Sdanielk1977 /* Check for the -readonly option */ 22788cbadb02Sdanielk1977 if( objc>3 && strcmp(Tcl_GetString(objv[2]), "-readonly")==0 ){ 22798cbadb02Sdanielk1977 isReadonly = 1; 22808cbadb02Sdanielk1977 } 22818cbadb02Sdanielk1977 22828cbadb02Sdanielk1977 if( objc!=(5+isReadonly) && objc!=(6+isReadonly) ){ 22838cbadb02Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?-readonly? ?DB? TABLE COLUMN ROWID"); 2284b4e9af9fSdanielk1977 return TCL_ERROR; 2285b4e9af9fSdanielk1977 } 2286b4e9af9fSdanielk1977 22878cbadb02Sdanielk1977 if( objc==(6+isReadonly) ){ 2288b4e9af9fSdanielk1977 zDb = Tcl_GetString(objv[2]); 2289b4e9af9fSdanielk1977 } 2290b4e9af9fSdanielk1977 zTable = Tcl_GetString(objv[objc-3]); 2291b4e9af9fSdanielk1977 zColumn = Tcl_GetString(objv[objc-2]); 2292b4e9af9fSdanielk1977 rc = Tcl_GetWideIntFromObj(interp, objv[objc-1], &iRow); 2293b4e9af9fSdanielk1977 2294b4e9af9fSdanielk1977 if( rc==TCL_OK ){ 22958cbadb02Sdanielk1977 rc = createIncrblobChannel( 22968cbadb02Sdanielk1977 interp, pDb, zDb, zTable, zColumn, iRow, isReadonly 22978cbadb02Sdanielk1977 ); 2298b4e9af9fSdanielk1977 } 229932a0d8bbSdanielk1977 #endif 2300b4e9af9fSdanielk1977 break; 2301b4e9af9fSdanielk1977 } 2302b4e9af9fSdanielk1977 2303b4e9af9fSdanielk1977 /* 2304f11bded5Sdrh ** $db interrupt 2305f11bded5Sdrh ** 2306f11bded5Sdrh ** Interrupt the execution of the inner-most SQL interpreter. This 2307f11bded5Sdrh ** causes the SQL statement to return an error of SQLITE_INTERRUPT. 2308f11bded5Sdrh */ 2309f11bded5Sdrh case DB_INTERRUPT: { 2310f11bded5Sdrh sqlite3_interrupt(pDb->db); 2311f11bded5Sdrh break; 2312f11bded5Sdrh } 2313f11bded5Sdrh 2314f11bded5Sdrh /* 231519e2d37fSdrh ** $db nullvalue ?STRING? 231619e2d37fSdrh ** 231719e2d37fSdrh ** Change text used when a NULL comes back from the database. If ?STRING? 231819e2d37fSdrh ** is not present, then the current string used for NULL is returned. 231919e2d37fSdrh ** If STRING is present, then STRING is returned. 232019e2d37fSdrh ** 232119e2d37fSdrh */ 232219e2d37fSdrh case DB_NULLVALUE: { 232319e2d37fSdrh if( objc!=2 && objc!=3 ){ 232419e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "NULLVALUE"); 232519e2d37fSdrh return TCL_ERROR; 232619e2d37fSdrh } 232719e2d37fSdrh if( objc==3 ){ 232819e2d37fSdrh int len; 232919e2d37fSdrh char *zNull = Tcl_GetStringFromObj(objv[2], &len); 233019e2d37fSdrh if( pDb->zNull ){ 233119e2d37fSdrh Tcl_Free(pDb->zNull); 233219e2d37fSdrh } 233319e2d37fSdrh if( zNull && len>0 ){ 233419e2d37fSdrh pDb->zNull = Tcl_Alloc( len + 1 ); 233519e2d37fSdrh strncpy(pDb->zNull, zNull, len); 233619e2d37fSdrh pDb->zNull[len] = '\0'; 233719e2d37fSdrh }else{ 233819e2d37fSdrh pDb->zNull = 0; 233919e2d37fSdrh } 234019e2d37fSdrh } 234119e2d37fSdrh Tcl_SetObjResult(interp, dbTextToObj(pDb->zNull)); 234219e2d37fSdrh break; 234319e2d37fSdrh } 234419e2d37fSdrh 234519e2d37fSdrh /* 2346af9ff33aSdrh ** $db last_insert_rowid 2347af9ff33aSdrh ** 2348af9ff33aSdrh ** Return an integer which is the ROWID for the most recent insert. 2349af9ff33aSdrh */ 2350af9ff33aSdrh case DB_LAST_INSERT_ROWID: { 2351af9ff33aSdrh Tcl_Obj *pResult; 2352f7e678d6Sdrh Tcl_WideInt rowid; 2353af9ff33aSdrh if( objc!=2 ){ 2354af9ff33aSdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 2355af9ff33aSdrh return TCL_ERROR; 2356af9ff33aSdrh } 23576f8a503dSdanielk1977 rowid = sqlite3_last_insert_rowid(pDb->db); 2358af9ff33aSdrh pResult = Tcl_GetObjResult(interp); 2359f7e678d6Sdrh Tcl_SetWideIntObj(pResult, rowid); 2360af9ff33aSdrh break; 2361af9ff33aSdrh } 2362af9ff33aSdrh 2363af9ff33aSdrh /* 23644a4c11aaSdan ** The DB_ONECOLUMN method is implemented together with DB_EXISTS. 23655d9d7576Sdrh */ 23661807ce37Sdrh 23671807ce37Sdrh /* $db progress ?N CALLBACK? 23681807ce37Sdrh ** 23691807ce37Sdrh ** Invoke the given callback every N virtual machine opcodes while executing 23701807ce37Sdrh ** queries. 23711807ce37Sdrh */ 23721807ce37Sdrh case DB_PROGRESS: { 23731807ce37Sdrh if( objc==2 ){ 23741807ce37Sdrh if( pDb->zProgress ){ 23751807ce37Sdrh Tcl_AppendResult(interp, pDb->zProgress, 0); 23765d9d7576Sdrh } 23771807ce37Sdrh }else if( objc==4 ){ 23781807ce37Sdrh char *zProgress; 23791807ce37Sdrh int len; 23801807ce37Sdrh int N; 23811807ce37Sdrh if( TCL_OK!=Tcl_GetIntFromObj(interp, objv[2], &N) ){ 23821807ce37Sdrh return TCL_ERROR; 23831807ce37Sdrh }; 23841807ce37Sdrh if( pDb->zProgress ){ 23851807ce37Sdrh Tcl_Free(pDb->zProgress); 23861807ce37Sdrh } 23871807ce37Sdrh zProgress = Tcl_GetStringFromObj(objv[3], &len); 23881807ce37Sdrh if( zProgress && len>0 ){ 23891807ce37Sdrh pDb->zProgress = Tcl_Alloc( len + 1 ); 23905bb3eb9bSdrh memcpy(pDb->zProgress, zProgress, len+1); 23911807ce37Sdrh }else{ 23921807ce37Sdrh pDb->zProgress = 0; 23931807ce37Sdrh } 23941807ce37Sdrh #ifndef SQLITE_OMIT_PROGRESS_CALLBACK 23951807ce37Sdrh if( pDb->zProgress ){ 23961807ce37Sdrh pDb->interp = interp; 23971807ce37Sdrh sqlite3_progress_handler(pDb->db, N, DbProgressHandler, pDb); 23981807ce37Sdrh }else{ 23991807ce37Sdrh sqlite3_progress_handler(pDb->db, 0, 0, 0); 24001807ce37Sdrh } 24011807ce37Sdrh #endif 24021807ce37Sdrh }else{ 24031807ce37Sdrh Tcl_WrongNumArgs(interp, 2, objv, "N CALLBACK"); 24041807ce37Sdrh return TCL_ERROR; 24055d9d7576Sdrh } 24065d9d7576Sdrh break; 24075d9d7576Sdrh } 24085d9d7576Sdrh 240919e2d37fSdrh /* $db profile ?CALLBACK? 241019e2d37fSdrh ** 241119e2d37fSdrh ** Make arrangements to invoke the CALLBACK routine after each SQL statement 241219e2d37fSdrh ** that has run. The text of the SQL and the amount of elapse time are 241319e2d37fSdrh ** appended to CALLBACK before the script is run. 241419e2d37fSdrh */ 241519e2d37fSdrh case DB_PROFILE: { 241619e2d37fSdrh if( objc>3 ){ 241719e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 241819e2d37fSdrh return TCL_ERROR; 241919e2d37fSdrh }else if( objc==2 ){ 242019e2d37fSdrh if( pDb->zProfile ){ 242119e2d37fSdrh Tcl_AppendResult(interp, pDb->zProfile, 0); 242219e2d37fSdrh } 242319e2d37fSdrh }else{ 242419e2d37fSdrh char *zProfile; 242519e2d37fSdrh int len; 242619e2d37fSdrh if( pDb->zProfile ){ 242719e2d37fSdrh Tcl_Free(pDb->zProfile); 242819e2d37fSdrh } 242919e2d37fSdrh zProfile = Tcl_GetStringFromObj(objv[2], &len); 243019e2d37fSdrh if( zProfile && len>0 ){ 243119e2d37fSdrh pDb->zProfile = Tcl_Alloc( len + 1 ); 24325bb3eb9bSdrh memcpy(pDb->zProfile, zProfile, len+1); 243319e2d37fSdrh }else{ 243419e2d37fSdrh pDb->zProfile = 0; 243519e2d37fSdrh } 243619e2d37fSdrh #ifndef SQLITE_OMIT_TRACE 243719e2d37fSdrh if( pDb->zProfile ){ 243819e2d37fSdrh pDb->interp = interp; 243919e2d37fSdrh sqlite3_profile(pDb->db, DbProfileHandler, pDb); 244019e2d37fSdrh }else{ 244119e2d37fSdrh sqlite3_profile(pDb->db, 0, 0); 244219e2d37fSdrh } 244319e2d37fSdrh #endif 244419e2d37fSdrh } 244519e2d37fSdrh break; 244619e2d37fSdrh } 244719e2d37fSdrh 24485d9d7576Sdrh /* 244922fbcb8dSdrh ** $db rekey KEY 245022fbcb8dSdrh ** 245122fbcb8dSdrh ** Change the encryption key on the currently open database. 245222fbcb8dSdrh */ 245322fbcb8dSdrh case DB_REKEY: { 245422fbcb8dSdrh int nKey; 245522fbcb8dSdrh void *pKey; 245622fbcb8dSdrh if( objc!=3 ){ 245722fbcb8dSdrh Tcl_WrongNumArgs(interp, 2, objv, "KEY"); 245822fbcb8dSdrh return TCL_ERROR; 245922fbcb8dSdrh } 246022fbcb8dSdrh pKey = Tcl_GetByteArrayFromObj(objv[2], &nKey); 24619eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 24622011d5f5Sdrh rc = sqlite3_rekey(pDb->db, pKey, nKey); 246322fbcb8dSdrh if( rc ){ 2464f20b21c8Sdanielk1977 Tcl_AppendResult(interp, sqlite3ErrStr(rc), 0); 246522fbcb8dSdrh rc = TCL_ERROR; 246622fbcb8dSdrh } 246722fbcb8dSdrh #endif 246822fbcb8dSdrh break; 246922fbcb8dSdrh } 247022fbcb8dSdrh 2471dc2c4915Sdrh /* $db restore ?DATABASE? FILENAME 2472dc2c4915Sdrh ** 2473dc2c4915Sdrh ** Open a database file named FILENAME. Transfer the content 2474dc2c4915Sdrh ** of FILENAME into the local database DATABASE (default: "main"). 2475dc2c4915Sdrh */ 2476dc2c4915Sdrh case DB_RESTORE: { 2477dc2c4915Sdrh const char *zSrcFile; 2478dc2c4915Sdrh const char *zDestDb; 2479dc2c4915Sdrh sqlite3 *pSrc; 2480dc2c4915Sdrh sqlite3_backup *pBackup; 2481dc2c4915Sdrh int nTimeout = 0; 2482dc2c4915Sdrh 2483dc2c4915Sdrh if( objc==3 ){ 2484dc2c4915Sdrh zDestDb = "main"; 2485dc2c4915Sdrh zSrcFile = Tcl_GetString(objv[2]); 2486dc2c4915Sdrh }else if( objc==4 ){ 2487dc2c4915Sdrh zDestDb = Tcl_GetString(objv[2]); 2488dc2c4915Sdrh zSrcFile = Tcl_GetString(objv[3]); 2489dc2c4915Sdrh }else{ 2490dc2c4915Sdrh Tcl_WrongNumArgs(interp, 2, objv, "?DATABASE? FILENAME"); 2491dc2c4915Sdrh return TCL_ERROR; 2492dc2c4915Sdrh } 2493dc2c4915Sdrh rc = sqlite3_open_v2(zSrcFile, &pSrc, SQLITE_OPEN_READONLY, 0); 2494dc2c4915Sdrh if( rc!=SQLITE_OK ){ 2495dc2c4915Sdrh Tcl_AppendResult(interp, "cannot open source database: ", 2496dc2c4915Sdrh sqlite3_errmsg(pSrc), (char*)0); 2497dc2c4915Sdrh sqlite3_close(pSrc); 2498dc2c4915Sdrh return TCL_ERROR; 2499dc2c4915Sdrh } 2500dc2c4915Sdrh pBackup = sqlite3_backup_init(pDb->db, zDestDb, pSrc, "main"); 2501dc2c4915Sdrh if( pBackup==0 ){ 2502dc2c4915Sdrh Tcl_AppendResult(interp, "restore failed: ", 2503dc2c4915Sdrh sqlite3_errmsg(pDb->db), (char*)0); 2504dc2c4915Sdrh sqlite3_close(pSrc); 2505dc2c4915Sdrh return TCL_ERROR; 2506dc2c4915Sdrh } 2507dc2c4915Sdrh while( (rc = sqlite3_backup_step(pBackup,100))==SQLITE_OK 2508dc2c4915Sdrh || rc==SQLITE_BUSY ){ 2509dc2c4915Sdrh if( rc==SQLITE_BUSY ){ 2510dc2c4915Sdrh if( nTimeout++ >= 3 ) break; 2511dc2c4915Sdrh sqlite3_sleep(100); 2512dc2c4915Sdrh } 2513dc2c4915Sdrh } 2514dc2c4915Sdrh sqlite3_backup_finish(pBackup); 2515dc2c4915Sdrh if( rc==SQLITE_DONE ){ 2516dc2c4915Sdrh rc = TCL_OK; 2517dc2c4915Sdrh }else if( rc==SQLITE_BUSY || rc==SQLITE_LOCKED ){ 2518dc2c4915Sdrh Tcl_AppendResult(interp, "restore failed: source database busy", 2519dc2c4915Sdrh (char*)0); 2520dc2c4915Sdrh rc = TCL_ERROR; 2521dc2c4915Sdrh }else{ 2522dc2c4915Sdrh Tcl_AppendResult(interp, "restore failed: ", 2523dc2c4915Sdrh sqlite3_errmsg(pDb->db), (char*)0); 2524dc2c4915Sdrh rc = TCL_ERROR; 2525dc2c4915Sdrh } 2526dc2c4915Sdrh sqlite3_close(pSrc); 2527dc2c4915Sdrh break; 2528dc2c4915Sdrh } 2529dc2c4915Sdrh 253022fbcb8dSdrh /* 2531d1d38488Sdrh ** $db status (step|sort) 2532d1d38488Sdrh ** 2533d1d38488Sdrh ** Display SQLITE_STMTSTATUS_FULLSCAN_STEP or 2534d1d38488Sdrh ** SQLITE_STMTSTATUS_SORT for the most recent eval. 2535d1d38488Sdrh */ 2536d1d38488Sdrh case DB_STATUS: { 2537d1d38488Sdrh int v; 2538d1d38488Sdrh const char *zOp; 2539d1d38488Sdrh if( objc!=3 ){ 2540d1d38488Sdrh Tcl_WrongNumArgs(interp, 2, objv, "(step|sort)"); 2541d1d38488Sdrh return TCL_ERROR; 2542d1d38488Sdrh } 2543d1d38488Sdrh zOp = Tcl_GetString(objv[2]); 2544d1d38488Sdrh if( strcmp(zOp, "step")==0 ){ 2545d1d38488Sdrh v = pDb->nStep; 2546d1d38488Sdrh }else if( strcmp(zOp, "sort")==0 ){ 2547d1d38488Sdrh v = pDb->nSort; 2548d1d38488Sdrh }else{ 2549d1d38488Sdrh Tcl_AppendResult(interp, "bad argument: should be step or sort", 2550d1d38488Sdrh (char*)0); 2551d1d38488Sdrh return TCL_ERROR; 2552d1d38488Sdrh } 2553d1d38488Sdrh Tcl_SetObjResult(interp, Tcl_NewIntObj(v)); 2554d1d38488Sdrh break; 2555d1d38488Sdrh } 2556d1d38488Sdrh 2557d1d38488Sdrh /* 2558bec3f402Sdrh ** $db timeout MILLESECONDS 2559bec3f402Sdrh ** 2560bec3f402Sdrh ** Delay for the number of milliseconds specified when a file is locked. 2561bec3f402Sdrh */ 25626d31316cSdrh case DB_TIMEOUT: { 2563bec3f402Sdrh int ms; 25646d31316cSdrh if( objc!=3 ){ 25656d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "MILLISECONDS"); 2566bec3f402Sdrh return TCL_ERROR; 256775897234Sdrh } 25686d31316cSdrh if( Tcl_GetIntFromObj(interp, objv[2], &ms) ) return TCL_ERROR; 25696f8a503dSdanielk1977 sqlite3_busy_timeout(pDb->db, ms); 25706d31316cSdrh break; 257175897234Sdrh } 2572b5a20d3cSdrh 25730f14e2ebSdrh /* 25740f14e2ebSdrh ** $db total_changes 25750f14e2ebSdrh ** 25760f14e2ebSdrh ** Return the number of rows that were modified, inserted, or deleted 25770f14e2ebSdrh ** since the database handle was created. 25780f14e2ebSdrh */ 25790f14e2ebSdrh case DB_TOTAL_CHANGES: { 25800f14e2ebSdrh Tcl_Obj *pResult; 25810f14e2ebSdrh if( objc!=2 ){ 25820f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 25830f14e2ebSdrh return TCL_ERROR; 25840f14e2ebSdrh } 25850f14e2ebSdrh pResult = Tcl_GetObjResult(interp); 25860f14e2ebSdrh Tcl_SetIntObj(pResult, sqlite3_total_changes(pDb->db)); 25870f14e2ebSdrh break; 25880f14e2ebSdrh } 25890f14e2ebSdrh 2590b5a20d3cSdrh /* $db trace ?CALLBACK? 2591b5a20d3cSdrh ** 2592b5a20d3cSdrh ** Make arrangements to invoke the CALLBACK routine for each SQL statement 2593b5a20d3cSdrh ** that is executed. The text of the SQL is appended to CALLBACK before 2594b5a20d3cSdrh ** it is executed. 2595b5a20d3cSdrh */ 2596b5a20d3cSdrh case DB_TRACE: { 2597b5a20d3cSdrh if( objc>3 ){ 2598b5a20d3cSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 2599b97759edSdrh return TCL_ERROR; 2600b5a20d3cSdrh }else if( objc==2 ){ 2601b5a20d3cSdrh if( pDb->zTrace ){ 2602b5a20d3cSdrh Tcl_AppendResult(interp, pDb->zTrace, 0); 2603b5a20d3cSdrh } 2604b5a20d3cSdrh }else{ 2605b5a20d3cSdrh char *zTrace; 2606b5a20d3cSdrh int len; 2607b5a20d3cSdrh if( pDb->zTrace ){ 2608b5a20d3cSdrh Tcl_Free(pDb->zTrace); 2609b5a20d3cSdrh } 2610b5a20d3cSdrh zTrace = Tcl_GetStringFromObj(objv[2], &len); 2611b5a20d3cSdrh if( zTrace && len>0 ){ 2612b5a20d3cSdrh pDb->zTrace = Tcl_Alloc( len + 1 ); 26135bb3eb9bSdrh memcpy(pDb->zTrace, zTrace, len+1); 2614b5a20d3cSdrh }else{ 2615b5a20d3cSdrh pDb->zTrace = 0; 2616b5a20d3cSdrh } 261719e2d37fSdrh #ifndef SQLITE_OMIT_TRACE 2618b5a20d3cSdrh if( pDb->zTrace ){ 2619b5a20d3cSdrh pDb->interp = interp; 26206f8a503dSdanielk1977 sqlite3_trace(pDb->db, DbTraceHandler, pDb); 2621b5a20d3cSdrh }else{ 26226f8a503dSdanielk1977 sqlite3_trace(pDb->db, 0, 0); 2623b5a20d3cSdrh } 262419e2d37fSdrh #endif 2625b5a20d3cSdrh } 2626b5a20d3cSdrh break; 2627b5a20d3cSdrh } 2628b5a20d3cSdrh 26293d21423cSdrh /* $db transaction [-deferred|-immediate|-exclusive] SCRIPT 26303d21423cSdrh ** 26313d21423cSdrh ** Start a new transaction (if we are not already in the midst of a 26323d21423cSdrh ** transaction) and execute the TCL script SCRIPT. After SCRIPT 26333d21423cSdrh ** completes, either commit the transaction or roll it back if SCRIPT 26343d21423cSdrh ** throws an exception. Or if no new transation was started, do nothing. 26353d21423cSdrh ** pass the exception on up the stack. 26363d21423cSdrh ** 26373d21423cSdrh ** This command was inspired by Dave Thomas's talk on Ruby at the 26383d21423cSdrh ** 2005 O'Reilly Open Source Convention (OSCON). 26393d21423cSdrh */ 26403d21423cSdrh case DB_TRANSACTION: { 26413d21423cSdrh Tcl_Obj *pScript; 2642cd38d520Sdanielk1977 const char *zBegin = "SAVEPOINT _tcl_transaction"; 26433d21423cSdrh if( objc!=3 && objc!=4 ){ 26443d21423cSdrh Tcl_WrongNumArgs(interp, 2, objv, "[TYPE] SCRIPT"); 26453d21423cSdrh return TCL_ERROR; 26463d21423cSdrh } 2647cd38d520Sdanielk1977 26484a4c11aaSdan if( pDb->nTransaction==0 && objc==4 ){ 26493d21423cSdrh static const char *TTYPE_strs[] = { 2650ce604012Sdrh "deferred", "exclusive", "immediate", 0 26513d21423cSdrh }; 26523d21423cSdrh enum TTYPE_enum { 26533d21423cSdrh TTYPE_DEFERRED, TTYPE_EXCLUSIVE, TTYPE_IMMEDIATE 26543d21423cSdrh }; 26553d21423cSdrh int ttype; 2656b5555e7eSdrh if( Tcl_GetIndexFromObj(interp, objv[2], TTYPE_strs, "transaction type", 26573d21423cSdrh 0, &ttype) ){ 26583d21423cSdrh return TCL_ERROR; 26593d21423cSdrh } 26603d21423cSdrh switch( (enum TTYPE_enum)ttype ){ 26613d21423cSdrh case TTYPE_DEFERRED: /* no-op */; break; 26623d21423cSdrh case TTYPE_EXCLUSIVE: zBegin = "BEGIN EXCLUSIVE"; break; 26633d21423cSdrh case TTYPE_IMMEDIATE: zBegin = "BEGIN IMMEDIATE"; break; 26643d21423cSdrh } 26653d21423cSdrh } 2666cd38d520Sdanielk1977 pScript = objv[objc-1]; 2667cd38d520Sdanielk1977 26684a4c11aaSdan /* Run the SQLite BEGIN command to open a transaction or savepoint. */ 26691f1549f8Sdrh pDb->disableAuth++; 2670cd38d520Sdanielk1977 rc = sqlite3_exec(pDb->db, zBegin, 0, 0, 0); 26711f1549f8Sdrh pDb->disableAuth--; 2672cd38d520Sdanielk1977 if( rc!=SQLITE_OK ){ 2673cd38d520Sdanielk1977 Tcl_AppendResult(interp, sqlite3_errmsg(pDb->db), 0); 2674cd38d520Sdanielk1977 return TCL_ERROR; 26753d21423cSdrh } 2676cd38d520Sdanielk1977 pDb->nTransaction++; 2677cd38d520Sdanielk1977 26784a4c11aaSdan /* If using NRE, schedule a callback to invoke the script pScript, then 26794a4c11aaSdan ** a second callback to commit (or rollback) the transaction or savepoint 26804a4c11aaSdan ** opened above. If not using NRE, evaluate the script directly, then 26814a4c11aaSdan ** call function DbTransPostCmd() to commit (or rollback) the transaction 26824a4c11aaSdan ** or savepoint. */ 26834a4c11aaSdan if( DbUseNre() ){ 26844a4c11aaSdan Tcl_NRAddCallback(interp, DbTransPostCmd, cd, 0, 0, 0); 26854a4c11aaSdan Tcl_NREvalObj(interp, pScript, 0); 26863d21423cSdrh }else{ 26874a4c11aaSdan rc = DbTransPostCmd(&cd, interp, Tcl_EvalObjEx(interp, pScript, 0)); 26883d21423cSdrh } 26893d21423cSdrh break; 26903d21423cSdrh } 26913d21423cSdrh 269294eb6a14Sdanielk1977 /* 2693404ca075Sdanielk1977 ** $db unlock_notify ?script? 2694404ca075Sdanielk1977 */ 2695404ca075Sdanielk1977 case DB_UNLOCK_NOTIFY: { 2696404ca075Sdanielk1977 #ifndef SQLITE_ENABLE_UNLOCK_NOTIFY 2697404ca075Sdanielk1977 Tcl_AppendResult(interp, "unlock_notify not available in this build", 0); 2698404ca075Sdanielk1977 rc = TCL_ERROR; 2699404ca075Sdanielk1977 #else 2700404ca075Sdanielk1977 if( objc!=2 && objc!=3 ){ 2701404ca075Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?SCRIPT?"); 2702404ca075Sdanielk1977 rc = TCL_ERROR; 2703404ca075Sdanielk1977 }else{ 2704404ca075Sdanielk1977 void (*xNotify)(void **, int) = 0; 2705404ca075Sdanielk1977 void *pNotifyArg = 0; 2706404ca075Sdanielk1977 2707404ca075Sdanielk1977 if( pDb->pUnlockNotify ){ 2708404ca075Sdanielk1977 Tcl_DecrRefCount(pDb->pUnlockNotify); 2709404ca075Sdanielk1977 pDb->pUnlockNotify = 0; 2710404ca075Sdanielk1977 } 2711404ca075Sdanielk1977 2712404ca075Sdanielk1977 if( objc==3 ){ 2713404ca075Sdanielk1977 xNotify = DbUnlockNotify; 2714404ca075Sdanielk1977 pNotifyArg = (void *)pDb; 2715404ca075Sdanielk1977 pDb->pUnlockNotify = objv[2]; 2716404ca075Sdanielk1977 Tcl_IncrRefCount(pDb->pUnlockNotify); 2717404ca075Sdanielk1977 } 2718404ca075Sdanielk1977 2719404ca075Sdanielk1977 if( sqlite3_unlock_notify(pDb->db, xNotify, pNotifyArg) ){ 2720404ca075Sdanielk1977 Tcl_AppendResult(interp, sqlite3_errmsg(pDb->db), 0); 2721404ca075Sdanielk1977 rc = TCL_ERROR; 2722404ca075Sdanielk1977 } 2723404ca075Sdanielk1977 } 2724404ca075Sdanielk1977 #endif 2725404ca075Sdanielk1977 break; 2726404ca075Sdanielk1977 } 2727404ca075Sdanielk1977 2728404ca075Sdanielk1977 /* 272994eb6a14Sdanielk1977 ** $db update_hook ?script? 273071fd80bfSdanielk1977 ** $db rollback_hook ?script? 273194eb6a14Sdanielk1977 */ 273271fd80bfSdanielk1977 case DB_UPDATE_HOOK: 273371fd80bfSdanielk1977 case DB_ROLLBACK_HOOK: { 273471fd80bfSdanielk1977 273571fd80bfSdanielk1977 /* set ppHook to point at pUpdateHook or pRollbackHook, depending on 273671fd80bfSdanielk1977 ** whether [$db update_hook] or [$db rollback_hook] was invoked. 273771fd80bfSdanielk1977 */ 273871fd80bfSdanielk1977 Tcl_Obj **ppHook; 273971fd80bfSdanielk1977 if( choice==DB_UPDATE_HOOK ){ 274071fd80bfSdanielk1977 ppHook = &pDb->pUpdateHook; 274171fd80bfSdanielk1977 }else{ 274271fd80bfSdanielk1977 ppHook = &pDb->pRollbackHook; 274371fd80bfSdanielk1977 } 274471fd80bfSdanielk1977 274594eb6a14Sdanielk1977 if( objc!=2 && objc!=3 ){ 274694eb6a14Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?SCRIPT?"); 274794eb6a14Sdanielk1977 return TCL_ERROR; 274894eb6a14Sdanielk1977 } 274971fd80bfSdanielk1977 if( *ppHook ){ 275071fd80bfSdanielk1977 Tcl_SetObjResult(interp, *ppHook); 275194eb6a14Sdanielk1977 if( objc==3 ){ 275271fd80bfSdanielk1977 Tcl_DecrRefCount(*ppHook); 275371fd80bfSdanielk1977 *ppHook = 0; 275494eb6a14Sdanielk1977 } 275594eb6a14Sdanielk1977 } 275694eb6a14Sdanielk1977 if( objc==3 ){ 275771fd80bfSdanielk1977 assert( !(*ppHook) ); 275894eb6a14Sdanielk1977 if( Tcl_GetCharLength(objv[2])>0 ){ 275971fd80bfSdanielk1977 *ppHook = objv[2]; 276071fd80bfSdanielk1977 Tcl_IncrRefCount(*ppHook); 276194eb6a14Sdanielk1977 } 276294eb6a14Sdanielk1977 } 276371fd80bfSdanielk1977 276471fd80bfSdanielk1977 sqlite3_update_hook(pDb->db, (pDb->pUpdateHook?DbUpdateHandler:0), pDb); 276571fd80bfSdanielk1977 sqlite3_rollback_hook(pDb->db,(pDb->pRollbackHook?DbRollbackHandler:0),pDb); 276671fd80bfSdanielk1977 276794eb6a14Sdanielk1977 break; 276894eb6a14Sdanielk1977 } 276994eb6a14Sdanielk1977 27704397de57Sdanielk1977 /* $db version 27714397de57Sdanielk1977 ** 27724397de57Sdanielk1977 ** Return the version string for this database. 27734397de57Sdanielk1977 */ 27744397de57Sdanielk1977 case DB_VERSION: { 27754397de57Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_libversion(), TCL_STATIC); 27764397de57Sdanielk1977 break; 27774397de57Sdanielk1977 } 27784397de57Sdanielk1977 27791067fe11Stpoindex 27806d31316cSdrh } /* End of the SWITCH statement */ 278122fbcb8dSdrh return rc; 278275897234Sdrh } 278375897234Sdrh 2784a2c8a95bSdrh #if SQLITE_TCL_NRE 2785a2c8a95bSdrh /* 2786a2c8a95bSdrh ** Adaptor that provides an objCmd interface to the NRE-enabled 2787a2c8a95bSdrh ** interface implementation. 2788a2c8a95bSdrh */ 2789a2c8a95bSdrh static int DbObjCmdAdaptor( 2790a2c8a95bSdrh void *cd, 2791a2c8a95bSdrh Tcl_Interp *interp, 2792a2c8a95bSdrh int objc, 2793a2c8a95bSdrh Tcl_Obj *const*objv 2794a2c8a95bSdrh ){ 2795a2c8a95bSdrh return Tcl_NRCallObjProc(interp, DbObjCmd, cd, objc, objv); 2796a2c8a95bSdrh } 2797a2c8a95bSdrh #endif /* SQLITE_TCL_NRE */ 2798a2c8a95bSdrh 279975897234Sdrh /* 28003570ad93Sdrh ** sqlite3 DBNAME FILENAME ?-vfs VFSNAME? ?-key KEY? ?-readonly BOOLEAN? 28019a6284c1Sdanielk1977 ** ?-create BOOLEAN? ?-nomutex BOOLEAN? 280275897234Sdrh ** 280375897234Sdrh ** This is the main Tcl command. When the "sqlite" Tcl command is 280475897234Sdrh ** invoked, this routine runs to process that command. 280575897234Sdrh ** 280675897234Sdrh ** The first argument, DBNAME, is an arbitrary name for a new 280775897234Sdrh ** database connection. This command creates a new command named 280875897234Sdrh ** DBNAME that is used to control that connection. The database 280975897234Sdrh ** connection is deleted when the DBNAME command is deleted. 281075897234Sdrh ** 28113570ad93Sdrh ** The second argument is the name of the database file. 2812fbc3eab8Sdrh ** 281375897234Sdrh */ 281422fbcb8dSdrh static int DbMain(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){ 2815bec3f402Sdrh SqliteDb *p; 281622fbcb8dSdrh void *pKey = 0; 281722fbcb8dSdrh int nKey = 0; 281822fbcb8dSdrh const char *zArg; 281975897234Sdrh char *zErrMsg; 28203570ad93Sdrh int i; 282122fbcb8dSdrh const char *zFile; 28223570ad93Sdrh const char *zVfs = 0; 2823d9da78a2Sdrh int flags; 2824882e8e4dSdrh Tcl_DString translatedFilename; 2825d9da78a2Sdrh 2826d9da78a2Sdrh /* In normal use, each TCL interpreter runs in a single thread. So 2827d9da78a2Sdrh ** by default, we can turn of mutexing on SQLite database connections. 2828d9da78a2Sdrh ** However, for testing purposes it is useful to have mutexes turned 2829d9da78a2Sdrh ** on. So, by default, mutexes default off. But if compiled with 2830d9da78a2Sdrh ** SQLITE_TCL_DEFAULT_FULLMUTEX then mutexes default on. 2831d9da78a2Sdrh */ 2832d9da78a2Sdrh #ifdef SQLITE_TCL_DEFAULT_FULLMUTEX 2833d9da78a2Sdrh flags = SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE | SQLITE_OPEN_FULLMUTEX; 2834d9da78a2Sdrh #else 2835d9da78a2Sdrh flags = SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE | SQLITE_OPEN_NOMUTEX; 2836d9da78a2Sdrh #endif 2837d9da78a2Sdrh 283822fbcb8dSdrh if( objc==2 ){ 283922fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[1], 0); 284022fbcb8dSdrh if( strcmp(zArg,"-version")==0 ){ 28416f8a503dSdanielk1977 Tcl_AppendResult(interp,sqlite3_version,0); 2842647cb0e1Sdrh return TCL_OK; 2843647cb0e1Sdrh } 28449eb9e26bSdrh if( strcmp(zArg,"-has-codec")==0 ){ 28459eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 284622fbcb8dSdrh Tcl_AppendResult(interp,"1",0); 284722fbcb8dSdrh #else 284822fbcb8dSdrh Tcl_AppendResult(interp,"0",0); 284922fbcb8dSdrh #endif 285022fbcb8dSdrh return TCL_OK; 285122fbcb8dSdrh } 2852fbc3eab8Sdrh } 28533570ad93Sdrh for(i=3; i+1<objc; i+=2){ 28543570ad93Sdrh zArg = Tcl_GetString(objv[i]); 285522fbcb8dSdrh if( strcmp(zArg,"-key")==0 ){ 28563570ad93Sdrh pKey = Tcl_GetByteArrayFromObj(objv[i+1], &nKey); 28573570ad93Sdrh }else if( strcmp(zArg, "-vfs")==0 ){ 28583570ad93Sdrh i++; 28593570ad93Sdrh zVfs = Tcl_GetString(objv[i]); 28603570ad93Sdrh }else if( strcmp(zArg, "-readonly")==0 ){ 28613570ad93Sdrh int b; 28623570ad93Sdrh if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 28633570ad93Sdrh if( b ){ 286433f4e02aSdrh flags &= ~(SQLITE_OPEN_READWRITE|SQLITE_OPEN_CREATE); 28653570ad93Sdrh flags |= SQLITE_OPEN_READONLY; 28663570ad93Sdrh }else{ 28673570ad93Sdrh flags &= ~SQLITE_OPEN_READONLY; 28683570ad93Sdrh flags |= SQLITE_OPEN_READWRITE; 28693570ad93Sdrh } 28703570ad93Sdrh }else if( strcmp(zArg, "-create")==0 ){ 28713570ad93Sdrh int b; 28723570ad93Sdrh if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 287333f4e02aSdrh if( b && (flags & SQLITE_OPEN_READONLY)==0 ){ 28743570ad93Sdrh flags |= SQLITE_OPEN_CREATE; 28753570ad93Sdrh }else{ 28763570ad93Sdrh flags &= ~SQLITE_OPEN_CREATE; 28773570ad93Sdrh } 28789a6284c1Sdanielk1977 }else if( strcmp(zArg, "-nomutex")==0 ){ 28799a6284c1Sdanielk1977 int b; 28809a6284c1Sdanielk1977 if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 28819a6284c1Sdanielk1977 if( b ){ 28829a6284c1Sdanielk1977 flags |= SQLITE_OPEN_NOMUTEX; 2883039963adSdrh flags &= ~SQLITE_OPEN_FULLMUTEX; 28849a6284c1Sdanielk1977 }else{ 28859a6284c1Sdanielk1977 flags &= ~SQLITE_OPEN_NOMUTEX; 28869a6284c1Sdanielk1977 } 2887039963adSdrh }else if( strcmp(zArg, "-fullmutex")==0 ){ 2888039963adSdrh int b; 2889039963adSdrh if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 2890039963adSdrh if( b ){ 2891039963adSdrh flags |= SQLITE_OPEN_FULLMUTEX; 2892039963adSdrh flags &= ~SQLITE_OPEN_NOMUTEX; 2893039963adSdrh }else{ 2894039963adSdrh flags &= ~SQLITE_OPEN_FULLMUTEX; 2895039963adSdrh } 28963570ad93Sdrh }else{ 28973570ad93Sdrh Tcl_AppendResult(interp, "unknown option: ", zArg, (char*)0); 28983570ad93Sdrh return TCL_ERROR; 289922fbcb8dSdrh } 290022fbcb8dSdrh } 29013570ad93Sdrh if( objc<3 || (objc&1)!=1 ){ 290222fbcb8dSdrh Tcl_WrongNumArgs(interp, 1, objv, 29033570ad93Sdrh "HANDLE FILENAME ?-vfs VFSNAME? ?-readonly BOOLEAN? ?-create BOOLEAN?" 2904039963adSdrh " ?-nomutex BOOLEAN? ?-fullmutex BOOLEAN?" 29059eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 29063570ad93Sdrh " ?-key CODECKEY?" 290722fbcb8dSdrh #endif 290822fbcb8dSdrh ); 290975897234Sdrh return TCL_ERROR; 291075897234Sdrh } 291175897234Sdrh zErrMsg = 0; 29124cdc9e84Sdrh p = (SqliteDb*)Tcl_Alloc( sizeof(*p) ); 291375897234Sdrh if( p==0 ){ 2914bec3f402Sdrh Tcl_SetResult(interp, "malloc failed", TCL_STATIC); 2915bec3f402Sdrh return TCL_ERROR; 2916bec3f402Sdrh } 2917bec3f402Sdrh memset(p, 0, sizeof(*p)); 291822fbcb8dSdrh zFile = Tcl_GetStringFromObj(objv[2], 0); 2919882e8e4dSdrh zFile = Tcl_TranslateFileName(interp, zFile, &translatedFilename); 29203570ad93Sdrh sqlite3_open_v2(zFile, &p->db, flags, zVfs); 2921882e8e4dSdrh Tcl_DStringFree(&translatedFilename); 292280290863Sdanielk1977 if( SQLITE_OK!=sqlite3_errcode(p->db) ){ 29239404d50eSdrh zErrMsg = sqlite3_mprintf("%s", sqlite3_errmsg(p->db)); 292480290863Sdanielk1977 sqlite3_close(p->db); 292580290863Sdanielk1977 p->db = 0; 292680290863Sdanielk1977 } 29272011d5f5Sdrh #ifdef SQLITE_HAS_CODEC 2928f3a65f7eSdrh if( p->db ){ 29292011d5f5Sdrh sqlite3_key(p->db, pKey, nKey); 2930f3a65f7eSdrh } 2931eb8ed70dSdrh #endif 2932bec3f402Sdrh if( p->db==0 ){ 293375897234Sdrh Tcl_SetResult(interp, zErrMsg, TCL_VOLATILE); 2934bec3f402Sdrh Tcl_Free((char*)p); 29359404d50eSdrh sqlite3_free(zErrMsg); 293675897234Sdrh return TCL_ERROR; 293775897234Sdrh } 2938fb7e7651Sdrh p->maxStmt = NUM_PREPARED_STMTS; 29395169bbc6Sdrh p->interp = interp; 294022fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[1], 0); 29414a4c11aaSdan if( DbUseNre() ){ 2942a2c8a95bSdrh Tcl_NRCreateCommand(interp, zArg, DbObjCmdAdaptor, DbObjCmd, 2943a2c8a95bSdrh (char*)p, DbDeleteCmd); 29444a4c11aaSdan }else{ 294522fbcb8dSdrh Tcl_CreateObjCommand(interp, zArg, DbObjCmd, (char*)p, DbDeleteCmd); 29464a4c11aaSdan } 294775897234Sdrh return TCL_OK; 294875897234Sdrh } 294975897234Sdrh 295075897234Sdrh /* 295190ca9753Sdrh ** Provide a dummy Tcl_InitStubs if we are using this as a static 295290ca9753Sdrh ** library. 295390ca9753Sdrh */ 295490ca9753Sdrh #ifndef USE_TCL_STUBS 295590ca9753Sdrh # undef Tcl_InitStubs 295690ca9753Sdrh # define Tcl_InitStubs(a,b,c) 295790ca9753Sdrh #endif 295890ca9753Sdrh 295990ca9753Sdrh /* 296029bc4615Sdrh ** Make sure we have a PACKAGE_VERSION macro defined. This will be 296129bc4615Sdrh ** defined automatically by the TEA makefile. But other makefiles 296229bc4615Sdrh ** do not define it. 296329bc4615Sdrh */ 296429bc4615Sdrh #ifndef PACKAGE_VERSION 296529bc4615Sdrh # define PACKAGE_VERSION SQLITE_VERSION 296629bc4615Sdrh #endif 296729bc4615Sdrh 296829bc4615Sdrh /* 296975897234Sdrh ** Initialize this module. 297075897234Sdrh ** 297175897234Sdrh ** This Tcl module contains only a single new Tcl command named "sqlite". 297275897234Sdrh ** (Hence there is no namespace. There is no point in using a namespace 297375897234Sdrh ** if the extension only supplies one new name!) The "sqlite" command is 297475897234Sdrh ** used to open a new SQLite database. See the DbMain() routine above 297575897234Sdrh ** for additional information. 297675897234Sdrh */ 2977ad6e1370Sdrh EXTERN int Sqlite3_Init(Tcl_Interp *interp){ 297892febd92Sdrh Tcl_InitStubs(interp, "8.4", 0); 2979ef4ac8f9Sdrh Tcl_CreateObjCommand(interp, "sqlite3", (Tcl_ObjCmdProc*)DbMain, 0, 0); 298029bc4615Sdrh Tcl_PkgProvide(interp, "sqlite3", PACKAGE_VERSION); 298149766d6cSdrh Tcl_CreateObjCommand(interp, "sqlite", (Tcl_ObjCmdProc*)DbMain, 0, 0); 298229bc4615Sdrh Tcl_PkgProvide(interp, "sqlite", PACKAGE_VERSION); 298390ca9753Sdrh return TCL_OK; 298490ca9753Sdrh } 2985ad6e1370Sdrh EXTERN int Tclsqlite3_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2986ad6e1370Sdrh EXTERN int Sqlite3_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2987ad6e1370Sdrh EXTERN int Tclsqlite3_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2988e2c3a659Sdrh EXTERN int Sqlite3_Unload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2989e2c3a659Sdrh EXTERN int Tclsqlite3_Unload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2990e2c3a659Sdrh EXTERN int Sqlite3_SafeUnload(Tcl_Interp *interp, int flags){ return TCL_OK; } 2991e2c3a659Sdrh EXTERN int Tclsqlite3_SafeUnload(Tcl_Interp *interp, int flags){ return TCL_OK;} 2992e2c3a659Sdrh 299349766d6cSdrh 299449766d6cSdrh #ifndef SQLITE_3_SUFFIX_ONLY 2995ad6e1370Sdrh EXTERN int Sqlite_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2996ad6e1370Sdrh EXTERN int Tclsqlite_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2997ad6e1370Sdrh EXTERN int Sqlite_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2998ad6e1370Sdrh EXTERN int Tclsqlite_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2999e2c3a659Sdrh EXTERN int Sqlite_Unload(Tcl_Interp *interp, int flags){ return TCL_OK; } 3000e2c3a659Sdrh EXTERN int Tclsqlite_Unload(Tcl_Interp *interp, int flags){ return TCL_OK; } 3001e2c3a659Sdrh EXTERN int Sqlite_SafeUnload(Tcl_Interp *interp, int flags){ return TCL_OK; } 3002e2c3a659Sdrh EXTERN int Tclsqlite_SafeUnload(Tcl_Interp *interp, int flags){ return TCL_OK;} 300349766d6cSdrh #endif 300475897234Sdrh 30053e27c026Sdrh #ifdef TCLSH 30063e27c026Sdrh /***************************************************************************** 300757a0227fSdrh ** All of the code that follows is used to build standalone TCL interpreters 300857a0227fSdrh ** that are statically linked with SQLite. Enable these by compiling 300957a0227fSdrh ** with -DTCLSH=n where n can be 1 or 2. An n of 1 generates a standard 301057a0227fSdrh ** tclsh but with SQLite built in. An n of 2 generates the SQLite space 301157a0227fSdrh ** analysis program. 301275897234Sdrh */ 3013348784efSdrh 301457a0227fSdrh #if defined(SQLITE_TEST) || defined(SQLITE_TCLMD5) 301557a0227fSdrh /* 301657a0227fSdrh * This code implements the MD5 message-digest algorithm. 301757a0227fSdrh * The algorithm is due to Ron Rivest. This code was 301857a0227fSdrh * written by Colin Plumb in 1993, no copyright is claimed. 301957a0227fSdrh * This code is in the public domain; do with it what you wish. 302057a0227fSdrh * 302157a0227fSdrh * Equivalent code is available from RSA Data Security, Inc. 302257a0227fSdrh * This code has been tested against that, and is equivalent, 302357a0227fSdrh * except that you don't need to include two pages of legalese 302457a0227fSdrh * with every copy. 302557a0227fSdrh * 302657a0227fSdrh * To compute the message digest of a chunk of bytes, declare an 302757a0227fSdrh * MD5Context structure, pass it to MD5Init, call MD5Update as 302857a0227fSdrh * needed on buffers full of bytes, and then call MD5Final, which 302957a0227fSdrh * will fill a supplied 16-byte array with the digest. 303057a0227fSdrh */ 303157a0227fSdrh 303257a0227fSdrh /* 303357a0227fSdrh * If compiled on a machine that doesn't have a 32-bit integer, 303457a0227fSdrh * you just set "uint32" to the appropriate datatype for an 303557a0227fSdrh * unsigned 32-bit integer. For example: 303657a0227fSdrh * 303757a0227fSdrh * cc -Duint32='unsigned long' md5.c 303857a0227fSdrh * 303957a0227fSdrh */ 304057a0227fSdrh #ifndef uint32 304157a0227fSdrh # define uint32 unsigned int 304257a0227fSdrh #endif 304357a0227fSdrh 304457a0227fSdrh struct MD5Context { 304557a0227fSdrh int isInit; 304657a0227fSdrh uint32 buf[4]; 304757a0227fSdrh uint32 bits[2]; 304857a0227fSdrh unsigned char in[64]; 304957a0227fSdrh }; 305057a0227fSdrh typedef struct MD5Context MD5Context; 305157a0227fSdrh 305257a0227fSdrh /* 305357a0227fSdrh * Note: this code is harmless on little-endian machines. 305457a0227fSdrh */ 305557a0227fSdrh static void byteReverse (unsigned char *buf, unsigned longs){ 305657a0227fSdrh uint32 t; 305757a0227fSdrh do { 305857a0227fSdrh t = (uint32)((unsigned)buf[3]<<8 | buf[2]) << 16 | 305957a0227fSdrh ((unsigned)buf[1]<<8 | buf[0]); 306057a0227fSdrh *(uint32 *)buf = t; 306157a0227fSdrh buf += 4; 306257a0227fSdrh } while (--longs); 306357a0227fSdrh } 306457a0227fSdrh /* The four core functions - F1 is optimized somewhat */ 306557a0227fSdrh 306657a0227fSdrh /* #define F1(x, y, z) (x & y | ~x & z) */ 306757a0227fSdrh #define F1(x, y, z) (z ^ (x & (y ^ z))) 306857a0227fSdrh #define F2(x, y, z) F1(z, x, y) 306957a0227fSdrh #define F3(x, y, z) (x ^ y ^ z) 307057a0227fSdrh #define F4(x, y, z) (y ^ (x | ~z)) 307157a0227fSdrh 307257a0227fSdrh /* This is the central step in the MD5 algorithm. */ 307357a0227fSdrh #define MD5STEP(f, w, x, y, z, data, s) \ 307457a0227fSdrh ( w += f(x, y, z) + data, w = w<<s | w>>(32-s), w += x ) 307557a0227fSdrh 307657a0227fSdrh /* 307757a0227fSdrh * The core of the MD5 algorithm, this alters an existing MD5 hash to 307857a0227fSdrh * reflect the addition of 16 longwords of new data. MD5Update blocks 307957a0227fSdrh * the data and converts bytes into longwords for this routine. 308057a0227fSdrh */ 308157a0227fSdrh static void MD5Transform(uint32 buf[4], const uint32 in[16]){ 308257a0227fSdrh register uint32 a, b, c, d; 308357a0227fSdrh 308457a0227fSdrh a = buf[0]; 308557a0227fSdrh b = buf[1]; 308657a0227fSdrh c = buf[2]; 308757a0227fSdrh d = buf[3]; 308857a0227fSdrh 308957a0227fSdrh MD5STEP(F1, a, b, c, d, in[ 0]+0xd76aa478, 7); 309057a0227fSdrh MD5STEP(F1, d, a, b, c, in[ 1]+0xe8c7b756, 12); 309157a0227fSdrh MD5STEP(F1, c, d, a, b, in[ 2]+0x242070db, 17); 309257a0227fSdrh MD5STEP(F1, b, c, d, a, in[ 3]+0xc1bdceee, 22); 309357a0227fSdrh MD5STEP(F1, a, b, c, d, in[ 4]+0xf57c0faf, 7); 309457a0227fSdrh MD5STEP(F1, d, a, b, c, in[ 5]+0x4787c62a, 12); 309557a0227fSdrh MD5STEP(F1, c, d, a, b, in[ 6]+0xa8304613, 17); 309657a0227fSdrh MD5STEP(F1, b, c, d, a, in[ 7]+0xfd469501, 22); 309757a0227fSdrh MD5STEP(F1, a, b, c, d, in[ 8]+0x698098d8, 7); 309857a0227fSdrh MD5STEP(F1, d, a, b, c, in[ 9]+0x8b44f7af, 12); 309957a0227fSdrh MD5STEP(F1, c, d, a, b, in[10]+0xffff5bb1, 17); 310057a0227fSdrh MD5STEP(F1, b, c, d, a, in[11]+0x895cd7be, 22); 310157a0227fSdrh MD5STEP(F1, a, b, c, d, in[12]+0x6b901122, 7); 310257a0227fSdrh MD5STEP(F1, d, a, b, c, in[13]+0xfd987193, 12); 310357a0227fSdrh MD5STEP(F1, c, d, a, b, in[14]+0xa679438e, 17); 310457a0227fSdrh MD5STEP(F1, b, c, d, a, in[15]+0x49b40821, 22); 310557a0227fSdrh 310657a0227fSdrh MD5STEP(F2, a, b, c, d, in[ 1]+0xf61e2562, 5); 310757a0227fSdrh MD5STEP(F2, d, a, b, c, in[ 6]+0xc040b340, 9); 310857a0227fSdrh MD5STEP(F2, c, d, a, b, in[11]+0x265e5a51, 14); 310957a0227fSdrh MD5STEP(F2, b, c, d, a, in[ 0]+0xe9b6c7aa, 20); 311057a0227fSdrh MD5STEP(F2, a, b, c, d, in[ 5]+0xd62f105d, 5); 311157a0227fSdrh MD5STEP(F2, d, a, b, c, in[10]+0x02441453, 9); 311257a0227fSdrh MD5STEP(F2, c, d, a, b, in[15]+0xd8a1e681, 14); 311357a0227fSdrh MD5STEP(F2, b, c, d, a, in[ 4]+0xe7d3fbc8, 20); 311457a0227fSdrh MD5STEP(F2, a, b, c, d, in[ 9]+0x21e1cde6, 5); 311557a0227fSdrh MD5STEP(F2, d, a, b, c, in[14]+0xc33707d6, 9); 311657a0227fSdrh MD5STEP(F2, c, d, a, b, in[ 3]+0xf4d50d87, 14); 311757a0227fSdrh MD5STEP(F2, b, c, d, a, in[ 8]+0x455a14ed, 20); 311857a0227fSdrh MD5STEP(F2, a, b, c, d, in[13]+0xa9e3e905, 5); 311957a0227fSdrh MD5STEP(F2, d, a, b, c, in[ 2]+0xfcefa3f8, 9); 312057a0227fSdrh MD5STEP(F2, c, d, a, b, in[ 7]+0x676f02d9, 14); 312157a0227fSdrh MD5STEP(F2, b, c, d, a, in[12]+0x8d2a4c8a, 20); 312257a0227fSdrh 312357a0227fSdrh MD5STEP(F3, a, b, c, d, in[ 5]+0xfffa3942, 4); 312457a0227fSdrh MD5STEP(F3, d, a, b, c, in[ 8]+0x8771f681, 11); 312557a0227fSdrh MD5STEP(F3, c, d, a, b, in[11]+0x6d9d6122, 16); 312657a0227fSdrh MD5STEP(F3, b, c, d, a, in[14]+0xfde5380c, 23); 312757a0227fSdrh MD5STEP(F3, a, b, c, d, in[ 1]+0xa4beea44, 4); 312857a0227fSdrh MD5STEP(F3, d, a, b, c, in[ 4]+0x4bdecfa9, 11); 312957a0227fSdrh MD5STEP(F3, c, d, a, b, in[ 7]+0xf6bb4b60, 16); 313057a0227fSdrh MD5STEP(F3, b, c, d, a, in[10]+0xbebfbc70, 23); 313157a0227fSdrh MD5STEP(F3, a, b, c, d, in[13]+0x289b7ec6, 4); 313257a0227fSdrh MD5STEP(F3, d, a, b, c, in[ 0]+0xeaa127fa, 11); 313357a0227fSdrh MD5STEP(F3, c, d, a, b, in[ 3]+0xd4ef3085, 16); 313457a0227fSdrh MD5STEP(F3, b, c, d, a, in[ 6]+0x04881d05, 23); 313557a0227fSdrh MD5STEP(F3, a, b, c, d, in[ 9]+0xd9d4d039, 4); 313657a0227fSdrh MD5STEP(F3, d, a, b, c, in[12]+0xe6db99e5, 11); 313757a0227fSdrh MD5STEP(F3, c, d, a, b, in[15]+0x1fa27cf8, 16); 313857a0227fSdrh MD5STEP(F3, b, c, d, a, in[ 2]+0xc4ac5665, 23); 313957a0227fSdrh 314057a0227fSdrh MD5STEP(F4, a, b, c, d, in[ 0]+0xf4292244, 6); 314157a0227fSdrh MD5STEP(F4, d, a, b, c, in[ 7]+0x432aff97, 10); 314257a0227fSdrh MD5STEP(F4, c, d, a, b, in[14]+0xab9423a7, 15); 314357a0227fSdrh MD5STEP(F4, b, c, d, a, in[ 5]+0xfc93a039, 21); 314457a0227fSdrh MD5STEP(F4, a, b, c, d, in[12]+0x655b59c3, 6); 314557a0227fSdrh MD5STEP(F4, d, a, b, c, in[ 3]+0x8f0ccc92, 10); 314657a0227fSdrh MD5STEP(F4, c, d, a, b, in[10]+0xffeff47d, 15); 314757a0227fSdrh MD5STEP(F4, b, c, d, a, in[ 1]+0x85845dd1, 21); 314857a0227fSdrh MD5STEP(F4, a, b, c, d, in[ 8]+0x6fa87e4f, 6); 314957a0227fSdrh MD5STEP(F4, d, a, b, c, in[15]+0xfe2ce6e0, 10); 315057a0227fSdrh MD5STEP(F4, c, d, a, b, in[ 6]+0xa3014314, 15); 315157a0227fSdrh MD5STEP(F4, b, c, d, a, in[13]+0x4e0811a1, 21); 315257a0227fSdrh MD5STEP(F4, a, b, c, d, in[ 4]+0xf7537e82, 6); 315357a0227fSdrh MD5STEP(F4, d, a, b, c, in[11]+0xbd3af235, 10); 315457a0227fSdrh MD5STEP(F4, c, d, a, b, in[ 2]+0x2ad7d2bb, 15); 315557a0227fSdrh MD5STEP(F4, b, c, d, a, in[ 9]+0xeb86d391, 21); 315657a0227fSdrh 315757a0227fSdrh buf[0] += a; 315857a0227fSdrh buf[1] += b; 315957a0227fSdrh buf[2] += c; 316057a0227fSdrh buf[3] += d; 316157a0227fSdrh } 316257a0227fSdrh 316357a0227fSdrh /* 316457a0227fSdrh * Start MD5 accumulation. Set bit count to 0 and buffer to mysterious 316557a0227fSdrh * initialization constants. 316657a0227fSdrh */ 316757a0227fSdrh static void MD5Init(MD5Context *ctx){ 316857a0227fSdrh ctx->isInit = 1; 316957a0227fSdrh ctx->buf[0] = 0x67452301; 317057a0227fSdrh ctx->buf[1] = 0xefcdab89; 317157a0227fSdrh ctx->buf[2] = 0x98badcfe; 317257a0227fSdrh ctx->buf[3] = 0x10325476; 317357a0227fSdrh ctx->bits[0] = 0; 317457a0227fSdrh ctx->bits[1] = 0; 317557a0227fSdrh } 317657a0227fSdrh 317757a0227fSdrh /* 317857a0227fSdrh * Update context to reflect the concatenation of another buffer full 317957a0227fSdrh * of bytes. 318057a0227fSdrh */ 318157a0227fSdrh static 318257a0227fSdrh void MD5Update(MD5Context *ctx, const unsigned char *buf, unsigned int len){ 318357a0227fSdrh uint32 t; 318457a0227fSdrh 318557a0227fSdrh /* Update bitcount */ 318657a0227fSdrh 318757a0227fSdrh t = ctx->bits[0]; 318857a0227fSdrh if ((ctx->bits[0] = t + ((uint32)len << 3)) < t) 318957a0227fSdrh ctx->bits[1]++; /* Carry from low to high */ 319057a0227fSdrh ctx->bits[1] += len >> 29; 319157a0227fSdrh 319257a0227fSdrh t = (t >> 3) & 0x3f; /* Bytes already in shsInfo->data */ 319357a0227fSdrh 319457a0227fSdrh /* Handle any leading odd-sized chunks */ 319557a0227fSdrh 319657a0227fSdrh if ( t ) { 319757a0227fSdrh unsigned char *p = (unsigned char *)ctx->in + t; 319857a0227fSdrh 319957a0227fSdrh t = 64-t; 320057a0227fSdrh if (len < t) { 320157a0227fSdrh memcpy(p, buf, len); 320257a0227fSdrh return; 320357a0227fSdrh } 320457a0227fSdrh memcpy(p, buf, t); 320557a0227fSdrh byteReverse(ctx->in, 16); 320657a0227fSdrh MD5Transform(ctx->buf, (uint32 *)ctx->in); 320757a0227fSdrh buf += t; 320857a0227fSdrh len -= t; 320957a0227fSdrh } 321057a0227fSdrh 321157a0227fSdrh /* Process data in 64-byte chunks */ 321257a0227fSdrh 321357a0227fSdrh while (len >= 64) { 321457a0227fSdrh memcpy(ctx->in, buf, 64); 321557a0227fSdrh byteReverse(ctx->in, 16); 321657a0227fSdrh MD5Transform(ctx->buf, (uint32 *)ctx->in); 321757a0227fSdrh buf += 64; 321857a0227fSdrh len -= 64; 321957a0227fSdrh } 322057a0227fSdrh 322157a0227fSdrh /* Handle any remaining bytes of data. */ 322257a0227fSdrh 322357a0227fSdrh memcpy(ctx->in, buf, len); 322457a0227fSdrh } 322557a0227fSdrh 322657a0227fSdrh /* 322757a0227fSdrh * Final wrapup - pad to 64-byte boundary with the bit pattern 322857a0227fSdrh * 1 0* (64-bit count of bits processed, MSB-first) 322957a0227fSdrh */ 323057a0227fSdrh static void MD5Final(unsigned char digest[16], MD5Context *ctx){ 323157a0227fSdrh unsigned count; 323257a0227fSdrh unsigned char *p; 323357a0227fSdrh 323457a0227fSdrh /* Compute number of bytes mod 64 */ 323557a0227fSdrh count = (ctx->bits[0] >> 3) & 0x3F; 323657a0227fSdrh 323757a0227fSdrh /* Set the first char of padding to 0x80. This is safe since there is 323857a0227fSdrh always at least one byte free */ 323957a0227fSdrh p = ctx->in + count; 324057a0227fSdrh *p++ = 0x80; 324157a0227fSdrh 324257a0227fSdrh /* Bytes of padding needed to make 64 bytes */ 324357a0227fSdrh count = 64 - 1 - count; 324457a0227fSdrh 324557a0227fSdrh /* Pad out to 56 mod 64 */ 324657a0227fSdrh if (count < 8) { 324757a0227fSdrh /* Two lots of padding: Pad the first block to 64 bytes */ 324857a0227fSdrh memset(p, 0, count); 324957a0227fSdrh byteReverse(ctx->in, 16); 325057a0227fSdrh MD5Transform(ctx->buf, (uint32 *)ctx->in); 325157a0227fSdrh 325257a0227fSdrh /* Now fill the next block with 56 bytes */ 325357a0227fSdrh memset(ctx->in, 0, 56); 325457a0227fSdrh } else { 325557a0227fSdrh /* Pad block to 56 bytes */ 325657a0227fSdrh memset(p, 0, count-8); 325757a0227fSdrh } 325857a0227fSdrh byteReverse(ctx->in, 14); 325957a0227fSdrh 326057a0227fSdrh /* Append length in bits and transform */ 326157a0227fSdrh ((uint32 *)ctx->in)[ 14 ] = ctx->bits[0]; 326257a0227fSdrh ((uint32 *)ctx->in)[ 15 ] = ctx->bits[1]; 326357a0227fSdrh 326457a0227fSdrh MD5Transform(ctx->buf, (uint32 *)ctx->in); 326557a0227fSdrh byteReverse((unsigned char *)ctx->buf, 4); 326657a0227fSdrh memcpy(digest, ctx->buf, 16); 326757a0227fSdrh memset(ctx, 0, sizeof(ctx)); /* In case it is sensitive */ 326857a0227fSdrh } 326957a0227fSdrh 327057a0227fSdrh /* 327157a0227fSdrh ** Convert a 128-bit MD5 digest into a 32-digit base-16 number. 327257a0227fSdrh */ 327357a0227fSdrh static void MD5DigestToBase16(unsigned char *digest, char *zBuf){ 327457a0227fSdrh static char const zEncode[] = "0123456789abcdef"; 327557a0227fSdrh int i, j; 327657a0227fSdrh 327757a0227fSdrh for(j=i=0; i<16; i++){ 327857a0227fSdrh int a = digest[i]; 327957a0227fSdrh zBuf[j++] = zEncode[(a>>4)&0xf]; 328057a0227fSdrh zBuf[j++] = zEncode[a & 0xf]; 328157a0227fSdrh } 328257a0227fSdrh zBuf[j] = 0; 328357a0227fSdrh } 328457a0227fSdrh 328557a0227fSdrh 328657a0227fSdrh /* 328757a0227fSdrh ** Convert a 128-bit MD5 digest into sequency of eight 5-digit integers 328857a0227fSdrh ** each representing 16 bits of the digest and separated from each 328957a0227fSdrh ** other by a "-" character. 329057a0227fSdrh */ 329157a0227fSdrh static void MD5DigestToBase10x8(unsigned char digest[16], char zDigest[50]){ 329257a0227fSdrh int i, j; 329357a0227fSdrh unsigned int x; 329457a0227fSdrh for(i=j=0; i<16; i+=2){ 329557a0227fSdrh x = digest[i]*256 + digest[i+1]; 329657a0227fSdrh if( i>0 ) zDigest[j++] = '-'; 329757a0227fSdrh sprintf(&zDigest[j], "%05u", x); 329857a0227fSdrh j += 5; 329957a0227fSdrh } 330057a0227fSdrh zDigest[j] = 0; 330157a0227fSdrh } 330257a0227fSdrh 330357a0227fSdrh /* 330457a0227fSdrh ** A TCL command for md5. The argument is the text to be hashed. The 330557a0227fSdrh ** Result is the hash in base64. 330657a0227fSdrh */ 330757a0227fSdrh static int md5_cmd(void*cd, Tcl_Interp *interp, int argc, const char **argv){ 330857a0227fSdrh MD5Context ctx; 330957a0227fSdrh unsigned char digest[16]; 331057a0227fSdrh char zBuf[50]; 331157a0227fSdrh void (*converter)(unsigned char*, char*); 331257a0227fSdrh 331357a0227fSdrh if( argc!=2 ){ 331457a0227fSdrh Tcl_AppendResult(interp,"wrong # args: should be \"", argv[0], 331557a0227fSdrh " TEXT\"", 0); 331657a0227fSdrh return TCL_ERROR; 331757a0227fSdrh } 331857a0227fSdrh MD5Init(&ctx); 331957a0227fSdrh MD5Update(&ctx, (unsigned char*)argv[1], (unsigned)strlen(argv[1])); 332057a0227fSdrh MD5Final(digest, &ctx); 332157a0227fSdrh converter = (void(*)(unsigned char*,char*))cd; 332257a0227fSdrh converter(digest, zBuf); 332357a0227fSdrh Tcl_AppendResult(interp, zBuf, (char*)0); 332457a0227fSdrh return TCL_OK; 332557a0227fSdrh } 332657a0227fSdrh 332757a0227fSdrh /* 332857a0227fSdrh ** A TCL command to take the md5 hash of a file. The argument is the 332957a0227fSdrh ** name of the file. 333057a0227fSdrh */ 333157a0227fSdrh static int md5file_cmd(void*cd, Tcl_Interp*interp, int argc, const char **argv){ 333257a0227fSdrh FILE *in; 333357a0227fSdrh MD5Context ctx; 333457a0227fSdrh void (*converter)(unsigned char*, char*); 333557a0227fSdrh unsigned char digest[16]; 333657a0227fSdrh char zBuf[10240]; 333757a0227fSdrh 333857a0227fSdrh if( argc!=2 ){ 333957a0227fSdrh Tcl_AppendResult(interp,"wrong # args: should be \"", argv[0], 334057a0227fSdrh " FILENAME\"", 0); 334157a0227fSdrh return TCL_ERROR; 334257a0227fSdrh } 334357a0227fSdrh in = fopen(argv[1],"rb"); 334457a0227fSdrh if( in==0 ){ 334557a0227fSdrh Tcl_AppendResult(interp,"unable to open file \"", argv[1], 334657a0227fSdrh "\" for reading", 0); 334757a0227fSdrh return TCL_ERROR; 334857a0227fSdrh } 334957a0227fSdrh MD5Init(&ctx); 335057a0227fSdrh for(;;){ 335157a0227fSdrh int n; 335257a0227fSdrh n = fread(zBuf, 1, sizeof(zBuf), in); 335357a0227fSdrh if( n<=0 ) break; 335457a0227fSdrh MD5Update(&ctx, (unsigned char*)zBuf, (unsigned)n); 335557a0227fSdrh } 335657a0227fSdrh fclose(in); 335757a0227fSdrh MD5Final(digest, &ctx); 335857a0227fSdrh converter = (void(*)(unsigned char*,char*))cd; 335957a0227fSdrh converter(digest, zBuf); 336057a0227fSdrh Tcl_AppendResult(interp, zBuf, (char*)0); 336157a0227fSdrh return TCL_OK; 336257a0227fSdrh } 336357a0227fSdrh 336457a0227fSdrh /* 336557a0227fSdrh ** Register the four new TCL commands for generating MD5 checksums 336657a0227fSdrh ** with the TCL interpreter. 336757a0227fSdrh */ 336857a0227fSdrh int Md5_Init(Tcl_Interp *interp){ 336957a0227fSdrh Tcl_CreateCommand(interp, "md5", (Tcl_CmdProc*)md5_cmd, 337057a0227fSdrh MD5DigestToBase16, 0); 337157a0227fSdrh Tcl_CreateCommand(interp, "md5-10x8", (Tcl_CmdProc*)md5_cmd, 337257a0227fSdrh MD5DigestToBase10x8, 0); 337357a0227fSdrh Tcl_CreateCommand(interp, "md5file", (Tcl_CmdProc*)md5file_cmd, 337457a0227fSdrh MD5DigestToBase16, 0); 337557a0227fSdrh Tcl_CreateCommand(interp, "md5file-10x8", (Tcl_CmdProc*)md5file_cmd, 337657a0227fSdrh MD5DigestToBase10x8, 0); 337757a0227fSdrh return TCL_OK; 337857a0227fSdrh } 337957a0227fSdrh #endif /* defined(SQLITE_TEST) || defined(SQLITE_TCLMD5) */ 338057a0227fSdrh 338157a0227fSdrh #if defined(SQLITE_TEST) 338257a0227fSdrh /* 338357a0227fSdrh ** During testing, the special md5sum() aggregate function is available. 338457a0227fSdrh ** inside SQLite. The following routines implement that function. 338557a0227fSdrh */ 338657a0227fSdrh static void md5step(sqlite3_context *context, int argc, sqlite3_value **argv){ 338757a0227fSdrh MD5Context *p; 338857a0227fSdrh int i; 338957a0227fSdrh if( argc<1 ) return; 339057a0227fSdrh p = sqlite3_aggregate_context(context, sizeof(*p)); 339157a0227fSdrh if( p==0 ) return; 339257a0227fSdrh if( !p->isInit ){ 339357a0227fSdrh MD5Init(p); 339457a0227fSdrh } 339557a0227fSdrh for(i=0; i<argc; i++){ 339657a0227fSdrh const char *zData = (char*)sqlite3_value_text(argv[i]); 339757a0227fSdrh if( zData ){ 339857a0227fSdrh MD5Update(p, (unsigned char*)zData, strlen(zData)); 339957a0227fSdrh } 340057a0227fSdrh } 340157a0227fSdrh } 340257a0227fSdrh static void md5finalize(sqlite3_context *context){ 340357a0227fSdrh MD5Context *p; 340457a0227fSdrh unsigned char digest[16]; 340557a0227fSdrh char zBuf[33]; 340657a0227fSdrh p = sqlite3_aggregate_context(context, sizeof(*p)); 340757a0227fSdrh MD5Final(digest,p); 340857a0227fSdrh MD5DigestToBase16(digest, zBuf); 340957a0227fSdrh sqlite3_result_text(context, zBuf, -1, SQLITE_TRANSIENT); 341057a0227fSdrh } 341157a0227fSdrh int Md5_Register(sqlite3 *db){ 341257a0227fSdrh int rc = sqlite3_create_function(db, "md5sum", -1, SQLITE_UTF8, 0, 0, 341357a0227fSdrh md5step, md5finalize); 341457a0227fSdrh sqlite3_overload_function(db, "md5sum", -1); /* To exercise this API */ 341557a0227fSdrh return rc; 341657a0227fSdrh } 341757a0227fSdrh #endif /* defined(SQLITE_TEST) */ 341857a0227fSdrh 341957a0227fSdrh 3420348784efSdrh /* 34213e27c026Sdrh ** If the macro TCLSH is one, then put in code this for the 34223e27c026Sdrh ** "main" routine that will initialize Tcl and take input from 34233570ad93Sdrh ** standard input, or if a file is named on the command line 34243570ad93Sdrh ** the TCL interpreter reads and evaluates that file. 3425348784efSdrh */ 34263e27c026Sdrh #if TCLSH==1 3427348784efSdrh static char zMainloop[] = 3428348784efSdrh "set line {}\n" 3429348784efSdrh "while {![eof stdin]} {\n" 3430348784efSdrh "if {$line!=\"\"} {\n" 3431348784efSdrh "puts -nonewline \"> \"\n" 3432348784efSdrh "} else {\n" 3433348784efSdrh "puts -nonewline \"% \"\n" 3434348784efSdrh "}\n" 3435348784efSdrh "flush stdout\n" 3436348784efSdrh "append line [gets stdin]\n" 3437348784efSdrh "if {[info complete $line]} {\n" 3438348784efSdrh "if {[catch {uplevel #0 $line} result]} {\n" 3439348784efSdrh "puts stderr \"Error: $result\"\n" 3440348784efSdrh "} elseif {$result!=\"\"} {\n" 3441348784efSdrh "puts $result\n" 3442348784efSdrh "}\n" 3443348784efSdrh "set line {}\n" 3444348784efSdrh "} else {\n" 3445348784efSdrh "append line \\n\n" 3446348784efSdrh "}\n" 3447348784efSdrh "}\n" 3448348784efSdrh ; 34493e27c026Sdrh #endif 34503e27c026Sdrh 3451348784efSdrh #define TCLSH_MAIN main /* Needed to fake out mktclapp */ 3452348784efSdrh int TCLSH_MAIN(int argc, char **argv){ 3453348784efSdrh Tcl_Interp *interp; 34540a549071Sdanielk1977 34550a549071Sdanielk1977 /* Call sqlite3_shutdown() once before doing anything else. This is to 34560a549071Sdanielk1977 ** test that sqlite3_shutdown() can be safely called by a process before 34570a549071Sdanielk1977 ** sqlite3_initialize() is. */ 34580a549071Sdanielk1977 sqlite3_shutdown(); 34590a549071Sdanielk1977 3460297ecf14Sdrh Tcl_FindExecutable(argv[0]); 3461348784efSdrh interp = Tcl_CreateInterp(); 346238f8271fSdrh Sqlite3_Init(interp); 346357a0227fSdrh #if defined(SQLITE_TEST) || defined(SQLITE_TCLMD5) 346457a0227fSdrh Md5_Init(interp); 346557a0227fSdrh #endif 3466d9b0257aSdrh #ifdef SQLITE_TEST 3467d1bf3512Sdrh { 34682f999a67Sdrh extern int Sqliteconfig_Init(Tcl_Interp*); 3469d1bf3512Sdrh extern int Sqlitetest1_Init(Tcl_Interp*); 34705c4d9703Sdrh extern int Sqlitetest2_Init(Tcl_Interp*); 34715c4d9703Sdrh extern int Sqlitetest3_Init(Tcl_Interp*); 3472a6064dcfSdrh extern int Sqlitetest4_Init(Tcl_Interp*); 3473998b56c3Sdanielk1977 extern int Sqlitetest5_Init(Tcl_Interp*); 34749c06c953Sdrh extern int Sqlitetest6_Init(Tcl_Interp*); 347529c636bcSdrh extern int Sqlitetest7_Init(Tcl_Interp*); 3476b9bb7c18Sdrh extern int Sqlitetest8_Init(Tcl_Interp*); 3477a713f2c3Sdanielk1977 extern int Sqlitetest9_Init(Tcl_Interp*); 34782366940dSdrh extern int Sqlitetestasync_Init(Tcl_Interp*); 34791409be69Sdrh extern int Sqlitetest_autoext_Init(Tcl_Interp*); 3480*0a7a9155Sdan extern int Sqlitetest_demovfs_Init(Tcl_Interp *); 3481984bfaa4Sdrh extern int Sqlitetest_func_Init(Tcl_Interp*); 348215926590Sdrh extern int Sqlitetest_hexio_Init(Tcl_Interp*); 3483e1ab2193Sdan extern int Sqlitetest_init_Init(Tcl_Interp*); 34842f999a67Sdrh extern int Sqlitetest_malloc_Init(Tcl_Interp*); 34851a9ed0b2Sdanielk1977 extern int Sqlitetest_mutex_Init(Tcl_Interp*); 34862f999a67Sdrh extern int Sqlitetestschema_Init(Tcl_Interp*); 34872f999a67Sdrh extern int Sqlitetestsse_Init(Tcl_Interp*); 34882f999a67Sdrh extern int Sqlitetesttclvar_Init(Tcl_Interp*); 348944918fa0Sdanielk1977 extern int SqlitetestThread_Init(Tcl_Interp*); 3490a15db353Sdanielk1977 extern int SqlitetestOnefile_Init(); 34915d1f5aa6Sdanielk1977 extern int SqlitetestOsinst_Init(Tcl_Interp*); 34920410302eSdanielk1977 extern int Sqlitetestbackup_Init(Tcl_Interp*); 3493522efc62Sdrh extern int Sqlitetestintarray_Init(Tcl_Interp*); 34942e66f0b9Sdrh 34952f999a67Sdrh Sqliteconfig_Init(interp); 34966490bebdSdanielk1977 Sqlitetest1_Init(interp); 34975c4d9703Sdrh Sqlitetest2_Init(interp); 3498de647130Sdrh Sqlitetest3_Init(interp); 3499fc57d7bfSdanielk1977 Sqlitetest4_Init(interp); 3500998b56c3Sdanielk1977 Sqlitetest5_Init(interp); 35019c06c953Sdrh Sqlitetest6_Init(interp); 350229c636bcSdrh Sqlitetest7_Init(interp); 3503b9bb7c18Sdrh Sqlitetest8_Init(interp); 3504a713f2c3Sdanielk1977 Sqlitetest9_Init(interp); 35052366940dSdrh Sqlitetestasync_Init(interp); 35061409be69Sdrh Sqlitetest_autoext_Init(interp); 3507*0a7a9155Sdan Sqlitetest_demovfs_Init(interp); 3508984bfaa4Sdrh Sqlitetest_func_Init(interp); 350915926590Sdrh Sqlitetest_hexio_Init(interp); 3510e1ab2193Sdan Sqlitetest_init_Init(interp); 35112f999a67Sdrh Sqlitetest_malloc_Init(interp); 35121a9ed0b2Sdanielk1977 Sqlitetest_mutex_Init(interp); 35132f999a67Sdrh Sqlitetestschema_Init(interp); 35142f999a67Sdrh Sqlitetesttclvar_Init(interp); 351544918fa0Sdanielk1977 SqlitetestThread_Init(interp); 3516a15db353Sdanielk1977 SqlitetestOnefile_Init(interp); 35175d1f5aa6Sdanielk1977 SqlitetestOsinst_Init(interp); 35180410302eSdanielk1977 Sqlitetestbackup_Init(interp); 3519522efc62Sdrh Sqlitetestintarray_Init(interp); 3520a15db353Sdanielk1977 352189dec819Sdrh #ifdef SQLITE_SSE 35222e66f0b9Sdrh Sqlitetestsse_Init(interp); 35232e66f0b9Sdrh #endif 3524d1bf3512Sdrh } 3525d1bf3512Sdrh #endif 3526c7285978Sdrh if( argc>=2 ){ 3527348784efSdrh int i; 3528ad42c3a3Sshess char zArgc[32]; 3529ad42c3a3Sshess sqlite3_snprintf(sizeof(zArgc), zArgc, "%d", argc-(3-TCLSH)); 3530ad42c3a3Sshess Tcl_SetVar(interp,"argc", zArgc, TCL_GLOBAL_ONLY); 3531348784efSdrh Tcl_SetVar(interp,"argv0",argv[1],TCL_GLOBAL_ONLY); 3532348784efSdrh Tcl_SetVar(interp,"argv", "", TCL_GLOBAL_ONLY); 353361212b69Sdrh for(i=3-TCLSH; i<argc; i++){ 3534348784efSdrh Tcl_SetVar(interp, "argv", argv[i], 3535348784efSdrh TCL_GLOBAL_ONLY | TCL_LIST_ELEMENT | TCL_APPEND_VALUE); 3536348784efSdrh } 3537c7285978Sdrh if( Tcl_EvalFile(interp, argv[1])!=TCL_OK ){ 35380de8c112Sdrh const char *zInfo = Tcl_GetVar(interp, "errorInfo", TCL_GLOBAL_ONLY); 3539a81c64a2Sdrh if( zInfo==0 ) zInfo = Tcl_GetStringResult(interp); 3540c61053b7Sdrh fprintf(stderr,"%s: %s\n", *argv, zInfo); 3541348784efSdrh return 1; 3542348784efSdrh } 35433e27c026Sdrh } 3544c7285978Sdrh if( argc<=1 ){ 3545348784efSdrh Tcl_GlobalEval(interp, zMainloop); 3546348784efSdrh } 3547348784efSdrh return 0; 3548348784efSdrh } 3549348784efSdrh #endif /* TCLSH */ 3550