os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/crash.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
# 2001 September 15
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.
sl@0
    12
#
sl@0
    13
# The focus of this file is testing the ability of the database to
sl@0
    14
# uses its rollback journal to recover intact (no database corruption)
sl@0
    15
# from a power failure during the middle of a COMMIT.  The OS interface
sl@0
    16
# modules are overloaded using the modified I/O routines found in test6.c.  
sl@0
    17
# These routines allow us to simulate the kind of file damage that 
sl@0
    18
# occurs after a power failure.
sl@0
    19
#
sl@0
    20
# $Id: crash.test,v 1.27 2008/01/08 15:18:52 drh Exp $
sl@0
    21
sl@0
    22
set testdir [file dirname $argv0]
sl@0
    23
source $testdir/tester.tcl
sl@0
    24
sl@0
    25
ifcapable !crashtest {
sl@0
    26
  finish_test
sl@0
    27
  return
sl@0
    28
}
sl@0
    29
sl@0
    30
set repeats 100
sl@0
    31
#set repeats 10
sl@0
    32
sl@0
    33
# The following procedure computes a "signature" for table "abc".  If
sl@0
    34
# abc changes in any way, the signature should change.  
sl@0
    35
proc signature {} {
sl@0
    36
  return [db eval {SELECT count(*), md5sum(a), md5sum(b), md5sum(c) FROM abc}]
sl@0
    37
}
sl@0
    38
proc signature2 {} {
sl@0
    39
  return [db eval {SELECT count(*), md5sum(a), md5sum(b), md5sum(c) FROM abc2}]
sl@0
    40
}
sl@0
    41
sl@0
    42
#--------------------------------------------------------------------------
sl@0
    43
# Simple crash test:
sl@0
    44
#
sl@0
    45
# crash-1.1: Create a database with a table with two rows.
sl@0
    46
# crash-1.2: Run a 'DELETE FROM abc WHERE a = 1' that crashes during
sl@0
    47
#            the first journal-sync.
sl@0
    48
# crash-1.3: Ensure the database is in the same state as after crash-1.1.
sl@0
    49
# crash-1.4: Run a 'DELETE FROM abc WHERE a = 1' that crashes during
sl@0
    50
#            the first database-sync.
sl@0
    51
# crash-1.5: Ensure the database is in the same state as after crash-1.1.
sl@0
    52
# crash-1.6: Run a 'DELETE FROM abc WHERE a = 1' that crashes during
sl@0
    53
#            the second journal-sync.
sl@0
    54
# crash-1.7: Ensure the database is in the same state as after crash-1.1.
sl@0
    55
#
sl@0
    56
# Tests 1.8 through 1.11 test for crashes on the third journal sync and
sl@0
    57
# second database sync.  Neither of these is required in such a small test
sl@0
    58
# case, so these tests are just to verify that the test infrastructure
sl@0
    59
# operates as expected.
sl@0
    60
#
sl@0
    61
do_test crash-1.1 {
sl@0
    62
  execsql {
sl@0
    63
    CREATE TABLE abc(a, b, c);
sl@0
    64
    INSERT INTO abc VALUES(1, 2, 3);
sl@0
    65
    INSERT INTO abc VALUES(4, 5, 6);
sl@0
    66
  }
sl@0
    67
  set ::sig [signature]
sl@0
    68
  expr 0
sl@0
    69
} {0}
sl@0
    70
for {set i 0} {$i<10} {incr i} {
sl@0
    71
  set seed [expr {int(abs(rand()*10000))}]
sl@0
    72
  do_test crash-1.2.$i {
sl@0
    73
    crashsql -delay 1 -file test.db-journal -seed $seed {
sl@0
    74
      DELETE FROM abc WHERE a = 1;
sl@0
    75
    }
sl@0
    76
  } {1 {child process exited abnormally}}
sl@0
    77
  do_test crash-1.3.$i {
sl@0
    78
    signature
sl@0
    79
  } $::sig
sl@0
    80
}
sl@0
    81
do_test crash-1.4 {
sl@0
    82
  crashsql -delay 1 -file test.db {
sl@0
    83
    DELETE FROM abc WHERE a = 1;
sl@0
    84
  }
sl@0
    85
} {1 {child process exited abnormally}}
sl@0
    86
do_test crash-1.5 {
sl@0
    87
  signature
sl@0
    88
} $::sig
sl@0
    89
