os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/attach2.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
# 2003 July 1
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 testing the ATTACH and DETACH commands
sl@0
    13
# and related functionality.
sl@0
    14
#
sl@0
    15
# $Id: attach2.test,v 1.38 2007/12/13 21:54:11 drh Exp $
sl@0
    16
#
sl@0
    17
sl@0
    18
set testdir [file dirname $argv0]
sl@0
    19
source $testdir/tester.tcl
sl@0
    20
sl@0
    21
ifcapable !attach {
sl@0
    22
  finish_test
sl@0
    23
  return
sl@0
    24
}
sl@0
    25
sl@0
    26
# Ticket #354
sl@0
    27
#
sl@0
    28
# Databases test.db and test2.db contain identical schemas.  Make
sl@0
    29
# sure we can attach test2.db from test.db.
sl@0
    30
#
sl@0
    31
do_test attach2-1.1 {
sl@0
    32
  db eval {
sl@0
    33
    CREATE TABLE t1(a,b);
sl@0
    34
    CREATE INDEX x1 ON t1(a);
sl@0
    35
  }
sl@0
    36
  file delete -force test2.db
sl@0
    37
  file delete -force test2.db-journal
sl@0
    38
  sqlite3 db2 test2.db
sl@0
    39
  db2 eval {
sl@0
    40
    CREATE TABLE t1(a,b);
sl@0
    41
    CREATE INDEX x1 ON t1(a);
sl@0
    42
  }
sl@0
    43
  catchsql {
sl@0
    44
    ATTACH 'test2.db' AS t2;
sl@0
    45
  }
sl@0
    46
} {0 {}}
sl@0
    47
sl@0
    48
# Ticket #514
sl@0
    49
#
sl@0
    50
proc db_list {db} {
sl@0
    51
  set list {}
sl@0
    52
  foreach {idx name file} [execsql {PRAGMA database_list} $db] {
sl@0
    53
    lappend list $idx $name
sl@0
    54
  }
sl@0
    55
  return $list
sl@0
    56
}
sl@0
    57
db eval {DETACH t2}
sl@0
    58
do_test attach2-2.1 {
sl@0
    59
  # lock test2.db then try to attach it.  This is no longer an error because
sl@0
    60
  # db2 just RESERVES the database.  It does not obtain a write-lock until
sl@0
    61
  # we COMMIT.
sl@0
    62
  db2 eval {BEGIN}
sl@0
    63
  db2 eval {UPDATE t1 SET a = 0 WHERE 0}
sl@0
    64
  catchsql {
sl@0
    65
    ATTACH 'test2.db' AS t2;
sl@0
    66
  }
sl@0
    67
} {0 {}}
sl@0
    68
ifcapable schema_pragmas {
sl@0
    69
do_test attach2-2.2 {
sl@0
    70
  # make sure test2.db did get attached.
sl@0
    71
  db_list db
sl@0
    72
} {0 main 2 t2}
sl@0
    73
} ;# ifcapable schema_pragmas
sl@0
    74
db2 eval {COMMIT}
sl@0
    75
sl@0
    76
do_test attach2-2.5 {
sl@0
    77
  # Make sure we can read test2.db from db
sl@0
    78
  catchsql {
sl@0
    79
    SELECT name FROM t2.sqlite_master;
sl@0
    80
  }
sl@0
    81
} {0 {t1 x1}}
sl@0
    82
do_test attach2-2.6 {
sl@0
    83
  # lock test2.db and try to read from it.  This should still work because
sl@0
    84
  # the lock is only a RESERVED lock which does not prevent reading.
sl@0
    85
  #
sl@0
    86
  db2 eval BEGIN
sl@0
    87
  db2 eval {UPDATE t1 SET a = 0 WHERE 0}
sl@0
    88
  catchsql {
sl@0
    89
    SELECT name FROM t2.sqlite_master;
sl@0
    90
  }
sl@0
    91
} {0 {t1 x1}}
sl@0
    92
