os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/tkt2409.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 June 13
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
# This file implements tests to verify that ticket #2409 has been
sl@0
    14
# fixed. More specifically, they verify that if SQLite cannot
sl@0
    15
# obtain an EXCLUSIVE lock while trying to spill the cache during
sl@0
    16
# any statement other than a COMMIT, an I/O error is returned instead
sl@0
    17
# of SQLITE_BUSY.
sl@0
    18
#
sl@0
    19
# $Id: tkt2409.test,v 1.6 2008/08/28 17:46:19 drh Exp $
sl@0
    20
sl@0
    21
# Test Outline:
sl@0
    22
#
sl@0
    23
#   tkt-2409-1.*: Cause a cache-spill during an INSERT that is within
sl@0
    24
#       a db transaction but does not start a statement transaction.
sl@0
    25
#       Verify that the transaction is automatically rolled back
sl@0
    26
#       and SQLITE_IOERR_BLOCKED is returned
sl@0
    27
#
sl@0
    28
#       UPDATE: As of the pcache modifications, failing to upgrade to
sl@0
    29
#       an exclusive lock when attempting a cache-spill is no longer an
sl@0
    30
#       error. The pcache module allocates more space and keeps working
sl@0
    31
#       in memory if this occurs.
sl@0
    32
#
sl@0
    33
#   tkt-2409-2.*: Cause a cache-spill while updating the change-counter
sl@0
    34
#       during a database COMMIT. Verify that the transaction is not
sl@0
    35
#       rolled back and SQLITE_BUSY is returned.
sl@0
    36
#
sl@0
    37
#   tkt-2409-3.*: Similar to 2409-1.*, but using many INSERT statements
sl@0
    38
#       within a transaction instead of just one.
sl@0
    39
#
sl@0
    40
#       UPDATE: Again, pcache now just keeps working in main memory.
sl@0
    41
#
sl@0
    42
#   tkt-2409-4.*: Similar to 2409-1.*, but rig it so that the
sl@0
    43
#       INSERT statement starts a statement transaction. Verify that
sl@0
    44
#       SQLITE_BUSY is returned and the transaction is not rolled back.
sl@0
    45
#
sl@0
    46
#       UPDATE: This time, SQLITE_BUSY is not returned. pcache just uses
sl@0
    47
#       more malloc()'d memory.
sl@0
    48
#
sl@0
    49
sl@0
    50
set testdir [file dirname $argv0]
sl@0
    51
source $testdir/tester.tcl
sl@0
    52
sl@0
    53
ifcapable !pager_pragmas {
sl@0
    54
  finish_test
sl@0
    55
  return
sl@0
    56
}
sl@0
    57
sl@0
    58
sqlite3_extended_result_codes $::DB 1
sl@0
    59
sl@0
    60
# Aquire a read-lock on the database using handle [db2].
sl@0
    61
#
sl@0
    62
proc read_lock_db {} {
sl@0
    63
  if {$::STMT eq ""} {
sl@0
    64
    set ::STMT [sqlite3_prepare db2 {SELECT rowid FROM sqlite_master} -1 TAIL]
sl@0
    65
    set rc [sqlite3_step $::STMT]
sl@0
    66
    if {$rc eq "SQLITE_ERROR"} {
sl@0
    67
      unread_lock_db
sl@0
    68
      read_lock_db
sl@0
    69
    }
sl@0
    70
  }
sl@0
    71
}
sl@0
    72
sl@0
    73
# Release any read-lock obtained using [read_lock_db]
sl@0
    74
#
sl@0
    75
proc unread_lock_db {} {
sl@0
    76
  if {$::STMT ne ""} {
sl@0
    77
    sqlite3_finalize $::STMT
sl@0
    78
    set ::STMT ""
sl@0
    79
  }
sl@0
    80
}
sl@0
    81
sl@0
    82
# Open the db handle used by [read_lock_db].
sl@0
    83
#
sl@0
    84
sqlite3 db2 test.db
sl@0
    85
set ::STMT ""
sl@0
    86
sl@0
    87
do_test tkt2409-1.1 {
sl@0
    88
  execsql {
sl@0
    89
    PRAGMA cache_size=10;
sl@0
    90
    CREATE TABLE t1(x TEXT UNIQUE NOT NULL, y BLOB);
sl@0
    91
  }
sl@0
    92
  read_lock_db
sl@0
    93
  set ::zShort [string repeat 0123456789 1]
sl@0
    94
  set ::zLong  [string repeat 0123456789 1500]
sl@0
    95
  catchsql {
sl@0
    96
    BEGIN;
sl@0
    97
    INSERT INTO t1 VALUES($::zShort, $::zLong);
sl@0
    98
  }
sl@0
    99
} {0 {}}
sl@0
   100
sl@0
   101
do_test tkt2409-1.2 {
sl@0
   102
  sqlite3_errcode $::DB
sl@0
   103
} {SQLITE_OK}
sl@0
   104
sl@0
   105
# Check the integrity of the cache.
sl@0
   106
#
sl@0
   107
integrity_check tkt2409-1.3
sl@0
   108
sl@0
   109
# Check that the transaction was rolled back. Because the INSERT
sl@0
   110
# statement in which the "I/O error" occured did not open a statement
sl@0
   111
# transaction, SQLite had no choice but to roll back the transaction.
sl@0
   112
#
sl@0
   113
