os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/exclusive2.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
# 2007 March 24
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
# $Id: exclusive2.test,v 1.9 2008/08/22 00:25:53 aswift Exp $
sl@0
    14
sl@0
    15
set testdir [file dirname $argv0]
sl@0
    16
source $testdir/tester.tcl
sl@0
    17
sl@0
    18
ifcapable {!pager_pragmas} {
sl@0
    19
  finish_test
sl@0
    20
  return
sl@0
    21
}
sl@0
    22
sl@0
    23
# This module does not work right if the cache spills at unexpected
sl@0
    24
# moments.  So disable the soft-heap-limit.
sl@0
    25
#
sl@0
    26
sqlite3_soft_heap_limit 0
sl@0
    27
sl@0
    28
proc pagerChangeCounter {filename new {fd ""}} {
sl@0
    29
  if {$fd==""} {
sl@0
    30
    set fd [open $filename RDWR]
sl@0
    31
    fconfigure $fd -translation binary -encoding binary
sl@0
    32
    set needClose 1
sl@0
    33
  } else {
sl@0
    34
    set needClose 0
sl@0
    35
  }
sl@0
    36
  if {$new ne ""} {
sl@0
    37
    seek $fd 24
sl@0
    38
    set a [expr {($new&0xFF000000)>>24}]
sl@0
    39
    set b [expr {($new&0x00FF0000)>>16}]
sl@0
    40
    set c [expr {($new&0x0000FF00)>>8}]
sl@0
    41
    set d [expr {($new&0x000000FF)}]
sl@0
    42
    puts -nonewline $fd [binary format cccc $a $b $c $d]
sl@0
    43
    flush $fd
sl@0
    44
  }
sl@0
    45
sl@0
    46
  seek $fd 24
sl@0
    47
  foreach {a b c d} [list 0 0 0 0] {}
sl@0
    48
  binary scan [read $fd 4] cccc a b c d
sl@0
    49
  set  ret [expr ($a&0x000000FF)<<24]
sl@0
    50
  incr ret [expr ($b&0x000000FF)<<16]
sl@0
    51
  incr ret [expr ($c&0x000000FF)<<8]
sl@0
    52
  incr ret [expr ($d&0x000000FF)<<0]
sl@0
    53
sl@0
    54
  if {$needClose} {close $fd}
sl@0
    55
  return $ret
sl@0
    56
}
sl@0
    57
sl@0
    58
proc readPagerChangeCounter {filename} {
sl@0
    59
  set fd [open $filename RDONLY]
sl@0
    60
  fconfigure $fd -translation binary -encoding binary
sl@0
    61
sl@0
    62
  seek $fd 24
sl@0
    63
  foreach {a b c d} [list 0 0 0 0] {}
sl@0
    64
  binary scan [read $fd 4] cccc a b c d
sl@0
    65
  set  ret [expr ($a&0x000000FF)<<24]
sl@0
    66
  incr ret [expr ($b&0x000000FF)<<16]
sl@0
    67
  incr ret [expr ($c&0x000000FF)<<8]
sl@0
    68
  incr ret [expr ($d&0x000000FF)<<0]
sl@0
    69
sl@0
    70
  close $fd
sl@0
    71
  return $ret
sl@0
    72
}
sl@0
    73
sl@0
    74
sl@0
    75
proc t1sig {{db db}} {
sl@0
    76
  execsql {SELECT count(*), md5sum(a) FROM t1} $db
sl@0
    77
}
sl@0
    78
do_test exclusive2-1.0 {
sl@0
    79
  readPagerChangeCounter test.db
sl@0
    80
} {0}
sl@0
    81
sl@0
    82
#-----------------------------------------------------------------------
sl@0
    83
# The following tests - exclusive2-1.X - check that:
sl@0
    84
#
sl@0
    85
# 1-3:   Build a database with connection 1, calculate a signature.
sl@0
    86
# 4-9:   Modify the database using a second connection in a way that
sl@0
    87
#        does not modify the freelist, then reset the pager change-counter
sl@0
    88
#        to the value it had before the modifications.
sl@0
    89
# 8:     Check that using the first connection, the database signature
sl@0
    90
#        is still the same. This is because it uses the in-memory cache.
sl@0
    91
#        It can't tell the db has changed because we reset the change-counter.
sl@0
    92
# 9:     Increment the change-counter.
sl@0
    93
# 10:    Ensure that the first connection now sees the updated database. It
sl@0
    94
#        sees the change-counter has been incremented and discards the 
sl@0
    95
#        invalid in-memory cache.
sl@0
    96
#
sl@0
    97
# This will only work if the database cache is large enough to hold 
sl@0
    98
# the entire database. In the case of 1024 byte pages, this means
sl@0
    99
# the cache size must be at least 17. Otherwise, some pages will be
sl@0
   100
# loaded from the database file in step 8.
sl@0
   101
#
sl@0
   102
