os/persistentdata/persistentstorage/sqlite3api/TEST/SRC/test_schema.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
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