os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/alter2.test
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
# 2005 February 18
sl@0
     2
#
sl@0
     3
# The author disclaims copyright to this source code.  In place of
sl@0
     4
# a legal notice, here is a blessing:
sl@0
     5
#
sl@0
     6
#    May you do good and not evil.
sl@0
     7
#    May you find forgiveness for yourself and forgive others.
sl@0
     8
#    May you share freely, never taking more than you give.
sl@0
     9
#
sl@0
    10
#*************************************************************************
sl@0
    11
# This file implements regression tests for SQLite library.  The
sl@0
    12
# focus of this script is testing that SQLite can handle a subtle 
sl@0
    13
# file format change that may be used in the future to implement
sl@0
    14
# "ALTER TABLE ... ADD COLUMN".
sl@0
    15
#
sl@0
    16
# $Id: alter2.test,v 1.13 2008/03/19 00:21:31 drh Exp $
sl@0
    17
#
sl@0
    18
sl@0
    19
set testdir [file dirname $argv0]
sl@0
    20
source $testdir/tester.tcl
sl@0
    21
sl@0
    22
# We have to have pragmas in order to do this test
sl@0
    23
ifcapable {!pragma} return
sl@0
    24
sl@0
    25
# These tests do not work if there is a codec. 
sl@0
    26
#
sl@0
    27
#if {[catch {sqlite3 -has_codec} r] || $r} return
sl@0
    28
sl@0
    29
# The file format change affects the way row-records stored in tables (but 
sl@0
    30
# not indices) are interpreted. Before version 3.1.3, a row-record for a 
sl@0
    31
# table with N columns was guaranteed to contain exactly N fields. As
sl@0
    32
# of version 3.1.3, the record may contain up to N fields. In this case
sl@0
    33
# the M fields that are present are the values for the left-most M 
sl@0
    34
# columns. The (N-M) rightmost columns contain NULL.
sl@0
    35
#
sl@0
    36
# If any records in the database contain less fields than their table
sl@0
    37
# has columns, then the file-format meta value should be set to (at least) 2. 
sl@0
    38
#
sl@0
    39
sl@0
    40
# This procedure sets the value of the file-format in file 'test.db'
sl@0
    41
# to $newval. Also, the schema cookie is incremented.
sl@0
    42
# 
sl@0
    43
proc set_file_format {newval} {
sl@0
    44
  hexio_write test.db 44 [hexio_render_int32 $newval]
sl@0
    45
  set schemacookie [hexio_get_int [hexio_read test.db 40 4]]
sl@0
    46
  incr schemacookie
sl@0
    47
  hexio_write test.db 40 [hexio_render_int32 $schemacookie]
sl@0
    48
  return {}
sl@0
    49
}
sl@0
    50
sl@0
    51
# This procedure returns the value of the file-format in file 'test.db'.
sl@0
    52
# 
sl@0
    53
proc get_file_format {{fname test.db}} {
sl@0
    54
  return [hexio_get_int [hexio_read $fname 44 4]]
sl@0
    55
}
sl@0
    56
sl@0
    57
# This procedure sets the SQL statement stored for table $tbl in the
sl@0
    58
# sqlite_master table of file 'test.db' to $sql. Also set the file format
sl@0
    59
# to the supplied value. This is 2 if the added column has a default that is
sl@0
    60
# NULL, or 3 otherwise. 
sl@0
    61
#
sl@0
    62
