os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/autovacuum.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 file is testing the SELECT statement.
sl@0
    13
#
sl@0
    14
# $Id: autovacuum.test,v 1.28 2008/09/10 10:57:28 danielk1977 Exp $
sl@0
    15
sl@0
    16
set testdir [file dirname $argv0]
sl@0
    17
source $testdir/tester.tcl
sl@0
    18
sl@0
    19
# If this build of the library does not support auto-vacuum, omit this
sl@0
    20
# whole file.
sl@0
    21
ifcapable {!autovacuum || !pragma} {
sl@0
    22
  finish_test
sl@0
    23
  return
sl@0
    24
}
sl@0
    25
sl@0
    26
# Return a string $len characters long. The returned string is $char repeated
sl@0
    27
# over and over. For example, [make_str abc 8] returns "abcabcab".
sl@0
    28
proc make_str {char len} {
sl@0
    29
  set str [string repeat $char. $len]
sl@0
    30
  return [string range $str 0 [expr $len-1]]
sl@0
    31
}
sl@0
    32
sl@0
    33
# Return the number of pages in the file test.db by looking at the file system.
sl@0
    34
proc file_pages {} {
sl@0
    35
  return [expr [file size test.db] / 1024]
sl@0
    36
}
sl@0
    37
sl@0
    38
#-------------------------------------------------------------------------
sl@0
    39
# Test cases autovacuum-1.* work as follows:
sl@0
    40
#
sl@0
    41
# 1. A table with a single indexed field is created.
sl@0
    42
# 2. Approximately 20 rows are inserted into the table. Each row is long 
sl@0
    43
#    enough such that it uses at least 2 overflow pages for both the table 
sl@0
    44
#    and index entry.
sl@0
    45
# 3. The rows are deleted in a psuedo-random order. Sometimes only one row
sl@0
    46
#    is deleted per transaction, sometimes more than one.
sl@0
    47
# 4. After each transaction the table data is checked to ensure it is correct
sl@0
    48
#    and a "PRAGMA integrity_check" is executed.
sl@0
    49
# 5. Once all the rows are deleted the file is checked to make sure it 
sl@0
    50
#    consists of exactly 4 pages.
sl@0
    51
#
sl@0
    52
# Steps 2-5 are repeated for a few different psuedo-random delete patterns 
sl@0
    53
# (defined by the $delete_orders list).
sl@0
    54
set delete_orders [list]
sl@0
    55
lappend delete_orders {1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20}
sl@0
    56
lappend delete_orders {20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1} 
sl@0
    57
lappend delete_orders {8 18 2 4 14 11 13 3 10 7 9 5 12 17 19 15 20 6 16 1}
sl@0
    58
lappend delete_orders {10 3 11 17 19 20 7 4 13 6 1 14 16 12 9 18 8 15 5 2}
sl@0
    59
lappend delete_orders {{1 2 3 4 5 6 7 8 9 10} {11 12 13 14 15 16 17 18 19 20}}
sl@0
    60
lappend delete_orders {{19 8 17 15} {16 11 9 14} {18 5 3 1} {13 20 7 2} {6 12}}
sl@0
    61
sl@0
    62
# The length of each table entry. 
sl@0
    63
# set ENTRY_LEN 3500
sl@0
    64
set ENTRY_LEN 3500
sl@0
    65
sl@0
    66
do_test autovacuum-1.1 {
sl@0
    67
  execsql {
sl@0
    68
    PRAGMA auto_vacuum = 1;
sl@0
    69
    CREATE TABLE av1(a);
sl@0
    70
    CREATE INDEX av1_idx ON av1(a);
sl@0
    71
  }
sl@0
    72
} {}
sl@0
    73
sl@0
    74
set tn 0
sl@0
    75