do_test exclusive2-1.1 {
sl@0
   103
  execsql {
sl@0
   104
    BEGIN;
sl@0
   105
    CREATE TABLE t1(a, b);
sl@0
   106
    INSERT INTO t1(a) VALUES(randstr(10, 400));
sl@0
   107
    INSERT INTO t1(a) VALUES(randstr(10, 400));
sl@0
   108
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   109
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   110
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   111
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   112
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   113
    COMMIT;
sl@0
   114
    SELECT count(*) FROM t1;
sl@0
   115
  }
sl@0
   116
} {64}
sl@0
   117
do_test exclusive2-1.2.1 {
sl@0
   118
  # Make sure the pager cache is large enough to store the 
sl@0
   119
  # entire database.
sl@0
   120
  set nPage [expr [file size test.db]/1024]
sl@0
   121
  if {$::SQLITE_DEFAULT_CACHE_SIZE < $nPage} {
sl@0
   122
    execsql "PRAGMA cache_size = $nPage"
sl@0
   123
  }
sl@0
   124
  expr {[execsql {PRAGMA cache_size}] >= $nPage}
sl@0
   125
} {1}
sl@0
   126
do_test exclusive2-1.2 {
sl@0
   127
  set ::sig [t1sig]
sl@0
   128
  readPagerChangeCounter test.db
sl@0
   129
} {1}
sl@0
   130
do_test exclusive2-1.3 {
sl@0
   131
  t1sig
sl@0
   132
} $::sig
sl@0
   133
do_test exclusive2-1.4 {
sl@0
   134
  sqlite3 db2 test.db
sl@0
   135
  t1sig db2
sl@0
   136
} $::sig
sl@0
   137
do_test exclusive2-1.5 {
sl@0
   138
  execsql {
sl@0
   139
    UPDATE t1 SET b=a, a=NULL;
sl@0
   140
  } db2
sl@0
   141
  expr {[t1sig db2] eq $::sig}
sl@0
   142
} 0
sl@0
   143
do_test exclusive2-1.6 {
sl@0
   144
  readPagerChangeCounter test.db
sl@0
   145
} {2}
sl@0
   146
do_test exclusive2-1.7 {
sl@0
   147
  pagerChangeCounter test.db 1
sl@0
   148
} {1}
sl@0
   149
do_test exclusive2-1.9 {
sl@0
   150
  t1sig
sl@0
   151
  expr {[t1sig] eq $::sig}
sl@0
   152
} {1}
sl@0
   153
do_test exclusive2-1.10 {
sl@0
   154
  pagerChangeCounter test.db 2
sl@0
   155
} {2}
sl@0
   156
do_test exclusive2-1.11 {
sl@0
   157
  expr {[t1sig] eq $::sig}
sl@0
   158
} {0}
sl@0
   159
sl@0
   160
#--------------------------------------------------------------------
sl@0
   161
# These tests - exclusive2-2.X - are similar to exclusive2-1.X, 
sl@0
   162
# except that they are run with locking_mode=EXCLUSIVE.
sl@0
   163
#
sl@0
   164
# 1-3:   Build a database with exclusive-access connection 1, 
sl@0
   165
#        calculate a signature.
sl@0
   166
# 4:     Corrupt the database by writing 10000 bytes of garbage
sl@0
   167
#        starting at the beginning of page 2. Check that connection 1
sl@0
   168
#        still works. It should be accessing the in-memory cache.
sl@0
   169
# 5-6:   Modify the dataase change-counter. Connection 1 still works
sl@0
   170
#        entirely from in-memory cache, because it doesn't check the
sl@0
   171
#        change-counter.
sl@0
   172
# 7-8    Set the locking-mode back to normal. After the db is unlocked,
sl@0
   173
#        SQLite detects the modified change-counter and discards the
sl@0
   174
#        in-memory cache. Then it finds the corruption caused in step 4....
sl@0
   175
#
sl@0
   176
# As above, this test is only applicable if the pager cache is
sl@0
   177
# large enough to hold the entire database. With 1024 byte pages,
sl@0
   178
# this means 19 pages.  We also need to disable the soft-heap-limit
sl@0
   179
# to prevent memory-induced cache spills.
sl@0
   180
#
sl@0
   181
do_test exclusive2-2.1 {
sl@0
   182
  execsql {PRAGMA locking_mode = exclusive;}
sl@0
   183
  execsql {
sl@0
   184
    BEGIN;
sl@0
   185
    DELETE FROM t1;
sl@0
   186
    INSERT INTO t1(a) VALUES(randstr(10, 400));
sl@0
   187
    INSERT INTO t1(a) VALUES(randstr(10, 400));
sl@0
   188
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   189
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   190
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   191
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   192
    INSERT INTO t1(a) SELECT randstr(10, 400) FROM t1;
sl@0
   193
    COMMIT;
sl@0
   194
    SELECT count(*) FROM t1;
sl@0
   195
  }
sl@0
   196
} {64}
sl@0
   197