do_test crash-1.6 {
sl@0
    90
  crashsql -delay 2 -file test.db-journal {
sl@0
    91
    DELETE FROM abc WHERE a = 1;
sl@0
    92
  }
sl@0
    93
} {1 {child process exited abnormally}}
sl@0
    94
do_test crash-1.7 {
sl@0
    95
  catchsql {
sl@0
    96
    SELECT * FROM abc;
sl@0
    97
  }
sl@0
    98
} {0 {1 2 3 4 5 6}}
sl@0
    99
sl@0
   100
do_test crash-1.8 {
sl@0
   101
  crashsql -delay 3 -file test.db-journal {
sl@0
   102
    DELETE FROM abc WHERE a = 1;
sl@0
   103
  }
sl@0
   104
} {0 {}}
sl@0
   105
do_test crash-1.9 {
sl@0
   106
  catchsql {
sl@0
   107
    SELECT * FROM abc;
sl@0
   108
  }
sl@0
   109
} {0 {4 5 6}}
sl@0
   110
do_test crash-1.10 {
sl@0
   111
  crashsql -delay 2 -file test.db {
sl@0
   112
    DELETE FROM abc WHERE a = 4;
sl@0
   113
  }
sl@0
   114
} {0 {}}
sl@0
   115
do_test crash-1.11 {
sl@0
   116
  catchsql {
sl@0
   117
    SELECT * FROM abc;
sl@0
   118
  }
sl@0
   119
} {0 {}}
sl@0
   120
sl@0
   121
#--------------------------------------------------------------------------
sl@0
   122
# The following tests test recovery when both the database file and the the
sl@0
   123
# journal file contain corrupt data. This can happen after pages are
sl@0
   124
# written to the database file before a transaction is committed due to
sl@0
   125
# cache-pressure.
sl@0
   126
#
sl@0
   127
# crash-2.1: Insert 18 pages of data into the database.
sl@0
   128
# crash-2.2: Check the database file size looks ok.
sl@0
   129
# crash-2.3: Delete 15 or so pages (with a 10 page page-cache), then crash.
sl@0
   130
# crash-2.4: Ensure the database is in the same state as after crash-2.1.
sl@0
   131
#
sl@0
   132
# Test cases crash-2.5 and crash-2.6 check that the database is OK if the 
sl@0
   133
# crash occurs during the main database file sync. But this isn't really
sl@0
   134
# different from the crash-1.* cases.
sl@0
   135
#
sl@0
   136
do_test crash-2.1 {
sl@0
   137
  execsql { BEGIN }
sl@0
   138
  for {set n 0} {$n < 1000} {incr n} {
sl@0
   139
    execsql "INSERT INTO abc VALUES($n, [expr 2*$n], [expr 3*$n])"
sl@0
   140
  }
sl@0
   141
  execsql { COMMIT }
sl@0
   142
  set ::sig [signature]
sl@0
   143
  execsql { SELECT sum(a), sum(b), sum(c) from abc }
sl@0
   144
} {499500 999000 1498500}
sl@0
   145
do_test crash-2.2 {
sl@0
   146
  expr ([file size test.db] / 1024)>16
sl@0
   147
} {1}
sl@0
   148
do_test crash-2.3 {
sl@0
   149
  crashsql -delay 2 -file test.db-journal {
sl@0
   150
    DELETE FROM abc WHERE a < 800;
sl@0
   151
  }
sl@0
   152
} {1 {child process exited abnormally}}
sl@0
   153
do_test crash-2.4 {
sl@0
   154
  signature
sl@0
   155
} $sig
sl@0
   156
do_test crash-2.5 {
sl@0
   157
  crashsql -delay 1 -file test.db {
sl@0
   158
    DELETE FROM abc WHERE a<800;
sl@0
   159
  }
sl@0
   160
} {1 {child process exited abnormally}}
sl@0
   161
do_test crash-2.6 {
sl@0
   162
  signature
sl@0
   163
} $sig
sl@0
   164
sl@0
   165
#--------------------------------------------------------------------------
sl@0
   166
# The crash-3.* test cases are essentially the same test as test case
sl@0
   167
# crash-2.*, but with a more complicated data set. 
sl@0
   168
#
sl@0
   169
# The test is repeated a few times with different seeds for the random
sl@0
   170
# number generator in the crashing executable. Because there is no way to
sl@0
   171
# seed the random number generator directly, some SQL is added to the test
sl@0
   172
# case to 'use up' a different quantity random numbers before the test SQL
sl@0
   173
# is executed.
sl@0
   174
#
sl@0
   175
sl@0
   176
