os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/misc7.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
# 2006 September 4
sl@0
     2
#
sl@0
     3
# Portions Copyright (c) 2007-2010 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
# This file implements regression tests for SQLite library.
sl@0
    14
#
sl@0
    15
# $Id: misc7.test,v 1.24 2008/08/22 13:57:39 pweilbacher Exp $
sl@0
    16
sl@0
    17
set testdir [file dirname $argv0]
sl@0
    18
source $testdir/tester.tcl
sl@0
    19
sl@0
    20
do_test misc7-1-misuse {
sl@0
    21
  c_misuse_test
sl@0
    22
} {}
sl@0
    23
sl@0
    24
do_test misc7-2 {
sl@0
    25
  c_realloc_test
sl@0
    26
} {}
sl@0
    27
sl@0
    28
do_test misc7-3 {
sl@0
    29
  c_collation_test
sl@0
    30
} {}
sl@0
    31
sl@0
    32
# Try to open a directory:
sl@0
    33
# Symbian OS: '/' in the file name replaced with '\\'
sl@0
    34
do_test misc7-4 {
sl@0
    35
  file delete mydir
sl@0
    36
  file mkdir mydir
sl@0
    37
  set rc [catch {
sl@0
    38
    sqlite3 db2 .\\mydir
sl@0
    39
  } msg]
sl@0
    40
  list $rc $msg
sl@0
    41
} {1 {unable to open database file}}
sl@0
    42
sl@0
    43
# Try to open a file with a directory where its journal file should be.
sl@0
    44
# Symbian OS: '/' in the file name replaced with '\\'
sl@0
    45
do_test misc7-5 {
sl@0
    46
  file delete mydir
sl@0
    47
  file mkdir mydir-journal
sl@0
    48
  sqlite3 db2 .\\mydir
sl@0
    49
  catchsql {
sl@0
    50
    CREATE TABLE abc(a, b, c);
sl@0
    51
  } db2
sl@0
    52
} {1 {unable to open database file}}
sl@0
    53
db2 close
sl@0
    54
sl@0
    55
#--------------------------------------------------------------------
sl@0
    56
# The following tests, misc7-6.* test the libraries behaviour when
sl@0
    57
# it cannot open a file. To force this condition, we use up all the
sl@0
    58
# file-descriptors before running sqlite. This probably only works
sl@0
    59
# on unix.
sl@0
    60
#
sl@0
    61
sl@0
    62
proc use_up_files {} {
sl@0
    63
  set ret [list]
sl@0
    64
  catch {
sl@0
    65
    while 1 { lappend ret [open test.db] }
sl@0
    66
  }
sl@0
    67
  return $ret
sl@0
    68
}
sl@0
    69
sl@0
    70
proc do_fileopen_test {prefix sql} {
sl@0
    71
  set fd_list [use_up_files]
sl@0
    72
  set ::go 1
sl@0
    73
  set ::n 1
sl@0
    74
  set ::sql $sql
sl@0
    75
  while {$::go} {
sl@0
    76
    catch {db close}
sl@0
    77
    do_test ${prefix}.${::n} {
sl@0
    78
      set rc [catch {
sl@0
    79
        sqlite db test.db
sl@0
    80
        db eval $::sql
sl@0
    81
      } msg]
sl@0
    82
      if {$rc == 0} {set ::go 0}
sl@0
    83
  
sl@0
    84
      expr {$rc == 0 || ($rc == 1 && [string first unable $msg]==0)}
sl@0
    85
    } 1
sl@0
    86
  
sl@0
    87
    close [lindex $fd_list 0]
sl@0
    88
    set fd_list [lrange $fd_list 1 end]
sl@0
    89
    incr ::n
sl@0
    90
  }
sl@0
    91
  foreach fd $fd_list {
sl@0
    92
    close $fd
sl@0
    93
  }
sl@0
    94
  db close
sl@0
    95
}
sl@0
    96
sl@0
    97
