os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/shared.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 December 30
sl@0
     2
#
sl@0
     3
# Portions Copyright (c) 2007-2008 Nokia Corporation and/or its subsidiaries. All rights reserved.
sl@0
     4
#
sl@0
     5
# The author disclaims copyright to this source code.  In place of
sl@0
     6
# a legal notice, here is a blessing:
sl@0
     7
#
sl@0
     8
#    May you do good and not evil.
sl@0
     9
#    May you find forgiveness for yourself and forgive others.
sl@0
    10
#    May you share freely, never taking more than you give.
sl@0
    11
#
sl@0
    12
#***********************************************************************
sl@0
    13
#
sl@0
    14
# $Id: shared.test,v 1.34 2008/07/12 14:52:20 drh Exp $
sl@0
    15
sl@0
    16
set testdir [file dirname $argv0]
sl@0
    17
source $testdir/tester.tcl
sl@0
    18
db close
sl@0
    19
sl@0
    20
# These tests cannot be run without the ATTACH command.
sl@0
    21
#
sl@0
    22
ifcapable !shared_cache||!attach {
sl@0
    23
  finish_test
sl@0
    24
  return
sl@0
    25
}
sl@0
    26
sl@0
    27
set ::enable_shared_cache [sqlite3_enable_shared_cache 1]
sl@0
    28
sl@0
    29
