xref: /sqlite-3.40.0/src/tclsqlite.c (revision fc57d7bf)
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 *************************************************************************
1275897234Sdrh ** A TCL Interface to SQLite
1375897234Sdrh **
14*fc57d7bfSdanielk1977 ** $Id: tclsqlite.c,v 1.70 2004/05/26 02:04:57 danielk1977 Exp $
1575897234Sdrh */
166d31316cSdrh #ifndef NO_TCL     /* Omit this whole file if TCL is unavailable */
176d31316cSdrh 
1806b2718aSdrh #include "sqliteInt.h"
1917a68934Sdrh #include "tcl.h"
2075897234Sdrh #include <stdlib.h>
2175897234Sdrh #include <string.h>
22ce927065Sdrh #include <assert.h>
2375897234Sdrh 
2475897234Sdrh /*
2598808babSdrh ** If TCL uses UTF-8 and SQLite is configured to use iso8859, then we
2698808babSdrh ** have to do a translation when going between the two.  Set the
2798808babSdrh ** UTF_TRANSLATION_NEEDED macro to indicate that we need to do
2898808babSdrh ** this translation.
2998808babSdrh */
3098808babSdrh #if defined(TCL_UTF_MAX) && !defined(SQLITE_UTF8)
3198808babSdrh # define UTF_TRANSLATION_NEEDED 1
3298808babSdrh #endif
3398808babSdrh 
3498808babSdrh /*
35cabb0819Sdrh ** New SQL functions can be created as TCL scripts.  Each such function
36cabb0819Sdrh ** is described by an instance of the following structure.
37cabb0819Sdrh */
38cabb0819Sdrh typedef struct SqlFunc SqlFunc;
39cabb0819Sdrh struct SqlFunc {
40cabb0819Sdrh   Tcl_Interp *interp;   /* The TCL interpret to execute the function */
41cabb0819Sdrh   char *zScript;        /* The script to be run */
42cabb0819Sdrh   SqlFunc *pNext;       /* Next function on the list of them all */
43cabb0819Sdrh };
44cabb0819Sdrh 
45cabb0819Sdrh /*
46bec3f402Sdrh ** There is one instance of this structure for each SQLite database
47bec3f402Sdrh ** that has been opened by the SQLite TCL interface.
48bec3f402Sdrh */
49bec3f402Sdrh typedef struct SqliteDb SqliteDb;
50bec3f402Sdrh struct SqliteDb {
51bec3f402Sdrh   sqlite *db;           /* The "real" database structure */
52bec3f402Sdrh   Tcl_Interp *interp;   /* The interpreter used for this database */
536d31316cSdrh   char *zBusy;          /* The busy callback routine */
54aa940eacSdrh   char *zCommit;        /* The commit hook callback routine */
55b5a20d3cSdrh   char *zTrace;         /* The trace callback routine */
56348bb5d6Sdanielk1977   char *zProgress;      /* The progress callback routine */
57e22a334bSdrh   char *zAuth;          /* The authorization callback routine */
58cabb0819Sdrh   SqlFunc *pFunc;       /* List of SQL functions */
596f8a503dSdanielk1977   int rc;               /* Return code of most recent sqlite3_exec() */
60bec3f402Sdrh };
61bec3f402Sdrh 
62bec3f402Sdrh /*
6375897234Sdrh ** An instance of this structure passes information thru the sqlite
6475897234Sdrh ** logic from the original TCL command into the callback routine.
6575897234Sdrh */
6675897234Sdrh typedef struct CallbackData CallbackData;
6775897234Sdrh struct CallbackData {
6875897234Sdrh   Tcl_Interp *interp;       /* The TCL interpreter */
6975897234Sdrh   char *zArray;             /* The array into which data is written */
706d31316cSdrh   Tcl_Obj *pCode;           /* The code to execute for each row */
71ce927065Sdrh   int once;                 /* Set for first callback only */
72960e8c63Sdrh   int tcl_rc;               /* Return code from TCL script */
7398808babSdrh   int nColName;             /* Number of entries in the azColName[] array */
7498808babSdrh   char **azColName;         /* Column names translated to UTF-8 */
7598808babSdrh };
76297ecf14Sdrh 
776d4abfbeSdrh #ifdef UTF_TRANSLATION_NEEDED
78297ecf14Sdrh /*
7975897234Sdrh ** Called for each row of the result.
806d4abfbeSdrh **
816d4abfbeSdrh ** This version is used when TCL expects UTF-8 data but the database
826d4abfbeSdrh ** uses the ISO8859 format.  A translation must occur from ISO8859 into
836d4abfbeSdrh ** UTF-8.
8475897234Sdrh */
8575897234Sdrh static int DbEvalCallback(
8675897234Sdrh   void *clientData,      /* An instance of CallbackData */
8775897234Sdrh   int nCol,              /* Number of columns in the result */
8875897234Sdrh   char ** azCol,         /* Data for each column */
8975897234Sdrh   char ** azN            /* Name for each column */
9075897234Sdrh ){
9175897234Sdrh   CallbackData *cbData = (CallbackData*)clientData;
9275897234Sdrh   int i, rc;
93297ecf14Sdrh   Tcl_DString dCol;
946d4abfbeSdrh   Tcl_DStringInit(&dCol);
95ce927065Sdrh   if( cbData->azColName==0 ){
96ce927065Sdrh     assert( cbData->once );
97ce927065Sdrh     cbData->once = 0;
98ce927065Sdrh     if( cbData->zArray[0] ){
996d4abfbeSdrh       Tcl_SetVar2(cbData->interp, cbData->zArray, "*", "", 0);
100ce927065Sdrh     }
1016d4abfbeSdrh     cbData->azColName = malloc( nCol*sizeof(char*) );
1026d4abfbeSdrh     if( cbData->azColName==0 ){ return 1; }
1036d4abfbeSdrh     cbData->nColName = nCol;
1046d4abfbeSdrh     for(i=0; i<nCol; i++){
1056d4abfbeSdrh       Tcl_ExternalToUtfDString(NULL, azN[i], -1, &dCol);
1066d4abfbeSdrh       cbData->azColName[i] = malloc( Tcl_DStringLength(&dCol) + 1 );
1076d4abfbeSdrh       if( cbData->azColName[i] ){
1086d4abfbeSdrh         strcpy(cbData->azColName[i], Tcl_DStringValue(&dCol));
109ce927065Sdrh       }else{
110ce927065Sdrh         return 1;
1116d4abfbeSdrh       }
112ce927065Sdrh       if( cbData->zArray[0] ){
113ce927065Sdrh         Tcl_SetVar2(cbData->interp, cbData->zArray, "*",
114ce927065Sdrh              Tcl_DStringValue(&dCol), TCL_LIST_ELEMENT|TCL_APPEND_VALUE);
115704027f1Sdrh         if( azN[nCol]!=0 ){
1165080aaa7Sdrh           Tcl_DString dType;
1175080aaa7Sdrh           Tcl_DStringInit(&dType);
118fa173a76Sdrh           Tcl_DStringAppend(&dType, "typeof:", -1);
119fa173a76Sdrh           Tcl_DStringAppend(&dType, Tcl_DStringValue(&dCol), -1);
120fa173a76Sdrh           Tcl_DStringFree(&dCol);
1215080aaa7Sdrh           Tcl_ExternalToUtfDString(NULL, azN[i+nCol], -1, &dCol);
122fa173a76Sdrh           Tcl_SetVar2(cbData->interp, cbData->zArray,
123fa173a76Sdrh                Tcl_DStringValue(&dType), Tcl_DStringValue(&dCol),
124fa173a76Sdrh                TCL_LIST_ELEMENT|TCL_APPEND_VALUE);
125fa173a76Sdrh           Tcl_DStringFree(&dType);
1266d4abfbeSdrh         }
1275080aaa7Sdrh       }
128fa173a76Sdrh 
1296d4abfbeSdrh       Tcl_DStringFree(&dCol);
1306d4abfbeSdrh     }
1316d4abfbeSdrh   }
1326d4abfbeSdrh   if( azCol!=0 ){
1336d4abfbeSdrh     if( cbData->zArray[0] ){
1346d4abfbeSdrh       for(i=0; i<nCol; i++){
1356d4abfbeSdrh         char *z = azCol[i];
1366d4abfbeSdrh         if( z==0 ) z = "";
1376d4abfbeSdrh         Tcl_DStringInit(&dCol);
1386d4abfbeSdrh         Tcl_ExternalToUtfDString(NULL, z, -1, &dCol);
1396d4abfbeSdrh         Tcl_SetVar2(cbData->interp, cbData->zArray, cbData->azColName[i],
1406d4abfbeSdrh               Tcl_DStringValue(&dCol), 0);
1416d4abfbeSdrh         Tcl_DStringFree(&dCol);
1426d4abfbeSdrh       }
1436d4abfbeSdrh     }else{
1446d4abfbeSdrh       for(i=0; i<nCol; i++){
1456d4abfbeSdrh         char *z = azCol[i];
1466d4abfbeSdrh         if( z==0 ) z = "";
1476d4abfbeSdrh         Tcl_DStringInit(&dCol);
1486d4abfbeSdrh         Tcl_ExternalToUtfDString(NULL, z, -1, &dCol);
1496d4abfbeSdrh         Tcl_SetVar(cbData->interp, cbData->azColName[i],
1506d4abfbeSdrh                    Tcl_DStringValue(&dCol), 0);
1516d4abfbeSdrh         Tcl_DStringFree(&dCol);
1526d4abfbeSdrh       }
1536d4abfbeSdrh     }
1546d4abfbeSdrh   }
1556d4abfbeSdrh   rc = Tcl_EvalObj(cbData->interp, cbData->pCode);
1566d4abfbeSdrh   if( rc==TCL_CONTINUE ) rc = TCL_OK;
1576d4abfbeSdrh   cbData->tcl_rc = rc;
1586d4abfbeSdrh   return rc!=TCL_OK;
1596d4abfbeSdrh }
1606d4abfbeSdrh #endif /* UTF_TRANSLATION_NEEDED */
1616d4abfbeSdrh 
1626d4abfbeSdrh #ifndef UTF_TRANSLATION_NEEDED
1636d4abfbeSdrh /*
1646d4abfbeSdrh ** Called for each row of the result.
1656d4abfbeSdrh **
1666d4abfbeSdrh ** This version is used when either of the following is true:
1676d4abfbeSdrh **
1686d4abfbeSdrh **    (1) This version of TCL uses UTF-8 and the data in the
1696d4abfbeSdrh **        SQLite database is already in the UTF-8 format.
1706d4abfbeSdrh **
1716d4abfbeSdrh **    (2) This version of TCL uses ISO8859 and the data in the
1726d4abfbeSdrh **        SQLite database is already in the ISO8859 format.
1736d4abfbeSdrh */
1746d4abfbeSdrh static int DbEvalCallback(
1756d4abfbeSdrh   void *clientData,      /* An instance of CallbackData */
1766d4abfbeSdrh   int nCol,              /* Number of columns in the result */
1776d4abfbeSdrh   char ** azCol,         /* Data for each column */
1786d4abfbeSdrh   char ** azN            /* Name for each column */
1796d4abfbeSdrh ){
1806d4abfbeSdrh   CallbackData *cbData = (CallbackData*)clientData;
1816d4abfbeSdrh   int i, rc;
1826a535340Sdrh   if( azCol==0 || (cbData->once && cbData->zArray[0]) ){
1839b0d0a8bSdrh     Tcl_SetVar2(cbData->interp, cbData->zArray, "*", "", 0);
18475897234Sdrh     for(i=0; i<nCol; i++){
18575897234Sdrh       Tcl_SetVar2(cbData->interp, cbData->zArray, "*", azN[i],
18675897234Sdrh          TCL_LIST_ELEMENT|TCL_APPEND_VALUE);
1875080aaa7Sdrh       if( azN[nCol] ){
1886f8a503dSdanielk1977         char *z = sqlite3_mprintf("typeof:%s", azN[i]);
1895080aaa7Sdrh         Tcl_SetVar2(cbData->interp, cbData->zArray, z, azN[i+nCol],
190fa173a76Sdrh            TCL_LIST_ELEMENT|TCL_APPEND_VALUE);
1916f8a503dSdanielk1977         sqlite3_freemem(z);
19275897234Sdrh       }
1935080aaa7Sdrh     }
1946a535340Sdrh     cbData->once = 0;
19575897234Sdrh   }
1966a535340Sdrh   if( azCol!=0 ){
1976a535340Sdrh     if( cbData->zArray[0] ){
19875897234Sdrh       for(i=0; i<nCol; i++){
199c61053b7Sdrh         char *z = azCol[i];
200c61053b7Sdrh         if( z==0 ) z = "";
201c61053b7Sdrh         Tcl_SetVar2(cbData->interp, cbData->zArray, azN[i], z, 0);
20275897234Sdrh       }
20375897234Sdrh     }else{
20475897234Sdrh       for(i=0; i<nCol; i++){
205c61053b7Sdrh         char *z = azCol[i];
206c61053b7Sdrh         if( z==0 ) z = "";
207c61053b7Sdrh         Tcl_SetVar(cbData->interp, azN[i], z, 0);
20875897234Sdrh       }
20975897234Sdrh     }
2106a535340Sdrh   }
2116d31316cSdrh   rc = Tcl_EvalObj(cbData->interp, cbData->pCode);
212960e8c63Sdrh   if( rc==TCL_CONTINUE ) rc = TCL_OK;
213960e8c63Sdrh   cbData->tcl_rc = rc;
214960e8c63Sdrh   return rc!=TCL_OK;
21575897234Sdrh }
2166d4abfbeSdrh #endif
21775897234Sdrh 
21875897234Sdrh /*
2196d31316cSdrh ** This is an alternative callback for database queries.  Instead
2206d31316cSdrh ** of invoking a TCL script to handle the result, this callback just
2216d31316cSdrh ** appends each column of the result to a list.  After the query
2226d31316cSdrh ** is complete, the list is returned.
2236d31316cSdrh */
2246d31316cSdrh static int DbEvalCallback2(
2256d31316cSdrh   void *clientData,      /* An instance of CallbackData */
2266d31316cSdrh   int nCol,              /* Number of columns in the result */
2276d31316cSdrh   char ** azCol,         /* Data for each column */
2286d31316cSdrh   char ** azN            /* Name for each column */
2296d31316cSdrh ){
2306d31316cSdrh   Tcl_Obj *pList = (Tcl_Obj*)clientData;
2316d31316cSdrh   int i;
2326a535340Sdrh   if( azCol==0 ) return 0;
2336d31316cSdrh   for(i=0; i<nCol; i++){
2346d31316cSdrh     Tcl_Obj *pElem;
2356d31316cSdrh     if( azCol[i] && *azCol[i] ){
236297ecf14Sdrh #ifdef UTF_TRANSLATION_NEEDED
237297ecf14Sdrh       Tcl_DString dCol;
238297ecf14Sdrh       Tcl_DStringInit(&dCol);
239297ecf14Sdrh       Tcl_ExternalToUtfDString(NULL, azCol[i], -1, &dCol);
240297ecf14Sdrh       pElem = Tcl_NewStringObj(Tcl_DStringValue(&dCol), -1);
241297ecf14Sdrh       Tcl_DStringFree(&dCol);
242297ecf14Sdrh #else
2436d31316cSdrh       pElem = Tcl_NewStringObj(azCol[i], -1);
244297ecf14Sdrh #endif
2456d31316cSdrh     }else{
2466d31316cSdrh       pElem = Tcl_NewObj();
2476d31316cSdrh     }
2486d31316cSdrh     Tcl_ListObjAppendElement(0, pList, pElem);
2496d31316cSdrh   }
2506d31316cSdrh   return 0;
2516d31316cSdrh }
2526d31316cSdrh 
2536d31316cSdrh /*
2545d9d7576Sdrh ** This is a second alternative callback for database queries.  A the
2555d9d7576Sdrh ** first column of the first row of the result is made the TCL result.
2565d9d7576Sdrh */
2575d9d7576Sdrh static int DbEvalCallback3(
2585d9d7576Sdrh   void *clientData,      /* An instance of CallbackData */
2595d9d7576Sdrh   int nCol,              /* Number of columns in the result */
2605d9d7576Sdrh   char ** azCol,         /* Data for each column */
2615d9d7576Sdrh   char ** azN            /* Name for each column */
2625d9d7576Sdrh ){
2635d9d7576Sdrh   Tcl_Interp *interp = (Tcl_Interp*)clientData;
2645d9d7576Sdrh   Tcl_Obj *pElem;
2655d9d7576Sdrh   if( azCol==0 ) return 1;
2665d9d7576Sdrh   if( nCol==0 ) return 1;
2675d9d7576Sdrh #ifdef UTF_TRANSLATION_NEEDED
2685d9d7576Sdrh   {
2695d9d7576Sdrh     Tcl_DString dCol;
2705d9d7576Sdrh     Tcl_DStringInit(&dCol);
2715d9d7576Sdrh     Tcl_ExternalToUtfDString(NULL, azCol[0], -1, &dCol);
2725d9d7576Sdrh     pElem = Tcl_NewStringObj(Tcl_DStringValue(&dCol), -1);
2735d9d7576Sdrh     Tcl_DStringFree(&dCol);
2745d9d7576Sdrh   }
2755d9d7576Sdrh #else
2765d9d7576Sdrh   pElem = Tcl_NewStringObj(azCol[0], -1);
2775d9d7576Sdrh #endif
2785d9d7576Sdrh   Tcl_SetObjResult(interp, pElem);
2795d9d7576Sdrh   return 1;
2805d9d7576Sdrh }
2815d9d7576Sdrh 
2825d9d7576Sdrh /*
28375897234Sdrh ** Called when the command is deleted.
28475897234Sdrh */
28575897234Sdrh static void DbDeleteCmd(void *db){
286bec3f402Sdrh   SqliteDb *pDb = (SqliteDb*)db;
2876f8a503dSdanielk1977   sqlite3_close(pDb->db);
288cabb0819Sdrh   while( pDb->pFunc ){
289cabb0819Sdrh     SqlFunc *pFunc = pDb->pFunc;
290cabb0819Sdrh     pDb->pFunc = pFunc->pNext;
291cabb0819Sdrh     Tcl_Free((char*)pFunc);
292cabb0819Sdrh   }
293bec3f402Sdrh   if( pDb->zBusy ){
294bec3f402Sdrh     Tcl_Free(pDb->zBusy);
295bec3f402Sdrh   }
296b5a20d3cSdrh   if( pDb->zTrace ){
297b5a20d3cSdrh     Tcl_Free(pDb->zTrace);
2980d1a643aSdrh   }
299e22a334bSdrh   if( pDb->zAuth ){
300e22a334bSdrh     Tcl_Free(pDb->zAuth);
301e22a334bSdrh   }
302bec3f402Sdrh   Tcl_Free((char*)pDb);
303bec3f402Sdrh }
304bec3f402Sdrh 
305bec3f402Sdrh /*
306bec3f402Sdrh ** This routine is called when a database file is locked while trying
307bec3f402Sdrh ** to execute SQL.
308bec3f402Sdrh */
309bec3f402Sdrh static int DbBusyHandler(void *cd, const char *zTable, int nTries){
310bec3f402Sdrh   SqliteDb *pDb = (SqliteDb*)cd;
311bec3f402Sdrh   int rc;
312bec3f402Sdrh   char zVal[30];
313bec3f402Sdrh   char *zCmd;
314bec3f402Sdrh   Tcl_DString cmd;
315bec3f402Sdrh 
316bec3f402Sdrh   Tcl_DStringInit(&cmd);
317bec3f402Sdrh   Tcl_DStringAppend(&cmd, pDb->zBusy, -1);
318bec3f402Sdrh   Tcl_DStringAppendElement(&cmd, zTable);
319bec3f402Sdrh   sprintf(zVal, " %d", nTries);
320bec3f402Sdrh   Tcl_DStringAppend(&cmd, zVal, -1);
321bec3f402Sdrh   zCmd = Tcl_DStringValue(&cmd);
322bec3f402Sdrh   rc = Tcl_Eval(pDb->interp, zCmd);
323bec3f402Sdrh   Tcl_DStringFree(&cmd);
324bec3f402Sdrh   if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){
325bec3f402Sdrh     return 0;
326bec3f402Sdrh   }
327bec3f402Sdrh   return 1;
32875897234Sdrh }
32975897234Sdrh 
33075897234Sdrh /*
331348bb5d6Sdanielk1977 ** This routine is invoked as the 'progress callback' for the database.
332348bb5d6Sdanielk1977 */
333348bb5d6Sdanielk1977 static int DbProgressHandler(void *cd){
334348bb5d6Sdanielk1977   SqliteDb *pDb = (SqliteDb*)cd;
335348bb5d6Sdanielk1977   int rc;
336348bb5d6Sdanielk1977 
337348bb5d6Sdanielk1977   assert( pDb->zProgress );
338348bb5d6Sdanielk1977   rc = Tcl_Eval(pDb->interp, pDb->zProgress);
339348bb5d6Sdanielk1977   if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){
340348bb5d6Sdanielk1977     return 1;
341348bb5d6Sdanielk1977   }
342348bb5d6Sdanielk1977   return 0;
343348bb5d6Sdanielk1977 }
344348bb5d6Sdanielk1977 
345348bb5d6Sdanielk1977 /*
346b5a20d3cSdrh ** This routine is called by the SQLite trace handler whenever a new
347b5a20d3cSdrh ** block of SQL is executed.  The TCL script in pDb->zTrace is executed.
3480d1a643aSdrh */
349b5a20d3cSdrh static void DbTraceHandler(void *cd, const char *zSql){
3500d1a643aSdrh   SqliteDb *pDb = (SqliteDb*)cd;
351b5a20d3cSdrh   Tcl_DString str;
3520d1a643aSdrh 
353b5a20d3cSdrh   Tcl_DStringInit(&str);
354b5a20d3cSdrh   Tcl_DStringAppend(&str, pDb->zTrace, -1);
355b5a20d3cSdrh   Tcl_DStringAppendElement(&str, zSql);
356b5a20d3cSdrh   Tcl_Eval(pDb->interp, Tcl_DStringValue(&str));
357b5a20d3cSdrh   Tcl_DStringFree(&str);
358b5a20d3cSdrh   Tcl_ResetResult(pDb->interp);
3590d1a643aSdrh }
3600d1a643aSdrh 
3610d1a643aSdrh /*
362aa940eacSdrh ** This routine is called when a transaction is committed.  The
363aa940eacSdrh ** TCL script in pDb->zCommit is executed.  If it returns non-zero or
364aa940eacSdrh ** if it throws an exception, the transaction is rolled back instead
365aa940eacSdrh ** of being committed.
366aa940eacSdrh */
367aa940eacSdrh static int DbCommitHandler(void *cd){
368aa940eacSdrh   SqliteDb *pDb = (SqliteDb*)cd;
369aa940eacSdrh   int rc;
370aa940eacSdrh 
371aa940eacSdrh   rc = Tcl_Eval(pDb->interp, pDb->zCommit);
372aa940eacSdrh   if( rc!=TCL_OK || atoi(Tcl_GetStringResult(pDb->interp)) ){
373aa940eacSdrh     return 1;
374aa940eacSdrh   }
375aa940eacSdrh   return 0;
376aa940eacSdrh }
377aa940eacSdrh 
378aa940eacSdrh /*
379cabb0819Sdrh ** This routine is called to evaluate an SQL function implemented
380cabb0819Sdrh ** using TCL script.
381cabb0819Sdrh */
3820ae8b831Sdanielk1977 static void tclSqlFunc(sqlite3_context *context, int argc, sqlite3_value **argv){
3836f8a503dSdanielk1977   SqlFunc *p = sqlite3_user_data(context);
384cabb0819Sdrh   Tcl_DString cmd;
385cabb0819Sdrh   int i;
386cabb0819Sdrh   int rc;
387cabb0819Sdrh 
388cabb0819Sdrh   Tcl_DStringInit(&cmd);
389cabb0819Sdrh   Tcl_DStringAppend(&cmd, p->zScript, -1);
390cabb0819Sdrh   for(i=0; i<argc; i++){
39151ad0ecdSdanielk1977     if( SQLITE3_NULL==sqlite3_value_type(argv[i]) ){
39251ad0ecdSdanielk1977       Tcl_DStringAppendElement(&cmd, "");
39351ad0ecdSdanielk1977     }else{
39451ad0ecdSdanielk1977       Tcl_DStringAppendElement(&cmd, sqlite3_value_data(argv[i]));
39551ad0ecdSdanielk1977     }
396cabb0819Sdrh   }
397cabb0819Sdrh   rc = Tcl_Eval(p->interp, Tcl_DStringValue(&cmd));
398cabb0819Sdrh   if( rc ){
3997e18c259Sdanielk1977     sqlite3_result_error(context, Tcl_GetStringResult(p->interp), -1);
400cabb0819Sdrh   }else{
4017e18c259Sdanielk1977     sqlite3_result_text(context, Tcl_GetStringResult(p->interp), -1, 1);
402cabb0819Sdrh   }
403cabb0819Sdrh }
404e22a334bSdrh #ifndef SQLITE_OMIT_AUTHORIZATION
405e22a334bSdrh /*
406e22a334bSdrh ** This is the authentication function.  It appends the authentication
407e22a334bSdrh ** type code and the two arguments to zCmd[] then invokes the result
408e22a334bSdrh ** on the interpreter.  The reply is examined to determine if the
409e22a334bSdrh ** authentication fails or succeeds.
410e22a334bSdrh */
411e22a334bSdrh static int auth_callback(
412e22a334bSdrh   void *pArg,
413e22a334bSdrh   int code,
414e22a334bSdrh   const char *zArg1,
415e22a334bSdrh   const char *zArg2,
416e22a334bSdrh   const char *zArg3,
417e22a334bSdrh   const char *zArg4
418e22a334bSdrh ){
419e22a334bSdrh   char *zCode;
420e22a334bSdrh   Tcl_DString str;
421e22a334bSdrh   int rc;
422e22a334bSdrh   const char *zReply;
423e22a334bSdrh   SqliteDb *pDb = (SqliteDb*)pArg;
424e22a334bSdrh 
425e22a334bSdrh   switch( code ){
426e22a334bSdrh     case SQLITE_COPY              : zCode="SQLITE_COPY"; break;
427e22a334bSdrh     case SQLITE_CREATE_INDEX      : zCode="SQLITE_CREATE_INDEX"; break;
428e22a334bSdrh     case SQLITE_CREATE_TABLE      : zCode="SQLITE_CREATE_TABLE"; break;
429e22a334bSdrh     case SQLITE_CREATE_TEMP_INDEX : zCode="SQLITE_CREATE_TEMP_INDEX"; break;
430e22a334bSdrh     case SQLITE_CREATE_TEMP_TABLE : zCode="SQLITE_CREATE_TEMP_TABLE"; break;
431e22a334bSdrh     case SQLITE_CREATE_TEMP_TRIGGER: zCode="SQLITE_CREATE_TEMP_TRIGGER"; break;
432e22a334bSdrh     case SQLITE_CREATE_TEMP_VIEW  : zCode="SQLITE_CREATE_TEMP_VIEW"; break;
433e22a334bSdrh     case SQLITE_CREATE_TRIGGER    : zCode="SQLITE_CREATE_TRIGGER"; break;
434e22a334bSdrh     case SQLITE_CREATE_VIEW       : zCode="SQLITE_CREATE_VIEW"; break;
435e22a334bSdrh     case SQLITE_DELETE            : zCode="SQLITE_DELETE"; break;
436e22a334bSdrh     case SQLITE_DROP_INDEX        : zCode="SQLITE_DROP_INDEX"; break;
437e22a334bSdrh     case SQLITE_DROP_TABLE        : zCode="SQLITE_DROP_TABLE"; break;
438e22a334bSdrh     case SQLITE_DROP_TEMP_INDEX   : zCode="SQLITE_DROP_TEMP_INDEX"; break;
439e22a334bSdrh     case SQLITE_DROP_TEMP_TABLE   : zCode="SQLITE_DROP_TEMP_TABLE"; break;
440e22a334bSdrh     case SQLITE_DROP_TEMP_TRIGGER : zCode="SQLITE_DROP_TEMP_TRIGGER"; break;
441e22a334bSdrh     case SQLITE_DROP_TEMP_VIEW    : zCode="SQLITE_DROP_TEMP_VIEW"; break;
442e22a334bSdrh     case SQLITE_DROP_TRIGGER      : zCode="SQLITE_DROP_TRIGGER"; break;
443e22a334bSdrh     case SQLITE_DROP_VIEW         : zCode="SQLITE_DROP_VIEW"; break;
444e22a334bSdrh     case SQLITE_INSERT            : zCode="SQLITE_INSERT"; break;
445e22a334bSdrh     case SQLITE_PRAGMA            : zCode="SQLITE_PRAGMA"; break;
446e22a334bSdrh     case SQLITE_READ              : zCode="SQLITE_READ"; break;
447e22a334bSdrh     case SQLITE_SELECT            : zCode="SQLITE_SELECT"; break;
448e22a334bSdrh     case SQLITE_TRANSACTION       : zCode="SQLITE_TRANSACTION"; break;
449e22a334bSdrh     case SQLITE_UPDATE            : zCode="SQLITE_UPDATE"; break;
45081e293b4Sdrh     case SQLITE_ATTACH            : zCode="SQLITE_ATTACH"; break;
45181e293b4Sdrh     case SQLITE_DETACH            : zCode="SQLITE_DETACH"; break;
452e22a334bSdrh     default                       : zCode="????"; break;
453e22a334bSdrh   }
454e22a334bSdrh   Tcl_DStringInit(&str);
455e22a334bSdrh   Tcl_DStringAppend(&str, pDb->zAuth, -1);
456e22a334bSdrh   Tcl_DStringAppendElement(&str, zCode);
457e22a334bSdrh   Tcl_DStringAppendElement(&str, zArg1 ? zArg1 : "");
458e22a334bSdrh   Tcl_DStringAppendElement(&str, zArg2 ? zArg2 : "");
459e22a334bSdrh   Tcl_DStringAppendElement(&str, zArg3 ? zArg3 : "");
460e22a334bSdrh   Tcl_DStringAppendElement(&str, zArg4 ? zArg4 : "");
461e22a334bSdrh   rc = Tcl_GlobalEval(pDb->interp, Tcl_DStringValue(&str));
462e22a334bSdrh   Tcl_DStringFree(&str);
463e22a334bSdrh   zReply = Tcl_GetStringResult(pDb->interp);
464e22a334bSdrh   if( strcmp(zReply,"SQLITE_OK")==0 ){
465e22a334bSdrh     rc = SQLITE_OK;
466e22a334bSdrh   }else if( strcmp(zReply,"SQLITE_DENY")==0 ){
467e22a334bSdrh     rc = SQLITE_DENY;
468e22a334bSdrh   }else if( strcmp(zReply,"SQLITE_IGNORE")==0 ){
469e22a334bSdrh     rc = SQLITE_IGNORE;
470e22a334bSdrh   }else{
471e22a334bSdrh     rc = 999;
472e22a334bSdrh   }
473e22a334bSdrh   return rc;
474e22a334bSdrh }
475e22a334bSdrh #endif /* SQLITE_OMIT_AUTHORIZATION */
476cabb0819Sdrh 
477cabb0819Sdrh /*
47875897234Sdrh ** The "sqlite" command below creates a new Tcl command for each
47975897234Sdrh ** connection it opens to an SQLite database.  This routine is invoked
48075897234Sdrh ** whenever one of those connection-specific commands is executed
48175897234Sdrh ** in Tcl.  For example, if you run Tcl code like this:
48275897234Sdrh **
48375897234Sdrh **       sqlite db1  "my_database"
48475897234Sdrh **       db1 close
48575897234Sdrh **
48675897234Sdrh ** The first command opens a connection to the "my_database" database
48775897234Sdrh ** and calls that connection "db1".  The second command causes this
48875897234Sdrh ** subroutine to be invoked.
48975897234Sdrh */
4906d31316cSdrh static int DbObjCmd(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){
491bec3f402Sdrh   SqliteDb *pDb = (SqliteDb*)cd;
4926d31316cSdrh   int choice;
49322fbcb8dSdrh   int rc = TCL_OK;
4940de8c112Sdrh   static const char *DB_strs[] = {
495b5a20d3cSdrh     "authorizer",         "busy",                   "changes",
496aa940eacSdrh     "close",              "commit_hook",            "complete",
497aa940eacSdrh     "errorcode",          "eval",                   "function",
498f146a776Srdc     "last_insert_rowid",  "last_statement_changes", "onecolumn",
499f146a776Srdc     "progress",           "rekey",                  "timeout",
500f146a776Srdc     "trace",
50122fbcb8dSdrh     0
5026d31316cSdrh   };
503411995dcSdrh   enum DB_enum {
504b5a20d3cSdrh     DB_AUTHORIZER,        DB_BUSY,                   DB_CHANGES,
505aa940eacSdrh     DB_CLOSE,             DB_COMMIT_HOOK,            DB_COMPLETE,
506aa940eacSdrh     DB_ERRORCODE,         DB_EVAL,                   DB_FUNCTION,
507f146a776Srdc     DB_LAST_INSERT_ROWID, DB_LAST_STATEMENT_CHANGES, DB_ONECOLUMN,
508f146a776Srdc     DB_PROGRESS,          DB_REKEY,                  DB_TIMEOUT,
509f146a776Srdc     DB_TRACE
5106d31316cSdrh   };
5116d31316cSdrh 
5126d31316cSdrh   if( objc<2 ){
5136d31316cSdrh     Tcl_WrongNumArgs(interp, 1, objv, "SUBCOMMAND ...");
51475897234Sdrh     return TCL_ERROR;
51575897234Sdrh   }
516411995dcSdrh   if( Tcl_GetIndexFromObj(interp, objv[1], DB_strs, "option", 0, &choice) ){
5176d31316cSdrh     return TCL_ERROR;
5186d31316cSdrh   }
5196d31316cSdrh 
520411995dcSdrh   switch( (enum DB_enum)choice ){
52175897234Sdrh 
522e22a334bSdrh   /*    $db authorizer ?CALLBACK?
523e22a334bSdrh   **
524e22a334bSdrh   ** Invoke the given callback to authorize each SQL operation as it is
525e22a334bSdrh   ** compiled.  5 arguments are appended to the callback before it is
526e22a334bSdrh   ** invoked:
527e22a334bSdrh   **
528e22a334bSdrh   **   (1) The authorization type (ex: SQLITE_CREATE_TABLE, SQLITE_INSERT, ...)
529e22a334bSdrh   **   (2) First descriptive name (depends on authorization type)
530e22a334bSdrh   **   (3) Second descriptive name
531e22a334bSdrh   **   (4) Name of the database (ex: "main", "temp")
532e22a334bSdrh   **   (5) Name of trigger that is doing the access
533e22a334bSdrh   **
534e22a334bSdrh   ** The callback should return on of the following strings: SQLITE_OK,
535e22a334bSdrh   ** SQLITE_IGNORE, or SQLITE_DENY.  Any other return value is an error.
536e22a334bSdrh   **
537e22a334bSdrh   ** If this method is invoked with no arguments, the current authorization
538e22a334bSdrh   ** callback string is returned.
539e22a334bSdrh   */
540e22a334bSdrh   case DB_AUTHORIZER: {
541e22a334bSdrh     if( objc>3 ){
542e22a334bSdrh       Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?");
543e22a334bSdrh     }else if( objc==2 ){
544b5a20d3cSdrh       if( pDb->zAuth ){
545e22a334bSdrh         Tcl_AppendResult(interp, pDb->zAuth, 0);
546e22a334bSdrh       }
547e22a334bSdrh     }else{
548e22a334bSdrh       char *zAuth;
549e22a334bSdrh       int len;
550e22a334bSdrh       if( pDb->zAuth ){
551e22a334bSdrh         Tcl_Free(pDb->zAuth);
552e22a334bSdrh       }
553e22a334bSdrh       zAuth = Tcl_GetStringFromObj(objv[2], &len);
554e22a334bSdrh       if( zAuth && len>0 ){
555e22a334bSdrh         pDb->zAuth = Tcl_Alloc( len + 1 );
556e22a334bSdrh         strcpy(pDb->zAuth, zAuth);
557e22a334bSdrh       }else{
558e22a334bSdrh         pDb->zAuth = 0;
559e22a334bSdrh       }
560e22a334bSdrh #ifndef SQLITE_OMIT_AUTHORIZATION
561e22a334bSdrh       if( pDb->zAuth ){
562e22a334bSdrh         pDb->interp = interp;
5636f8a503dSdanielk1977         sqlite3_set_authorizer(pDb->db, auth_callback, pDb);
564e22a334bSdrh       }else{
5656f8a503dSdanielk1977         sqlite3_set_authorizer(pDb->db, 0, 0);
566e22a334bSdrh       }
567e22a334bSdrh #endif
568e22a334bSdrh     }
569e22a334bSdrh     break;
570e22a334bSdrh   }
571e22a334bSdrh 
572bec3f402Sdrh   /*    $db busy ?CALLBACK?
573bec3f402Sdrh   **
574bec3f402Sdrh   ** Invoke the given callback if an SQL statement attempts to open
575bec3f402Sdrh   ** a locked database file.
576bec3f402Sdrh   */
5776d31316cSdrh   case DB_BUSY: {
5786d31316cSdrh     if( objc>3 ){
5796d31316cSdrh       Tcl_WrongNumArgs(interp, 2, objv, "CALLBACK");
580bec3f402Sdrh       return TCL_ERROR;
5816d31316cSdrh     }else if( objc==2 ){
582bec3f402Sdrh       if( pDb->zBusy ){
583bec3f402Sdrh         Tcl_AppendResult(interp, pDb->zBusy, 0);
584bec3f402Sdrh       }
585bec3f402Sdrh     }else{
5866d31316cSdrh       char *zBusy;
5876d31316cSdrh       int len;
588bec3f402Sdrh       if( pDb->zBusy ){
589bec3f402Sdrh         Tcl_Free(pDb->zBusy);
5906d31316cSdrh       }
5916d31316cSdrh       zBusy = Tcl_GetStringFromObj(objv[2], &len);
5926d31316cSdrh       if( zBusy && len>0 ){
5936d31316cSdrh         pDb->zBusy = Tcl_Alloc( len + 1 );
5946d31316cSdrh         strcpy(pDb->zBusy, zBusy);
5956d31316cSdrh       }else{
596bec3f402Sdrh         pDb->zBusy = 0;
597bec3f402Sdrh       }
598bec3f402Sdrh       if( pDb->zBusy ){
599bec3f402Sdrh         pDb->interp = interp;
6006f8a503dSdanielk1977         sqlite3_busy_handler(pDb->db, DbBusyHandler, pDb);
6016d31316cSdrh       }else{
6026f8a503dSdanielk1977         sqlite3_busy_handler(pDb->db, 0, 0);
603bec3f402Sdrh       }
604bec3f402Sdrh     }
6056d31316cSdrh     break;
6066d31316cSdrh   }
607bec3f402Sdrh 
608348bb5d6Sdanielk1977   /*    $db progress ?N CALLBACK?
609348bb5d6Sdanielk1977   **
610348bb5d6Sdanielk1977   ** Invoke the given callback every N virtual machine opcodes while executing
611348bb5d6Sdanielk1977   ** queries.
612348bb5d6Sdanielk1977   */
613348bb5d6Sdanielk1977   case DB_PROGRESS: {
614348bb5d6Sdanielk1977     if( objc==2 ){
615348bb5d6Sdanielk1977       if( pDb->zProgress ){
616348bb5d6Sdanielk1977         Tcl_AppendResult(interp, pDb->zProgress, 0);
617348bb5d6Sdanielk1977       }
618348bb5d6Sdanielk1977     }else if( objc==4 ){
619348bb5d6Sdanielk1977       char *zProgress;
620348bb5d6Sdanielk1977       int len;
621348bb5d6Sdanielk1977       int N;
622348bb5d6Sdanielk1977       if( TCL_OK!=Tcl_GetIntFromObj(interp, objv[2], &N) ){
623348bb5d6Sdanielk1977 	return TCL_ERROR;
624348bb5d6Sdanielk1977       };
625348bb5d6Sdanielk1977       if( pDb->zProgress ){
626348bb5d6Sdanielk1977         Tcl_Free(pDb->zProgress);
627348bb5d6Sdanielk1977       }
628348bb5d6Sdanielk1977       zProgress = Tcl_GetStringFromObj(objv[3], &len);
629348bb5d6Sdanielk1977       if( zProgress && len>0 ){
630348bb5d6Sdanielk1977         pDb->zProgress = Tcl_Alloc( len + 1 );
631348bb5d6Sdanielk1977         strcpy(pDb->zProgress, zProgress);
632348bb5d6Sdanielk1977       }else{
633348bb5d6Sdanielk1977         pDb->zProgress = 0;
634348bb5d6Sdanielk1977       }
635348bb5d6Sdanielk1977 #ifndef SQLITE_OMIT_PROGRESS_CALLBACK
636348bb5d6Sdanielk1977       if( pDb->zProgress ){
637348bb5d6Sdanielk1977         pDb->interp = interp;
6386f8a503dSdanielk1977         sqlite3_progress_handler(pDb->db, N, DbProgressHandler, pDb);
639348bb5d6Sdanielk1977       }else{
6406f8a503dSdanielk1977         sqlite3_progress_handler(pDb->db, 0, 0, 0);
641348bb5d6Sdanielk1977       }
642348bb5d6Sdanielk1977 #endif
643348bb5d6Sdanielk1977     }else{
644348bb5d6Sdanielk1977       Tcl_WrongNumArgs(interp, 2, objv, "N CALLBACK");
645348bb5d6Sdanielk1977       return TCL_ERROR;
646348bb5d6Sdanielk1977     }
647348bb5d6Sdanielk1977     break;
648348bb5d6Sdanielk1977   }
649348bb5d6Sdanielk1977 
650c8d30ac1Sdrh   /*
651c8d30ac1Sdrh   **     $db changes
652c8d30ac1Sdrh   **
653c8d30ac1Sdrh   ** Return the number of rows that were modified, inserted, or deleted by
654c8d30ac1Sdrh   ** the most recent "eval".
655c8d30ac1Sdrh   */
656c8d30ac1Sdrh   case DB_CHANGES: {
657c8d30ac1Sdrh     Tcl_Obj *pResult;
658c8d30ac1Sdrh     int nChange;
659c8d30ac1Sdrh     if( objc!=2 ){
660c8d30ac1Sdrh       Tcl_WrongNumArgs(interp, 2, objv, "");
661c8d30ac1Sdrh       return TCL_ERROR;
662c8d30ac1Sdrh     }
6636f8a503dSdanielk1977     nChange = sqlite3_changes(pDb->db);
664c8d30ac1Sdrh     pResult = Tcl_GetObjResult(interp);
665c8d30ac1Sdrh     Tcl_SetIntObj(pResult, nChange);
666c8d30ac1Sdrh     break;
667c8d30ac1Sdrh   }
668c8d30ac1Sdrh 
669f146a776Srdc   /*
670f146a776Srdc   **     $db last_statement_changes
671f146a776Srdc   **
672f146a776Srdc   ** Return the number of rows that were modified, inserted, or deleted by
673f146a776Srdc   ** the last statment to complete execution (excluding changes due to
674f146a776Srdc   ** triggers)
675f146a776Srdc   */
676f146a776Srdc   case DB_LAST_STATEMENT_CHANGES: {
677f146a776Srdc     Tcl_Obj *pResult;
678f146a776Srdc     int lsChange;
679f146a776Srdc     if( objc!=2 ){
680f146a776Srdc       Tcl_WrongNumArgs(interp, 2, objv, "");
681f146a776Srdc       return TCL_ERROR;
682f146a776Srdc     }
6836f8a503dSdanielk1977     lsChange = sqlite3_last_statement_changes(pDb->db);
684f146a776Srdc     pResult = Tcl_GetObjResult(interp);
685f146a776Srdc     Tcl_SetIntObj(pResult, lsChange);
686f146a776Srdc     break;
687f146a776Srdc   }
688f146a776Srdc 
68975897234Sdrh   /*    $db close
69075897234Sdrh   **
69175897234Sdrh   ** Shutdown the database
69275897234Sdrh   */
6936d31316cSdrh   case DB_CLOSE: {
6946d31316cSdrh     Tcl_DeleteCommand(interp, Tcl_GetStringFromObj(objv[0], 0));
6956d31316cSdrh     break;
6966d31316cSdrh   }
69775897234Sdrh 
698aa940eacSdrh   /*    $db commit_hook ?CALLBACK?
699aa940eacSdrh   **
700aa940eacSdrh   ** Invoke the given callback just before committing every SQL transaction.
701aa940eacSdrh   ** If the callback throws an exception or returns non-zero, then the
702aa940eacSdrh   ** transaction is aborted.  If CALLBACK is an empty string, the callback
703aa940eacSdrh   ** is disabled.
704aa940eacSdrh   */
705aa940eacSdrh   case DB_COMMIT_HOOK: {
706aa940eacSdrh     if( objc>3 ){
707aa940eacSdrh       Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?");
708aa940eacSdrh     }else if( objc==2 ){
709aa940eacSdrh       if( pDb->zCommit ){
710aa940eacSdrh         Tcl_AppendResult(interp, pDb->zCommit, 0);
711aa940eacSdrh       }
712aa940eacSdrh     }else{
713aa940eacSdrh       char *zCommit;
714aa940eacSdrh       int len;
715aa940eacSdrh       if( pDb->zCommit ){
716aa940eacSdrh         Tcl_Free(pDb->zCommit);
717aa940eacSdrh       }
718aa940eacSdrh       zCommit = Tcl_GetStringFromObj(objv[2], &len);
719aa940eacSdrh       if( zCommit && len>0 ){
720aa940eacSdrh         pDb->zCommit = Tcl_Alloc( len + 1 );
721aa940eacSdrh         strcpy(pDb->zCommit, zCommit);
722aa940eacSdrh       }else{
723aa940eacSdrh         pDb->zCommit = 0;
724aa940eacSdrh       }
725aa940eacSdrh       if( pDb->zCommit ){
726aa940eacSdrh         pDb->interp = interp;
7276f8a503dSdanielk1977         sqlite3_commit_hook(pDb->db, DbCommitHandler, pDb);
728aa940eacSdrh       }else{
7296f8a503dSdanielk1977         sqlite3_commit_hook(pDb->db, 0, 0);
730aa940eacSdrh       }
731aa940eacSdrh     }
732aa940eacSdrh     break;
733aa940eacSdrh   }
734aa940eacSdrh 
73575897234Sdrh   /*    $db complete SQL
73675897234Sdrh   **
73775897234Sdrh   ** Return TRUE if SQL is a complete SQL statement.  Return FALSE if
73875897234Sdrh   ** additional lines of input are needed.  This is similar to the
73975897234Sdrh   ** built-in "info complete" command of Tcl.
74075897234Sdrh   */
7416d31316cSdrh   case DB_COMPLETE: {
7426d31316cSdrh     Tcl_Obj *pResult;
7436d31316cSdrh     int isComplete;
7446d31316cSdrh     if( objc!=3 ){
7456d31316cSdrh       Tcl_WrongNumArgs(interp, 2, objv, "SQL");
74675897234Sdrh       return TCL_ERROR;
74775897234Sdrh     }
7486f8a503dSdanielk1977     isComplete = sqlite3_complete( Tcl_GetStringFromObj(objv[2], 0) );
7496d31316cSdrh     pResult = Tcl_GetObjResult(interp);
7506d31316cSdrh     Tcl_SetBooleanObj(pResult, isComplete);
7516d31316cSdrh     break;
7526d31316cSdrh   }
75375897234Sdrh 
75475897234Sdrh   /*
755dcd997eaSdrh   **    $db errorcode
756dcd997eaSdrh   **
757dcd997eaSdrh   ** Return the numeric error code that was returned by the most recent
7586f8a503dSdanielk1977   ** call to sqlite3_exec().
759dcd997eaSdrh   */
760dcd997eaSdrh   case DB_ERRORCODE: {
761dcd997eaSdrh     Tcl_SetObjResult(interp, Tcl_NewIntObj(pDb->rc));
762dcd997eaSdrh     break;
763dcd997eaSdrh   }
764dcd997eaSdrh 
765dcd997eaSdrh   /*
76675897234Sdrh   **    $db eval $sql ?array {  ...code... }?
76775897234Sdrh   **
76875897234Sdrh   ** The SQL statement in $sql is evaluated.  For each row, the values are
769bec3f402Sdrh   ** placed in elements of the array named "array" and ...code... is executed.
77075897234Sdrh   ** If "array" and "code" are omitted, then no callback is every invoked.
77175897234Sdrh   ** If "array" is an empty string, then the values are placed in variables
77275897234Sdrh   ** that have the same name as the fields extracted by the query.
77375897234Sdrh   */
7746d31316cSdrh   case DB_EVAL: {
77575897234Sdrh     CallbackData cbData;
77675897234Sdrh     char *zErrMsg;
7776d31316cSdrh     char *zSql;
778297ecf14Sdrh #ifdef UTF_TRANSLATION_NEEDED
779297ecf14Sdrh     Tcl_DString dSql;
7806d4abfbeSdrh     int i;
781297ecf14Sdrh #endif
78275897234Sdrh 
7836d31316cSdrh     if( objc!=5 && objc!=3 ){
7846d31316cSdrh       Tcl_WrongNumArgs(interp, 2, objv, "SQL ?ARRAY-NAME CODE?");
78575897234Sdrh       return TCL_ERROR;
78675897234Sdrh     }
787bec3f402Sdrh     pDb->interp = interp;
7886d31316cSdrh     zSql = Tcl_GetStringFromObj(objv[2], 0);
789297ecf14Sdrh #ifdef UTF_TRANSLATION_NEEDED
790297ecf14Sdrh     Tcl_DStringInit(&dSql);
791297ecf14Sdrh     Tcl_UtfToExternalDString(NULL, zSql, -1, &dSql);
792297ecf14Sdrh     zSql = Tcl_DStringValue(&dSql);
793297ecf14Sdrh #endif
7946d31316cSdrh     Tcl_IncrRefCount(objv[2]);
7956d31316cSdrh     if( objc==5 ){
79675897234Sdrh       cbData.interp = interp;
797dcc581ccSdrh       cbData.once = 1;
7986d31316cSdrh       cbData.zArray = Tcl_GetStringFromObj(objv[3], 0);
7996d31316cSdrh       cbData.pCode = objv[4];
800960e8c63Sdrh       cbData.tcl_rc = TCL_OK;
8016d4abfbeSdrh       cbData.nColName = 0;
8026d4abfbeSdrh       cbData.azColName = 0;
80375897234Sdrh       zErrMsg = 0;
8046d31316cSdrh       Tcl_IncrRefCount(objv[3]);
8056d31316cSdrh       Tcl_IncrRefCount(objv[4]);
8066f8a503dSdanielk1977       rc = sqlite3_exec(pDb->db, zSql, DbEvalCallback, &cbData, &zErrMsg);
8076d31316cSdrh       Tcl_DecrRefCount(objv[4]);
8086d31316cSdrh       Tcl_DecrRefCount(objv[3]);
809960e8c63Sdrh       if( cbData.tcl_rc==TCL_BREAK ){ cbData.tcl_rc = TCL_OK; }
81075897234Sdrh     }else{
8116d31316cSdrh       Tcl_Obj *pList = Tcl_NewObj();
812960e8c63Sdrh       cbData.tcl_rc = TCL_OK;
8136f8a503dSdanielk1977       rc = sqlite3_exec(pDb->db, zSql, DbEvalCallback2, pList, &zErrMsg);
8146d31316cSdrh       Tcl_SetObjResult(interp, pList);
81575897234Sdrh     }
816dcd997eaSdrh     pDb->rc = rc;
817b798fa64Sdrh     if( rc==SQLITE_ABORT ){
818b798fa64Sdrh       if( zErrMsg ) free(zErrMsg);
819b798fa64Sdrh       rc = cbData.tcl_rc;
820b798fa64Sdrh     }else if( zErrMsg ){
82175897234Sdrh       Tcl_SetResult(interp, zErrMsg, TCL_VOLATILE);
82275897234Sdrh       free(zErrMsg);
823960e8c63Sdrh       rc = TCL_ERROR;
824b798fa64Sdrh     }else if( rc!=SQLITE_OK ){
8256f8a503dSdanielk1977       Tcl_AppendResult(interp, sqlite3_error_string(rc), 0);
8266d4abfbeSdrh       rc = TCL_ERROR;
827960e8c63Sdrh     }else{
82875897234Sdrh     }
8296d31316cSdrh     Tcl_DecrRefCount(objv[2]);
830297ecf14Sdrh #ifdef UTF_TRANSLATION_NEEDED
831297ecf14Sdrh     Tcl_DStringFree(&dSql);
8326d4abfbeSdrh     if( objc==5 && cbData.azColName ){
8336d4abfbeSdrh       for(i=0; i<cbData.nColName; i++){
8346d4abfbeSdrh         if( cbData.azColName[i] ) free(cbData.azColName[i]);
8356d4abfbeSdrh       }
8366d4abfbeSdrh       free(cbData.azColName);
837ce927065Sdrh       cbData.azColName = 0;
8386d4abfbeSdrh     }
839297ecf14Sdrh #endif
84075897234Sdrh     return rc;
8416d31316cSdrh   }
842bec3f402Sdrh 
843bec3f402Sdrh   /*
844cabb0819Sdrh   **     $db function NAME SCRIPT
845cabb0819Sdrh   **
846cabb0819Sdrh   ** Create a new SQL function called NAME.  Whenever that function is
847cabb0819Sdrh   ** called, invoke SCRIPT to evaluate the function.
848cabb0819Sdrh   */
849cabb0819Sdrh   case DB_FUNCTION: {
850cabb0819Sdrh     SqlFunc *pFunc;
851cabb0819Sdrh     char *zName;
852cabb0819Sdrh     char *zScript;
853cabb0819Sdrh     int nScript;
854cabb0819Sdrh     if( objc!=4 ){
855cabb0819Sdrh       Tcl_WrongNumArgs(interp, 2, objv, "NAME SCRIPT");
856cabb0819Sdrh       return TCL_ERROR;
857cabb0819Sdrh     }
858cabb0819Sdrh     zName = Tcl_GetStringFromObj(objv[2], 0);
859cabb0819Sdrh     zScript = Tcl_GetStringFromObj(objv[3], &nScript);
860cabb0819Sdrh     pFunc = (SqlFunc*)Tcl_Alloc( sizeof(*pFunc) + nScript + 1 );
861cabb0819Sdrh     if( pFunc==0 ) return TCL_ERROR;
862cabb0819Sdrh     pFunc->interp = interp;
863cabb0819Sdrh     pFunc->pNext = pDb->pFunc;
864cabb0819Sdrh     pFunc->zScript = (char*)&pFunc[1];
865cabb0819Sdrh     strcpy(pFunc->zScript, zScript);
8666f8a503dSdanielk1977     sqlite3_create_function(pDb->db, zName, -1, tclSqlFunc, pFunc);
8676f8a503dSdanielk1977     sqlite3_function_type(pDb->db, zName, SQLITE_NUMERIC);
868cabb0819Sdrh     break;
869cabb0819Sdrh   }
870cabb0819Sdrh 
871cabb0819Sdrh   /*
872af9ff33aSdrh   **     $db last_insert_rowid
873af9ff33aSdrh   **
874af9ff33aSdrh   ** Return an integer which is the ROWID for the most recent insert.
875af9ff33aSdrh   */
876af9ff33aSdrh   case DB_LAST_INSERT_ROWID: {
877af9ff33aSdrh     Tcl_Obj *pResult;
878af9ff33aSdrh     int rowid;
879af9ff33aSdrh     if( objc!=2 ){
880af9ff33aSdrh       Tcl_WrongNumArgs(interp, 2, objv, "");
881af9ff33aSdrh       return TCL_ERROR;
882af9ff33aSdrh     }
8836f8a503dSdanielk1977     rowid = sqlite3_last_insert_rowid(pDb->db);
884af9ff33aSdrh     pResult = Tcl_GetObjResult(interp);
885af9ff33aSdrh     Tcl_SetIntObj(pResult, rowid);
886af9ff33aSdrh     break;
887af9ff33aSdrh   }
888af9ff33aSdrh 
889af9ff33aSdrh   /*
8905d9d7576Sdrh   **     $db onecolumn SQL
8915d9d7576Sdrh   **
8925d9d7576Sdrh   ** Return a single column from a single row of the given SQL query.
8935d9d7576Sdrh   */
8945d9d7576Sdrh   case DB_ONECOLUMN: {
8955d9d7576Sdrh     char *zSql;
8965d9d7576Sdrh     char *zErrMsg = 0;
8975d9d7576Sdrh     if( objc!=3 ){
8985d9d7576Sdrh       Tcl_WrongNumArgs(interp, 2, objv, "SQL");
8995d9d7576Sdrh       return TCL_ERROR;
9005d9d7576Sdrh     }
9015d9d7576Sdrh     zSql = Tcl_GetStringFromObj(objv[2], 0);
9026f8a503dSdanielk1977     rc = sqlite3_exec(pDb->db, zSql, DbEvalCallback3, interp, &zErrMsg);
9035d9d7576Sdrh     if( rc==SQLITE_ABORT ){
90422fbcb8dSdrh       rc = SQLITE_OK;
9055d9d7576Sdrh     }else if( zErrMsg ){
9065d9d7576Sdrh       Tcl_SetResult(interp, zErrMsg, TCL_VOLATILE);
9075d9d7576Sdrh       free(zErrMsg);
9085d9d7576Sdrh       rc = TCL_ERROR;
9095d9d7576Sdrh     }else if( rc!=SQLITE_OK ){
9106f8a503dSdanielk1977       Tcl_AppendResult(interp, sqlite3_error_string(rc), 0);
9115d9d7576Sdrh       rc = TCL_ERROR;
9125d9d7576Sdrh     }
9135d9d7576Sdrh     break;
9145d9d7576Sdrh   }
9155d9d7576Sdrh 
9165d9d7576Sdrh   /*
91722fbcb8dSdrh   **     $db rekey KEY
91822fbcb8dSdrh   **
91922fbcb8dSdrh   ** Change the encryption key on the currently open database.
92022fbcb8dSdrh   */
92122fbcb8dSdrh   case DB_REKEY: {
92222fbcb8dSdrh     int nKey;
92322fbcb8dSdrh     void *pKey;
92422fbcb8dSdrh     if( objc!=3 ){
92522fbcb8dSdrh       Tcl_WrongNumArgs(interp, 2, objv, "KEY");
92622fbcb8dSdrh       return TCL_ERROR;
92722fbcb8dSdrh     }
92822fbcb8dSdrh     pKey = Tcl_GetByteArrayFromObj(objv[2], &nKey);
9299eb9e26bSdrh #ifdef SQLITE_HAS_CODEC
93022fbcb8dSdrh     rc = sqlite_rekey(pDb->db, pKey, nKey);
93122fbcb8dSdrh     if( rc ){
9326f8a503dSdanielk1977       Tcl_AppendResult(interp, sqlite3_error_string(rc), 0);
93322fbcb8dSdrh       rc = TCL_ERROR;
93422fbcb8dSdrh     }
93522fbcb8dSdrh #endif
93622fbcb8dSdrh     break;
93722fbcb8dSdrh   }
93822fbcb8dSdrh 
93922fbcb8dSdrh   /*
940bec3f402Sdrh   **     $db timeout MILLESECONDS
941bec3f402Sdrh   **
942bec3f402Sdrh   ** Delay for the number of milliseconds specified when a file is locked.
943bec3f402Sdrh   */
9446d31316cSdrh   case DB_TIMEOUT: {
945bec3f402Sdrh     int ms;
9466d31316cSdrh     if( objc!=3 ){
9476d31316cSdrh       Tcl_WrongNumArgs(interp, 2, objv, "MILLISECONDS");
948bec3f402Sdrh       return TCL_ERROR;
94975897234Sdrh     }
9506d31316cSdrh     if( Tcl_GetIntFromObj(interp, objv[2], &ms) ) return TCL_ERROR;
9516f8a503dSdanielk1977     sqlite3_busy_timeout(pDb->db, ms);
9526d31316cSdrh     break;
95375897234Sdrh   }
954b5a20d3cSdrh 
955b5a20d3cSdrh   /*    $db trace ?CALLBACK?
956b5a20d3cSdrh   **
957b5a20d3cSdrh   ** Make arrangements to invoke the CALLBACK routine for each SQL statement
958b5a20d3cSdrh   ** that is executed.  The text of the SQL is appended to CALLBACK before
959b5a20d3cSdrh   ** it is executed.
960b5a20d3cSdrh   */
961b5a20d3cSdrh   case DB_TRACE: {
962b5a20d3cSdrh     if( objc>3 ){
963b5a20d3cSdrh       Tcl_WrongNumArgs(interp, 2, objv, "?CALLBACK?");
964b5a20d3cSdrh     }else if( objc==2 ){
965b5a20d3cSdrh       if( pDb->zTrace ){
966b5a20d3cSdrh         Tcl_AppendResult(interp, pDb->zTrace, 0);
967b5a20d3cSdrh       }
968b5a20d3cSdrh     }else{
969b5a20d3cSdrh       char *zTrace;
970b5a20d3cSdrh       int len;
971b5a20d3cSdrh       if( pDb->zTrace ){
972b5a20d3cSdrh         Tcl_Free(pDb->zTrace);
973b5a20d3cSdrh       }
974b5a20d3cSdrh       zTrace = Tcl_GetStringFromObj(objv[2], &len);
975b5a20d3cSdrh       if( zTrace && len>0 ){
976b5a20d3cSdrh         pDb->zTrace = Tcl_Alloc( len + 1 );
977b5a20d3cSdrh         strcpy(pDb->zTrace, zTrace);
978b5a20d3cSdrh       }else{
979b5a20d3cSdrh         pDb->zTrace = 0;
980b5a20d3cSdrh       }
981b5a20d3cSdrh       if( pDb->zTrace ){
982b5a20d3cSdrh         pDb->interp = interp;
9836f8a503dSdanielk1977         sqlite3_trace(pDb->db, DbTraceHandler, pDb);
984b5a20d3cSdrh       }else{
9856f8a503dSdanielk1977         sqlite3_trace(pDb->db, 0, 0);
986b5a20d3cSdrh       }
987b5a20d3cSdrh     }
988b5a20d3cSdrh     break;
989b5a20d3cSdrh   }
990b5a20d3cSdrh 
9916d31316cSdrh   } /* End of the SWITCH statement */
99222fbcb8dSdrh   return rc;
99375897234Sdrh }
99475897234Sdrh 
99575897234Sdrh /*
99622fbcb8dSdrh **   sqlite DBNAME FILENAME ?MODE? ?-key KEY?
99775897234Sdrh **
99875897234Sdrh ** This is the main Tcl command.  When the "sqlite" Tcl command is
99975897234Sdrh ** invoked, this routine runs to process that command.
100075897234Sdrh **
100175897234Sdrh ** The first argument, DBNAME, is an arbitrary name for a new
100275897234Sdrh ** database connection.  This command creates a new command named
100375897234Sdrh ** DBNAME that is used to control that connection.  The database
100475897234Sdrh ** connection is deleted when the DBNAME command is deleted.
100575897234Sdrh **
100675897234Sdrh ** The second argument is the name of the directory that contains
100775897234Sdrh ** the sqlite database that is to be accessed.
1008fbc3eab8Sdrh **
1009fbc3eab8Sdrh ** For testing purposes, we also support the following:
1010fbc3eab8Sdrh **
1011fbc3eab8Sdrh **  sqlite -encoding
1012fbc3eab8Sdrh **
1013fbc3eab8Sdrh **       Return the encoding used by LIKE and GLOB operators.  Choices
1014fbc3eab8Sdrh **       are UTF-8 and iso8859.
1015fbc3eab8Sdrh **
1016647cb0e1Sdrh **  sqlite -version
1017647cb0e1Sdrh **
1018647cb0e1Sdrh **       Return the version number of the SQLite library.
1019647cb0e1Sdrh **
1020fbc3eab8Sdrh **  sqlite -tcl-uses-utf
1021fbc3eab8Sdrh **
1022fbc3eab8Sdrh **       Return "1" if compiled with a Tcl uses UTF-8.  Return "0" if
1023fbc3eab8Sdrh **       not.  Used by tests to make sure the library was compiled
1024fbc3eab8Sdrh **       correctly.
102575897234Sdrh */
102622fbcb8dSdrh static int DbMain(void *cd, Tcl_Interp *interp, int objc,Tcl_Obj *const*objv){
1027bec3f402Sdrh   SqliteDb *p;
102822fbcb8dSdrh   void *pKey = 0;
102922fbcb8dSdrh   int nKey = 0;
103022fbcb8dSdrh   const char *zArg;
103175897234Sdrh   char *zErrMsg;
103222fbcb8dSdrh   const char *zFile;
103380290863Sdanielk1977   const char *zOpts[2] = {0, 0};
103406b2718aSdrh   char zBuf[80];
103522fbcb8dSdrh   if( objc==2 ){
103622fbcb8dSdrh     zArg = Tcl_GetStringFromObj(objv[1], 0);
103722fbcb8dSdrh     if( strcmp(zArg,"-encoding")==0 ){
10386f8a503dSdanielk1977       Tcl_AppendResult(interp,sqlite3_encoding,0);
1039fbc3eab8Sdrh       return TCL_OK;
1040fbc3eab8Sdrh     }
104122fbcb8dSdrh     if( strcmp(zArg,"-version")==0 ){
10426f8a503dSdanielk1977       Tcl_AppendResult(interp,sqlite3_version,0);
1043647cb0e1Sdrh       return TCL_OK;
1044647cb0e1Sdrh     }
10459eb9e26bSdrh     if( strcmp(zArg,"-has-codec")==0 ){
10469eb9e26bSdrh #ifdef SQLITE_HAS_CODEC
104722fbcb8dSdrh       Tcl_AppendResult(interp,"1",0);
104822fbcb8dSdrh #else
104922fbcb8dSdrh       Tcl_AppendResult(interp,"0",0);
105022fbcb8dSdrh #endif
105122fbcb8dSdrh       return TCL_OK;
105222fbcb8dSdrh     }
105322fbcb8dSdrh     if( strcmp(zArg,"-tcl-uses-utf")==0 ){
1054fbc3eab8Sdrh #ifdef TCL_UTF_MAX
1055fbc3eab8Sdrh       Tcl_AppendResult(interp,"1",0);
1056fbc3eab8Sdrh #else
1057fbc3eab8Sdrh       Tcl_AppendResult(interp,"0",0);
1058fbc3eab8Sdrh #endif
1059fbc3eab8Sdrh       return TCL_OK;
1060fbc3eab8Sdrh     }
1061fbc3eab8Sdrh   }
106222fbcb8dSdrh   if( objc==5 || objc==6 ){
106322fbcb8dSdrh     zArg = Tcl_GetStringFromObj(objv[objc-2], 0);
106422fbcb8dSdrh     if( strcmp(zArg,"-key")==0 ){
106522fbcb8dSdrh       pKey = Tcl_GetByteArrayFromObj(objv[objc-1], &nKey);
106622fbcb8dSdrh       objc -= 2;
106722fbcb8dSdrh     }
106822fbcb8dSdrh   }
106922fbcb8dSdrh   if( objc!=3 && objc!=4 ){
107022fbcb8dSdrh     Tcl_WrongNumArgs(interp, 1, objv,
10719eb9e26bSdrh #ifdef SQLITE_HAS_CODEC
10729eb9e26bSdrh       "HANDLE FILENAME ?-key CODEC-KEY?"
107322fbcb8dSdrh #else
107422fbcb8dSdrh       "HANDLE FILENAME ?MODE?"
107522fbcb8dSdrh #endif
107622fbcb8dSdrh     );
107775897234Sdrh     return TCL_ERROR;
107875897234Sdrh   }
107975897234Sdrh   zErrMsg = 0;
10804cdc9e84Sdrh   p = (SqliteDb*)Tcl_Alloc( sizeof(*p) );
108175897234Sdrh   if( p==0 ){
1082bec3f402Sdrh     Tcl_SetResult(interp, "malloc failed", TCL_STATIC);
1083bec3f402Sdrh     return TCL_ERROR;
1084bec3f402Sdrh   }
1085bec3f402Sdrh   memset(p, 0, sizeof(*p));
108622fbcb8dSdrh   zFile = Tcl_GetStringFromObj(objv[2], 0);
10879eb9e26bSdrh #ifdef SQLITE_HAS_CODEC
10886f8a503dSdanielk1977   p->db = sqlite3_open_encrypted(zFile, pKey, nKey, 0, &zErrMsg);
1089eb8ed70dSdrh #else
109080290863Sdanielk1977   if( objc>3 ){
109180290863Sdanielk1977     zOpts[0] = Tcl_GetString(objv[3]);
109280290863Sdanielk1977   }
109380290863Sdanielk1977   sqlite3_open(zFile, &p->db, zOpts);
109480290863Sdanielk1977   if( SQLITE_OK!=sqlite3_errcode(p->db) ){
109580290863Sdanielk1977     zErrMsg = strdup(sqlite3_errmsg(p->db));
109680290863Sdanielk1977     sqlite3_close(p->db);
109780290863Sdanielk1977     p->db = 0;
109880290863Sdanielk1977   }
1099eb8ed70dSdrh #endif
1100bec3f402Sdrh   if( p->db==0 ){
110175897234Sdrh     Tcl_SetResult(interp, zErrMsg, TCL_VOLATILE);
1102bec3f402Sdrh     Tcl_Free((char*)p);
110375897234Sdrh     free(zErrMsg);
110475897234Sdrh     return TCL_ERROR;
110575897234Sdrh   }
110622fbcb8dSdrh   zArg = Tcl_GetStringFromObj(objv[1], 0);
110722fbcb8dSdrh   Tcl_CreateObjCommand(interp, zArg, DbObjCmd, (char*)p, DbDeleteCmd);
1108c22bd47dSdrh 
110906b2718aSdrh   /* The return value is the value of the sqlite* pointer
111006b2718aSdrh   */
111106b2718aSdrh   sprintf(zBuf, "%p", p->db);
11125e5377fbSdrh   if( strncmp(zBuf,"0x",2) ){
11135e5377fbSdrh     sprintf(zBuf, "0x%p", p->db);
11145e5377fbSdrh   }
111506b2718aSdrh   Tcl_AppendResult(interp, zBuf, 0);
111606b2718aSdrh 
1117c22bd47dSdrh   /* If compiled with SQLITE_TEST turned on, then register the "md5sum"
111806b2718aSdrh   ** SQL function.
1119c22bd47dSdrh   */
112028b4e489Sdrh #ifdef SQLITE_TEST
112128b4e489Sdrh   {
112228b4e489Sdrh     extern void Md5_Register(sqlite*);
112328b4e489Sdrh     Md5_Register(p->db);
112428b4e489Sdrh    }
112528b4e489Sdrh #endif
112675897234Sdrh   return TCL_OK;
112775897234Sdrh }
112875897234Sdrh 
112975897234Sdrh /*
113090ca9753Sdrh ** Provide a dummy Tcl_InitStubs if we are using this as a static
113190ca9753Sdrh ** library.
113290ca9753Sdrh */
113390ca9753Sdrh #ifndef USE_TCL_STUBS
113490ca9753Sdrh # undef  Tcl_InitStubs
113590ca9753Sdrh # define Tcl_InitStubs(a,b,c)
113690ca9753Sdrh #endif
113790ca9753Sdrh 
113890ca9753Sdrh /*
113975897234Sdrh ** Initialize this module.
114075897234Sdrh **
114175897234Sdrh ** This Tcl module contains only a single new Tcl command named "sqlite".
114275897234Sdrh ** (Hence there is no namespace.  There is no point in using a namespace
114375897234Sdrh ** if the extension only supplies one new name!)  The "sqlite" command is
114475897234Sdrh ** used to open a new SQLite database.  See the DbMain() routine above
114575897234Sdrh ** for additional information.
114675897234Sdrh */
114775897234Sdrh int Sqlite_Init(Tcl_Interp *interp){
114890ca9753Sdrh   Tcl_InitStubs(interp, "8.0", 0);
114922fbcb8dSdrh   Tcl_CreateObjCommand(interp, "sqlite", (Tcl_ObjCmdProc*)DbMain, 0, 0);
11506d4abfbeSdrh   Tcl_PkgProvide(interp, "sqlite", "2.0");
115190ca9753Sdrh   return TCL_OK;
115290ca9753Sdrh }
115390ca9753Sdrh int Tclsqlite_Init(Tcl_Interp *interp){
115490ca9753Sdrh   Tcl_InitStubs(interp, "8.0", 0);
115522fbcb8dSdrh   Tcl_CreateObjCommand(interp, "sqlite", (Tcl_ObjCmdProc*)DbMain, 0, 0);
11566d4abfbeSdrh   Tcl_PkgProvide(interp, "sqlite", "2.0");
115775897234Sdrh   return TCL_OK;
115875897234Sdrh }
115975897234Sdrh int Sqlite_SafeInit(Tcl_Interp *interp){
116075897234Sdrh   return TCL_OK;
116175897234Sdrh }
116290ca9753Sdrh int Tclsqlite_SafeInit(Tcl_Interp *interp){
116390ca9753Sdrh   return TCL_OK;
116490ca9753Sdrh }
116575897234Sdrh 
11663cebbde3Sdrh #if 0
116775897234Sdrh /*
116875897234Sdrh ** If compiled using mktclapp, this routine runs to initialize
116975897234Sdrh ** everything.
117075897234Sdrh */
117175897234Sdrh int Et_AppInit(Tcl_Interp *interp){
117275897234Sdrh   return Sqlite_Init(interp);
117375897234Sdrh }
11743cebbde3Sdrh #endif
1175348784efSdrh 
1176348784efSdrh /*
1177348784efSdrh ** If the macro TCLSH is defined and is one, then put in code for the
1178348784efSdrh ** "main" routine that will initialize Tcl.
1179348784efSdrh */
1180348784efSdrh #if defined(TCLSH) && TCLSH==1
1181348784efSdrh static char zMainloop[] =
1182348784efSdrh   "set line {}\n"
1183348784efSdrh   "while {![eof stdin]} {\n"
1184348784efSdrh     "if {$line!=\"\"} {\n"
1185348784efSdrh       "puts -nonewline \"> \"\n"
1186348784efSdrh     "} else {\n"
1187348784efSdrh       "puts -nonewline \"% \"\n"
1188348784efSdrh     "}\n"
1189348784efSdrh     "flush stdout\n"
1190348784efSdrh     "append line [gets stdin]\n"
1191348784efSdrh     "if {[info complete $line]} {\n"
1192348784efSdrh       "if {[catch {uplevel #0 $line} result]} {\n"
1193348784efSdrh         "puts stderr \"Error: $result\"\n"
1194348784efSdrh       "} elseif {$result!=\"\"} {\n"
1195348784efSdrh         "puts $result\n"
1196348784efSdrh       "}\n"
1197348784efSdrh       "set line {}\n"
1198348784efSdrh     "} else {\n"
1199348784efSdrh       "append line \\n\n"
1200348784efSdrh     "}\n"
1201348784efSdrh   "}\n"
1202348784efSdrh ;
1203348784efSdrh 
1204348784efSdrh #define TCLSH_MAIN main   /* Needed to fake out mktclapp */
1205348784efSdrh int TCLSH_MAIN(int argc, char **argv){
1206348784efSdrh   Tcl_Interp *interp;
1207297ecf14Sdrh   Tcl_FindExecutable(argv[0]);
1208348784efSdrh   interp = Tcl_CreateInterp();
12094adee20fSdanielk1977   Sqlite_Init(interp);
1210d9b0257aSdrh #ifdef SQLITE_TEST
1211d1bf3512Sdrh   {
1212d1bf3512Sdrh     extern int Sqlitetest1_Init(Tcl_Interp*);
12135c4d9703Sdrh     extern int Sqlitetest2_Init(Tcl_Interp*);
12145c4d9703Sdrh     extern int Sqlitetest3_Init(Tcl_Interp*);
1215a6064dcfSdrh     extern int Sqlitetest4_Init(Tcl_Interp*);
1216998b56c3Sdanielk1977     extern int Sqlitetest5_Init(Tcl_Interp*);
1217efc251daSdrh     extern int Md5_Init(Tcl_Interp*);
12186490bebdSdanielk1977     Sqlitetest1_Init(interp);
12195c4d9703Sdrh     Sqlitetest2_Init(interp);
1220de647130Sdrh     Sqlitetest3_Init(interp);
1221*fc57d7bfSdanielk1977     Sqlitetest4_Init(interp);
1222998b56c3Sdanielk1977     Sqlitetest5_Init(interp);
1223efc251daSdrh     Md5_Init(interp);
1224d1bf3512Sdrh   }
1225d1bf3512Sdrh #endif
1226348784efSdrh   if( argc>=2 ){
1227348784efSdrh     int i;
1228348784efSdrh     Tcl_SetVar(interp,"argv0",argv[1],TCL_GLOBAL_ONLY);
1229348784efSdrh     Tcl_SetVar(interp,"argv", "", TCL_GLOBAL_ONLY);
1230348784efSdrh     for(i=2; i<argc; i++){
1231348784efSdrh       Tcl_SetVar(interp, "argv", argv[i],
1232348784efSdrh           TCL_GLOBAL_ONLY | TCL_LIST_ELEMENT | TCL_APPEND_VALUE);
1233348784efSdrh     }
1234348784efSdrh     if( Tcl_EvalFile(interp, argv[1])!=TCL_OK ){
12350de8c112Sdrh       const char *zInfo = Tcl_GetVar(interp, "errorInfo", TCL_GLOBAL_ONLY);
1236c61053b7Sdrh       if( zInfo==0 ) zInfo = interp->result;
1237c61053b7Sdrh       fprintf(stderr,"%s: %s\n", *argv, zInfo);
1238348784efSdrh       return 1;
1239348784efSdrh     }
1240348784efSdrh   }else{
1241348784efSdrh     Tcl_GlobalEval(interp, zMainloop);
1242348784efSdrh   }
1243348784efSdrh   return 0;
1244348784efSdrh }
1245348784efSdrh #endif /* TCLSH */
12466d31316cSdrh 
12476d31316cSdrh #endif /* !defined(NO_TCL) */
12484adee20fSdanielk1977 
12494adee20fSdanielk1977 
12504adee20fSdanielk1977 
1251