execsql { CREATE TABLE abc(a PRIMARY KEY, b, c); }
sl@0
    98
db close
sl@0
    99
sl@0
   100
if {$tcl_platform(platform)!="windows" && $tcl_platform(platform)!="symbian"} {
sl@0
   101
  do_fileopen_test misc7-6.1 {
sl@0
   102
    BEGIN;
sl@0
   103
    INSERT INTO abc VALUES(1, 2, 3);
sl@0
   104
    INSERT INTO abc VALUES(2, 3, 4);
sl@0
   105
    INSERT INTO abc SELECT a+2, b, c FROM abc;
sl@0
   106
    COMMIT;
sl@0
   107
  }
sl@0
   108
  
sl@0
   109
  do_fileopen_test misc7-6.2 {
sl@0
   110
    PRAGMA temp.cache_size = 1000;
sl@0
   111
  }
sl@0
   112
}
sl@0
   113
sl@0
   114
#
sl@0
   115
# End of tests for out-of-file-descriptors condition.
sl@0
   116
#--------------------------------------------------------------------
sl@0
   117
sl@0
   118
sqlite3 db test.db
sl@0
   119
execsql {
sl@0
   120
  DELETE FROM abc;
sl@0
   121
  INSERT INTO abc VALUES(1, 2, 3);
sl@0
   122
  INSERT INTO abc VALUES(2, 3, 4);
sl@0
   123
  INSERT INTO abc SELECT a+2, b, c FROM abc;
sl@0
   124
}
sl@0
   125
  
sl@0
   126
sl@0
   127
#--------------------------------------------------------------------
sl@0
   128
# Test that the sqlite3_busy_timeout call seems to delay approximately
sl@0
   129
# the right amount of time.
sl@0
   130
#
sl@0
   131
do_test misc7-7.0 {
sl@0
   132
  sqlite3 db2 test.db
sl@0
   133
  sqlite3_busy_timeout [sqlite3_connection_pointer db] 2000
sl@0
   134
  execsql {
sl@0
   135
    BEGIN EXCLUSIVE;
sl@0
   136
  } db2
sl@0
   137
sl@0
   138
  # Now db2 has an exclusive lock on the database file, and db has
sl@0
   139
  # a busy-timeout of 2000 milliseconds. So check that trying to
sl@0
   140
  # access the database using connection db delays for at least 1500 ms.
sl@0
   141
  #
sl@0
   142
  set tm [time {
sl@0
   143
    set result [catchsql {
sl@0
   144
        SELECT * FROM sqlite_master;
sl@0
   145
      } db]
sl@0
   146
  }]
sl@0
   147
  set delay [lindex $tm 0]  ;# In microseconds
sl@0
   148
  lappend result [expr {$delay>1500000 && $delay<4000000}]
sl@0
   149
} {1 {database is locked} 1}
sl@0
   150
db2 close
sl@0
   151
sl@0
   152
#--------------------------------------------------------------------
sl@0
   153
# Test that nothing goes horribly wrong when attaching a database
sl@0
   154
# after the omit_readlock pragma has been exercised.
sl@0
   155
#
sl@0
   156
do_test misc7-7.1 {
sl@0
   157
  file delete -force test2.db
sl@0
   158
  file delete -force test2.db-journal
sl@0
   159
  execsql {
sl@0
   160
    PRAGMA omit_readlock = 1;
sl@0
   161
    ATTACH 'test2.db' AS aux;
sl@0
   162
    CREATE TABLE aux.hello(world);
sl@0
   163
    SELECT name FROM aux.sqlite_master;
sl@0
   164
  }
sl@0
   165
} {hello}
sl@0
   166
do_test misc7-7.2 {
sl@0
   167
  execsql {
sl@0
   168
    DETACH aux;
sl@0
   169
  }
sl@0
   170
} {}
sl@0
   171
sl@0
   172
# Test the UTF-16 version of the "out of memory" message (used when
sl@0
   173
# malloc fails during sqlite3_open() ).
sl@0
   174
