os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/lock2.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.  The
sl@0
    12
# focus of this script is database locks between competing processes.
sl@0
    13
#
sl@0
    14
# $Id: lock2.test,v 1.9 2007/12/13 21:54:11 drh Exp $
sl@0
    15
sl@0
    16
sl@0
    17
set testdir [file dirname $argv0]
sl@0
    18
source $testdir/tester.tcl
sl@0
    19
sl@0
    20
# Launch another testfixture process to be controlled by this one. A
sl@0
    21
# channel name is returned that may be passed as the first argument to proc
sl@0
    22
# 'testfixture' to execute a command. The child testfixture process is shut
sl@0
    23
# down by closing the channel.
sl@0
    24
proc launch_testfixture {} {
sl@0
    25
  set prg [info nameofexec]
sl@0
    26
  if {$prg eq ""} {
sl@0
    27
    set prg [file join . testfixture]
sl@0
    28
  }
sl@0
    29
  set chan [open "|$prg tf_main.tcl" r+]
sl@0
    30
  fconfigure $chan -buffering line
sl@0
    31
  return $chan
sl@0
    32
}
sl@0
    33
sl@0
    34
# Execute a command in a child testfixture process, connected by two-way
sl@0
    35
# channel $chan. Return the result of the command, or an error message.
sl@0
    36
proc testfixture {chan cmd} {
sl@0
    37
  puts $chan $cmd
sl@0
    38
  puts $chan OVER
sl@0
    39
  set r ""
sl@0
    40
  while { 1 } {
sl@0
    41
    set line [gets $chan]
sl@0
    42
    if { $line == "OVER" } { 
sl@0
    43
      return $r
sl@0
    44
    }
sl@0
    45
    append r $line
sl@0
    46
  }
sl@0
    47
}
sl@0
    48
sl@0
    49
# Write the main loop for the child testfixture processes into file
sl@0
    50
# tf_main.tcl. The parent (this script) interacts with the child processes
sl@0
    51
# via a two way pipe. The parent writes a script to the stdin of the child
sl@0
    52
# process, followed by the word "OVER" on a line of its own. The child
sl@0
    53
# process evaluates the script and writes the results to stdout, followed
sl@0
    54
# by an "OVER" of its own.
sl@0
    55
set f [open tf_main.tcl w]
sl@0
    56
puts $f {
sl@0
    57
  set l [open log w]
sl@0
    58
  set script ""
sl@0
    59
  while {![eof stdin]} {
sl@0
    60
    flush stdout
sl@0
    61
    set line [gets stdin]
sl@0
    62
    puts $l "READ $line"
sl@0
    63
    if { $line == "OVER" } {
sl@0
    64
      catch {eval $script} result
sl@0
    65
      puts $result
sl@0
    66
      puts $l "WRITE $result"
sl@0
    67
      puts OVER
sl@0
    68
      puts $l "WRITE OVER"
sl@0
    69
      flush stdout
sl@0
    70
      set script ""
sl@0
    71
    } else {
sl@0
    72
      append script $line
sl@0
    73
      append script " ; "
sl@0
    74
    }
sl@0
    75
  }
sl@0
    76
  close $l
sl@0
    77
}
sl@0
    78
close $f
sl@0
    79
sl@0
    80
# Simple locking test case:
sl@0
    81
#
sl@0
    82
# lock2-1.1: Connect a second process to the database.
sl@0
    83
# lock2-1.2: Establish a RESERVED lock with this process.
sl@0
    84
# lock2-1.3: Get a SHARED lock with the second process.
sl@0
    85
# lock2-1.4: Try for a RESERVED lock with process 2. This fails.
sl@0
    86
# lock2-1.5: Try to upgrade the first process to EXCLUSIVE, this fails so
sl@0
    87
#            it gets PENDING.
sl@0
    88
# lock2-1.6: Release the SHARED lock held by the second process. 
sl@0
    89