foreach delete_order $delete_orders {
sl@0
    76
  incr tn
sl@0
    77
sl@0
    78
  # Set up the table.
sl@0
    79
  set ::tbl_data [list]
sl@0
    80
  foreach i [lsort -integer [eval concat $delete_order]] {
sl@0
    81
    execsql "INSERT INTO av1 (oid, a) VALUES($i, '[make_str $i $ENTRY_LEN]')"
sl@0
    82
    lappend ::tbl_data [make_str $i $ENTRY_LEN]
sl@0
    83
  }
sl@0
    84
sl@0
    85
  # Make sure the integrity check passes with the initial data.
sl@0
    86
  ifcapable {integrityck} {
sl@0
    87
    do_test autovacuum-1.$tn.1 {
sl@0
    88
      execsql {
sl@0
    89
        pragma integrity_check
sl@0
    90
      }
sl@0
    91
    } {ok}
sl@0
    92
  }
sl@0
    93
sl@0
    94
  foreach delete $delete_order {
sl@0
    95
    # Delete one set of rows from the table.
sl@0
    96
    do_test autovacuum-1.$tn.($delete).1 {
sl@0
    97
      execsql "
sl@0
    98
        DELETE FROM av1 WHERE oid = [join $delete " OR oid = "]
sl@0
    99
      "
sl@0
   100
    } {}
sl@0
   101
sl@0
   102
    # Do the integrity check.
sl@0
   103
    ifcapable {integrityck} {
sl@0
   104
      do_test autovacuum-1.$tn.($delete).2 {
sl@0
   105
        execsql {
sl@0
   106
          pragma integrity_check
sl@0
   107
        }
sl@0
   108
      } {ok}
sl@0
   109
    }
sl@0
   110
    # Ensure the data remaining in the table is what was expected.
sl@0
   111
    foreach d $delete {
sl@0
   112
      set idx [lsearch $::tbl_data [make_str $d $ENTRY_LEN]]
sl@0
   113
      set ::tbl_data [lreplace $::tbl_data $idx $idx]
sl@0
   114
    }
sl@0
   115
    do_test autovacuum-1.$tn.($delete).3 {
sl@0
   116
      execsql {
sl@0
   117
        select a from av1
sl@0
   118
      }
sl@0
   119
    } $::tbl_data
sl@0
   120
  }
sl@0
   121
sl@0
   122
  # All rows have been deleted. Ensure the file has shrunk to 4 pages.
sl@0
   123
  do_test autovacuum-1.$tn.3 {
sl@0
   124
    file_pages
sl@0
   125
  } {4}
sl@0
   126
}
sl@0
   127
sl@0
   128
#---------------------------------------------------------------------------
sl@0
   129
# Tests cases autovacuum-2.* test that root pages are allocated 
sl@0
   130
# and deallocated correctly at the start of the file. Operation is roughly as
sl@0
   131
# follows:
sl@0
   132
#
sl@0
   133
# autovacuum-2.1.*: Drop the tables that currently exist in the database.
sl@0
   134
# autovacuum-2.2.*: Create some tables. Ensure that data pages can be
sl@0
   135
#                   moved correctly to make space for new root-pages.
sl@0
   136
# autovacuum-2.3.*: Drop one of the tables just created (not the last one),
sl@0
   137
#                   and check that one of the other tables is moved to
sl@0
   138
#                   the free root-page location.
sl@0
   139
# autovacuum-2.4.*: Check that a table can be created correctly when the
sl@0
   140
#                   root-page it requires is on the free-list.
sl@0
   141
# autovacuum-2.5.*: Check that a table with indices can be dropped. This
sl@0
   142
#                   is slightly tricky because dropping one of the
sl@0
   143
#                   indices/table btrees could move the root-page of another.
sl@0
   144
#                   The code-generation layer of SQLite overcomes this problem
sl@0
   145
#                   by dropping the btrees in descending order of root-pages.
sl@0
   146
#                   This test ensures that this actually happens.
sl@0
   147
#
sl@0
   148
do_test autovacuum-2.1.1 {
sl@0
   149
  execsql {
sl@0
   150
    DROP TABLE av1;
sl@0
   151
  }
sl@0
   152
} {}
sl@0
   153
do_test autovacuum-2.1.2 {
sl@0
   154
  file_pages
sl@0
   155
} {1}
sl@0
   156
sl@0
   157
# Create a table and put some data in it.
sl@0
   158
do_test autovacuum-2.2.1 {
sl@0
   159
  execsql {
sl@0
   160
    CREATE TABLE av1(x);
sl@0
   161
    SELECT rootpage FROM sqlite_master ORDER BY rootpage;
sl@0
   162
  }
sl@0
   163
} {3}
sl@0
   164
do_test autovacuum-2.2.2 {
sl@0
   165
  execsql "
sl@0
   166
    INSERT INTO av1 VALUES('[make_str abc 3000]');
sl@0
   167
    INSERT INTO av1 VALUES('[make_str def 3000]');
sl@0
   168
    INSERT INTO av1 VALUES('[make_str ghi 3000]');
sl@0
   169
    INSERT INTO av1 VALUES('[make_str jkl 3000]');
sl@0
   170
  "
sl@0
   171
  set ::av1_data [db eval {select * from av1}]
sl@0
   172
  file_pages
sl@0
   173
} {15}
sl@0
   174