proc alter_table {tbl sql {file_format 2}} {
sl@0
    63
  sqlite3 dbat test.db
sl@0
    64
  set s [string map {' ''} $sql]
sl@0
    65
  set t [string map {' ''} $tbl]
sl@0
    66
  dbat eval [subst {
sl@0
    67
    PRAGMA writable_schema = 1;
sl@0
    68
    UPDATE sqlite_master SET sql = '$s' WHERE name = '$t' AND type = 'table';
sl@0
    69
    PRAGMA writable_schema = 0;
sl@0
    70
  }]
sl@0
    71
  dbat close
sl@0
    72
  set_file_format 2
sl@0
    73
}
sl@0
    74
sl@0
    75
#-----------------------------------------------------------------------
sl@0
    76
# Some basic tests to make sure short rows are handled.
sl@0
    77
#
sl@0
    78
do_test alter2-1.1 {
sl@0
    79
  execsql {
sl@0
    80
    CREATE TABLE abc(a, b);
sl@0
    81
    INSERT INTO abc VALUES(1, 2);
sl@0
    82
    INSERT INTO abc VALUES(3, 4);
sl@0
    83
    INSERT INTO abc VALUES(5, 6);
sl@0
    84
  }
sl@0
    85
} {}
sl@0
    86
do_test alter2-1.2 {
sl@0
    87
  # ALTER TABLE abc ADD COLUMN c;
sl@0
    88
  alter_table abc {CREATE TABLE abc(a, b, c);}
sl@0
    89
} {}
sl@0
    90
do_test alter2-1.3 {
sl@0
    91
  execsql {
sl@0
    92
    SELECT * FROM abc;
sl@0
    93
  }
sl@0
    94
} {1 2 {} 3 4 {} 5 6 {}}
sl@0
    95
do_test alter2-1.4 {
sl@0
    96
  execsql {
sl@0
    97
    UPDATE abc SET c = 10 WHERE a = 1;
sl@0
    98
    SELECT * FROM abc;
sl@0
    99
  }
sl@0
   100
} {1 2 10 3 4 {} 5 6 {}}
sl@0
   101
do_test alter2-1.5 {
sl@0
   102
  execsql {
sl@0
   103
    CREATE INDEX abc_i ON abc(c);
sl@0
   104
  }
sl@0
   105
} {}
sl@0
   106
do_test alter2-1.6 {
sl@0
   107
  execsql {
sl@0
   108
    SELECT c FROM abc ORDER BY c;
sl@0
   109
  }
sl@0
   110
} {{} {} 10}
sl@0
   111
do_test alter2-1.7 {
sl@0
   112
  execsql {
sl@0
   113
    SELECT * FROM abc WHERE c = 10;
sl@0
   114
  }
sl@0
   115
} {1 2 10}
sl@0
   116
do_test alter2-1.8 {
sl@0
   117
  execsql {
sl@0
   118
    SELECT sum(a), c FROM abc GROUP BY c;
sl@0
   119
  }
sl@0
   120
} {8 {} 1 10}
sl@0
   121
do_test alter2-1.9 {
sl@0
   122
  # ALTER TABLE abc ADD COLUMN d;
sl@0
   123
  alter_table abc {CREATE TABLE abc(a, b, c, d);}
sl@0
   124
  execsql { SELECT * FROM abc; }
sl@0
   125
  execsql {
sl@0
   126
    UPDATE abc SET d = 11 WHERE c IS NULL AND a<4;
sl@0
   127
    SELECT * FROM abc;
sl@0
   128
  }
sl@0
   129
} {1 2 10 {} 3 4 {} 11 5 6 {} {}}
sl@0
   130
do_test alter2-1.10 {
sl@0
   131
  execsql {
sl@0
   132
    SELECT typeof(d) FROM abc;
sl@0
   133
  }
sl@0
   134
} {null integer null}
sl@0
   135
do_test alter2-1.99 {
sl@0
   136
  execsql {
sl@0
   137
    DROP TABLE abc;
sl@0
   138
  }
sl@0
   139
} {}
sl@0
   140
sl@0
   141
#-----------------------------------------------------------------------
sl@0
   142
# Test that views work when the underlying table structure is changed.
sl@0
   143
#
sl@0
   144
ifcapable view {
sl@0
   145
  do_test alter2-2.1 {
sl@0
   146
    execsql {
sl@0
   147
      CREATE TABLE abc2(a, b, c);
sl@0
   148
      INSERT INTO abc2 VALUES(1, 2, 10);
sl@0
   149
      INSERT INTO abc2 VALUES(3, 4, NULL);
sl@0
   150
      INSERT INTO abc2 VALUES(5, 6, NULL);
sl@0
   151
      CREATE VIEW abc2_v AS SELECT * FROM abc2;
sl@0
   152
      SELECT * FROM abc2_v;
sl@0
   153
    }
sl@0
   154
  } {1 2 10 3 4 {} 5 6 {}}
sl@0
   155
  do_test alter2-2.2 {
sl@0
   156
    # ALTER TABLE abc ADD COLUMN d;
sl@0
   157
    alter_table abc2 {CREATE TABLE abc2(a, b, c, d);}
sl@0
   158
    execsql {
sl@0
   159
      SELECT * FROM abc2_v;
sl@0
   160
    }
sl@0
   161
  } {1 2 10 {} 3 4 {} {} 5 6 {} {}}
sl@0
   162
  do_test alter2-2.3 {
sl@0
   163
    execsql {
sl@0
   164
      DROP TABLE abc2;
sl@0
   165
      DROP VIEW abc2_v;
sl@0
   166
    }
sl@0
   167
  } {}
sl@0
   168
}
sl@0
   169
sl@0
   170
#-----------------------------------------------------------------------
sl@0
   171
# Test that triggers work when a short row is copied to the old.*
sl@0
   172
# trigger pseudo-table.
sl@0
   173
#
sl@0
   174
ifcapable trigger {
sl@0
   175
  do_test alter2-3.1 {
sl@0
   176
    execsql {
sl@0
   177
      CREATE TABLE abc3(a, b);
sl@0
   178
      CREATE TABLE blog(o, n);
sl@0
   179
      CREATE TRIGGER abc3_t AFTER UPDATE OF b ON abc3 BEGIN
sl@0
   180
        INSERT INTO blog VALUES(old.b, new.b);
sl@0
   181
      END;
sl@0
   182
    }
sl@0
   183
  } {}
sl@0
   184
  do_test alter2-3.2 {
sl@0
   185
    execsql {
sl@0
   186
      INSERT INTO abc3 VALUES(1, 4);
sl@0
   187
      UPDATE abc3 SET b = 2 WHERE b = 4;
sl@0
   188
      SELECT * FROM blog;
sl@0
   189
    }
sl@0
   190
  } {4 2}
sl@0
   191
  do_test alter2-3.3 {
sl@0
   192
    execsql {
sl@0
   193
      INSERT INTO abc3 VALUES(3, 4);
sl@0
   194
      INSERT INTO abc3 VALUES(5, 6);
sl@0
   195
    }
sl@0
   196
    alter_table abc3 {CREATE TABLE abc3(a, b, c);}
sl@0
   197
    execsql {
sl@0
   198
      SELECT * FROM abc3;
sl@0
   199
    }
sl@0
   200
  } {1 2 {} 3 4 {} 5 6 {}}
sl@0
   201
  do_test alter2-3.4 {
sl@0
   202
    execsql {
sl@0
   203
      UPDATE abc3 SET b = b*2 WHERE a<4;
sl@0
   204
      SELECT * FROM abc3;
sl@0
   205
    }
sl@0
   206
  } {1 4 {} 3 8 {} 5 6 {}}
sl@0
   207
  do_test alter2-3.5 {
sl@0
   208
    execsql {
sl@0
   209
      SELECT * FROM blog;
sl@0
   210
    }
sl@0
   211
  } {4 2 2 4 4 8}
sl@0
   212
sl@0
   213
  do_test alter2-3.6 {
sl@0
   214
    execsql {
sl@0
   215
      CREATE TABLE clog(o, n);
sl@0
   216
      CREATE TRIGGER abc3_t2 AFTER UPDATE OF c ON abc3 BEGIN
sl@0
   217
        INSERT INTO clog VALUES(old.c, new.c);
sl@0
   218
      END;
sl@0
   219
      UPDATE abc3 SET c = a*2;
sl@0
   220
      SELECT * FROM clog;
sl@0
   221
    }
sl@0
   222
  } {{} 2 {} 6 {} 10}
sl@0
   223
}
sl@0
   224
sl@0
   225
#---------------------------------------------------------------------
sl@0
   226
# Check that an error occurs if the database is upgraded to a file
sl@0
   227
# format that SQLite does not support (in this case 5). Note: The 
sl@0
   228
# file format is checked each time the schema is read, so changing the
sl@0
   229
# file format requires incrementing the schema cookie.
sl@0
   230
#
sl@0
   231
do_test alter2-4.1 {
sl@0
   232
  db close
sl@0
   233
  set_file_format 5
sl@0
   234
  sqlite3 db test.db
sl@0
   235
} {}
sl@0
   236
do_test alter2-4.2 {
sl@0
   237
  # We have to run two queries here because the Tcl interface uses
sl@0
   238
  # sqlite3_prepare_v2(). In this case, the first query encounters an 
sl@0
   239
  # SQLITE_SCHEMA error. Then, when trying to recompile the statement, the
sl@0
   240
  # "unsupported file format" error is encountered. So the error code
sl@0
   241
  # returned is SQLITE_SCHEMA, not SQLITE_ERROR as required by the following
sl@0
   242
  # test case.
sl@0
   243
  #
sl@0
   244
  # When the query is attempted a second time, the same error message is
sl@0
   245
  # returned but the error code is SQLITE_ERROR, because the unsupported
sl@0
   246
  # file format was detected during a call to sqlite3_prepare(), not
sl@0
   247
  # sqlite3_step().
sl@0
   248
  #
sl@0
   249
  catchsql { SELECT * FROM sqlite_master; }
sl@0
   250
  catchsql { SELECT * FROM sqlite_master; }
sl@0
   251
} {1 {unsupported file format}}
sl@0
   252
do_test alter2-4.3 {
sl@0
   253
  sqlite3_errcode db
sl@0
   254
} {SQLITE_ERROR}
sl@0
   255
do_test alter2-4.4 {
sl@0
   256
  set ::DB [sqlite3_connection_pointer db]
sl@0
   257
  catchsql {
sl@0
   258
    SELECT * FROM sqlite_master;
sl@0
   259
  }
sl@0
   260
} {1 {unsupported file format}}
sl@0
   261
do_test alter2-4.5 {
sl@0
   262
  sqlite3_errcode db
sl@0
   263
} {SQLITE_ERROR}
sl@0
   264
sl@0
   265
#---------------------------------------------------------------------
sl@0
   266
# Check that executing VACUUM on a file with file-format version 2
sl@0
   267
# resets the file format to 1.
sl@0
   268
#
sl@0
   269
set default_file_format [expr $SQLITE_DEFAULT_FILE_FORMAT==4 ? 4 : 1]
sl@0
   270
ifcapable vacuum {
sl@0
   271
  do_test alter2-5.1 {
sl@0
   272
    set_file_format 2
sl@0
   273
    db close
sl@0
   274
    sqlite3 db test.db
sl@0
   275
    execsql {SELECT 1 FROM sqlite_master LIMIT 1;}
sl@0
   276
    get_file_format
sl@0
   277
  } {2}
sl@0
   278
  do_test alter2-5.2 {
sl@0
   279
    execsql {    
sl@0
   280
      VACUUM;
sl@0
   281
    }
sl@0
   282
  } {}
sl@0
   283
  do_test alter2-5.3 {
sl@0
   284
    get_file_format
sl@0
   285
  } $default_file_format
sl@0
   286
}
sl@0
   287
 
sl@0
   288
#---------------------------------------------------------------------
sl@0
   289
# Test that when a database with file-format 2 is opened, new 
sl@0
   290
# databases are still created with file-format 1.
sl@0
   291
#
sl@0
   292
do_test alter2-6.1 {
sl@0
   293
  db close
sl@0
   294
  set_file_format 2
sl@0
   295
  sqlite3 db test.db
sl@0
   296
  get_file_format
sl@0
   297
} {2}
sl@0
   298
ifcapable attach {
sl@0
   299
  do_test alter2-6.2 {
sl@0
   300
    file delete -force test2.db-journal
sl@0
   301
    file delete -force test2.db
sl@0
   302
    execsql {
sl@0
   303
      ATTACH 'test2.db' AS aux;
sl@0
   304
      CREATE TABLE aux.t1(a, b);
sl@0
   305
    }
sl@0
   306
    get_file_format test2.db
sl@0
   307
  } $default_file_format
sl@0
   308
}
sl@0
   309
do_test alter2-6.3 {
sl@0
   310
  execsql {
sl@0
   311
    CREATE TABLE t1(a, b);
sl@0
   312
  }
sl@0
   313
  get_file_format 
sl@0
   314
} {2}
sl@0
   315
sl@0
   316
#---------------------------------------------------------------------
sl@0
   317
# Test that types and values for columns added with default values 
sl@0
   318
# other than NULL work with SELECT statements.
sl@0
   319
#
sl@0
   320
do_test alter2-7.1 {
sl@0
   321
  execsql {
sl@0
   322
    DROP TABLE t1;
sl@0
   323
    CREATE TABLE t1(a);
sl@0
   324
    INSERT INTO t1 VALUES(1);
sl@0
   325
    INSERT INTO t1 VALUES(2);
sl@0
   326
    INSERT INTO t1 VALUES(3);
sl@0
   327
    INSERT INTO t1 VALUES(4);
sl@0
   328
    SELECT * FROM t1;
sl@0
   329
  }
sl@0
   330
} {1 2 3 4}
sl@0
   331
do_test alter2-7.2 {
sl@0
   332
  set sql {CREATE TABLE t1(a, b DEFAULT '123', c INTEGER DEFAULT '123')}
sl@0
   333
  alter_table t1 $sql 3
sl@0
   334
  execsql {
sl@0
   335
    SELECT * FROM t1 LIMIT 1;
sl@0
   336
  }
sl@0
   337
} {1 123 123}
sl@0
   338
do_test alter2-7.3 {
sl@0
   339
  execsql {
sl@0
   340
    SELECT a, typeof(a), b, typeof(b), c, typeof(c) FROM t1 LIMIT 1;
sl@0
   341
  }
sl@0
   342
} {1 integer 123 text 123 integer}
sl@0
   343
do_test alter2-7.4 {
sl@0
   344
  execsql {
sl@0
   345
    SELECT a, typeof(a), b, typeof(b), c, typeof(c) FROM t1 LIMIT 1;
sl@0
   346
  }
sl@0
   347
} {1 integer 123 text 123 integer}
sl@0
   348
do_test alter2-7.5 {
sl@0
   349
  set sql {CREATE TABLE t1(a, b DEFAULT -123.0, c VARCHAR(10) default 5)}
sl@0
   350
  alter_table t1 $sql 3
sl@0
   351
  execsql {
sl@0
   352
    SELECT a, typeof(a), b, typeof(b), c, typeof(c) FROM t1 LIMIT 1;
sl@0
   353
  }
sl@0
   354
} {1 integer -123 integer 5 text}
sl@0
   355
sl@0
   356
#-----------------------------------------------------------------------
sl@0
   357
# Test that UPDATE trigger tables work with default values, and that when
sl@0
   358
# a row is updated the default values are correctly transfered to the 
sl@0
   359
# new row.
sl@0
   360
# 
sl@0
   361
ifcapable trigger {
sl@0
   362
db function set_val {set ::val}
sl@0
   363
  do_test alter2-8.1 {
sl@0
   364
    execsql {
sl@0
   365
      CREATE TRIGGER trig1 BEFORE UPDATE ON t1 BEGIN
sl@0
   366
      SELECT set_val(
sl@0
   367
          old.b||' '||typeof(old.b)||' '||old.c||' '||typeof(old.c)||' '||
sl@0
   368
          new.b||' '||typeof(new.b)||' '||new.c||' '||typeof(new.c) 
sl@0
   369
      );
sl@0
   370
      END;
sl@0
   371
    }
sl@0
   372
    list
sl@0
   373
  } {}
sl@0
   374
}
sl@0
   375
do_test alter2-8.2 {
sl@0
   376
  execsql {
sl@0
   377
    UPDATE t1 SET c = 10 WHERE a = 1;
sl@0
   378
    SELECT a, typeof(a), b, typeof(b), c, typeof(c) FROM t1 LIMIT 1;
sl@0
   379
  }
sl@0
   380
} {1 integer -123 integer 10 text}
sl@0
   381
ifcapable trigger {
sl@0
   382
  do_test alter2-8.3 {
sl@0
   383
    set ::val
sl@0
   384
  } {-123 integer 5 text -123 integer 10 text}
sl@0
   385
}
sl@0
   386
sl@0
   387
#-----------------------------------------------------------------------
sl@0
   388
# Test that DELETE trigger tables work with default values, and that when
sl@0
   389
# a row is updated the default values are correctly transfered to the 
sl@0
   390
# new row.
sl@0
   391
# 
sl@0
   392
ifcapable trigger {
sl@0
   393
  do_test alter2-9.1 {
sl@0
   394
    execsql {
sl@0
   395
      CREATE TRIGGER trig2 BEFORE DELETE ON t1 BEGIN
sl@0
   396
      SELECT set_val(
sl@0
   397
          old.b||' '||typeof(old.b)||' '||old.c||' '||typeof(old.c)
sl@0
   398
      );
sl@0
   399
      END;
sl@0
   400
    }
sl@0
   401
    list
sl@0
   402
  } {}
sl@0
   403
  do_test alter2-9.2 {
sl@0
   404
    execsql {
sl@0
   405
      DELETE FROM t1 WHERE a = 2;
sl@0
   406
    }
sl@0
   407
    set ::val
sl@0
   408
  } {-123 integer 5 text}
sl@0
   409
}
sl@0
   410
sl@0
   411
#-----------------------------------------------------------------------
sl@0
   412
# Test creating an index on a column added with a default value. 
sl@0
   413
#
sl@0
   414
ifcapable bloblit {
sl@0
   415
  do_test alter2-10.1 {
sl@0
   416
    execsql {
sl@0
   417
      CREATE TABLE t2(a);
sl@0
   418
      INSERT INTO t2 VALUES('a');
sl@0
   419
      INSERT INTO t2 VALUES('b');
sl@0
   420
      INSERT INTO t2 VALUES('c');
sl@0
   421
      INSERT INTO t2 VALUES('d');
sl@0
   422
    }
sl@0
   423
    alter_table t2 {CREATE TABLE t2(a, b DEFAULT X'ABCD', c DEFAULT NULL);} 3
sl@0
   424
    catchsql {
sl@0
   425
      SELECT * FROM sqlite_master;
sl@0
   426
    }
sl@0
   427
    execsql {
sl@0
   428
      SELECT quote(a), quote(b), quote(c) FROM t2 LIMIT 1;
sl@0
   429
    }
sl@0
   430
  } {'a' X'ABCD' NULL}
sl@0
   431
  do_test alter2-10.2 {
sl@0
   432
    execsql {
sl@0
   433
      CREATE INDEX i1 ON t2(b);
sl@0
   434
      SELECT a FROM t2 WHERE b = X'ABCD';
sl@0
   435
    }
sl@0
   436
  } {a b c d}
sl@0
   437
  do_test alter2-10.3 {
sl@0
   438
    execsql {
sl@0
   439
      DELETE FROM t2 WHERE a = 'c';
sl@0
   440
      SELECT a FROM t2 WHERE b = X'ABCD';
sl@0
   441
    }
sl@0
   442
  } {a b d}
sl@0
   443
  do_test alter2-10.4 {
sl@0
   444
    execsql {
sl@0
   445
      SELECT count(b) FROM t2 WHERE b = X'ABCD';
sl@0
   446
    }
sl@0
   447
  } {3}
sl@0
   448
}
sl@0
   449
sl@0
   450
finish_test