do_test attach2-2.7 {
sl@0
    93
  # but we can still read from test1.db even though test2.db is locked.
sl@0
    94
  catchsql {
sl@0
    95
    SELECT name FROM main.sqlite_master;
sl@0
    96
  }
sl@0
    97
} {0 {t1 x1}}
sl@0
    98
do_test attach2-2.8 {
sl@0
    99
  # start a transaction on test.db even though test2.db is locked.
sl@0
   100
  catchsql {
sl@0
   101
    BEGIN;
sl@0
   102
    INSERT INTO t1 VALUES(8,9);
sl@0
   103
  }
sl@0
   104
} {0 {}}
sl@0
   105
do_test attach2-2.9 {
sl@0
   106
  execsql {
sl@0
   107
    SELECT * FROM t1
sl@0
   108
  }
sl@0
   109
} {8 9}
sl@0
   110
do_test attach2-2.10 {
sl@0
   111
  # now try to write to test2.db.  the write should fail
sl@0
   112
  catchsql {
sl@0
   113
    INSERT INTO t2.t1 VALUES(1,2);
sl@0
   114
  }
sl@0
   115
} {1 {database is locked}}
sl@0
   116
do_test attach2-2.11 {
sl@0
   117
  # when the write failed in the previous test, the transaction should
sl@0
   118
  # have rolled back.
sl@0
   119
  # 
sl@0
   120
  # Update for version 3: A transaction is no longer rolled back if a
sl@0
   121
  #                       database is found to be busy.
sl@0
   122
  execsql {rollback}
sl@0
   123
  db2 eval ROLLBACK
sl@0
   124
  execsql {
sl@0
   125
    SELECT * FROM t1
sl@0
   126
  }
sl@0
   127
} {}
sl@0
   128
do_test attach2-2.12 {
sl@0
   129
  catchsql {
sl@0
   130
    COMMIT
sl@0
   131
  }
sl@0
   132
} {1 {cannot commit - no transaction is active}}
sl@0
   133
sl@0
   134
# Ticket #574:  Make sure it works using the non-callback API
sl@0
   135
#
sl@0
   136
do_test attach2-3.1 {
sl@0
   137
  set DB [sqlite3_connection_pointer db]
sl@0
   138
  set rc [catch {sqlite3_prepare $DB "ATTACH 'test2.db' AS t2" -1 TAIL} VM]
sl@0
   139
  if {$rc} {lappend rc $VM}
sl@0
   140
  sqlite3_step $VM
sl@0
   141
  sqlite3_finalize $VM
sl@0
   142
  set rc
sl@0
   143
} {0}
sl@0
   144
do_test attach2-3.2 {
sl@0
   145
  set rc [catch {sqlite3_prepare $DB "DETACH t2" -1 TAIL} VM]
sl@0
   146
  if {$rc} {lappend rc $VM}
sl@0
   147
  sqlite3_step $VM
sl@0
   148
  sqlite3_finalize $VM
sl@0
   149
  set rc
sl@0
   150
} {0}
sl@0
   151
sl@0
   152
db close
sl@0
   153
for {set i 2} {$i<=15} {incr i} {
sl@0
   154
  catch {db$i close}
sl@0
   155
}
sl@0
   156
sl@0
   157
# A procedure to verify the status of locks on a database.
sl@0
   158
#
sl@0
   159
proc lock_status {testnum db expected_result} {
sl@0
   160
  # If the database was compiled with OMIT_TEMPDB set, then 
sl@0
   161
  # the lock_status list will not contain an entry for the temp
sl@0
   162
  # db. But the test code doesn't know this, so its easiest 
sl@0
   163
  # to filter it out of the $expected_result list here.
sl@0
   164
  ifcapable !tempdb {
sl@0
   165
    set expected_result [concat \
sl@0
   166
        [lrange $expected_result 0 1] \
sl@0
   167
        [lrange $expected_result 4 end] \
sl@0
   168
    ]
sl@0
   169
  }
sl@0
   170
  do_test attach2-$testnum [subst {
sl@0
   171
    $db cache flush  ;# The lock_status pragma should not be cached
sl@0
   172
    execsql {PRAGMA lock_status} $db
sl@0
   173
  }] $expected_result
sl@0
   174
}
sl@0
   175
