sl@0
|
1 |
/*
|
sl@0
|
2 |
** 2006 June 10
|
sl@0
|
3 |
**
|
sl@0
|
4 |
** The author disclaims copyright to this source code. In place of
|
sl@0
|
5 |
** a legal notice, here is a blessing:
|
sl@0
|
6 |
**
|
sl@0
|
7 |
** May you do good and not evil.
|
sl@0
|
8 |
** May you find forgiveness for yourself and forgive others.
|
sl@0
|
9 |
** May you share freely, never taking more than you give.
|
sl@0
|
10 |
**
|
sl@0
|
11 |
*************************************************************************
|
sl@0
|
12 |
** Code for testing the virtual table interfaces. This code
|
sl@0
|
13 |
** is not included in the SQLite library. It is used for automated
|
sl@0
|
14 |
** testing of the SQLite library.
|
sl@0
|
15 |
**
|
sl@0
|
16 |
** $Id: test_schema.c,v 1.15 2008/07/07 14:50:14 drh Exp $
|
sl@0
|
17 |
*/
|
sl@0
|
18 |
|
sl@0
|
19 |
/* The code in this file defines a sqlite3 virtual-table module that
|
sl@0
|
20 |
** provides a read-only view of the current database schema. There is one
|
sl@0
|
21 |
** row in the schema table for each column in the database schema.
|
sl@0
|
22 |
*/
|
sl@0
|
23 |
#define SCHEMA \
|
sl@0
|
24 |
"CREATE TABLE x(" \
|
sl@0
|
25 |
"database," /* Name of database (i.e. main, temp etc.) */ \
|
sl@0
|
26 |
"tablename," /* Name of table */ \
|
sl@0
|
27 |
"cid," /* Column number (from left-to-right, 0 upward) */ \
|
sl@0
|
28 |
"name," /* Column name */ \
|
sl@0
|
29 |
"type," /* Specified type (i.e. VARCHAR(32)) */ \
|
sl@0
|
30 |
"not_null," /* Boolean. True if NOT NULL was specified */ \
|
sl@0
|
31 |
"dflt_value," /* Default value for this column */ \
|
sl@0
|
32 |
"pk" /* True if this column is part of the primary key */ \
|
sl@0
|
33 |
")"
|
sl@0
|
34 |
|
sl@0
|
35 |
/* If SQLITE_TEST is defined this code is preprocessed for use as part
|
sl@0
|
36 |
** of the sqlite test binary "testfixture". Otherwise it is preprocessed
|
sl@0
|
37 |
** to be compiled into an sqlite dynamic extension.
|
sl@0
|
38 |
*/
|
sl@0
|
39 |
#ifdef SQLITE_TEST
|
sl@0
|
40 |
#include "sqliteInt.h"
|
sl@0
|
41 |
#include "tcl.h"
|
sl@0
|
42 |
#else
|
sl@0
|
43 |
#include "sqlite3ext.h"
|
sl@0
|
44 |
SQLITE_EXTENSION_INIT1
|
sl@0
|
45 |
#endif
|
sl@0
|
46 |
|
sl@0
|
47 |
#include <stdlib.h>
|
sl@0
|
48 |
#include <string.h>
|
sl@0
|
49 |
#include <assert.h>
|
sl@0
|
50 |
|
sl@0
|
51 |
typedef struct schema_vtab schema_vtab;
|
sl@0
|
52 |
typedef struct schema_cursor schema_cursor;
|
sl@0
|
53 |
|
sl@0
|
54 |
/* A schema table object */
|
sl@0
|
55 |
struct schema_vtab {
|
sl@0
|
56 |
sqlite3_vtab base;
|
sl@0
|
57 |
sqlite3 *db;
|
sl@0
|
58 |
};
|
sl@0
|
59 |
|
sl@0
|
60 |
/* A schema table cursor object */
|
sl@0
|
61 |
struct schema_cursor {
|
sl@0
|
62 |
sqlite3_vtab_cursor base;
|
sl@0
|
63 |
sqlite3_stmt *pDbList;
|
sl@0
|
64 |
sqlite3_stmt *pTableList;
|
sl@0
|
65 |
sqlite3_stmt *pColumnList;
|
sl@0
|
66 |
int rowid;
|
sl@0
|
67 |
};
|
sl@0
|
68 |
|
sl@0
|
69 |
/*
|
sl@0
|
70 |
** None of this works unless we have virtual tables.
|
sl@0
|
71 |
*/
|
sl@0
|
72 |
#ifndef SQLITE_OMIT_VIRTUALTABLE
|
sl@0
|
73 |
|
sl@0
|
74 |
/*
|
sl@0
|
75 |
** Table destructor for the schema module.
|
sl@0
|
76 |
*/
|
sl@0
|
77 |
static int schemaDestroy(sqlite3_vtab *pVtab){
|
sl@0
|
78 |
sqlite3_free(pVtab);
|
sl@0
|
79 |
return 0;
|
sl@0
|
80 |
}
|
sl@0
|
81 |
|
sl@0
|
82 |
/*
|
sl@0
|
83 |
** Table constructor for the schema module.
|
sl@0
|
84 |
*/
|
sl@0
|
85 |
static int schemaCreate(
|
sl@0
|
86 |
sqlite3 *db,
|
sl@0
|
87 |
void *pAux,
|
sl@0
|
88 |
int argc, const char *const*argv,
|
sl@0
|
89 |
sqlite3_vtab **ppVtab,
|
sl@0
|
90 |
char **pzErr
|
sl@0
|
91 |
){
|
sl@0
|
92 |
int rc = SQLITE_NOMEM;
|
sl@0
|
93 |
schema_vtab *pVtab = sqlite3_malloc(sizeof(schema_vtab));
|
sl@0
|
94 |
if( pVtab ){
|
sl@0
|
95 |
memset(pVtab, 0, sizeof(schema_vtab));
|
sl@0
|
96 |
pVtab->db = db;
|
sl@0
|
97 |
#ifndef SQLITE_OMIT_VIRTUALTABLE
|
sl@0
|
98 |
rc = sqlite3_declare_vtab(db, SCHEMA);
|
sl@0
|
99 |
#endif
|
sl@0
|
100 |
}
|
sl@0
|
101 |
*ppVtab = (sqlite3_vtab *)pVtab;
|
sl@0
|
102 |
return rc;
|
sl@0
|
103 |
}
|
sl@0
|
104 |
|
sl@0
|
105 |
/*
|
sl@0
|
106 |
** Open a new cursor on the schema table.
|
sl@0
|
107 |
*/
|
sl@0
|
108 |
static int schemaOpen(sqlite3_vtab *pVTab, sqlite3_vtab_cursor **ppCursor){
|
sl@0
|
109 |
int rc = SQLITE_NOMEM;
|
sl@0
|
110 |
schema_cursor *pCur;
|
sl@0
|
111 |
pCur = sqlite3_malloc(sizeof(schema_cursor));
|
sl@0
|
112 |
if( pCur ){
|
sl@0
|
113 |
memset(pCur, 0, sizeof(schema_cursor));
|
sl@0
|
114 |
*ppCursor = (sqlite3_vtab_cursor *)pCur;
|
sl@0
|
115 |
rc = SQLITE_OK;
|
sl@0
|
116 |
}
|
sl@0
|
117 |
return rc;
|
sl@0
|
118 |
}
|
sl@0
|
119 |
|
sl@0
|
120 |
/*
|
sl@0
|
121 |
** Close a schema table cursor.
|
sl@0
|
122 |
*/
|
sl@0
|
123 |
static int schemaClose(sqlite3_vtab_cursor *cur){
|
sl@0
|
124 |
schema_cursor *pCur = (schema_cursor *)cur;
|
sl@0
|
125 |
sqlite3_finalize(pCur->pDbList);
|
sl@0
|
126 |
sqlite3_finalize(pCur->pTableList);
|
sl@0
|
127 |
sqlite3_finalize(pCur->pColumnList);
|
sl@0
|
128 |
sqlite3_free(pCur);
|
sl@0
|
129 |
return SQLITE_OK;
|
sl@0
|
130 |
}
|
sl@0
|
131 |
|
sl@0
|
132 |
/*
|
sl@0
|
133 |
** Retrieve a column of data.
|
sl@0
|
134 |
*/
|
sl@0
|
135 |
static int schemaColumn(sqlite3_vtab_cursor *cur, sqlite3_context *ctx, int i){
|
sl@0
|
136 |
schema_cursor *pCur = (schema_cursor *)cur;
|
sl@0
|
137 |
switch( i ){
|
sl@0
|
138 |
case 0:
|
sl@0
|
139 |
sqlite3_result_value(ctx, sqlite3_column_value(pCur->pDbList, 1));
|
sl@0
|
140 |
break;
|
sl@0
|
141 |
case 1:
|
sl@0
|
142 |
sqlite3_result_value(ctx, sqlite3_column_value(pCur->pTableList, 0));
|
sl@0
|
143 |
break;
|
sl@0
|
144 |
default:
|
sl@0
|
145 |
sqlite3_result_value(ctx, sqlite3_column_value(pCur->pColumnList, i-2));
|
sl@0
|
146 |
break;
|
sl@0
|
147 |
}
|
sl@0
|
148 |
return SQLITE_OK;
|
sl@0
|
149 |
}
|
sl@0
|
150 |
|
sl@0
|
151 |
/*
|
sl@0
|
152 |
** Retrieve the current rowid.
|
sl@0
|
153 |
*/
|
sl@0
|
154 |
static int schemaRowid(sqlite3_vtab_cursor *cur, sqlite_int64 *pRowid){
|
sl@0
|
155 |
schema_cursor *pCur = (schema_cursor *)cur;
|
sl@0
|
156 |
*pRowid = pCur->rowid;
|
sl@0
|
157 |
return SQLITE_OK;
|
sl@0
|
158 |
}
|
sl@0
|
159 |
|
sl@0
|
160 |
static int finalize(sqlite3_stmt **ppStmt){
|
sl@0
|
161 |
int rc = sqlite3_finalize(*ppStmt);
|
sl@0
|
162 |
*ppStmt = 0;
|
sl@0
|
163 |
return rc;
|
sl@0
|
164 |
}
|
sl@0
|
165 |
|
sl@0
|
166 |
static int schemaEof(sqlite3_vtab_cursor *cur){
|
sl@0
|
167 |
schema_cursor *pCur = (schema_cursor *)cur;
|
sl@0
|
168 |
return (pCur->pDbList ? 0 : 1);
|
sl@0
|
169 |
}
|
sl@0
|
170 |
|
sl@0
|
171 |
/*
|
sl@0
|
172 |
** Advance the cursor to the next row.
|
sl@0
|
173 |
*/
|
sl@0
|
174 |
static int schemaNext(sqlite3_vtab_cursor *cur){
|
sl@0
|
175 |
int rc = SQLITE_OK;
|
sl@0
|
176 |
schema_cursor *pCur = (schema_cursor *)cur;
|
sl@0
|
177 |
schema_vtab *pVtab = (schema_vtab *)(cur->pVtab);
|
sl@0
|
178 |
char *zSql = 0;
|
sl@0
|
179 |
|
sl@0
|
180 |
while( !pCur->pColumnList || SQLITE_ROW!=sqlite3_step(pCur->pColumnList) ){
|
sl@0
|
181 |
if( SQLITE_OK!=(rc = finalize(&pCur->pColumnList)) ) goto next_exit;
|
sl@0
|
182 |
|
sl@0
|
183 |
while( !pCur->pTableList || SQLITE_ROW!=sqlite3_step(pCur->pTableList) ){
|
sl@0
|
184 |
if( SQLITE_OK!=(rc = finalize(&pCur->pTableList)) ) goto next_exit;
|
sl@0
|
185 |
|
sl@0
|
186 |
assert(pCur->pDbList);
|
sl@0
|
187 |
while( SQLITE_ROW!=sqlite3_step(pCur->pDbList) ){
|
sl@0
|
188 |
rc = finalize(&pCur->pDbList);
|
sl@0
|
189 |
goto next_exit;
|
sl@0
|
190 |
}
|
sl@0
|
191 |
|
sl@0
|
192 |
/* Set zSql to the SQL to pull the list of tables from the
|
sl@0
|
193 |
** sqlite_master (or sqlite_temp_master) table of the database
|
sl@0
|
194 |
** identfied by the row pointed to by the SQL statement pCur->pDbList
|
sl@0
|
195 |
** (iterating through a "PRAGMA database_list;" statement).
|
sl@0
|
196 |
*/
|
sl@0
|
197 |
if( sqlite3_column_int(pCur->pDbList, 0)==1 ){
|
sl@0
|
198 |
zSql = sqlite3_mprintf(
|
sl@0
|
199 |
"SELECT name FROM sqlite_temp_master WHERE type='table'"
|
sl@0
|
200 |
);
|
sl@0
|
201 |
}else{
|
sl@0
|
202 |
sqlite3_stmt *pDbList = pCur->pDbList;
|
sl@0
|
203 |
zSql = sqlite3_mprintf(
|
sl@0
|
204 |
"SELECT name FROM %Q.sqlite_master WHERE type='table'",
|
sl@0
|
205 |
sqlite3_column_text(pDbList, 1)
|
sl@0
|
206 |
);
|
sl@0
|
207 |
}
|
sl@0
|
208 |
if( !zSql ){
|
sl@0
|
209 |
rc = SQLITE_NOMEM;
|
sl@0
|
210 |
goto next_exit;
|
sl@0
|
211 |
}
|
sl@0
|
212 |
|
sl@0
|
213 |
rc = sqlite3_prepare(pVtab->db, zSql, -1, &pCur->pTableList, 0);
|
sl@0
|
214 |
sqlite3_free(zSql);
|
sl@0
|
215 |
if( rc!=SQLITE_OK ) goto next_exit;
|
sl@0
|
216 |
}
|
sl@0
|
217 |
|
sl@0
|
218 |
/* Set zSql to the SQL to the table_info pragma for the table currently
|
sl@0
|
219 |
** identified by the rows pointed to by statements pCur->pDbList and
|
sl@0
|
220 |
** pCur->pTableList.
|
sl@0
|
221 |
*/
|
sl@0
|
222 |
zSql = sqlite3_mprintf("PRAGMA %Q.table_info(%Q)",
|
sl@0
|
223 |
sqlite3_column_text(pCur->pDbList, 1),
|
sl@0
|
224 |
sqlite3_column_text(pCur->pTableList, 0)
|
sl@0
|
225 |
);
|
sl@0
|
226 |
|
sl@0
|
227 |
if( !zSql ){
|
sl@0
|
228 |
rc = SQLITE_NOMEM;
|
sl@0
|
229 |
goto next_exit;
|
sl@0
|
230 |
}
|
sl@0
|
231 |
rc = sqlite3_prepare(pVtab->db, zSql, -1, &pCur->pColumnList, 0);
|
sl@0
|
232 |
sqlite3_free(zSql);
|
sl@0
|
233 |
if( rc!=SQLITE_OK ) goto next_exit;
|
sl@0
|
234 |
}
|
sl@0
|
235 |
pCur->rowid++;
|
sl@0
|
236 |
|
sl@0
|
237 |
next_exit:
|
sl@0
|
238 |
/* TODO: Handle rc */
|
sl@0
|
239 |
return rc;
|
sl@0
|
240 |
}
|
sl@0
|
241 |
|
sl@0
|
242 |
/*
|
sl@0
|
243 |
** Reset a schema table cursor.
|
sl@0
|
244 |
*/
|
sl@0
|
245 |
static int schemaFilter(
|
sl@0
|
246 |
sqlite3_vtab_cursor *pVtabCursor,
|
sl@0
|
247 |
int idxNum, const char *idxStr,
|
sl@0
|
248 |
int argc, sqlite3_value **argv
|
sl@0
|
249 |
){
|
sl@0
|
250 |
int rc;
|
sl@0
|
251 |
schema_vtab *pVtab = (schema_vtab *)(pVtabCursor->pVtab);
|
sl@0
|
252 |
schema_cursor *pCur = (schema_cursor *)pVtabCursor;
|
sl@0
|
253 |
pCur->rowid = 0;
|
sl@0
|
254 |
finalize(&pCur->pTableList);
|
sl@0
|
255 |
finalize(&pCur->pColumnList);
|
sl@0
|
256 |
finalize(&pCur->pDbList);
|
sl@0
|
257 |
rc = sqlite3_prepare(pVtab->db,"PRAGMA database_list", -1, &pCur->pDbList, 0);
|
sl@0
|
258 |
return (rc==SQLITE_OK ? schemaNext(pVtabCursor) : rc);
|
sl@0
|
259 |
}
|
sl@0
|
260 |
|
sl@0
|
261 |
/*
|
sl@0
|
262 |
** Analyse the WHERE condition.
|
sl@0
|
263 |
*/
|
sl@0
|
264 |
static int schemaBestIndex(sqlite3_vtab *tab, sqlite3_index_info *pIdxInfo){
|
sl@0
|
265 |
return SQLITE_OK;
|
sl@0
|
266 |
}
|
sl@0
|
267 |
|
sl@0
|
268 |
/*
|
sl@0
|
269 |
** A virtual table module that merely echos method calls into TCL
|
sl@0
|
270 |
** variables.
|
sl@0
|
271 |
*/
|
sl@0
|
272 |
static sqlite3_module schemaModule = {
|
sl@0
|
273 |
0, /* iVersion */
|
sl@0
|
274 |
schemaCreate,
|
sl@0
|
275 |
schemaCreate,
|
sl@0
|
276 |
schemaBestIndex,
|
sl@0
|
277 |
schemaDestroy,
|
sl@0
|
278 |
schemaDestroy,
|
sl@0
|
279 |
schemaOpen, /* xOpen - open a cursor */
|
sl@0
|
280 |
schemaClose, /* xClose - close a cursor */
|
sl@0
|
281 |
schemaFilter, /* xFilter - configure scan constraints */
|
sl@0
|
282 |
schemaNext, /* xNext - advance a cursor */
|
sl@0
|
283 |
schemaEof, /* xEof */
|
sl@0
|
284 |
schemaColumn, /* xColumn - read data */
|
sl@0
|
285 |
schemaRowid, /* xRowid - read data */
|
sl@0
|
286 |
0, /* xUpdate */
|
sl@0
|
287 |
0, /* xBegin */
|
sl@0
|
288 |
0, /* xSync */
|
sl@0
|
289 |
0, /* xCommit */
|
sl@0
|
290 |
0, /* xRollback */
|
sl@0
|
291 |
0, /* xFindMethod */
|
sl@0
|
292 |
0, /* xRename */
|
sl@0
|
293 |
};
|
sl@0
|
294 |
|
sl@0
|
295 |
#endif /* !defined(SQLITE_OMIT_VIRTUALTABLE) */
|
sl@0
|
296 |
|
sl@0
|
297 |
#ifdef SQLITE_TEST
|
sl@0
|
298 |
|
sl@0
|
299 |
/*
|
sl@0
|
300 |
** Decode a pointer to an sqlite3 object.
|
sl@0
|
301 |
*/
|
sl@0
|
302 |
extern int getDbPointer(Tcl_Interp *interp, const char *zA, sqlite3 **ppDb);
|
sl@0
|
303 |
|
sl@0
|
304 |
/*
|
sl@0
|
305 |
** Register the schema virtual table module.
|
sl@0
|
306 |
*/
|
sl@0
|
307 |
static int register_schema_module(
|
sl@0
|
308 |
ClientData clientData, /* Not used */
|
sl@0
|
309 |
Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
|
sl@0
|
310 |
int objc, /* Number of arguments */
|
sl@0
|
311 |
Tcl_Obj *CONST objv[] /* Command arguments */
|
sl@0
|
312 |
){
|
sl@0
|
313 |
sqlite3 *db;
|
sl@0
|
314 |
if( objc!=2 ){
|
sl@0
|
315 |
Tcl_WrongNumArgs(interp, 1, objv, "DB");
|
sl@0
|
316 |
return TCL_ERROR;
|
sl@0
|
317 |
}
|
sl@0
|
318 |
if( getDbPointer(interp, Tcl_GetString(objv[1]), &db) ) return TCL_ERROR;
|
sl@0
|
319 |
#ifndef SQLITE_OMIT_VIRTUALTABLE
|
sl@0
|
320 |
sqlite3_create_module(db, "schema", &schemaModule, 0);
|
sl@0
|
321 |
#endif
|
sl@0
|
322 |
return TCL_OK;
|
sl@0
|
323 |
}
|
sl@0
|
324 |
|
sl@0
|
325 |
/*
|
sl@0
|
326 |
** Register commands with the TCL interpreter.
|
sl@0
|
327 |
*/
|
sl@0
|
328 |
int Sqlitetestschema_Init(Tcl_Interp *interp){
|
sl@0
|
329 |
static struct {
|
sl@0
|
330 |
char *zName;
|
sl@0
|
331 |
Tcl_ObjCmdProc *xProc;
|
sl@0
|
332 |
void *clientData;
|
sl@0
|
333 |
} aObjCmd[] = {
|
sl@0
|
334 |
{ "register_schema_module", register_schema_module, 0 },
|
sl@0
|
335 |
};
|
sl@0
|
336 |
int i;
|
sl@0
|
337 |
for(i=0; i<sizeof(aObjCmd)/sizeof(aObjCmd[0]); i++){
|
sl@0
|
338 |
Tcl_CreateObjCommand(interp, aObjCmd[i].zName,
|
sl@0
|
339 |
aObjCmd[i].xProc, aObjCmd[i].clientData, 0);
|
sl@0
|
340 |
}
|
sl@0
|
341 |
return TCL_OK;
|
sl@0
|
342 |
}
|
sl@0
|
343 |
|
sl@0
|
344 |
#else
|
sl@0
|
345 |
|
sl@0
|
346 |
/*
|
sl@0
|
347 |
** Extension load function.
|
sl@0
|
348 |
*/
|
sl@0
|
349 |
int sqlite3_extension_init(
|
sl@0
|
350 |
sqlite3 *db,
|
sl@0
|
351 |
char **pzErrMsg,
|
sl@0
|
352 |
const sqlite3_api_routines *pApi
|
sl@0
|
353 |
){
|
sl@0
|
354 |
SQLITE_EXTENSION_INIT2(pApi);
|
sl@0
|
355 |
#ifndef SQLITE_OMIT_VIRTUALTABLE
|
sl@0
|
356 |
sqlite3_create_module(db, "schema", &schemaModule, 0);
|
sl@0
|
357 |
#endif
|
sl@0
|
358 |
return 0;
|
sl@0
|
359 |
}
|
sl@0
|
360 |
|
sl@0
|
361 |
#endif
|