# Make sure the file is much bigger than the pager-cache (10 pages). This
sl@0
   177
# ensures that cache-spills happen regularly.
sl@0
   178
do_test crash-3.0 {
sl@0
   179
  execsql {
sl@0
   180
    INSERT INTO abc SELECT * FROM abc;
sl@0
   181
    INSERT INTO abc SELECT * FROM abc;
sl@0
   182
    INSERT INTO abc SELECT * FROM abc;
sl@0
   183
    INSERT INTO abc SELECT * FROM abc;
sl@0
   184
    INSERT INTO abc SELECT * FROM abc;
sl@0
   185
  }
sl@0
   186
  expr ([file size test.db] / 1024) > 450
sl@0
   187
} {1}
sl@0
   188
for {set i 1} {$i < $repeats} {incr i} {
sl@0
   189
  set sig [signature]
sl@0
   190
  do_test crash-3.$i.1 {
sl@0
   191
     set seed [expr {int(abs(rand()*10000))}]
sl@0
   192
     crashsql -delay [expr $i%5 + 1] -file test.db-journal -seed $seed "
sl@0
   193
       BEGIN;
sl@0
   194
       SELECT random() FROM abc LIMIT $i;
sl@0
   195
       INSERT INTO abc VALUES(randstr(10,10), 0, 0);
sl@0
   196
       DELETE FROM abc WHERE random()%10!=0;
sl@0
   197
       COMMIT;
sl@0
   198
     "
sl@0
   199
  } {1 {child process exited abnormally}}
sl@0
   200
  do_test crash-3.$i.2 {
sl@0
   201
    signature
sl@0
   202
  } $sig
sl@0
   203
} 
sl@0
   204
sl@0
   205
#--------------------------------------------------------------------------
sl@0
   206
# The following test cases - crash-4.* - test the correct recovery of the
sl@0
   207
# database when a crash occurs during a multi-file transaction.
sl@0
   208
#
sl@0
   209
# crash-4.1.*: Test recovery when crash occurs during sync() of the 
sl@0
   210
#              main database journal file.
sl@0
   211
# crash-4.2.*: Test recovery when crash occurs during sync() of an 
sl@0
   212
#              attached database journal file.
sl@0
   213
# crash-4.3.*: Test recovery when crash occurs during sync() of the master
sl@0
   214
#              journal file. 
sl@0
   215
#
sl@0
   216