foreach av [list 0 1] {
sl@0
    30
sl@0
    31
# Open the database connection and execute the auto-vacuum pragma
sl@0
    32
file delete -force test.db
sl@0
    33
sqlite3 db test.db
sl@0
    34
sl@0
    35
ifcapable autovacuum {
sl@0
    36
  do_test shared-[expr $av+1].1.0 {
sl@0
    37
    execsql "pragma auto_vacuum=$::av"
sl@0
    38
    execsql {pragma auto_vacuum}
sl@0
    39
  } "$av"
sl@0
    40
} else {
sl@0
    41
  if {$av} {
sl@0
    42
    db close
sl@0
    43
    break
sl@0
    44
  }
sl@0
    45
}
sl@0
    46
sl@0
    47
# $av is currently 0 if this loop iteration is to test with auto-vacuum turned
sl@0
    48
# off, and 1 if it is turned on. Increment it so that (1 -> no auto-vacuum) 
sl@0
    49
# and (2 -> auto-vacuum). The sole reason for this is so that it looks nicer
sl@0
    50
# when we use this variable as part of test-case names.
sl@0
    51
#
sl@0
    52
incr av
sl@0
    53
sl@0
    54
# Test organization:
sl@0
    55
#
sl@0
    56
# shared-1.*: Simple test to verify basic sanity of table level locking when
sl@0
    57
#             two connections share a pager cache.
sl@0
    58
# shared-2.*: Test that a read transaction can co-exist with a 
sl@0
    59
#             write-transaction, including a simple test to ensure the 
sl@0
    60
#             external locking protocol is still working.
sl@0
    61
# shared-3.*: Simple test of read-uncommitted mode.
sl@0
    62
# shared-4.*: Check that the schema is locked and unlocked correctly.
sl@0
    63
# shared-5.*: Test that creating/dropping schema items works when databases
sl@0
    64
#             are attached in different orders to different handles.
sl@0
    65
# shared-6.*: Locking, UNION ALL queries and sub-queries.
sl@0
    66
# shared-7.*: Autovacuum and shared-cache.
sl@0
    67
# shared-8.*: Tests related to the text encoding of shared-cache databases.
sl@0
    68
# shared-9.*: TEMP triggers and shared-cache databases.
sl@0
    69
# shared-10.*: Tests of sqlite3_close().
sl@0
    70
# shared-11.*: Test transaction locking.
sl@0
    71
#
sl@0
    72
sl@0
    73
do_test shared-$av.1.1 {
sl@0
    74
  # Open a second database on the file test.db. It should use the same pager
sl@0
    75
  # cache and schema as the original connection. Verify that only 1 file is 
sl@0
    76
  # opened.
sl@0
    77
  sqlite3 db2 test.db
sl@0
    78
  set ::sqlite_open_file_count
sl@0
    79
} {1}
sl@0
    80
do_test shared-$av.1.2 {
sl@0
    81
  # Add a table and a single row of data via the first connection. 
sl@0
    82
  # Ensure that the second connection can see them.
sl@0
    83
  execsql {
sl@0
    84
    CREATE TABLE abc(a, b, c);
sl@0
    85
    INSERT INTO abc VALUES(1, 2, 3);
sl@0
    86
  } db
sl@0
    87
  execsql {
sl@0
    88
    SELECT * FROM abc;
sl@0
    89
  } db2
sl@0
    90
} {1 2 3}
sl@0
    91
do_test shared-$av.1.3 {
sl@0
    92
  # Have the first connection begin a transaction and obtain a read-lock
sl@0
    93
  # on table abc. This should not prevent the second connection from 
sl@0
    94
  # querying abc.
sl@0
    95
  execsql {
sl@0
    96
    BEGIN;
sl@0
    97
    SELECT * FROM abc;
sl@0
    98
  }
sl@0
    99
  execsql {
sl@0
   100
    SELECT * FROM abc;
sl@0
   101
  } db2
sl@0
   102
} {1 2 3}
sl@0
   103
do_test shared-$av.1.4 {
sl@0
   104
  # Try to insert a row into abc via connection 2. This should fail because
sl@0
   105
  # of the read-lock connection 1 is holding on table abc (obtained in the
sl@0
   106
  # previous test case).
sl@0
   107
  catchsql {
sl@0
   108
    INSERT INTO abc VALUES(4, 5, 6);
sl@0
   109
  } db2
sl@0
   110
} {1 {database table is locked: abc}}
sl@0
   111
do_test shared-$av.1.5 {
sl@0
   112
  # Using connection 2 (the one without the open transaction), try to create
sl@0
   113
  # a new table. This should fail because of the open read transaction 
sl@0
   114
  # held by connection 1.
sl@0
   115
  catchsql {
sl@0
   116
    CREATE TABLE def(d, e, f);
sl@0
   117
  } db2
sl@0
   118
} {1 {database table is locked: sqlite_master}}
sl@0
   119
do_test shared-$av.1.6 {
sl@0
   120
  # Upgrade connection 1's transaction to a write transaction. Create
sl@0
   121
  # a new table - def - and insert a row into it. Because the connection 1
sl@0
   122
  # transaction modifies the schema, it should not be possible for 
sl@0
   123
  # connection 2 to access the database at all until the connection 1 
sl@0
   124
  # has finished the transaction.
sl@0
   125
  execsql {
sl@0
   126
    CREATE TABLE def(d, e, f);
sl@0
   127
    INSERT INTO def VALUES('IV', 'V', 'VI');
sl@0
   128
  }
sl@0
   129
} {}
sl@0
   130
do_test shared-$av.1.7 {
sl@0
   131
  # Read from the sqlite_master table with connection 1 (inside the 
sl@0
   132
  # transaction). Then test that we can not do this with connection 2. This
sl@0
   133
  # is because of the schema-modified lock established by connection 1 
sl@0
   134
  # in the previous test case.
sl@0
   135
  execsql {
sl@0
   136
    SELECT * FROM sqlite_master;
sl@0
   137
  }
sl@0
   138
  catchsql {
sl@0
   139
    SELECT * FROM sqlite_master;
sl@0
   140
  } db2
sl@0
   141
} {1 {database schema is locked: main}}
sl@0
   142
do_test shared-$av.1.8 {
sl@0
   143
  # Commit the connection 1 transaction.
sl@0
   144
  execsql {
sl@0
   145
    COMMIT;
sl@0
   146
  }
sl@0
   147
} {}
sl@0
   148
sl@0
   149
do_test shared-$av.2.1 {
sl@0
   150
  # Open connection db3 to the database. Use a different path to the same
sl@0
   151
  # file so that db3 does *not* share the same pager cache as db and db2
sl@0
   152
  # (there should be two open file handles).
sl@0
   153
  if {$::tcl_platform(platform)=="unix"} {
sl@0
   154
    sqlite3 db3 ./test.db
sl@0
   155
  } else {
sl@0
   156
    sqlite3 db3 TEST.DB
sl@0
   157
  }
sl@0
   158
  set ::sqlite_open_file_count
sl@0
   159
} {2}
sl@0
   160
do_test shared-$av.2.2 {
sl@0
   161
  # Start read transactions on db and db2 (the shared pager cache). Ensure
sl@0
   162
  # db3 cannot write to the database.
sl@0
   163
  execsql {
sl@0
   164
    BEGIN;
sl@0
   165
    SELECT * FROM abc;
sl@0
   166
  }
sl@0
   167
  execsql {
sl@0
   168
    BEGIN;
sl@0
   169
    SELECT * FROM abc;
sl@0
   170
  } db2
sl@0
   171
  catchsql {
sl@0
   172
    INSERT INTO abc VALUES(1, 2, 3);
sl@0
   173
  } db2
sl@0
   174
} {1 {database table is locked: abc}}
sl@0
   175
do_test shared-$av.2.3 {
sl@0
   176
  # Turn db's transaction into a write-transaction. db3 should still be
sl@0
   177
  # able to read from table def (but will not see the new row). Connection
sl@0
   178
  # db2 should not be able to read def (because of the write-lock).
sl@0
   179
sl@0
   180
# Todo: The failed "INSERT INTO abc ..." statement in the above test
sl@0
   181
# has started a write-transaction on db2 (should this be so?). This 
sl@0
   182
# would prevent connection db from starting a write-transaction. So roll the
sl@0
   183
# db2 transaction back and replace it with a new read transaction.
sl@0
   184
  execsql {
sl@0
   185
    ROLLBACK;
sl@0
   186
    BEGIN;
sl@0
   187
    SELECT * FROM abc;
sl@0
   188
  } db2
sl@0
   189
sl@0
   190
  execsql {
sl@0
   191
    INSERT INTO def VALUES('VII', 'VIII', 'IX');
sl@0
   192
  }
sl@0
   193
  concat [
sl@0
   194
    catchsql { SELECT * FROM def; } db3
sl@0
   195
  ] [
sl@0
   196
    catchsql { SELECT * FROM def; } db2
sl@0
   197
  ]
sl@0
   198
} {0 {IV V VI} 1 {database table is locked: def}}
sl@0
   199
do_test shared-$av.2.4 {
sl@0
   200
  # Commit the open transaction on db. db2 still holds a read-transaction.
sl@0
   201
  # This should prevent db3 from writing to the database, but not from 
sl@0
   202
  # reading.
sl@0
   203
  execsql {
sl@0
   204
    COMMIT;
sl@0
   205
  }
sl@0
   206
  concat [
sl@0
   207
    catchsql { SELECT * FROM def; } db3
sl@0
   208
  ] [
sl@0
   209
    catchsql { INSERT INTO def VALUES('X', 'XI', 'XII'); } db3
sl@0
   210
  ]
sl@0
   211
} {0 {IV V VI VII VIII IX} 1 {database is locked}}
sl@0
   212
sl@0
   213
catchsql COMMIT db2
sl@0
   214
sl@0
   215
do_test shared-$av.3.1.1 {
sl@0
   216
  # This test case starts a linear scan of table 'seq' using a 
sl@0
   217
  # read-uncommitted connection. In the middle of the scan, rows are added
sl@0
   218
  # to the end of the seq table (ahead of the current cursor position).
sl@0
   219
  # The uncommitted rows should be included in the results of the scan.
sl@0
   220
  execsql "
sl@0
   221
    CREATE TABLE seq(i PRIMARY KEY, x);
sl@0
   222
    INSERT INTO seq VALUES(1, '[string repeat X 500]');
sl@0
   223
    INSERT INTO seq VALUES(2, '[string repeat X 500]');
sl@0
   224
  "
sl@0
   225
  execsql {SELECT * FROM sqlite_master} db2
sl@0
   226
  execsql {PRAGMA read_uncommitted = 1} db2
sl@0
   227
sl@0
   228
  set ret [list]
sl@0
   229
  db2 eval {SELECT i FROM seq ORDER BY i} {
sl@0
   230
    if {$i < 4} {
sl@0
   231
      set max [execsql {SELECT max(i) FROM seq}]
sl@0
   232
      db eval {
sl@0
   233
        INSERT INTO seq SELECT i + :max, x FROM seq;
sl@0
   234
      }
sl@0
   235
    }
sl@0
   236
    lappend ret $i
sl@0
   237
  }
sl@0
   238
  set ret
sl@0
   239
} {1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16}
sl@0
   240
do_test shared-$av.3.1.2 {
sl@0
   241
  # Another linear scan through table seq using a read-uncommitted connection.
sl@0
   242
  # This time, delete each row as it is read. Should not affect the results of
sl@0
   243
  # the scan, but the table should be empty after the scan is concluded 
sl@0
   244
  # (test 3.1.3 verifies this).
sl@0
   245
  set ret [list]
sl@0
   246
  db2 eval {SELECT i FROM seq} {
sl@0
   247
    db eval {DELETE FROM seq WHERE i = :i}
sl@0
   248
    lappend ret $i
sl@0
   249
  }
sl@0
   250
  set ret
sl@0
   251
} {1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16}
sl@0
   252
do_test shared-$av.3.1.3 {
sl@0
   253
  execsql {
sl@0
   254
    SELECT * FROM seq;
sl@0
   255
  }
sl@0
   256
} {}
sl@0
   257
sl@0
   258
catch {db close}
sl@0
   259
catch {db2 close}
sl@0
   260
catch {db3 close}
sl@0
   261
sl@0
   262
#--------------------------------------------------------------------------
sl@0
   263
# Tests shared-4.* test that the schema locking rules are applied 
sl@0
   264
# correctly. i.e.:
sl@0
   265
#
sl@0
   266
# 1. All transactions require a read-lock on the schemas of databases they
sl@0
   267
#    access.
sl@0
   268
# 2. Transactions that modify a database schema require a write-lock on that
sl@0
   269
#    schema.
sl@0
   270
# 3. It is not possible to compile a statement while another handle has a 
sl@0
   271
#    write-lock on the schema.
sl@0
   272
#
sl@0
   273
sl@0
   274
# Open two database handles db and db2. Each has a single attach database
sl@0
   275
# (as well as main):
sl@0
   276
#
sl@0
   277
#     db.main   ->   ./test.db
sl@0
   278
#     db.test2  ->   ./test2.db
sl@0
   279
#     db2.main  ->   ./test2.db
sl@0
   280
#     db2.test  ->   ./test.db
sl@0
   281
#
sl@0
   282
file delete -force test.db
sl@0
   283
file delete -force test2.db
sl@0
   284
file delete -force test2.db-journal
sl@0
   285
sqlite3 db  test.db
sl@0
   286
sqlite3 db2 test2.db
sl@0
   287
do_test shared-$av.4.1.1 {
sl@0
   288
  set sqlite_open_file_count
sl@0
   289
} {2}
sl@0
   290
do_test shared-$av.4.1.2 {
sl@0
   291
  execsql {ATTACH 'test2.db' AS test2}
sl@0
   292
  set sqlite_open_file_count
sl@0
   293
} {2}
sl@0
   294
do_test shared-$av.4.1.3 {
sl@0
   295
  execsql {ATTACH 'test.db' AS test} db2
sl@0
   296
  set sqlite_open_file_count
sl@0
   297
} {2}
sl@0
   298
sl@0
   299
# Sanity check: Create a table in ./test.db via handle db, and test that handle
sl@0
   300
# db2 can "see" the new table immediately. A handle using a seperate pager
sl@0
   301
# cache would have to reload the database schema before this were possible.
sl@0
   302
#
sl@0
   303
do_test shared-$av.4.2.1 {
sl@0
   304
  execsql {
sl@0
   305
    CREATE TABLE abc(a, b, c);
sl@0
   306
    CREATE TABLE def(d, e, f);
sl@0
   307
    INSERT INTO abc VALUES('i', 'ii', 'iii');
sl@0
   308
    INSERT INTO def VALUES('I', 'II', 'III');
sl@0
   309
  }
sl@0
   310
} {}
sl@0
   311
do_test shared-$av.4.2.2 {
sl@0
   312
  execsql {
sl@0
   313
    SELECT * FROM test.abc;
sl@0
   314
  } db2
sl@0
   315
} {i ii iii}
sl@0
   316
sl@0
   317
# Open a read-transaction and read from table abc via handle 2. Check that
sl@0
   318
# handle 1 can read table abc. Check that handle 1 cannot modify table abc
sl@0
   319
# or the database schema. Then check that handle 1 can modify table def.
sl@0
   320
#
sl@0
   321
do_test shared-$av.4.3.1 {
sl@0
   322
  execsql {
sl@0
   323
    BEGIN;
sl@0
   324
    SELECT * FROM test.abc;
sl@0
   325
  } db2
sl@0
   326
} {i ii iii}
sl@0
   327
do_test shared-$av.4.3.2 {
sl@0
   328
  catchsql {
sl@0
   329
    INSERT INTO abc VALUES('iv', 'v', 'vi');
sl@0
   330
  }
sl@0
   331
} {1 {database table is locked: abc}}
sl@0
   332
do_test shared-$av.4.3.3 {
sl@0
   333
  catchsql {
sl@0
   334
    CREATE TABLE ghi(g, h, i);
sl@0
   335
  }
sl@0
   336
} {1 {database table is locked: sqlite_master}}
sl@0
   337
do_test shared-$av.4.3.3 {
sl@0
   338
  catchsql {
sl@0
   339
    INSERT INTO def VALUES('IV', 'V', 'VI');
sl@0
   340
  }
sl@0
   341
} {0 {}}
sl@0
   342
do_test shared-$av.4.3.4 {
sl@0
   343
  # Cleanup: commit the transaction opened by db2.
sl@0
   344
  execsql {
sl@0
   345
    COMMIT
sl@0
   346
  } db2
sl@0
   347
} {}
sl@0
   348
sl@0
   349
# Open a write-transaction using handle 1 and modify the database schema.
sl@0
   350
# Then try to execute a compiled statement to read from the same 
sl@0
   351
# database via handle 2 (fails to get the lock on sqlite_master). Also
sl@0
   352
# try to compile a read of the same database using handle 2 (also fails).
sl@0
   353
# Finally, compile a read of the other database using handle 2. This
sl@0
   354
# should also fail.
sl@0
   355
#
sl@0
   356
ifcapable compound {
sl@0
   357
  do_test shared-$av.4.4.1.2 {
sl@0
   358
    # Sanity check 1: Check that the schema is what we think it is when viewed
sl@0
   359
    # via handle 1.
sl@0
   360
    execsql {
sl@0
   361
      CREATE TABLE test2.ghi(g, h, i);
sl@0
   362
      SELECT 'test.db:'||name FROM sqlite_master 
sl@0
   363
      UNION ALL
sl@0
   364
      SELECT 'test2.db:'||name FROM test2.sqlite_master;
sl@0
   365
    }
sl@0
   366
  } {test.db:abc test.db:def test2.db:ghi}
sl@0
   367
  do_test shared-$av.4.4.1.2 {
sl@0
   368
    # Sanity check 2: Check that the schema is what we think it is when viewed
sl@0
   369
    # via handle 2.
sl@0
   370
    execsql {
sl@0
   371
      SELECT 'test2.db:'||name FROM sqlite_master 
sl@0
   372
      UNION ALL
sl@0
   373
      SELECT 'test.db:'||name FROM test.sqlite_master;
sl@0
   374
    } db2
sl@0
   375
  } {test2.db:ghi test.db:abc test.db:def}
sl@0
   376
}
sl@0
   377
sl@0
   378
do_test shared-$av.4.4.2 {
sl@0
   379
  set ::DB2 [sqlite3_connection_pointer db2]
sl@0
   380
  set sql {SELECT * FROM abc}
sl@0
   381
  set ::STMT1 [sqlite3_prepare $::DB2 $sql -1 DUMMY]
sl@0
   382
  execsql {
sl@0
   383
    BEGIN;
sl@0
   384
    CREATE TABLE jkl(j, k, l);
sl@0
   385
  }
sl@0
   386
  sqlite3_step $::STMT1
sl@0
   387
} {SQLITE_ERROR}
sl@0
   388
do_test shared-$av.4.4.3 {
sl@0
   389
  sqlite3_finalize $::STMT1
sl@0
   390
} {SQLITE_LOCKED}
sl@0
   391
do_test shared-$av.4.4.4 {
sl@0
   392
  set rc [catch {
sl@0
   393
    set ::STMT1 [sqlite3_prepare $::DB2 $sql -1 DUMMY]
sl@0
   394
  } msg]
sl@0
   395
  list $rc $msg
sl@0
   396
} {1 {(6) database schema is locked: test}}
sl@0
   397
do_test shared-$av.4.4.5 {
sl@0
   398
  set rc [catch {
sl@0
   399
    set ::STMT1 [sqlite3_prepare $::DB2 "SELECT * FROM ghi" -1 DUMMY]
sl@0
   400
  } msg]
sl@0
   401
  list $rc $msg
sl@0
   402
} {1 {(6) database schema is locked: test}}
sl@0
   403
sl@0
   404
sl@0
   405
catch {db2 close}
sl@0
   406
catch {db close}
sl@0
   407
sl@0
   408
#--------------------------------------------------------------------------
sl@0
   409
# Tests shared-5.* 
sl@0
   410
#
sl@0
   411
foreach db [list test.db test1.db test2.db test3.db] {
sl@0
   412
  file delete -force $db ${db}-journal
sl@0
   413
}
sl@0
   414
do_test shared-$av.5.1.1 {
sl@0
   415
  sqlite3 db1 test.db
sl@0
   416
  sqlite3 db2 test.db
sl@0
   417
  execsql {
sl@0
   418
    ATTACH 'test1.db' AS test1;
sl@0
   419
    ATTACH 'test2.db' AS test2;
sl@0
   420
    ATTACH 'test3.db' AS test3;
sl@0
   421
  } db1
sl@0
   422
  execsql {
sl@0
   423
    ATTACH 'test3.db' AS test3;
sl@0
   424
    ATTACH 'test2.db' AS test2;
sl@0
   425
    ATTACH 'test1.db' AS test1;
sl@0
   426
  } db2
sl@0
   427
} {}
sl@0
   428
do_test shared-$av.5.1.2 {
sl@0
   429
  execsql {
sl@0
   430
    CREATE TABLE test1.t1(a, b);
sl@0
   431
    CREATE INDEX test1.i1 ON t1(a, b);
sl@0
   432
  } db1
sl@0
   433
} {}
sl@0
   434
ifcapable view {
sl@0
   435
  do_test shared-$av.5.1.3 {
sl@0
   436
    execsql {
sl@0
   437
      CREATE VIEW test1.v1 AS SELECT * FROM t1;
sl@0
   438
    } db1
sl@0
   439
  } {}
sl@0
   440
}
sl@0
   441
ifcapable trigger {
sl@0
   442
  do_test shared-$av.5.1.4 {
sl@0
   443
    execsql {
sl@0
   444
      CREATE TRIGGER test1.trig1 AFTER INSERT ON t1 BEGIN
sl@0
   445
        INSERT INTO t1 VALUES(new.a, new.b);
sl@0
   446
      END;
sl@0
   447
    } db1
sl@0
   448
  } {}
sl@0
   449
}
sl@0
   450
do_test shared-$av.5.1.5 {
sl@0
   451
  execsql {
sl@0
   452
    DROP INDEX i1;
sl@0
   453
  } db2
sl@0
   454
} {}
sl@0
   455
ifcapable view {
sl@0
   456
  do_test shared-$av.5.1.6 {
sl@0
   457
    execsql {
sl@0
   458
      DROP VIEW v1;
sl@0
   459
    } db2
sl@0
   460
  } {}
sl@0
   461
}
sl@0
   462
ifcapable trigger {
sl@0
   463
  do_test shared-$av.5.1.7 {
sl@0
   464
    execsql {
sl@0
   465
      DROP TRIGGER trig1;
sl@0
   466
    } db2
sl@0
   467
  } {}
sl@0
   468
}
sl@0
   469
do_test shared-$av.5.1.8 {
sl@0
   470
  execsql {
sl@0
   471
    DROP TABLE t1;
sl@0
   472
  } db2
sl@0
   473
} {}
sl@0
   474
ifcapable compound {
sl@0
   475
  do_test shared-$av.5.1.9 {
sl@0
   476
    execsql {
sl@0
   477
      SELECT * FROM sqlite_master UNION ALL SELECT * FROM test1.sqlite_master
sl@0
   478
    } db1
sl@0
   479
  } {}
sl@0
   480
}
sl@0
   481
sl@0
   482
#--------------------------------------------------------------------------
sl@0
   483
# Tests shared-6.* test that a query obtains all the read-locks it needs
sl@0
   484
# before starting execution of the query. This means that there is no chance
sl@0
   485
# some rows of data will be returned before a lock fails and SQLITE_LOCK
sl@0
   486
# is returned.
sl@0
   487
#
sl@0
   488
do_test shared-$av.6.1.1 {
sl@0
   489
  execsql {
sl@0
   490
    CREATE TABLE t1(a, b);
sl@0
   491
    CREATE TABLE t2(a, b);
sl@0
   492
    INSERT INTO t1 VALUES(1, 2);
sl@0
   493
    INSERT INTO t2 VALUES(3, 4);
sl@0
   494
  } db1
sl@0
   495
} {}
sl@0
   496
ifcapable compound {
sl@0
   497
  do_test shared-$av.6.1.2 {
sl@0
   498
    execsql {
sl@0
   499
      SELECT * FROM t1 UNION ALL SELECT * FROM t2;
sl@0
   500
    } db2
sl@0
   501
  } {1 2 3 4}
sl@0
   502
}
sl@0
   503
do_test shared-$av.6.1.3 {
sl@0
   504
  # Establish a write lock on table t2 via connection db2. Then make a 
sl@0
   505
  # UNION all query using connection db1 that first accesses t1, followed 
sl@0
   506
  # by t2. If the locks are grabbed at the start of the statement (as 
sl@0
   507
  # they should be), no rows are returned. If (as was previously the case)
sl@0
   508
  # they are grabbed as the tables are accessed, the t1 rows will be 
sl@0
   509
  # returned before the query fails.
sl@0
   510
  #
sl@0
   511
  execsql {
sl@0
   512
    BEGIN;
sl@0
   513
    INSERT INTO t2 VALUES(5, 6);
sl@0
   514
  } db2
sl@0
   515
  set ret [list]
sl@0
   516
  catch {
sl@0
   517
    db1 eval {SELECT * FROM t1 UNION ALL SELECT * FROM t2} {
sl@0
   518
      lappend ret $a $b
sl@0
   519
    }
sl@0
   520
  }
sl@0
   521
  set ret
sl@0
   522
} {}
sl@0
   523
do_test shared-$av.6.1.4 {
sl@0
   524
  execsql {
sl@0
   525
    COMMIT;
sl@0
   526
    BEGIN;
sl@0
   527
    INSERT INTO t1 VALUES(7, 8);
sl@0
   528
  } db2
sl@0
   529
  set ret [list]
sl@0
   530
  catch {
sl@0
   531
    db1 eval {
sl@0
   532
      SELECT (CASE WHEN a>4 THEN (SELECT a FROM t1) ELSE 0 END) AS d FROM t2;
sl@0
   533
    } {
sl@0
   534
      lappend ret $d
sl@0
   535
    }
sl@0
   536
  }
sl@0
   537
  set ret
sl@0
   538
} {}
sl@0
   539
sl@0
   540
catch {db1 close}
sl@0
   541
catch {db2 close}
sl@0
   542
foreach f [list test.db test2.db] {
sl@0
   543
  file delete -force $f ${f}-journal
sl@0
   544
}
sl@0
   545
sl@0
   546
#--------------------------------------------------------------------------
sl@0
   547
# Tests shared-7.* test auto-vacuum does not invalidate cursors from
sl@0
   548
# other shared-cache users when it reorganizes the database on 
sl@0
   549
# COMMIT.
sl@0
   550
#
sl@0
   551
do_test shared-$av.7.1 {
sl@0
   552
  # This test case sets up a test database in auto-vacuum mode consisting 
sl@0
   553
  # of two tables, t1 and t2. Both have a single index. Table t1 is 
sl@0
   554
  # populated first (so consists of pages toward the start of the db file), 
sl@0
   555
  # t2 second (pages toward the end of the file). 
sl@0
   556
  sqlite3 db test.db
sl@0
   557
  sqlite3 db2 test.db
sl@0
   558
  execsql {
sl@0
   559
    BEGIN;
sl@0
   560
    CREATE TABLE t1(a PRIMARY KEY, b);
sl@0
   561
    CREATE TABLE t2(a PRIMARY KEY, b);
sl@0
   562
  }
sl@0
   563
  set ::contents {}
sl@0
   564
  for {set i 0} {$i < 100} {incr i} {
sl@0
   565
    set a [string repeat "$i " 20]
sl@0
   566
    set b [string repeat "$i " 20]
sl@0
   567
    db eval {
sl@0
   568
      INSERT INTO t1 VALUES(:a, :b);
sl@0
   569
    }
sl@0
   570
    lappend ::contents [list [expr $i+1] $a $b]
sl@0
   571
  }
sl@0
   572
  execsql {
sl@0
   573
    INSERT INTO t2 SELECT * FROM t1;
sl@0
   574
    COMMIT;
sl@0
   575
  }
sl@0
   576
} {}
sl@0
   577
sl@0
   578
#
sl@0
   579
# Symbian: "stack overflow" if "shared-$av.7.2" is executed
sl@0
   580
#   
sl@0
   581
if {$tcl_platform(platform)!="symbian"} {
sl@0
   582
	do_test shared-$av.7.2 {
sl@0
   583
	  # This test case deletes the contents of table t1 (the one at the start of
sl@0
   584
	  # the file) while many cursors are open on table t2 and its index. All of
sl@0
   585
	  # the non-root pages will be moved from the end to the start of the file
sl@0
   586
	  # when the DELETE is committed - this test verifies that moving the pages
sl@0
   587
	  # does not disturb the open cursors.
sl@0
   588
	  #
sl@0
   589
sl@0
   590
	  proc lockrow {db tbl oids body} {
sl@0
   591
	    set ret [list]
sl@0
   592
	    db eval "SELECT oid AS i, a, b FROM $tbl ORDER BY a" {
sl@0
   593
	      if {$i==[lindex $oids 0]} {
sl@0
   594
	        set noids [lrange $oids 1 end]
sl@0
   595
	        if {[llength $noids]==0} {
sl@0
   596
	          set subret [eval $body]
sl@0
   597
	        } else {
sl@0
   598
	          set subret [lockrow $db $tbl $noids $body]
sl@0
   599
	        }
sl@0
   600
	      }
sl@0
   601
	      lappend ret [list $i $a $b]
sl@0
   602
	    }
sl@0
   603
	    return [linsert $subret 0 $ret]
sl@0
   604
	  }
sl@0
   605
	  proc locktblrows {db tbl body} {
sl@0
   606
	    set oids [db eval "SELECT oid FROM $tbl"]
sl@0
   607
	    lockrow $db $tbl $oids $body
sl@0
   608
	  }
sl@0
   609
sl@0
   610
	  set scans [locktblrows db t2 {
sl@0
   611
	    execsql {
sl@0
   612
	      DELETE FROM t1;
sl@0
   613
	    } db2
sl@0
   614
	  }]
sl@0
   615
	  set error 0
sl@0
   616
sl@0
   617
	  # Test that each SELECT query returned the expected contents of t2.
sl@0
   618
	  foreach s $scans {
sl@0
   619
	    if {[lsort -integer -index 0 $s]!=$::contents} {
sl@0
   620
	      set error 1
sl@0
   621
	    }
sl@0
   622
	  }
sl@0
   623
	  set error
sl@0
   624
	} {0}
sl@0
   625
}
sl@0
   626
sl@0
   627
catch {db close}
sl@0
   628
catch {db2 close}
sl@0
   629
unset -nocomplain contents
sl@0
   630
sl@0
   631
#--------------------------------------------------------------------------
sl@0
   632
# The following tests try to trick the shared-cache code into assuming
sl@0
   633
# the wrong encoding for a database.
sl@0
   634
#
sl@0
   635
file delete -force test.db test.db-journal
sl@0
   636
ifcapable utf16 {
sl@0
   637
  do_test shared-$av.8.1.1 {
sl@0
   638
    sqlite3 db test.db
sl@0
   639
    execsql {
sl@0
   640
      PRAGMA encoding = 'UTF-16';
sl@0
   641
      SELECT * FROM sqlite_master;
sl@0
   642
    }
sl@0
   643
  } {}
sl@0
   644
  do_test shared-$av.8.1.2 {
sl@0
   645
    string range [execsql {PRAGMA encoding;}] 0 end-2
sl@0
   646
  } {UTF-16}
sl@0
   647
sl@0
   648
  do_test shared-$av.8.1.3 {
sl@0
   649
    sqlite3 db2 test.db
sl@0
   650
    execsql {
sl@0
   651
      PRAGMA encoding = 'UTF-8';
sl@0
   652
      CREATE TABLE abc(a, b, c);
sl@0
   653
    } db2
sl@0
   654
  } {}
sl@0
   655
  do_test shared-$av.8.1.4 {
sl@0
   656
    execsql {
sl@0
   657
      SELECT * FROM sqlite_master;
sl@0
   658
    }
sl@0
   659
  } "table abc abc [expr $AUTOVACUUM?3:2] {CREATE TABLE abc(a, b, c)}"
sl@0
   660
  do_test shared-$av.8.1.5 {
sl@0
   661
    db2 close
sl@0
   662
    execsql {
sl@0
   663
      PRAGMA encoding;
sl@0
   664
    }
sl@0
   665
  } {UTF-8}
sl@0
   666
sl@0
   667
  file delete -force test2.db test2.db-journal
sl@0
   668
  do_test shared-$av.8.2.1 {
sl@0
   669
    execsql {
sl@0
   670
      ATTACH 'test2.db' AS aux;
sl@0
   671
      SELECT * FROM aux.sqlite_master;
sl@0
   672
    }
sl@0
   673
  } {}
sl@0
   674
  do_test shared-$av.8.2.2 {
sl@0
   675
    sqlite3 db2 test2.db
sl@0
   676
    execsql {
sl@0
   677
      PRAGMA encoding = 'UTF-16';
sl@0
   678
      CREATE TABLE def(d, e, f);
sl@0
   679
    } db2
sl@0
   680
    string range [execsql {PRAGMA encoding;} db2] 0 end-2
sl@0
   681
  } {UTF-16}
sl@0
   682
sl@0
   683
  catch {db close}
sl@0
   684
  catch {db2 close}
sl@0
   685
  file delete -force test.db test2.db
sl@0
   686
sl@0
   687
  do_test shared-$av.8.3.2 {
sl@0
   688
    sqlite3 db test.db
sl@0
   689
    execsql { CREATE TABLE def(d, e, f) }
sl@0
   690
    execsql { PRAGMA encoding }
sl@0
   691
  } {UTF-8}
sl@0
   692
  do_test shared-$av.8.3.3 {
sl@0
   693
    set zDb16 "[encoding convertto unicode test.db]\x00\x00"
sl@0
   694
    set db16 [sqlite3_open16 $zDb16 {}]
sl@0
   695
sl@0
   696
    set stmt [sqlite3_prepare $db16 "SELECT sql FROM sqlite_master" -1 DUMMY]
sl@0
   697
    sqlite3_step $stmt
sl@0
   698
    set sql [sqlite3_column_text $stmt 0]
sl@0
   699
    sqlite3_finalize $stmt
sl@0
   700
    set sql
sl@0
   701
  } {CREATE TABLE def(d, e, f)}
sl@0
   702
  do_test shared-$av.8.3.4 {
sl@0
   703
    set stmt [sqlite3_prepare $db16 "PRAGMA encoding" -1 DUMMY]
sl@0
   704
    sqlite3_step $stmt
sl@0
   705
    set enc [sqlite3_column_text $stmt 0]
sl@0
   706
    sqlite3_finalize $stmt
sl@0
   707
    set enc
sl@0
   708
  } {UTF-8}
sl@0
   709
sl@0
   710
  sqlite3_close $db16
sl@0
   711
sl@0
   712
# Bug #2547 is causing this to fail.
sl@0
   713
if 0 {
sl@0
   714
  do_test shared-$av.8.2.3 {
sl@0
   715
    catchsql {
sl@0
   716
      SELECT * FROM aux.sqlite_master;
sl@0
   717
    }
sl@0
   718
  } {1 {attached databases must use the same text encoding as main database}}
sl@0
   719
}
sl@0
   720
}
sl@0
   721
sl@0
   722
catch {db close}
sl@0
   723
catch {db2 close}
sl@0
   724
file delete -force test.db test2.db
sl@0
   725
sl@0
   726
#---------------------------------------------------------------------------
sl@0
   727
# The following tests - shared-9.* - test interactions between TEMP triggers
sl@0
   728
# and shared-schemas.
sl@0
   729
#
sl@0
   730
ifcapable trigger&&tempdb {
sl@0
   731
sl@0
   732
do_test shared-$av.9.1 {
sl@0
   733
  sqlite3 db test.db
sl@0
   734
  sqlite3 db2 test.db
sl@0
   735
  execsql {
sl@0
   736
    CREATE TABLE abc(a, b, c);
sl@0
   737
    CREATE TABLE abc_mirror(a, b, c);
sl@0
   738
    CREATE TEMP TRIGGER BEFORE INSERT ON abc BEGIN 
sl@0
   739
      INSERT INTO abc_mirror(a, b, c) VALUES(new.a, new.b, new.c);
sl@0
   740
    END;
sl@0
   741
    INSERT INTO abc VALUES(1, 2, 3);
sl@0
   742
    SELECT * FROM abc_mirror;
sl@0
   743
  }
sl@0
   744
} {1 2 3}
sl@0
   745
do_test shared-$av.9.2 {
sl@0
   746
  execsql {
sl@0
   747
    INSERT INTO abc VALUES(4, 5, 6);
sl@0
   748
    SELECT * FROM abc_mirror;
sl@0
   749
  } db2
sl@0
   750
} {1 2 3}
sl@0
   751
do_test shared-$av.9.3 {
sl@0
   752
  db close
sl@0
   753
  db2 close
sl@0
   754
} {}
sl@0
   755
sl@0
   756
} ; # End shared-9.*
sl@0
   757
sl@0
   758
#---------------------------------------------------------------------------
sl@0
   759
# The following tests - shared-10.* - test that the library behaves 
sl@0
   760
# correctly when a connection to a shared-cache is closed. 
sl@0
   761
#
sl@0
   762
do_test shared-$av.10.1 {
sl@0
   763
  # Create a small sample database with two connections to it (db and db2).
sl@0
   764
  file delete -force test.db
sl@0
   765
  sqlite3 db  test.db
sl@0
   766
  sqlite3 db2 test.db
sl@0
   767
  execsql {
sl@0
   768
    CREATE TABLE ab(a PRIMARY KEY, b);
sl@0
   769
    CREATE TABLE de(d PRIMARY KEY, e);
sl@0
   770
    INSERT INTO ab VALUES('Chiang Mai', 100000);
sl@0
   771
    INSERT INTO ab VALUES('Bangkok', 8000000);
sl@0
   772
    INSERT INTO de VALUES('Ubon', 120000);
sl@0
   773
    INSERT INTO de VALUES('Khon Kaen', 200000);
sl@0
   774
  }
sl@0
   775
} {}
sl@0
   776
do_test shared-$av.10.2 {
sl@0
   777
  # Open a read-transaction with the first connection, a write-transaction
sl@0
   778
  # with the second.
sl@0
   779
  execsql {
sl@0
   780
    BEGIN;
sl@0
   781
    SELECT * FROM ab;
sl@0
   782
  }
sl@0
   783
  execsql {
sl@0
   784
    BEGIN;
sl@0
   785
    INSERT INTO de VALUES('Pataya', 30000);
sl@0
   786
  } db2
sl@0
   787
} {}
sl@0
   788
do_test shared-$av.10.3 {
sl@0
   789
  # An external connection should be able to read the database, but not
sl@0
   790
  # prepare a write operation.
sl@0
   791
  if {$::tcl_platform(platform)=="unix"} {
sl@0
   792
    sqlite3 db3 ./test.db
sl@0
   793
  } else {
sl@0
   794
    sqlite3 db3 TEST.DB
sl@0
   795
  }
sl@0
   796
  execsql {
sl@0
   797
    SELECT * FROM ab;
sl@0
   798
  } db3
sl@0
   799
  catchsql {
sl@0
   800
    BEGIN;
sl@0
   801
    INSERT INTO de VALUES('Pataya', 30000);
sl@0
   802
  } db3
sl@0
   803
} {1 {database is locked}}
sl@0
   804
do_test shared-$av.10.4 {
sl@0
   805
  # Close the connection with the write-transaction open
sl@0
   806
  db2 close
sl@0
   807
} {}
sl@0
   808
do_test shared-$av.10.5 {
sl@0
   809
  # Test that the db2 transaction has been automatically rolled back.
sl@0
   810
  # If it has not the ('Pataya', 30000) entry will still be in the table.
sl@0
   811
  execsql {
sl@0
   812
    SELECT * FROM de;
sl@0
   813
  }
sl@0
   814
} {Ubon 120000 {Khon Kaen} 200000}
sl@0
   815
do_test shared-$av.10.5 {
sl@0
   816
  # Closing db2 should have dropped the shared-cache back to a read-lock.
sl@0
   817
  # So db3 should be able to prepare a write...
sl@0
   818
  catchsql {INSERT INTO de VALUES('Pataya', 30000);} db3
sl@0
   819
} {0 {}}
sl@0
   820
do_test shared-$av.10.6 {
sl@0
   821
  # ... but not commit it.
sl@0
   822
  catchsql {COMMIT} db3
sl@0
   823
} {1 {database is locked}}
sl@0
   824
do_test shared-$av.10.7 {
sl@0
   825
  # Commit the (read-only) db transaction. Check via db3 to make sure the 
sl@0
   826
  # contents of table "de" are still as they should be.
sl@0
   827
  execsql {
sl@0
   828
    COMMIT;
sl@0
   829
  }
sl@0
   830
  execsql {
sl@0
   831
    SELECT * FROM de;
sl@0
   832
  } db3
sl@0
   833
} {Ubon 120000 {Khon Kaen} 200000 Pataya 30000}
sl@0
   834
do_test shared-$av.10.9 {
sl@0
   835
  # Commit the external transaction.
sl@0
   836
  catchsql {COMMIT} db3
sl@0
   837
} {0 {}}
sl@0
   838
integrity_check shared-$av.10.10
sl@0
   839
do_test shared-$av.10.11 {
sl@0
   840
  db close
sl@0
   841
  db3 close
sl@0
   842
} {}
sl@0
   843
sl@0
   844
do_test shared-$av.11.1 {
sl@0
   845
  file delete -force test.db
sl@0
   846
  sqlite3 db  test.db
sl@0
   847
  sqlite3 db2 test.db
sl@0
   848
  execsql {
sl@0
   849
    CREATE TABLE abc(a, b, c);
sl@0
   850
    CREATE TABLE abc2(a, b, c);
sl@0
   851
    BEGIN;
sl@0
   852
    INSERT INTO abc VALUES(1, 2, 3);
sl@0
   853
  }
sl@0
   854
} {}
sl@0
   855
do_test shared-$av.11.2 {
sl@0
   856
  catchsql {BEGIN;} db2
sl@0
   857
  catchsql {SELECT * FROM abc;} db2
sl@0
   858
} {1 {database table is locked: abc}}
sl@0
   859
do_test shared-$av.11.3 {
sl@0
   860
  catchsql {BEGIN} db2
sl@0
   861
} {1 {cannot start a transaction within a transaction}}
sl@0
   862
do_test shared-$av.11.4 {
sl@0
   863
  catchsql {SELECT * FROM abc2;} db2
sl@0
   864
} {0 {}}
sl@0
   865
do_test shared-$av.11.5 {
sl@0
   866
  catchsql {INSERT INTO abc2 VALUES(1, 2, 3);} db2
sl@0
   867
} {1 {database is locked}}
sl@0
   868
do_test shared-$av.11.6 {
sl@0
   869
  catchsql {SELECT * FROM abc2}
sl@0
   870
} {0 {}}
sl@0
   871
do_test shared-$av.11.6 {
sl@0
   872
  execsql {
sl@0
   873
    ROLLBACK;
sl@0
   874
    PRAGMA read_uncommitted = 1;
sl@0
   875
  } db2
sl@0
   876
} {}
sl@0
   877
do_test shared-$av.11.7 {
sl@0
   878
  execsql {
sl@0
   879
    INSERT INTO abc2 VALUES(4, 5, 6);
sl@0
   880
    INSERT INTO abc2 VALUES(7, 8, 9);
sl@0
   881
  }
sl@0
   882
} {}
sl@0
   883
do_test shared-$av.11.8 {
sl@0
   884
  set res [list]
sl@0
   885
  db2 eval {
sl@0
   886
    SELECT abc.a as I, abc2.a as II FROM abc, abc2;
sl@0
   887
  } {
sl@0
   888
    execsql {
sl@0
   889
      DELETE FROM abc WHERE 1;
sl@0
   890
    }
sl@0
   891
    lappend res $I $II
sl@0
   892
  }
sl@0
   893
  set res
sl@0
   894
} {1 4 {} 7}
sl@0
   895
if {[llength [info command sqlite3_shared_cache_report]]==1} {
sl@0
   896
  do_test shared-$av.11.9 {
sl@0
   897
    string tolower [sqlite3_shared_cache_report]
sl@0
   898
  } [string tolower [list [file nativename [file normalize test.db]] 2]]
sl@0
   899
}
sl@0
   900
sl@0
   901
do_test shared-$av.11.11 {
sl@0
   902
  db close
sl@0
   903
  db2 close
sl@0
   904
} {}
sl@0
   905
sl@0
   906
# This tests that if it is impossible to free any pages, SQLite will
sl@0
   907
# exceed the limit set by PRAGMA cache_size.
sl@0
   908
file delete -force test.db test.db-journal
sl@0
   909
sqlite3 db test.db 
sl@0
   910
ifcapable pager_pragmas {
sl@0
   911
  do_test shared-$av.12.1 {
sl@0
   912
    execsql {
sl@0
   913
      PRAGMA cache_size = 10;
sl@0
   914
      PRAGMA cache_size;
sl@0
   915
    }
sl@0
   916
  } {10}
sl@0
   917
}
sl@0
   918
do_test shared-$av.12.2 {
sl@0
   919
  set ::db_handles [list]
sl@0
   920
  for {set i 1} {$i < 15} {incr i} {
sl@0
   921
    lappend ::db_handles db$i
sl@0
   922
    sqlite3 db$i test.db 
sl@0
   923
    execsql "CREATE TABLE db${i}(a, b, c)" db$i 
sl@0
   924
    execsql "INSERT INTO db${i} VALUES(1, 2, 3)"
sl@0
   925
  }
sl@0
   926
} {}
sl@0
   927
proc nested_select {handles} {
sl@0
   928
  [lindex $handles 0] eval "SELECT * FROM [lindex $handles 0]" {
sl@0
   929
    lappend ::res $a $b $c
sl@0
   930
    if {[llength $handles]>1} {
sl@0
   931
      nested_select [lrange $handles 1 end]
sl@0
   932
    }
sl@0
   933
  }
sl@0
   934
}
sl@0
   935
do_test shared-$av.12.3 {
sl@0
   936
  set ::res [list]
sl@0
   937
  nested_select $::db_handles
sl@0
   938
  set ::res
sl@0
   939
} [string range [string repeat "1 2 3 " [llength $::db_handles]] 0 end-1]
sl@0
   940
sl@0
   941
do_test shared-$av.12.X {
sl@0
   942
  db close
sl@0
   943
  foreach h $::db_handles { 
sl@0
   944
    $h close
sl@0
   945
  }
sl@0
   946
} {}
sl@0
   947
sl@0
   948
# Internally, locks are acquired on shared B-Tree structures in the order
sl@0
   949
# that the structures appear in the virtual memory address space. This
sl@0
   950
# test case attempts to cause the order of the structures in memory 
sl@0
   951
# to be different from the order in which they are attached to a given
sl@0
   952
# database handle. This covers an extra line or two.
sl@0
   953
#
sl@0
   954
do_test shared-$av.13.1 {
sl@0
   955
  file delete -force test2.db test3.db test4.db test5.db
sl@0
   956
  sqlite3 db :memory:
sl@0
   957
  execsql {
sl@0
   958
    ATTACH 'test2.db' AS aux2;
sl@0
   959
    ATTACH 'test3.db' AS aux3;
sl@0
   960
    ATTACH 'test4.db' AS aux4;
sl@0
   961
    ATTACH 'test5.db' AS aux5;
sl@0
   962
    DETACH aux2;
sl@0
   963
    DETACH aux3;
sl@0
   964
    DETACH aux4;
sl@0
   965
    ATTACH 'test2.db' AS aux2;
sl@0
   966
    ATTACH 'test3.db' AS aux3;
sl@0
   967
    ATTACH 'test4.db' AS aux4;
sl@0
   968
  }
sl@0
   969
} {}
sl@0
   970
do_test shared-$av.13.2 {
sl@0
   971
  execsql {
sl@0
   972
    CREATE TABLE t1(a, b, c);
sl@0
   973
    CREATE TABLE aux2.t2(a, b, c);
sl@0
   974
    CREATE TABLE aux3.t3(a, b, c);
sl@0
   975
    CREATE TABLE aux4.t4(a, b, c);
sl@0
   976
    CREATE TABLE aux5.t5(a, b, c);
sl@0
   977
    SELECT count(*) FROM 
sl@0
   978
      aux2.sqlite_master, 
sl@0
   979
      aux3.sqlite_master, 
sl@0
   980
      aux4.sqlite_master, 
sl@0
   981
      aux5.sqlite_master
sl@0
   982
  }
sl@0
   983
} {1}
sl@0
   984
do_test shared-$av.13.3 {
sl@0
   985
  db close
sl@0
   986
} {}
sl@0
   987
sl@0
   988
# Test that nothing horrible happens if a connection to a shared B-Tree 
sl@0
   989
# structure is closed while some other connection has an open cursor.
sl@0
   990
#
sl@0
   991
do_test shared-$av.14.1 {
sl@0
   992
  sqlite3 db test.db
sl@0
   993
  sqlite3 db2 test.db
sl@0
   994
  execsql {SELECT name FROM sqlite_master}
sl@0
   995
} {db1 db2 db3 db4 db5 db6 db7 db8 db9 db10 db11 db12 db13 db14}
sl@0
   996
do_test shared-$av.14.2 {
sl@0
   997
  set res [list]
sl@0
   998
  db eval {SELECT name FROM sqlite_master} {
sl@0
   999
    if {$name eq "db7"} {
sl@0
  1000
      db2 close
sl@0
  1001
    }
sl@0
  1002
    lappend res $name
sl@0
  1003
  }
sl@0
  1004
  set res
sl@0
  1005
} {db1 db2 db3 db4 db5 db6 db7 db8 db9 db10 db11 db12 db13 db14}
sl@0
  1006
do_test shared-$av.14.3 {
sl@0
  1007
  db close
sl@0
  1008
} {}
sl@0
  1009
sl@0
  1010
}
sl@0
  1011
sl@0
  1012
sqlite3_enable_shared_cache $::enable_shared_cache
sl@0
  1013
finish_test