sl@0
   175
# Create another table. Check it is located immediately after the first.
sl@0
   176
# This test case moves the second page in an over-flow chain.
sl@0
   177
do_test autovacuum-2.2.3 {
sl@0
   178
  execsql {
sl@0
   179
    CREATE TABLE av2(x);
sl@0
   180
    SELECT rootpage FROM sqlite_master ORDER BY rootpage;
sl@0
   181
  }
sl@0
   182
} {3 4}
sl@0
   183
do_test autovacuum-2.2.4 {
sl@0
   184
  file_pages
sl@0
   185
} {16}
sl@0
   186
sl@0
   187
# Create another table. Check it is located immediately after the second.
sl@0
   188
# This test case moves the first page in an over-flow chain.
sl@0
   189
do_test autovacuum-2.2.5 {
sl@0
   190
  execsql {
sl@0
   191
    CREATE TABLE av3(x);
sl@0
   192
    SELECT rootpage FROM sqlite_master ORDER BY rootpage;
sl@0
   193
  }
sl@0
   194
} {3 4 5}
sl@0
   195
do_test autovacuum-2.2.6 {
sl@0
   196
  file_pages
sl@0
   197
} {17}
sl@0
   198
sl@0
   199
# Create another table. Check it is located immediately after the second.
sl@0
   200
# This test case moves a btree leaf page.
sl@0
   201
do_test autovacuum-2.2.7 {
sl@0
   202
  execsql {
sl@0
   203
    CREATE TABLE av4(x);
sl@0
   204
    SELECT rootpage FROM sqlite_master ORDER BY rootpage;
sl@0
   205
  }
sl@0
   206
} {3 4 5 6}
sl@0
   207
do_test autovacuum-2.2.8 {
sl@0
   208
  file_pages
sl@0
   209
} {18}
sl@0
   210
do_test autovacuum-2.2.9 {
sl@0
   211
  execsql {
sl@0
   212
    select * from av1
sl@0
   213
  }
sl@0
   214
} $av1_data
sl@0
   215
sl@0
   216
do_test autovacuum-2.3.1 {
sl@0
   217
  execsql {
sl@0
   218
    INSERT INTO av2 SELECT 'av1' || x FROM av1;
sl@0
   219
    INSERT INTO av3 SELECT 'av2' || x FROM av1;
sl@0
   220
    INSERT INTO av4 SELECT 'av3' || x FROM av1;
sl@0
   221
  }
sl@0
   222
  set ::av2_data [execsql {select x from av2}]
sl@0
   223
  set ::av3_data [execsql {select x from av3}]
sl@0
   224
  set ::av4_data [execsql {select x from av4}]
sl@0
   225
  file_pages
sl@0
   226
} {54}
sl@0
   227
do_test autovacuum-2.3.2 {
sl@0
   228
  execsql {
sl@0
   229
    DROP TABLE av2;
sl@0
   230
    SELECT rootpage FROM sqlite_master ORDER BY rootpage;
sl@0
   231
  }
sl@0
   232
} {3 4 5}
sl@0
   233
do_test autovacuum-2.3.3 {
sl@0
   234
  file_pages
sl@0
   235
} {41}
sl@0
   236
do_test autovacuum-2.3.4 {
sl@0
   237
  execsql {
sl@0
   238
    SELECT x FROM av3;
sl@0
   239
  }
sl@0
   240
} $::av3_data
sl@0
   241
do_test autovacuum-2.3.5 {
sl@0
   242
  execsql {
sl@0
   243
    SELECT x FROM av4;
sl@0
   244
  }
sl@0
   245
} $::av4_data
sl@0
   246
sl@0
   247
# Drop all the tables in the file. This puts all pages except the first 2
sl@0
   248
# (the sqlite_master root-page and the first pointer map page) on the 
sl@0
   249
# free-list.
sl@0
   250
do_test autovacuum-2.4.1 {
sl@0
   251
  execsql {
sl@0
   252
    DROP TABLE av1;
sl@0
   253
    DROP TABLE av3;
sl@0
   254
    BEGIN;
sl@0
   255
    DROP TABLE av4;
sl@0
   256
  }
sl@0
   257
  file_pages
sl@0
   258
} {15}
sl@0
   259
do_test autovacuum-2.4.2 {
sl@0
   260
  for {set i 3} {$i<=10} {incr i} {
sl@0
   261
    execsql "CREATE TABLE av$i (x)"
sl@0
   262
  }
sl@0
   263
  file_pages
sl@0
   264
} {15}
sl@0
   265
