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*b4e9af9fSdanielk1977 ** $Id: tclsqlite.c,v 1.180 2007/05/01 17:49:49 danielk1977 Exp $ 1675897234Sdrh */ 17bd08af48Sdrh #include "tcl.h" 18*b4e9af9fSdanielk1977 #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" 26895d7472Sdrh # include "hash.h" 2775897234Sdrh # include <stdlib.h> 2875897234Sdrh # include <string.h> 29ce927065Sdrh # include <assert.h> 30d1e4733dSdrh # include <ctype.h> 31bd08af48Sdrh #endif 3275897234Sdrh 33ad6e1370Sdrh /* 34ad6e1370Sdrh * Windows needs to know which symbols to export. Unix does not. 35ad6e1370Sdrh * BUILD_sqlite should be undefined for Unix. 36ad6e1370Sdrh */ 37ad6e1370Sdrh #ifdef BUILD_sqlite 38ad6e1370Sdrh #undef TCL_STORAGE_CLASS 39ad6e1370Sdrh #define TCL_STORAGE_CLASS DLLEXPORT 40ad6e1370Sdrh #endif /* BUILD_sqlite */ 4129bc4615Sdrh 42a21c6b6fSdanielk1977 #define NUM_PREPARED_STMTS 10 43fb7e7651Sdrh #define MAX_PREPARED_STMTS 100 44fb7e7651Sdrh 4575897234Sdrh /* 4698808babSdrh ** If TCL uses UTF-8 and SQLite is configured to use iso8859, then we 4798808babSdrh ** have to do a translation when going between the two. Set the 4898808babSdrh ** UTF_TRANSLATION_NEEDED macro to indicate that we need to do 4998808babSdrh ** this translation. 5098808babSdrh */ 5198808babSdrh #if defined(TCL_UTF_MAX) && !defined(SQLITE_UTF8) 5298808babSdrh # define UTF_TRANSLATION_NEEDED 1 5398808babSdrh #endif 5498808babSdrh 5598808babSdrh /* 56cabb0819Sdrh ** New SQL functions can be created as TCL scripts. Each such function 57cabb0819Sdrh ** is described by an instance of the following structure. 58cabb0819Sdrh */ 59cabb0819Sdrh typedef struct SqlFunc SqlFunc; 60cabb0819Sdrh struct SqlFunc { 61cabb0819Sdrh Tcl_Interp *interp; /* The TCL interpret to execute the function */ 62d1e4733dSdrh Tcl_Obj *pScript; /* The Tcl_Obj representation of the script */ 63d1e4733dSdrh int useEvalObjv; /* True if it is safe to use Tcl_EvalObjv */ 64d1e4733dSdrh char *zName; /* Name of this function */ 65cabb0819Sdrh SqlFunc *pNext; /* Next function on the list of them all */ 66cabb0819Sdrh }; 67cabb0819Sdrh 68cabb0819Sdrh /* 690202b29eSdanielk1977 ** New collation sequences function can be created as TCL scripts. Each such 700202b29eSdanielk1977 ** function is described by an instance of the following structure. 710202b29eSdanielk1977 */ 720202b29eSdanielk1977 typedef struct SqlCollate SqlCollate; 730202b29eSdanielk1977 struct SqlCollate { 740202b29eSdanielk1977 Tcl_Interp *interp; /* The TCL interpret to execute the function */ 750202b29eSdanielk1977 char *zScript; /* The script to be run */ 760202b29eSdanielk1977 SqlCollate *pNext; /* Next function on the list of them all */ 770202b29eSdanielk1977 }; 780202b29eSdanielk1977 790202b29eSdanielk1977 /* 80fb7e7651Sdrh ** Prepared statements are cached for faster execution. Each prepared 81fb7e7651Sdrh ** statement is described by an instance of the following structure. 82fb7e7651Sdrh */ 83fb7e7651Sdrh typedef struct SqlPreparedStmt SqlPreparedStmt; 84fb7e7651Sdrh struct SqlPreparedStmt { 85fb7e7651Sdrh SqlPreparedStmt *pNext; /* Next in linked list */ 86fb7e7651Sdrh SqlPreparedStmt *pPrev; /* Previous on the list */ 87fb7e7651Sdrh sqlite3_stmt *pStmt; /* The prepared statement */ 88fb7e7651Sdrh int nSql; /* chars in zSql[] */ 89fb7e7651Sdrh char zSql[1]; /* Text of the SQL statement */ 90fb7e7651Sdrh }; 91fb7e7651Sdrh 92fb7e7651Sdrh /* 93bec3f402Sdrh ** There is one instance of this structure for each SQLite database 94bec3f402Sdrh ** that has been opened by the SQLite TCL interface. 95bec3f402Sdrh */ 96bec3f402Sdrh typedef struct SqliteDb SqliteDb; 97bec3f402Sdrh struct SqliteDb { 98dddca286Sdrh sqlite3 *db; /* The "real" database structure. MUST BE FIRST */ 99bec3f402Sdrh Tcl_Interp *interp; /* The interpreter used for this database */ 1006d31316cSdrh char *zBusy; /* The busy callback routine */ 101aa940eacSdrh char *zCommit; /* The commit hook callback routine */ 102b5a20d3cSdrh char *zTrace; /* The trace callback routine */ 10319e2d37fSdrh char *zProfile; /* The profile callback routine */ 104348bb5d6Sdanielk1977 char *zProgress; /* The progress callback routine */ 105e22a334bSdrh char *zAuth; /* The authorization callback routine */ 10655c45f2eSdanielk1977 char *zNull; /* Text to substitute for an SQL NULL value */ 107cabb0819Sdrh SqlFunc *pFunc; /* List of SQL functions */ 10894eb6a14Sdanielk1977 Tcl_Obj *pUpdateHook; /* Update hook script (if any) */ 10971fd80bfSdanielk1977 Tcl_Obj *pRollbackHook; /* Rollback hook script (if any) */ 1100202b29eSdanielk1977 SqlCollate *pCollate; /* List of SQL collation functions */ 1116f8a503dSdanielk1977 int rc; /* Return code of most recent sqlite3_exec() */ 1127cedc8d4Sdanielk1977 Tcl_Obj *pCollateNeeded; /* Collation needed script */ 113fb7e7651Sdrh SqlPreparedStmt *stmtList; /* List of prepared statements*/ 114fb7e7651Sdrh SqlPreparedStmt *stmtLast; /* Last statement in the list */ 115fb7e7651Sdrh int maxStmt; /* The next maximum number of stmtList */ 116fb7e7651Sdrh int nStmt; /* Number of statements in stmtList */ 11798808babSdrh }; 118297ecf14Sdrh 119*b4e9af9fSdanielk1977 typedef struct IncrblobChannel IncrblobChannel; 120*b4e9af9fSdanielk1977 struct IncrblobChannel { 121*b4e9af9fSdanielk1977 sqlite3_blob *pBlob; 122*b4e9af9fSdanielk1977 int iSeek; /* Current seek offset */ 123*b4e9af9fSdanielk1977 }; 124*b4e9af9fSdanielk1977 125*b4e9af9fSdanielk1977 /* 126*b4e9af9fSdanielk1977 ** Close an incremental blob channel. 127*b4e9af9fSdanielk1977 */ 128*b4e9af9fSdanielk1977 static int incrblobClose(ClientData instanceData, Tcl_Interp *interp){ 129*b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 130*b4e9af9fSdanielk1977 sqlite3_blob_close(p->pBlob); 131*b4e9af9fSdanielk1977 Tcl_Free((char *)p); 132*b4e9af9fSdanielk1977 return TCL_OK; 133*b4e9af9fSdanielk1977 } 134*b4e9af9fSdanielk1977 135*b4e9af9fSdanielk1977 /* 136*b4e9af9fSdanielk1977 ** Read data from an incremental blob channel. 137*b4e9af9fSdanielk1977 */ 138*b4e9af9fSdanielk1977 static int incrblobInput( 139*b4e9af9fSdanielk1977 ClientData instanceData, 140*b4e9af9fSdanielk1977 char *buf, 141*b4e9af9fSdanielk1977 int bufSize, 142*b4e9af9fSdanielk1977 int *errorCodePtr 143*b4e9af9fSdanielk1977 ){ 144*b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 145*b4e9af9fSdanielk1977 int nRead = bufSize; /* Number of bytes to read */ 146*b4e9af9fSdanielk1977 int nBlob; /* Total size of the blob */ 147*b4e9af9fSdanielk1977 int rc; /* sqlite error code */ 148*b4e9af9fSdanielk1977 149*b4e9af9fSdanielk1977 nBlob = sqlite3_blob_bytes(p->pBlob); 150*b4e9af9fSdanielk1977 if( (p->iSeek+nRead)>nBlob ){ 151*b4e9af9fSdanielk1977 nRead = nBlob-p->iSeek; 152*b4e9af9fSdanielk1977 } 153*b4e9af9fSdanielk1977 if( nRead<=0 ){ 154*b4e9af9fSdanielk1977 return 0; 155*b4e9af9fSdanielk1977 } 156*b4e9af9fSdanielk1977 157*b4e9af9fSdanielk1977 rc = sqlite3_blob_read(p->pBlob, (void *)buf, nRead, p->iSeek); 158*b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 159*b4e9af9fSdanielk1977 *errorCodePtr = rc; 160*b4e9af9fSdanielk1977 return -1; 161*b4e9af9fSdanielk1977 } 162*b4e9af9fSdanielk1977 163*b4e9af9fSdanielk1977 p->iSeek += nRead; 164*b4e9af9fSdanielk1977 return nRead; 165*b4e9af9fSdanielk1977 } 166*b4e9af9fSdanielk1977 167*b4e9af9fSdanielk1977 static int incrblobOutput( 168*b4e9af9fSdanielk1977 ClientData instanceData, 169*b4e9af9fSdanielk1977 CONST char *buf, 170*b4e9af9fSdanielk1977 int toWrite, 171*b4e9af9fSdanielk1977 int *errorCodePtr 172*b4e9af9fSdanielk1977 ){ 173*b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 174*b4e9af9fSdanielk1977 int nWrite = toWrite; /* Number of bytes to write */ 175*b4e9af9fSdanielk1977 int nBlob; /* Total size of the blob */ 176*b4e9af9fSdanielk1977 int rc; /* sqlite error code */ 177*b4e9af9fSdanielk1977 178*b4e9af9fSdanielk1977 nBlob = sqlite3_blob_bytes(p->pBlob); 179*b4e9af9fSdanielk1977 if( (p->iSeek+nWrite)>nBlob ){ 180*b4e9af9fSdanielk1977 *errorCodePtr = EINVAL; 181*b4e9af9fSdanielk1977 return -1; 182*b4e9af9fSdanielk1977 } 183*b4e9af9fSdanielk1977 if( nWrite<=0 ){ 184*b4e9af9fSdanielk1977 return 0; 185*b4e9af9fSdanielk1977 } 186*b4e9af9fSdanielk1977 187*b4e9af9fSdanielk1977 rc = sqlite3_blob_write(p->pBlob, (void *)buf, nWrite, p->iSeek); 188*b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 189*b4e9af9fSdanielk1977 *errorCodePtr = EIO; 190*b4e9af9fSdanielk1977 return -1; 191*b4e9af9fSdanielk1977 } 192*b4e9af9fSdanielk1977 193*b4e9af9fSdanielk1977 p->iSeek += nWrite; 194*b4e9af9fSdanielk1977 return nWrite; 195*b4e9af9fSdanielk1977 } 196*b4e9af9fSdanielk1977 197*b4e9af9fSdanielk1977 /* 198*b4e9af9fSdanielk1977 ** Seek an incremental blob channel. 199*b4e9af9fSdanielk1977 */ 200*b4e9af9fSdanielk1977 static int incrblobSeek( 201*b4e9af9fSdanielk1977 ClientData instanceData, 202*b4e9af9fSdanielk1977 long offset, 203*b4e9af9fSdanielk1977 int seekMode, 204*b4e9af9fSdanielk1977 int *errorCodePtr 205*b4e9af9fSdanielk1977 ){ 206*b4e9af9fSdanielk1977 IncrblobChannel *p = (IncrblobChannel *)instanceData; 207*b4e9af9fSdanielk1977 208*b4e9af9fSdanielk1977 switch( seekMode ){ 209*b4e9af9fSdanielk1977 case SEEK_SET: 210*b4e9af9fSdanielk1977 p->iSeek = offset; 211*b4e9af9fSdanielk1977 break; 212*b4e9af9fSdanielk1977 case SEEK_CUR: 213*b4e9af9fSdanielk1977 p->iSeek += offset; 214*b4e9af9fSdanielk1977 break; 215*b4e9af9fSdanielk1977 case SEEK_END: 216*b4e9af9fSdanielk1977 p->iSeek = sqlite3_blob_bytes(p->pBlob) + offset; 217*b4e9af9fSdanielk1977 break; 218*b4e9af9fSdanielk1977 219*b4e9af9fSdanielk1977 default: assert(!"Bad seekMode"); 220*b4e9af9fSdanielk1977 } 221*b4e9af9fSdanielk1977 222*b4e9af9fSdanielk1977 return p->iSeek; 223*b4e9af9fSdanielk1977 } 224*b4e9af9fSdanielk1977 225*b4e9af9fSdanielk1977 226*b4e9af9fSdanielk1977 static void incrblobWatch(ClientData instanceData, int mode){ 227*b4e9af9fSdanielk1977 /* NO-OP */ 228*b4e9af9fSdanielk1977 } 229*b4e9af9fSdanielk1977 static int incrblobHandle(ClientData instanceData, int dir, ClientData *hPtr){ 230*b4e9af9fSdanielk1977 return TCL_ERROR; 231*b4e9af9fSdanielk1977 } 232*b4e9af9fSdanielk1977 233*b4e9af9fSdanielk1977 static Tcl_ChannelType IncrblobChannelType = { 234*b4e9af9fSdanielk1977 "incrblob", /* typeName */ 235*b4e9af9fSdanielk1977 TCL_CHANNEL_VERSION_2, /* version */ 236*b4e9af9fSdanielk1977 incrblobClose, /* closeProc */ 237*b4e9af9fSdanielk1977 incrblobInput, /* inputProc */ 238*b4e9af9fSdanielk1977 incrblobOutput, /* outputProc */ 239*b4e9af9fSdanielk1977 incrblobSeek, /* seekProc */ 240*b4e9af9fSdanielk1977 0, /* setOptionProc */ 241*b4e9af9fSdanielk1977 0, /* getOptionProc */ 242*b4e9af9fSdanielk1977 incrblobWatch, /* watchProc (this is a no-op) */ 243*b4e9af9fSdanielk1977 incrblobHandle, /* getHandleProc (always returns error) */ 244*b4e9af9fSdanielk1977 0, /* close2Proc */ 245*b4e9af9fSdanielk1977 0, /* blockModeProc */ 246*b4e9af9fSdanielk1977 0, /* flushProc */ 247*b4e9af9fSdanielk1977 0, /* handlerProc */ 248*b4e9af9fSdanielk1977 0, /* wideSeekProc */ 249*b4e9af9fSdanielk1977 0, /* threadActionProc */ 250*b4e9af9fSdanielk1977 }; 251*b4e9af9fSdanielk1977 252*b4e9af9fSdanielk1977 /* 253*b4e9af9fSdanielk1977 ** Create a new incrblob channel. 254*b4e9af9fSdanielk1977 */ 255*b4e9af9fSdanielk1977 static int createIncrblobChannel( 256*b4e9af9fSdanielk1977 Tcl_Interp *interp, 257*b4e9af9fSdanielk1977 SqliteDb *pDb, 258*b4e9af9fSdanielk1977 const char *zDb, 259*b4e9af9fSdanielk1977 const char *zTable, 260*b4e9af9fSdanielk1977 const char *zColumn, 261*b4e9af9fSdanielk1977 sqlite_int64 iRow 262*b4e9af9fSdanielk1977 ){ 263*b4e9af9fSdanielk1977 IncrblobChannel *p; 264*b4e9af9fSdanielk1977 sqlite3_blob *pBlob; 265*b4e9af9fSdanielk1977 int rc; 266*b4e9af9fSdanielk1977 Tcl_Channel channel; 267*b4e9af9fSdanielk1977 int flags = TCL_READABLE|TCL_WRITABLE; 268*b4e9af9fSdanielk1977 269*b4e9af9fSdanielk1977 /* This variable is used to name the channels: "incrblob_[incr count]" */ 270*b4e9af9fSdanielk1977 static int count = 0; 271*b4e9af9fSdanielk1977 char zChannel[64]; 272*b4e9af9fSdanielk1977 273*b4e9af9fSdanielk1977 rc = sqlite3_blob_open(pDb->db, zDb, zTable, zColumn, iRow, 1, &pBlob); 274*b4e9af9fSdanielk1977 if( rc!=SQLITE_OK ){ 275*b4e9af9fSdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 276*b4e9af9fSdanielk1977 return TCL_ERROR; 277*b4e9af9fSdanielk1977 } 278*b4e9af9fSdanielk1977 279*b4e9af9fSdanielk1977 p = (IncrblobChannel *)Tcl_Alloc(sizeof(IncrblobChannel)); 280*b4e9af9fSdanielk1977 p->iSeek = 0; 281*b4e9af9fSdanielk1977 p->pBlob = pBlob; 282*b4e9af9fSdanielk1977 283*b4e9af9fSdanielk1977 sprintf(zChannel, "incrblob_%d", ++count); 284*b4e9af9fSdanielk1977 channel = Tcl_CreateChannel(&IncrblobChannelType, zChannel, p, flags); 285*b4e9af9fSdanielk1977 Tcl_RegisterChannel(interp, channel); 286*b4e9af9fSdanielk1977 287*b4e9af9fSdanielk1977 Tcl_SetResult(interp, (char *)Tcl_GetChannelName(channel), TCL_VOLATILE); 288*b4e9af9fSdanielk1977 return TCL_OK; 289*b4e9af9fSdanielk1977 } 290*b4e9af9fSdanielk1977 2916d31316cSdrh /* 292d1e4733dSdrh ** Look at the script prefix in pCmd. We will be executing this script 293d1e4733dSdrh ** after first appending one or more arguments. This routine analyzes 294d1e4733dSdrh ** the script to see if it is safe to use Tcl_EvalObjv() on the script 295d1e4733dSdrh ** rather than the more general Tcl_EvalEx(). Tcl_EvalObjv() is much 296d1e4733dSdrh ** faster. 297d1e4733dSdrh ** 298d1e4733dSdrh ** Scripts that are safe to use with Tcl_EvalObjv() consists of a 299d1e4733dSdrh ** command name followed by zero or more arguments with no [...] or $ 300d1e4733dSdrh ** or {...} or ; to be seen anywhere. Most callback scripts consist 301d1e4733dSdrh ** of just a single procedure name and they meet this requirement. 302d1e4733dSdrh */ 303d1e4733dSdrh static int safeToUseEvalObjv(Tcl_Interp *interp, Tcl_Obj *pCmd){ 304d1e4733dSdrh /* We could try to do something with Tcl_Parse(). But we will instead 305d1e4733dSdrh ** just do a search for forbidden characters. If any of the forbidden 306d1e4733dSdrh ** characters appear in pCmd, we will report the string as unsafe. 307d1e4733dSdrh */ 308d1e4733dSdrh const char *z; 309d1e4733dSdrh int n; 310d1e4733dSdrh z = Tcl_GetStringFromObj(pCmd, &n); 311d1e4733dSdrh while( n-- > 0 ){ 312d1e4733dSdrh int c = *(z++); 313d1e4733dSdrh if( c=='$' || c=='[' || c==';' ) return 0; 314d1e4733dSdrh } 315d1e4733dSdrh return 1; 316d1e4733dSdrh } 317d1e4733dSdrh 318d1e4733dSdrh /* 319d1e4733dSdrh ** Find an SqlFunc structure with the given name. Or create a new 320d1e4733dSdrh ** one if an existing one cannot be found. Return a pointer to the 321d1e4733dSdrh ** structure. 322d1e4733dSdrh */ 323d1e4733dSdrh static SqlFunc *findSqlFunc(SqliteDb *pDb, const char *zName){ 324d1e4733dSdrh SqlFunc *p, *pNew; 325d1e4733dSdrh int i; 326d1e4733dSdrh pNew = (SqlFunc*)Tcl_Alloc( sizeof(*pNew) + strlen(zName) + 1 ); 327d1e4733dSdrh pNew->zName = (char*)&pNew[1]; 328d1e4733dSdrh for(i=0; zName[i]; i++){ pNew->zName[i] = tolower(zName[i]); } 329d1e4733dSdrh pNew->zName[i] = 0; 330d1e4733dSdrh for(p=pDb->pFunc; p; p=p->pNext){ 331d1e4733dSdrh if( strcmp(p->zName, pNew->zName)==0 ){ 332d1e4733dSdrh Tcl_Free((char*)pNew); 333d1e4733dSdrh return p; 334d1e4733dSdrh } 335d1e4733dSdrh } 336d1e4733dSdrh pNew->interp = pDb->interp; 337d1e4733dSdrh pNew->pScript = 0; 338d1e4733dSdrh pNew->pNext = pDb->pFunc; 339d1e4733dSdrh pDb->pFunc = pNew; 340d1e4733dSdrh return pNew; 341d1e4733dSdrh } 342d1e4733dSdrh 343d1e4733dSdrh /* 344fb7e7651Sdrh ** Finalize and free a list of prepared statements 345fb7e7651Sdrh */ 346fb7e7651Sdrh static void flushStmtCache( SqliteDb *pDb ){ 347fb7e7651Sdrh SqlPreparedStmt *pPreStmt; 348fb7e7651Sdrh 349fb7e7651Sdrh while( pDb->stmtList ){ 350fb7e7651Sdrh sqlite3_finalize( pDb->stmtList->pStmt ); 351fb7e7651Sdrh pPreStmt = pDb->stmtList; 352fb7e7651Sdrh pDb->stmtList = pDb->stmtList->pNext; 353fb7e7651Sdrh Tcl_Free( (char*)pPreStmt ); 354fb7e7651Sdrh } 355fb7e7651Sdrh pDb->nStmt = 0; 356fb7e7651Sdrh pDb->stmtLast = 0; 357fb7e7651Sdrh } 358fb7e7651Sdrh 359fb7e7651Sdrh /* 360895d7472Sdrh ** TCL calls this procedure when an sqlite3 database command is 361895d7472Sdrh ** deleted. 36275897234Sdrh */ 36375897234Sdrh static void DbDeleteCmd(void *db){ 364bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)db; 365fb7e7651Sdrh flushStmtCache(pDb); 3666f8a503dSdanielk1977 sqlite3_close(pDb->db); 367cabb0819Sdrh while( pDb->pFunc ){ 368cabb0819Sdrh SqlFunc *pFunc = pDb->pFunc; 369cabb0819Sdrh pDb->pFunc = pFunc->pNext; 370d1e4733dSdrh Tcl_DecrRefCount(pFunc->pScript); 371cabb0819Sdrh Tcl_Free((char*)pFunc); 372cabb0819Sdrh } 3730202b29eSdanielk1977 while( pDb->pCollate ){ 3740202b29eSdanielk1977 SqlCollate *pCollate = pDb->pCollate; 3750202b29eSdanielk1977 pDb->pCollate = pCollate->pNext; 3760202b29eSdanielk1977 Tcl_Free((char*)pCollate); 3770202b29eSdanielk1977 } 378bec3f402Sdrh if( pDb->zBusy ){ 379bec3f402Sdrh Tcl_Free(pDb->zBusy); 380bec3f402Sdrh } 381b5a20d3cSdrh if( pDb->zTrace ){ 382b5a20d3cSdrh Tcl_Free(pDb->zTrace); 3830d1a643aSdrh } 38419e2d37fSdrh if( pDb->zProfile ){ 38519e2d37fSdrh Tcl_Free(pDb->zProfile); 38619e2d37fSdrh } 387e22a334bSdrh if( pDb->zAuth ){ 388e22a334bSdrh Tcl_Free(pDb->zAuth); 389e22a334bSdrh } 39055c45f2eSdanielk1977 if( pDb->zNull ){ 39155c45f2eSdanielk1977 Tcl_Free(pDb->zNull); 39255c45f2eSdanielk1977 } 39394eb6a14Sdanielk1977 if( pDb->pUpdateHook ){ 39494eb6a14Sdanielk1977 Tcl_DecrRefCount(pDb->pUpdateHook); 39594eb6a14Sdanielk1977 } 39671fd80bfSdanielk1977 if( pDb->pRollbackHook ){ 39771fd80bfSdanielk1977 Tcl_DecrRefCount(pDb->pRollbackHook); 39871fd80bfSdanielk1977 } 39994eb6a14Sdanielk1977 if( pDb->pCollateNeeded ){ 40094eb6a14Sdanielk1977 Tcl_DecrRefCount(pDb->pCollateNeeded); 40194eb6a14Sdanielk1977 } 402bec3f402Sdrh Tcl_Free((char*)pDb); 403bec3f402Sdrh } 404bec3f402Sdrh 405bec3f402Sdrh /* 406bec3f402Sdrh ** This routine is called when a database file is locked while trying 407bec3f402Sdrh ** to execute SQL. 408bec3f402Sdrh */ 4092a764eb0Sdanielk1977 static int DbBusyHandler(void *cd, int nTries){ 410bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)cd; 411bec3f402Sdrh int rc; 412bec3f402Sdrh char zVal[30]; 413bec3f402Sdrh 414bec3f402Sdrh sprintf(zVal, "%d", nTries); 415d1e4733dSdrh rc = Tcl_VarEval(pDb->interp, pDb->zBusy, " ", zVal, (char*)0); 416bec3f402Sdrh if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 417bec3f402Sdrh return 0; 418bec3f402Sdrh } 419bec3f402Sdrh return 1; 42075897234Sdrh } 42175897234Sdrh 42275897234Sdrh /* 423348bb5d6Sdanielk1977 ** This routine is invoked as the 'progress callback' for the database. 424348bb5d6Sdanielk1977 */ 425348bb5d6Sdanielk1977 static int DbProgressHandler(void *cd){ 426348bb5d6Sdanielk1977 SqliteDb *pDb = (SqliteDb*)cd; 427348bb5d6Sdanielk1977 int rc; 428348bb5d6Sdanielk1977 429348bb5d6Sdanielk1977 assert( pDb->zProgress ); 430348bb5d6Sdanielk1977 rc = Tcl_Eval(pDb->interp, pDb->zProgress); 431348bb5d6Sdanielk1977 if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 432348bb5d6Sdanielk1977 return 1; 433348bb5d6Sdanielk1977 } 434348bb5d6Sdanielk1977 return 0; 435348bb5d6Sdanielk1977 } 436348bb5d6Sdanielk1977 437d1167393Sdrh #ifndef SQLITE_OMIT_TRACE 438348bb5d6Sdanielk1977 /* 439b5a20d3cSdrh ** This routine is called by the SQLite trace handler whenever a new 440b5a20d3cSdrh ** block of SQL is executed. The TCL script in pDb->zTrace is executed. 4410d1a643aSdrh */ 442b5a20d3cSdrh static void DbTraceHandler(void *cd, const char *zSql){ 4430d1a643aSdrh SqliteDb *pDb = (SqliteDb*)cd; 444b5a20d3cSdrh Tcl_DString str; 4450d1a643aSdrh 446b5a20d3cSdrh Tcl_DStringInit(&str); 447b5a20d3cSdrh Tcl_DStringAppend(&str, pDb->zTrace, -1); 448b5a20d3cSdrh Tcl_DStringAppendElement(&str, zSql); 449b5a20d3cSdrh Tcl_Eval(pDb->interp, Tcl_DStringValue(&str)); 450b5a20d3cSdrh Tcl_DStringFree(&str); 451b5a20d3cSdrh Tcl_ResetResult(pDb->interp); 4520d1a643aSdrh } 453d1167393Sdrh #endif 4540d1a643aSdrh 455d1167393Sdrh #ifndef SQLITE_OMIT_TRACE 4560d1a643aSdrh /* 45719e2d37fSdrh ** This routine is called by the SQLite profile handler after a statement 45819e2d37fSdrh ** SQL has executed. The TCL script in pDb->zProfile is evaluated. 45919e2d37fSdrh */ 46019e2d37fSdrh static void DbProfileHandler(void *cd, const char *zSql, sqlite_uint64 tm){ 46119e2d37fSdrh SqliteDb *pDb = (SqliteDb*)cd; 46219e2d37fSdrh Tcl_DString str; 46319e2d37fSdrh char zTm[100]; 46419e2d37fSdrh 46519e2d37fSdrh sqlite3_snprintf(sizeof(zTm)-1, zTm, "%lld", tm); 46619e2d37fSdrh Tcl_DStringInit(&str); 46719e2d37fSdrh Tcl_DStringAppend(&str, pDb->zProfile, -1); 46819e2d37fSdrh Tcl_DStringAppendElement(&str, zSql); 46919e2d37fSdrh Tcl_DStringAppendElement(&str, zTm); 47019e2d37fSdrh Tcl_Eval(pDb->interp, Tcl_DStringValue(&str)); 47119e2d37fSdrh Tcl_DStringFree(&str); 47219e2d37fSdrh Tcl_ResetResult(pDb->interp); 47319e2d37fSdrh } 474d1167393Sdrh #endif 47519e2d37fSdrh 47619e2d37fSdrh /* 477aa940eacSdrh ** This routine is called when a transaction is committed. The 478aa940eacSdrh ** TCL script in pDb->zCommit is executed. If it returns non-zero or 479aa940eacSdrh ** if it throws an exception, the transaction is rolled back instead 480aa940eacSdrh ** of being committed. 481aa940eacSdrh */ 482aa940eacSdrh static int DbCommitHandler(void *cd){ 483aa940eacSdrh SqliteDb *pDb = (SqliteDb*)cd; 484aa940eacSdrh int rc; 485aa940eacSdrh 486aa940eacSdrh rc = Tcl_Eval(pDb->interp, pDb->zCommit); 487aa940eacSdrh if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){ 488aa940eacSdrh return 1; 489aa940eacSdrh } 490aa940eacSdrh return 0; 491aa940eacSdrh } 492aa940eacSdrh 49371fd80bfSdanielk1977 static void DbRollbackHandler(void *clientData){ 49471fd80bfSdanielk1977 SqliteDb *pDb = (SqliteDb*)clientData; 49571fd80bfSdanielk1977 assert(pDb->pRollbackHook); 49671fd80bfSdanielk1977 if( TCL_OK!=Tcl_EvalObjEx(pDb->interp, pDb->pRollbackHook, 0) ){ 49771fd80bfSdanielk1977 Tcl_BackgroundError(pDb->interp); 49871fd80bfSdanielk1977 } 49971fd80bfSdanielk1977 } 50071fd80bfSdanielk1977 50194eb6a14Sdanielk1977 static void DbUpdateHandler( 50294eb6a14Sdanielk1977 void *p, 50394eb6a14Sdanielk1977 int op, 50494eb6a14Sdanielk1977 const char *zDb, 50594eb6a14Sdanielk1977 const char *zTbl, 50694eb6a14Sdanielk1977 sqlite_int64 rowid 50794eb6a14Sdanielk1977 ){ 50894eb6a14Sdanielk1977 SqliteDb *pDb = (SqliteDb *)p; 50994eb6a14Sdanielk1977 Tcl_Obj *pCmd; 51094eb6a14Sdanielk1977 51194eb6a14Sdanielk1977 assert( pDb->pUpdateHook ); 51294eb6a14Sdanielk1977 assert( op==SQLITE_INSERT || op==SQLITE_UPDATE || op==SQLITE_DELETE ); 51394eb6a14Sdanielk1977 51494eb6a14Sdanielk1977 pCmd = Tcl_DuplicateObj(pDb->pUpdateHook); 51594eb6a14Sdanielk1977 Tcl_IncrRefCount(pCmd); 51694eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj( 51794eb6a14Sdanielk1977 ( (op==SQLITE_INSERT)?"INSERT":(op==SQLITE_UPDATE)?"UPDATE":"DELETE"), -1)); 51894eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj(zDb, -1)); 51994eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewStringObj(zTbl, -1)); 52094eb6a14Sdanielk1977 Tcl_ListObjAppendElement(0, pCmd, Tcl_NewWideIntObj(rowid)); 52194eb6a14Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pCmd, TCL_EVAL_DIRECT); 52294eb6a14Sdanielk1977 } 52394eb6a14Sdanielk1977 5247cedc8d4Sdanielk1977 static void tclCollateNeeded( 5257cedc8d4Sdanielk1977 void *pCtx, 5269bb575fdSdrh sqlite3 *db, 5277cedc8d4Sdanielk1977 int enc, 5287cedc8d4Sdanielk1977 const char *zName 5297cedc8d4Sdanielk1977 ){ 5307cedc8d4Sdanielk1977 SqliteDb *pDb = (SqliteDb *)pCtx; 5317cedc8d4Sdanielk1977 Tcl_Obj *pScript = Tcl_DuplicateObj(pDb->pCollateNeeded); 5327cedc8d4Sdanielk1977 Tcl_IncrRefCount(pScript); 5337cedc8d4Sdanielk1977 Tcl_ListObjAppendElement(0, pScript, Tcl_NewStringObj(zName, -1)); 5347cedc8d4Sdanielk1977 Tcl_EvalObjEx(pDb->interp, pScript, 0); 5357cedc8d4Sdanielk1977 Tcl_DecrRefCount(pScript); 5367cedc8d4Sdanielk1977 } 5377cedc8d4Sdanielk1977 538aa940eacSdrh /* 5390202b29eSdanielk1977 ** This routine is called to evaluate an SQL collation function implemented 5400202b29eSdanielk1977 ** using TCL script. 5410202b29eSdanielk1977 */ 5420202b29eSdanielk1977 static int tclSqlCollate( 5430202b29eSdanielk1977 void *pCtx, 5440202b29eSdanielk1977 int nA, 5450202b29eSdanielk1977 const void *zA, 5460202b29eSdanielk1977 int nB, 5470202b29eSdanielk1977 const void *zB 5480202b29eSdanielk1977 ){ 5490202b29eSdanielk1977 SqlCollate *p = (SqlCollate *)pCtx; 5500202b29eSdanielk1977 Tcl_Obj *pCmd; 5510202b29eSdanielk1977 5520202b29eSdanielk1977 pCmd = Tcl_NewStringObj(p->zScript, -1); 5530202b29eSdanielk1977 Tcl_IncrRefCount(pCmd); 5540202b29eSdanielk1977 Tcl_ListObjAppendElement(p->interp, pCmd, Tcl_NewStringObj(zA, nA)); 5550202b29eSdanielk1977 Tcl_ListObjAppendElement(p->interp, pCmd, Tcl_NewStringObj(zB, nB)); 556d1e4733dSdrh Tcl_EvalObjEx(p->interp, pCmd, TCL_EVAL_DIRECT); 5570202b29eSdanielk1977 Tcl_DecrRefCount(pCmd); 5580202b29eSdanielk1977 return (atoi(Tcl_GetStringResult(p->interp))); 5590202b29eSdanielk1977 } 5600202b29eSdanielk1977 5610202b29eSdanielk1977 /* 562cabb0819Sdrh ** This routine is called to evaluate an SQL function implemented 563cabb0819Sdrh ** using TCL script. 564cabb0819Sdrh */ 5650ae8b831Sdanielk1977 static void tclSqlFunc(sqlite3_context *context, int argc, sqlite3_value**argv){ 5666f8a503dSdanielk1977 SqlFunc *p = sqlite3_user_data(context); 567d1e4733dSdrh Tcl_Obj *pCmd; 568cabb0819Sdrh int i; 569cabb0819Sdrh int rc; 570cabb0819Sdrh 571d1e4733dSdrh if( argc==0 ){ 572d1e4733dSdrh /* If there are no arguments to the function, call Tcl_EvalObjEx on the 573d1e4733dSdrh ** script object directly. This allows the TCL compiler to generate 574d1e4733dSdrh ** bytecode for the command on the first invocation and thus make 575d1e4733dSdrh ** subsequent invocations much faster. */ 576d1e4733dSdrh pCmd = p->pScript; 577d1e4733dSdrh Tcl_IncrRefCount(pCmd); 578d1e4733dSdrh rc = Tcl_EvalObjEx(p->interp, pCmd, 0); 579d1e4733dSdrh Tcl_DecrRefCount(pCmd); 58051ad0ecdSdanielk1977 }else{ 581d1e4733dSdrh /* If there are arguments to the function, make a shallow copy of the 582d1e4733dSdrh ** script object, lappend the arguments, then evaluate the copy. 583d1e4733dSdrh ** 584d1e4733dSdrh ** By "shallow" copy, we mean a only the outer list Tcl_Obj is duplicated. 585d1e4733dSdrh ** The new Tcl_Obj contains pointers to the original list elements. 586d1e4733dSdrh ** That way, when Tcl_EvalObjv() is run and shimmers the first element 587d1e4733dSdrh ** of the list to tclCmdNameType, that alternate representation will 588d1e4733dSdrh ** be preserved and reused on the next invocation. 589d1e4733dSdrh */ 590d1e4733dSdrh Tcl_Obj **aArg; 591d1e4733dSdrh int nArg; 592d1e4733dSdrh if( Tcl_ListObjGetElements(p->interp, p->pScript, &nArg, &aArg) ){ 593d1e4733dSdrh sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 594d1e4733dSdrh return; 595d1e4733dSdrh } 596d1e4733dSdrh pCmd = Tcl_NewListObj(nArg, aArg); 597d1e4733dSdrh Tcl_IncrRefCount(pCmd); 598d1e4733dSdrh for(i=0; i<argc; i++){ 599d1e4733dSdrh sqlite3_value *pIn = argv[i]; 600d1e4733dSdrh Tcl_Obj *pVal; 601d1e4733dSdrh 602d1e4733dSdrh /* Set pVal to contain the i'th column of this row. */ 603d1e4733dSdrh switch( sqlite3_value_type(pIn) ){ 604d1e4733dSdrh case SQLITE_BLOB: { 605d1e4733dSdrh int bytes = sqlite3_value_bytes(pIn); 606d1e4733dSdrh pVal = Tcl_NewByteArrayObj(sqlite3_value_blob(pIn), bytes); 607d1e4733dSdrh break; 608d1e4733dSdrh } 609d1e4733dSdrh case SQLITE_INTEGER: { 610d1e4733dSdrh sqlite_int64 v = sqlite3_value_int64(pIn); 611d1e4733dSdrh if( v>=-2147483647 && v<=2147483647 ){ 612d1e4733dSdrh pVal = Tcl_NewIntObj(v); 613d1e4733dSdrh }else{ 614d1e4733dSdrh pVal = Tcl_NewWideIntObj(v); 615d1e4733dSdrh } 616d1e4733dSdrh break; 617d1e4733dSdrh } 618d1e4733dSdrh case SQLITE_FLOAT: { 619d1e4733dSdrh double r = sqlite3_value_double(pIn); 620d1e4733dSdrh pVal = Tcl_NewDoubleObj(r); 621d1e4733dSdrh break; 622d1e4733dSdrh } 623d1e4733dSdrh case SQLITE_NULL: { 624d1e4733dSdrh pVal = Tcl_NewStringObj("", 0); 625d1e4733dSdrh break; 626d1e4733dSdrh } 627d1e4733dSdrh default: { 628d1e4733dSdrh int bytes = sqlite3_value_bytes(pIn); 62900fd957bSdanielk1977 pVal = Tcl_NewStringObj((char *)sqlite3_value_text(pIn), bytes); 630d1e4733dSdrh break; 63151ad0ecdSdanielk1977 } 632cabb0819Sdrh } 633d1e4733dSdrh rc = Tcl_ListObjAppendElement(p->interp, pCmd, pVal); 634d1e4733dSdrh if( rc ){ 635d1e4733dSdrh Tcl_DecrRefCount(pCmd); 636d1e4733dSdrh sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 637d1e4733dSdrh return; 638d1e4733dSdrh } 639d1e4733dSdrh } 640d1e4733dSdrh if( !p->useEvalObjv ){ 641d1e4733dSdrh /* Tcl_EvalObjEx() will automatically call Tcl_EvalObjv() if pCmd 642d1e4733dSdrh ** is a list without a string representation. To prevent this from 643d1e4733dSdrh ** happening, make sure pCmd has a valid string representation */ 644d1e4733dSdrh Tcl_GetString(pCmd); 645d1e4733dSdrh } 646d1e4733dSdrh rc = Tcl_EvalObjEx(p->interp, pCmd, TCL_EVAL_DIRECT); 647d1e4733dSdrh Tcl_DecrRefCount(pCmd); 648d1e4733dSdrh } 649562e8d3cSdanielk1977 650c7f269d5Sdrh if( rc && rc!=TCL_RETURN ){ 6517e18c259Sdanielk1977 sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1); 652cabb0819Sdrh }else{ 653c7f269d5Sdrh Tcl_Obj *pVar = Tcl_GetObjResult(p->interp); 654c7f269d5Sdrh int n; 655c7f269d5Sdrh u8 *data; 656c7f269d5Sdrh char *zType = pVar->typePtr ? pVar->typePtr->name : ""; 657c7f269d5Sdrh char c = zType[0]; 658df0bddaeSdrh if( c=='b' && strcmp(zType,"bytearray")==0 && pVar->bytes==0 ){ 659d1e4733dSdrh /* Only return a BLOB type if the Tcl variable is a bytearray and 660df0bddaeSdrh ** has no string representation. */ 661c7f269d5Sdrh data = Tcl_GetByteArrayFromObj(pVar, &n); 662c7f269d5Sdrh sqlite3_result_blob(context, data, n, SQLITE_TRANSIENT); 663c7f269d5Sdrh }else if( (c=='b' && strcmp(zType,"boolean")==0) || 664c7f269d5Sdrh (c=='i' && strcmp(zType,"int")==0) ){ 665c7f269d5Sdrh Tcl_GetIntFromObj(0, pVar, &n); 666c7f269d5Sdrh sqlite3_result_int(context, n); 667c7f269d5Sdrh }else if( c=='d' && strcmp(zType,"double")==0 ){ 668c7f269d5Sdrh double r; 669c7f269d5Sdrh Tcl_GetDoubleFromObj(0, pVar, &r); 670c7f269d5Sdrh sqlite3_result_double(context, r); 671df0bddaeSdrh }else if( c=='w' && strcmp(zType,"wideInt")==0 ){ 672df0bddaeSdrh Tcl_WideInt v; 673df0bddaeSdrh Tcl_GetWideIntFromObj(0, pVar, &v); 674df0bddaeSdrh sqlite3_result_int64(context, v); 675c7f269d5Sdrh }else{ 67600fd957bSdanielk1977 data = (unsigned char *)Tcl_GetStringFromObj(pVar, &n); 67700fd957bSdanielk1977 sqlite3_result_text(context, (char *)data, n, SQLITE_TRANSIENT); 678c7f269d5Sdrh } 679cabb0819Sdrh } 680cabb0819Sdrh } 681895d7472Sdrh 682e22a334bSdrh #ifndef SQLITE_OMIT_AUTHORIZATION 683e22a334bSdrh /* 684e22a334bSdrh ** This is the authentication function. It appends the authentication 685e22a334bSdrh ** type code and the two arguments to zCmd[] then invokes the result 686e22a334bSdrh ** on the interpreter. The reply is examined to determine if the 687e22a334bSdrh ** authentication fails or succeeds. 688e22a334bSdrh */ 689e22a334bSdrh static int auth_callback( 690e22a334bSdrh void *pArg, 691e22a334bSdrh int code, 692e22a334bSdrh const char *zArg1, 693e22a334bSdrh const char *zArg2, 694e22a334bSdrh const char *zArg3, 695e22a334bSdrh const char *zArg4 696e22a334bSdrh ){ 697e22a334bSdrh char *zCode; 698e22a334bSdrh Tcl_DString str; 699e22a334bSdrh int rc; 700e22a334bSdrh const char *zReply; 701e22a334bSdrh SqliteDb *pDb = (SqliteDb*)pArg; 702e22a334bSdrh 703e22a334bSdrh switch( code ){ 704e22a334bSdrh case SQLITE_COPY : zCode="SQLITE_COPY"; break; 705e22a334bSdrh case SQLITE_CREATE_INDEX : zCode="SQLITE_CREATE_INDEX"; break; 706e22a334bSdrh case SQLITE_CREATE_TABLE : zCode="SQLITE_CREATE_TABLE"; break; 707e22a334bSdrh case SQLITE_CREATE_TEMP_INDEX : zCode="SQLITE_CREATE_TEMP_INDEX"; break; 708e22a334bSdrh case SQLITE_CREATE_TEMP_TABLE : zCode="SQLITE_CREATE_TEMP_TABLE"; break; 709e22a334bSdrh case SQLITE_CREATE_TEMP_TRIGGER: zCode="SQLITE_CREATE_TEMP_TRIGGER"; break; 710e22a334bSdrh case SQLITE_CREATE_TEMP_VIEW : zCode="SQLITE_CREATE_TEMP_VIEW"; break; 711e22a334bSdrh case SQLITE_CREATE_TRIGGER : zCode="SQLITE_CREATE_TRIGGER"; break; 712e22a334bSdrh case SQLITE_CREATE_VIEW : zCode="SQLITE_CREATE_VIEW"; break; 713e22a334bSdrh case SQLITE_DELETE : zCode="SQLITE_DELETE"; break; 714e22a334bSdrh case SQLITE_DROP_INDEX : zCode="SQLITE_DROP_INDEX"; break; 715e22a334bSdrh case SQLITE_DROP_TABLE : zCode="SQLITE_DROP_TABLE"; break; 716e22a334bSdrh case SQLITE_DROP_TEMP_INDEX : zCode="SQLITE_DROP_TEMP_INDEX"; break; 717e22a334bSdrh case SQLITE_DROP_TEMP_TABLE : zCode="SQLITE_DROP_TEMP_TABLE"; break; 718e22a334bSdrh case SQLITE_DROP_TEMP_TRIGGER : zCode="SQLITE_DROP_TEMP_TRIGGER"; break; 719e22a334bSdrh case SQLITE_DROP_TEMP_VIEW : zCode="SQLITE_DROP_TEMP_VIEW"; break; 720e22a334bSdrh case SQLITE_DROP_TRIGGER : zCode="SQLITE_DROP_TRIGGER"; break; 721e22a334bSdrh case SQLITE_DROP_VIEW : zCode="SQLITE_DROP_VIEW"; break; 722e22a334bSdrh case SQLITE_INSERT : zCode="SQLITE_INSERT"; break; 723e22a334bSdrh case SQLITE_PRAGMA : zCode="SQLITE_PRAGMA"; break; 724e22a334bSdrh case SQLITE_READ : zCode="SQLITE_READ"; break; 725e22a334bSdrh case SQLITE_SELECT : zCode="SQLITE_SELECT"; break; 726e22a334bSdrh case SQLITE_TRANSACTION : zCode="SQLITE_TRANSACTION"; break; 727e22a334bSdrh case SQLITE_UPDATE : zCode="SQLITE_UPDATE"; break; 72881e293b4Sdrh case SQLITE_ATTACH : zCode="SQLITE_ATTACH"; break; 72981e293b4Sdrh case SQLITE_DETACH : zCode="SQLITE_DETACH"; break; 7301c8c23ccSdanielk1977 case SQLITE_ALTER_TABLE : zCode="SQLITE_ALTER_TABLE"; break; 7311d54df88Sdanielk1977 case SQLITE_REINDEX : zCode="SQLITE_REINDEX"; break; 732e6e04969Sdrh case SQLITE_ANALYZE : zCode="SQLITE_ANALYZE"; break; 733f1a381e7Sdanielk1977 case SQLITE_CREATE_VTABLE : zCode="SQLITE_CREATE_VTABLE"; break; 734f1a381e7Sdanielk1977 case SQLITE_DROP_VTABLE : zCode="SQLITE_DROP_VTABLE"; break; 7355169bbc6Sdrh case SQLITE_FUNCTION : zCode="SQLITE_FUNCTION"; break; 736e22a334bSdrh default : zCode="????"; break; 737e22a334bSdrh } 738e22a334bSdrh Tcl_DStringInit(&str); 739e22a334bSdrh Tcl_DStringAppend(&str, pDb->zAuth, -1); 740e22a334bSdrh Tcl_DStringAppendElement(&str, zCode); 741e22a334bSdrh Tcl_DStringAppendElement(&str, zArg1 ? zArg1 : ""); 742e22a334bSdrh Tcl_DStringAppendElement(&str, zArg2 ? zArg2 : ""); 743e22a334bSdrh Tcl_DStringAppendElement(&str, zArg3 ? zArg3 : ""); 744e22a334bSdrh Tcl_DStringAppendElement(&str, zArg4 ? zArg4 : ""); 745e22a334bSdrh rc = Tcl_GlobalEval(pDb->interp, Tcl_DStringValue(&str)); 746e22a334bSdrh Tcl_DStringFree(&str); 747e22a334bSdrh zReply = Tcl_GetStringResult(pDb->interp); 748e22a334bSdrh if( strcmp(zReply,"SQLITE_OK")==0 ){ 749e22a334bSdrh rc = SQLITE_OK; 750e22a334bSdrh }else if( strcmp(zReply,"SQLITE_DENY")==0 ){ 751e22a334bSdrh rc = SQLITE_DENY; 752e22a334bSdrh }else if( strcmp(zReply,"SQLITE_IGNORE")==0 ){ 753e22a334bSdrh rc = SQLITE_IGNORE; 754e22a334bSdrh }else{ 755e22a334bSdrh rc = 999; 756e22a334bSdrh } 757e22a334bSdrh return rc; 758e22a334bSdrh } 759e22a334bSdrh #endif /* SQLITE_OMIT_AUTHORIZATION */ 760cabb0819Sdrh 761cabb0819Sdrh /* 762ef2cb63eSdanielk1977 ** zText is a pointer to text obtained via an sqlite3_result_text() 763ef2cb63eSdanielk1977 ** or similar interface. This routine returns a Tcl string object, 764ef2cb63eSdanielk1977 ** reference count set to 0, containing the text. If a translation 765ef2cb63eSdanielk1977 ** between iso8859 and UTF-8 is required, it is preformed. 766ef2cb63eSdanielk1977 */ 767ef2cb63eSdanielk1977 static Tcl_Obj *dbTextToObj(char const *zText){ 768ef2cb63eSdanielk1977 Tcl_Obj *pVal; 769ef2cb63eSdanielk1977 #ifdef UTF_TRANSLATION_NEEDED 770ef2cb63eSdanielk1977 Tcl_DString dCol; 771ef2cb63eSdanielk1977 Tcl_DStringInit(&dCol); 772ef2cb63eSdanielk1977 Tcl_ExternalToUtfDString(NULL, zText, -1, &dCol); 773ef2cb63eSdanielk1977 pVal = Tcl_NewStringObj(Tcl_DStringValue(&dCol), -1); 774ef2cb63eSdanielk1977 Tcl_DStringFree(&dCol); 775ef2cb63eSdanielk1977 #else 776ef2cb63eSdanielk1977 pVal = Tcl_NewStringObj(zText, -1); 777ef2cb63eSdanielk1977 #endif 778ef2cb63eSdanielk1977 return pVal; 779ef2cb63eSdanielk1977 } 780ef2cb63eSdanielk1977 781ef2cb63eSdanielk1977 /* 7821067fe11Stpoindex ** This routine reads a line of text from FILE in, stores 7831067fe11Stpoindex ** the text in memory obtained from malloc() and returns a pointer 7841067fe11Stpoindex ** to the text. NULL is returned at end of file, or if malloc() 7851067fe11Stpoindex ** fails. 7861067fe11Stpoindex ** 7871067fe11Stpoindex ** The interface is like "readline" but no command-line editing 7881067fe11Stpoindex ** is done. 7891067fe11Stpoindex ** 7901067fe11Stpoindex ** copied from shell.c from '.import' command 7911067fe11Stpoindex */ 7921067fe11Stpoindex static char *local_getline(char *zPrompt, FILE *in){ 7931067fe11Stpoindex char *zLine; 7941067fe11Stpoindex int nLine; 7951067fe11Stpoindex int n; 7961067fe11Stpoindex int eol; 7971067fe11Stpoindex 7981067fe11Stpoindex nLine = 100; 7991067fe11Stpoindex zLine = malloc( nLine ); 8001067fe11Stpoindex if( zLine==0 ) return 0; 8011067fe11Stpoindex n = 0; 8021067fe11Stpoindex eol = 0; 8031067fe11Stpoindex while( !eol ){ 8041067fe11Stpoindex if( n+100>nLine ){ 8051067fe11Stpoindex nLine = nLine*2 + 100; 8061067fe11Stpoindex zLine = realloc(zLine, nLine); 8071067fe11Stpoindex if( zLine==0 ) return 0; 8081067fe11Stpoindex } 8091067fe11Stpoindex if( fgets(&zLine[n], nLine - n, in)==0 ){ 8101067fe11Stpoindex if( n==0 ){ 8111067fe11Stpoindex free(zLine); 8121067fe11Stpoindex return 0; 8131067fe11Stpoindex } 8141067fe11Stpoindex zLine[n] = 0; 8151067fe11Stpoindex eol = 1; 8161067fe11Stpoindex break; 8171067fe11Stpoindex } 8181067fe11Stpoindex while( zLine[n] ){ n++; } 8191067fe11Stpoindex if( n>0 && zLine[n-1]=='\n' ){ 8201067fe11Stpoindex n--; 8211067fe11Stpoindex zLine[n] = 0; 8221067fe11Stpoindex eol = 1; 8231067fe11Stpoindex } 8241067fe11Stpoindex } 8251067fe11Stpoindex zLine = realloc( zLine, n+1 ); 8261067fe11Stpoindex return zLine; 8271067fe11Stpoindex } 8281067fe11Stpoindex 8291067fe11Stpoindex /* 83075897234Sdrh ** The "sqlite" command below creates a new Tcl command for each 83175897234Sdrh ** connection it opens to an SQLite database. This routine is invoked 83275897234Sdrh ** whenever one of those connection-specific commands is executed 83375897234Sdrh ** in Tcl. For example, if you run Tcl code like this: 83475897234Sdrh ** 8359bb575fdSdrh ** sqlite3 db1 "my_database" 83675897234Sdrh ** db1 close 83775897234Sdrh ** 83875897234Sdrh ** The first command opens a connection to the "my_database" database 83975897234Sdrh ** and calls that connection "db1". The second command causes this 84075897234Sdrh ** subroutine to be invoked. 84175897234Sdrh */ 8426d31316cSdrh static int DbObjCmd(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){ 843bec3f402Sdrh SqliteDb *pDb = (SqliteDb*)cd; 8446d31316cSdrh int choice; 84522fbcb8dSdrh int rc = TCL_OK; 8460de8c112Sdrh static const char *DB_strs[] = { 847fb7e7651Sdrh "authorizer", "busy", "cache", 848fb7e7651Sdrh "changes", "close", "collate", 849fb7e7651Sdrh "collation_needed", "commit_hook", "complete", 8504144905bSdrh "copy", "enable_load_extension","errorcode", 8514144905bSdrh "eval", "exists", "function", 852*b4e9af9fSdanielk1977 "incrblob", 853f11bded5Sdrh "interrupt", "last_insert_rowid", "nullvalue", 854f11bded5Sdrh "onecolumn", "profile", "progress", 855f11bded5Sdrh "rekey", "rollback_hook", "timeout", 856f11bded5Sdrh "total_changes", "trace", "transaction", 857f11bded5Sdrh "update_hook", "version", 0 8586d31316cSdrh }; 859411995dcSdrh enum DB_enum { 860fb7e7651Sdrh DB_AUTHORIZER, DB_BUSY, DB_CACHE, 861fb7e7651Sdrh DB_CHANGES, DB_CLOSE, DB_COLLATE, 862fb7e7651Sdrh DB_COLLATION_NEEDED, DB_COMMIT_HOOK, DB_COMPLETE, 8634144905bSdrh DB_COPY, DB_ENABLE_LOAD_EXTENSION,DB_ERRORCODE, 8644144905bSdrh DB_EVAL, DB_EXISTS, DB_FUNCTION, 865*b4e9af9fSdanielk1977 DB_INCRBLOB, 866f11bded5Sdrh DB_INTERRUPT, DB_LAST_INSERT_ROWID,DB_NULLVALUE, 867f11bded5Sdrh DB_ONECOLUMN, DB_PROFILE, DB_PROGRESS, 868f11bded5Sdrh DB_REKEY, DB_ROLLBACK_HOOK, DB_TIMEOUT, 869f11bded5Sdrh DB_TOTAL_CHANGES, DB_TRACE, DB_TRANSACTION, 870f11bded5Sdrh DB_UPDATE_HOOK, DB_VERSION, 8716d31316cSdrh }; 8721067fe11Stpoindex /* don't leave trailing commas on DB_enum, it confuses the AIX xlc compiler */ 8736d31316cSdrh 8746d31316cSdrh if( objc<2 ){ 8756d31316cSdrh Tcl_WrongNumArgs(interp, 1, objv, "SUBCOMMAND ..."); 87675897234Sdrh return TCL_ERROR; 87775897234Sdrh } 878411995dcSdrh if( Tcl_GetIndexFromObj(interp, objv[1], DB_strs, "option", 0, &choice) ){ 8796d31316cSdrh return TCL_ERROR; 8806d31316cSdrh } 8816d31316cSdrh 882411995dcSdrh switch( (enum DB_enum)choice ){ 88375897234Sdrh 884e22a334bSdrh /* $db authorizer ?CALLBACK? 885e22a334bSdrh ** 886e22a334bSdrh ** Invoke the given callback to authorize each SQL operation as it is 887e22a334bSdrh ** compiled. 5 arguments are appended to the callback before it is 888e22a334bSdrh ** invoked: 889e22a334bSdrh ** 890e22a334bSdrh ** (1) The authorization type (ex: SQLITE_CREATE_TABLE, SQLITE_INSERT, ...) 891e22a334bSdrh ** (2) First descriptive name (depends on authorization type) 892e22a334bSdrh ** (3) Second descriptive name 893e22a334bSdrh ** (4) Name of the database (ex: "main", "temp") 894e22a334bSdrh ** (5) Name of trigger that is doing the access 895e22a334bSdrh ** 896e22a334bSdrh ** The callback should return on of the following strings: SQLITE_OK, 897e22a334bSdrh ** SQLITE_IGNORE, or SQLITE_DENY. Any other return value is an error. 898e22a334bSdrh ** 899e22a334bSdrh ** If this method is invoked with no arguments, the current authorization 900e22a334bSdrh ** callback string is returned. 901e22a334bSdrh */ 902e22a334bSdrh case DB_AUTHORIZER: { 9031211de37Sdrh #ifdef SQLITE_OMIT_AUTHORIZATION 9041211de37Sdrh Tcl_AppendResult(interp, "authorization not available in this build", 0); 9051211de37Sdrh return TCL_ERROR; 9061211de37Sdrh #else 907e22a334bSdrh if( objc>3 ){ 908e22a334bSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 9090f14e2ebSdrh return TCL_ERROR; 910e22a334bSdrh }else if( objc==2 ){ 911b5a20d3cSdrh if( pDb->zAuth ){ 912e22a334bSdrh Tcl_AppendResult(interp, pDb->zAuth, 0); 913e22a334bSdrh } 914e22a334bSdrh }else{ 915e22a334bSdrh char *zAuth; 916e22a334bSdrh int len; 917e22a334bSdrh if( pDb->zAuth ){ 918e22a334bSdrh Tcl_Free(pDb->zAuth); 919e22a334bSdrh } 920e22a334bSdrh zAuth = Tcl_GetStringFromObj(objv[2], &len); 921e22a334bSdrh if( zAuth && len>0 ){ 922e22a334bSdrh pDb->zAuth = Tcl_Alloc( len + 1 ); 923e22a334bSdrh strcpy(pDb->zAuth, zAuth); 924e22a334bSdrh }else{ 925e22a334bSdrh pDb->zAuth = 0; 926e22a334bSdrh } 927e22a334bSdrh if( pDb->zAuth ){ 928e22a334bSdrh pDb->interp = interp; 9296f8a503dSdanielk1977 sqlite3_set_authorizer(pDb->db, auth_callback, pDb); 930e22a334bSdrh }else{ 9316f8a503dSdanielk1977 sqlite3_set_authorizer(pDb->db, 0, 0); 932e22a334bSdrh } 933e22a334bSdrh } 9341211de37Sdrh #endif 935e22a334bSdrh break; 936e22a334bSdrh } 937e22a334bSdrh 938bec3f402Sdrh /* $db busy ?CALLBACK? 939bec3f402Sdrh ** 940bec3f402Sdrh ** Invoke the given callback if an SQL statement attempts to open 941bec3f402Sdrh ** a locked database file. 942bec3f402Sdrh */ 9436d31316cSdrh case DB_BUSY: { 9446d31316cSdrh if( objc>3 ){ 9456d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "CALLBACK"); 946bec3f402Sdrh return TCL_ERROR; 9476d31316cSdrh }else if( objc==2 ){ 948bec3f402Sdrh if( pDb->zBusy ){ 949bec3f402Sdrh Tcl_AppendResult(interp, pDb->zBusy, 0); 950bec3f402Sdrh } 951bec3f402Sdrh }else{ 9526d31316cSdrh char *zBusy; 9536d31316cSdrh int len; 954bec3f402Sdrh if( pDb->zBusy ){ 955bec3f402Sdrh Tcl_Free(pDb->zBusy); 9566d31316cSdrh } 9576d31316cSdrh zBusy = Tcl_GetStringFromObj(objv[2], &len); 9586d31316cSdrh if( zBusy && len>0 ){ 9596d31316cSdrh pDb->zBusy = Tcl_Alloc( len + 1 ); 9606d31316cSdrh strcpy(pDb->zBusy, zBusy); 9616d31316cSdrh }else{ 962bec3f402Sdrh pDb->zBusy = 0; 963bec3f402Sdrh } 964bec3f402Sdrh if( pDb->zBusy ){ 965bec3f402Sdrh pDb->interp = interp; 9666f8a503dSdanielk1977 sqlite3_busy_handler(pDb->db, DbBusyHandler, pDb); 9676d31316cSdrh }else{ 9686f8a503dSdanielk1977 sqlite3_busy_handler(pDb->db, 0, 0); 969bec3f402Sdrh } 970bec3f402Sdrh } 9716d31316cSdrh break; 9726d31316cSdrh } 973bec3f402Sdrh 974fb7e7651Sdrh /* $db cache flush 975fb7e7651Sdrh ** $db cache size n 976fb7e7651Sdrh ** 977fb7e7651Sdrh ** Flush the prepared statement cache, or set the maximum number of 978fb7e7651Sdrh ** cached statements. 979fb7e7651Sdrh */ 980fb7e7651Sdrh case DB_CACHE: { 981fb7e7651Sdrh char *subCmd; 982fb7e7651Sdrh int n; 983fb7e7651Sdrh 984fb7e7651Sdrh if( objc<=2 ){ 985fb7e7651Sdrh Tcl_WrongNumArgs(interp, 1, objv, "cache option ?arg?"); 986fb7e7651Sdrh return TCL_ERROR; 987fb7e7651Sdrh } 988fb7e7651Sdrh subCmd = Tcl_GetStringFromObj( objv[2], 0 ); 989fb7e7651Sdrh if( *subCmd=='f' && strcmp(subCmd,"flush")==0 ){ 990fb7e7651Sdrh if( objc!=3 ){ 991fb7e7651Sdrh Tcl_WrongNumArgs(interp, 2, objv, "flush"); 992fb7e7651Sdrh return TCL_ERROR; 993fb7e7651Sdrh }else{ 994fb7e7651Sdrh flushStmtCache( pDb ); 995fb7e7651Sdrh } 996fb7e7651Sdrh }else if( *subCmd=='s' && strcmp(subCmd,"size")==0 ){ 997fb7e7651Sdrh if( objc!=4 ){ 998fb7e7651Sdrh Tcl_WrongNumArgs(interp, 2, objv, "size n"); 999fb7e7651Sdrh return TCL_ERROR; 1000fb7e7651Sdrh }else{ 1001fb7e7651Sdrh if( TCL_ERROR==Tcl_GetIntFromObj(interp, objv[3], &n) ){ 1002fb7e7651Sdrh Tcl_AppendResult( interp, "cannot convert \"", 1003fb7e7651Sdrh Tcl_GetStringFromObj(objv[3],0), "\" to integer", 0); 1004fb7e7651Sdrh return TCL_ERROR; 1005fb7e7651Sdrh }else{ 1006fb7e7651Sdrh if( n<0 ){ 1007fb7e7651Sdrh flushStmtCache( pDb ); 1008fb7e7651Sdrh n = 0; 1009fb7e7651Sdrh }else if( n>MAX_PREPARED_STMTS ){ 1010fb7e7651Sdrh n = MAX_PREPARED_STMTS; 1011fb7e7651Sdrh } 1012fb7e7651Sdrh pDb->maxStmt = n; 1013fb7e7651Sdrh } 1014fb7e7651Sdrh } 1015fb7e7651Sdrh }else{ 1016fb7e7651Sdrh Tcl_AppendResult( interp, "bad option \"", 1017fb7e7651Sdrh Tcl_GetStringFromObj(objv[0],0), "\": must be flush or size", 0); 1018fb7e7651Sdrh return TCL_ERROR; 1019fb7e7651Sdrh } 1020fb7e7651Sdrh break; 1021fb7e7651Sdrh } 1022fb7e7651Sdrh 1023b28af71aSdanielk1977 /* $db changes 1024c8d30ac1Sdrh ** 1025c8d30ac1Sdrh ** Return the number of rows that were modified, inserted, or deleted by 1026b28af71aSdanielk1977 ** the most recent INSERT, UPDATE or DELETE statement, not including 1027b28af71aSdanielk1977 ** any changes made by trigger programs. 1028c8d30ac1Sdrh */ 1029c8d30ac1Sdrh case DB_CHANGES: { 1030c8d30ac1Sdrh Tcl_Obj *pResult; 1031c8d30ac1Sdrh if( objc!=2 ){ 1032c8d30ac1Sdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 1033c8d30ac1Sdrh return TCL_ERROR; 1034c8d30ac1Sdrh } 1035c8d30ac1Sdrh pResult = Tcl_GetObjResult(interp); 1036b28af71aSdanielk1977 Tcl_SetIntObj(pResult, sqlite3_changes(pDb->db)); 1037f146a776Srdc break; 1038f146a776Srdc } 1039f146a776Srdc 104075897234Sdrh /* $db close 104175897234Sdrh ** 104275897234Sdrh ** Shutdown the database 104375897234Sdrh */ 10446d31316cSdrh case DB_CLOSE: { 10456d31316cSdrh Tcl_DeleteCommand(interp, Tcl_GetStringFromObj(objv[0], 0)); 10466d31316cSdrh break; 10476d31316cSdrh } 104875897234Sdrh 10490f14e2ebSdrh /* 10500f14e2ebSdrh ** $db collate NAME SCRIPT 10510f14e2ebSdrh ** 10520f14e2ebSdrh ** Create a new SQL collation function called NAME. Whenever 10530f14e2ebSdrh ** that function is called, invoke SCRIPT to evaluate the function. 10540f14e2ebSdrh */ 10550f14e2ebSdrh case DB_COLLATE: { 10560f14e2ebSdrh SqlCollate *pCollate; 10570f14e2ebSdrh char *zName; 10580f14e2ebSdrh char *zScript; 10590f14e2ebSdrh int nScript; 10600f14e2ebSdrh if( objc!=4 ){ 10610f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, "NAME SCRIPT"); 10620f14e2ebSdrh return TCL_ERROR; 10630f14e2ebSdrh } 10640f14e2ebSdrh zName = Tcl_GetStringFromObj(objv[2], 0); 10650f14e2ebSdrh zScript = Tcl_GetStringFromObj(objv[3], &nScript); 10660f14e2ebSdrh pCollate = (SqlCollate*)Tcl_Alloc( sizeof(*pCollate) + nScript + 1 ); 10670f14e2ebSdrh if( pCollate==0 ) return TCL_ERROR; 10680f14e2ebSdrh pCollate->interp = interp; 10690f14e2ebSdrh pCollate->pNext = pDb->pCollate; 10700f14e2ebSdrh pCollate->zScript = (char*)&pCollate[1]; 10710f14e2ebSdrh pDb->pCollate = pCollate; 10720f14e2ebSdrh strcpy(pCollate->zScript, zScript); 10730f14e2ebSdrh if( sqlite3_create_collation(pDb->db, zName, SQLITE_UTF8, 10740f14e2ebSdrh pCollate, tclSqlCollate) ){ 10759636c4e1Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 10760f14e2ebSdrh return TCL_ERROR; 10770f14e2ebSdrh } 10780f14e2ebSdrh break; 10790f14e2ebSdrh } 10800f14e2ebSdrh 10810f14e2ebSdrh /* 10820f14e2ebSdrh ** $db collation_needed SCRIPT 10830f14e2ebSdrh ** 10840f14e2ebSdrh ** Create a new SQL collation function called NAME. Whenever 10850f14e2ebSdrh ** that function is called, invoke SCRIPT to evaluate the function. 10860f14e2ebSdrh */ 10870f14e2ebSdrh case DB_COLLATION_NEEDED: { 10880f14e2ebSdrh if( objc!=3 ){ 10890f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, "SCRIPT"); 10900f14e2ebSdrh return TCL_ERROR; 10910f14e2ebSdrh } 10920f14e2ebSdrh if( pDb->pCollateNeeded ){ 10930f14e2ebSdrh Tcl_DecrRefCount(pDb->pCollateNeeded); 10940f14e2ebSdrh } 10950f14e2ebSdrh pDb->pCollateNeeded = Tcl_DuplicateObj(objv[2]); 10960f14e2ebSdrh Tcl_IncrRefCount(pDb->pCollateNeeded); 10970f14e2ebSdrh sqlite3_collation_needed(pDb->db, pDb, tclCollateNeeded); 10980f14e2ebSdrh break; 10990f14e2ebSdrh } 11000f14e2ebSdrh 110119e2d37fSdrh /* $db commit_hook ?CALLBACK? 110219e2d37fSdrh ** 110319e2d37fSdrh ** Invoke the given callback just before committing every SQL transaction. 110419e2d37fSdrh ** If the callback throws an exception or returns non-zero, then the 110519e2d37fSdrh ** transaction is aborted. If CALLBACK is an empty string, the callback 110619e2d37fSdrh ** is disabled. 110719e2d37fSdrh */ 110819e2d37fSdrh case DB_COMMIT_HOOK: { 110919e2d37fSdrh if( objc>3 ){ 111019e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 111119e2d37fSdrh return TCL_ERROR; 111219e2d37fSdrh }else if( objc==2 ){ 111319e2d37fSdrh if( pDb->zCommit ){ 111419e2d37fSdrh Tcl_AppendResult(interp, pDb->zCommit, 0); 111519e2d37fSdrh } 111619e2d37fSdrh }else{ 111719e2d37fSdrh char *zCommit; 111819e2d37fSdrh int len; 111919e2d37fSdrh if( pDb->zCommit ){ 112019e2d37fSdrh Tcl_Free(pDb->zCommit); 112119e2d37fSdrh } 112219e2d37fSdrh zCommit = Tcl_GetStringFromObj(objv[2], &len); 112319e2d37fSdrh if( zCommit && len>0 ){ 112419e2d37fSdrh pDb->zCommit = Tcl_Alloc( len + 1 ); 112519e2d37fSdrh strcpy(pDb->zCommit, zCommit); 112619e2d37fSdrh }else{ 112719e2d37fSdrh pDb->zCommit = 0; 112819e2d37fSdrh } 112919e2d37fSdrh if( pDb->zCommit ){ 113019e2d37fSdrh pDb->interp = interp; 113119e2d37fSdrh sqlite3_commit_hook(pDb->db, DbCommitHandler, pDb); 113219e2d37fSdrh }else{ 113319e2d37fSdrh sqlite3_commit_hook(pDb->db, 0, 0); 113419e2d37fSdrh } 113519e2d37fSdrh } 113619e2d37fSdrh break; 113719e2d37fSdrh } 113819e2d37fSdrh 113975897234Sdrh /* $db complete SQL 114075897234Sdrh ** 114175897234Sdrh ** Return TRUE if SQL is a complete SQL statement. Return FALSE if 114275897234Sdrh ** additional lines of input are needed. This is similar to the 114375897234Sdrh ** built-in "info complete" command of Tcl. 114475897234Sdrh */ 11456d31316cSdrh case DB_COMPLETE: { 1146ccae6026Sdrh #ifndef SQLITE_OMIT_COMPLETE 11476d31316cSdrh Tcl_Obj *pResult; 11486d31316cSdrh int isComplete; 11496d31316cSdrh if( objc!=3 ){ 11506d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL"); 115175897234Sdrh return TCL_ERROR; 115275897234Sdrh } 11536f8a503dSdanielk1977 isComplete = sqlite3_complete( Tcl_GetStringFromObj(objv[2], 0) ); 11546d31316cSdrh pResult = Tcl_GetObjResult(interp); 11556d31316cSdrh Tcl_SetBooleanObj(pResult, isComplete); 1156ccae6026Sdrh #endif 11576d31316cSdrh break; 11586d31316cSdrh } 115975897234Sdrh 116019e2d37fSdrh /* $db copy conflict-algorithm table filename ?SEPARATOR? ?NULLINDICATOR? 116119e2d37fSdrh ** 116219e2d37fSdrh ** Copy data into table from filename, optionally using SEPARATOR 116319e2d37fSdrh ** as column separators. If a column contains a null string, or the 116419e2d37fSdrh ** value of NULLINDICATOR, a NULL is inserted for the column. 116519e2d37fSdrh ** conflict-algorithm is one of the sqlite conflict algorithms: 116619e2d37fSdrh ** rollback, abort, fail, ignore, replace 116719e2d37fSdrh ** On success, return the number of lines processed, not necessarily same 116819e2d37fSdrh ** as 'db changes' due to conflict-algorithm selected. 116919e2d37fSdrh ** 117019e2d37fSdrh ** This code is basically an implementation/enhancement of 117119e2d37fSdrh ** the sqlite3 shell.c ".import" command. 117219e2d37fSdrh ** 117319e2d37fSdrh ** This command usage is equivalent to the sqlite2.x COPY statement, 117419e2d37fSdrh ** which imports file data into a table using the PostgreSQL COPY file format: 117519e2d37fSdrh ** $db copy $conflit_algo $table_name $filename \t \\N 117619e2d37fSdrh */ 117719e2d37fSdrh case DB_COPY: { 117819e2d37fSdrh char *zTable; /* Insert data into this table */ 117919e2d37fSdrh char *zFile; /* The file from which to extract data */ 118019e2d37fSdrh char *zConflict; /* The conflict algorithm to use */ 118119e2d37fSdrh sqlite3_stmt *pStmt; /* A statement */ 118219e2d37fSdrh int rc; /* Result code */ 118319e2d37fSdrh int nCol; /* Number of columns in the table */ 118419e2d37fSdrh int nByte; /* Number of bytes in an SQL string */ 118519e2d37fSdrh int i, j; /* Loop counters */ 118619e2d37fSdrh int nSep; /* Number of bytes in zSep[] */ 118719e2d37fSdrh int nNull; /* Number of bytes in zNull[] */ 118819e2d37fSdrh char *zSql; /* An SQL statement */ 118919e2d37fSdrh char *zLine; /* A single line of input from the file */ 119019e2d37fSdrh char **azCol; /* zLine[] broken up into columns */ 119119e2d37fSdrh char *zCommit; /* How to commit changes */ 119219e2d37fSdrh FILE *in; /* The input file */ 119319e2d37fSdrh int lineno = 0; /* Line number of input file */ 119419e2d37fSdrh char zLineNum[80]; /* Line number print buffer */ 119519e2d37fSdrh Tcl_Obj *pResult; /* interp result */ 119619e2d37fSdrh 119719e2d37fSdrh char *zSep; 119819e2d37fSdrh char *zNull; 119919e2d37fSdrh if( objc<5 || objc>7 ){ 120019e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, 120119e2d37fSdrh "CONFLICT-ALGORITHM TABLE FILENAME ?SEPARATOR? ?NULLINDICATOR?"); 120219e2d37fSdrh return TCL_ERROR; 120319e2d37fSdrh } 120419e2d37fSdrh if( objc>=6 ){ 120519e2d37fSdrh zSep = Tcl_GetStringFromObj(objv[5], 0); 120619e2d37fSdrh }else{ 120719e2d37fSdrh zSep = "\t"; 120819e2d37fSdrh } 120919e2d37fSdrh if( objc>=7 ){ 121019e2d37fSdrh zNull = Tcl_GetStringFromObj(objv[6], 0); 121119e2d37fSdrh }else{ 121219e2d37fSdrh zNull = ""; 121319e2d37fSdrh } 121419e2d37fSdrh zConflict = Tcl_GetStringFromObj(objv[2], 0); 121519e2d37fSdrh zTable = Tcl_GetStringFromObj(objv[3], 0); 121619e2d37fSdrh zFile = Tcl_GetStringFromObj(objv[4], 0); 121719e2d37fSdrh nSep = strlen(zSep); 121819e2d37fSdrh nNull = strlen(zNull); 121919e2d37fSdrh if( nSep==0 ){ 122019e2d37fSdrh Tcl_AppendResult(interp,"Error: non-null separator required for copy",0); 122119e2d37fSdrh return TCL_ERROR; 122219e2d37fSdrh } 122319e2d37fSdrh if(sqlite3StrICmp(zConflict, "rollback") != 0 && 122419e2d37fSdrh sqlite3StrICmp(zConflict, "abort" ) != 0 && 122519e2d37fSdrh sqlite3StrICmp(zConflict, "fail" ) != 0 && 122619e2d37fSdrh sqlite3StrICmp(zConflict, "ignore" ) != 0 && 122719e2d37fSdrh sqlite3StrICmp(zConflict, "replace" ) != 0 ) { 122819e2d37fSdrh Tcl_AppendResult(interp, "Error: \"", zConflict, 122919e2d37fSdrh "\", conflict-algorithm must be one of: rollback, " 123019e2d37fSdrh "abort, fail, ignore, or replace", 0); 123119e2d37fSdrh return TCL_ERROR; 123219e2d37fSdrh } 123319e2d37fSdrh zSql = sqlite3_mprintf("SELECT * FROM '%q'", zTable); 123419e2d37fSdrh if( zSql==0 ){ 123519e2d37fSdrh Tcl_AppendResult(interp, "Error: no such table: ", zTable, 0); 123619e2d37fSdrh return TCL_ERROR; 123719e2d37fSdrh } 123819e2d37fSdrh nByte = strlen(zSql); 12393e701a18Sdrh rc = sqlite3_prepare(pDb->db, zSql, -1, &pStmt, 0); 124019e2d37fSdrh sqlite3_free(zSql); 124119e2d37fSdrh if( rc ){ 124219e2d37fSdrh Tcl_AppendResult(interp, "Error: ", sqlite3_errmsg(pDb->db), 0); 124319e2d37fSdrh nCol = 0; 124419e2d37fSdrh }else{ 124519e2d37fSdrh nCol = sqlite3_column_count(pStmt); 124619e2d37fSdrh } 124719e2d37fSdrh sqlite3_finalize(pStmt); 124819e2d37fSdrh if( nCol==0 ) { 124919e2d37fSdrh return TCL_ERROR; 125019e2d37fSdrh } 125119e2d37fSdrh zSql = malloc( nByte + 50 + nCol*2 ); 125219e2d37fSdrh if( zSql==0 ) { 125319e2d37fSdrh Tcl_AppendResult(interp, "Error: can't malloc()", 0); 125419e2d37fSdrh return TCL_ERROR; 125519e2d37fSdrh } 125619e2d37fSdrh sqlite3_snprintf(nByte+50, zSql, "INSERT OR %q INTO '%q' VALUES(?", 125719e2d37fSdrh zConflict, zTable); 125819e2d37fSdrh j = strlen(zSql); 125919e2d37fSdrh for(i=1; i<nCol; i++){ 126019e2d37fSdrh zSql[j++] = ','; 126119e2d37fSdrh zSql[j++] = '?'; 126219e2d37fSdrh } 126319e2d37fSdrh zSql[j++] = ')'; 126419e2d37fSdrh zSql[j] = 0; 12653e701a18Sdrh rc = sqlite3_prepare(pDb->db, zSql, -1, &pStmt, 0); 126619e2d37fSdrh free(zSql); 126719e2d37fSdrh if( rc ){ 126819e2d37fSdrh Tcl_AppendResult(interp, "Error: ", sqlite3_errmsg(pDb->db), 0); 126919e2d37fSdrh sqlite3_finalize(pStmt); 127019e2d37fSdrh return TCL_ERROR; 127119e2d37fSdrh } 127219e2d37fSdrh in = fopen(zFile, "rb"); 127319e2d37fSdrh if( in==0 ){ 127419e2d37fSdrh Tcl_AppendResult(interp, "Error: cannot open file: ", zFile, NULL); 127519e2d37fSdrh sqlite3_finalize(pStmt); 127619e2d37fSdrh return TCL_ERROR; 127719e2d37fSdrh } 127819e2d37fSdrh azCol = malloc( sizeof(azCol[0])*(nCol+1) ); 127919e2d37fSdrh if( azCol==0 ) { 128019e2d37fSdrh Tcl_AppendResult(interp, "Error: can't malloc()", 0); 128143617e9aSdrh fclose(in); 128219e2d37fSdrh return TCL_ERROR; 128319e2d37fSdrh } 12843752785fSdrh (void)sqlite3_exec(pDb->db, "BEGIN", 0, 0, 0); 128519e2d37fSdrh zCommit = "COMMIT"; 128619e2d37fSdrh while( (zLine = local_getline(0, in))!=0 ){ 128719e2d37fSdrh char *z; 128819e2d37fSdrh i = 0; 128919e2d37fSdrh lineno++; 129019e2d37fSdrh azCol[0] = zLine; 129119e2d37fSdrh for(i=0, z=zLine; *z; z++){ 129219e2d37fSdrh if( *z==zSep[0] && strncmp(z, zSep, nSep)==0 ){ 129319e2d37fSdrh *z = 0; 129419e2d37fSdrh i++; 129519e2d37fSdrh if( i<nCol ){ 129619e2d37fSdrh azCol[i] = &z[nSep]; 129719e2d37fSdrh z += nSep-1; 129819e2d37fSdrh } 129919e2d37fSdrh } 130019e2d37fSdrh } 130119e2d37fSdrh if( i+1!=nCol ){ 130219e2d37fSdrh char *zErr; 130319e2d37fSdrh zErr = malloc(200 + strlen(zFile)); 1304c1f4494eSdrh if( zErr ){ 1305955de52cSdanielk1977 sprintf(zErr, 1306955de52cSdanielk1977 "Error: %s line %d: expected %d columns of data but found %d", 130719e2d37fSdrh zFile, lineno, nCol, i+1); 130819e2d37fSdrh Tcl_AppendResult(interp, zErr, 0); 130919e2d37fSdrh free(zErr); 1310c1f4494eSdrh } 131119e2d37fSdrh zCommit = "ROLLBACK"; 131219e2d37fSdrh break; 131319e2d37fSdrh } 131419e2d37fSdrh for(i=0; i<nCol; i++){ 131519e2d37fSdrh /* check for null data, if so, bind as null */ 131619e2d37fSdrh if ((nNull>0 && strcmp(azCol[i], zNull)==0) || strlen(azCol[i])==0) { 131719e2d37fSdrh sqlite3_bind_null(pStmt, i+1); 131819e2d37fSdrh }else{ 131919e2d37fSdrh sqlite3_bind_text(pStmt, i+1, azCol[i], -1, SQLITE_STATIC); 132019e2d37fSdrh } 132119e2d37fSdrh } 132219e2d37fSdrh sqlite3_step(pStmt); 132319e2d37fSdrh rc = sqlite3_reset(pStmt); 132419e2d37fSdrh free(zLine); 132519e2d37fSdrh if( rc!=SQLITE_OK ){ 132619e2d37fSdrh Tcl_AppendResult(interp,"Error: ", sqlite3_errmsg(pDb->db), 0); 132719e2d37fSdrh zCommit = "ROLLBACK"; 132819e2d37fSdrh break; 132919e2d37fSdrh } 133019e2d37fSdrh } 133119e2d37fSdrh free(azCol); 133219e2d37fSdrh fclose(in); 133319e2d37fSdrh sqlite3_finalize(pStmt); 13343752785fSdrh (void)sqlite3_exec(pDb->db, zCommit, 0, 0, 0); 133519e2d37fSdrh 133619e2d37fSdrh if( zCommit[0] == 'C' ){ 133719e2d37fSdrh /* success, set result as number of lines processed */ 133819e2d37fSdrh pResult = Tcl_GetObjResult(interp); 133919e2d37fSdrh Tcl_SetIntObj(pResult, lineno); 134019e2d37fSdrh rc = TCL_OK; 134119e2d37fSdrh }else{ 134219e2d37fSdrh /* failure, append lineno where failed */ 134319e2d37fSdrh sprintf(zLineNum,"%d",lineno); 134419e2d37fSdrh Tcl_AppendResult(interp,", failed while processing line: ",zLineNum,0); 134519e2d37fSdrh rc = TCL_ERROR; 134619e2d37fSdrh } 134719e2d37fSdrh break; 134819e2d37fSdrh } 134919e2d37fSdrh 135075897234Sdrh /* 13514144905bSdrh ** $db enable_load_extension BOOLEAN 13524144905bSdrh ** 13534144905bSdrh ** Turn the extension loading feature on or off. It if off by 13544144905bSdrh ** default. 13554144905bSdrh */ 13564144905bSdrh case DB_ENABLE_LOAD_EXTENSION: { 1357f533acc0Sdrh #ifndef SQLITE_OMIT_LOAD_EXTENSION 13584144905bSdrh int onoff; 13594144905bSdrh if( objc!=3 ){ 13604144905bSdrh Tcl_WrongNumArgs(interp, 2, objv, "BOOLEAN"); 13614144905bSdrh return TCL_ERROR; 13624144905bSdrh } 13634144905bSdrh if( Tcl_GetBooleanFromObj(interp, objv[2], &onoff) ){ 13644144905bSdrh return TCL_ERROR; 13654144905bSdrh } 13664144905bSdrh sqlite3_enable_load_extension(pDb->db, onoff); 13674144905bSdrh break; 1368f533acc0Sdrh #else 1369f533acc0Sdrh Tcl_AppendResult(interp, "extension loading is turned off at compile-time", 1370f533acc0Sdrh 0); 1371f533acc0Sdrh return TCL_ERROR; 1372f533acc0Sdrh #endif 13734144905bSdrh } 13744144905bSdrh 13754144905bSdrh /* 1376dcd997eaSdrh ** $db errorcode 1377dcd997eaSdrh ** 1378dcd997eaSdrh ** Return the numeric error code that was returned by the most recent 13796f8a503dSdanielk1977 ** call to sqlite3_exec(). 1380dcd997eaSdrh */ 1381dcd997eaSdrh case DB_ERRORCODE: { 1382f3ce83f5Sdanielk1977 Tcl_SetObjResult(interp, Tcl_NewIntObj(sqlite3_errcode(pDb->db))); 1383dcd997eaSdrh break; 1384dcd997eaSdrh } 1385dcd997eaSdrh 1386dcd997eaSdrh /* 1387895d7472Sdrh ** $db eval $sql ?array? ?{ ...code... }? 13881807ce37Sdrh ** $db onecolumn $sql 138975897234Sdrh ** 139075897234Sdrh ** The SQL statement in $sql is evaluated. For each row, the values are 1391bec3f402Sdrh ** placed in elements of the array named "array" and ...code... is executed. 139275897234Sdrh ** If "array" and "code" are omitted, then no callback is every invoked. 139375897234Sdrh ** If "array" is an empty string, then the values are placed in variables 139475897234Sdrh ** that have the same name as the fields extracted by the query. 13951807ce37Sdrh ** 13961807ce37Sdrh ** The onecolumn method is the equivalent of: 13971807ce37Sdrh ** lindex [$db eval $sql] 0 139875897234Sdrh */ 13991807ce37Sdrh case DB_ONECOLUMN: 140097f2ebc1Sdrh case DB_EVAL: 140197f2ebc1Sdrh case DB_EXISTS: { 14029d74b4c5Sdrh char const *zSql; /* Next SQL statement to execute */ 14039d74b4c5Sdrh char const *zLeft; /* What is left after first stmt in zSql */ 14049d74b4c5Sdrh sqlite3_stmt *pStmt; /* Compiled SQL statment */ 140592febd92Sdrh Tcl_Obj *pArray; /* Name of array into which results are written */ 140692febd92Sdrh Tcl_Obj *pScript; /* Script to run for each result set */ 14071d895039Sdrh Tcl_Obj **apParm; /* Parameters that need a Tcl_DecrRefCount() */ 14081d895039Sdrh int nParm; /* Number of entries used in apParm[] */ 14091d895039Sdrh Tcl_Obj *aParm[10]; /* Static space for apParm[] in the common case */ 14101807ce37Sdrh Tcl_Obj *pRet; /* Value to be returned */ 1411fb7e7651Sdrh SqlPreparedStmt *pPreStmt; /* Pointer to a prepared statement */ 1412fb7e7651Sdrh int rc2; 1413ef2cb63eSdanielk1977 141497f2ebc1Sdrh if( choice==DB_EVAL ){ 141592febd92Sdrh if( objc<3 || objc>5 ){ 1416895d7472Sdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL ?ARRAY-NAME? ?SCRIPT?"); 141730ccda10Sdanielk1977 return TCL_ERROR; 141830ccda10Sdanielk1977 } 14191807ce37Sdrh pRet = Tcl_NewObj(); 14201807ce37Sdrh Tcl_IncrRefCount(pRet); 142197f2ebc1Sdrh }else{ 142297f2ebc1Sdrh if( objc!=3 ){ 142397f2ebc1Sdrh Tcl_WrongNumArgs(interp, 2, objv, "SQL"); 142497f2ebc1Sdrh return TCL_ERROR; 142597f2ebc1Sdrh } 142697f2ebc1Sdrh if( choice==DB_EXISTS ){ 1427446a9b82Sdrh pRet = Tcl_NewBooleanObj(0); 1428446a9b82Sdrh Tcl_IncrRefCount(pRet); 1429446a9b82Sdrh }else{ 1430446a9b82Sdrh pRet = 0; 143197f2ebc1Sdrh } 14321807ce37Sdrh } 143392febd92Sdrh if( objc==3 ){ 143492febd92Sdrh pArray = pScript = 0; 143592febd92Sdrh }else if( objc==4 ){ 143692febd92Sdrh pArray = 0; 143792febd92Sdrh pScript = objv[3]; 143892febd92Sdrh }else{ 143992febd92Sdrh pArray = objv[3]; 144092febd92Sdrh if( Tcl_GetString(pArray)[0]==0 ) pArray = 0; 144192febd92Sdrh pScript = objv[4]; 144292febd92Sdrh } 144330ccda10Sdanielk1977 14441d895039Sdrh Tcl_IncrRefCount(objv[2]); 144530ccda10Sdanielk1977 zSql = Tcl_GetStringFromObj(objv[2], 0); 144690b6bb19Sdrh while( rc==TCL_OK && zSql[0] ){ 144792febd92Sdrh int i; /* Loop counter */ 1448fb7e7651Sdrh int nVar; /* Number of bind parameters in the pStmt */ 144992febd92Sdrh int nCol; /* Number of columns in the result set */ 145092febd92Sdrh Tcl_Obj **apColName = 0; /* Array of column names */ 1451fb7e7651Sdrh int len; /* String length of zSql */ 145230ccda10Sdanielk1977 1453fb7e7651Sdrh /* Try to find a SQL statement that has already been compiled and 1454fb7e7651Sdrh ** which matches the next sequence of SQL. 1455fb7e7651Sdrh */ 1456fb7e7651Sdrh pStmt = 0; 1457fb7e7651Sdrh pPreStmt = pDb->stmtList; 1458fb7e7651Sdrh len = strlen(zSql); 1459fb7e7651Sdrh if( pPreStmt && sqlite3_expired(pPreStmt->pStmt) ){ 1460fb7e7651Sdrh flushStmtCache(pDb); 1461fb7e7651Sdrh pPreStmt = 0; 1462fb7e7651Sdrh } 1463fb7e7651Sdrh for(; pPreStmt; pPreStmt=pPreStmt->pNext){ 1464fb7e7651Sdrh int n = pPreStmt->nSql; 1465fb7e7651Sdrh if( len>=n 1466fb7e7651Sdrh && memcmp(pPreStmt->zSql, zSql, n)==0 1467fb7e7651Sdrh && (zSql[n]==0 || zSql[n-1]==';') 1468fb7e7651Sdrh ){ 1469fb7e7651Sdrh pStmt = pPreStmt->pStmt; 1470fb7e7651Sdrh zLeft = &zSql[pPreStmt->nSql]; 1471fb7e7651Sdrh 1472fb7e7651Sdrh /* When a prepared statement is found, unlink it from the 1473fb7e7651Sdrh ** cache list. It will later be added back to the beginning 1474fb7e7651Sdrh ** of the cache list in order to implement LRU replacement. 1475fb7e7651Sdrh */ 1476fb7e7651Sdrh if( pPreStmt->pPrev ){ 1477fb7e7651Sdrh pPreStmt->pPrev->pNext = pPreStmt->pNext; 1478fb7e7651Sdrh }else{ 1479fb7e7651Sdrh pDb->stmtList = pPreStmt->pNext; 1480fb7e7651Sdrh } 1481fb7e7651Sdrh if( pPreStmt->pNext ){ 1482fb7e7651Sdrh pPreStmt->pNext->pPrev = pPreStmt->pPrev; 1483fb7e7651Sdrh }else{ 1484fb7e7651Sdrh pDb->stmtLast = pPreStmt->pPrev; 1485fb7e7651Sdrh } 1486fb7e7651Sdrh pDb->nStmt--; 1487fb7e7651Sdrh break; 1488fb7e7651Sdrh } 1489fb7e7651Sdrh } 1490fb7e7651Sdrh 1491fb7e7651Sdrh /* If no prepared statement was found. Compile the SQL text 1492fb7e7651Sdrh */ 1493fb7e7651Sdrh if( pStmt==0 ){ 149430ccda10Sdanielk1977 if( SQLITE_OK!=sqlite3_prepare(pDb->db, zSql, -1, &pStmt, &zLeft) ){ 1495ef2cb63eSdanielk1977 Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 149630ccda10Sdanielk1977 rc = TCL_ERROR; 149730ccda10Sdanielk1977 break; 149830ccda10Sdanielk1977 } 149992febd92Sdrh if( pStmt==0 ){ 150092febd92Sdrh if( SQLITE_OK!=sqlite3_errcode(pDb->db) ){ 1501fb7e7651Sdrh /* A compile-time error in the statement 1502fb7e7651Sdrh */ 150392febd92Sdrh Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 150492febd92Sdrh rc = TCL_ERROR; 150592febd92Sdrh break; 150692febd92Sdrh }else{ 1507fb7e7651Sdrh /* The statement was a no-op. Continue to the next statement 1508fb7e7651Sdrh ** in the SQL string. 1509fb7e7651Sdrh */ 151092febd92Sdrh zSql = zLeft; 151192febd92Sdrh continue; 151292febd92Sdrh } 151392febd92Sdrh } 1514fb7e7651Sdrh assert( pPreStmt==0 ); 1515fb7e7651Sdrh } 151630ccda10Sdanielk1977 1517fb7e7651Sdrh /* Bind values to parameters that begin with $ or : 1518fb7e7651Sdrh */ 151992febd92Sdrh nVar = sqlite3_bind_parameter_count(pStmt); 15201d895039Sdrh nParm = 0; 15211d895039Sdrh if( nVar>sizeof(aParm)/sizeof(aParm[0]) ){ 15221d895039Sdrh apParm = (Tcl_Obj**)Tcl_Alloc(nVar*sizeof(apParm[0])); 15231d895039Sdrh }else{ 15241d895039Sdrh apParm = aParm; 15251d895039Sdrh } 152692febd92Sdrh for(i=1; i<=nVar; i++){ 152792febd92Sdrh const char *zVar = sqlite3_bind_parameter_name(pStmt, i); 1528c8f9079cSdrh if( zVar!=0 && (zVar[0]=='$' || zVar[0]==':') ){ 152992febd92Sdrh Tcl_Obj *pVar = Tcl_GetVar2Ex(interp, &zVar[1], 0, 0); 153092febd92Sdrh if( pVar ){ 153192febd92Sdrh int n; 153292febd92Sdrh u8 *data; 153392febd92Sdrh char *zType = pVar->typePtr ? pVar->typePtr->name : ""; 153492febd92Sdrh char c = zType[0]; 1535df0bddaeSdrh if( c=='b' && strcmp(zType,"bytearray")==0 && pVar->bytes==0 ){ 1536df0bddaeSdrh /* Only load a BLOB type if the Tcl variable is a bytearray and 1537df0bddaeSdrh ** has no string representation. */ 153892febd92Sdrh data = Tcl_GetByteArrayFromObj(pVar, &n); 153992febd92Sdrh sqlite3_bind_blob(pStmt, i, data, n, SQLITE_STATIC); 15401d895039Sdrh Tcl_IncrRefCount(pVar); 15411d895039Sdrh apParm[nParm++] = pVar; 154292febd92Sdrh }else if( (c=='b' && strcmp(zType,"boolean")==0) || 154392febd92Sdrh (c=='i' && strcmp(zType,"int")==0) ){ 154492febd92Sdrh Tcl_GetIntFromObj(interp, pVar, &n); 154592febd92Sdrh sqlite3_bind_int(pStmt, i, n); 154692febd92Sdrh }else if( c=='d' && strcmp(zType,"double")==0 ){ 154792febd92Sdrh double r; 154892febd92Sdrh Tcl_GetDoubleFromObj(interp, pVar, &r); 154992febd92Sdrh sqlite3_bind_double(pStmt, i, r); 1550df0bddaeSdrh }else if( c=='w' && strcmp(zType,"wideInt")==0 ){ 1551df0bddaeSdrh Tcl_WideInt v; 1552df0bddaeSdrh Tcl_GetWideIntFromObj(interp, pVar, &v); 1553df0bddaeSdrh sqlite3_bind_int64(pStmt, i, v); 155492febd92Sdrh }else{ 155500fd957bSdanielk1977 data = (unsigned char *)Tcl_GetStringFromObj(pVar, &n); 155600fd957bSdanielk1977 sqlite3_bind_text(pStmt, i, (char *)data, n, SQLITE_STATIC); 15571d895039Sdrh Tcl_IncrRefCount(pVar); 15581d895039Sdrh apParm[nParm++] = pVar; 155992febd92Sdrh } 1560fb7e7651Sdrh }else{ 1561fb7e7651Sdrh sqlite3_bind_null( pStmt, i ); 156292febd92Sdrh } 156392febd92Sdrh } 156492febd92Sdrh } 156592febd92Sdrh 156692febd92Sdrh /* Compute column names */ 156792febd92Sdrh nCol = sqlite3_column_count(pStmt); 156892febd92Sdrh if( pScript ){ 156992febd92Sdrh apColName = (Tcl_Obj**)Tcl_Alloc( sizeof(Tcl_Obj*)*nCol ); 157092febd92Sdrh if( apColName==0 ) break; 157192febd92Sdrh for(i=0; i<nCol; i++){ 157292febd92Sdrh apColName[i] = dbTextToObj(sqlite3_column_name(pStmt,i)); 157392febd92Sdrh Tcl_IncrRefCount(apColName[i]); 157492febd92Sdrh } 157592febd92Sdrh } 157692febd92Sdrh 157792febd92Sdrh /* If results are being stored in an array variable, then create 157892febd92Sdrh ** the array(*) entry for that array 157992febd92Sdrh */ 158092febd92Sdrh if( pArray ){ 158130ccda10Sdanielk1977 Tcl_Obj *pColList = Tcl_NewObj(); 15823ced14a6Sdrh Tcl_Obj *pStar = Tcl_NewStringObj("*", -1); 158330ccda10Sdanielk1977 Tcl_IncrRefCount(pColList); 158492febd92Sdrh for(i=0; i<nCol; i++){ 158592febd92Sdrh Tcl_ListObjAppendElement(interp, pColList, apColName[i]); 158630ccda10Sdanielk1977 } 15873ced14a6Sdrh Tcl_ObjSetVar2(interp, pArray, pStar, pColList,0); 15883ced14a6Sdrh Tcl_DecrRefCount(pColList); 15893ced14a6Sdrh Tcl_DecrRefCount(pStar); 159030ccda10Sdanielk1977 } 159130ccda10Sdanielk1977 159292febd92Sdrh /* Execute the SQL 159392febd92Sdrh */ 159490b6bb19Sdrh while( rc==TCL_OK && pStmt && SQLITE_ROW==sqlite3_step(pStmt) ){ 159592febd92Sdrh for(i=0; i<nCol; i++){ 159630ccda10Sdanielk1977 Tcl_Obj *pVal; 159730ccda10Sdanielk1977 159830ccda10Sdanielk1977 /* Set pVal to contain the i'th column of this row. */ 159992febd92Sdrh switch( sqlite3_column_type(pStmt, i) ){ 160092febd92Sdrh case SQLITE_BLOB: { 16013fd0a736Sdanielk1977 int bytes = sqlite3_column_bytes(pStmt, i); 16023fd0a736Sdanielk1977 pVal = Tcl_NewByteArrayObj(sqlite3_column_blob(pStmt, i), bytes); 160392febd92Sdrh break; 160492febd92Sdrh } 160592febd92Sdrh case SQLITE_INTEGER: { 160692febd92Sdrh sqlite_int64 v = sqlite3_column_int64(pStmt, i); 160792febd92Sdrh if( v>=-2147483647 && v<=2147483647 ){ 160892febd92Sdrh pVal = Tcl_NewIntObj(v); 160992febd92Sdrh }else{ 161092febd92Sdrh pVal = Tcl_NewWideIntObj(v); 161192febd92Sdrh } 161292febd92Sdrh break; 161392febd92Sdrh } 161492febd92Sdrh case SQLITE_FLOAT: { 161592febd92Sdrh double r = sqlite3_column_double(pStmt, i); 161692febd92Sdrh pVal = Tcl_NewDoubleObj(r); 161792febd92Sdrh break; 161892febd92Sdrh } 161955c45f2eSdanielk1977 case SQLITE_NULL: { 162055c45f2eSdanielk1977 pVal = dbTextToObj(pDb->zNull); 162155c45f2eSdanielk1977 break; 162255c45f2eSdanielk1977 } 162392febd92Sdrh default: { 162400fd957bSdanielk1977 pVal = dbTextToObj((char *)sqlite3_column_text(pStmt, i)); 162592febd92Sdrh break; 162692febd92Sdrh } 162730ccda10Sdanielk1977 } 162830ccda10Sdanielk1977 162992febd92Sdrh if( pScript ){ 163092febd92Sdrh if( pArray==0 ){ 163192febd92Sdrh Tcl_ObjSetVar2(interp, apColName[i], 0, pVal, 0); 163230ccda10Sdanielk1977 }else{ 163392febd92Sdrh Tcl_ObjSetVar2(interp, pArray, apColName[i], pVal, 0); 163430ccda10Sdanielk1977 } 16351807ce37Sdrh }else if( choice==DB_ONECOLUMN ){ 163697f2ebc1Sdrh assert( pRet==0 ); 16371807ce37Sdrh if( pRet==0 ){ 16381807ce37Sdrh pRet = pVal; 16391807ce37Sdrh Tcl_IncrRefCount(pRet); 16401807ce37Sdrh } 164190b6bb19Sdrh rc = TCL_BREAK; 164297f2ebc1Sdrh i = nCol; 164397f2ebc1Sdrh }else if( choice==DB_EXISTS ){ 1644446a9b82Sdrh Tcl_DecrRefCount(pRet); 1645446a9b82Sdrh pRet = Tcl_NewBooleanObj(1); 1646446a9b82Sdrh Tcl_IncrRefCount(pRet); 164797f2ebc1Sdrh rc = TCL_BREAK; 164897f2ebc1Sdrh i = nCol; 164930ccda10Sdanielk1977 }else{ 165030ccda10Sdanielk1977 Tcl_ListObjAppendElement(interp, pRet, pVal); 165130ccda10Sdanielk1977 } 165230ccda10Sdanielk1977 } 165330ccda10Sdanielk1977 165492febd92Sdrh if( pScript ){ 165592febd92Sdrh rc = Tcl_EvalObjEx(interp, pScript, 0); 165690b6bb19Sdrh if( rc==TCL_CONTINUE ){ 165790b6bb19Sdrh rc = TCL_OK; 165830ccda10Sdanielk1977 } 165930ccda10Sdanielk1977 } 166090b6bb19Sdrh } 166190b6bb19Sdrh if( rc==TCL_BREAK ){ 166290b6bb19Sdrh rc = TCL_OK; 166390b6bb19Sdrh } 166430ccda10Sdanielk1977 166592febd92Sdrh /* Free the column name objects */ 166692febd92Sdrh if( pScript ){ 166792febd92Sdrh for(i=0; i<nCol; i++){ 166892febd92Sdrh Tcl_DecrRefCount(apColName[i]); 166992febd92Sdrh } 167092febd92Sdrh Tcl_Free((char*)apColName); 167192febd92Sdrh } 167292febd92Sdrh 16731d895039Sdrh /* Free the bound string and blob parameters */ 16741d895039Sdrh for(i=0; i<nParm; i++){ 16751d895039Sdrh Tcl_DecrRefCount(apParm[i]); 16761d895039Sdrh } 16771d895039Sdrh if( apParm!=aParm ){ 16781d895039Sdrh Tcl_Free((char*)apParm); 16791d895039Sdrh } 16801d895039Sdrh 1681fb7e7651Sdrh /* Reset the statement. If the result code is SQLITE_SCHEMA, then 1682fb7e7651Sdrh ** flush the statement cache and try the statement again. 168392febd92Sdrh */ 1684fb7e7651Sdrh rc2 = sqlite3_reset(pStmt); 1685fb7e7651Sdrh if( SQLITE_SCHEMA==rc2 ){ 1686fb7e7651Sdrh /* After a schema change, flush the cache and try to run the 1687fb7e7651Sdrh ** statement again 1688fb7e7651Sdrh */ 1689fb7e7651Sdrh flushStmtCache( pDb ); 1690fb7e7651Sdrh sqlite3_finalize(pStmt); 1691fb7e7651Sdrh if( pPreStmt ) Tcl_Free((char*)pPreStmt); 169230ccda10Sdanielk1977 continue; 1693fb7e7651Sdrh }else if( SQLITE_OK!=rc2 ){ 1694fb7e7651Sdrh /* If a run-time error occurs, report the error and stop reading 1695fb7e7651Sdrh ** the SQL 1696fb7e7651Sdrh */ 1697ef2cb63eSdanielk1977 Tcl_SetObjResult(interp, dbTextToObj(sqlite3_errmsg(pDb->db))); 1698fb7e7651Sdrh sqlite3_finalize(pStmt); 169930ccda10Sdanielk1977 rc = TCL_ERROR; 1700fb7e7651Sdrh if( pPreStmt ) Tcl_Free((char*)pPreStmt); 170130ccda10Sdanielk1977 break; 1702fb7e7651Sdrh }else if( pDb->maxStmt<=0 ){ 1703fb7e7651Sdrh /* If the cache is turned off, deallocated the statement */ 1704fb7e7651Sdrh if( pPreStmt ) Tcl_Free((char*)pPreStmt); 1705fb7e7651Sdrh sqlite3_finalize(pStmt); 1706fb7e7651Sdrh }else{ 1707fb7e7651Sdrh /* Everything worked and the cache is operational. 1708fb7e7651Sdrh ** Create a new SqlPreparedStmt structure if we need one. 1709fb7e7651Sdrh ** (If we already have one we can just reuse it.) 1710fb7e7651Sdrh */ 1711fb7e7651Sdrh if( pPreStmt==0 ){ 1712fb7e7651Sdrh len = zLeft - zSql; 1713fb7e7651Sdrh pPreStmt = (SqlPreparedStmt*)Tcl_Alloc( sizeof(*pPreStmt) + len ); 1714fb7e7651Sdrh if( pPreStmt==0 ) return TCL_ERROR; 1715fb7e7651Sdrh pPreStmt->pStmt = pStmt; 1716fb7e7651Sdrh pPreStmt->nSql = len; 1717fb7e7651Sdrh memcpy(pPreStmt->zSql, zSql, len); 1718fb7e7651Sdrh pPreStmt->zSql[len] = 0; 171930ccda10Sdanielk1977 } 172030ccda10Sdanielk1977 1721fb7e7651Sdrh /* Add the prepared statement to the beginning of the cache list 1722fb7e7651Sdrh */ 1723fb7e7651Sdrh pPreStmt->pNext = pDb->stmtList; 1724fb7e7651Sdrh pPreStmt->pPrev = 0; 1725fb7e7651Sdrh if( pDb->stmtList ){ 1726fb7e7651Sdrh pDb->stmtList->pPrev = pPreStmt; 1727fb7e7651Sdrh } 1728fb7e7651Sdrh pDb->stmtList = pPreStmt; 1729fb7e7651Sdrh if( pDb->stmtLast==0 ){ 1730fb7e7651Sdrh assert( pDb->nStmt==0 ); 1731fb7e7651Sdrh pDb->stmtLast = pPreStmt; 1732fb7e7651Sdrh }else{ 1733fb7e7651Sdrh assert( pDb->nStmt>0 ); 1734fb7e7651Sdrh } 1735fb7e7651Sdrh pDb->nStmt++; 1736fb7e7651Sdrh 1737fb7e7651Sdrh /* If we have too many statement in cache, remove the surplus from the 1738fb7e7651Sdrh ** end of the cache list. 1739fb7e7651Sdrh */ 1740fb7e7651Sdrh while( pDb->nStmt>pDb->maxStmt ){ 1741fb7e7651Sdrh sqlite3_finalize(pDb->stmtLast->pStmt); 1742fb7e7651Sdrh pDb->stmtLast = pDb->stmtLast->pPrev; 1743fb7e7651Sdrh Tcl_Free((char*)pDb->stmtLast->pNext); 1744fb7e7651Sdrh pDb->stmtLast->pNext = 0; 1745fb7e7651Sdrh pDb->nStmt--; 1746fb7e7651Sdrh } 1747fb7e7651Sdrh } 1748fb7e7651Sdrh 1749fb7e7651Sdrh /* Proceed to the next statement */ 175030ccda10Sdanielk1977 zSql = zLeft; 175130ccda10Sdanielk1977 } 17521d895039Sdrh Tcl_DecrRefCount(objv[2]); 175330ccda10Sdanielk1977 17541807ce37Sdrh if( pRet ){ 1755ef2cb63eSdanielk1977 if( rc==TCL_OK ){ 175630ccda10Sdanielk1977 Tcl_SetObjResult(interp, pRet); 175730ccda10Sdanielk1977 } 1758ef2cb63eSdanielk1977 Tcl_DecrRefCount(pRet); 1759050be329Sdrh }else if( rc==TCL_OK ){ 1760050be329Sdrh Tcl_ResetResult(interp); 17611807ce37Sdrh } 176230ccda10Sdanielk1977 break; 176330ccda10Sdanielk1977 } 1764bec3f402Sdrh 1765bec3f402Sdrh /* 1766cabb0819Sdrh ** $db function NAME SCRIPT 1767cabb0819Sdrh ** 1768cabb0819Sdrh ** Create a new SQL function called NAME. Whenever that function is 1769cabb0819Sdrh ** called, invoke SCRIPT to evaluate the function. 1770cabb0819Sdrh */ 1771cabb0819Sdrh case DB_FUNCTION: { 1772cabb0819Sdrh SqlFunc *pFunc; 1773d1e4733dSdrh Tcl_Obj *pScript; 1774cabb0819Sdrh char *zName; 1775cabb0819Sdrh if( objc!=4 ){ 1776cabb0819Sdrh Tcl_WrongNumArgs(interp, 2, objv, "NAME SCRIPT"); 1777cabb0819Sdrh return TCL_ERROR; 1778cabb0819Sdrh } 1779cabb0819Sdrh zName = Tcl_GetStringFromObj(objv[2], 0); 1780d1e4733dSdrh pScript = objv[3]; 1781d1e4733dSdrh pFunc = findSqlFunc(pDb, zName); 1782cabb0819Sdrh if( pFunc==0 ) return TCL_ERROR; 1783d1e4733dSdrh if( pFunc->pScript ){ 1784d1e4733dSdrh Tcl_DecrRefCount(pFunc->pScript); 1785d1e4733dSdrh } 1786d1e4733dSdrh pFunc->pScript = pScript; 1787d1e4733dSdrh Tcl_IncrRefCount(pScript); 1788d1e4733dSdrh pFunc->useEvalObjv = safeToUseEvalObjv(interp, pScript); 1789c8c1158bSdanielk1977 rc = sqlite3_create_function(pDb->db, zName, -1, SQLITE_UTF8, 1790d8123366Sdanielk1977 pFunc, tclSqlFunc, 0, 0); 1791fb7e7651Sdrh if( rc!=SQLITE_OK ){ 1792fb7e7651Sdrh rc = TCL_ERROR; 17939636c4e1Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_errmsg(pDb->db), TCL_VOLATILE); 1794fb7e7651Sdrh }else{ 1795fb7e7651Sdrh /* Must flush any cached statements */ 1796fb7e7651Sdrh flushStmtCache( pDb ); 1797fb7e7651Sdrh } 1798cabb0819Sdrh break; 1799cabb0819Sdrh } 1800cabb0819Sdrh 1801cabb0819Sdrh /* 1802*b4e9af9fSdanielk1977 ** $db incrblob ?DB? TABLE COLUMN ROWID 1803*b4e9af9fSdanielk1977 */ 1804*b4e9af9fSdanielk1977 case DB_INCRBLOB: { 1805*b4e9af9fSdanielk1977 const char *zDb = "main"; 1806*b4e9af9fSdanielk1977 const char *zTable; 1807*b4e9af9fSdanielk1977 const char *zColumn; 1808*b4e9af9fSdanielk1977 sqlite_int64 iRow; 1809*b4e9af9fSdanielk1977 1810*b4e9af9fSdanielk1977 if( objc!=5 && objc!=6 ){ 1811*b4e9af9fSdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?DB? TABLE ROWID"); 1812*b4e9af9fSdanielk1977 return TCL_ERROR; 1813*b4e9af9fSdanielk1977 } 1814*b4e9af9fSdanielk1977 1815*b4e9af9fSdanielk1977 if( objc==6 ){ 1816*b4e9af9fSdanielk1977 zDb = Tcl_GetString(objv[2]); 1817*b4e9af9fSdanielk1977 } 1818*b4e9af9fSdanielk1977 zTable = Tcl_GetString(objv[objc-3]); 1819*b4e9af9fSdanielk1977 zColumn = Tcl_GetString(objv[objc-2]); 1820*b4e9af9fSdanielk1977 rc = Tcl_GetWideIntFromObj(interp, objv[objc-1], &iRow); 1821*b4e9af9fSdanielk1977 1822*b4e9af9fSdanielk1977 if( rc==TCL_OK ){ 1823*b4e9af9fSdanielk1977 rc = createIncrblobChannel(interp, pDb, zDb, zTable, zColumn, iRow); 1824*b4e9af9fSdanielk1977 } 1825*b4e9af9fSdanielk1977 break; 1826*b4e9af9fSdanielk1977 } 1827*b4e9af9fSdanielk1977 1828*b4e9af9fSdanielk1977 /* 1829f11bded5Sdrh ** $db interrupt 1830f11bded5Sdrh ** 1831f11bded5Sdrh ** Interrupt the execution of the inner-most SQL interpreter. This 1832f11bded5Sdrh ** causes the SQL statement to return an error of SQLITE_INTERRUPT. 1833f11bded5Sdrh */ 1834f11bded5Sdrh case DB_INTERRUPT: { 1835f11bded5Sdrh sqlite3_interrupt(pDb->db); 1836f11bded5Sdrh break; 1837f11bded5Sdrh } 1838f11bded5Sdrh 1839f11bded5Sdrh /* 184019e2d37fSdrh ** $db nullvalue ?STRING? 184119e2d37fSdrh ** 184219e2d37fSdrh ** Change text used when a NULL comes back from the database. If ?STRING? 184319e2d37fSdrh ** is not present, then the current string used for NULL is returned. 184419e2d37fSdrh ** If STRING is present, then STRING is returned. 184519e2d37fSdrh ** 184619e2d37fSdrh */ 184719e2d37fSdrh case DB_NULLVALUE: { 184819e2d37fSdrh if( objc!=2 && objc!=3 ){ 184919e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "NULLVALUE"); 185019e2d37fSdrh return TCL_ERROR; 185119e2d37fSdrh } 185219e2d37fSdrh if( objc==3 ){ 185319e2d37fSdrh int len; 185419e2d37fSdrh char *zNull = Tcl_GetStringFromObj(objv[2], &len); 185519e2d37fSdrh if( pDb->zNull ){ 185619e2d37fSdrh Tcl_Free(pDb->zNull); 185719e2d37fSdrh } 185819e2d37fSdrh if( zNull && len>0 ){ 185919e2d37fSdrh pDb->zNull = Tcl_Alloc( len + 1 ); 186019e2d37fSdrh strncpy(pDb->zNull, zNull, len); 186119e2d37fSdrh pDb->zNull[len] = '\0'; 186219e2d37fSdrh }else{ 186319e2d37fSdrh pDb->zNull = 0; 186419e2d37fSdrh } 186519e2d37fSdrh } 186619e2d37fSdrh Tcl_SetObjResult(interp, dbTextToObj(pDb->zNull)); 186719e2d37fSdrh break; 186819e2d37fSdrh } 186919e2d37fSdrh 187019e2d37fSdrh /* 1871af9ff33aSdrh ** $db last_insert_rowid 1872af9ff33aSdrh ** 1873af9ff33aSdrh ** Return an integer which is the ROWID for the most recent insert. 1874af9ff33aSdrh */ 1875af9ff33aSdrh case DB_LAST_INSERT_ROWID: { 1876af9ff33aSdrh Tcl_Obj *pResult; 1877f7e678d6Sdrh Tcl_WideInt rowid; 1878af9ff33aSdrh if( objc!=2 ){ 1879af9ff33aSdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 1880af9ff33aSdrh return TCL_ERROR; 1881af9ff33aSdrh } 18826f8a503dSdanielk1977 rowid = sqlite3_last_insert_rowid(pDb->db); 1883af9ff33aSdrh pResult = Tcl_GetObjResult(interp); 1884f7e678d6Sdrh Tcl_SetWideIntObj(pResult, rowid); 1885af9ff33aSdrh break; 1886af9ff33aSdrh } 1887af9ff33aSdrh 1888af9ff33aSdrh /* 18891807ce37Sdrh ** The DB_ONECOLUMN method is implemented together with DB_EVAL. 18905d9d7576Sdrh */ 18911807ce37Sdrh 18921807ce37Sdrh /* $db progress ?N CALLBACK? 18931807ce37Sdrh ** 18941807ce37Sdrh ** Invoke the given callback every N virtual machine opcodes while executing 18951807ce37Sdrh ** queries. 18961807ce37Sdrh */ 18971807ce37Sdrh case DB_PROGRESS: { 18981807ce37Sdrh if( objc==2 ){ 18991807ce37Sdrh if( pDb->zProgress ){ 19001807ce37Sdrh Tcl_AppendResult(interp, pDb->zProgress, 0); 19015d9d7576Sdrh } 19021807ce37Sdrh }else if( objc==4 ){ 19031807ce37Sdrh char *zProgress; 19041807ce37Sdrh int len; 19051807ce37Sdrh int N; 19061807ce37Sdrh if( TCL_OK!=Tcl_GetIntFromObj(interp, objv[2], &N) ){ 19071807ce37Sdrh return TCL_ERROR; 19081807ce37Sdrh }; 19091807ce37Sdrh if( pDb->zProgress ){ 19101807ce37Sdrh Tcl_Free(pDb->zProgress); 19111807ce37Sdrh } 19121807ce37Sdrh zProgress = Tcl_GetStringFromObj(objv[3], &len); 19131807ce37Sdrh if( zProgress && len>0 ){ 19141807ce37Sdrh pDb->zProgress = Tcl_Alloc( len + 1 ); 19151807ce37Sdrh strcpy(pDb->zProgress, zProgress); 19161807ce37Sdrh }else{ 19171807ce37Sdrh pDb->zProgress = 0; 19181807ce37Sdrh } 19191807ce37Sdrh #ifndef SQLITE_OMIT_PROGRESS_CALLBACK 19201807ce37Sdrh if( pDb->zProgress ){ 19211807ce37Sdrh pDb->interp = interp; 19221807ce37Sdrh sqlite3_progress_handler(pDb->db, N, DbProgressHandler, pDb); 19231807ce37Sdrh }else{ 19241807ce37Sdrh sqlite3_progress_handler(pDb->db, 0, 0, 0); 19251807ce37Sdrh } 19261807ce37Sdrh #endif 19271807ce37Sdrh }else{ 19281807ce37Sdrh Tcl_WrongNumArgs(interp, 2, objv, "N CALLBACK"); 19291807ce37Sdrh return TCL_ERROR; 19305d9d7576Sdrh } 19315d9d7576Sdrh break; 19325d9d7576Sdrh } 19335d9d7576Sdrh 193419e2d37fSdrh /* $db profile ?CALLBACK? 193519e2d37fSdrh ** 193619e2d37fSdrh ** Make arrangements to invoke the CALLBACK routine after each SQL statement 193719e2d37fSdrh ** that has run. The text of the SQL and the amount of elapse time are 193819e2d37fSdrh ** appended to CALLBACK before the script is run. 193919e2d37fSdrh */ 194019e2d37fSdrh case DB_PROFILE: { 194119e2d37fSdrh if( objc>3 ){ 194219e2d37fSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 194319e2d37fSdrh return TCL_ERROR; 194419e2d37fSdrh }else if( objc==2 ){ 194519e2d37fSdrh if( pDb->zProfile ){ 194619e2d37fSdrh Tcl_AppendResult(interp, pDb->zProfile, 0); 194719e2d37fSdrh } 194819e2d37fSdrh }else{ 194919e2d37fSdrh char *zProfile; 195019e2d37fSdrh int len; 195119e2d37fSdrh if( pDb->zProfile ){ 195219e2d37fSdrh Tcl_Free(pDb->zProfile); 195319e2d37fSdrh } 195419e2d37fSdrh zProfile = Tcl_GetStringFromObj(objv[2], &len); 195519e2d37fSdrh if( zProfile && len>0 ){ 195619e2d37fSdrh pDb->zProfile = Tcl_Alloc( len + 1 ); 195719e2d37fSdrh strcpy(pDb->zProfile, zProfile); 195819e2d37fSdrh }else{ 195919e2d37fSdrh pDb->zProfile = 0; 196019e2d37fSdrh } 196119e2d37fSdrh #ifndef SQLITE_OMIT_TRACE 196219e2d37fSdrh if( pDb->zProfile ){ 196319e2d37fSdrh pDb->interp = interp; 196419e2d37fSdrh sqlite3_profile(pDb->db, DbProfileHandler, pDb); 196519e2d37fSdrh }else{ 196619e2d37fSdrh sqlite3_profile(pDb->db, 0, 0); 196719e2d37fSdrh } 196819e2d37fSdrh #endif 196919e2d37fSdrh } 197019e2d37fSdrh break; 197119e2d37fSdrh } 197219e2d37fSdrh 19735d9d7576Sdrh /* 197422fbcb8dSdrh ** $db rekey KEY 197522fbcb8dSdrh ** 197622fbcb8dSdrh ** Change the encryption key on the currently open database. 197722fbcb8dSdrh */ 197822fbcb8dSdrh case DB_REKEY: { 197922fbcb8dSdrh int nKey; 198022fbcb8dSdrh void *pKey; 198122fbcb8dSdrh if( objc!=3 ){ 198222fbcb8dSdrh Tcl_WrongNumArgs(interp, 2, objv, "KEY"); 198322fbcb8dSdrh return TCL_ERROR; 198422fbcb8dSdrh } 198522fbcb8dSdrh pKey = Tcl_GetByteArrayFromObj(objv[2], &nKey); 19869eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 19872011d5f5Sdrh rc = sqlite3_rekey(pDb->db, pKey, nKey); 198822fbcb8dSdrh if( rc ){ 1989f20b21c8Sdanielk1977 Tcl_AppendResult(interp, sqlite3ErrStr(rc), 0); 199022fbcb8dSdrh rc = TCL_ERROR; 199122fbcb8dSdrh } 199222fbcb8dSdrh #endif 199322fbcb8dSdrh break; 199422fbcb8dSdrh } 199522fbcb8dSdrh 199622fbcb8dSdrh /* 1997bec3f402Sdrh ** $db timeout MILLESECONDS 1998bec3f402Sdrh ** 1999bec3f402Sdrh ** Delay for the number of milliseconds specified when a file is locked. 2000bec3f402Sdrh */ 20016d31316cSdrh case DB_TIMEOUT: { 2002bec3f402Sdrh int ms; 20036d31316cSdrh if( objc!=3 ){ 20046d31316cSdrh Tcl_WrongNumArgs(interp, 2, objv, "MILLISECONDS"); 2005bec3f402Sdrh return TCL_ERROR; 200675897234Sdrh } 20076d31316cSdrh if( Tcl_GetIntFromObj(interp, objv[2], &ms) ) return TCL_ERROR; 20086f8a503dSdanielk1977 sqlite3_busy_timeout(pDb->db, ms); 20096d31316cSdrh break; 201075897234Sdrh } 2011b5a20d3cSdrh 20120f14e2ebSdrh /* 20130f14e2ebSdrh ** $db total_changes 20140f14e2ebSdrh ** 20150f14e2ebSdrh ** Return the number of rows that were modified, inserted, or deleted 20160f14e2ebSdrh ** since the database handle was created. 20170f14e2ebSdrh */ 20180f14e2ebSdrh case DB_TOTAL_CHANGES: { 20190f14e2ebSdrh Tcl_Obj *pResult; 20200f14e2ebSdrh if( objc!=2 ){ 20210f14e2ebSdrh Tcl_WrongNumArgs(interp, 2, objv, ""); 20220f14e2ebSdrh return TCL_ERROR; 20230f14e2ebSdrh } 20240f14e2ebSdrh pResult = Tcl_GetObjResult(interp); 20250f14e2ebSdrh Tcl_SetIntObj(pResult, sqlite3_total_changes(pDb->db)); 20260f14e2ebSdrh break; 20270f14e2ebSdrh } 20280f14e2ebSdrh 2029b5a20d3cSdrh /* $db trace ?CALLBACK? 2030b5a20d3cSdrh ** 2031b5a20d3cSdrh ** Make arrangements to invoke the CALLBACK routine for each SQL statement 2032b5a20d3cSdrh ** that is executed. The text of the SQL is appended to CALLBACK before 2033b5a20d3cSdrh ** it is executed. 2034b5a20d3cSdrh */ 2035b5a20d3cSdrh case DB_TRACE: { 2036b5a20d3cSdrh if( objc>3 ){ 2037b5a20d3cSdrh Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?"); 2038b97759edSdrh return TCL_ERROR; 2039b5a20d3cSdrh }else if( objc==2 ){ 2040b5a20d3cSdrh if( pDb->zTrace ){ 2041b5a20d3cSdrh Tcl_AppendResult(interp, pDb->zTrace, 0); 2042b5a20d3cSdrh } 2043b5a20d3cSdrh }else{ 2044b5a20d3cSdrh char *zTrace; 2045b5a20d3cSdrh int len; 2046b5a20d3cSdrh if( pDb->zTrace ){ 2047b5a20d3cSdrh Tcl_Free(pDb->zTrace); 2048b5a20d3cSdrh } 2049b5a20d3cSdrh zTrace = Tcl_GetStringFromObj(objv[2], &len); 2050b5a20d3cSdrh if( zTrace && len>0 ){ 2051b5a20d3cSdrh pDb->zTrace = Tcl_Alloc( len + 1 ); 2052b5a20d3cSdrh strcpy(pDb->zTrace, zTrace); 2053b5a20d3cSdrh }else{ 2054b5a20d3cSdrh pDb->zTrace = 0; 2055b5a20d3cSdrh } 205619e2d37fSdrh #ifndef SQLITE_OMIT_TRACE 2057b5a20d3cSdrh if( pDb->zTrace ){ 2058b5a20d3cSdrh pDb->interp = interp; 20596f8a503dSdanielk1977 sqlite3_trace(pDb->db, DbTraceHandler, pDb); 2060b5a20d3cSdrh }else{ 20616f8a503dSdanielk1977 sqlite3_trace(pDb->db, 0, 0); 2062b5a20d3cSdrh } 206319e2d37fSdrh #endif 2064b5a20d3cSdrh } 2065b5a20d3cSdrh break; 2066b5a20d3cSdrh } 2067b5a20d3cSdrh 20683d21423cSdrh /* $db transaction [-deferred|-immediate|-exclusive] SCRIPT 20693d21423cSdrh ** 20703d21423cSdrh ** Start a new transaction (if we are not already in the midst of a 20713d21423cSdrh ** transaction) and execute the TCL script SCRIPT. After SCRIPT 20723d21423cSdrh ** completes, either commit the transaction or roll it back if SCRIPT 20733d21423cSdrh ** throws an exception. Or if no new transation was started, do nothing. 20743d21423cSdrh ** pass the exception on up the stack. 20753d21423cSdrh ** 20763d21423cSdrh ** This command was inspired by Dave Thomas's talk on Ruby at the 20773d21423cSdrh ** 2005 O'Reilly Open Source Convention (OSCON). 20783d21423cSdrh */ 20793d21423cSdrh case DB_TRANSACTION: { 20803d21423cSdrh int inTrans; 20813d21423cSdrh Tcl_Obj *pScript; 20823d21423cSdrh const char *zBegin = "BEGIN"; 20833d21423cSdrh if( objc!=3 && objc!=4 ){ 20843d21423cSdrh Tcl_WrongNumArgs(interp, 2, objv, "[TYPE] SCRIPT"); 20853d21423cSdrh return TCL_ERROR; 20863d21423cSdrh } 20873d21423cSdrh if( objc==3 ){ 20883d21423cSdrh pScript = objv[2]; 20893d21423cSdrh } else { 20903d21423cSdrh static const char *TTYPE_strs[] = { 2091ce604012Sdrh "deferred", "exclusive", "immediate", 0 20923d21423cSdrh }; 20933d21423cSdrh enum TTYPE_enum { 20943d21423cSdrh TTYPE_DEFERRED, TTYPE_EXCLUSIVE, TTYPE_IMMEDIATE 20953d21423cSdrh }; 20963d21423cSdrh int ttype; 2097b5555e7eSdrh if( Tcl_GetIndexFromObj(interp, objv[2], TTYPE_strs, "transaction type", 20983d21423cSdrh 0, &ttype) ){ 20993d21423cSdrh return TCL_ERROR; 21003d21423cSdrh } 21013d21423cSdrh switch( (enum TTYPE_enum)ttype ){ 21023d21423cSdrh case TTYPE_DEFERRED: /* no-op */; break; 21033d21423cSdrh case TTYPE_EXCLUSIVE: zBegin = "BEGIN EXCLUSIVE"; break; 21043d21423cSdrh case TTYPE_IMMEDIATE: zBegin = "BEGIN IMMEDIATE"; break; 21053d21423cSdrh } 21063d21423cSdrh pScript = objv[3]; 21073d21423cSdrh } 21083d21423cSdrh inTrans = !sqlite3_get_autocommit(pDb->db); 21093d21423cSdrh if( !inTrans ){ 21103752785fSdrh (void)sqlite3_exec(pDb->db, zBegin, 0, 0, 0); 21113d21423cSdrh } 21123d21423cSdrh rc = Tcl_EvalObjEx(interp, pScript, 0); 21133d21423cSdrh if( !inTrans ){ 21143d21423cSdrh const char *zEnd; 21153d21423cSdrh if( rc==TCL_ERROR ){ 21163d21423cSdrh zEnd = "ROLLBACK"; 21173d21423cSdrh } else { 21183d21423cSdrh zEnd = "COMMIT"; 21193d21423cSdrh } 21203752785fSdrh (void)sqlite3_exec(pDb->db, zEnd, 0, 0, 0); 21213d21423cSdrh } 21223d21423cSdrh break; 21233d21423cSdrh } 21243d21423cSdrh 212594eb6a14Sdanielk1977 /* 212694eb6a14Sdanielk1977 ** $db update_hook ?script? 212771fd80bfSdanielk1977 ** $db rollback_hook ?script? 212894eb6a14Sdanielk1977 */ 212971fd80bfSdanielk1977 case DB_UPDATE_HOOK: 213071fd80bfSdanielk1977 case DB_ROLLBACK_HOOK: { 213171fd80bfSdanielk1977 213271fd80bfSdanielk1977 /* set ppHook to point at pUpdateHook or pRollbackHook, depending on 213371fd80bfSdanielk1977 ** whether [$db update_hook] or [$db rollback_hook] was invoked. 213471fd80bfSdanielk1977 */ 213571fd80bfSdanielk1977 Tcl_Obj **ppHook; 213671fd80bfSdanielk1977 if( choice==DB_UPDATE_HOOK ){ 213771fd80bfSdanielk1977 ppHook = &pDb->pUpdateHook; 213871fd80bfSdanielk1977 }else{ 213971fd80bfSdanielk1977 ppHook = &pDb->pRollbackHook; 214071fd80bfSdanielk1977 } 214171fd80bfSdanielk1977 214294eb6a14Sdanielk1977 if( objc!=2 && objc!=3 ){ 214394eb6a14Sdanielk1977 Tcl_WrongNumArgs(interp, 2, objv, "?SCRIPT?"); 214494eb6a14Sdanielk1977 return TCL_ERROR; 214594eb6a14Sdanielk1977 } 214671fd80bfSdanielk1977 if( *ppHook ){ 214771fd80bfSdanielk1977 Tcl_SetObjResult(interp, *ppHook); 214894eb6a14Sdanielk1977 if( objc==3 ){ 214971fd80bfSdanielk1977 Tcl_DecrRefCount(*ppHook); 215071fd80bfSdanielk1977 *ppHook = 0; 215194eb6a14Sdanielk1977 } 215294eb6a14Sdanielk1977 } 215394eb6a14Sdanielk1977 if( objc==3 ){ 215471fd80bfSdanielk1977 assert( !(*ppHook) ); 215594eb6a14Sdanielk1977 if( Tcl_GetCharLength(objv[2])>0 ){ 215671fd80bfSdanielk1977 *ppHook = objv[2]; 215771fd80bfSdanielk1977 Tcl_IncrRefCount(*ppHook); 215894eb6a14Sdanielk1977 } 215994eb6a14Sdanielk1977 } 216071fd80bfSdanielk1977 216171fd80bfSdanielk1977 sqlite3_update_hook(pDb->db, (pDb->pUpdateHook?DbUpdateHandler:0), pDb); 216271fd80bfSdanielk1977 sqlite3_rollback_hook(pDb->db,(pDb->pRollbackHook?DbRollbackHandler:0),pDb); 216371fd80bfSdanielk1977 216494eb6a14Sdanielk1977 break; 216594eb6a14Sdanielk1977 } 216694eb6a14Sdanielk1977 21674397de57Sdanielk1977 /* $db version 21684397de57Sdanielk1977 ** 21694397de57Sdanielk1977 ** Return the version string for this database. 21704397de57Sdanielk1977 */ 21714397de57Sdanielk1977 case DB_VERSION: { 21724397de57Sdanielk1977 Tcl_SetResult(interp, (char *)sqlite3_libversion(), TCL_STATIC); 21734397de57Sdanielk1977 break; 21744397de57Sdanielk1977 } 21754397de57Sdanielk1977 21761067fe11Stpoindex 21776d31316cSdrh } /* End of the SWITCH statement */ 217822fbcb8dSdrh return rc; 217975897234Sdrh } 218075897234Sdrh 218175897234Sdrh /* 21829bb575fdSdrh ** sqlite3 DBNAME FILENAME ?MODE? ?-key KEY? 218375897234Sdrh ** 218475897234Sdrh ** This is the main Tcl command. When the "sqlite" Tcl command is 218575897234Sdrh ** invoked, this routine runs to process that command. 218675897234Sdrh ** 218775897234Sdrh ** The first argument, DBNAME, is an arbitrary name for a new 218875897234Sdrh ** database connection. This command creates a new command named 218975897234Sdrh ** DBNAME that is used to control that connection. The database 219075897234Sdrh ** connection is deleted when the DBNAME command is deleted. 219175897234Sdrh ** 219275897234Sdrh ** The second argument is the name of the directory that contains 219375897234Sdrh ** the sqlite database that is to be accessed. 2194fbc3eab8Sdrh ** 2195fbc3eab8Sdrh ** For testing purposes, we also support the following: 2196fbc3eab8Sdrh ** 21979bb575fdSdrh ** sqlite3 -encoding 2198fbc3eab8Sdrh ** 2199fbc3eab8Sdrh ** Return the encoding used by LIKE and GLOB operators. Choices 2200fbc3eab8Sdrh ** are UTF-8 and iso8859. 2201fbc3eab8Sdrh ** 22029bb575fdSdrh ** sqlite3 -version 2203647cb0e1Sdrh ** 2204647cb0e1Sdrh ** Return the version number of the SQLite library. 2205647cb0e1Sdrh ** 22069bb575fdSdrh ** sqlite3 -tcl-uses-utf 2207fbc3eab8Sdrh ** 2208fbc3eab8Sdrh ** Return "1" if compiled with a Tcl uses UTF-8. Return "0" if 2209fbc3eab8Sdrh ** not. Used by tests to make sure the library was compiled 2210fbc3eab8Sdrh ** correctly. 221175897234Sdrh */ 221222fbcb8dSdrh static int DbMain(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){ 2213bec3f402Sdrh SqliteDb *p; 221422fbcb8dSdrh void *pKey = 0; 221522fbcb8dSdrh int nKey = 0; 221622fbcb8dSdrh const char *zArg; 221775897234Sdrh char *zErrMsg; 221822fbcb8dSdrh const char *zFile; 2219882e8e4dSdrh Tcl_DString translatedFilename; 222022fbcb8dSdrh if( objc==2 ){ 222122fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[1], 0); 222222fbcb8dSdrh if( strcmp(zArg,"-version")==0 ){ 22236f8a503dSdanielk1977 Tcl_AppendResult(interp,sqlite3_version,0); 2224647cb0e1Sdrh return TCL_OK; 2225647cb0e1Sdrh } 22269eb9e26bSdrh if( strcmp(zArg,"-has-codec")==0 ){ 22279eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 222822fbcb8dSdrh Tcl_AppendResult(interp,"1",0); 222922fbcb8dSdrh #else 223022fbcb8dSdrh Tcl_AppendResult(interp,"0",0); 223122fbcb8dSdrh #endif 223222fbcb8dSdrh return TCL_OK; 223322fbcb8dSdrh } 223422fbcb8dSdrh if( strcmp(zArg,"-tcl-uses-utf")==0 ){ 2235fbc3eab8Sdrh #ifdef TCL_UTF_MAX 2236fbc3eab8Sdrh Tcl_AppendResult(interp,"1",0); 2237fbc3eab8Sdrh #else 2238fbc3eab8Sdrh Tcl_AppendResult(interp,"0",0); 2239fbc3eab8Sdrh #endif 2240fbc3eab8Sdrh return TCL_OK; 2241fbc3eab8Sdrh } 2242fbc3eab8Sdrh } 224322fbcb8dSdrh if( objc==5 || objc==6 ){ 224422fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[objc-2], 0); 224522fbcb8dSdrh if( strcmp(zArg,"-key")==0 ){ 224622fbcb8dSdrh pKey = Tcl_GetByteArrayFromObj(objv[objc-1], &nKey); 224722fbcb8dSdrh objc -= 2; 224822fbcb8dSdrh } 224922fbcb8dSdrh } 225022fbcb8dSdrh if( objc!=3 && objc!=4 ){ 225122fbcb8dSdrh Tcl_WrongNumArgs(interp, 1, objv, 22529eb9e26bSdrh #ifdef SQLITE_HAS_CODEC 22539eb9e26bSdrh "HANDLE FILENAME ?-key CODEC-KEY?" 225422fbcb8dSdrh #else 225522fbcb8dSdrh "HANDLE FILENAME ?MODE?" 225622fbcb8dSdrh #endif 225722fbcb8dSdrh ); 225875897234Sdrh return TCL_ERROR; 225975897234Sdrh } 226075897234Sdrh zErrMsg = 0; 22614cdc9e84Sdrh p = (SqliteDb*)Tcl_Alloc( sizeof(*p) ); 226275897234Sdrh if( p==0 ){ 2263bec3f402Sdrh Tcl_SetResult(interp, "malloc failed", TCL_STATIC); 2264bec3f402Sdrh return TCL_ERROR; 2265bec3f402Sdrh } 2266bec3f402Sdrh memset(p, 0, sizeof(*p)); 226722fbcb8dSdrh zFile = Tcl_GetStringFromObj(objv[2], 0); 2268882e8e4dSdrh zFile = Tcl_TranslateFileName(interp, zFile, &translatedFilename); 22694f057f90Sdanielk1977 sqlite3_open(zFile, &p->db); 2270882e8e4dSdrh Tcl_DStringFree(&translatedFilename); 227180290863Sdanielk1977 if( SQLITE_OK!=sqlite3_errcode(p->db) ){ 22729404d50eSdrh zErrMsg = sqlite3_mprintf("%s", sqlite3_errmsg(p->db)); 227380290863Sdanielk1977 sqlite3_close(p->db); 227480290863Sdanielk1977 p->db = 0; 227580290863Sdanielk1977 } 22762011d5f5Sdrh #ifdef SQLITE_HAS_CODEC 22772011d5f5Sdrh sqlite3_key(p->db, pKey, nKey); 2278eb8ed70dSdrh #endif 2279bec3f402Sdrh if( p->db==0 ){ 228075897234Sdrh Tcl_SetResult(interp, zErrMsg, TCL_VOLATILE); 2281bec3f402Sdrh Tcl_Free((char*)p); 22829404d50eSdrh sqlite3_free(zErrMsg); 228375897234Sdrh return TCL_ERROR; 228475897234Sdrh } 2285fb7e7651Sdrh p->maxStmt = NUM_PREPARED_STMTS; 22865169bbc6Sdrh p->interp = interp; 228722fbcb8dSdrh zArg = Tcl_GetStringFromObj(objv[1], 0); 228822fbcb8dSdrh Tcl_CreateObjCommand(interp, zArg, DbObjCmd, (char*)p, DbDeleteCmd); 2289c22bd47dSdrh 2290c22bd47dSdrh /* If compiled with SQLITE_TEST turned on, then register the "md5sum" 229106b2718aSdrh ** SQL function. 2292c22bd47dSdrh */ 229328b4e489Sdrh #ifdef SQLITE_TEST 229428b4e489Sdrh { 22959bb575fdSdrh extern void Md5_Register(sqlite3*); 22963b93bc8cSdrh #ifdef SQLITE_MEMDEBUG 229713073931Sdanielk1977 int mallocfail = sqlite3_iMallocFail; 229813073931Sdanielk1977 sqlite3_iMallocFail = 0; 22995b59af85Sdanielk1977 #endif 230028b4e489Sdrh Md5_Register(p->db); 23013b93bc8cSdrh #ifdef SQLITE_MEMDEBUG 230213073931Sdanielk1977 sqlite3_iMallocFail = mallocfail; 23035b59af85Sdanielk1977 #endif 230428b4e489Sdrh } 230528b4e489Sdrh #endif 230675897234Sdrh return TCL_OK; 230775897234Sdrh } 230875897234Sdrh 230975897234Sdrh /* 231090ca9753Sdrh ** Provide a dummy Tcl_InitStubs if we are using this as a static 231190ca9753Sdrh ** library. 231290ca9753Sdrh */ 231390ca9753Sdrh #ifndef USE_TCL_STUBS 231490ca9753Sdrh # undef Tcl_InitStubs 231590ca9753Sdrh # define Tcl_InitStubs(a,b,c) 231690ca9753Sdrh #endif 231790ca9753Sdrh 231890ca9753Sdrh /* 231929bc4615Sdrh ** Make sure we have a PACKAGE_VERSION macro defined. This will be 232029bc4615Sdrh ** defined automatically by the TEA makefile. But other makefiles 232129bc4615Sdrh ** do not define it. 232229bc4615Sdrh */ 232329bc4615Sdrh #ifndef PACKAGE_VERSION 232429bc4615Sdrh # define PACKAGE_VERSION SQLITE_VERSION 232529bc4615Sdrh #endif 232629bc4615Sdrh 232729bc4615Sdrh /* 232875897234Sdrh ** Initialize this module. 232975897234Sdrh ** 233075897234Sdrh ** This Tcl module contains only a single new Tcl command named "sqlite". 233175897234Sdrh ** (Hence there is no namespace. There is no point in using a namespace 233275897234Sdrh ** if the extension only supplies one new name!) The "sqlite" command is 233375897234Sdrh ** used to open a new SQLite database. See the DbMain() routine above 233475897234Sdrh ** for additional information. 233575897234Sdrh */ 2336ad6e1370Sdrh EXTERN int Sqlite3_Init(Tcl_Interp *interp){ 233792febd92Sdrh Tcl_InitStubs(interp, "8.4", 0); 2338ef4ac8f9Sdrh Tcl_CreateObjCommand(interp, "sqlite3", (Tcl_ObjCmdProc*)DbMain, 0, 0); 233929bc4615Sdrh Tcl_PkgProvide(interp, "sqlite3", PACKAGE_VERSION); 234049766d6cSdrh Tcl_CreateObjCommand(interp, "sqlite", (Tcl_ObjCmdProc*)DbMain, 0, 0); 234129bc4615Sdrh Tcl_PkgProvide(interp, "sqlite", PACKAGE_VERSION); 234290ca9753Sdrh return TCL_OK; 234390ca9753Sdrh } 2344ad6e1370Sdrh EXTERN int Tclsqlite3_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2345ad6e1370Sdrh EXTERN int Sqlite3_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2346ad6e1370Sdrh EXTERN int Tclsqlite3_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 234749766d6cSdrh 234849766d6cSdrh #ifndef SQLITE_3_SUFFIX_ONLY 2349ad6e1370Sdrh EXTERN int Sqlite_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2350ad6e1370Sdrh EXTERN int Tclsqlite_Init(Tcl_Interp *interp){ return Sqlite3_Init(interp); } 2351ad6e1370Sdrh EXTERN int Sqlite_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 2352ad6e1370Sdrh EXTERN int Tclsqlite_SafeInit(Tcl_Interp *interp){ return TCL_OK; } 235349766d6cSdrh #endif 235475897234Sdrh 23553e27c026Sdrh #ifdef TCLSH 23563e27c026Sdrh /***************************************************************************** 23573e27c026Sdrh ** The code that follows is used to build standalone TCL interpreters 235875897234Sdrh */ 2359348784efSdrh 2360348784efSdrh /* 23613e27c026Sdrh ** If the macro TCLSH is one, then put in code this for the 23623e27c026Sdrh ** "main" routine that will initialize Tcl and take input from 23633e27c026Sdrh ** standard input. 2364348784efSdrh */ 23653e27c026Sdrh #if TCLSH==1 2366348784efSdrh static char zMainloop[] = 2367348784efSdrh "set line {}\n" 2368348784efSdrh "while {![eof stdin]} {\n" 2369348784efSdrh "if {$line!=\"\"} {\n" 2370348784efSdrh "puts -nonewline \"> \"\n" 2371348784efSdrh "} else {\n" 2372348784efSdrh "puts -nonewline \"% \"\n" 2373348784efSdrh "}\n" 2374348784efSdrh "flush stdout\n" 2375348784efSdrh "append line [gets stdin]\n" 2376348784efSdrh "if {[info complete $line]} {\n" 2377348784efSdrh "if {[catch {uplevel #0 $line} result]} {\n" 2378348784efSdrh "puts stderr \"Error: $result\"\n" 2379348784efSdrh "} elseif {$result!=\"\"} {\n" 2380348784efSdrh "puts $result\n" 2381348784efSdrh "}\n" 2382348784efSdrh "set line {}\n" 2383348784efSdrh "} else {\n" 2384348784efSdrh "append line \\n\n" 2385348784efSdrh "}\n" 2386348784efSdrh "}\n" 2387348784efSdrh ; 23883e27c026Sdrh #endif 23893e27c026Sdrh 23903e27c026Sdrh /* 23913e27c026Sdrh ** If the macro TCLSH is two, then get the main loop code out of 23923e27c026Sdrh ** the separate file "spaceanal_tcl.h". 23933e27c026Sdrh */ 23943e27c026Sdrh #if TCLSH==2 23953e27c026Sdrh static char zMainloop[] = 23963e27c026Sdrh #include "spaceanal_tcl.h" 23973e27c026Sdrh ; 23983e27c026Sdrh #endif 2399348784efSdrh 2400348784efSdrh #define TCLSH_MAIN main /* Needed to fake out mktclapp */ 2401348784efSdrh int TCLSH_MAIN(int argc, char **argv){ 2402348784efSdrh Tcl_Interp *interp; 2403297ecf14Sdrh Tcl_FindExecutable(argv[0]); 2404348784efSdrh interp = Tcl_CreateInterp(); 240538f8271fSdrh Sqlite3_Init(interp); 2406d9b0257aSdrh #ifdef SQLITE_TEST 2407d1bf3512Sdrh { 2408d1bf3512Sdrh extern int Sqlitetest1_Init(Tcl_Interp*); 24095c4d9703Sdrh extern int Sqlitetest2_Init(Tcl_Interp*); 24105c4d9703Sdrh extern int Sqlitetest3_Init(Tcl_Interp*); 2411a6064dcfSdrh extern int Sqlitetest4_Init(Tcl_Interp*); 2412998b56c3Sdanielk1977 extern int Sqlitetest5_Init(Tcl_Interp*); 24139c06c953Sdrh extern int Sqlitetest6_Init(Tcl_Interp*); 241429c636bcSdrh extern int Sqlitetest7_Init(Tcl_Interp*); 2415b9bb7c18Sdrh extern int Sqlitetest8_Init(Tcl_Interp*); 2416a713f2c3Sdanielk1977 extern int Sqlitetest9_Init(Tcl_Interp*); 2417efc251daSdrh extern int Md5_Init(Tcl_Interp*); 24182e66f0b9Sdrh extern int Sqlitetestsse_Init(Tcl_Interp*); 24192366940dSdrh extern int Sqlitetestasync_Init(Tcl_Interp*); 24204be8b51eSdrh extern int Sqlitetesttclvar_Init(Tcl_Interp*); 2421954ce99cSdanielk1977 extern int Sqlitetestschema_Init(Tcl_Interp*); 24221409be69Sdrh extern int Sqlitetest_autoext_Init(Tcl_Interp*); 242315926590Sdrh extern int Sqlitetest_hexio_Init(Tcl_Interp*); 24242e66f0b9Sdrh 24256490bebdSdanielk1977 Sqlitetest1_Init(interp); 24265c4d9703Sdrh Sqlitetest2_Init(interp); 2427de647130Sdrh Sqlitetest3_Init(interp); 2428fc57d7bfSdanielk1977 Sqlitetest4_Init(interp); 2429998b56c3Sdanielk1977 Sqlitetest5_Init(interp); 24309c06c953Sdrh Sqlitetest6_Init(interp); 243129c636bcSdrh Sqlitetest7_Init(interp); 2432b9bb7c18Sdrh Sqlitetest8_Init(interp); 2433a713f2c3Sdanielk1977 Sqlitetest9_Init(interp); 24342366940dSdrh Sqlitetestasync_Init(interp); 24354be8b51eSdrh Sqlitetesttclvar_Init(interp); 2436954ce99cSdanielk1977 Sqlitetestschema_Init(interp); 24371409be69Sdrh Sqlitetest_autoext_Init(interp); 243815926590Sdrh Sqlitetest_hexio_Init(interp); 2439efc251daSdrh Md5_Init(interp); 244089dec819Sdrh #ifdef SQLITE_SSE 24412e66f0b9Sdrh Sqlitetestsse_Init(interp); 24422e66f0b9Sdrh #endif 2443d1bf3512Sdrh } 2444d1bf3512Sdrh #endif 24453e27c026Sdrh if( argc>=2 || TCLSH==2 ){ 2446348784efSdrh int i; 2447ad42c3a3Sshess char zArgc[32]; 2448ad42c3a3Sshess sqlite3_snprintf(sizeof(zArgc), zArgc, "%d", argc-(3-TCLSH)); 2449ad42c3a3Sshess Tcl_SetVar(interp,"argc", zArgc, TCL_GLOBAL_ONLY); 2450348784efSdrh Tcl_SetVar(interp,"argv0",argv[1],TCL_GLOBAL_ONLY); 2451348784efSdrh Tcl_SetVar(interp,"argv", "", TCL_GLOBAL_ONLY); 245261212b69Sdrh for(i=3-TCLSH; i<argc; i++){ 2453348784efSdrh Tcl_SetVar(interp, "argv", argv[i], 2454348784efSdrh TCL_GLOBAL_ONLY | TCL_LIST_ELEMENT | TCL_APPEND_VALUE); 2455348784efSdrh } 24563e27c026Sdrh if( TCLSH==1 && Tcl_EvalFile(interp, argv[1])!=TCL_OK ){ 24570de8c112Sdrh const char *zInfo = Tcl_GetVar(interp, "errorInfo", TCL_GLOBAL_ONLY); 2458c61053b7Sdrh if( zInfo==0 ) zInfo = interp->result; 2459c61053b7Sdrh fprintf(stderr,"%s: %s\n", *argv, zInfo); 2460348784efSdrh return 1; 2461348784efSdrh } 24623e27c026Sdrh } 24633e27c026Sdrh if( argc<=1 || TCLSH==2 ){ 2464348784efSdrh Tcl_GlobalEval(interp, zMainloop); 2465348784efSdrh } 2466348784efSdrh return 0; 2467348784efSdrh } 2468348784efSdrh #endif /* TCLSH */ 2469