do_test exclusive2-2.2.1 {
sl@0
   198
  # Make sure the pager cache is large enough to store the 
sl@0
   199
  # entire database.
sl@0
   200
  set nPage [expr [file size test.db]/1024]
sl@0
   201
  if {$::SQLITE_DEFAULT_CACHE_SIZE < $nPage} {
sl@0
   202
    execsql "PRAGMA cache_size = $nPage"
sl@0
   203
  }
sl@0
   204
  expr {[execsql {PRAGMA cache_size}] >= $nPage}
sl@0
   205
} {1}
sl@0
   206
do_test exclusive2-2.2 {
sl@0
   207
  set ::sig [t1sig]
sl@0
   208
  readPagerChangeCounter test.db
sl@0
   209
} {3}
sl@0
   210
do_test exclusive2-2.3 {
sl@0
   211
  t1sig
sl@0
   212
} $::sig
sl@0
   213
sl@0
   214
do_test exclusive2-2.4 {
sl@0
   215
  set ::fd [open test.db RDWR]
sl@0
   216
  fconfigure $::fd -translation binary
sl@0
   217
  seek $::fd 1024
sl@0
   218
  puts -nonewline $::fd [string repeat [binary format c 0] 10000]
sl@0
   219
  flush $::fd
sl@0
   220
  t1sig
sl@0
   221
} $::sig
sl@0
   222
sl@0
   223
do_test exclusive2-2.5 {
sl@0
   224
  pagerChangeCounter test.db 5 $::fd
sl@0
   225
} {5}
sl@0
   226
do_test exclusive2-2.6 {
sl@0
   227
  t1sig
sl@0
   228
} $::sig
sl@0
   229
do_test exclusive2-2.7 {
sl@0
   230
  execsql {PRAGMA locking_mode = normal}
sl@0
   231
  t1sig
sl@0
   232
} $::sig
sl@0
   233
sl@0
   234
do_test exclusive2-2.8 {
sl@0
   235
  set rc [catch {t1sig} msg]
sl@0
   236
  list $rc $msg
sl@0
   237
} {1 {database disk image is malformed}}
sl@0
   238
sl@0
   239
#--------------------------------------------------------------------
sl@0
   240
# These tests - exclusive2-3.X - verify that the pager change-counter
sl@0
   241
# is only incremented by the first change when in exclusive access
sl@0
   242
# mode. In normal mode, the change-counter is incremented once
sl@0
   243
# per write-transaction.
sl@0
   244
#
sl@0
   245
sl@0
   246
db close
sl@0
   247
db2 close
sl@0
   248
catch {close $::fd}
sl@0
   249
file delete -force test.db
sl@0
   250
file delete -force test.db-journal
sl@0
   251
sl@0
   252
do_test exclusive2-3.0 {
sl@0
   253
  sqlite3 db test.db
sl@0
   254
  execsql {
sl@0
   255
    BEGIN;
sl@0
   256
    CREATE TABLE t1(a UNIQUE);
sl@0
   257
    INSERT INTO t1 VALUES(randstr(10, 400));
sl@0
   258
    INSERT INTO t1 VALUES(randstr(10, 400));
sl@0
   259
    COMMIT;
sl@0
   260
  }
sl@0
   261
  readPagerChangeCounter test.db
sl@0
   262
} {1}
sl@0
   263
do_test exclusive2-3.1 {
sl@0
   264
  execsql {
sl@0
   265
    INSERT INTO t1 VALUES(randstr(10, 400));
sl@0
   266
  }
sl@0
   267
  readPagerChangeCounter test.db
sl@0
   268
} {2}
sl@0
   269
do_test exclusive2-3.2 {
sl@0
   270
  execsql {
sl@0
   271
    INSERT INTO t1 VALUES(randstr(10, 400));
sl@0
   272
  }
sl@0
   273
  readPagerChangeCounter test.db
sl@0
   274
} {3}
sl@0
   275
do_test exclusive2-3.3 {
sl@0
   276
  execsql {
sl@0
   277
    PRAGMA locking_mode = exclusive;
sl@0
   278
    INSERT INTO t1 VALUES(randstr(10, 400));
sl@0
   279
  }
sl@0
   280
  readPagerChangeCounter test.db
sl@0
   281
} {4}
sl@0
   282
do_test exclusive2-3.4 {
sl@0
   283
  execsql {
sl@0
   284
    INSERT INTO t1 VALUES(randstr(10, 400));
sl@0
   285
  }
sl@0
   286
  readPagerChangeCounter test.db
sl@0
   287
} {4}
sl@0
   288
do_test exclusive2-3.5 {
sl@0
   289
  execsql {
sl@0
   290
    PRAGMA locking_mode = normal;
sl@0
   291
    INSERT INTO t1 VALUES(randstr(10, 400));
sl@0
   292
  }
sl@0
   293
  readPagerChangeCounter test.db
sl@0
   294
} {4}
sl@0
   295
do_test exclusive2-3.6 {
sl@0
   296
  execsql {
sl@0
   297
    INSERT INTO t1 VALUES(randstr(10, 400));
sl@0
   298
  }
sl@0
   299
  readPagerChangeCounter test.db
sl@0
   300
} {5}
sl@0
   301
sqlite3_soft_heap_limit $soft_limit
sl@0
   302
sl@0
   303
finish_test