do_test autovacuum-2.4.3 {
sl@0
   266
  execsql {
sl@0
   267
    SELECT rootpage FROM sqlite_master ORDER by rootpage
sl@0
   268
  }
sl@0
   269
} {3 4 5 6 7 8 9 10}
sl@0
   270
sl@0
   271
# Right now there are 5 free pages in the database. Consume and then free
sl@0
   272
# a 520 pages. Then create 520 tables. This ensures that at least some of the
sl@0
   273
# desired root-pages reside on the second free-list trunk page, and that the
sl@0
   274
# trunk itself is required at some point.
sl@0
   275
do_test autovacuum-2.4.4 {
sl@0
   276
  execsql "
sl@0
   277
    INSERT INTO av3 VALUES ('[make_str abcde [expr 1020*520 + 500]]');
sl@0
   278
    DELETE FROM av3;
sl@0
   279
  "
sl@0
   280
} {}
sl@0
   281
set root_page_list [list]
sl@0
   282
set pending_byte_page [expr ($::sqlite_pending_byte / 1024) + 1]
sl@0
   283
for {set i 3} {$i<=532} {incr i} {
sl@0
   284
  # 207 and 412 are pointer-map pages.
sl@0
   285
  if { $i!=207 && $i!=412 && $i != $pending_byte_page} {
sl@0
   286
    lappend root_page_list $i
sl@0
   287
  }
sl@0
   288
}
sl@0
   289
if {$i >= $pending_byte_page} {
sl@0
   290
  lappend root_page_list $i
sl@0
   291
}
sl@0
   292
do_test autovacuum-2.4.5 {
sl@0
   293
  for {set i 11} {$i<=530} {incr i} {
sl@0
   294
    execsql "CREATE TABLE av$i (x)"
sl@0
   295
  }
sl@0
   296
  execsql {
sl@0
   297
    SELECT rootpage FROM sqlite_master ORDER by rootpage
sl@0
   298
  }
sl@0
   299
} $root_page_list
sl@0
   300
sl@0
   301
# Just for fun, delete all those tables and see if the database is 1 page.
sl@0
   302
do_test autovacuum-2.4.6 {
sl@0
   303
  execsql COMMIT;
sl@0
   304
  file_pages
sl@0
   305
} [expr 561 + (($i >= $pending_byte_page)?1:0)]
sl@0
   306
integrity_check autovacuum-2.4.6
sl@0
   307
do_test autovacuum-2.4.7 {
sl@0
   308
  execsql BEGIN
sl@0
   309
  for {set i 3} {$i<=530} {incr i} {
sl@0
   310
    execsql "DROP TABLE av$i"
sl@0
   311
  }
sl@0
   312
  execsql COMMIT
sl@0
   313
  file_pages
sl@0
   314
} 1
sl@0
   315
sl@0
   316
# Create some tables with indices to drop.
sl@0
   317
do_test autovacuum-2.5.1 {
sl@0
   318
  execsql {
sl@0
   319
    CREATE TABLE av1(a PRIMARY KEY, b, c);
sl@0
   320
    INSERT INTO av1 VALUES('av1 a', 'av1 b', 'av1 c');
sl@0
   321
sl@0
   322
    CREATE TABLE av2(a PRIMARY KEY, b, c);
sl@0
   323
    CREATE INDEX av2_i1 ON av2(b);
sl@0
   324
    CREATE INDEX av2_i2 ON av2(c);
sl@0
   325
    INSERT INTO av2 VALUES('av2 a', 'av2 b', 'av2 c');
sl@0
   326
sl@0
   327
    CREATE TABLE av3(a PRIMARY KEY, b, c);
sl@0
   328
    CREATE INDEX av3_i1 ON av3(b);
sl@0
   329
    INSERT INTO av3 VALUES('av3 a', 'av3 b', 'av3 c');
sl@0
   330
sl@0
   331
    CREATE TABLE av4(a, b, c);
sl@0
   332
    CREATE INDEX av4_i1 ON av4(a);
sl@0
   333
    CREATE INDEX av4_i2 ON av4(b);
sl@0
   334
    CREATE INDEX av4_i3 ON av4(c);
sl@0
   335
    CREATE INDEX av4_i4 ON av4(a, b, c);
sl@0
   336
    INSERT INTO av4 VALUES('av4 a', 'av4 b', 'av4 c');
sl@0
   337
  }
sl@0
   338
} {}
sl@0
   339