set sqlite_os_trace 0
sl@0
   176
sl@0
   177
# Tests attach2-4.* test that read-locks work correctly with attached
sl@0
   178
# databases.
sl@0
   179
do_test attach2-4.1 {
sl@0
   180
  sqlite3 db test.db
sl@0
   181
  sqlite3 db2 test.db
sl@0
   182
  execsql {ATTACH 'test2.db' as file2}
sl@0
   183
  execsql {ATTACH 'test2.db' as file2} db2
sl@0
   184
} {}
sl@0
   185
sl@0
   186
lock_status 4.1.1 db {main unlocked temp closed file2 unlocked}
sl@0
   187
lock_status 4.1.2 db2 {main unlocked temp closed file2 unlocked}
sl@0
   188
sl@0
   189
do_test attach2-4.2 {
sl@0
   190
  # Handle 'db' read-locks test.db
sl@0
   191
  execsql {BEGIN}
sl@0
   192
  execsql {SELECT * FROM t1}
sl@0
   193
  # Lock status:
sl@0
   194
  #    db  - shared(main)
sl@0
   195
  #    db2 -
sl@0
   196
} {}
sl@0
   197
sl@0
   198
lock_status 4.2.1 db {main shared temp closed file2 unlocked}
sl@0
   199
lock_status 4.2.2 db2 {main unlocked temp closed file2 unlocked}
sl@0
   200
sl@0
   201
do_test attach2-4.3 {
sl@0
   202
  # The read lock held by db does not prevent db2 from reading test.db
sl@0
   203
  execsql {SELECT * FROM t1} db2
sl@0
   204
} {}
sl@0
   205
sl@0
   206
lock_status 4.3.1 db {main shared temp closed file2 unlocked}
sl@0
   207
lock_status 4.3.2 db2 {main unlocked temp closed file2 unlocked}
sl@0
   208
sl@0
   209
do_test attach2-4.4 {
sl@0
   210
  # db is holding a read lock on test.db, so we should not be able
sl@0
   211
  # to commit a write to test.db from db2
sl@0
   212
  catchsql {
sl@0
   213
    INSERT INTO t1 VALUES(1, 2)
sl@0
   214
  } db2 
sl@0
   215
} {1 {database is locked}}
sl@0
   216
sl@0
   217
lock_status 4.4.1 db {main shared temp closed file2 unlocked}
sl@0
   218
lock_status 4.4.2 db2 {main unlocked temp closed file2 unlocked}
sl@0
   219
sl@0
   220
# We have to make sure that the cache_size and the soft_heap_limit
sl@0
   221
# are large enough to hold the entire change in memory.  If either
sl@0
   222
# is set too small, then changes will spill to the database, forcing
sl@0
   223
# a reserved lock to promote to exclusive.  That will mess up our
sl@0
   224
# test results. 
sl@0
   225
sl@0
   226
set soft_limit [sqlite3_soft_heap_limit 0]
sl@0
   227
sl@0
   228
sl@0
   229
do_test attach2-4.5 {
sl@0
   230
  # Handle 'db2' reserves file2.
sl@0
   231
  execsql {BEGIN} db2
sl@0
   232
  execsql {INSERT INTO file2.t1 VALUES(1, 2)} db2
sl@0
   233
  # Lock status:
sl@0
   234
  #    db  - shared(main)
sl@0
   235
  #    db2 - reserved(file2)
sl@0
   236
} {}
sl@0
   237
sl@0
   238
lock_status 4.5.1 db {main shared temp closed file2 unlocked}
sl@0
   239