ifcapable attach {
sl@0
   217
  do_test crash-4.0 {
sl@0
   218
    file delete -force test2.db
sl@0
   219
    file delete -force test2.db-journal
sl@0
   220
    execsql {
sl@0
   221
      ATTACH 'test2.db' AS aux;
sl@0
   222
      PRAGMA aux.default_cache_size = 10;
sl@0
   223
      CREATE TABLE aux.abc2 AS SELECT 2*a as a, 2*b as b, 2*c as c FROM abc;
sl@0
   224
    }
sl@0
   225
    expr ([file size test2.db] / 1024) > 450
sl@0
   226
  } {1}
sl@0
   227
  
sl@0
   228
  set fin 0
sl@0
   229
  for {set i 1} {$i<$repeats} {incr i} {
sl@0
   230
    set seed [expr {int(abs(rand()*10000))}]
sl@0
   231
    set sig [signature]
sl@0
   232
    set sig2 [signature2]
sl@0
   233
    do_test crash-4.1.$i.1 {
sl@0
   234
       set c [crashsql -delay $i -file test.db-journal -seed $::seed "
sl@0
   235
         ATTACH 'test2.db' AS aux;
sl@0
   236
         BEGIN;
sl@0
   237
         SELECT randstr($i,$i) FROM abc LIMIT $i;
sl@0
   238
         INSERT INTO abc VALUES(randstr(10,10), 0, 0);
sl@0
   239
         DELETE FROM abc WHERE random()%10!=0;
sl@0
   240
         INSERT INTO abc2 VALUES(randstr(10,10), 0, 0);
sl@0
   241
         DELETE FROM abc2 WHERE random()%10!=0;
sl@0
   242
         COMMIT;
sl@0
   243
       "]
sl@0
   244
       if { $c == {0 {}} } {
sl@0
   245
         set ::fin 1
sl@0
   246
         set c {1 {child process exited abnormally}}
sl@0
   247
       }
sl@0
   248
       set c
sl@0
   249
    } {1 {child process exited abnormally}}
sl@0
   250
    if {$::fin} break
sl@0
   251
    do_test crash-4.1.$i.2 {
sl@0
   252
      signature
sl@0
   253
    } $sig
sl@0
   254
    do_test crash-4.1.$i.3 {
sl@0
   255
      signature2
sl@0
   256
    } $sig2
sl@0
   257
  } 
sl@0
   258
  set i 0
sl@0
   259
  set fin 0
sl@0
   260
  while {[incr i]} {
sl@0
   261
    set seed [expr {int(abs(rand()*10000))}]
sl@0
   262
    set sig [signature]
sl@0
   263
    set sig2 [signature2]
sl@0
   264
    set ::fin 0
sl@0
   265
    do_test crash-4.2.$i.1 {
sl@0
   266
       set c [crashsql -delay $i -file test2.db-journal -seed $::seed "
sl@0
   267
         ATTACH 'test2.db' AS aux;
sl@0
   268
         BEGIN;
sl@0
   269
         SELECT randstr($i,$i) FROM abc LIMIT $i;
sl@0
   270
         INSERT INTO abc VALUES(randstr(10,10), 0, 0);
sl@0
   271
         DELETE FROM abc WHERE random()%10!=0;
sl@0
   272
         INSERT INTO abc2 VALUES(randstr(10,10), 0, 0);
sl@0
   273
         DELETE FROM abc2 WHERE random()%10!=0;
sl@0
   274
         COMMIT;
sl@0
   275
       "]
sl@0
   276
       if { $c == {0 {}} } {
sl@0
   277
         set ::fin 1
sl@0
   278
         set c {1 {child process exited abnormally}}
sl@0
   279
       }
sl@0
   280
       set c
sl@0
   281
    } {1 {child process exited abnormally}}
sl@0
   282
    if { $::fin } break
sl@0
   283
    do_test crash-4.2.$i.2 {
sl@0
   284
      signature
sl@0
   285
    } $sig
sl@0
   286
    do_test crash-4.2.$i.3 {
sl@0
   287
      signature2
sl@0
   288
    } $sig2
sl@0
   289
  } 
sl@0
   290
  for {set i 1} {$i < 5} {incr i} {
sl@0
   291
    set sig [signature]
sl@0
   292
    set sig2 [signature2]
sl@0
   293
    do_test crash-4.3.$i.1 {
sl@0
   294
       crashsql -delay 1 -file test.db-mj* "
sl@0
   295
         ATTACH 'test2.db' AS aux;
sl@0
   296
         BEGIN;
sl@0
   297
         SELECT random() FROM abc LIMIT $i;
sl@0
   298
         INSERT INTO abc VALUES(randstr(10,10), 0, 0);
sl@0
   299
         DELETE FROM abc WHERE random()%10!=0;
sl@0
   300
         INSERT INTO abc2 VALUES(randstr(10,10), 0, 0);
sl@0
   301
         DELETE FROM abc2 WHERE random()%10!=0;
sl@0
   302
         COMMIT;
sl@0
   303
       "
sl@0
   304
    } {1 {child process exited abnormally}}
sl@0
   305
    do_test crash-4.3.$i.2 {
sl@0
   306
      signature
sl@0
   307
    } $sig
sl@0
   308
    do_test crash-4.3.$i.3 {
sl@0
   309
      signature2
sl@0
   310
    } $sig2
sl@0
   311
  }
sl@0
   312
}
sl@0
   313
sl@0
   314
#--------------------------------------------------------------------------
sl@0
   315
# The following test cases - crash-5.* - exposes a bug that existed in the
sl@0
   316
# sqlite3pager_movepage() API used by auto-vacuum databases.
sl@0
   317
# database when a crash occurs during a multi-file transaction. See comments
sl@0
   318
# in test crash-5.3 for details.
sl@0
   319
#
sl@0
   320
db close
sl@0
   321
file delete -force test.db
sl@0
   322
sqlite3 db test.db
sl@0
   323
do_test crash-5.1 {
sl@0
   324
  execsql {
sl@0
   325
    CREATE TABLE abc(a, b, c);                          -- Root page 3
sl@0
   326
    INSERT INTO abc VALUES(randstr(1500,1500), 0, 0);   -- Overflow page 4
sl@0
   327
    INSERT INTO abc SELECT * FROM abc;
sl@0
   328
    INSERT INTO abc SELECT * FROM abc;
sl@0
   329
    INSERT INTO abc SELECT * FROM abc;
sl@0
   330
  }
sl@0
   331
} {}
sl@0
   332
do_test crash-5.2 {
sl@0
   333
  expr [file size test.db] / 1024
sl@0
   334
} [expr [string match [execsql {pragma auto_vacuum}] 1] ? 11 : 10]
sl@0
   335
set sig [signature]
sl@0
   336
do_test crash-5.3 {
sl@0
   337
# The SQL below is used to expose a bug that existed in
sl@0
   338
# sqlite3pager_movepage() during development of the auto-vacuum feature. It
sl@0
   339
# functions as follows:
sl@0
   340
# 
sl@0
   341
# 1: Begin a transaction.
sl@0
   342
# 2: Put page 4 on the free-list (was the overflow page for the row deleted).
sl@0
   343
# 3: Write data to page 4 (it becomes the overflow page for the row inserted).
sl@0
   344
#    The old page 4 data has been written to the journal file, but the
sl@0
   345
#    journal file has not been sync()hronized.
sl@0
   346
# 4: Create a table, which calls sqlite3pager_movepage() to move page 4
sl@0
   347
#    to the end of the database (page 12) to make room for the new root-page.
sl@0
   348
# 5: Put pressure on the pager-cache. This results in page 4 being written
sl@0
   349
#    to the database file to make space in the cache to load a new page. The
sl@0
   350
#    bug was that page 4 was written to the database file before the journal
sl@0
   351
#    is sync()hronized.
sl@0
   352
# 6: Commit. A crash occurs during the sync of the journal file.
sl@0
   353
#
sl@0
   354
# End result: Before the bug was fixed, data has been written to page 4 of the
sl@0
   355
# database file and the journal file does not contain trustworthy rollback
sl@0
   356
# data for this page.
sl@0
   357
#
sl@0
   358
  crashsql -delay 1 -file test.db-journal {
sl@0
   359
    BEGIN;                                             -- 1
sl@0
   360
    DELETE FROM abc WHERE oid = 1;                     -- 2
sl@0
   361
    INSERT INTO abc VALUES(randstr(1500,1500), 0, 0);  -- 3
sl@0
   362
    CREATE TABLE abc2(a, b, c);                        -- 4
sl@0
   363
    SELECT * FROM abc;                                 -- 5
sl@0
   364
    COMMIT;                                            -- 6
sl@0
   365
  }
sl@0
   366
} {1 {child process exited abnormally}}
sl@0
   367
integrity_check crash-5.4
sl@0
   368
do_test crash-5.5 {
sl@0
   369
  signature
sl@0
   370
} $sig
sl@0
   371
sl@0
   372
#--------------------------------------------------------------------------
sl@0
   373
# The following test cases - crash-6.* - test that a DROP TABLE operation
sl@0
   374
# is correctly rolled back in the event of a crash while the database file
sl@0
   375
# is being written. This is mainly to test that all pages are written to the
sl@0
   376
# journal file before truncation in an auto-vacuum database.
sl@0
   377
#
sl@0
   378
do_test crash-6.1 {
sl@0
   379
  crashsql -delay 1 -file test.db {
sl@0
   380
    DROP TABLE abc;
sl@0
   381
  }
sl@0
   382
} {1 {child process exited abnormally}}
sl@0
   383
do_test crash-6.2 {
sl@0
   384
  signature
sl@0
   385
} $sig
sl@0
   386
sl@0
   387
#--------------------------------------------------------------------------
sl@0
   388
# These test cases test the case where the master journal file name is 
sl@0
   389
# corrupted slightly so that the corruption has to be detected by the
sl@0
   390
# checksum.
sl@0
   391
do_test crash-7.1 {
sl@0
   392
  crashsql -delay 1 -file test.db {
sl@0
   393
    ATTACH 'test2.db' AS aux;
sl@0
   394
    BEGIN;
sl@0
   395
    INSERT INTO abc VALUES(randstr(1500,1500), 0, 0);
sl@0
   396
    INSERT INTO abc2 VALUES(randstr(1500,1500), 0, 0);
sl@0
   397
    COMMIT;
sl@0
   398
  }
sl@0
   399
sl@0
   400
  # Change the checksum value for the master journal name.
sl@0
   401
  set f [open test.db-journal a]
sl@0
   402
  fconfigure $f -encoding binary
sl@0
   403
  seek $f [expr [file size test.db-journal] - 12]
sl@0
   404
  puts -nonewline $f "\00\00\00\00"
sl@0
   405
  close $f
sl@0
   406
} {}
sl@0
   407
do_test crash-7.2 {
sl@0
   408
  signature
sl@0
   409
} $sig
sl@0
   410
sl@0
   411
finish_test