sl@0
   340
do_test autovacuum-2.5.2 {
sl@0
   341
  execsql {
sl@0
   342
    SELECT name, rootpage FROM sqlite_master;
sl@0
   343
  }
sl@0
   344
} [list av1 3  sqlite_autoindex_av1_1 4 \
sl@0
   345
        av2 5  sqlite_autoindex_av2_1 6 av2_i1 7 av2_i2 8 \
sl@0
   346
        av3 9 sqlite_autoindex_av3_1 10 av3_i1 11 \
sl@0
   347
        av4 12 av4_i1 13 av4_i2 14 av4_i3 15 av4_i4 16 \
sl@0
   348
]
sl@0
   349
sl@0
   350
# The following 4 tests are SELECT queries that use the indices created.
sl@0
   351
# If the root-pages in the internal schema are not updated correctly when
sl@0
   352
# a table or indice is moved, these queries will fail. They are repeated
sl@0
   353
# after each table is dropped (i.e. as test cases 2.5.*.[1..4]).
sl@0
   354
do_test autovacuum-2.5.2.1 {
sl@0
   355
  execsql {
sl@0
   356
    SELECT * FROM av1 WHERE a = 'av1 a';
sl@0
   357
  }
sl@0
   358
} {{av1 a} {av1 b} {av1 c}}
sl@0
   359
do_test autovacuum-2.5.2.2 {
sl@0
   360
  execsql {
sl@0
   361
    SELECT * FROM av2 WHERE a = 'av2 a' AND b = 'av2 b' AND c = 'av2 c'
sl@0
   362
  }
sl@0
   363
} {{av2 a} {av2 b} {av2 c}}
sl@0
   364
do_test autovacuum-2.5.2.3 {
sl@0
   365
  execsql {
sl@0
   366
    SELECT * FROM av3 WHERE a = 'av3 a' AND b = 'av3 b';
sl@0
   367
  }
sl@0
   368
} {{av3 a} {av3 b} {av3 c}}
sl@0
   369
do_test autovacuum-2.5.2.4 {
sl@0
   370
  execsql {
sl@0
   371
    SELECT * FROM av4 WHERE a = 'av4 a' AND b = 'av4 b' AND c = 'av4 c';
sl@0
   372
  }
sl@0
   373
} {{av4 a} {av4 b} {av4 c}}
sl@0
   374
sl@0
   375
# Drop table av3. Indices av4_i2, av4_i3 and av4_i4 are moved to fill the two
sl@0
   376
# root pages vacated. The operation proceeds as:
sl@0
   377
# Step 1: Delete av3_i1 (root-page 11). Move root-page of av4_i4 to page 11.
sl@0
   378
# Step 2: Delete av3 (root-page 10). Move root-page of av4_i3 to page 10.
sl@0
   379
# Step 3: Delete sqlite_autoindex_av1_3 (root-page 9). Move av4_i2 to page 9.
sl@0
   380
do_test autovacuum-2.5.3 {
sl@0
   381
  execsql {
sl@0
   382
    DROP TABLE av3;
sl@0
   383
    SELECT name, rootpage FROM sqlite_master;
sl@0
   384
  }
sl@0
   385
} [list av1 3  sqlite_autoindex_av1_1 4 \
sl@0
   386
        av2 5  sqlite_autoindex_av2_1 6 av2_i1 7 av2_i2 8 \
sl@0
   387
        av4 12 av4_i1 13 av4_i2 9 av4_i3 10 av4_i4 11 \
sl@0
   388
]
sl@0
   389
do_test autovacuum-2.5.3.1 {
sl@0
   390
  execsql {
sl@0
   391
    SELECT * FROM av1 WHERE a = 'av1 a';
sl@0
   392
  }
sl@0
   393
} {{av1 a} {av1 b} {av1 c}}
sl@0
   394
do_test autovacuum-2.5.3.2 {
sl@0
   395
  execsql {
sl@0
   396
    SELECT * FROM av2 WHERE a = 'av2 a' AND b = 'av2 b' AND c = 'av2 c'
sl@0
   397
  }
sl@0
   398
} {{av2 a} {av2 b} {av2 c}}
sl@0
   399
do_test autovacuum-2.5.3.3 {
sl@0
   400
  execsql {
sl@0
   401
    SELECT * FROM av4 WHERE a = 'av4 a' AND b = 'av4 b' AND c = 'av4 c';
sl@0
   402
  }
sl@0
   403
} {{av4 a} {av4 b} {av4 c}}
sl@0
   404
sl@0
   405