# lock2-1.7: Attempt to reaquire a SHARED lock with the second process.
sl@0
    90
#            this fails due to the PENDING lock.
sl@0
    91
# lock2-1.8: Ensure the first process can now upgrade to EXCLUSIVE.
sl@0
    92
#
sl@0
    93
do_test lock2-1.1 {
sl@0
    94
  set ::tf1 [launch_testfixture]
sl@0
    95
  testfixture $::tf1 "set sqlite_pending_byte $::sqlite_pending_byte"
sl@0
    96
  testfixture $::tf1 {
sl@0
    97
    sqlite3 db test.db -key xyzzy
sl@0
    98
    db eval {select * from sqlite_master}
sl@0
    99
  }
sl@0
   100
} {}
sl@0
   101
do_test lock2-1.1.1 {
sl@0
   102
  execsql {pragma lock_status}
sl@0
   103
} {main unlocked temp closed}
sl@0
   104
sqlite3_soft_heap_limit 0
sl@0
   105
do_test lock2-1.2 {
sl@0
   106
  execsql {
sl@0
   107
    BEGIN;
sl@0
   108
    CREATE TABLE abc(a, b, c);
sl@0
   109
  }
sl@0
   110
} {}
sl@0
   111
do_test lock2-1.3 {
sl@0
   112
  testfixture $::tf1 {
sl@0
   113
    db eval {
sl@0
   114
      BEGIN;
sl@0
   115
      SELECT * FROM sqlite_master;
sl@0
   116
    }
sl@0
   117
  }
sl@0
   118
} {}
sl@0
   119
do_test lock2-1.4 {
sl@0
   120
  testfixture $::tf1 {
sl@0
   121
    db eval {
sl@0
   122
      CREATE TABLE def(d, e, f)
sl@0
   123
    }
sl@0
   124
  }
sl@0
   125
} {database is locked}
sl@0
   126
do_test lock2-1.5 {
sl@0
   127
  catchsql {
sl@0
   128
    COMMIT;
sl@0
   129
  }
sl@0
   130
} {1 {database is locked}}
sl@0
   131
do_test lock2-1.6 {
sl@0
   132
  testfixture $::tf1 {
sl@0
   133
    db eval {
sl@0
   134
      SELECT * FROM sqlite_master;
sl@0
   135
      COMMIT;
sl@0
   136
    }
sl@0
   137
  }
sl@0
   138
} {}
sl@0
   139
do_test lock2-1.7 {
sl@0
   140
  testfixture $::tf1 {
sl@0
   141
    db eval {
sl@0
   142
      BEGIN;
sl@0
   143
      SELECT * FROM sqlite_master;
sl@0
   144
    }
sl@0
   145
  }
sl@0
   146
} {database is locked}
sl@0
   147
do_test lock2-1.8 {
sl@0
   148
  catchsql {
sl@0
   149
    COMMIT;
sl@0
   150
  }
sl@0
   151
} {0 {}}
sl@0
   152
do_test lock2-1.9 {
sl@0
   153
  execsql {
sl@0
   154
    SELECT * FROM sqlite_master;
sl@0
   155
  }
sl@0
   156
} "table abc abc [expr $AUTOVACUUM?3:2] {CREATE TABLE abc(a, b, c)}"
sl@0
   157
do_test lock2-1.10 {
sl@0
   158
  testfixture $::tf1 {
sl@0
   159
    db eval {
sl@0
   160
      SELECT * FROM sqlite_master;
sl@0
   161
    }
sl@0
   162
  }
sl@0
   163
} "table abc abc [expr $AUTOVACUUM?3:2] {CREATE TABLE abc(a, b, c)}"
sl@0
   164
sl@0
   165
catch {testfixture $::tf1 {db close}}
sl@0
   166
catch {close $::tf1}
sl@0
   167
sqlite3_soft_heap_limit $soft_limit
sl@0
   168
sl@0
   169
finish_test