#
sl@0
   175
ifcapable utf16 {
sl@0
   176
  do_test misc7-8 {
sl@0
   177
    encoding convertfrom unicode [sqlite3_errmsg16 0x00000000]
sl@0
   178
  } {out of memory}
sl@0
   179
}
sl@0
   180
sl@0
   181
do_test misc7-9 {
sl@0
   182
  execsql {
sl@0
   183
    SELECT * 
sl@0
   184
    FROM (SELECT name+1 AS one FROM sqlite_master LIMIT 1 OFFSET 1) 
sl@0
   185
    WHERE one LIKE 'hello%';
sl@0
   186
  }
sl@0
   187
} {}
sl@0
   188
sl@0
   189
#--------------------------------------------------------------------
sl@0
   190
# Improve coverage for vtab code.
sl@0
   191
#
sl@0
   192
ifcapable vtab {
sl@0
   193
  # Run some debug code to improve reported coverage
sl@0
   194
  #
sl@0
   195
sl@0
   196
  # set sqlite_where_trace 1
sl@0
   197
  do_test misc7-10 {
sl@0
   198
    register_echo_module [sqlite3_connection_pointer db]
sl@0
   199
    execsql {
sl@0
   200
      CREATE VIRTUAL TABLE t1 USING echo(abc);
sl@0
   201
      SELECT a FROM t1 WHERE a = 1 ORDER BY b;
sl@0
   202
    }
sl@0
   203
  } {1}
sl@0
   204
  set sqlite_where_trace 0
sl@0
   205
sl@0
   206
  # Specify an ORDER BY clause that cannot be indexed.
sl@0
   207
  do_test misc7-11 {
sl@0
   208
    execsql {
sl@0
   209
      SELECT t1.a, t2.a FROM t1, t1 AS t2 ORDER BY 2 LIMIT 1;
sl@0
   210
    }
sl@0
   211
  } {1 1}
sl@0
   212
sl@0
   213
  # The whole point of this is to test an error code other than
sl@0
   214
  # SQLITE_NOMEM from the vtab xBestIndex callback.
sl@0
   215
  #
sl@0
   216
  do_ioerr_test misc7-12 -tclprep {
sl@0
   217
    sqlite3 db2 test.db
sl@0
   218
    register_echo_module [sqlite3_connection_pointer db2]
sl@0
   219
    db2 eval {
sl@0
   220
      CREATE TABLE abc(a PRIMARY KEY, b, c);
sl@0
   221
      INSERT INTO abc VALUES(1, 2, 3);
sl@0
   222
      CREATE VIRTUAL TABLE t1 USING echo(abc);
sl@0
   223
    }
sl@0
   224
    db2 close
sl@0
   225
  } -tclbody {
sl@0
   226
    register_echo_module [sqlite3_connection_pointer db]
sl@0
   227
    execsql {SELECT * FROM t1 WHERE a = 1;}
sl@0
   228
  } 
sl@0
   229
sl@0
   230
  # The case where the virtual table module returns a very large number
sl@0
   231
  # as the cost of a scan (greater than SQLITE_BIG_DOUBLE in the code).
sl@0
   232
  #
sl@0
   233
  do_test misc7-13 {
sl@0
   234
    sqlite3 db test.db
sl@0
   235
    register_echo_module [sqlite3_connection_pointer db]
sl@0
   236
    set ::echo_module_cost 2.0e+99
sl@0
   237
    execsql {SELECT * FROM t1 WHERE a = 1;}
sl@0
   238
  } {1 2 3}
sl@0
   239
  unset ::echo_module_cost
sl@0
   240
}
sl@0
   241
sl@0
   242
db close
sl@0
   243
file delete -force test.db
sl@0
   244
file delete -force test.db-journal
sl@0
   245
sqlite3 db test.db
sl@0
   246
sl@0
   247