# Drop table av1:
sl@0
   406
# Step 1: Delete av1 (root page 4). Root-page of av4_i1 fills the gap.
sl@0
   407
# Step 2: Delete sqlite_autoindex_av1_1 (root page 3). Move av4 to the gap.
sl@0
   408
do_test autovacuum-2.5.4 {
sl@0
   409
  execsql {
sl@0
   410
    DROP TABLE av1;
sl@0
   411
    SELECT name, rootpage FROM sqlite_master;
sl@0
   412
  }
sl@0
   413
} [list av2 5  sqlite_autoindex_av2_1 6 av2_i1 7 av2_i2 8 \
sl@0
   414
        av4 3 av4_i1 4 av4_i2 9 av4_i3 10 av4_i4 11 \
sl@0
   415
]
sl@0
   416
do_test autovacuum-2.5.4.2 {
sl@0
   417
  execsql {
sl@0
   418
    SELECT * FROM av2 WHERE a = 'av2 a' AND b = 'av2 b' AND c = 'av2 c'
sl@0
   419
  }
sl@0
   420
} {{av2 a} {av2 b} {av2 c}}
sl@0
   421
do_test autovacuum-2.5.4.4 {
sl@0
   422
  execsql {
sl@0
   423
    SELECT * FROM av4 WHERE a = 'av4 a' AND b = 'av4 b' AND c = 'av4 c';
sl@0
   424
  }
sl@0
   425
} {{av4 a} {av4 b} {av4 c}}
sl@0
   426
sl@0
   427
# Drop table av4:
sl@0
   428
# Step 1: Delete av4_i4.
sl@0
   429
# Step 2: Delete av4_i3.
sl@0
   430
# Step 3: Delete av4_i2.
sl@0
   431
# Step 4: Delete av4_i1. av2_i2 replaces it.
sl@0
   432
# Step 5: Delete av4. av2_i1 replaces it.
sl@0
   433
do_test autovacuum-2.5.5 {
sl@0
   434
  execsql {
sl@0
   435
    DROP TABLE av4;
sl@0
   436
    SELECT name, rootpage FROM sqlite_master;
sl@0
   437
  }
sl@0
   438
} [list av2 5 sqlite_autoindex_av2_1 6 av2_i1 3 av2_i2 4]
sl@0
   439
do_test autovacuum-2.5.5.2 {
sl@0
   440
  execsql {
sl@0
   441
    SELECT * FROM av2 WHERE a = 'av2 a' AND b = 'av2 b' AND c = 'av2 c'
sl@0
   442
  }
sl@0
   443
} {{av2 a} {av2 b} {av2 c}}
sl@0
   444
sl@0
   445
#--------------------------------------------------------------------------
sl@0
   446
# Test cases autovacuum-3.* test the operation of the "PRAGMA auto_vacuum"
sl@0
   447
# command.
sl@0
   448
#
sl@0
   449
do_test autovacuum-3.1 {
sl@0
   450
  execsql {
sl@0
   451
    PRAGMA auto_vacuum;
sl@0
   452
  }
sl@0
   453
} {1}
sl@0
   454
do_test autovacuum-3.2 {
sl@0
   455
  db close
sl@0
   456
  sqlite3 db test.db
sl@0
   457
  execsql {
sl@0
   458
    PRAGMA auto_vacuum;
sl@0
   459
  }
sl@0
   460
} {1}
sl@0
   461
do_test autovacuum-3.3 {
sl@0
   462
  execsql {
sl@0
   463
    PRAGMA auto_vacuum = 0;
sl@0
   464
    PRAGMA auto_vacuum;
sl@0
   465
  }
sl@0
   466
} {1}
sl@0
   467
sl@0
   468
do_test autovacuum-3.4 {
sl@0
   469
  db close
sl@0
   470
  file delete -force test.db
sl@0
   471
  sqlite3 db test.db
sl@0
   472
  execsql {
sl@0
   473
    PRAGMA auto_vacuum;
sl@0
   474
  }
sl@0
   475
} $AUTOVACUUM
sl@0
   476
do_test autovacuum-3.5 {
sl@0
   477
  execsql {
sl@0
   478
    CREATE TABLE av1(x);
sl@0
   479
    PRAGMA auto_vacuum;
sl@0
   480
  }
sl@0
   481
} $AUTOVACUUM
sl@0
   482
do_test autovacuum-3.6 {
sl@0
   483
  execsql {
sl@0
   484
    PRAGMA auto_vacuum = 1;
sl@0
   485
    PRAGMA auto_vacuum;
sl@0
   486
  }
sl@0
   487
} [expr $AUTOVACUUM ? 1 : 0]
sl@0
   488