lock_status 4.5.2 db2 {main unlocked temp closed file2 reserved}
sl@0
   240
sl@0
   241
do_test attach2-4.6.1 {
sl@0
   242
  # Reads are allowed against a reserved database.
sl@0
   243
  catchsql {
sl@0
   244
    SELECT * FROM file2.t1;
sl@0
   245
  }
sl@0
   246
  # Lock status:
sl@0
   247
  #    db  - shared(main), shared(file2)
sl@0
   248
  #    db2 - reserved(file2)
sl@0
   249
} {0 {}}
sl@0
   250
sl@0
   251
lock_status 4.6.1.1 db {main shared temp closed file2 shared}
sl@0
   252
lock_status 4.6.1.2 db2 {main unlocked temp closed file2 reserved}
sl@0
   253
sl@0
   254
do_test attach2-4.6.2 {
sl@0
   255
  # Writes against a reserved database are not allowed.
sl@0
   256
  catchsql {
sl@0
   257
    UPDATE file2.t1 SET a=0;
sl@0
   258
  }
sl@0
   259
} {1 {database is locked}}
sl@0
   260
sl@0
   261
lock_status 4.6.2.1 db {main shared temp closed file2 shared}
sl@0
   262
lock_status 4.6.2.2 db2 {main unlocked temp closed file2 reserved}
sl@0
   263
sl@0
   264
do_test attach2-4.7 {
sl@0
   265
  # Ensure handle 'db' retains the lock on the main file after
sl@0
   266
  # failing to obtain a write-lock on file2.
sl@0
   267
  catchsql {
sl@0
   268
    INSERT INTO t1 VALUES(1, 2)
sl@0
   269
  } db2 
sl@0
   270
} {0 {}}
sl@0
   271
sl@0
   272
lock_status 4.7.1 db {main shared temp closed file2 shared}
sl@0
   273
lock_status 4.7.2 db2 {main reserved temp closed file2 reserved}
sl@0
   274
sl@0
   275
do_test attach2-4.8 {
sl@0
   276
  # We should still be able to read test.db from db2
sl@0
   277
  execsql {SELECT * FROM t1} db2
sl@0
   278
} {1 2}
sl@0
   279
sl@0
   280
lock_status 4.8.1 db {main shared temp closed file2 shared}
sl@0
   281
lock_status 4.8.2 db2 {main reserved temp closed file2 reserved}
sl@0
   282
sl@0
   283
do_test attach2-4.9 {
sl@0
   284
  # Try to upgrade the handle 'db' lock.
sl@0
   285
  catchsql {
sl@0
   286
    INSERT INTO t1 VALUES(1, 2)
sl@0
   287
  }
sl@0
   288
} {1 {database is locked}}
sl@0
   289
sl@0
   290
lock_status 4.9.1 db {main shared temp closed file2 shared}
sl@0
   291
lock_status 4.9.2 db2 {main reserved temp closed file2 reserved}
sl@0
   292
sl@0
   293
do_test attach2-4.10 {
sl@0
   294
  # We cannot commit db2 while db is holding a read-lock
sl@0
   295
  catchsql {COMMIT} db2
sl@0
   296
} {1 {database is locked}}
sl@0
   297
sl@0
   298
lock_status 4.10.1 db {main shared temp closed file2 shared}
sl@0
   299
lock_status 4.10.2 db2 {main pending temp closed file2 reserved}
sl@0
   300
sl@0
   301
set sqlite_os_trace 0
sl@0
   302
do_test attach2-4.11 {
sl@0
   303
  # db is able to commit.
sl@0
   304
  catchsql {COMMIT}
sl@0
   305
} {0 {}}
sl@0
   306
sl@0
   307
lock_status 4.11.1 db {main unlocked temp closed file2 unlocked}
sl@0
   308
lock_status 4.11.2 db2 {main pending temp closed file2 reserved}
sl@0
   309
sl@0
   310