ifcapable explain {
sl@0
   248
  do_test misc7-14.1 {
sl@0
   249
    execsql {
sl@0
   250
      CREATE TABLE abc(a PRIMARY KEY, b, c);
sl@0
   251
    }
sl@0
   252
    execsql {
sl@0
   253
      EXPLAIN QUERY PLAN SELECT * FROM abc AS t2 WHERE rowid = 1;
sl@0
   254
    }
sl@0
   255
  } {0 0 {TABLE abc AS t2 USING PRIMARY KEY}}
sl@0
   256
  do_test misc7-14.2 {
sl@0
   257
    execsql {
sl@0
   258
      EXPLAIN QUERY PLAN SELECT * FROM abc AS t2 WHERE a = 1;
sl@0
   259
    }
sl@0
   260
  } {0 0 {TABLE abc AS t2 WITH INDEX sqlite_autoindex_abc_1}}
sl@0
   261
  do_test misc7-14.3 {
sl@0
   262
    execsql {
sl@0
   263
      EXPLAIN QUERY PLAN SELECT * FROM abc AS t2 ORDER BY a;
sl@0
   264
    }
sl@0
   265
  } {0 0 {TABLE abc AS t2 WITH INDEX sqlite_autoindex_abc_1 ORDER BY}}
sl@0
   266
}
sl@0
   267
sl@0
   268
db close
sl@0
   269
file delete -force test.db
sl@0
   270
file delete -force test.db-journal
sl@0
   271
sqlite3 db test.db
sl@0
   272
sl@0
   273
#--------------------------------------------------------------------
sl@0
   274
# This is all to force the pager_remove_from_stmt_list() function
sl@0
   275
# (inside pager.c) to remove a pager from the middle of the
sl@0
   276
# statement-list.
sl@0
   277
#
sl@0
   278
do_test misc7-15.1 {
sl@0
   279
  execsql {
sl@0
   280
    PRAGMA cache_size = 10;
sl@0
   281
    BEGIN;
sl@0
   282
    CREATE TABLE abc(a PRIMARY KEY, b, c);
sl@0
   283
    INSERT INTO abc 
sl@0
   284
    VALUES(randstr(100,100), randstr(100,100), randstr(100,100));
sl@0
   285
    INSERT INTO abc SELECT 
sl@0
   286
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   287
    INSERT INTO abc SELECT 
sl@0
   288
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   289
    INSERT INTO abc SELECT 
sl@0
   290
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   291
    INSERT INTO abc SELECT 
sl@0
   292
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   293
    INSERT INTO abc SELECT 
sl@0
   294
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   295
    INSERT INTO abc SELECT 
sl@0
   296
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   297
    INSERT INTO abc SELECT 
sl@0
   298
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   299
    INSERT INTO abc SELECT 
sl@0
   300
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   301
    COMMIT;
sl@0
   302
  }
sl@0
   303
  expr {[file size test.db]>10240}
sl@0
   304
} {1}
sl@0
   305
do_test misc7-15.2 {
sl@0
   306
  execsql {
sl@0
   307
    DELETE FROM abc WHERE rowid > 12;
sl@0
   308
    INSERT INTO abc SELECT 
sl@0
   309
            randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   310
  }
sl@0
   311
} {}
sl@0
   312
sl@0
   313
db close
sl@0
   314
file delete -force test.db
sl@0
   315
file delete -force test.db-journal
sl@0
   316
sqlite3 db test.db
sl@0
   317
sl@0
   318