do_test tkt2409-1.4 {
sl@0
   114
  unread_lock_db
sl@0
   115
  catchsql { ROLLBACK }
sl@0
   116
} {0 {}}
sl@0
   117
sl@0
   118
set ::zShort [string repeat 0123456789 1]
sl@0
   119
set ::zLong  [string repeat 0123456789 1500]
sl@0
   120
set ::rc 1
sl@0
   121
for {set iCache 10} {$::rc} {incr iCache} {
sl@0
   122
  execsql "PRAGMA cache_size = $iCache"
sl@0
   123
  do_test tkt2409-2.1.$iCache {
sl@0
   124
    read_lock_db
sl@0
   125
    set ::rc [catch {
sl@0
   126
      execsql {
sl@0
   127
        BEGIN;
sl@0
   128
        INSERT INTO t1 VALUES($::zShort, $::zLong);
sl@0
   129
      }
sl@0
   130
    } msg]
sl@0
   131
    expr {($::rc == 1 && $msg eq "disk I/O error") || $::rc == 0}
sl@0
   132
  } {1}
sl@0
   133
}
sl@0
   134
sl@0
   135
do_test tkt2409-2.2 {
sl@0
   136
  catchsql {
sl@0
   137
    ROLLBACK;
sl@0
   138
    BEGIN;
sl@0
   139
    INSERT INTO t1 VALUES($::zShort, $::zLong);
sl@0
   140
    COMMIT;
sl@0
   141
  }
sl@0
   142
} {1 {database is locked}}
sl@0
   143
sl@0
   144
do_test tkt2409-2.3 {
sl@0
   145
  unread_lock_db
sl@0
   146
  catchsql {
sl@0
   147
    COMMIT;
sl@0
   148
  }
sl@0
   149
} {0 {}}
sl@0
   150
sl@0
   151
sl@0
   152
do_test tkt2409-3.1 {
sl@0
   153
  db close
sl@0
   154
  set ::DB [sqlite3 db test.db; sqlite3_connection_pointer db]
sl@0
   155
  sqlite3_extended_result_codes $::DB 1
sl@0
   156
  execsql {
sl@0
   157
    PRAGMA cache_size=10;
sl@0
   158
    DELETE FROM t1;
sl@0
   159
  }
sl@0
   160
  read_lock_db
sl@0
   161
  set ::zShort [string repeat 0123456789 1]
sl@0
   162
  set ::zLong  [string repeat 0123456789 1500]
sl@0
   163
  catchsql {
sl@0
   164
    BEGIN;
sl@0
   165
    INSERT INTO t1 SELECT $::zShort, $::zLong;
sl@0
   166
  }
sl@0
   167
} {0 {}}
sl@0
   168
sl@0
   169
do_test tkt2409-3.2 {
sl@0
   170
  sqlite3_errcode $::DB
sl@0
   171
} {SQLITE_OK}
sl@0
   172
sl@0
   173
# Check the integrity of the cache.
sl@0
   174
#
sl@0
   175
integrity_check tkt2409-3.3
sl@0
   176
sl@0
   177
# Check that the transaction was rolled back. Because the INSERT
sl@0
   178
# statement in which the "I/O error" occured did not open a statement
sl@0
   179
# transaction, SQLite had no choice but to roll back the transaction.
sl@0
   180
#
sl@0
   181
do_test tkt2409-3.4 {
sl@0
   182
  unread_lock_db
sl@0
   183
  catchsql { ROLLBACK }
sl@0
   184
} {0 {}}
sl@0
   185
integrity_check tkt2409-3.5
sl@0
   186
sl@0
   187
expr {srand(1)}
sl@0
   188
do_test tkt2409-4.1 {
sl@0
   189
  execsql {
sl@0
   190
    PRAGMA cache_size=20;
sl@0
   191
    DROP TABLE t1;
sl@0
   192
    CREATE TABLE t1 (x TEXT UNIQUE NOT NULL);
sl@0
   193
  }
sl@0
   194
sl@0
   195
  unset -nocomplain t1
sl@0
   196
  array unset t1
sl@0
   197
  set t1(0) 1
sl@0
   198
  set sql ""
sl@0
   199
  for {set i 0} {$i<5000} {incr i} {
sl@0
   200
    set r 0
sl@0
   201
    while {[info exists t1($r)]} {
sl@0
   202
      set r [expr {int(rand()*1000000000)}]
sl@0
   203
    }
sl@0
   204
    set t1($r) 1
sl@0
   205
    append sql "INSERT INTO t1 VALUES('some-text-$r');"
sl@0
   206
  }
sl@0
   207
sl@0
   208
  read_lock_db
sl@0
   209
  execsql BEGIN
sl@0
   210
  catchsql $sql
sl@0
   211
} {0 {}}
sl@0
   212
sl@0
   213
do_test tkt2409-4.2 {
sl@0
   214
  sqlite3_errcode $::DB
sl@0
   215
} {SQLITE_OK}
sl@0
   216
sl@0
   217
# Check the integrity of the cache.
sl@0
   218
#
sl@0
   219
integrity_check tkt2409-4.3
sl@0
   220
sl@0
   221
do_test tkt2409-4.4 {
sl@0
   222
  catchsql { ROLLBACK }
sl@0
   223
} {0 {}}
sl@0
   224
integrity_check tkt2409-4.5
sl@0
   225
sl@0
   226
unread_lock_db
sl@0
   227
db2 close
sl@0
   228
unset -nocomplain t1
sl@0
   229
finish_test