175897234Sdrh /* 2b19a2bc6Sdrh ** 2001 September 15 375897234Sdrh ** 4b19a2bc6Sdrh ** The author disclaims copyright to this source code. In place of 5b19a2bc6Sdrh ** a legal notice, here is a blessing: 675897234Sdrh ** 7b19a2bc6Sdrh ** May you do good and not evil. 8b19a2bc6Sdrh ** May you find forgiveness for yourself and forgive others. 9b19a2bc6Sdrh ** May you share freely, never taking more than you give. 1075897234Sdrh ** 1175897234Sdrh ************************************************************************* 12bd08af48Sdrh ** A TCL Interface to SQLite. Append this file to sqlite3.c and 13bd08af48Sdrh ** compile the whole thing to build a TCL-enabled version of SQLite. 1475897234Sdrh ** 15*33f4e02aSdrh ** $Id: tclsqlite.c,v 1.201 2007/09/03 15:19:35 drh Exp $ 1675897234Sdrh */ 17bd08af48Sdrh #include "tcl.h" 18b4e9af9fSdanielk1977 #include <errno.h> 196d31316cSdrh 20bd08af48Sdrh /* 21bd08af48Sdrh ** Some additional include files are needed if this file is not 22bd08af48Sdrh ** appended to the amalgamation. 23bd08af48Sdrh */ 24bd08af48Sdrh #ifndef SQLITE_AMALGAMATION 2506b2718aSdrh # include "sqliteInt.h" 2675897234Sdrh # include <stdlib.h> 2775897234Sdrh # include <string.h> 28ce927065Sdrh # include <assert.h> 29d1e4733dSdrh # include <ctype.h> 30bd08af48Sdrh #endif 3175897234Sdrh 32ad6e1370Sdrh /* 33ad6e1370Sdrh * Windows needs to know which symbols to export. Unix does not. 34ad6e1370Sdrh * BUILD_sqlite should be undefined for Unix. 35ad6e1370Sdrh */ 36ad6e1370Sdrh #ifdef BUILD_sqlite 37ad6e1370Sdrh #undef TCL_STORAGE_CLASS 38ad6e1370Sdrh #define TCL_STORAGE_CLASS DLLEXPORT 39ad6e1370Sdrh #endif /* BUILD_sqlite */ 4029bc4615Sdrh 41a21c6b6fSdanielk1977 #define NUM_PREPARED_STMTS 10 42fb7e7651Sdrh #define MAX_PREPARED_STMTS 100 43fb7e7651Sdrh 4475897234Sdrh /* 4598808babSdrh ** If TCL uses UTF-8 and SQLite is configured to use iso8859, then we 4698808babSdrh ** have to do a translation when going between the two. Set the 4798808babSdrh ** UTF_TRANSLATION_NEEDED macro to indicate that we need to do 4898808babSdrh ** this translation. 4998808babSdrh */ 5098808babSdrh #if defined(TCL_UTF_MAX) && !defined(SQLITE_UTF8) 5198808babSdrh # define UTF_TRANSLATION_NEEDED 1 5298808babSdrh #endif 5398808babSdrh 5498808babSdrh /* 55cabb0819Sdrh ** New SQL functions can be created as TCL scripts. Each such function 56cabb0819Sdrh ** is described by an instance of the following structure. 57cabb0819Sdrh */ 58cabb0819Sdrh typedef struct SqlFunc SqlFunc; 59cabb0819Sdrh struct SqlFunc { 60cabb0819Sdrh Tcl_Interp *interp; /* The TCL interpret to execute the function */ 61d1e4733dSdrh Tcl_Obj *pScript; /* The Tcl_Obj representation of the script */ 62d1e4733dSdrh int useEvalObjv; /* True if it is safe to use Tcl_EvalObjv */ 63d1e4733dSdrh char *zName; /* Name of this function */ 64cabb0819Sdrh SqlFunc *pNext; /* Next function on the list of them all */ 65cabb0819Sdrh }; 66cabb0819Sdrh 67cabb0819Sdrh /* 680202b29eSdanielk1977 ** New collation sequences function can be created as TCL scripts. Each such 690202b29eSdanielk1977 ** function is described by an instance of the following structure. 700202b29eSdanielk1977 */ 710202b29eSdanielk1977 typedef struct SqlCollate SqlCollate; 720202b29eSdanielk1977 struct SqlCollate { 730202b29eSdanielk1977 Tcl_Interp *interp; /* The TCL interpret to execute the function */ 740202b29eSdanielk1977 char *zScript; /* The script to be run */ 750202b29eSdanielk1977 SqlCollate *pNext; /* Next function on the list of them all */ 760202b29eSdanielk1977 }; 770202b29eSdanielk1977 780202b29eSdanielk1977 /* 79fb7e7651Sdrh ** Prepared statements are cached for faster execution. Each prepared 80fb7e7651Sdrh ** statement is described by an instance of the following structure. 81fb7e7651Sdrh */ 82fb7e7651Sdrh typedef struct SqlPreparedStmt SqlPreparedStmt; 83fb7e7651Sdrh struct SqlPreparedStmt { 84fb7e7651Sdrh SqlPreparedStmt *pNext; /* Next in linked list */ 85fb7e7651Sdrh SqlPreparedStmt *pPrev; /* Previous on the list */ 86fb7e7651Sdrh sqlite3_stmt *pStmt; /* The prepared statement */ 87fb7e7651Sdrh int nSql; /* chars in zSql[] */ 88fb7e7651Sdrh char zSql[1]; /* Text of the SQL statement */ 89fb7e7651Sdrh }; 90fb7e7651Sdrh 91d0441796Sdanielk1977 typedef struct IncrblobChannel IncrblobChannel; 92d0441796Sdanielk1977 93fb7e7651Sdrh /* 94bec3f402Sdrh ** There is one instance of this structure for each SQLite database 95bec3f402Sdrh ** that has been opened by the SQLite TCL interface. 96bec3f402Sdrh */ 97bec3f402Sdrh typedef struct SqliteDb SqliteDb; 98bec3f402Sdrh struct SqliteDb { 99dddca286Sdrh sqlite3 *db; /* The "real" database structure. MUST BE FIRST */ 100bec3f402Sdrh Tcl_Interp *interp; /* The interpreter used for this database */ 1016d31316cSdrh char *zBusy; /* The busy callback routine */ 102aa940eacSdrh char *zCommit; /* The commit hook callback routine */ 103b5a20d3cSdrh char *zTrace; /* The trace callback routine */ 10419e2d37fSdrh char *zProfile; /* The profile callback routine */ 105348bb5d6Sdanielk1977 char *zProgress; /* The progress callback routine */ 106e22a334bSdrh char *zAuth; /* The authorization callback routine */ 10755c45f2eSdanielk1977 char *zNull; /* Text to substitute for an SQL NULL value */ 108cabb0819Sdrh SqlFunc *pFunc; /* List of SQL functions */ 10994eb6a14Sdanielk1977 Tcl_Obj *pUpdateHook; /* Update hook script (if any) */ 11071fd80bfSdanielk1977 Tcl_Obj *pRollbackHook; /* Rollback hook script (if any) */ 1110202b29eSdanielk1977 SqlCollate *pCollate; /* List of SQL collation functions */ 1126f8a503dSdanielk1977 int rc; /* Return code of most recent sqlite3_exec() */ 1137cedc8d4Sdanielk1977 Tcl_Obj *pCollateNeeded; /* Collation needed script */ 114fb7e7651Sdrh SqlPreparedStmt *stmtList; /* List of prepared statements*/ 115fb7e7651Sdrh SqlPreparedStmt *stmtLast; /* Last statement in the list */ 116fb7e7651Sdrh int maxStmt; /* The next maximum number of stmtList */ 117fb7e7651Sdrh int nStmt; /* Number of statements in stmtList */ 118d0441796Sdanielk1977 IncrblobChannel *pIncrblob;/* Linked list of open incrblob channels */ 11998808babSdrh }; 120297ecf14Sdrh 121b4e9af9fSdanielk1977 struct IncrblobChannel { 122d0441796Sdanielk1977 sqlite3_blob *pBlob; /* sqlite3 blob handle */ 123dcbb5d3fSdanielk1977 SqliteDb *pDb; /* Associated database connection */ 124b4e9af9fSdanielk1977 int iSeek; /* Current seek offset */ 125d0441796Sdanielk1977 Tcl_Channel channel; /* Channel identifier */ 126d0441796Sdanielk1977 IncrblobChannel *pNext; /* Linked list of all open incrblob channels */ 127d0441796Sdanielk1977 IncrblobChannel *pPrev; /* Linked list of all open incrblob channels */ 128b4e9af9fSdanielk1977 }; 129b4e9af9fSdanielk1977 13032a0d8bbSdanielk1977 #ifndef SQLITE_OMIT_INCRBLOB 131b4e9af9fSdanielk1977 /* 132d0441796Sdanielk1977 ** Close all incrblob channels opened using database connection pDb. 133d0441796Sdanielk1977 ** This is called when shutting down the database connection. 134d0441796Sdanielk1977 */ 135d0441796Sdanielk1977 static void closeIncrblobChannels(SqliteDb *pDb){ 136d0441796Sdanielk1977 IncrblobChannel *p; 137d0441796Sdanielk1977 IncrblobChannel *pNext; 138d0441796Sdanielk1977 139d0441796Sdanielk1977 for(p=pDb->pIncrblob; p; p=pNext){ 140d0441796Sdanielk1977 pNext = p->pNext; 141d0441796Sdanielk1977 142d0441796Sdanielk1977 /* Note: Calling unregister here call Tcl_Close on the incrblob channel, 143d0441796Sdanielk1977 ** which deletes the IncrblobChannel structure at *p. So do not 144d0441796Sdanielk1977 ** call Tcl_Free() here. 145d0441796Sdanielk1977 */ 146d0441796Sdanielk1977 Tcl_UnregisterChannel(pDb->interp, p->channel); 147d0441796Sdanielk1977 } 148d0441796Sdanielk1977 } 149d0441796Sdanielk1977 150d0441796Sdanielk1977 /* 151b4e9af9fSdanielk1977 ** Close an incremental blob channel. 152b4e9af9fSdanielk1977 */ 153b4e9af9fSdanielk1977 static int incrblobClose(ClientData instanceData, Tcl_Interp *interp){ 154b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 15592d4d7a9Sdanielk1977 int rc = sqlite3_blob_close(p->pBlob); 15692d4d7a9Sdanielk1977 sqlite3 *db = p->pDb->db; 157d0441796Sdanielk1977 158d0441796Sdanielk1977 /* Remove the channel from the SqliteDb.pIncrblob list. */ 159d0441796Sdanielk1977 if( p->pNext ){ 160d0441796Sdanielk1977 p->pNext->pPrev = p->pPrev; 161d0441796Sdanielk1977 } 162d0441796Sdanielk1977 if( p->pPrev ){ 163d0441796Sdanielk1977 p->pPrev->pNext = p->pNext; 164d0441796Sdanielk1977 } 165d0441796Sdanielk1977 if( p->pDb->pIncrblob==p ){ 166d0441796Sdanielk1977 p->pDb->pIncrblob = p->pNext; 167d0441796Sdanielk1977 } 168d0441796Sdanielk1977 16992d4d7a9Sdanielk1977 /* Free the IncrblobChannel structure */ 170b4e9af9fSdanielk1977 Tcl_Free((char *)p); 17192d4d7a9Sdanielk1977 17292d4d7a9Sdanielk1977 if( rc!=SQLITE_OK ){ 17392d4d7a9Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(db), TCL_VOLATILE); 17492d4d7a9Sdanielk1977 return TCL_ERROR; 17592d4d7a9Sdanielk1977 } 176b4e9af9fSdanielk1977 return TCL_OK; 177b4e9af9fSdanielk1977 } 178b4e9af9fSdanielk1977 179b4e9af9fSdanielk1977 /* 180b4e9af9fSdanielk1977 ** Read data from an incremental blob channel. 181b4e9af9fSdanielk1977 */ 182b4e9af9fSdanielk1977 static int incrblobInput( 183b4e9af9fSdanielk1977 ClientData instanceData, 184b4e9af9fSdanielk1977 char *buf, 185b4e9af9fSdanielk1977 int bufSize, 186b4e9af9fSdanielk1977 int *errorCodePtr 187b4e9af9fSdanielk1977 ){ 188b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 189b4e9af9fSdanielk1977 int nRead = bufSize; /* Number of bytes to read */ 190b4e9af9fSdanielk1977 int nBlob; /* Total size of the blob */ 191b4e9af9fSdanielk1977 int rc; /* sqlite error code */ 192b4e9af9fSdanielk1977 193b4e9af9fSdanielk1977 nBlob = sqlite3_blob_bytes(p->pBlob); 194b4e9af9fSdanielk1977 if( (p->iSeek+nRead)>nBlob ){ 195b4e9af9fSdanielk1977 nRead = nBlob-p->iSeek; 196b4e9af9fSdanielk1977 } 197b4e9af9fSdanielk1977 if( nRead<=0 ){ 198b4e9af9fSdanielk1977 return 0; 199b4e9af9fSdanielk1977 } 200b4e9af9fSdanielk1977 201b4e9af9fSdanielk1977 rc = sqlite3_blob_read(p->pBlob, (void *)buf, nRead, p->iSeek); 202b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 203b4e9af9fSdanielk1977 *errorCodePtr = rc; 204b4e9af9fSdanielk1977 return -1; 205b4e9af9fSdanielk1977 } 206b4e9af9fSdanielk1977 207b4e9af9fSdanielk1977 p->iSeek += nRead; 208b4e9af9fSdanielk1977 return nRead; 209b4e9af9fSdanielk1977 } 210b4e9af9fSdanielk1977 211d0441796Sdanielk1977 /* 212d0441796Sdanielk1977 ** Write data to an incremental blob channel. 213d0441796Sdanielk1977 */ 214b4e9af9fSdanielk1977 static int incrblobOutput( 215b4e9af9fSdanielk1977 ClientData instanceData, 216b4e9af9fSdanielk1977 CONST char *buf, 217b4e9af9fSdanielk1977 int toWrite, 218b4e9af9fSdanielk1977 int *errorCodePtr 219b4e9af9fSdanielk1977 ){ 220b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 221b4e9af9fSdanielk1977 int nWrite = toWrite; /* Number of bytes to write */ 222b4e9af9fSdanielk1977 int nBlob; /* Total size of the blob */ 223b4e9af9fSdanielk1977 int rc; /* sqlite error code */ 224b4e9af9fSdanielk1977 225b4e9af9fSdanielk1977 nBlob = sqlite3_blob_bytes(p->pBlob); 226b4e9af9fSdanielk1977 if( (p->iSeek+nWrite)>nBlob ){ 227b4e9af9fSdanielk1977 *errorCodePtr = EINVAL; 228b4e9af9fSdanielk1977 return -1; 229b4e9af9fSdanielk1977 } 230b4e9af9fSdanielk1977 if( nWrite<=0 ){ 231b4e9af9fSdanielk1977 return 0; 232b4e9af9fSdanielk1977 } 233b4e9af9fSdanielk1977 234b4e9af9fSdanielk1977 rc = sqlite3_blob_write(p->pBlob, (void *)buf, nWrite, p->iSeek); 235b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 236b4e9af9fSdanielk1977 *errorCodePtr = EIO; 237b4e9af9fSdanielk1977 return -1; 238b4e9af9fSdanielk1977 } 239b4e9af9fSdanielk1977 240b4e9af9fSdanielk1977 p->iSeek += nWrite; 241b4e9af9fSdanielk1977 return nWrite; 242b4e9af9fSdanielk1977 } 243b4e9af9fSdanielk1977 244b4e9af9fSdanielk1977 /* 245b4e9af9fSdanielk1977 ** Seek an incremental blob channel. 246b4e9af9fSdanielk1977 */ 247b4e9af9fSdanielk1977 static int incrblobSeek( 248b4e9af9fSdanielk1977 ClientData instanceData, 249b4e9af9fSdanielk1977 long offset, 250b4e9af9fSdanielk1977 int seekMode, 251b4e9af9fSdanielk1977 int *errorCodePtr 252b4e9af9fSdanielk1977 ){ 253b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 254b4e9af9fSdanielk1977 255b4e9af9fSdanielk1977 switch( seekMode ){ 256b4e9af9fSdanielk1977 case SEEK_SET: 257b4e9af9fSdanielk1977 p->iSeek = offset; 258b4e9af9fSdanielk1977 break; 259b4e9af9fSdanielk1977 case SEEK_CUR: 260b4e9af9fSdanielk1977 p->iSeek += offset; 261b4e9af9fSdanielk1977 break; 262b4e9af9fSdanielk1977 case SEEK_END: 263b4e9af9fSdanielk1977 p->iSeek = sqlite3_blob_bytes(p->pBlob) + offset; 264b4e9af9fSdanielk1977 break; 265b4e9af9fSdanielk1977 266b4e9af9fSdanielk1977 default: assert(!"Bad seekMode"); 267b4e9af9fSdanielk1977 } 268b4e9af9fSdanielk1977 269b4e9af9fSdanielk1977 return p->iSeek; 270b4e9af9fSdanielk1977 } 271b4e9af9fSdanielk1977 272b4e9af9fSdanielk1977 273b4e9af9fSdanielk1977 static void incrblobWatch(ClientData instanceData, int mode){ 274b4e9af9fSdanielk1977 /* NO-OP */ 275b4e9af9fSdanielk1977 } 276b4e9af9fSdanielk1977 static int incrblobHandle(ClientData instanceData, int dir, ClientData *hPtr){ 277b4e9af9fSdanielk1977 return TCL_ERROR; 278b4e9af9fSdanielk1977 } 279b4e9af9fSdanielk1977 280b4e9af9fSdanielk1977 static Tcl_ChannelType IncrblobChannelType = { 281b4e9af9fSdanielk1977 "incrblob", /* typeName */ 282b4e9af9fSdanielk1977 TCL_CHANNEL_VERSION_2, /* version */ 283b4e9af9fSdanielk1977 incrblobClose, /* closeProc */ 284b4e9af9fSdanielk1977 incrblobInput, /* inputProc */ 285b4e9af9fSdanielk1977 incrblobOutput, /* outputProc */ 286b4e9af9fSdanielk1977 incrblobSeek, /* seekProc */ 287b4e9af9fSdanielk1977 0, /* setOptionProc */ 288b4e9af9fSdanielk1977 0, /* getOptionProc */ 289b4e9af9fSdanielk1977 incrblobWatch, /* watchProc (this is a no-op) */ 290b4e9af9fSdanielk1977 incrblobHandle, /* getHandleProc (always returns error) */ 291b4e9af9fSdanielk1977 0, /* close2Proc */ 292b4e9af9fSdanielk1977 0, /* blockModeProc */ 293b4e9af9fSdanielk1977 0, /* flushProc */ 294b4e9af9fSdanielk1977 0, /* handlerProc */ 295b4e9af9fSdanielk1977 0, /* wideSeekProc */ 296b4e9af9fSdanielk1977 }; 297b4e9af9fSdanielk1977 298b4e9af9fSdanielk1977 /* 299b4e9af9fSdanielk1977 ** Create a new incrblob channel. 300b4e9af9fSdanielk1977 */ 301b4e9af9fSdanielk1977 static int createIncrblobChannel( 302b4e9af9fSdanielk1977 Tcl_Interp *interp, 303b4e9af9fSdanielk1977 SqliteDb *pDb, 304b4e9af9fSdanielk1977 const char *zDb, 305b4e9af9fSdanielk1977 const char *zTable, 306b4e9af9fSdanielk1977 const char *zColumn, 3078cbadb02Sdanielk1977 sqlite_int64 iRow, 3088cbadb02Sdanielk1977 int isReadonly 309b4e9af9fSdanielk1977 ){ 310b4e9af9fSdanielk1977 IncrblobChannel *p; 3118cbadb02Sdanielk1977 sqlite3 *db = pDb->db; 312b4e9af9fSdanielk1977 sqlite3_blob *pBlob; 313b4e9af9fSdanielk1977 int rc; 3148cbadb02Sdanielk1977 int flags = TCL_READABLE|(isReadonly ? 0 : TCL_WRITABLE); 315b4e9af9fSdanielk1977 316b4e9af9fSdanielk1977 /* This variable is used to name the channels: "incrblob_[incr count]" */ 317b4e9af9fSdanielk1977 static int count = 0; 318b4e9af9fSdanielk1977 char zChannel[64]; 319b4e9af9fSdanielk1977 3208cbadb02Sdanielk1977 rc = sqlite3_blob_open(db, zDb, zTable, zColumn, iRow, !isReadonly, &pBlob); 321b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 322b4e9af9fSdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 323b4e9af9fSdanielk1977 return TCL_ERROR; 324b4e9af9fSdanielk1977 } 325b4e9af9fSdanielk1977 326b4e9af9fSdanielk1977 p = (IncrblobChannel *)Tcl_Alloc(sizeof(IncrblobChannel)); 327b4e9af9fSdanielk1977 p->iSeek = 0; 328b4e9af9fSdanielk1977 p->pBlob = pBlob; 329b4e9af9fSdanielk1977 3305bb3eb9bSdrh sqlite3_snprintf(sizeof(zChannel), zChannel, "incrblob_%d", ++count); 331d0441796Sdanielk1977 p->channel = Tcl_CreateChannel(&IncrblobChannelType, zChannel, p, flags); 332d0441796Sdanielk1977 Tcl_RegisterChannel(interp, p->channel); 333b4e9af9fSdanielk1977 334d0441796Sdanielk1977 /* Link the new channel into the SqliteDb.pIncrblob list. */ 335d0441796Sdanielk1977 p->pNext = pDb->pIncrblob; 336d0441796Sdanielk1977 p->pPrev = 0; 337d0441796Sdanielk1977 if( p->pNext ){ 338d0441796Sdanielk1977 p->pNext->pPrev = p; 339d0441796Sdanielk1977 } 340d0441796Sdanielk1977 pDb->pIncrblob = p; 341d0441796Sdanielk1977 p->pDb = pDb; 342d0441796Sdanielk1977 343d0441796Sdanielk1977 Tcl_SetResult(interp, (char *)Tcl_GetChannelName(p->channel), TCL_VOLATILE); 344b4e9af9fSdanielk1977 return TCL_OK; 345b4e9af9fSdanielk1977 } 34632a0d8bbSdanielk1977 #else /* else clause for "#ifndef SQLITE_OMIT_INCRBLOB" */ 34732a0d8bbSdanielk1977 #define closeIncrblobChannels(pDb) 34832a0d8bbSdanielk1977 #endif 349b4e9af9fSdanielk1977 3506d31316cSdrh /* 351d1e4733dSdrh ** Look at the script prefix in pCmd. We will be executing this script 352d1e4733dSdrh ** after first appending one or more arguments. This routine analyzes 353d1e4733dSdrh ** the script to see if it is safe to use Tcl_EvalObjv() on the script 354d1e4733dSdrh ** rather than the more general Tcl_EvalEx(). Tcl_EvalObjv() is much 355d1e4733dSdrh ** faster. 356d1e4733dSdrh ** 357d1e4733dSdrh ** Scripts that are safe to use with Tcl_EvalObjv() consists of a 358d1e4733dSdrh ** command name followed by zero or more arguments with no [...] or $ 359d1e4733dSdrh ** or {...} or ; to be seen anywhere. Most callback scripts consist 360d1e4733dSdrh ** of just a single procedure name and they meet this requirement. 361d1e4733dSdrh */ 362d1e4733dSdrh static int safeToUseEvalObjv(Tcl_Interp *interp, Tcl_Obj *pCmd){ 363d1e4733dSdrh /* We could try to do something with Tcl_Parse(). But we will instead 364d1e4733dSdrh ** just do a search for forbidden characters. If any of the forbidden 365d1e4733dSdrh ** characters appear in pCmd, we will report the string as unsafe. 366d1e4733dSdrh */ 367d1e4733dSdrh const char *z; 368d1e4733dSdrh int n; 369d1e4733dSdrh z = Tcl_GetStringFromObj(pCmd, &n); 370d1e4733dSdrh while( n-- > 0 ){ 371d1e4733dSdrh int c = *(z++); 372d1e4733dSdrh if( c=='$' || c=='[' || c==';' ) return 0; 373d1e4733dSdrh } 374d1e4733dSdrh return 1; 375d1e4733dSdrh } 376d1e4733dSdrh 377d1e4733dSdrh /* 378d1e4733dSdrh ** Find an SqlFunc structure with the given name. Or create a new 379d1e4733dSdrh ** one if an existing one cannot be found. Return a pointer to the 380d1e4733dSdrh ** structure. 381d1e4733dSdrh */ 382d1e4733dSdrh static SqlFunc *findSqlFunc(SqliteDb *pDb, const char *zName){ 383d1e4733dSdrh SqlFunc *p, *pNew; 384d1e4733dSdrh int i; 385d1e4733dSdrh pNew = (SqlFunc*)Tcl_Alloc( sizeof(*pNew) + strlen(zName) + 1 ); 386d1e4733dSdrh pNew->zName = (char*)&pNew[1]; 387d1e4733dSdrh for(i=0; zName[i]; i++){ pNew->zName[i] = tolower(zName[i]); } 388d1e4733dSdrh pNew->zName[i] = 0; 389d1e4733dSdrh for(p=pDb->pFunc; p; p=p->pNext){ 390d1e4733dSdrh if( strcmp(p->zName, pNew->zName)==0 ){ 391d1e4733dSdrh Tcl_Free((char*)pNew); 392d1e4733dSdrh return p; 393d1e4733dSdrh } 394d1e4733dSdrh } 395d1e4733dSdrh pNew->interp = pDb->interp; 396d1e4733dSdrh pNew->pScript = 0; 397d1e4733dSdrh pNew->pNext = pDb->pFunc; 398d1e4733dSdrh pDb->pFunc = pNew; 399d1e4733dSdrh return pNew; 400d1e4733dSdrh } 401d1e4733dSdrh 402d1e4733dSdrh /* 403fb7e7651Sdrh ** Finalize and free a list of prepared statements 404fb7e7651Sdrh */ 405fb7e7651Sdrh static void flushStmtCache( SqliteDb *pDb ){ 406fb7e7651Sdrh SqlPreparedStmt *pPreStmt; 407fb7e7651Sdrh 408fb7e7651Sdrh while( pDb->stmtList ){ 409fb7e7651Sdrh sqlite3_finalize( pDb->stmtList->pStmt ); 410fb7e7651Sdrh pPreStmt = pDb->stmtList; 411fb7e7651Sdrh pDb->stmtList = pDb->stmtList->pNext; 412fb7e7651Sdrh Tcl_Free( (char*)pPreStmt ); 413fb7e7651Sdrh } 414fb7e7651Sdrh pDb->nStmt = 0; 415fb7e7651Sdrh pDb->stmtLast = 0; 416fb7e7651Sdrh } 417fb7e7651Sdrh 418fb7e7651Sdrh /* 419895d7472Sdrh ** TCL calls this procedure when an sqlite3 database command is 420895d7472Sdrh ** deleted. 42175897234Sdrh */ 42275897234Sdrh static void DbDeleteCmd(void *db){ 423bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)db; 424fb7e7651Sdrh flushStmtCache(pDb); 425d0441796Sdanielk1977 closeIncrblobChannels(pDb); 4266f8a503dSdanielk1977 sqlite3_close(pDb->db); 427cabb0819Sdrh while( pDb->pFunc ){ 428cabb0819Sdrh SqlFunc *pFunc = pDb->pFunc; 429cabb0819Sdrh pDb->pFunc = pFunc->pNext; 430d1e4733dSdrh Tcl_DecrRefCount(pFunc->pScript); 431cabb0819Sdrh Tcl_Free((char*)pFunc); 432cabb0819Sdrh } 4330202b29eSdanielk1977 while( pDb->pCollate ){ 4340202b29eSdanielk1977 SqlCollate *pCollate = pDb->pCollate; 4350202b29eSdanielk1977 pDb->pCollate = pCollate->pNext; 4360202b29eSdanielk1977 Tcl_Free((char*)pCollate); 4370202b29eSdanielk1977 } 438bec3f402Sdrh if( pDb->zBusy ){ 439bec3f402Sdrh Tcl_Free(pDb->zBusy); 440bec3f402Sdrh } 441b5a20d3cSdrh if( pDb->zTrace ){ 442b5a20d3cSdrh Tcl_Free(pDb->zTrace); 4430d1a643aSdrh } 44419e2d37fSdrh if( pDb->zProfile ){ 44519e2d37fSdrh Tcl_Free(pDb->zProfile); 44619e2d37fSdrh } 447e22a334bSdrh if( pDb->zAuth ){ 448e22a334bSdrh Tcl_Free(pDb->zAuth); 449e22a334bSdrh } 45055c45f2eSdanielk1977 if( pDb->zNull ){ 45155c45f2eSdanielk1977 Tcl_Free(pDb->zNull); 45255c45f2eSdanielk1977 } 45394eb6a14Sdanielk1977 if( pDb->pUpdateHook ){ 45494eb6a14Sdanielk1977 Tcl_DecrRefCount(pDb->pUpdateHook); 45594eb6a14Sdanielk1977 } 45671fd80bfSdanielk1977 if( pDb->pRollbackHook ){ 45771fd80bfSdanielk1977 Tcl_DecrRefCount(pDb->pRollbackHook); 45871fd80bfSdanielk1977 } 45994eb6a14Sdanielk1977 if( pDb->pCollateNeeded ){ 46094eb6a14Sdanielk1977 Tcl_DecrRefCount(pDb->pCollateNeeded); 46194eb6a14Sdanielk1977 } 462bec3f402Sdrh Tcl_Free((char*)pDb); 463bec3f402Sdrh } 464bec3f402Sdrh 465bec3f402Sdrh /* 466bec3f402Sdrh ** This routine is called when a database file is locked while trying 467bec3f402Sdrh ** to execute SQL. 468bec3f402Sdrh */ 4692a764eb0Sdanielk1977 static int DbBusyHandler(void *cd, int nTries){ 470bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)cd; 471bec3f402Sdrh int rc; 472bec3f402Sdrh char zVal[30]; 473bec3f402Sdrh 4745bb3eb9bSdrh sqlite3_snprintf(sizeof(zVal), zVal, "%d", nTries); 475d1e4733dSdrh rc = Tcl_VarEval(pDb->interp, pDb->zBusy, " ", zVal, (char*)0); 476bec3f402Sdrh if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 477bec3f402Sdrh return 0; 478bec3f402Sdrh } 479bec3f402Sdrh return 1; 48075897234Sdrh } 48175897234Sdrh 48275897234Sdrh /* 483348bb5d6Sdanielk1977 ** This routine is invoked as the 'progress callback' for the database. 484348bb5d6Sdanielk1977 */ 485348bb5d6Sdanielk1977 static int DbProgressHandler(void *cd){ 486348bb5d6Sdanielk1977 SqliteDb *pDb = (SqliteDb*)cd; 487348bb5d6Sdanielk1977 int rc; 488348bb5d6Sdanielk1977 489348bb5d6Sdanielk1977 assert( pDb->zProgress ); 490348bb5d6Sdanielk1977 rc = Tcl_Eval(pDb->interp, pDb->zProgress); 491348bb5d6Sdanielk1977 if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 492348bb5d6Sdanielk1977 return 1; 493348bb5d6Sdanielk1977 } 494348bb5d6Sdanielk1977 return 0; 495348bb5d6Sdanielk1977 } 496348bb5d6Sdanielk1977 497d1167393Sdrh #ifndef SQLITE_OMIT_TRACE 498348bb5d6Sdanielk1977 /* 499b5a20d3cSdrh ** This routine is called by the SQLite trace handler whenever a new 500b5a20d3cSdrh ** block of SQL is executed. The TCL script in pDb->zTrace is executed. 5010d1a643aSdrh */ 502b5a20d3cSdrh static void DbTraceHandler(void *cd, const char *zSql){ 5030d1a643aSdrh SqliteDb *pDb = (SqliteDb*)cd; 504b5a20d3cSdrh Tcl_DString str; 5050d1a643aSdrh 506b5a20d3cSdrh Tcl_DStringInit(&str); 507b5a20d3cSdrh Tcl_DStringAppend(&str, pDb->zTrace, -1); 508b5a20d3cSdrh Tcl_DStringAppendElement(&str, zSql); 509b5a20d3cSdrh Tcl_Eval(pDb->interp, Tcl_DStringValue(&str)); 510b5a20d3cSdrh Tcl_DStringFree(&str); 511b5a20d3cSdrh Tcl_ResetResult(pDb->interp); 5120d1a643aSdrh } 513d1167393Sdrh #endif 5140d1a643aSdrh 515d1167393Sdrh #ifndef SQLITE_OMIT_TRACE 5160d1a643aSdrh /* 51719e2d37fSdrh ** This routine is called by the SQLite profile handler after a statement 51819e2d37fSdrh ** SQL has executed. The TCL script in pDb->zProfile is evaluated. 51919e2d37fSdrh */ 52019e2d37fSdrh static void DbProfileHandler(void *cd, const char *zSql, sqlite_uint64 tm){ 52119e2d37fSdrh SqliteDb *pDb = (SqliteDb*)cd; 52219e2d37fSdrh Tcl_DString str; 52319e2d37fSdrh char zTm[100]; 52419e2d37fSdrh 52519e2d37fSdrh sqlite3_snprintf(sizeof(zTm)-1, zTm, "%lld", tm); 52619e2d37fSdrh Tcl_DStringInit(&str); 52719e2d37fSdrh Tcl_DStringAppend(&str, pDb->zProfile, -1); 52819e2d37fSdrh Tcl_DStringAppendElement(&str, zSql); 52919e2d37fSdrh Tcl_DStringAppendElement(&str, zTm); 53019e2d37fSdrh Tcl_Eval(pDb->interp, Tcl_DStringValue(&str)); 53119e2d37fSdrh Tcl_DStringFree(&str); 53219e2d37fSdrh Tcl_ResetResult(pDb->interp); 53319e2d37fSdrh } 534d1167393Sdrh #endif 53519e2d37fSdrh 53619e2d37fSdrh /* 537aa940eacSdrh ** This routine is called when a transaction is committed. The 538aa940eacSdrh ** TCL script in pDb->zCommit is executed. If it returns non-zero or 539aa940eacSdrh ** if it throws an exception, the transaction is rolled back instead 540aa940eacSdrh ** of being committed. 541aa940eacSdrh */ 542aa940eacSdrh static int DbCommitHandler(void *cd){ 543aa940eacSdrh SqliteDb *pDb = (SqliteDb*)cd; 544aa940eacSdrh int rc; 545aa940eacSdrh 546aa940eacSdrh rc = Tcl_Eval(pDb->interp, pDb->zCommit); 547aa940eacSdrh if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 548aa940eacSdrh return 1; 549aa940eacSdrh } 550aa940eacSdrh return 0; 551aa940eacSdrh } 552aa940eacSdrh 55371fd80bfSdanielk1977 static void DbRollbackHandler(void *clientData){ 55471fd80bfSdanielk1977 SqliteDb *pDb = (SqliteDb*)clientData; 55571fd80bfSdanielk1977 assert(pDb->pRollbackHook); 55671fd80bfSdanielk1977 if( TCL_OK!=Tcl_EvalObjEx(pDb->interp, pDb->pRollbackHook, 0) ){ 55771fd80bfSdanielk1977 Tcl_BackgroundError(pDb->interp); 55871fd80bfSdanielk1977 } 55971fd80bfSdanielk1977 } 56071fd80bfSdanielk1977 56194eb6a14Sdanielk1977 static void DbUpdateHandler( 56294eb6a14Sdanielk1977 void *p, 56394eb6a14Sdanielk1977 int op, 56494eb6a14Sdanielk1977 const char *zDb, 56594eb6a14Sdanielk1977 const char *zTbl, 56694eb6a14Sdanielk1977 sqlite_int64 rowid 56794eb6a14Sdanielk1977 ){ 56894eb6a14Sdanielk1977 SqliteDb *pDb = (SqliteDb *)p; 56994eb6a14Sdanielk1977 Tcl_Obj *pCmd; 57094eb6a14Sdanielk1977 57194eb6a14Sdanielk1977 assert( pDb->pUpdateHook ); 57294eb6a14Sdanielk1977 assert( op==SQLITE_INSERT || op==SQLITE_UPDATE || op==SQLITE_DELETE ); 57394eb6a14Sdanielk1977 57494eb6a14Sdanielk1977 pCmd = Tcl_DuplicateObj(pDb->pUpdateHook); 57594eb6a14Sdanielk1977 Tcl_IncrRefCount(pCmd); 57694eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj( 57794eb6a14Sdanielk1977 ( (op==SQLITE_INSERT)?"INSERT":(op==SQLITE_UPDATE)?"UPDATE":"DELETE"), -1)); 57894eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj(zDb, -1)); 57994eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj(zTbl, -1)); 58094eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewWideIntObj(rowid)); 58194eb6a14Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pCmd, TCL_EVAL_DIRECT); 58294eb6a14Sdanielk1977 } 58394eb6a14Sdanielk1977 5847cedc8d4Sdanielk1977 static void tclCollateNeeded( 5857cedc8d4Sdanielk1977 void *pCtx, 5869bb575fdSdrh sqlite3 *db, 5877cedc8d4Sdanielk1977 int enc, 5887cedc8d4Sdanielk1977 const char *zName 5897cedc8d4Sdanielk1977 ){ 5907cedc8d4Sdanielk1977 SqliteDb *pDb = (SqliteDb *)pCtx; 5917cedc8d4Sdanielk1977 Tcl_Obj *pScript = Tcl_DuplicateObj(pDb->pCollateNeeded); 5927cedc8d4Sdanielk1977 Tcl_IncrRefCount(pScript); 5937cedc8d4Sdanielk1977 Tcl_ListObjAppendElement(0, pScript, Tcl_NewStringObj(zName, -1)); 5947cedc8d4Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pScript, 0); 5957cedc8d4Sdanielk1977 Tcl_DecrRefCount(pScript); 5967cedc8d4Sdanielk1977 } 5977cedc8d4Sdanielk1977 598aa940eacSdrh /* 5990202b29eSdanielk1977 ** This routine is called to evaluate an SQL collation function implemented 6000202b29eSdanielk1977 ** using TCL script. 6010202b29eSdanielk1977 */ 6020202b29eSdanielk1977 static int tclSqlCollate( 6030202b29eSdanielk1977 void *pCtx, 6040202b29eSdanielk1977 int nA, 6050202b29eSdanielk1977 const void *zA, 6060202b29eSdanielk1977 int nB, 6070202b29eSdanielk1977 const void *zB 6080202b29eSdanielk1977 ){ 6090202b29eSdanielk1977 SqlCollate *p = (SqlCollate *)pCtx; 6100202b29eSdanielk1977 Tcl_Obj *pCmd; 6110202b29eSdanielk1977 6120202b29eSdanielk1977 pCmd = Tcl_NewStringObj(p->zScript, -1); 6130202b29eSdanielk1977 Tcl_IncrRefCount(pCmd); 6140202b29eSdanielk1977 Tcl_ListObjAppendElement(p->interp, pCmd, Tcl_NewStringObj(zA, nA)); 6150202b29eSdanielk1977 Tcl_ListObjAppendElement(p->interp, pCmd, Tcl_NewStringObj(zB, nB)); 616d1e4733dSdrh Tcl_EvalObjEx(p->interp, pCmd, TCL_EVAL_DIRECT); 6170202b29eSdanielk1977 Tcl_DecrRefCount(pCmd); 6180202b29eSdanielk1977 return (atoi(Tcl_GetStringResult(p->interp))); 6190202b29eSdanielk1977 } 6200202b29eSdanielk1977 6210202b29eSdanielk1977 /* 622cabb0819Sdrh ** This routine is called to evaluate an SQL function implemented 623cabb0819Sdrh ** using TCL script. 624cabb0819Sdrh */ 6250ae8b831Sdanielk1977 static void tclSqlFunc(sqlite3_context *context, int argc, sqlite3_value**argv){ 6266f8a503dSdanielk1977 SqlFunc *p = sqlite3_user_data(context); 627d1e4733dSdrh Tcl_Obj *pCmd; 628cabb0819Sdrh int i; 629cabb0819Sdrh int rc; 630cabb0819Sdrh 631d1e4733dSdrh if( argc==0 ){ 632d1e4733dSdrh /* If there are no arguments to the function, call Tcl_EvalObjEx on the 633d1e4733dSdrh ** script object directly. This allows the TCL compiler to generate 634d1e4733dSdrh ** bytecode for the command on the first invocation and thus make 635d1e4733dSdrh ** subsequent invocations much faster. */ 636d1e4733dSdrh pCmd = p->pScript; 637d1e4733dSdrh Tcl_IncrRefCount(pCmd); 638d1e4733dSdrh rc = Tcl_EvalObjEx(p->interp, pCmd, 0); 639d1e4733dSdrh Tcl_DecrRefCount(pCmd); 64051ad0ecdSdanielk1977 }else{ 641d1e4733dSdrh /* If there are arguments to the function, make a shallow copy of the 642d1e4733dSdrh ** script object, lappend the arguments, then evaluate the copy. 643d1e4733dSdrh ** 644d1e4733dSdrh ** By "shallow" copy, we mean a only the outer list Tcl_Obj is duplicated. 645d1e4733dSdrh ** The new Tcl_Obj contains pointers to the original list elements. 646d1e4733dSdrh ** That way, when Tcl_EvalObjv() is run and shimmers the first element 647d1e4733dSdrh ** of the list to tclCmdNameType, that alternate representation will 648d1e4733dSdrh ** be preserved and reused on the next invocation. 649d1e4733dSdrh */ 650d1e4733dSdrh Tcl_Obj **aArg; 651d1e4733dSdrh int nArg; 652d1e4733dSdrh if( Tcl_ListObjGetElements(p->interp, p->pScript, &nArg, &aArg) ){ 653d1e4733dSdrh sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 654d1e4733dSdrh return; 655d1e4733dSdrh } 656d1e4733dSdrh pCmd = Tcl_NewListObj(nArg, aArg); 657d1e4733dSdrh Tcl_IncrRefCount(pCmd); 658d1e4733dSdrh for(i=0; i<argc; i++){ 659d1e4733dSdrh sqlite3_value *pIn = argv[i]; 660d1e4733dSdrh Tcl_Obj *pVal; 661d1e4733dSdrh 662d1e4733dSdrh /* Set pVal to contain the i'th column of this row. */ 663d1e4733dSdrh switch( sqlite3_value_type(pIn) ){ 664d1e4733dSdrh case SQLITE_BLOB: { 665d1e4733dSdrh int bytes = sqlite3_value_bytes(pIn); 666d1e4733dSdrh pVal = Tcl_NewByteArrayObj(sqlite3_value_blob(pIn), bytes); 667d1e4733dSdrh break; 668d1e4733dSdrh } 669d1e4733dSdrh case SQLITE_INTEGER: { 670d1e4733dSdrh sqlite_int64 v = sqlite3_value_int64(pIn); 671d1e4733dSdrh if( v>=-2147483647 && v<=2147483647 ){ 672d1e4733dSdrh pVal = Tcl_NewIntObj(v); 673d1e4733dSdrh }else{ 674d1e4733dSdrh pVal = Tcl_NewWideIntObj(v); 675d1e4733dSdrh } 676d1e4733dSdrh break; 677d1e4733dSdrh } 678d1e4733dSdrh case SQLITE_FLOAT: { 679d1e4733dSdrh double r = sqlite3_value_double(pIn); 680d1e4733dSdrh pVal = Tcl_NewDoubleObj(r); 681d1e4733dSdrh break; 682d1e4733dSdrh } 683d1e4733dSdrh case SQLITE_NULL: { 684d1e4733dSdrh pVal = Tcl_NewStringObj("", 0); 685d1e4733dSdrh break; 686d1e4733dSdrh } 687d1e4733dSdrh default: { 688d1e4733dSdrh int bytes = sqlite3_value_bytes(pIn); 68900fd957bSdanielk1977 pVal = Tcl_NewStringObj((char *)sqlite3_value_text(pIn), bytes); 690d1e4733dSdrh break; 69151ad0ecdSdanielk1977 } 692cabb0819Sdrh } 693d1e4733dSdrh rc = Tcl_ListObjAppendElement(p->interp, pCmd, pVal); 694d1e4733dSdrh if( rc ){ 695d1e4733dSdrh Tcl_DecrRefCount(pCmd); 696d1e4733dSdrh sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 697d1e4733dSdrh return; 698d1e4733dSdrh } 699d1e4733dSdrh } 700d1e4733dSdrh if( !p->useEvalObjv ){ 701d1e4733dSdrh /* Tcl_EvalObjEx() will automatically call Tcl_EvalObjv() if pCmd 702d1e4733dSdrh ** is a list without a string representation. To prevent this from 703d1e4733dSdrh ** happening, make sure pCmd has a valid string representation */ 704d1e4733dSdrh Tcl_GetString(pCmd); 705d1e4733dSdrh } 706d1e4733dSdrh rc = Tcl_EvalObjEx(p->interp, pCmd, TCL_EVAL_DIRECT); 707d1e4733dSdrh Tcl_DecrRefCount(pCmd); 708d1e4733dSdrh } 709562e8d3cSdanielk1977 710c7f269d5Sdrh if( rc && rc!=TCL_RETURN ){ 7117e18c259Sdanielk1977 sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 712cabb0819Sdrh }else{ 713c7f269d5Sdrh Tcl_Obj *pVar = Tcl_GetObjResult(p->interp); 714c7f269d5Sdrh int n; 715c7f269d5Sdrh u8 *data; 716c7f269d5Sdrh char *zType = pVar->typePtr ? pVar->typePtr->name : ""; 717c7f269d5Sdrh char c = zType[0]; 718df0bddaeSdrh if( c=='b' && strcmp(zType,"bytearray")==0 && pVar->bytes==0 ){ 719d1e4733dSdrh /* Only return a BLOB type if the Tcl variable is a bytearray and 720df0bddaeSdrh ** has no string representation. */ 721c7f269d5Sdrh data = Tcl_GetByteArrayFromObj(pVar, &n); 722c7f269d5Sdrh sqlite3_result_blob(context, data, n, SQLITE_TRANSIENT); 723985e0c63Sdrh }else if( c=='b' && strcmp(zType,"boolean")==0 ){ 724c7f269d5Sdrh Tcl_GetIntFromObj(0, pVar, &n); 725c7f269d5Sdrh sqlite3_result_int(context, n); 726c7f269d5Sdrh }else if( c=='d' && strcmp(zType,"double")==0 ){ 727c7f269d5Sdrh double r; 728c7f269d5Sdrh Tcl_GetDoubleFromObj(0, pVar, &r); 729c7f269d5Sdrh sqlite3_result_double(context, r); 730985e0c63Sdrh }else if( (c=='w' && strcmp(zType,"wideInt")==0) || 731985e0c63Sdrh (c=='i' && strcmp(zType,"int")==0) ){ 732df0bddaeSdrh Tcl_WideInt v; 733df0bddaeSdrh Tcl_GetWideIntFromObj(0, pVar, &v); 734df0bddaeSdrh sqlite3_result_int64(context, v); 735c7f269d5Sdrh }else{ 73600fd957bSdanielk1977 data = (unsigned char *)Tcl_GetStringFromObj(pVar, &n); 73700fd957bSdanielk1977 sqlite3_result_text(context, (char *)data, n, SQLITE_TRANSIENT); 738c7f269d5Sdrh } 739cabb0819Sdrh } 740cabb0819Sdrh } 741895d7472Sdrh 742e22a334bSdrh #ifndef SQLITE_OMIT_AUTHORIZATION 743e22a334bSdrh /* 744e22a334bSdrh ** This is the authentication function. It appends the authentication 745e22a334bSdrh ** type code and the two arguments to zCmd[] then invokes the result 746e22a334bSdrh ** on the interpreter. The reply is examined to determine if the 747e22a334bSdrh ** authentication fails or succeeds. 748e22a334bSdrh */ 749e22a334bSdrh static int auth_callback( 750e22a334bSdrh void *pArg, 751e22a334bSdrh int code, 752e22a334bSdrh const char *zArg1, 753e22a334bSdrh const char *zArg2, 754e22a334bSdrh const char *zArg3, 755e22a334bSdrh const char *zArg4 756e22a334bSdrh ){ 757e22a334bSdrh char *zCode; 758e22a334bSdrh Tcl_DString str; 759e22a334bSdrh int rc; 760e22a334bSdrh const char *zReply; 761e22a334bSdrh SqliteDb *pDb = (SqliteDb*)pArg; 762e22a334bSdrh 763e22a334bSdrh switch( code ){ 764e22a334bSdrh case SQLITE_COPY : zCode="SQLITE_COPY"; break; 765e22a334bSdrh case SQLITE_CREATE_INDEX : zCode="SQLITE_CREATE_INDEX"; break; 766e22a334bSdrh case SQLITE_CREATE_TABLE : zCode="SQLITE_CREATE_TABLE"; break; 767e22a334bSdrh case SQLITE_CREATE_TEMP_INDEX : zCode="SQLITE_CREATE_TEMP_INDEX"; break; 768e22a334bSdrh case SQLITE_CREATE_TEMP_TABLE : zCode="SQLITE_CREATE_TEMP_TABLE"; break; 769e22a334bSdrh case SQLITE_CREATE_TEMP_TRIGGER: zCode="SQLITE_CREATE_TEMP_TRIGGER"; break; 770e22a334bSdrh case SQLITE_CREATE_TEMP_VIEW : zCode="SQLITE_CREATE_TEMP_VIEW"; break; 771e22a334bSdrh case SQLITE_CREATE_TRIGGER : zCode="SQLITE_CREATE_TRIGGER"; break; 772e22a334bSdrh case SQLITE_CREATE_VIEW : zCode="SQLITE_CREATE_VIEW"; break; 773e22a334bSdrh case SQLITE_DELETE : zCode="SQLITE_DELETE"; break; 774e22a334bSdrh case SQLITE_DROP_INDEX : zCode="SQLITE_DROP_INDEX"; break; 775e22a334bSdrh case SQLITE_DROP_TABLE : zCode="SQLITE_DROP_TABLE"; break; 776e22a334bSdrh case SQLITE_DROP_TEMP_INDEX : zCode="SQLITE_DROP_TEMP_INDEX"; break; 777e22a334bSdrh case SQLITE_DROP_TEMP_TABLE : zCode="SQLITE_DROP_TEMP_TABLE"; break; 778e22a334bSdrh case SQLITE_DROP_TEMP_TRIGGER : zCode="SQLITE_DROP_TEMP_TRIGGER"; break; 779e22a334bSdrh case SQLITE_DROP_TEMP_VIEW : zCode="SQLITE_DROP_TEMP_VIEW"; break; 780e22a334bSdrh case SQLITE_DROP_TRIGGER : zCode="SQLITE_DROP_TRIGGER"; break; 781e22a334bSdrh case SQLITE_DROP_VIEW : zCode="SQLITE_DROP_VIEW"; break; 782e22a334bSdrh case SQLITE_INSERT : zCode="SQLITE_INSERT"; break; 783e22a334bSdrh case SQLITE_PRAGMA : zCode="SQLITE_PRAGMA"; break; 784e22a334bSdrh case SQLITE_READ : zCode="SQLITE_READ"; break; 785e22a334bSdrh case SQLITE_SELECT : zCode="SQLITE_SELECT"; break; 786e22a334bSdrh case SQLITE_TRANSACTION : zCode="SQLITE_TRANSACTION"; break; 787e22a334bSdrh case SQLITE_UPDATE : zCode="SQLITE_UPDATE"; break; 78881e293b4Sdrh case SQLITE_ATTACH : zCode="SQLITE_ATTACH"; break; 78981e293b4Sdrh case SQLITE_DETACH : zCode="SQLITE_DETACH"; break; 7901c8c23ccSdanielk1977 case SQLITE_ALTER_TABLE : zCode="SQLITE_ALTER_TABLE"; break; 7911d54df88Sdanielk1977 case SQLITE_REINDEX : zCode="SQLITE_REINDEX"; break; 792e6e04969Sdrh case SQLITE_ANALYZE : zCode="SQLITE_ANALYZE"; break; 793f1a381e7Sdanielk1977 case SQLITE_CREATE_VTABLE : zCode="SQLITE_CREATE_VTABLE"; break; 794f1a381e7Sdanielk1977 case SQLITE_DROP_VTABLE : zCode="SQLITE_DROP_VTABLE"; break; 7955169bbc6Sdrh case SQLITE_FUNCTION : zCode="SQLITE_FUNCTION"; break; 796e22a334bSdrh default : zCode="????"; break; 797e22a334bSdrh } 798e22a334bSdrh Tcl_DStringInit(&str); 799e22a334bSdrh Tcl_DStringAppend(&str, pDb->zAuth, -1); 800e22a334bSdrh Tcl_DStringAppendElement(&str, zCode); 801e22a334bSdrh Tcl_DStringAppendElement(&str, zArg1 ? zArg1 : ""); 802e22a334bSdrh Tcl_DStringAppendElement(&str, zArg2 ? zArg2 : ""); 803e22a334bSdrh Tcl_DStringAppendElement(&str, zArg3 ? zArg3 : ""); 804e22a334bSdrh Tcl_DStringAppendElement(&str, zArg4 ? zArg4 : ""); 805e22a334bSdrh rc = Tcl_GlobalEval(pDb->interp, Tcl_DStringValue(&str)); 806e22a334bSdrh Tcl_DStringFree(&str); 807e22a334bSdrh zReply = Tcl_GetStringResult(pDb->interp); 808e22a334bSdrh if( strcmp(zReply,"SQLITE_OK")==0 ){ 809e22a334bSdrh rc = SQLITE_OK; 810e22a334bSdrh }else if( strcmp(zReply,"SQLITE_DENY")==0 ){ 811e22a334bSdrh rc = SQLITE_DENY; 812e22a334bSdrh }else if( strcmp(zReply,"SQLITE_IGNORE")==0 ){ 813e22a334bSdrh rc = SQLITE_IGNORE; 814e22a334bSdrh }else{ 815e22a334bSdrh rc = 999; 816e22a334bSdrh } 817e22a334bSdrh return rc; 818e22a334bSdrh } 819e22a334bSdrh #endif /* SQLITE_OMIT_AUTHORIZATION */ 820cabb0819Sdrh 821cabb0819Sdrh /* 822ef2cb63eSdanielk1977 ** zText is a pointer to text obtained via an sqlite3_result_text() 823ef2cb63eSdanielk1977 ** or similar interface. This routine returns a Tcl string object, 824ef2cb63eSdanielk1977 ** reference count set to 0, containing the text. If a translation 825ef2cb63eSdanielk1977 ** between iso8859 and UTF-8 is required, it is preformed. 826ef2cb63eSdanielk1977 */ 827ef2cb63eSdanielk1977 static Tcl_Obj *dbTextToObj(char const *zText){ 828ef2cb63eSdanielk1977 Tcl_Obj *pVal; 829ef2cb63eSdanielk1977 #ifdef UTF_TRANSLATION_NEEDED 830ef2cb63eSdanielk1977 Tcl_DString dCol; 831ef2cb63eSdanielk1977 Tcl_DStringInit(&dCol); 832ef2cb63eSdanielk1977 Tcl_ExternalToUtfDString(NULL, zText, -1, &dCol); 833ef2cb63eSdanielk1977 pVal = Tcl_NewStringObj(Tcl_DStringValue(&dCol), -1); 834ef2cb63eSdanielk1977 Tcl_DStringFree(&dCol); 835ef2cb63eSdanielk1977 #else 836ef2cb63eSdanielk1977 pVal = Tcl_NewStringObj(zText, -1); 837ef2cb63eSdanielk1977 #endif 838ef2cb63eSdanielk1977 return pVal; 839ef2cb63eSdanielk1977 } 840ef2cb63eSdanielk1977 841ef2cb63eSdanielk1977 /* 8421067fe11Stpoindex ** This routine reads a line of text from FILE in, stores 8431067fe11Stpoindex ** the text in memory obtained from malloc() and returns a pointer 8441067fe11Stpoindex ** to the text. NULL is returned at end of file, or if malloc() 8451067fe11Stpoindex ** fails. 8461067fe11Stpoindex ** 8471067fe11Stpoindex ** The interface is like "readline" but no command-line editing 8481067fe11Stpoindex ** is done. 8491067fe11Stpoindex ** 8501067fe11Stpoindex ** copied from shell.c from '.import' command 8511067fe11Stpoindex */ 8521067fe11Stpoindex static char *local_getline(char *zPrompt, FILE *in){ 8531067fe11Stpoindex char *zLine; 8541067fe11Stpoindex int nLine; 8551067fe11Stpoindex int n; 8561067fe11Stpoindex int eol; 8571067fe11Stpoindex 8581067fe11Stpoindex nLine = 100; 8591067fe11Stpoindex zLine = malloc( nLine ); 8601067fe11Stpoindex if( zLine==0 ) return 0; 8611067fe11Stpoindex n = 0; 8621067fe11Stpoindex eol = 0; 8631067fe11Stpoindex while( !eol ){ 8641067fe11Stpoindex if( n+100>nLine ){ 8651067fe11Stpoindex nLine = nLine*2 + 100; 8661067fe11Stpoindex zLine = realloc(zLine, nLine); 8671067fe11Stpoindex if( zLine==0 ) return 0; 8681067fe11Stpoindex } 8691067fe11Stpoindex if( fgets(&zLine[n], nLine - n, in)==0 ){ 8701067fe11Stpoindex if( n==0 ){ 8711067fe11Stpoindex free(zLine); 8721067fe11Stpoindex return 0; 8731067fe11Stpoindex } 8741067fe11Stpoindex zLine[n] = 0; 8751067fe11Stpoindex eol = 1; 8761067fe11Stpoindex break; 8771067fe11Stpoindex } 8781067fe11Stpoindex while( zLine[n] ){ n++; } 8791067fe11Stpoindex if( n>0 && zLine[n-1]=='\n' ){ 8801067fe11Stpoindex n--; 8811067fe11Stpoindex zLine[n] = 0; 8821067fe11Stpoindex eol = 1; 8831067fe11Stpoindex } 8841067fe11Stpoindex } 8851067fe11Stpoindex zLine = realloc( zLine, n+1 ); 8861067fe11Stpoindex return zLine; 8871067fe11Stpoindex } 8881067fe11Stpoindex 8891067fe11Stpoindex /* 89075897234Sdrh ** The "sqlite" command below creates a new Tcl command for each 89175897234Sdrh ** connection it opens to an SQLite database. This routine is invoked 89275897234Sdrh ** whenever one of those connection-specific commands is executed 89375897234Sdrh ** in Tcl. For example, if you run Tcl code like this: 89475897234Sdrh ** 8959bb575fdSdrh ** sqlite3 db1 "my_database" 89675897234Sdrh ** db1 close 89775897234Sdrh ** 89875897234Sdrh ** The first command opens a connection to the "my_database" database 89975897234Sdrh ** and calls that connection "db1". The second command causes this 90075897234Sdrh ** subroutine to be invoked. 90175897234Sdrh */ 9026d31316cSdrh static int DbObjCmd(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){ 903bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)cd; 9046d31316cSdrh int choice; 90522fbcb8dSdrh int rc = TCL_OK; 9060de8c112Sdrh static const char *DB_strs[] = { 907fb7e7651Sdrh "authorizer", "busy", "cache", 908fb7e7651Sdrh "changes", "close", "collate", 909fb7e7651Sdrh "collation_needed", "commit_hook", "complete", 9104144905bSdrh "copy", "enable_load_extension","errorcode", 9114144905bSdrh "eval", "exists", "function", 91259fffd02Sdrh "incrblob", "interrupt", "last_insert_rowid", 91359fffd02Sdrh "nullvalue", "onecolumn", "profile", 91459fffd02Sdrh "progress", "rekey", "rollback_hook", 91559fffd02Sdrh "timeout", "total_changes", "trace", 91659fffd02Sdrh "transaction", "update_hook", "version", 91759fffd02Sdrh 0 9186d31316cSdrh }; 919411995dcSdrh enum DB_enum { 920fb7e7651Sdrh DB_AUTHORIZER, DB_BUSY, DB_CACHE, 921fb7e7651Sdrh DB_CHANGES, DB_CLOSE, DB_COLLATE, 922fb7e7651Sdrh DB_COLLATION_NEEDED, DB_COMMIT_HOOK, DB_COMPLETE, 9234144905bSdrh DB_COPY, DB_ENABLE_LOAD_EXTENSION,DB_ERRORCODE, 9244144905bSdrh DB_EVAL, DB_EXISTS, DB_FUNCTION, 92559fffd02Sdrh DB_INCRBLOB, DB_INTERRUPT, DB_LAST_INSERT_ROWID, 92659fffd02Sdrh DB_NULLVALUE, DB_ONECOLUMN, DB_PROFILE, 92759fffd02Sdrh DB_PROGRESS, DB_REKEY, DB_ROLLBACK_HOOK, 92859fffd02Sdrh DB_TIMEOUT, DB_TOTAL_CHANGES, DB_TRACE, 92959fffd02Sdrh DB_TRANSACTION, DB_UPDATE_HOOK, DB_VERSION 9306d31316cSdrh }; 9311067fe11Stpoindex /* don't leave trailing commas on DB_enum, it confuses the AIX xlc compiler */ 9326d31316cSdrh 9336d31316cSdrh if( objc<2 ){ 9346d31316cSdrh Tcl_WrongNumArgs(interp, 1, objv, "SUBCOMMAND ..."); 93575897234Sdrh return TCL_ERROR; 93675897234Sdrh } 937411995dcSdrh if( Tcl_GetIndexFromObj(interp, objv[1], DB_strs, "option", 0, &choice) ){ 9386d31316cSdrh return TCL_ERROR; 9396d31316cSdrh } 9406d31316cSdrh 941411995dcSdrh switch( (enum DB_enum)choice ){ 94275897234Sdrh 943e22a334bSdrh /* $db authorizer ?CALLBACK? 944e22a334bSdrh ** 945e22a334bSdrh ** Invoke the given callback to authorize each SQL operation as it is 946e22a334bSdrh ** compiled. 5 arguments are appended to the callback before it is 947e22a334bSdrh ** invoked: 948e22a334bSdrh ** 949e22a334bSdrh ** (1) The authorization type (ex: SQLITE_CREATE_TABLE, SQLITE_INSERT, ...) 950e22a334bSdrh ** (2) First descriptive name (depends on authorization type) 951e22a334bSdrh ** (3) Second descriptive name 952e22a334bSdrh ** (4) Name of the database (ex: "main", "temp") 953e22a334bSdrh ** (5) Name of trigger that is doing the access 954e22a334bSdrh ** 955e22a334bSdrh ** The callback should return on of the following strings: SQLITE_OK, 956e22a334bSdrh ** SQLITE_IGNORE, or SQLITE_DENY. Any other return value is an error. 957e22a334bSdrh ** 958e22a334bSdrh ** If this method is invoked with no arguments, the current authorization 959e22a334bSdrh ** callback string is returned. 960e22a334bSdrh */ 961e22a334bSdrh case DB_AUTHORIZER: { 9621211de37Sdrh #ifdef SQLITE_OMIT_AUTHORIZATION 9631211de37Sdrh Tcl_AppendResult(interp, "authorization not available in this build", 0); 9641211de37Sdrh return TCL_ERROR; 9651211de37Sdrh #else 966e22a334bSdrh if( objc>3 ){ 967e22a334bSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 9680f14e2ebSdrh return TCL_ERROR; 969e22a334bSdrh }else if( objc==2 ){ 970b5a20d3cSdrh if( pDb->zAuth ){ 971e22a334bSdrh Tcl_AppendResult(interp, pDb->zAuth, 0); 972e22a334bSdrh } 973e22a334bSdrh }else{ 974e22a334bSdrh char *zAuth; 975e22a334bSdrh int len; 976e22a334bSdrh if( pDb->zAuth ){ 977e22a334bSdrh Tcl_Free(pDb->zAuth); 978e22a334bSdrh } 979e22a334bSdrh zAuth = Tcl_GetStringFromObj(objv[2], &len); 980e22a334bSdrh if( zAuth && len>0 ){ 981e22a334bSdrh pDb->zAuth = Tcl_Alloc( len + 1 ); 9825bb3eb9bSdrh memcpy(pDb->zAuth, zAuth, len+1); 983e22a334bSdrh }else{ 984e22a334bSdrh pDb->zAuth = 0; 985e22a334bSdrh } 986e22a334bSdrh if( pDb->zAuth ){ 987e22a334bSdrh pDb->interp = interp; 9886f8a503dSdanielk1977 sqlite3_set_authorizer(pDb->db, auth_callback, pDb); 989e22a334bSdrh }else{ 9906f8a503dSdanielk1977 sqlite3_set_authorizer(pDb->db, 0, 0); 991e22a334bSdrh } 992e22a334bSdrh } 9931211de37Sdrh #endif 994e22a334bSdrh break; 995e22a334bSdrh } 996e22a334bSdrh 997bec3f402Sdrh /* $db busy ?CALLBACK? 998bec3f402Sdrh ** 999bec3f402Sdrh ** Invoke the given callback if an SQL statement attempts to open 1000bec3f402Sdrh ** a locked database file. 1001bec3f402Sdrh */ 10026d31316cSdrh case DB_BUSY: { 10036d31316cSdrh if( objc>3 ){ 10046d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "CALLBACK"); 1005bec3f402Sdrh return TCL_ERROR; 10066d31316cSdrh }else if( objc==2 ){ 1007bec3f402Sdrh if( pDb->zBusy ){ 1008bec3f402Sdrh Tcl_AppendResult(interp, pDb->zBusy, 0); 1009bec3f402Sdrh } 1010bec3f402Sdrh }else{ 10116d31316cSdrh char *zBusy; 10126d31316cSdrh int len; 1013bec3f402Sdrh if( pDb->zBusy ){ 1014bec3f402Sdrh Tcl_Free(pDb->zBusy); 10156d31316cSdrh } 10166d31316cSdrh zBusy = Tcl_GetStringFromObj(objv[2], &len); 10176d31316cSdrh if( zBusy && len>0 ){ 10186d31316cSdrh pDb->zBusy = Tcl_Alloc( len + 1 ); 10195bb3eb9bSdrh memcpy(pDb->zBusy, zBusy, len+1); 10206d31316cSdrh }else{ 1021bec3f402Sdrh pDb->zBusy = 0; 1022bec3f402Sdrh } 1023bec3f402Sdrh if( pDb->zBusy ){ 1024bec3f402Sdrh pDb->interp = interp; 10256f8a503dSdanielk1977 sqlite3_busy_handler(pDb->db, DbBusyHandler, pDb); 10266d31316cSdrh }else{ 10276f8a503dSdanielk1977 sqlite3_busy_handler(pDb->db, 0, 0); 1028bec3f402Sdrh } 1029bec3f402Sdrh } 10306d31316cSdrh break; 10316d31316cSdrh } 1032bec3f402Sdrh 1033fb7e7651Sdrh /* $db cache flush 1034fb7e7651Sdrh ** $db cache size n 1035fb7e7651Sdrh ** 1036fb7e7651Sdrh ** Flush the prepared statement cache, or set the maximum number of 1037fb7e7651Sdrh ** cached statements. 1038fb7e7651Sdrh */ 1039fb7e7651Sdrh case DB_CACHE: { 1040fb7e7651Sdrh char *subCmd; 1041fb7e7651Sdrh int n; 1042fb7e7651Sdrh 1043fb7e7651Sdrh if( objc<=2 ){ 1044fb7e7651Sdrh Tcl_WrongNumArgs(interp, 1, objv, "cache option ?arg?"); 1045fb7e7651Sdrh return TCL_ERROR; 1046fb7e7651Sdrh } 1047fb7e7651Sdrh subCmd = Tcl_GetStringFromObj( objv[2], 0 ); 1048fb7e7651Sdrh if( *subCmd=='f' && strcmp(subCmd,"flush")==0 ){ 1049fb7e7651Sdrh if( objc!=3 ){ 1050fb7e7651Sdrh Tcl_WrongNumArgs(interp, 2, objv, "flush"); 1051fb7e7651Sdrh return TCL_ERROR; 1052fb7e7651Sdrh }else{ 1053fb7e7651Sdrh flushStmtCache( pDb ); 1054fb7e7651Sdrh } 1055fb7e7651Sdrh }else if( *subCmd=='s' && strcmp(subCmd,"size")==0 ){ 1056fb7e7651Sdrh if( objc!=4 ){ 1057fb7e7651Sdrh Tcl_WrongNumArgs(interp, 2, objv, "size n"); 1058fb7e7651Sdrh return TCL_ERROR; 1059fb7e7651Sdrh }else{ 1060fb7e7651Sdrh if( TCL_ERROR==Tcl_GetIntFromObj(interp, objv[3], &n) ){ 1061fb7e7651Sdrh Tcl_AppendResult( interp, "cannot convert \"", 1062fb7e7651Sdrh Tcl_GetStringFromObj(objv[3],0), "\" to integer", 0); 1063fb7e7651Sdrh return TCL_ERROR; 1064fb7e7651Sdrh }else{ 1065fb7e7651Sdrh if( n<0 ){ 1066fb7e7651Sdrh flushStmtCache( pDb ); 1067fb7e7651Sdrh n = 0; 1068fb7e7651Sdrh }else if( n>MAX_PREPARED_STMTS ){ 1069fb7e7651Sdrh n = MAX_PREPARED_STMTS; 1070fb7e7651Sdrh } 1071fb7e7651Sdrh pDb->maxStmt = n; 1072fb7e7651Sdrh } 1073fb7e7651Sdrh } 1074fb7e7651Sdrh }else{ 1075fb7e7651Sdrh Tcl_AppendResult( interp, "bad option \"", 1076fb7e7651Sdrh Tcl_GetStringFromObj(objv[0],0), "\": must be flush or size", 0); 1077fb7e7651Sdrh return TCL_ERROR; 1078fb7e7651Sdrh } 1079fb7e7651Sdrh break; 1080fb7e7651Sdrh } 1081fb7e7651Sdrh 1082b28af71aSdanielk1977 /* $db changes 1083c8d30ac1Sdrh ** 1084c8d30ac1Sdrh ** Return the number of rows that were modified, inserted, or deleted by 1085b28af71aSdanielk1977 ** the most recent INSERT, UPDATE or DELETE statement, not including 1086b28af71aSdanielk1977 ** any changes made by trigger programs. 1087c8d30ac1Sdrh */ 1088c8d30ac1Sdrh case DB_CHANGES: { 1089c8d30ac1Sdrh Tcl_Obj *pResult; 1090c8d30ac1Sdrh if( objc!=2 ){ 1091c8d30ac1Sdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 1092c8d30ac1Sdrh return TCL_ERROR; 1093c8d30ac1Sdrh } 1094c8d30ac1Sdrh pResult = Tcl_GetObjResult(interp); 1095b28af71aSdanielk1977 Tcl_SetIntObj(pResult, sqlite3_changes(pDb->db)); 1096f146a776Srdc break; 1097f146a776Srdc } 1098f146a776Srdc 109975897234Sdrh /* $db close 110075897234Sdrh ** 110175897234Sdrh ** Shutdown the database 110275897234Sdrh */ 11036d31316cSdrh case DB_CLOSE: { 11046d31316cSdrh Tcl_DeleteCommand(interp, Tcl_GetStringFromObj(objv[0], 0)); 11056d31316cSdrh break; 11066d31316cSdrh } 110775897234Sdrh 11080f14e2ebSdrh /* 11090f14e2ebSdrh ** $db collate NAME SCRIPT 11100f14e2ebSdrh ** 11110f14e2ebSdrh ** Create a new SQL collation function called NAME. Whenever 11120f14e2ebSdrh ** that function is called, invoke SCRIPT to evaluate the function. 11130f14e2ebSdrh */ 11140f14e2ebSdrh case DB_COLLATE: { 11150f14e2ebSdrh SqlCollate *pCollate; 11160f14e2ebSdrh char *zName; 11170f14e2ebSdrh char *zScript; 11180f14e2ebSdrh int nScript; 11190f14e2ebSdrh if( objc!=4 ){ 11200f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, "NAME SCRIPT"); 11210f14e2ebSdrh return TCL_ERROR; 11220f14e2ebSdrh } 11230f14e2ebSdrh zName = Tcl_GetStringFromObj(objv[2], 0); 11240f14e2ebSdrh zScript = Tcl_GetStringFromObj(objv[3], &nScript); 11250f14e2ebSdrh pCollate = (SqlCollate*)Tcl_Alloc( sizeof(*pCollate) + nScript + 1 ); 11260f14e2ebSdrh if( pCollate==0 ) return TCL_ERROR; 11270f14e2ebSdrh pCollate->interp = interp; 11280f14e2ebSdrh pCollate->pNext = pDb->pCollate; 11290f14e2ebSdrh pCollate->zScript = (char*)&pCollate[1]; 11300f14e2ebSdrh pDb->pCollate = pCollate; 11315bb3eb9bSdrh memcpy(pCollate->zScript, zScript, nScript+1); 11320f14e2ebSdrh if( sqlite3_create_collation(pDb->db, zName, SQLITE_UTF8, 11330f14e2ebSdrh pCollate, tclSqlCollate) ){ 11349636c4e1Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 11350f14e2ebSdrh return TCL_ERROR; 11360f14e2ebSdrh } 11370f14e2ebSdrh break; 11380f14e2ebSdrh } 11390f14e2ebSdrh 11400f14e2ebSdrh /* 11410f14e2ebSdrh ** $db collation_needed SCRIPT 11420f14e2ebSdrh ** 11430f14e2ebSdrh ** Create a new SQL collation function called NAME. Whenever 11440f14e2ebSdrh ** that function is called, invoke SCRIPT to evaluate the function. 11450f14e2ebSdrh */ 11460f14e2ebSdrh case DB_COLLATION_NEEDED: { 11470f14e2ebSdrh if( objc!=3 ){ 11480f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, "SCRIPT"); 11490f14e2ebSdrh return TCL_ERROR; 11500f14e2ebSdrh } 11510f14e2ebSdrh if( pDb->pCollateNeeded ){ 11520f14e2ebSdrh Tcl_DecrRefCount(pDb->pCollateNeeded); 11530f14e2ebSdrh } 11540f14e2ebSdrh pDb->pCollateNeeded = Tcl_DuplicateObj(objv[2]); 11550f14e2ebSdrh Tcl_IncrRefCount(pDb->pCollateNeeded); 11560f14e2ebSdrh sqlite3_collation_needed(pDb->db, pDb, tclCollateNeeded); 11570f14e2ebSdrh break; 11580f14e2ebSdrh } 11590f14e2ebSdrh 116019e2d37fSdrh /* $db commit_hook ?CALLBACK? 116119e2d37fSdrh ** 116219e2d37fSdrh ** Invoke the given callback just before committing every SQL transaction. 116319e2d37fSdrh ** If the callback throws an exception or returns non-zero, then the 116419e2d37fSdrh ** transaction is aborted. If CALLBACK is an empty string, the callback 116519e2d37fSdrh ** is disabled. 116619e2d37fSdrh */ 116719e2d37fSdrh case DB_COMMIT_HOOK: { 116819e2d37fSdrh if( objc>3 ){ 116919e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 117019e2d37fSdrh return TCL_ERROR; 117119e2d37fSdrh }else if( objc==2 ){ 117219e2d37fSdrh if( pDb->zCommit ){ 117319e2d37fSdrh Tcl_AppendResult(interp, pDb->zCommit, 0); 117419e2d37fSdrh } 117519e2d37fSdrh }else{ 117619e2d37fSdrh char *zCommit; 117719e2d37fSdrh int len; 117819e2d37fSdrh if( pDb->zCommit ){ 117919e2d37fSdrh Tcl_Free(pDb->zCommit); 118019e2d37fSdrh } 118119e2d37fSdrh zCommit = Tcl_GetStringFromObj(objv[2], &len); 118219e2d37fSdrh if( zCommit && len>0 ){ 118319e2d37fSdrh pDb->zCommit = Tcl_Alloc( len + 1 ); 11845bb3eb9bSdrh memcpy(pDb->zCommit, zCommit, len+1); 118519e2d37fSdrh }else{ 118619e2d37fSdrh pDb->zCommit = 0; 118719e2d37fSdrh } 118819e2d37fSdrh if( pDb->zCommit ){ 118919e2d37fSdrh pDb->interp = interp; 119019e2d37fSdrh sqlite3_commit_hook(pDb->db, DbCommitHandler, pDb); 119119e2d37fSdrh }else{ 119219e2d37fSdrh sqlite3_commit_hook(pDb->db, 0, 0); 119319e2d37fSdrh } 119419e2d37fSdrh } 119519e2d37fSdrh break; 119619e2d37fSdrh } 119719e2d37fSdrh 119875897234Sdrh /* $db complete SQL 119975897234Sdrh ** 120075897234Sdrh ** Return TRUE if SQL is a complete SQL statement. Return FALSE if 120175897234Sdrh ** additional lines of input are needed. This is similar to the 120275897234Sdrh ** built-in "info complete" command of Tcl. 120375897234Sdrh */ 12046d31316cSdrh case DB_COMPLETE: { 1205ccae6026Sdrh #ifndef SQLITE_OMIT_COMPLETE 12066d31316cSdrh Tcl_Obj *pResult; 12076d31316cSdrh int isComplete; 12086d31316cSdrh if( objc!=3 ){ 12096d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL"); 121075897234Sdrh return TCL_ERROR; 121175897234Sdrh } 12126f8a503dSdanielk1977 isComplete = sqlite3_complete( Tcl_GetStringFromObj(objv[2], 0) ); 12136d31316cSdrh pResult = Tcl_GetObjResult(interp); 12146d31316cSdrh Tcl_SetBooleanObj(pResult, isComplete); 1215ccae6026Sdrh #endif 12166d31316cSdrh break; 12176d31316cSdrh } 121875897234Sdrh 121919e2d37fSdrh /* $db copy conflict-algorithm table filename ?SEPARATOR? ?NULLINDICATOR? 122019e2d37fSdrh ** 122119e2d37fSdrh ** Copy data into table from filename, optionally using SEPARATOR 122219e2d37fSdrh ** as column separators. If a column contains a null string, or the 122319e2d37fSdrh ** value of NULLINDICATOR, a NULL is inserted for the column. 122419e2d37fSdrh ** conflict-algorithm is one of the sqlite conflict algorithms: 122519e2d37fSdrh ** rollback, abort, fail, ignore, replace 122619e2d37fSdrh ** On success, return the number of lines processed, not necessarily same 122719e2d37fSdrh ** as 'db changes' due to conflict-algorithm selected. 122819e2d37fSdrh ** 122919e2d37fSdrh ** This code is basically an implementation/enhancement of 123019e2d37fSdrh ** the sqlite3 shell.c ".import" command. 123119e2d37fSdrh ** 123219e2d37fSdrh ** This command usage is equivalent to the sqlite2.x COPY statement, 123319e2d37fSdrh ** which imports file data into a table using the PostgreSQL COPY file format: 123419e2d37fSdrh ** $db copy $conflit_algo $table_name $filename \t \\N 123519e2d37fSdrh */ 123619e2d37fSdrh case DB_COPY: { 123719e2d37fSdrh char *zTable; /* Insert data into this table */ 123819e2d37fSdrh char *zFile; /* The file from which to extract data */ 123919e2d37fSdrh char *zConflict; /* The conflict algorithm to use */ 124019e2d37fSdrh sqlite3_stmt *pStmt; /* A statement */ 124119e2d37fSdrh int rc; /* Result code */ 124219e2d37fSdrh int nCol; /* Number of columns in the table */ 124319e2d37fSdrh int nByte; /* Number of bytes in an SQL string */ 124419e2d37fSdrh int i, j; /* Loop counters */ 124519e2d37fSdrh int nSep; /* Number of bytes in zSep[] */ 124619e2d37fSdrh int nNull; /* Number of bytes in zNull[] */ 124719e2d37fSdrh char *zSql; /* An SQL statement */ 124819e2d37fSdrh char *zLine; /* A single line of input from the file */ 124919e2d37fSdrh char **azCol; /* zLine[] broken up into columns */ 125019e2d37fSdrh char *zCommit; /* How to commit changes */ 125119e2d37fSdrh FILE *in; /* The input file */ 125219e2d37fSdrh int lineno = 0; /* Line number of input file */ 125319e2d37fSdrh char zLineNum[80]; /* Line number print buffer */ 125419e2d37fSdrh Tcl_Obj *pResult; /* interp result */ 125519e2d37fSdrh 125619e2d37fSdrh char *zSep; 125719e2d37fSdrh char *zNull; 125819e2d37fSdrh if( objc<5 || objc>7 ){ 125919e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, 126019e2d37fSdrh "CONFLICT-ALGORITHM TABLE FILENAME ?SEPARATOR? ?NULLINDICATOR?"); 126119e2d37fSdrh return TCL_ERROR; 126219e2d37fSdrh } 126319e2d37fSdrh if( objc>=6 ){ 126419e2d37fSdrh zSep = Tcl_GetStringFromObj(objv[5], 0); 126519e2d37fSdrh }else{ 126619e2d37fSdrh zSep = "\t"; 126719e2d37fSdrh } 126819e2d37fSdrh if( objc>=7 ){ 126919e2d37fSdrh zNull = Tcl_GetStringFromObj(objv[6], 0); 127019e2d37fSdrh }else{ 127119e2d37fSdrh zNull = ""; 127219e2d37fSdrh } 127319e2d37fSdrh zConflict = Tcl_GetStringFromObj(objv[2], 0); 127419e2d37fSdrh zTable = Tcl_GetStringFromObj(objv[3], 0); 127519e2d37fSdrh zFile = Tcl_GetStringFromObj(objv[4], 0); 127619e2d37fSdrh nSep = strlen(zSep); 127719e2d37fSdrh nNull = strlen(zNull); 127819e2d37fSdrh if( nSep==0 ){ 127919e2d37fSdrh Tcl_AppendResult(interp,"Error: non-null separator required for copy",0); 128019e2d37fSdrh return TCL_ERROR; 128119e2d37fSdrh } 128219e2d37fSdrh if(sqlite3StrICmp(zConflict, "rollback") != 0 && 128319e2d37fSdrh sqlite3StrICmp(zConflict, "abort" ) != 0 && 128419e2d37fSdrh sqlite3StrICmp(zConflict, "fail" ) != 0 && 128519e2d37fSdrh sqlite3StrICmp(zConflict, "ignore" ) != 0 && 128619e2d37fSdrh sqlite3StrICmp(zConflict, "replace" ) != 0 ) { 128719e2d37fSdrh Tcl_AppendResult(interp, "Error: \"", zConflict, 128819e2d37fSdrh "\", conflict-algorithm must be one of: rollback, " 128919e2d37fSdrh "abort, fail, ignore, or replace", 0); 129019e2d37fSdrh return TCL_ERROR; 129119e2d37fSdrh } 129219e2d37fSdrh zSql = sqlite3_mprintf("SELECT * FROM '%q'", zTable); 129319e2d37fSdrh if( zSql==0 ){ 129419e2d37fSdrh Tcl_AppendResult(interp, "Error: no such table: ", zTable, 0); 129519e2d37fSdrh return TCL_ERROR; 129619e2d37fSdrh } 129719e2d37fSdrh nByte = strlen(zSql); 12983e701a18Sdrh rc = sqlite3_prepare(pDb->db, zSql, -1, &pStmt, 0); 129919e2d37fSdrh sqlite3_free(zSql); 130019e2d37fSdrh if( rc ){ 130119e2d37fSdrh Tcl_AppendResult(interp, "Error: ", sqlite3_errmsg(pDb->db), 0); 130219e2d37fSdrh nCol = 0; 130319e2d37fSdrh }else{ 130419e2d37fSdrh nCol = sqlite3_column_count(pStmt); 130519e2d37fSdrh } 130619e2d37fSdrh sqlite3_finalize(pStmt); 130719e2d37fSdrh if( nCol==0 ) { 130819e2d37fSdrh return TCL_ERROR; 130919e2d37fSdrh } 131019e2d37fSdrh zSql = malloc( nByte + 50 + nCol*2 ); 131119e2d37fSdrh if( zSql==0 ) { 131219e2d37fSdrh Tcl_AppendResult(interp, "Error: can't malloc()", 0); 131319e2d37fSdrh return TCL_ERROR; 131419e2d37fSdrh } 131519e2d37fSdrh sqlite3_snprintf(nByte+50, zSql, "INSERT OR %q INTO '%q' VALUES(?", 131619e2d37fSdrh zConflict, zTable); 131719e2d37fSdrh j = strlen(zSql); 131819e2d37fSdrh for(i=1; i<nCol; i++){ 131919e2d37fSdrh zSql[j++] = ','; 132019e2d37fSdrh zSql[j++] = '?'; 132119e2d37fSdrh } 132219e2d37fSdrh zSql[j++] = ')'; 132319e2d37fSdrh zSql[j] = 0; 13243e701a18Sdrh rc = sqlite3_prepare(pDb->db, zSql, -1, &pStmt, 0); 132519e2d37fSdrh free(zSql); 132619e2d37fSdrh if( rc ){ 132719e2d37fSdrh Tcl_AppendResult(interp, "Error: ", sqlite3_errmsg(pDb->db), 0); 132819e2d37fSdrh sqlite3_finalize(pStmt); 132919e2d37fSdrh return TCL_ERROR; 133019e2d37fSdrh } 133119e2d37fSdrh in = fopen(zFile, "rb"); 133219e2d37fSdrh if( in==0 ){ 133319e2d37fSdrh Tcl_AppendResult(interp, "Error: cannot open file: ", zFile, NULL); 133419e2d37fSdrh sqlite3_finalize(pStmt); 133519e2d37fSdrh return TCL_ERROR; 133619e2d37fSdrh } 133719e2d37fSdrh azCol = malloc( sizeof(azCol[0])*(nCol+1) ); 133819e2d37fSdrh if( azCol==0 ) { 133919e2d37fSdrh Tcl_AppendResult(interp, "Error: can't malloc()", 0); 134043617e9aSdrh fclose(in); 134119e2d37fSdrh return TCL_ERROR; 134219e2d37fSdrh } 13433752785fSdrh (void)sqlite3_exec(pDb->db, "BEGIN", 0, 0, 0); 134419e2d37fSdrh zCommit = "COMMIT"; 134519e2d37fSdrh while( (zLine = local_getline(0, in))!=0 ){ 134619e2d37fSdrh char *z; 134719e2d37fSdrh i = 0; 134819e2d37fSdrh lineno++; 134919e2d37fSdrh azCol[0] = zLine; 135019e2d37fSdrh for(i=0, z=zLine; *z; z++){ 135119e2d37fSdrh if( *z==zSep[0] && strncmp(z, zSep, nSep)==0 ){ 135219e2d37fSdrh *z = 0; 135319e2d37fSdrh i++; 135419e2d37fSdrh if( i<nCol ){ 135519e2d37fSdrh azCol[i] = &z[nSep]; 135619e2d37fSdrh z += nSep-1; 135719e2d37fSdrh } 135819e2d37fSdrh } 135919e2d37fSdrh } 136019e2d37fSdrh if( i+1!=nCol ){ 136119e2d37fSdrh char *zErr; 13625bb3eb9bSdrh int nErr = strlen(zFile) + 200; 13635bb3eb9bSdrh zErr = malloc(nErr); 1364c1f4494eSdrh if( zErr ){ 13655bb3eb9bSdrh sqlite3_snprintf(nErr, zErr, 1366955de52cSdanielk1977 "Error: %s line %d: expected %d columns of data but found %d", 136719e2d37fSdrh zFile, lineno, nCol, i+1); 136819e2d37fSdrh Tcl_AppendResult(interp, zErr, 0); 136919e2d37fSdrh free(zErr); 1370c1f4494eSdrh } 137119e2d37fSdrh zCommit = "ROLLBACK"; 137219e2d37fSdrh break; 137319e2d37fSdrh } 137419e2d37fSdrh for(i=0; i<nCol; i++){ 137519e2d37fSdrh /* check for null data, if so, bind as null */ 137619e2d37fSdrh if ((nNull>0 && strcmp(azCol[i], zNull)==0) || strlen(azCol[i])==0) { 137719e2d37fSdrh sqlite3_bind_null(pStmt, i+1); 137819e2d37fSdrh }else{ 137919e2d37fSdrh sqlite3_bind_text(pStmt, i+1, azCol[i], -1, SQLITE_STATIC); 138019e2d37fSdrh } 138119e2d37fSdrh } 138219e2d37fSdrh sqlite3_step(pStmt); 138319e2d37fSdrh rc = sqlite3_reset(pStmt); 138419e2d37fSdrh free(zLine); 138519e2d37fSdrh if( rc!=SQLITE_OK ){ 138619e2d37fSdrh Tcl_AppendResult(interp,"Error: ", sqlite3_errmsg(pDb->db), 0); 138719e2d37fSdrh zCommit = "ROLLBACK"; 138819e2d37fSdrh break; 138919e2d37fSdrh } 139019e2d37fSdrh } 139119e2d37fSdrh free(azCol); 139219e2d37fSdrh fclose(in); 139319e2d37fSdrh sqlite3_finalize(pStmt); 13943752785fSdrh (void)sqlite3_exec(pDb->db, zCommit, 0, 0, 0); 139519e2d37fSdrh 139619e2d37fSdrh if( zCommit[0] == 'C' ){ 139719e2d37fSdrh /* success, set result as number of lines processed */ 139819e2d37fSdrh pResult = Tcl_GetObjResult(interp); 139919e2d37fSdrh Tcl_SetIntObj(pResult, lineno); 140019e2d37fSdrh rc = TCL_OK; 140119e2d37fSdrh }else{ 140219e2d37fSdrh /* failure, append lineno where failed */ 14035bb3eb9bSdrh sqlite3_snprintf(sizeof(zLineNum), zLineNum,"%d",lineno); 140419e2d37fSdrh Tcl_AppendResult(interp,", failed while processing line: ",zLineNum,0); 140519e2d37fSdrh rc = TCL_ERROR; 140619e2d37fSdrh } 140719e2d37fSdrh break; 140819e2d37fSdrh } 140919e2d37fSdrh 141075897234Sdrh /* 14114144905bSdrh ** $db enable_load_extension BOOLEAN 14124144905bSdrh ** 14134144905bSdrh ** Turn the extension loading feature on or off. It if off by 14144144905bSdrh ** default. 14154144905bSdrh */ 14164144905bSdrh case DB_ENABLE_LOAD_EXTENSION: { 1417f533acc0Sdrh #ifndef SQLITE_OMIT_LOAD_EXTENSION 14184144905bSdrh int onoff; 14194144905bSdrh if( objc!=3 ){ 14204144905bSdrh Tcl_WrongNumArgs(interp, 2, objv, "BOOLEAN"); 14214144905bSdrh return TCL_ERROR; 14224144905bSdrh } 14234144905bSdrh if( Tcl_GetBooleanFromObj(interp, objv[2], &onoff) ){ 14244144905bSdrh return TCL_ERROR; 14254144905bSdrh } 14264144905bSdrh sqlite3_enable_load_extension(pDb->db, onoff); 14274144905bSdrh break; 1428f533acc0Sdrh #else 1429f533acc0Sdrh Tcl_AppendResult(interp, "extension loading is turned off at compile-time", 1430f533acc0Sdrh 0); 1431f533acc0Sdrh return TCL_ERROR; 1432f533acc0Sdrh #endif 14334144905bSdrh } 14344144905bSdrh 14354144905bSdrh /* 1436dcd997eaSdrh ** $db errorcode 1437dcd997eaSdrh ** 1438dcd997eaSdrh ** Return the numeric error code that was returned by the most recent 14396f8a503dSdanielk1977 ** call to sqlite3_exec(). 1440dcd997eaSdrh */ 1441dcd997eaSdrh case DB_ERRORCODE: { 1442f3ce83f5Sdanielk1977 Tcl_SetObjResult(interp, Tcl_NewIntObj(sqlite3_errcode(pDb->db))); 1443dcd997eaSdrh break; 1444dcd997eaSdrh } 1445dcd997eaSdrh 1446dcd997eaSdrh /* 1447895d7472Sdrh ** $db eval $sql ?array? ?{ ...code... }? 14481807ce37Sdrh ** $db onecolumn $sql 144975897234Sdrh ** 145075897234Sdrh ** The SQL statement in $sql is evaluated. For each row, the values are 1451bec3f402Sdrh ** placed in elements of the array named "array" and ...code... is executed. 145275897234Sdrh ** If "array" and "code" are omitted, then no callback is every invoked. 145375897234Sdrh ** If "array" is an empty string, then the values are placed in variables 145475897234Sdrh ** that have the same name as the fields extracted by the query. 14551807ce37Sdrh ** 14561807ce37Sdrh ** The onecolumn method is the equivalent of: 14571807ce37Sdrh ** lindex [$db eval $sql] 0 145875897234Sdrh */ 14591807ce37Sdrh case DB_ONECOLUMN: 146097f2ebc1Sdrh case DB_EVAL: 146197f2ebc1Sdrh case DB_EXISTS: { 14629d74b4c5Sdrh char const *zSql; /* Next SQL statement to execute */ 14639d74b4c5Sdrh char const *zLeft; /* What is left after first stmt in zSql */ 14649d74b4c5Sdrh sqlite3_stmt *pStmt; /* Compiled SQL statment */ 146592febd92Sdrh Tcl_Obj *pArray; /* Name of array into which results are written */ 146692febd92Sdrh Tcl_Obj *pScript; /* Script to run for each result set */ 14671d895039Sdrh Tcl_Obj **apParm; /* Parameters that need a Tcl_DecrRefCount() */ 14681d895039Sdrh int nParm; /* Number of entries used in apParm[] */ 14691d895039Sdrh Tcl_Obj *aParm[10]; /* Static space for apParm[] in the common case */ 14701807ce37Sdrh Tcl_Obj *pRet; /* Value to be returned */ 1471fb7e7651Sdrh SqlPreparedStmt *pPreStmt; /* Pointer to a prepared statement */ 1472fb7e7651Sdrh int rc2; 1473ef2cb63eSdanielk1977 147497f2ebc1Sdrh if( choice==DB_EVAL ){ 147592febd92Sdrh if( objc<3 || objc>5 ){ 1476895d7472Sdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL ?ARRAY-NAME? ?SCRIPT?"); 147730ccda10Sdanielk1977 return TCL_ERROR; 147830ccda10Sdanielk1977 } 14791807ce37Sdrh pRet = Tcl_NewObj(); 14801807ce37Sdrh Tcl_IncrRefCount(pRet); 148197f2ebc1Sdrh }else{ 148297f2ebc1Sdrh if( objc!=3 ){ 148397f2ebc1Sdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL"); 148497f2ebc1Sdrh return TCL_ERROR; 148597f2ebc1Sdrh } 148697f2ebc1Sdrh if( choice==DB_EXISTS ){ 1487446a9b82Sdrh pRet = Tcl_NewBooleanObj(0); 1488446a9b82Sdrh Tcl_IncrRefCount(pRet); 1489446a9b82Sdrh }else{ 1490446a9b82Sdrh pRet = 0; 149197f2ebc1Sdrh } 14921807ce37Sdrh } 149392febd92Sdrh if( objc==3 ){ 149492febd92Sdrh pArray = pScript = 0; 149592febd92Sdrh }else if( objc==4 ){ 149692febd92Sdrh pArray = 0; 149792febd92Sdrh pScript = objv[3]; 149892febd92Sdrh }else{ 149992febd92Sdrh pArray = objv[3]; 150092febd92Sdrh if( Tcl_GetString(pArray)[0]==0 ) pArray = 0; 150192febd92Sdrh pScript = objv[4]; 150292febd92Sdrh } 150330ccda10Sdanielk1977 15041d895039Sdrh Tcl_IncrRefCount(objv[2]); 150530ccda10Sdanielk1977 zSql = Tcl_GetStringFromObj(objv[2], 0); 150690b6bb19Sdrh while( rc==TCL_OK && zSql[0] ){ 150792febd92Sdrh int i; /* Loop counter */ 1508fb7e7651Sdrh int nVar; /* Number of bind parameters in the pStmt */ 150992febd92Sdrh int nCol; /* Number of columns in the result set */ 151092febd92Sdrh Tcl_Obj **apColName = 0; /* Array of column names */ 1511fb7e7651Sdrh int len; /* String length of zSql */ 151230ccda10Sdanielk1977 1513fb7e7651Sdrh /* Try to find a SQL statement that has already been compiled and 1514fb7e7651Sdrh ** which matches the next sequence of SQL. 1515fb7e7651Sdrh */ 1516fb7e7651Sdrh pStmt = 0; 1517fb7e7651Sdrh pPreStmt = pDb->stmtList; 1518fb7e7651Sdrh len = strlen(zSql); 1519fb7e7651Sdrh if( pPreStmt && sqlite3_expired(pPreStmt->pStmt) ){ 1520fb7e7651Sdrh flushStmtCache(pDb); 1521fb7e7651Sdrh pPreStmt = 0; 1522fb7e7651Sdrh } 1523fb7e7651Sdrh for(; pPreStmt; pPreStmt=pPreStmt->pNext){ 1524fb7e7651Sdrh int n = pPreStmt->nSql; 1525fb7e7651Sdrh if( len>=n 1526fb7e7651Sdrh && memcmp(pPreStmt->zSql, zSql, n)==0 1527fb7e7651Sdrh && (zSql[n]==0 || zSql[n-1]==';') 1528fb7e7651Sdrh ){ 1529fb7e7651Sdrh pStmt = pPreStmt->pStmt; 1530fb7e7651Sdrh zLeft = &zSql[pPreStmt->nSql]; 1531fb7e7651Sdrh 1532fb7e7651Sdrh /* When a prepared statement is found, unlink it from the 1533fb7e7651Sdrh ** cache list. It will later be added back to the beginning 1534fb7e7651Sdrh ** of the cache list in order to implement LRU replacement. 1535fb7e7651Sdrh */ 1536fb7e7651Sdrh if( pPreStmt->pPrev ){ 1537fb7e7651Sdrh pPreStmt->pPrev->pNext = pPreStmt->pNext; 1538fb7e7651Sdrh }else{ 1539fb7e7651Sdrh pDb->stmtList = pPreStmt->pNext; 1540fb7e7651Sdrh } 1541fb7e7651Sdrh if( pPreStmt->pNext ){ 1542fb7e7651Sdrh pPreStmt->pNext->pPrev = pPreStmt->pPrev; 1543fb7e7651Sdrh }else{ 1544fb7e7651Sdrh pDb->stmtLast = pPreStmt->pPrev; 1545fb7e7651Sdrh } 1546fb7e7651Sdrh pDb->nStmt--; 1547fb7e7651Sdrh break; 1548fb7e7651Sdrh } 1549fb7e7651Sdrh } 1550fb7e7651Sdrh 1551fb7e7651Sdrh /* If no prepared statement was found. Compile the SQL text 1552fb7e7651Sdrh */ 1553fb7e7651Sdrh if( pStmt==0 ){ 155430ccda10Sdanielk1977 if( SQLITE_OK!=sqlite3_prepare(pDb->db, zSql, -1, &pStmt, &zLeft) ){ 1555ef2cb63eSdanielk1977 Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 155630ccda10Sdanielk1977 rc = TCL_ERROR; 155730ccda10Sdanielk1977 break; 155830ccda10Sdanielk1977 } 155992febd92Sdrh if( pStmt==0 ){ 156092febd92Sdrh if( SQLITE_OK!=sqlite3_errcode(pDb->db) ){ 1561fb7e7651Sdrh /* A compile-time error in the statement 1562fb7e7651Sdrh */ 156392febd92Sdrh Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 156492febd92Sdrh rc = TCL_ERROR; 156592febd92Sdrh break; 156692febd92Sdrh }else{ 1567fb7e7651Sdrh /* The statement was a no-op. Continue to the next statement 1568fb7e7651Sdrh ** in the SQL string. 1569fb7e7651Sdrh */ 157092febd92Sdrh zSql = zLeft; 157192febd92Sdrh continue; 157292febd92Sdrh } 157392febd92Sdrh } 1574fb7e7651Sdrh assert( pPreStmt==0 ); 1575fb7e7651Sdrh } 157630ccda10Sdanielk1977 1577fb7e7651Sdrh /* Bind values to parameters that begin with $ or : 1578fb7e7651Sdrh */ 157992febd92Sdrh nVar = sqlite3_bind_parameter_count(pStmt); 15801d895039Sdrh nParm = 0; 15811d895039Sdrh if( nVar>sizeof(aParm)/sizeof(aParm[0]) ){ 15821d895039Sdrh apParm = (Tcl_Obj**)Tcl_Alloc(nVar*sizeof(apParm[0])); 15831d895039Sdrh }else{ 15841d895039Sdrh apParm = aParm; 15851d895039Sdrh } 158692febd92Sdrh for(i=1; i<=nVar; i++){ 158792febd92Sdrh const char *zVar = sqlite3_bind_parameter_name(pStmt, i); 15884f5e80f9Sdrh if( zVar!=0 && (zVar[0]=='$' || zVar[0]==':' || zVar[0]=='@') ){ 158992febd92Sdrh Tcl_Obj *pVar = Tcl_GetVar2Ex(interp, &zVar[1], 0, 0); 159092febd92Sdrh if( pVar ){ 159192febd92Sdrh int n; 159292febd92Sdrh u8 *data; 159392febd92Sdrh char *zType = pVar->typePtr ? pVar->typePtr->name : ""; 159492febd92Sdrh char c = zType[0]; 15951c747819Sdrh if( zVar[0]=='@' || 15961c747819Sdrh (c=='b' && strcmp(zType,"bytearray")==0 && pVar->bytes==0) ){ 15971c747819Sdrh /* Load a BLOB type if the Tcl variable is a bytearray and 15981c747819Sdrh ** it has no string representation or the host 15994f5e80f9Sdrh ** parameter name begins with "@". */ 160092febd92Sdrh data = Tcl_GetByteArrayFromObj(pVar, &n); 160192febd92Sdrh sqlite3_bind_blob(pStmt, i, data, n, SQLITE_STATIC); 16021d895039Sdrh Tcl_IncrRefCount(pVar); 16031d895039Sdrh apParm[nParm++] = pVar; 1604985e0c63Sdrh }else if( c=='b' && strcmp(zType,"boolean")==0 ){ 160592febd92Sdrh Tcl_GetIntFromObj(interp, pVar, &n); 160692febd92Sdrh sqlite3_bind_int(pStmt, i, n); 160792febd92Sdrh }else if( c=='d' && strcmp(zType,"double")==0 ){ 160892febd92Sdrh double r; 160992febd92Sdrh Tcl_GetDoubleFromObj(interp, pVar, &r); 161092febd92Sdrh sqlite3_bind_double(pStmt, i, r); 1611985e0c63Sdrh }else if( (c=='w' && strcmp(zType,"wideInt")==0) || 1612985e0c63Sdrh (c=='i' && strcmp(zType,"int")==0) ){ 1613df0bddaeSdrh Tcl_WideInt v; 1614df0bddaeSdrh Tcl_GetWideIntFromObj(interp, pVar, &v); 1615df0bddaeSdrh sqlite3_bind_int64(pStmt, i, v); 161692febd92Sdrh }else{ 161700fd957bSdanielk1977 data = (unsigned char *)Tcl_GetStringFromObj(pVar, &n); 161800fd957bSdanielk1977 sqlite3_bind_text(pStmt, i, (char *)data, n, SQLITE_STATIC); 16191d895039Sdrh Tcl_IncrRefCount(pVar); 16201d895039Sdrh apParm[nParm++] = pVar; 162192febd92Sdrh } 1622fb7e7651Sdrh }else{ 1623fb7e7651Sdrh sqlite3_bind_null( pStmt, i ); 162492febd92Sdrh } 162592febd92Sdrh } 162692febd92Sdrh } 162792febd92Sdrh 162892febd92Sdrh /* Compute column names */ 162992febd92Sdrh nCol = sqlite3_column_count(pStmt); 163092febd92Sdrh if( pScript ){ 163192febd92Sdrh apColName = (Tcl_Obj**)Tcl_Alloc( sizeof(Tcl_Obj*)*nCol ); 163292febd92Sdrh if( apColName==0 ) break; 163392febd92Sdrh for(i=0; i<nCol; i++){ 163492febd92Sdrh apColName[i] = dbTextToObj(sqlite3_column_name(pStmt,i)); 163592febd92Sdrh Tcl_IncrRefCount(apColName[i]); 163692febd92Sdrh } 163792febd92Sdrh } 163892febd92Sdrh 163992febd92Sdrh /* If results are being stored in an array variable, then create 164092febd92Sdrh ** the array(*) entry for that array 164192febd92Sdrh */ 164292febd92Sdrh if( pArray ){ 164330ccda10Sdanielk1977 Tcl_Obj *pColList = Tcl_NewObj(); 16443ced14a6Sdrh Tcl_Obj *pStar = Tcl_NewStringObj("*", -1); 164530ccda10Sdanielk1977 Tcl_IncrRefCount(pColList); 164692febd92Sdrh for(i=0; i<nCol; i++){ 164792febd92Sdrh Tcl_ListObjAppendElement(interp, pColList, apColName[i]); 164830ccda10Sdanielk1977 } 16493ced14a6Sdrh Tcl_ObjSetVar2(interp, pArray, pStar, pColList,0); 16503ced14a6Sdrh Tcl_DecrRefCount(pColList); 165130ccda10Sdanielk1977 } 165230ccda10Sdanielk1977 165392febd92Sdrh /* Execute the SQL 165492febd92Sdrh */ 165590b6bb19Sdrh while( rc==TCL_OK && pStmt && SQLITE_ROW==sqlite3_step(pStmt) ){ 165692febd92Sdrh for(i=0; i<nCol; i++){ 165730ccda10Sdanielk1977 Tcl_Obj *pVal; 165830ccda10Sdanielk1977 165930ccda10Sdanielk1977 /* Set pVal to contain the i'th column of this row. */ 166092febd92Sdrh switch( sqlite3_column_type(pStmt, i) ){ 166192febd92Sdrh case SQLITE_BLOB: { 16623fd0a736Sdanielk1977 int bytes = sqlite3_column_bytes(pStmt, i); 16633fd0a736Sdanielk1977 pVal = Tcl_NewByteArrayObj(sqlite3_column_blob(pStmt, i), bytes); 166492febd92Sdrh break; 166592febd92Sdrh } 166692febd92Sdrh case SQLITE_INTEGER: { 166792febd92Sdrh sqlite_int64 v = sqlite3_column_int64(pStmt, i); 166892febd92Sdrh if( v>=-2147483647 && v<=2147483647 ){ 166992febd92Sdrh pVal = Tcl_NewIntObj(v); 167092febd92Sdrh }else{ 167192febd92Sdrh pVal = Tcl_NewWideIntObj(v); 167292febd92Sdrh } 167392febd92Sdrh break; 167492febd92Sdrh } 167592febd92Sdrh case SQLITE_FLOAT: { 167692febd92Sdrh double r = sqlite3_column_double(pStmt, i); 167792febd92Sdrh pVal = Tcl_NewDoubleObj(r); 167892febd92Sdrh break; 167992febd92Sdrh } 168055c45f2eSdanielk1977 case SQLITE_NULL: { 168155c45f2eSdanielk1977 pVal = dbTextToObj(pDb->zNull); 168255c45f2eSdanielk1977 break; 168355c45f2eSdanielk1977 } 168492febd92Sdrh default: { 168500fd957bSdanielk1977 pVal = dbTextToObj((char *)sqlite3_column_text(pStmt, i)); 168692febd92Sdrh break; 168792febd92Sdrh } 168830ccda10Sdanielk1977 } 168930ccda10Sdanielk1977 169092febd92Sdrh if( pScript ){ 169192febd92Sdrh if( pArray==0 ){ 169292febd92Sdrh Tcl_ObjSetVar2(interp, apColName[i], 0, pVal, 0); 169330ccda10Sdanielk1977 }else{ 169492febd92Sdrh Tcl_ObjSetVar2(interp, pArray, apColName[i], pVal, 0); 169530ccda10Sdanielk1977 } 16961807ce37Sdrh }else if( choice==DB_ONECOLUMN ){ 169797f2ebc1Sdrh assert( pRet==0 ); 16981807ce37Sdrh if( pRet==0 ){ 16991807ce37Sdrh pRet = pVal; 17001807ce37Sdrh Tcl_IncrRefCount(pRet); 17011807ce37Sdrh } 170290b6bb19Sdrh rc = TCL_BREAK; 170397f2ebc1Sdrh i = nCol; 170497f2ebc1Sdrh }else if( choice==DB_EXISTS ){ 1705446a9b82Sdrh Tcl_DecrRefCount(pRet); 1706446a9b82Sdrh pRet = Tcl_NewBooleanObj(1); 1707446a9b82Sdrh Tcl_IncrRefCount(pRet); 170897f2ebc1Sdrh rc = TCL_BREAK; 170997f2ebc1Sdrh i = nCol; 171030ccda10Sdanielk1977 }else{ 171130ccda10Sdanielk1977 Tcl_ListObjAppendElement(interp, pRet, pVal); 171230ccda10Sdanielk1977 } 171330ccda10Sdanielk1977 } 171430ccda10Sdanielk1977 171592febd92Sdrh if( pScript ){ 171692febd92Sdrh rc = Tcl_EvalObjEx(interp, pScript, 0); 171790b6bb19Sdrh if( rc==TCL_CONTINUE ){ 171890b6bb19Sdrh rc = TCL_OK; 171930ccda10Sdanielk1977 } 172030ccda10Sdanielk1977 } 172190b6bb19Sdrh } 172290b6bb19Sdrh if( rc==TCL_BREAK ){ 172390b6bb19Sdrh rc = TCL_OK; 172490b6bb19Sdrh } 172530ccda10Sdanielk1977 172692febd92Sdrh /* Free the column name objects */ 172792febd92Sdrh if( pScript ){ 172892febd92Sdrh for(i=0; i<nCol; i++){ 172992febd92Sdrh Tcl_DecrRefCount(apColName[i]); 173092febd92Sdrh } 173192febd92Sdrh Tcl_Free((char*)apColName); 173292febd92Sdrh } 173392febd92Sdrh 17341d895039Sdrh /* Free the bound string and blob parameters */ 17351d895039Sdrh for(i=0; i<nParm; i++){ 17361d895039Sdrh Tcl_DecrRefCount(apParm[i]); 17371d895039Sdrh } 17381d895039Sdrh if( apParm!=aParm ){ 17391d895039Sdrh Tcl_Free((char*)apParm); 17401d895039Sdrh } 17411d895039Sdrh 1742fb7e7651Sdrh /* Reset the statement. If the result code is SQLITE_SCHEMA, then 1743fb7e7651Sdrh ** flush the statement cache and try the statement again. 174492febd92Sdrh */ 1745fb7e7651Sdrh rc2 = sqlite3_reset(pStmt); 1746fb7e7651Sdrh if( SQLITE_SCHEMA==rc2 ){ 1747fb7e7651Sdrh /* After a schema change, flush the cache and try to run the 1748fb7e7651Sdrh ** statement again 1749fb7e7651Sdrh */ 1750fb7e7651Sdrh flushStmtCache( pDb ); 1751fb7e7651Sdrh sqlite3_finalize(pStmt); 1752fb7e7651Sdrh if( pPreStmt ) Tcl_Free((char*)pPreStmt); 175330ccda10Sdanielk1977 continue; 1754fb7e7651Sdrh }else if( SQLITE_OK!=rc2 ){ 1755fb7e7651Sdrh /* If a run-time error occurs, report the error and stop reading 1756fb7e7651Sdrh ** the SQL 1757fb7e7651Sdrh */ 1758ef2cb63eSdanielk1977 Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 1759fb7e7651Sdrh sqlite3_finalize(pStmt); 176030ccda10Sdanielk1977 rc = TCL_ERROR; 1761fb7e7651Sdrh if( pPreStmt ) Tcl_Free((char*)pPreStmt); 176230ccda10Sdanielk1977 break; 1763fb7e7651Sdrh }else if( pDb->maxStmt<=0 ){ 1764fb7e7651Sdrh /* If the cache is turned off, deallocated the statement */ 1765fb7e7651Sdrh if( pPreStmt ) Tcl_Free((char*)pPreStmt); 1766fb7e7651Sdrh sqlite3_finalize(pStmt); 1767fb7e7651Sdrh }else{ 1768fb7e7651Sdrh /* Everything worked and the cache is operational. 1769fb7e7651Sdrh ** Create a new SqlPreparedStmt structure if we need one. 1770fb7e7651Sdrh ** (If we already have one we can just reuse it.) 1771fb7e7651Sdrh */ 1772fb7e7651Sdrh if( pPreStmt==0 ){ 1773fb7e7651Sdrh len = zLeft - zSql; 1774fb7e7651Sdrh pPreStmt = (SqlPreparedStmt*)Tcl_Alloc( sizeof(*pPreStmt) + len ); 1775fb7e7651Sdrh if( pPreStmt==0 ) return TCL_ERROR; 1776fb7e7651Sdrh pPreStmt->pStmt = pStmt; 1777fb7e7651Sdrh pPreStmt->nSql = len; 1778fb7e7651Sdrh memcpy(pPreStmt->zSql, zSql, len); 1779fb7e7651Sdrh pPreStmt->zSql[len] = 0; 178030ccda10Sdanielk1977 } 178130ccda10Sdanielk1977 1782fb7e7651Sdrh /* Add the prepared statement to the beginning of the cache list 1783fb7e7651Sdrh */ 1784fb7e7651Sdrh pPreStmt->pNext = pDb->stmtList; 1785fb7e7651Sdrh pPreStmt->pPrev = 0; 1786fb7e7651Sdrh if( pDb->stmtList ){ 1787fb7e7651Sdrh pDb->stmtList->pPrev = pPreStmt; 1788fb7e7651Sdrh } 1789fb7e7651Sdrh pDb->stmtList = pPreStmt; 1790fb7e7651Sdrh if( pDb->stmtLast==0 ){ 1791fb7e7651Sdrh assert( pDb->nStmt==0 ); 1792fb7e7651Sdrh pDb->stmtLast = pPreStmt; 1793fb7e7651Sdrh }else{ 1794fb7e7651Sdrh assert( pDb->nStmt>0 ); 1795fb7e7651Sdrh } 1796fb7e7651Sdrh pDb->nStmt++; 1797fb7e7651Sdrh 1798fb7e7651Sdrh /* If we have too many statement in cache, remove the surplus from the 1799fb7e7651Sdrh ** end of the cache list. 1800fb7e7651Sdrh */ 1801fb7e7651Sdrh while( pDb->nStmt>pDb->maxStmt ){ 1802fb7e7651Sdrh sqlite3_finalize(pDb->stmtLast->pStmt); 1803fb7e7651Sdrh pDb->stmtLast = pDb->stmtLast->pPrev; 1804fb7e7651Sdrh Tcl_Free((char*)pDb->stmtLast->pNext); 1805fb7e7651Sdrh pDb->stmtLast->pNext = 0; 1806fb7e7651Sdrh pDb->nStmt--; 1807fb7e7651Sdrh } 1808fb7e7651Sdrh } 1809fb7e7651Sdrh 1810fb7e7651Sdrh /* Proceed to the next statement */ 181130ccda10Sdanielk1977 zSql = zLeft; 181230ccda10Sdanielk1977 } 18131d895039Sdrh Tcl_DecrRefCount(objv[2]); 181430ccda10Sdanielk1977 18151807ce37Sdrh if( pRet ){ 1816ef2cb63eSdanielk1977 if( rc==TCL_OK ){ 181730ccda10Sdanielk1977 Tcl_SetObjResult(interp, pRet); 181830ccda10Sdanielk1977 } 1819ef2cb63eSdanielk1977 Tcl_DecrRefCount(pRet); 1820050be329Sdrh }else if( rc==TCL_OK ){ 1821050be329Sdrh Tcl_ResetResult(interp); 18221807ce37Sdrh } 182330ccda10Sdanielk1977 break; 182430ccda10Sdanielk1977 } 1825bec3f402Sdrh 1826bec3f402Sdrh /* 1827cabb0819Sdrh ** $db function NAME SCRIPT 1828cabb0819Sdrh ** 1829cabb0819Sdrh ** Create a new SQL function called NAME. Whenever that function is 1830cabb0819Sdrh ** called, invoke SCRIPT to evaluate the function. 1831cabb0819Sdrh */ 1832cabb0819Sdrh case DB_FUNCTION: { 1833cabb0819Sdrh SqlFunc *pFunc; 1834d1e4733dSdrh Tcl_Obj *pScript; 1835cabb0819Sdrh char *zName; 1836cabb0819Sdrh if( objc!=4 ){ 1837cabb0819Sdrh Tcl_WrongNumArgs(interp, 2, objv, "NAME SCRIPT"); 1838cabb0819Sdrh return TCL_ERROR; 1839cabb0819Sdrh } 1840cabb0819Sdrh zName = Tcl_GetStringFromObj(objv[2], 0); 1841d1e4733dSdrh pScript = objv[3]; 1842d1e4733dSdrh pFunc = findSqlFunc(pDb, zName); 1843cabb0819Sdrh if( pFunc==0 ) return TCL_ERROR; 1844d1e4733dSdrh if( pFunc->pScript ){ 1845d1e4733dSdrh Tcl_DecrRefCount(pFunc->pScript); 1846d1e4733dSdrh } 1847d1e4733dSdrh pFunc->pScript = pScript; 1848d1e4733dSdrh Tcl_IncrRefCount(pScript); 1849d1e4733dSdrh pFunc->useEvalObjv = safeToUseEvalObjv(interp, pScript); 1850c8c1158bSdanielk1977 rc = sqlite3_create_function(pDb->db, zName, -1, SQLITE_UTF8, 1851d8123366Sdanielk1977 pFunc, tclSqlFunc, 0, 0); 1852fb7e7651Sdrh if( rc!=SQLITE_OK ){ 1853fb7e7651Sdrh rc = TCL_ERROR; 18549636c4e1Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 1855fb7e7651Sdrh }else{ 1856fb7e7651Sdrh /* Must flush any cached statements */ 1857fb7e7651Sdrh flushStmtCache( pDb ); 1858fb7e7651Sdrh } 1859cabb0819Sdrh break; 1860cabb0819Sdrh } 1861cabb0819Sdrh 1862cabb0819Sdrh /* 18638cbadb02Sdanielk1977 ** $db incrblob ?-readonly? ?DB? TABLE COLUMN ROWID 1864b4e9af9fSdanielk1977 */ 1865b4e9af9fSdanielk1977 case DB_INCRBLOB: { 186632a0d8bbSdanielk1977 #ifdef SQLITE_OMIT_INCRBLOB 186732a0d8bbSdanielk1977 Tcl_AppendResult(interp, "incrblob not available in this build", 0); 186832a0d8bbSdanielk1977 return TCL_ERROR; 186932a0d8bbSdanielk1977 #else 18708cbadb02Sdanielk1977 int isReadonly = 0; 1871b4e9af9fSdanielk1977 const char *zDb = "main"; 1872b4e9af9fSdanielk1977 const char *zTable; 1873b4e9af9fSdanielk1977 const char *zColumn; 1874b4e9af9fSdanielk1977 sqlite_int64 iRow; 1875b4e9af9fSdanielk1977 18768cbadb02Sdanielk1977 /* Check for the -readonly option */ 18778cbadb02Sdanielk1977 if( objc>3 && strcmp(Tcl_GetString(objv[2]), "-readonly")==0 ){ 18788cbadb02Sdanielk1977 isReadonly = 1; 18798cbadb02Sdanielk1977 } 18808cbadb02Sdanielk1977 18818cbadb02Sdanielk1977 if( objc!=(5+isReadonly) && objc!=(6+isReadonly) ){ 18828cbadb02Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?-readonly? ?DB? TABLE COLUMN ROWID"); 1883b4e9af9fSdanielk1977 return TCL_ERROR; 1884b4e9af9fSdanielk1977 } 1885b4e9af9fSdanielk1977 18868cbadb02Sdanielk1977 if( objc==(6+isReadonly) ){ 1887b4e9af9fSdanielk1977 zDb = Tcl_GetString(objv[2]); 1888b4e9af9fSdanielk1977 } 1889b4e9af9fSdanielk1977 zTable = Tcl_GetString(objv[objc-3]); 1890b4e9af9fSdanielk1977 zColumn = Tcl_GetString(objv[objc-2]); 1891b4e9af9fSdanielk1977 rc = Tcl_GetWideIntFromObj(interp, objv[objc-1], &iRow); 1892b4e9af9fSdanielk1977 1893b4e9af9fSdanielk1977 if( rc==TCL_OK ){ 18948cbadb02Sdanielk1977 rc = createIncrblobChannel( 18958cbadb02Sdanielk1977 interp, pDb, zDb, zTable, zColumn, iRow, isReadonly 18968cbadb02Sdanielk1977 ); 1897b4e9af9fSdanielk1977 } 189832a0d8bbSdanielk1977 #endif 1899b4e9af9fSdanielk1977 break; 1900b4e9af9fSdanielk1977 } 1901b4e9af9fSdanielk1977 1902b4e9af9fSdanielk1977 /* 1903f11bded5Sdrh ** $db interrupt 1904f11bded5Sdrh ** 1905f11bded5Sdrh ** Interrupt the execution of the inner-most SQL interpreter. This 1906f11bded5Sdrh ** causes the SQL statement to return an error of SQLITE_INTERRUPT. 1907f11bded5Sdrh */ 1908f11bded5Sdrh case DB_INTERRUPT: { 1909f11bded5Sdrh sqlite3_interrupt(pDb->db); 1910f11bded5Sdrh break; 1911f11bded5Sdrh } 1912f11bded5Sdrh 1913f11bded5Sdrh /* 191419e2d37fSdrh ** $db nullvalue ?STRING? 191519e2d37fSdrh ** 191619e2d37fSdrh ** Change text used when a NULL comes back from the database. If ?STRING? 191719e2d37fSdrh ** is not present, then the current string used for NULL is returned. 191819e2d37fSdrh ** If STRING is present, then STRING is returned. 191919e2d37fSdrh ** 192019e2d37fSdrh */ 192119e2d37fSdrh case DB_NULLVALUE: { 192219e2d37fSdrh if( objc!=2 && objc!=3 ){ 192319e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "NULLVALUE"); 192419e2d37fSdrh return TCL_ERROR; 192519e2d37fSdrh } 192619e2d37fSdrh if( objc==3 ){ 192719e2d37fSdrh int len; 192819e2d37fSdrh char *zNull = Tcl_GetStringFromObj(objv[2], &len); 192919e2d37fSdrh if( pDb->zNull ){ 193019e2d37fSdrh Tcl_Free(pDb->zNull); 193119e2d37fSdrh } 193219e2d37fSdrh if( zNull && len>0 ){ 193319e2d37fSdrh pDb->zNull = Tcl_Alloc( len + 1 ); 193419e2d37fSdrh strncpy(pDb->zNull, zNull, len); 193519e2d37fSdrh pDb->zNull[len] = '\0'; 193619e2d37fSdrh }else{ 193719e2d37fSdrh pDb->zNull = 0; 193819e2d37fSdrh } 193919e2d37fSdrh } 194019e2d37fSdrh Tcl_SetObjResult(interp, dbTextToObj(pDb->zNull)); 194119e2d37fSdrh break; 194219e2d37fSdrh } 194319e2d37fSdrh 194419e2d37fSdrh /* 1945af9ff33aSdrh ** $db last_insert_rowid 1946af9ff33aSdrh ** 1947af9ff33aSdrh ** Return an integer which is the ROWID for the most recent insert. 1948af9ff33aSdrh */ 1949af9ff33aSdrh case DB_LAST_INSERT_ROWID: { 1950af9ff33aSdrh Tcl_Obj *pResult; 1951f7e678d6Sdrh Tcl_WideInt rowid; 1952af9ff33aSdrh if( objc!=2 ){ 1953af9ff33aSdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 1954af9ff33aSdrh return TCL_ERROR; 1955af9ff33aSdrh } 19566f8a503dSdanielk1977 rowid = sqlite3_last_insert_rowid(pDb->db); 1957af9ff33aSdrh pResult = Tcl_GetObjResult(interp); 1958f7e678d6Sdrh Tcl_SetWideIntObj(pResult, rowid); 1959af9ff33aSdrh break; 1960af9ff33aSdrh } 1961af9ff33aSdrh 1962af9ff33aSdrh /* 19631807ce37Sdrh ** The DB_ONECOLUMN method is implemented together with DB_EVAL. 19645d9d7576Sdrh */ 19651807ce37Sdrh 19661807ce37Sdrh /* $db progress ?N CALLBACK? 19671807ce37Sdrh ** 19681807ce37Sdrh ** Invoke the given callback every N virtual machine opcodes while executing 19691807ce37Sdrh ** queries. 19701807ce37Sdrh */ 19711807ce37Sdrh case DB_PROGRESS: { 19721807ce37Sdrh if( objc==2 ){ 19731807ce37Sdrh if( pDb->zProgress ){ 19741807ce37Sdrh Tcl_AppendResult(interp, pDb->zProgress, 0); 19755d9d7576Sdrh } 19761807ce37Sdrh }else if( objc==4 ){ 19771807ce37Sdrh char *zProgress; 19781807ce37Sdrh int len; 19791807ce37Sdrh int N; 19801807ce37Sdrh if( TCL_OK!=Tcl_GetIntFromObj(interp, objv[2], &N) ){ 19811807ce37Sdrh return TCL_ERROR; 19821807ce37Sdrh }; 19831807ce37Sdrh if( pDb->zProgress ){ 19841807ce37Sdrh Tcl_Free(pDb->zProgress); 19851807ce37Sdrh } 19861807ce37Sdrh zProgress = Tcl_GetStringFromObj(objv[3], &len); 19871807ce37Sdrh if( zProgress && len>0 ){ 19881807ce37Sdrh pDb->zProgress = Tcl_Alloc( len + 1 ); 19895bb3eb9bSdrh memcpy(pDb->zProgress, zProgress, len+1); 19901807ce37Sdrh }else{ 19911807ce37Sdrh pDb->zProgress = 0; 19921807ce37Sdrh } 19931807ce37Sdrh #ifndef SQLITE_OMIT_PROGRESS_CALLBACK 19941807ce37Sdrh if( pDb->zProgress ){ 19951807ce37Sdrh pDb->interp = interp; 19961807ce37Sdrh sqlite3_progress_handler(pDb->db, N, DbProgressHandler, pDb); 19971807ce37Sdrh }else{ 19981807ce37Sdrh sqlite3_progress_handler(pDb->db, 0, 0, 0); 19991807ce37Sdrh } 20001807ce37Sdrh #endif 20011807ce37Sdrh }else{ 20021807ce37Sdrh Tcl_WrongNumArgs(interp, 2, objv, "N CALLBACK"); 20031807ce37Sdrh return TCL_ERROR; 20045d9d7576Sdrh } 20055d9d7576Sdrh break; 20065d9d7576Sdrh } 20075d9d7576Sdrh 200819e2d37fSdrh /* $db profile ?CALLBACK? 200919e2d37fSdrh ** 201019e2d37fSdrh ** Make arrangements to invoke the CALLBACK routine after each SQL statement 201119e2d37fSdrh ** that has run. The text of the SQL and the amount of elapse time are 201219e2d37fSdrh ** appended to CALLBACK before the script is run. 201319e2d37fSdrh */ 201419e2d37fSdrh case DB_PROFILE: { 201519e2d37fSdrh if( objc>3 ){ 201619e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 201719e2d37fSdrh return TCL_ERROR; 201819e2d37fSdrh }else if( objc==2 ){ 201919e2d37fSdrh if( pDb->zProfile ){ 202019e2d37fSdrh Tcl_AppendResult(interp, pDb->zProfile, 0); 202119e2d37fSdrh } 202219e2d37fSdrh }else{ 202319e2d37fSdrh char *zProfile; 202419e2d37fSdrh int len; 202519e2d37fSdrh if( pDb->zProfile ){ 202619e2d37fSdrh Tcl_Free(pDb->zProfile); 202719e2d37fSdrh } 202819e2d37fSdrh zProfile = Tcl_GetStringFromObj(objv[2], &len); 202919e2d37fSdrh if( zProfile && len>0 ){ 203019e2d37fSdrh pDb->zProfile = Tcl_Alloc( len + 1 ); 20315bb3eb9bSdrh memcpy(pDb->zProfile, zProfile, len+1); 203219e2d37fSdrh }else{ 203319e2d37fSdrh pDb->zProfile = 0; 203419e2d37fSdrh } 203519e2d37fSdrh #ifndef SQLITE_OMIT_TRACE 203619e2d37fSdrh if( pDb->zProfile ){ 203719e2d37fSdrh pDb->interp = interp; 203819e2d37fSdrh sqlite3_profile(pDb->db, DbProfileHandler, pDb); 203919e2d37fSdrh }else{ 204019e2d37fSdrh sqlite3_profile(pDb->db, 0, 0); 204119e2d37fSdrh } 204219e2d37fSdrh #endif 204319e2d37fSdrh } 204419e2d37fSdrh break; 204519e2d37fSdrh } 204619e2d37fSdrh 20475d9d7576Sdrh /* 204822fbcb8dSdrh ** $db rekey KEY 204922fbcb8dSdrh ** 205022fbcb8dSdrh ** Change the encryption key on the currently open database. 205122fbcb8dSdrh */ 205222fbcb8dSdrh case DB_REKEY: { 205322fbcb8dSdrh int nKey; 205422fbcb8dSdrh void *pKey; 205522fbcb8dSdrh if( objc!=3 ){ 205622fbcb8dSdrh Tcl_WrongNumArgs(interp, 2, objv, "KEY"); 205722fbcb8dSdrh return TCL_ERROR; 205822fbcb8dSdrh } 205922fbcb8dSdrh pKey = Tcl_GetByteArrayFromObj(objv[2], &nKey); 20609eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 20612011d5f5Sdrh rc = sqlite3_rekey(pDb->db, pKey, nKey); 206222fbcb8dSdrh if( rc ){ 2063f20b21c8Sdanielk1977 Tcl_AppendResult(interp, sqlite3ErrStr(rc), 0); 206422fbcb8dSdrh rc = TCL_ERROR; 206522fbcb8dSdrh } 206622fbcb8dSdrh #endif 206722fbcb8dSdrh break; 206822fbcb8dSdrh } 206922fbcb8dSdrh 207022fbcb8dSdrh /* 2071bec3f402Sdrh ** $db timeout MILLESECONDS 2072bec3f402Sdrh ** 2073bec3f402Sdrh ** Delay for the number of milliseconds specified when a file is locked. 2074bec3f402Sdrh */ 20756d31316cSdrh case DB_TIMEOUT: { 2076bec3f402Sdrh int ms; 20776d31316cSdrh if( objc!=3 ){ 20786d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "MILLISECONDS"); 2079bec3f402Sdrh return TCL_ERROR; 208075897234Sdrh } 20816d31316cSdrh if( Tcl_GetIntFromObj(interp, objv[2], &ms) ) return TCL_ERROR; 20826f8a503dSdanielk1977 sqlite3_busy_timeout(pDb->db, ms); 20836d31316cSdrh break; 208475897234Sdrh } 2085b5a20d3cSdrh 20860f14e2ebSdrh /* 20870f14e2ebSdrh ** $db total_changes 20880f14e2ebSdrh ** 20890f14e2ebSdrh ** Return the number of rows that were modified, inserted, or deleted 20900f14e2ebSdrh ** since the database handle was created. 20910f14e2ebSdrh */ 20920f14e2ebSdrh case DB_TOTAL_CHANGES: { 20930f14e2ebSdrh Tcl_Obj *pResult; 20940f14e2ebSdrh if( objc!=2 ){ 20950f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 20960f14e2ebSdrh return TCL_ERROR; 20970f14e2ebSdrh } 20980f14e2ebSdrh pResult = Tcl_GetObjResult(interp); 20990f14e2ebSdrh Tcl_SetIntObj(pResult, sqlite3_total_changes(pDb->db)); 21000f14e2ebSdrh break; 21010f14e2ebSdrh } 21020f14e2ebSdrh 2103b5a20d3cSdrh /* $db trace ?CALLBACK? 2104b5a20d3cSdrh ** 2105b5a20d3cSdrh ** Make arrangements to invoke the CALLBACK routine for each SQL statement 2106b5a20d3cSdrh ** that is executed. The text of the SQL is appended to CALLBACK before 2107b5a20d3cSdrh ** it is executed. 2108b5a20d3cSdrh */ 2109b5a20d3cSdrh case DB_TRACE: { 2110b5a20d3cSdrh if( objc>3 ){ 2111b5a20d3cSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 2112b97759edSdrh return TCL_ERROR; 2113b5a20d3cSdrh }else if( objc==2 ){ 2114b5a20d3cSdrh if( pDb->zTrace ){ 2115b5a20d3cSdrh Tcl_AppendResult(interp, pDb->zTrace, 0); 2116b5a20d3cSdrh } 2117b5a20d3cSdrh }else{ 2118b5a20d3cSdrh char *zTrace; 2119b5a20d3cSdrh int len; 2120b5a20d3cSdrh if( pDb->zTrace ){ 2121b5a20d3cSdrh Tcl_Free(pDb->zTrace); 2122b5a20d3cSdrh } 2123b5a20d3cSdrh zTrace = Tcl_GetStringFromObj(objv[2], &len); 2124b5a20d3cSdrh if( zTrace && len>0 ){ 2125b5a20d3cSdrh pDb->zTrace = Tcl_Alloc( len + 1 ); 21265bb3eb9bSdrh memcpy(pDb->zTrace, zTrace, len+1); 2127b5a20d3cSdrh }else{ 2128b5a20d3cSdrh pDb->zTrace = 0; 2129b5a20d3cSdrh } 213019e2d37fSdrh #ifndef SQLITE_OMIT_TRACE 2131b5a20d3cSdrh if( pDb->zTrace ){ 2132b5a20d3cSdrh pDb->interp = interp; 21336f8a503dSdanielk1977 sqlite3_trace(pDb->db, DbTraceHandler, pDb); 2134b5a20d3cSdrh }else{ 21356f8a503dSdanielk1977 sqlite3_trace(pDb->db, 0, 0); 2136b5a20d3cSdrh } 213719e2d37fSdrh #endif 2138b5a20d3cSdrh } 2139b5a20d3cSdrh break; 2140b5a20d3cSdrh } 2141b5a20d3cSdrh 21423d21423cSdrh /* $db transaction [-deferred|-immediate|-exclusive] SCRIPT 21433d21423cSdrh ** 21443d21423cSdrh ** Start a new transaction (if we are not already in the midst of a 21453d21423cSdrh ** transaction) and execute the TCL script SCRIPT. After SCRIPT 21463d21423cSdrh ** completes, either commit the transaction or roll it back if SCRIPT 21473d21423cSdrh ** throws an exception. Or if no new transation was started, do nothing. 21483d21423cSdrh ** pass the exception on up the stack. 21493d21423cSdrh ** 21503d21423cSdrh ** This command was inspired by Dave Thomas's talk on Ruby at the 21513d21423cSdrh ** 2005 O'Reilly Open Source Convention (OSCON). 21523d21423cSdrh */ 21533d21423cSdrh case DB_TRANSACTION: { 21543d21423cSdrh int inTrans; 21553d21423cSdrh Tcl_Obj *pScript; 21563d21423cSdrh const char *zBegin = "BEGIN"; 21573d21423cSdrh if( objc!=3 && objc!=4 ){ 21583d21423cSdrh Tcl_WrongNumArgs(interp, 2, objv, "[TYPE] SCRIPT"); 21593d21423cSdrh return TCL_ERROR; 21603d21423cSdrh } 21613d21423cSdrh if( objc==3 ){ 21623d21423cSdrh pScript = objv[2]; 21633d21423cSdrh } else { 21643d21423cSdrh static const char *TTYPE_strs[] = { 2165ce604012Sdrh "deferred", "exclusive", "immediate", 0 21663d21423cSdrh }; 21673d21423cSdrh enum TTYPE_enum { 21683d21423cSdrh TTYPE_DEFERRED, TTYPE_EXCLUSIVE, TTYPE_IMMEDIATE 21693d21423cSdrh }; 21703d21423cSdrh int ttype; 2171b5555e7eSdrh if( Tcl_GetIndexFromObj(interp, objv[2], TTYPE_strs, "transaction type", 21723d21423cSdrh 0, &ttype) ){ 21733d21423cSdrh return TCL_ERROR; 21743d21423cSdrh } 21753d21423cSdrh switch( (enum TTYPE_enum)ttype ){ 21763d21423cSdrh case TTYPE_DEFERRED: /* no-op */; break; 21773d21423cSdrh case TTYPE_EXCLUSIVE: zBegin = "BEGIN EXCLUSIVE"; break; 21783d21423cSdrh case TTYPE_IMMEDIATE: zBegin = "BEGIN IMMEDIATE"; break; 21793d21423cSdrh } 21803d21423cSdrh pScript = objv[3]; 21813d21423cSdrh } 21823d21423cSdrh inTrans = !sqlite3_get_autocommit(pDb->db); 21833d21423cSdrh if( !inTrans ){ 21843752785fSdrh (void)sqlite3_exec(pDb->db, zBegin, 0, 0, 0); 21853d21423cSdrh } 21863d21423cSdrh rc = Tcl_EvalObjEx(interp, pScript, 0); 21873d21423cSdrh if( !inTrans ){ 21883d21423cSdrh const char *zEnd; 21893d21423cSdrh if( rc==TCL_ERROR ){ 21903d21423cSdrh zEnd = "ROLLBACK"; 21913d21423cSdrh } else { 21923d21423cSdrh zEnd = "COMMIT"; 21933d21423cSdrh } 2194109b4350Sdrh if( sqlite3_exec(pDb->db, zEnd, 0, 0, 0) ){ 2195109b4350Sdrh sqlite3_exec(pDb->db, "ROLLBACK", 0, 0, 0); 2196109b4350Sdrh } 21973d21423cSdrh } 21983d21423cSdrh break; 21993d21423cSdrh } 22003d21423cSdrh 220194eb6a14Sdanielk1977 /* 220294eb6a14Sdanielk1977 ** $db update_hook ?script? 220371fd80bfSdanielk1977 ** $db rollback_hook ?script? 220494eb6a14Sdanielk1977 */ 220571fd80bfSdanielk1977 case DB_UPDATE_HOOK: 220671fd80bfSdanielk1977 case DB_ROLLBACK_HOOK: { 220771fd80bfSdanielk1977 220871fd80bfSdanielk1977 /* set ppHook to point at pUpdateHook or pRollbackHook, depending on 220971fd80bfSdanielk1977 ** whether [$db update_hook] or [$db rollback_hook] was invoked. 221071fd80bfSdanielk1977 */ 221171fd80bfSdanielk1977 Tcl_Obj **ppHook; 221271fd80bfSdanielk1977 if( choice==DB_UPDATE_HOOK ){ 221371fd80bfSdanielk1977 ppHook = &pDb->pUpdateHook; 221471fd80bfSdanielk1977 }else{ 221571fd80bfSdanielk1977 ppHook = &pDb->pRollbackHook; 221671fd80bfSdanielk1977 } 221771fd80bfSdanielk1977 221894eb6a14Sdanielk1977 if( objc!=2 && objc!=3 ){ 221994eb6a14Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?SCRIPT?"); 222094eb6a14Sdanielk1977 return TCL_ERROR; 222194eb6a14Sdanielk1977 } 222271fd80bfSdanielk1977 if( *ppHook ){ 222371fd80bfSdanielk1977 Tcl_SetObjResult(interp, *ppHook); 222494eb6a14Sdanielk1977 if( objc==3 ){ 222571fd80bfSdanielk1977 Tcl_DecrRefCount(*ppHook); 222671fd80bfSdanielk1977 *ppHook = 0; 222794eb6a14Sdanielk1977 } 222894eb6a14Sdanielk1977 } 222994eb6a14Sdanielk1977 if( objc==3 ){ 223071fd80bfSdanielk1977 assert( !(*ppHook) ); 223194eb6a14Sdanielk1977 if( Tcl_GetCharLength(objv[2])>0 ){ 223271fd80bfSdanielk1977 *ppHook = objv[2]; 223371fd80bfSdanielk1977 Tcl_IncrRefCount(*ppHook); 223494eb6a14Sdanielk1977 } 223594eb6a14Sdanielk1977 } 223671fd80bfSdanielk1977 223771fd80bfSdanielk1977 sqlite3_update_hook(pDb->db, (pDb->pUpdateHook?DbUpdateHandler:0), pDb); 223871fd80bfSdanielk1977 sqlite3_rollback_hook(pDb->db,(pDb->pRollbackHook?DbRollbackHandler:0),pDb); 223971fd80bfSdanielk1977 224094eb6a14Sdanielk1977 break; 224194eb6a14Sdanielk1977 } 224294eb6a14Sdanielk1977 22434397de57Sdanielk1977 /* $db version 22444397de57Sdanielk1977 ** 22454397de57Sdanielk1977 ** Return the version string for this database. 22464397de57Sdanielk1977 */ 22474397de57Sdanielk1977 case DB_VERSION: { 22484397de57Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_libversion(), TCL_STATIC); 22494397de57Sdanielk1977 break; 22504397de57Sdanielk1977 } 22514397de57Sdanielk1977 22521067fe11Stpoindex 22536d31316cSdrh } /* End of the SWITCH statement */ 225422fbcb8dSdrh return rc; 225575897234Sdrh } 225675897234Sdrh 225775897234Sdrh /* 22583570ad93Sdrh ** sqlite3 DBNAME FILENAME ?-vfs VFSNAME? ?-key KEY? ?-readonly BOOLEAN? 22593570ad93Sdrh ** ?-create BOOLEAN? 226075897234Sdrh ** 226175897234Sdrh ** This is the main Tcl command. When the "sqlite" Tcl command is 226275897234Sdrh ** invoked, this routine runs to process that command. 226375897234Sdrh ** 226475897234Sdrh ** The first argument, DBNAME, is an arbitrary name for a new 226575897234Sdrh ** database connection. This command creates a new command named 226675897234Sdrh ** DBNAME that is used to control that connection. The database 226775897234Sdrh ** connection is deleted when the DBNAME command is deleted. 226875897234Sdrh ** 22693570ad93Sdrh ** The second argument is the name of the database file. 2270fbc3eab8Sdrh ** 227175897234Sdrh */ 227222fbcb8dSdrh static int DbMain(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){ 2273bec3f402Sdrh SqliteDb *p; 227422fbcb8dSdrh void *pKey = 0; 227522fbcb8dSdrh int nKey = 0; 227622fbcb8dSdrh const char *zArg; 227775897234Sdrh char *zErrMsg; 22783570ad93Sdrh int i; 227922fbcb8dSdrh const char *zFile; 22803570ad93Sdrh const char *zVfs = 0; 22813570ad93Sdrh int flags = SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE; 2282882e8e4dSdrh Tcl_DString translatedFilename; 228322fbcb8dSdrh if( objc==2 ){ 228422fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[1], 0); 228522fbcb8dSdrh if( strcmp(zArg,"-version")==0 ){ 22866f8a503dSdanielk1977 Tcl_AppendResult(interp,sqlite3_version,0); 2287647cb0e1Sdrh return TCL_OK; 2288647cb0e1Sdrh } 22899eb9e26bSdrh if( strcmp(zArg,"-has-codec")==0 ){ 22909eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 229122fbcb8dSdrh Tcl_AppendResult(interp,"1",0); 229222fbcb8dSdrh #else 229322fbcb8dSdrh Tcl_AppendResult(interp,"0",0); 229422fbcb8dSdrh #endif 229522fbcb8dSdrh return TCL_OK; 229622fbcb8dSdrh } 2297fbc3eab8Sdrh } 22983570ad93Sdrh for(i=3; i+1<objc; i+=2){ 22993570ad93Sdrh zArg = Tcl_GetString(objv[i]); 230022fbcb8dSdrh if( strcmp(zArg,"-key")==0 ){ 23013570ad93Sdrh pKey = Tcl_GetByteArrayFromObj(objv[i+1], &nKey); 23023570ad93Sdrh }else if( strcmp(zArg, "-vfs")==0 ){ 23033570ad93Sdrh i++; 23043570ad93Sdrh zVfs = Tcl_GetString(objv[i]); 23053570ad93Sdrh }else if( strcmp(zArg, "-readonly")==0 ){ 23063570ad93Sdrh int b; 23073570ad93Sdrh if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 23083570ad93Sdrh if( b ){ 2309*33f4e02aSdrh flags &= ~(SQLITE_OPEN_READWRITE|SQLITE_OPEN_CREATE); 23103570ad93Sdrh flags |= SQLITE_OPEN_READONLY; 23113570ad93Sdrh }else{ 23123570ad93Sdrh flags &= ~SQLITE_OPEN_READONLY; 23133570ad93Sdrh flags |= SQLITE_OPEN_READWRITE; 23143570ad93Sdrh } 23153570ad93Sdrh }else if( strcmp(zArg, "-create")==0 ){ 23163570ad93Sdrh int b; 23173570ad93Sdrh if( Tcl_GetBooleanFromObj(interp, objv[i+1], &b) ) return TCL_ERROR; 2318*33f4e02aSdrh if( b && (flags & SQLITE_OPEN_READONLY)==0 ){ 23193570ad93Sdrh flags |= SQLITE_OPEN_CREATE; 23203570ad93Sdrh }else{ 23213570ad93Sdrh flags &= ~SQLITE_OPEN_CREATE; 23223570ad93Sdrh } 23233570ad93Sdrh }else{ 23243570ad93Sdrh Tcl_AppendResult(interp, "unknown option: ", zArg, (char*)0); 23253570ad93Sdrh return TCL_ERROR; 232622fbcb8dSdrh } 232722fbcb8dSdrh } 23283570ad93Sdrh if( objc<3 || (objc&1)!=1 ){ 232922fbcb8dSdrh Tcl_WrongNumArgs(interp, 1, objv, 23303570ad93Sdrh "HANDLE FILENAME ?-vfs VFSNAME? ?-readonly BOOLEAN? ?-create BOOLEAN?" 23319eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 23323570ad93Sdrh " ?-key CODECKEY?" 233322fbcb8dSdrh #endif 233422fbcb8dSdrh ); 233575897234Sdrh return TCL_ERROR; 233675897234Sdrh } 233775897234Sdrh zErrMsg = 0; 23384cdc9e84Sdrh p = (SqliteDb*)Tcl_Alloc( sizeof(*p) ); 233975897234Sdrh if( p==0 ){ 2340bec3f402Sdrh Tcl_SetResult(interp, "malloc failed", TCL_STATIC); 2341bec3f402Sdrh return TCL_ERROR; 2342bec3f402Sdrh } 2343bec3f402Sdrh memset(p, 0, sizeof(*p)); 234422fbcb8dSdrh zFile = Tcl_GetStringFromObj(objv[2], 0); 2345882e8e4dSdrh zFile = Tcl_TranslateFileName(interp, zFile, &translatedFilename); 23463570ad93Sdrh sqlite3_open_v2(zFile, &p->db, flags, zVfs); 2347882e8e4dSdrh Tcl_DStringFree(&translatedFilename); 234880290863Sdanielk1977 if( SQLITE_OK!=sqlite3_errcode(p->db) ){ 23499404d50eSdrh zErrMsg = sqlite3_mprintf("%s", sqlite3_errmsg(p->db)); 235080290863Sdanielk1977 sqlite3_close(p->db); 235180290863Sdanielk1977 p->db = 0; 235280290863Sdanielk1977 } 2353f3a65f7eSdrh #ifdef SQLITE_TEST 2354f3a65f7eSdrh if( p->db ){ 2355f3a65f7eSdrh extern int Md5_Register(sqlite3*); 2356f3a65f7eSdrh if( Md5_Register(p->db)==SQLITE_NOMEM ){ 2357f3a65f7eSdrh zErrMsg = sqlite3_mprintf("%s", sqlite3_errmsg(p->db)); 2358f3a65f7eSdrh sqlite3_close(p->db); 2359f3a65f7eSdrh p->db = 0; 2360f3a65f7eSdrh } 2361f3a65f7eSdrh } 2362f3a65f7eSdrh #endif 23632011d5f5Sdrh #ifdef SQLITE_HAS_CODEC 2364f3a65f7eSdrh if( p->db ){ 23652011d5f5Sdrh sqlite3_key(p->db, pKey, nKey); 2366f3a65f7eSdrh } 2367eb8ed70dSdrh #endif 2368bec3f402Sdrh if( p->db==0 ){ 236975897234Sdrh Tcl_SetResult(interp, zErrMsg, TCL_VOLATILE); 2370bec3f402Sdrh Tcl_Free((char*)p); 23719404d50eSdrh sqlite3_free(zErrMsg); 237275897234Sdrh return TCL_ERROR; 237375897234Sdrh } 2374fb7e7651Sdrh p->maxStmt = NUM_PREPARED_STMTS; 23755169bbc6Sdrh p->interp = interp; 237622fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[1], 0); 237722fbcb8dSdrh Tcl_CreateObjCommand(interp, zArg, DbObjCmd, (char*)p, DbDeleteCmd); 237875897234Sdrh return TCL_OK; 237975897234Sdrh } 238075897234Sdrh 238175897234Sdrh /* 238290ca9753Sdrh ** Provide a dummy Tcl_InitStubs if we are using this as a static 238390ca9753Sdrh ** library. 238490ca9753Sdrh */ 238590ca9753Sdrh #ifndef USE_TCL_STUBS 238690ca9753Sdrh # undef Tcl_InitStubs 238790ca9753Sdrh # define Tcl_InitStubs(a,b,c) 238890ca9753Sdrh #endif 238990ca9753Sdrh 239090ca9753Sdrh /* 239129bc4615Sdrh ** Make sure we have a PACKAGE_VERSION macro defined. This will be 239229bc4615Sdrh ** defined automatically by the TEA makefile. But other makefiles 239329bc4615Sdrh ** do not define it. 239429bc4615Sdrh */ 239529bc4615Sdrh #ifndef PACKAGE_VERSION 239629bc4615Sdrh # define PACKAGE_VERSION SQLITE_VERSION 239729bc4615Sdrh #endif 239829bc4615Sdrh 239929bc4615Sdrh /* 240075897234Sdrh ** Initialize this module. 240175897234Sdrh ** 240275897234Sdrh ** This Tcl module contains only a single new Tcl command named "sqlite". 240375897234Sdrh ** (Hence there is no namespace. There is no point in using a namespace 240475897234Sdrh ** if the extension only supplies one new name!) The "sqlite" command is 240575897234Sdrh ** used to open a new SQLite database. See the DbMain() routine above 240675897234Sdrh ** for additional information. 240775897234Sdrh */ 2408ad6e1370Sdrh EXTERN int Sqlite3_Init(Tcl_Interp *interp){ 240992febd92Sdrh Tcl_InitStubs(interp, "8.4", 0); 2410ef4ac8f9Sdrh Tcl_CreateObjCommand(interp, "sqlite3", (Tcl_ObjCmdProc*)DbMain, 0, 0); 241129bc4615Sdrh Tcl_PkgProvide(interp, "sqlite3", PACKAGE_VERSION); 241249766d6cSdrh Tcl_CreateObjCommand(interp, "sqlite", (Tcl_ObjCmdProc*)DbMain, 0, 0); 241329bc4615Sdrh Tcl_PkgProvide(interp, "sqlite", PACKAGE_VERSION); 241490ca9753Sdrh return TCL_OK; 241590ca9753Sdrh } 2416ad6e1370Sdrh EXTERN int Tclsqlite3_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2417ad6e1370Sdrh EXTERN int Sqlite3_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2418ad6e1370Sdrh EXTERN int Tclsqlite3_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 241949766d6cSdrh 242049766d6cSdrh #ifndef SQLITE_3_SUFFIX_ONLY 2421ad6e1370Sdrh EXTERN int Sqlite_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2422ad6e1370Sdrh EXTERN int Tclsqlite_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2423ad6e1370Sdrh EXTERN int Sqlite_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2424ad6e1370Sdrh EXTERN int Tclsqlite_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 242549766d6cSdrh #endif 242675897234Sdrh 24273e27c026Sdrh #ifdef TCLSH 24283e27c026Sdrh /***************************************************************************** 24293e27c026Sdrh ** The code that follows is used to build standalone TCL interpreters 24303570ad93Sdrh ** that are statically linked with SQLite. 243175897234Sdrh */ 2432348784efSdrh 2433348784efSdrh /* 24343e27c026Sdrh ** If the macro TCLSH is one, then put in code this for the 24353e27c026Sdrh ** "main" routine that will initialize Tcl and take input from 24363570ad93Sdrh ** standard input, or if a file is named on the command line 24373570ad93Sdrh ** the TCL interpreter reads and evaluates that file. 2438348784efSdrh */ 24393e27c026Sdrh #if TCLSH==1 2440348784efSdrh static char zMainloop[] = 2441348784efSdrh "set line {}\n" 2442348784efSdrh "while {![eof stdin]} {\n" 2443348784efSdrh "if {$line!=\"\"} {\n" 2444348784efSdrh "puts -nonewline \"> \"\n" 2445348784efSdrh "} else {\n" 2446348784efSdrh "puts -nonewline \"% \"\n" 2447348784efSdrh "}\n" 2448348784efSdrh "flush stdout\n" 2449348784efSdrh "append line [gets stdin]\n" 2450348784efSdrh "if {[info complete $line]} {\n" 2451348784efSdrh "if {[catch {uplevel #0 $line} result]} {\n" 2452348784efSdrh "puts stderr \"Error: $result\"\n" 2453348784efSdrh "} elseif {$result!=\"\"} {\n" 2454348784efSdrh "puts $result\n" 2455348784efSdrh "}\n" 2456348784efSdrh "set line {}\n" 2457348784efSdrh "} else {\n" 2458348784efSdrh "append line \\n\n" 2459348784efSdrh "}\n" 2460348784efSdrh "}\n" 2461348784efSdrh ; 24623e27c026Sdrh #endif 24633e27c026Sdrh 24643e27c026Sdrh /* 24653e27c026Sdrh ** If the macro TCLSH is two, then get the main loop code out of 24663e27c026Sdrh ** the separate file "spaceanal_tcl.h". 24673e27c026Sdrh */ 24683e27c026Sdrh #if TCLSH==2 24693e27c026Sdrh static char zMainloop[] = 24703e27c026Sdrh #include "spaceanal_tcl.h" 24713e27c026Sdrh ; 24723e27c026Sdrh #endif 2473348784efSdrh 2474348784efSdrh #define TCLSH_MAIN main /* Needed to fake out mktclapp */ 2475348784efSdrh int TCLSH_MAIN(int argc, char **argv){ 2476348784efSdrh Tcl_Interp *interp; 2477297ecf14Sdrh Tcl_FindExecutable(argv[0]); 2478348784efSdrh interp = Tcl_CreateInterp(); 247938f8271fSdrh Sqlite3_Init(interp); 2480d9b0257aSdrh #ifdef SQLITE_TEST 2481d1bf3512Sdrh { 24822f999a67Sdrh extern int Md5_Init(Tcl_Interp*); 24832f999a67Sdrh extern int Sqliteconfig_Init(Tcl_Interp*); 2484d1bf3512Sdrh extern int Sqlitetest1_Init(Tcl_Interp*); 24855c4d9703Sdrh extern int Sqlitetest2_Init(Tcl_Interp*); 24865c4d9703Sdrh extern int Sqlitetest3_Init(Tcl_Interp*); 2487a6064dcfSdrh extern int Sqlitetest4_Init(Tcl_Interp*); 2488998b56c3Sdanielk1977 extern int Sqlitetest5_Init(Tcl_Interp*); 24899c06c953Sdrh extern int Sqlitetest6_Init(Tcl_Interp*); 249029c636bcSdrh extern int Sqlitetest7_Init(Tcl_Interp*); 2491b9bb7c18Sdrh extern int Sqlitetest8_Init(Tcl_Interp*); 2492a713f2c3Sdanielk1977 extern int Sqlitetest9_Init(Tcl_Interp*); 24932366940dSdrh extern int Sqlitetestasync_Init(Tcl_Interp*); 24941409be69Sdrh extern int Sqlitetest_autoext_Init(Tcl_Interp*); 249515926590Sdrh extern int Sqlitetest_hexio_Init(Tcl_Interp*); 24962f999a67Sdrh extern int Sqlitetest_malloc_Init(Tcl_Interp*); 24972f999a67Sdrh extern int Sqlitetestschema_Init(Tcl_Interp*); 24982f999a67Sdrh extern int Sqlitetestsse_Init(Tcl_Interp*); 24992f999a67Sdrh extern int Sqlitetesttclvar_Init(Tcl_Interp*); 25002e66f0b9Sdrh 25012f999a67Sdrh Md5_Init(interp); 25022f999a67Sdrh Sqliteconfig_Init(interp); 25036490bebdSdanielk1977 Sqlitetest1_Init(interp); 25045c4d9703Sdrh Sqlitetest2_Init(interp); 2505de647130Sdrh Sqlitetest3_Init(interp); 2506fc57d7bfSdanielk1977 Sqlitetest4_Init(interp); 2507998b56c3Sdanielk1977 Sqlitetest5_Init(interp); 25089c06c953Sdrh Sqlitetest6_Init(interp); 250929c636bcSdrh Sqlitetest7_Init(interp); 2510b9bb7c18Sdrh Sqlitetest8_Init(interp); 2511a713f2c3Sdanielk1977 Sqlitetest9_Init(interp); 25122366940dSdrh Sqlitetestasync_Init(interp); 25131409be69Sdrh Sqlitetest_autoext_Init(interp); 251415926590Sdrh Sqlitetest_hexio_Init(interp); 25152f999a67Sdrh Sqlitetest_malloc_Init(interp); 25162f999a67Sdrh Sqlitetestschema_Init(interp); 25172f999a67Sdrh Sqlitetesttclvar_Init(interp); 251889dec819Sdrh #ifdef SQLITE_SSE 25192e66f0b9Sdrh Sqlitetestsse_Init(interp); 25202e66f0b9Sdrh #endif 2521d1bf3512Sdrh } 2522d1bf3512Sdrh #endif 25233e27c026Sdrh if( argc>=2 || TCLSH==2 ){ 2524348784efSdrh int i; 2525ad42c3a3Sshess char zArgc[32]; 2526ad42c3a3Sshess sqlite3_snprintf(sizeof(zArgc), zArgc, "%d", argc-(3-TCLSH)); 2527ad42c3a3Sshess Tcl_SetVar(interp,"argc", zArgc, TCL_GLOBAL_ONLY); 2528348784efSdrh Tcl_SetVar(interp,"argv0",argv[1],TCL_GLOBAL_ONLY); 2529348784efSdrh Tcl_SetVar(interp,"argv", "", TCL_GLOBAL_ONLY); 253061212b69Sdrh for(i=3-TCLSH; i<argc; i++){ 2531348784efSdrh Tcl_SetVar(interp, "argv", argv[i], 2532348784efSdrh TCL_GLOBAL_ONLY | TCL_LIST_ELEMENT | TCL_APPEND_VALUE); 2533348784efSdrh } 25343e27c026Sdrh if( TCLSH==1 && Tcl_EvalFile(interp, argv[1])!=TCL_OK ){ 25350de8c112Sdrh const char *zInfo = Tcl_GetVar(interp, "errorInfo", TCL_GLOBAL_ONLY); 2536c61053b7Sdrh if( zInfo==0 ) zInfo = interp->result; 2537c61053b7Sdrh fprintf(stderr,"%s: %s\n", *argv, zInfo); 2538348784efSdrh return 1; 2539348784efSdrh } 25403e27c026Sdrh } 25413e27c026Sdrh if( argc<=1 || TCLSH==2 ){ 2542348784efSdrh Tcl_GlobalEval(interp, zMainloop); 2543348784efSdrh } 2544348784efSdrh return 0; 2545348784efSdrh } 2546348784efSdrh #endif /* TCLSH */ 2547