do_test autovacuum-3.7 {
sl@0
   489
  execsql {
sl@0
   490
    DROP TABLE av1;
sl@0
   491
  }
sl@0
   492
  file_pages
sl@0
   493
} [expr $AUTOVACUUM?1:2]
sl@0
   494
sl@0
   495
sl@0
   496
#-----------------------------------------------------------------------
sl@0
   497
# Test that if a statement transaction around a CREATE INDEX statement is
sl@0
   498
# rolled back no corruption occurs.
sl@0
   499
#
sl@0
   500
do_test autovacuum-4.0 {
sl@0
   501
  # The last round of tests may have left the db in non-autovacuum mode.
sl@0
   502
  # Reset everything just in case.
sl@0
   503
  #
sl@0
   504
  db close
sl@0
   505
  file delete -force test.db test.db-journal
sl@0
   506
  sqlite3 db test.db
sl@0
   507
  execsql {
sl@0
   508
    PRAGMA auto_vacuum = 1;
sl@0
   509
    PRAGMA auto_vacuum;
sl@0
   510
  }
sl@0
   511
} {1}
sl@0
   512
do_test autovacuum-4.1 {
sl@0
   513
  execsql {
sl@0
   514
    CREATE TABLE av1(a, b);
sl@0
   515
    BEGIN;
sl@0
   516
  }
sl@0
   517
  for {set i 0} {$i<100} {incr i} {
sl@0
   518
    execsql "INSERT INTO av1 VALUES($i, '[string repeat X 200]');"
sl@0
   519
  }
sl@0
   520
  execsql "INSERT INTO av1 VALUES(99, '[string repeat X 200]');"
sl@0
   521
  execsql {
sl@0
   522
    SELECT sum(a) FROM av1;
sl@0
   523
  }
sl@0
   524
} {5049}
sl@0
   525
do_test autovacuum-4.2 {
sl@0
   526
  catchsql {
sl@0
   527
    CREATE UNIQUE INDEX av1_i ON av1(a);
sl@0
   528
  }
sl@0
   529
} {1 {indexed columns are not unique}}
sl@0
   530
do_test autovacuum-4.3 {
sl@0
   531
  execsql {
sl@0
   532
    SELECT sum(a) FROM av1;
sl@0
   533
  }
sl@0
   534
} {5049}
sl@0
   535
do_test autovacuum-4.4 {
sl@0
   536
  execsql {
sl@0
   537
    COMMIT;
sl@0
   538
  }
sl@0
   539
} {}
sl@0
   540
sl@0
   541
ifcapable integrityck {
sl@0
   542
sl@0
   543
# Ticket #1727
sl@0
   544
do_test autovacuum-5.1 {
sl@0
   545
  db close
sl@0
   546
  sqlite3 db :memory:
sl@0
   547
  db eval {
sl@0
   548
    PRAGMA auto_vacuum=1;
sl@0
   549
    CREATE TABLE t1(a);
sl@0
   550
    CREATE TABLE t2(a);
sl@0
   551
    DROP TABLE t1;
sl@0
   552
    PRAGMA integrity_check;
sl@0
   553
  }
sl@0
   554
} ok
sl@0
   555
sl@0
   556
}
sl@0
   557
sl@0
   558
# Ticket #1728.
sl@0
   559
#
sl@0
   560
# In autovacuum mode, when tables or indices are deleted, the rootpage
sl@0
   561
# values in the symbol table have to be updated.  There was a bug in this
sl@0
   562
# logic so that if an index/table was moved twice, the second move might
sl@0
   563
# not occur.  This would leave the internal symbol table in an inconsistent
sl@0
   564
# state causing subsequent statements to fail.
sl@0
   565
#
sl@0
   566
# The problem is difficult to reproduce.  The sequence of statements in
sl@0
   567
# the following test are carefully designed make it occur and thus to
sl@0
   568
# verify that this very obscure bug has been resolved.
sl@0
   569
# 
sl@0
   570