do_test attach2-4.12 {
sl@0
   311
  # Now we can commit db2
sl@0
   312
  catchsql {COMMIT} db2
sl@0
   313
} {0 {}}
sl@0
   314
sl@0
   315
lock_status 4.12.1 db {main unlocked temp closed file2 unlocked}
sl@0
   316
lock_status 4.12.2 db2 {main unlocked temp closed file2 unlocked}
sl@0
   317
sl@0
   318
do_test attach2-4.13 {
sl@0
   319
  execsql {SELECT * FROM file2.t1}
sl@0
   320
} {1 2}
sl@0
   321
do_test attach2-4.14 {
sl@0
   322
  execsql {INSERT INTO t1 VALUES(1, 2)}
sl@0
   323
} {}
sl@0
   324
do_test attach2-4.15 {
sl@0
   325
  execsql {SELECT * FROM t1} db2
sl@0
   326
} {1 2 1 2}
sl@0
   327
sl@0
   328
db close
sl@0
   329
db2 close
sl@0
   330
file delete -force test2.db
sl@0
   331
sqlite3_soft_heap_limit $soft_limit
sl@0
   332
sl@0
   333
# These tests - attach2-5.* - check that the master journal file is deleted
sl@0
   334
# correctly when a multi-file transaction is committed or rolled back.
sl@0
   335
#
sl@0
   336
# Update: It's not actually created if a rollback occurs, so that test
sl@0
   337
# doesn't really prove too much.
sl@0
   338
foreach f [glob test.db*] {file delete -force $f}
sl@0
   339
do_test attach2-5.1 {
sl@0
   340
  sqlite3 db test.db
sl@0
   341
  execsql {
sl@0
   342
    ATTACH 'test.db2' AS aux;
sl@0
   343
  }
sl@0
   344
} {}
sl@0
   345
do_test attach2-5.2 {
sl@0
   346
  execsql {
sl@0
   347
    BEGIN;
sl@0
   348
    CREATE TABLE tbl(a, b, c);
sl@0
   349
    CREATE TABLE aux.tbl(a, b, c);
sl@0
   350
    COMMIT;
sl@0
   351
  }
sl@0
   352
} {}
sl@0
   353
do_test attach2-5.3 {
sl@0
   354
  lsort [glob test.db*]
sl@0
   355
} {test.db test.db2}
sl@0
   356
do_test attach2-5.4 {
sl@0
   357
  execsql {
sl@0
   358
    BEGIN;
sl@0
   359
    DROP TABLE aux.tbl;
sl@0
   360
    DROP TABLE tbl;
sl@0
   361
    ROLLBACK;
sl@0
   362
  }
sl@0
   363
} {}
sl@0
   364
do_test attach2-5.5 {
sl@0
   365
  lsort [glob test.db*]
sl@0
   366
} {test.db test.db2}
sl@0
   367
sl@0
   368
# Check that a database cannot be ATTACHed or DETACHed during a transaction.
sl@0
   369
do_test attach2-6.1 {
sl@0
   370
  execsql {
sl@0
   371
    BEGIN;
sl@0
   372
  }
sl@0
   373
} {}
sl@0
   374
do_test attach2-6.2 {
sl@0
   375
  catchsql {
sl@0
   376
    ATTACH 'test3.db' as aux2;
sl@0
   377
  }
sl@0
   378
} {1 {cannot ATTACH database within transaction}}
sl@0
   379
sl@0
   380
do_test attach2-6.3 {
sl@0
   381
  catchsql {
sl@0
   382
    DETACH aux;
sl@0
   383
  }
sl@0
   384
} {1 {cannot DETACH database within transaction}}
sl@0
   385
do_test attach2-6.4 {
sl@0
   386
  execsql {
sl@0
   387
    COMMIT;
sl@0
   388
    DETACH aux;
sl@0
   389
  }
sl@0
   390
} {}
sl@0
   391
sl@0
   392
db close
sl@0
   393
sl@0
   394
finish_test