do_ioerr_test misc7-16 -sqlprep {
sl@0
   319
   PRAGMA cache_size = 10;
sl@0
   320
   PRAGMA default_cache_size = 10;
sl@0
   321
   CREATE TABLE t3(a, b, UNIQUE(a, b));
sl@0
   322
   INSERT INTO t3 VALUES( randstr(100, 100), randstr(100, 100) );
sl@0
   323
   INSERT INTO t3 SELECT randstr(100, 100), randstr(100, 100) FROM t3;
sl@0
   324
   INSERT INTO t3 SELECT randstr(100, 100), randstr(100, 100) FROM t3;
sl@0
   325
   INSERT INTO t3 SELECT randstr(100, 100), randstr(100, 100) FROM t3;
sl@0
   326
   INSERT INTO t3 SELECT randstr(100, 100), randstr(100, 100) FROM t3;
sl@0
   327
   INSERT INTO t3 SELECT randstr(100, 100), randstr(100, 100) FROM t3;
sl@0
   328
   UPDATE t3 
sl@0
   329
   SET b = 'hello world'
sl@0
   330
   WHERE rowid >= (SELECT max(rowid)-1 FROM t3);
sl@0
   331
} -tclbody {
sl@0
   332
  set rc [catch {db eval {
sl@0
   333
    BEGIN;
sl@0
   334
      PRAGMA cache_size = 10;
sl@0
   335
      INSERT INTO t3 VALUES( randstr(100, 100), randstr(100, 100) );
sl@0
   336
      UPDATE t3 SET a = b;
sl@0
   337
    COMMIT;
sl@0
   338
  }} msg]
sl@0
   339
sl@0
   340
  if {!$rc || ($rc && [string first "columns" $msg]==0)} {
sl@0
   341
    set msg
sl@0
   342
  } else {
sl@0
   343
    error $msg
sl@0
   344
  }
sl@0
   345
}
sl@0
   346
sl@0
   347
sqlite3 db test.db
sl@0
   348
sl@0
   349
do_test misc7-16.X {
sl@0
   350
  execsql {
sl@0
   351
    SELECT count(*) FROM t3;
sl@0
   352
  }
sl@0
   353
} {32}
sl@0
   354
sl@0
   355
set sqlite_pager_n_sort_bucket 4
sl@0
   356
do_test misc7-17 {
sl@0
   357
  execsql {
sl@0
   358
    PRAGMA integrity_check;
sl@0
   359
    VACUUM;
sl@0
   360
    PRAGMA integrity_check;
sl@0
   361
  }
sl@0
   362
} {ok ok}
sl@0
   363
set sqlite_pager_n_sort_bucket 0
sl@0
   364
sl@0
   365
#----------------------------------------------------------------------
sl@0
   366
# Test the situation where a hot-journal is discovered but write-access
sl@0
   367
# to it is denied. This should return SQLITE_BUSY.
sl@0
   368
#
sl@0
   369
# These tests do not work on windows due to restrictions in the
sl@0
   370
# windows file system.
sl@0
   371
#
sl@0
   372
# TODO: Not clear why this test is crashing tclsqlite3.exe
sl@0
   373
#
sl@0
   374