ifcapable integrityck&&memorydb {
sl@0
   571
sl@0
   572
do_test autovacuum-6.1 {
sl@0
   573
  db close
sl@0
   574
  sqlite3 db :memory:
sl@0
   575
  db eval {
sl@0
   576
    PRAGMA auto_vacuum=1;
sl@0
   577
    CREATE TABLE t1(a, b);
sl@0
   578
    CREATE INDEX i1 ON t1(a);
sl@0
   579
    CREATE TABLE t2(a);
sl@0
   580
    CREATE INDEX i2 ON t2(a);
sl@0
   581
    CREATE TABLE t3(a);
sl@0
   582
    CREATE INDEX i3 ON t2(a);
sl@0
   583
    CREATE INDEX x ON t1(b);
sl@0
   584
    DROP TABLE t3;
sl@0
   585
    PRAGMA integrity_check;
sl@0
   586
    DROP TABLE t2;
sl@0
   587
    PRAGMA integrity_check;
sl@0
   588
    DROP TABLE t1;
sl@0
   589
    PRAGMA integrity_check;
sl@0
   590
  }
sl@0
   591
} {ok ok ok}
sl@0
   592
sl@0
   593
}
sl@0
   594
sl@0
   595
#---------------------------------------------------------------------
sl@0
   596
# Test cases autovacuum-7.X test the case where a page must be moved
sl@0
   597
# and the destination location collides with at least one other
sl@0
   598
# entry in the page hash-table (internal to the pager.c module. 
sl@0
   599
#
sl@0
   600
do_test autovacuum-7.1 {
sl@0
   601
  db close
sl@0
   602
  file delete -force test.db
sl@0
   603
  file delete -force test.db-journal
sl@0
   604
  sqlite3 db test.db
sl@0
   605
sl@0
   606
  execsql {
sl@0
   607
    PRAGMA auto_vacuum=1;
sl@0
   608
    CREATE TABLE t1(a, b, PRIMARY KEY(a, b));
sl@0
   609
    INSERT INTO t1 VALUES(randstr(400,400),randstr(400,400));
sl@0
   610
    INSERT INTO t1 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 2
sl@0
   611
    INSERT INTO t1 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 4
sl@0
   612
    INSERT INTO t1 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 8
sl@0
   613
    INSERT INTO t1 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 16
sl@0
   614
    INSERT INTO t1 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 32
sl@0
   615
  }
sl@0
   616
sl@0
   617
  expr {[file size test.db] / 1024}
sl@0
   618
} {73}
sl@0
   619
sl@0
   620
do_test autovacuum-7.2 {
sl@0
   621
  execsql {
sl@0
   622
    CREATE TABLE t2(a, b, PRIMARY KEY(a, b));
sl@0
   623
    INSERT INTO t2 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 2
sl@0
   624
    CREATE TABLE t3(a, b, PRIMARY KEY(a, b));
sl@0
   625
    INSERT INTO t3 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 2
sl@0
   626
    CREATE TABLE t4(a, b, PRIMARY KEY(a, b));
sl@0
   627
    INSERT INTO t4 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 2
sl@0
   628
    CREATE TABLE t5(a, b, PRIMARY KEY(a, b));
sl@0
   629
    INSERT INTO t5 SELECT randstr(400,400), randstr(400,400) FROM t1; -- 2
sl@0
   630
  }
sl@0
   631
  expr {[file size test.db] / 1024}
sl@0
   632
} {354}
sl@0
   633
sl@0
   634
do_test autovacuum-7.3 {
sl@0
   635
  db close
sl@0
   636
  sqlite3 db test.db
sl@0
   637
  execsql {
sl@0
   638
    BEGIN;
sl@0
   639
    DELETE FROM t4;
sl@0
   640
    COMMIT;
sl@0
   641
    SELECT count(*) FROM t1;
sl@0
   642
  }
sl@0
   643
  expr {[file size test.db] / 1024}
sl@0
   644
} {286}
sl@0
   645
sl@0
   646
#------------------------------------------------------------------------
sl@0
   647
# Additional tests.
sl@0
   648
#
sl@0
   649
# Try to determine the autovacuum setting for a database that is locked.
sl@0
   650
#
sl@0
   651
do_test autovacuum-8.1 {
sl@0
   652
  db close
sl@0
   653
  sqlite3 db test.db
sl@0
   654
  sqlite3 db2 test.db
sl@0
   655
  db eval {PRAGMA auto_vacuum}
sl@0
   656
} {1}
sl@0
   657
do_test autovacuum-8.2 {
sl@0
   658
  db eval {BEGIN EXCLUSIVE}
sl@0
   659
  catchsql {PRAGMA auto_vacuum} db2
sl@0
   660
} {1 {database is locked}}
sl@0
   661
catch {db2 close}
sl@0
   662
catch {db eval {COMMIT}}
sl@0
   663
    
sl@0
   664
sl@0
   665
finish_test