if {$tcl_platform(platform)!="windows" && $tcl_platform(platform)!="symbian" && $tcl_platform(platform)!="os2"} {
sl@0
   375
sl@0
   376
  # Some network filesystems (ex: AFP) do not support setting read-only
sl@0
   377
  # permissions.  Only run these tests if full unix permission setting
sl@0
   378
  # capabilities are supported.
sl@0
   379
  #
sl@0
   380
  file attributes test.db -permissions rw-r--r--
sl@0
   381
  if {[file attributes test.db -permissions]==0644} {
sl@0
   382
sl@0
   383
    do_test misc7-17.1 {
sl@0
   384
      execsql {
sl@0
   385
        BEGIN;
sl@0
   386
        DELETE FROM t3 WHERE (oid%3)==0;
sl@0
   387
      }
sl@0
   388
      copy_file test.db bak.db
sl@0
   389
      copy_file test.db-journal bak.db-journal
sl@0
   390
      execsql {
sl@0
   391
        COMMIT;
sl@0
   392
      }
sl@0
   393
    
sl@0
   394
      db close
sl@0
   395
      copy_file bak.db test.db
sl@0
   396
      copy_file bak.db-journal test.db-journal
sl@0
   397
      sqlite3 db test.db
sl@0
   398
    
sl@0
   399
      catch {file attributes test.db-journal -permissions r--------}
sl@0
   400
      catch {file attributes test.db-journal -readonly 1}
sl@0
   401
      catchsql {
sl@0
   402
        SELECT count(*) FROM t3;
sl@0
   403
      }
sl@0
   404
    } {1 {database is locked}}
sl@0
   405
    do_test misc7-17.2 {
sl@0
   406
      # Note that the -readonly flag must be cleared before the -permissions
sl@0
   407
      # are set. Otherwise, when using tcl 8.5 on mac, the fact that the 
sl@0
   408
      # -readonly flag is set causes the attempt to set the permissions
sl@0
   409
      # to fail.
sl@0
   410
      catch {file attributes test.db-journal -readonly 0}
sl@0
   411
      catch {file attributes test.db-journal -permissions rw-------}
sl@0
   412
      catchsql {
sl@0
   413
        SELECT count(*) FROM t3;
sl@0
   414
      }
sl@0
   415
    } {0 32}
sl@0
   416
    
sl@0
   417
    set ::pending_byte_page [expr ($::sqlite_pending_byte / 1024) + 1]
sl@0
   418
    do_test misc7-17.3 {
sl@0
   419
      db eval {
sl@0
   420
        pragma writable_schema = true;
sl@0
   421
        UPDATE sqlite_master 
sl@0
   422
          SET rootpage = $pending_byte_page
sl@0
   423
          WHERE type = 'table' AND name = 't3';
sl@0
   424
      }
sl@0
   425
      execsql {
sl@0
   426
        SELECT rootpage FROM sqlite_master WHERE type = 'table' AND name = 't3';
sl@0
   427
      }
sl@0
   428
    } $::pending_byte_page
sl@0
   429
    
sl@0
   430
    do_test misc7-17.4 {
sl@0
   431
      db close
sl@0
   432
      sqlite3 db test.db
sl@0
   433
      catchsql {
sl@0
   434
        SELECT count(*) FROM t3;
sl@0
   435
      } 
sl@0
   436
    } {1 {database disk image is malformed}}
sl@0
   437
  }
sl@0
   438
}
sl@0
   439
sl@0
   440
# Ticket #2470
sl@0
   441
#
sl@0
   442
do_test misc7-18.1 {
sl@0
   443
  execsql {
sl@0
   444
    CREATE TABLE table_1 (col_10);
sl@0
   445
    CREATE TABLE table_2 (
sl@0
   446
      col_1, col_2, col_3, col_4, col_5,
sl@0
   447
      col_6, col_7, col_8, col_9, col_10
sl@0
   448
    );
sl@0
   449
    SELECT a.col_10
sl@0
   450
    FROM
sl@0
   451
      (SELECT table_1.col_10 AS col_10 FROM table_1) a,
sl@0
   452
      (SELECT table_1.col_10, table_2.col_9 AS qcol_9
sl@0
   453
         FROM table_1, table_2
sl@0
   454
        GROUP BY table_1.col_10, qcol_9);
sl@0
   455
  }
sl@0
   456
} {}
sl@0
   457
sl@0
   458
# Testing boundary conditions on sqlite3_status()
sl@0
   459
#
sl@0
   460
do_test misc7-19.1 {
sl@0
   461
  sqlite3_status -1 0
sl@0
   462
} {21 0 0}
sl@0
   463
do_test misc7-19.2 {
sl@0
   464
  sqlite3_status 1000 0
sl@0
   465
} {21 0 0}
sl@0
   466
sl@0
   467
sl@0
   468
# sqlite3_global_recover() is a no-op.  But we might as well test it
sl@0
   469
# if only to get the test coverage.
sl@0
   470
#
sl@0
   471
do_test misc7-20.1 {
sl@0
   472
  sqlite3_global_recover
sl@0
   473
} {SQLITE_OK}
sl@0
   474
sl@0
   475
sl@0
   476
db close
sl@0
   477
file delete -force test.db
sl@0
   478
sl@0
   479
finish_test