os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/incrblob.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
# 2007 May 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
#
sl@0
    12
# $Id: incrblob.test,v 1.21 2008/09/11 11:28:00 danielk1977 Exp $
sl@0
    13
#
sl@0
    14
sl@0
    15
set testdir [file dirname $argv0]
sl@0
    16
source $testdir/tester.tcl
sl@0
    17
sl@0
    18
ifcapable {!autovacuum || !pragma || !incrblob} {
sl@0
    19
  finish_test
sl@0
    20
  return
sl@0
    21
}
sl@0
    22
sl@0
    23
do_test incrblob-1.1 {
sl@0
    24
  execsql {
sl@0
    25
    CREATE TABLE blobs(k PRIMARY KEY, v BLOB);
sl@0
    26
    INSERT INTO blobs VALUES('one', X'0102030405060708090A');
sl@0
    27
    INSERT INTO blobs VALUES('two', X'0A090807060504030201');
sl@0
    28
  }
sl@0
    29
} {}
sl@0
    30
sl@0
    31
do_test incrblob-1.2.1 {
sl@0
    32
  set ::blob [db incrblob blobs v 1]
sl@0
    33
  string match incrblob_* $::blob
sl@0
    34
} {1}
sl@0
    35
unset -nocomplain data
sl@0
    36
do_test incrblob-1.2.2 {
sl@0
    37
  binary scan [read $::blob] c* data
sl@0
    38
  set data
sl@0
    39
} {1 2 3 4 5 6 7 8 9 10}
sl@0
    40
do_test incrblob-1.2.3 {
sl@0
    41
  seek $::blob 0
sl@0
    42
  puts -nonewline $::blob "1234567890"
sl@0
    43
  flush $::blob
sl@0
    44
} {}
sl@0
    45
do_test incrblob-1.2.4 {
sl@0
    46
  seek $::blob 0
sl@0
    47
  binary scan [read $::blob] c* data
sl@0
    48
  set data
sl@0
    49
} {49 50 51 52 53 54 55 56 57 48}
sl@0
    50
do_test incrblob-1.2.5 {
sl@0
    51
  close $::blob
sl@0
    52
} {}
sl@0
    53
do_test incrblob-1.2.6 {
sl@0
    54
  execsql {
sl@0
    55
    SELECT v FROM blobs WHERE rowid = 1;
sl@0
    56
  }
sl@0
    57
} {1234567890}
sl@0
    58
sl@0
    59
#--------------------------------------------------------------------
sl@0
    60
# Test cases incrblob-1.3.X check that it is possible to read and write
sl@0
    61
# regions of a blob that lie on overflow pages.
sl@0
    62
#
sl@0
    63
do_test incrblob-1.3.1 {
sl@0
    64
  set ::str "[string repeat . 10000]"
sl@0
    65
  execsql {
sl@0
    66
    INSERT INTO blobs(rowid, k, v) VALUES(3, 'three', $::str);
sl@0
    67
  }
sl@0
    68
} {}
sl@0
    69
sl@0
    70
do_test incrblob-1.3.2 {
sl@0
    71
  set ::blob [db incrblob blobs v 3]
sl@0
    72
  seek $::blob 8500
sl@0
    73
  read $::blob 10
sl@0
    74
} {..........}
sl@0
    75
do_test incrblob-1.3.3 {
sl@0
    76
  seek $::blob 8500
sl@0
    77
  puts -nonewline $::blob 1234567890
sl@0
    78
} {}
sl@0
    79
do_test incrblob-1.3.4 {
sl@0
    80
  seek $::blob 8496
sl@0
    81
  read $::blob 10
sl@0
    82
} {....123456}
sl@0
    83
do_test incrblob-1.3.10 {
sl@0
    84
  close $::blob
sl@0
    85
} {}
sl@0
    86
sl@0
    87
#------------------------------------------------------------------------
sl@0
    88
# incrblob-2.*: 
sl@0
    89
#
sl@0
    90
# Test that the following operations use ptrmap pages to reduce
sl@0
    91
# unnecessary reads:
sl@0
    92
#
sl@0
    93
#     * Reading near the end of a blob,
sl@0
    94
#     * Writing near the end of a blob, and
sl@0
    95
#     * SELECT a column value that is located on an overflow page.
sl@0
    96
#
sl@0
    97
proc nRead {db} {
sl@0
    98
  set bt [btree_from_db $db]
sl@0
    99
  db_enter $db
sl@0
   100
  array set stats [btree_pager_stats $bt]
sl@0
   101
  db_leave $db
sl@0
   102
  return $stats(read)
sl@0
   103
}
sl@0
   104
proc nWrite {db} {
sl@0
   105
  set bt [btree_from_db $db]
sl@0
   106
  db_enter $db
sl@0
   107
  array set stats [btree_pager_stats $bt]
sl@0
   108
  db_leave $db
sl@0
   109
  return $stats(write)
sl@0
   110
}
sl@0
   111
sl@0
   112
sqlite3_soft_heap_limit 0
sl@0
   113
sl@0
   114
foreach AutoVacuumMode [list 0 1] {
sl@0
   115
sl@0
   116
  if {$AutoVacuumMode>0} {
sl@0
   117
    ifcapable !autovacuum {
sl@0
   118
      break
sl@0
   119
    }
sl@0
   120
  }
sl@0
   121
sl@0
   122
  db close
sl@0
   123
  file delete -force test.db test.db-journal
sl@0
   124
sl@0
   125
  sqlite3 db test.db
sl@0
   126
  execsql "PRAGMA auto_vacuum = $AutoVacuumMode"
sl@0
   127
sl@0
   128
  do_test incrblob-2.$AutoVacuumMode.1 {
sl@0
   129
    set ::str [string repeat abcdefghij 2900]
sl@0
   130
    execsql {
sl@0
   131
      BEGIN;
sl@0
   132
      CREATE TABLE blobs(k PRIMARY KEY, v BLOB, i INTEGER);
sl@0
   133
      DELETE FROM blobs;
sl@0
   134
      INSERT INTO blobs VALUES('one', $::str || randstr(500,500), 45);
sl@0
   135
      COMMIT;
sl@0
   136
    }
sl@0
   137
    expr [file size test.db]/1024
sl@0
   138
  } [expr 31 + $AutoVacuumMode]
sl@0
   139
sl@0
   140
  ifcapable autovacuum {
sl@0
   141
    do_test incrblob-2.$AutoVacuumMode.2 {
sl@0
   142
      execsql {
sl@0
   143
        PRAGMA auto_vacuum;
sl@0
   144
      }
sl@0
   145
    } $AutoVacuumMode
sl@0
   146
  }
sl@0
   147
sl@0
   148
  do_test incrblob-2.$AutoVacuumMode.3 {
sl@0
   149
    # Open and close the db to make sure the page cache is empty.
sl@0
   150
    db close
sl@0
   151
    sqlite3 db test.db
sl@0
   152
  
sl@0
   153
    # Read the last 20 bytes of the blob via a blob handle.
sl@0
   154
    set ::blob [db incrblob blobs v 1]
sl@0
   155
    seek $::blob -20 end
sl@0
   156
    set ::fragment [read $::blob]
sl@0
   157
    close $::blob
sl@0
   158
  
sl@0
   159
    # If the database is not in auto-vacuum mode, the whole of
sl@0
   160
    # the overflow-chain must be scanned. In auto-vacuum mode,
sl@0
   161
    # sqlite uses the ptrmap pages to avoid reading the other pages.
sl@0
   162
    #
sl@0
   163
    nRead db
sl@0
   164
  } [expr $AutoVacuumMode ? 4 : 30]
sl@0
   165
sl@0
   166
  do_test incrblob-2.$AutoVacuumMode.4 {
sl@0
   167
    string range [db one {SELECT v FROM blobs}] end-19 end
sl@0
   168
  } $::fragment
sl@0
   169
sl@0
   170
  do_test incrblob-2.$AutoVacuumMode.5 {
sl@0
   171
    # Open and close the db to make sure the page cache is empty.
sl@0
   172
    db close
sl@0
   173
    sqlite3 db test.db
sl@0
   174
  
sl@0
   175
    # Write the second-to-last 20 bytes of the blob via a blob handle.
sl@0
   176
    #
sl@0
   177
    set ::blob [db incrblob blobs v 1]
sl@0
   178
    seek $::blob -40 end
sl@0
   179
    puts -nonewline $::blob "1234567890abcdefghij"
sl@0
   180
    flush $::blob
sl@0
   181
  
sl@0
   182
    # If the database is not in auto-vacuum mode, the whole of
sl@0
   183
    # the overflow-chain must be scanned. In auto-vacuum mode,
sl@0
   184
    # sqlite uses the ptrmap pages to avoid reading the other pages.
sl@0
   185
    #
sl@0
   186
    nRead db
sl@0
   187
  } [expr $AutoVacuumMode ? 4 : 30]
sl@0
   188
sl@0
   189
  # Pages 1 (the write-counter) and 32 (the blob data) were written.
sl@0
   190
  do_test incrblob-2.$AutoVacuumMode.6 {
sl@0
   191
    close $::blob
sl@0
   192
    nWrite db
sl@0
   193
  } 2
sl@0
   194
sl@0
   195
  do_test incrblob-2.$AutoVacuumMode.7 {
sl@0
   196
    string range [db one {SELECT v FROM blobs}] end-39 end-20
sl@0
   197
  } "1234567890abcdefghij"
sl@0
   198
sl@0
   199
  do_test incrblob-2.$AutoVacuumMode.8 {
sl@0
   200
    # Open and close the db to make sure the page cache is empty.
sl@0
   201
    db close
sl@0
   202
    sqlite3 db test.db
sl@0
   203
sl@0
   204
    execsql { SELECT i FROM blobs } 
sl@0
   205
  } {45}
sl@0
   206
sl@0
   207
  do_test incrblob-2.$AutoVacuumMode.9 {
sl@0
   208
    nRead db
sl@0
   209
  } [expr $AutoVacuumMode ? 4 : 30]
sl@0
   210
}
sl@0
   211
sqlite3_soft_heap_limit $soft_limit
sl@0
   212
sl@0
   213
#------------------------------------------------------------------------
sl@0
   214
# incrblob-3.*: 
sl@0
   215
#
sl@0
   216
# Test the outcome of trying to write to a read-only blob handle.
sl@0
   217
#
sl@0
   218
do_test incrblob-3.1 {
sl@0
   219
  set ::blob [db incrblob -readonly blobs v 1]
sl@0
   220
  seek $::blob -40 end
sl@0
   221
  read $::blob 20
sl@0
   222
} "1234567890abcdefghij"
sl@0
   223
do_test incrblob-3.2 {
sl@0
   224
  seek $::blob 0
sl@0
   225
  set rc [catch {
sl@0
   226
    puts -nonewline $::blob "helloworld"
sl@0
   227
  } msg]
sl@0
   228
  close $::blob
sl@0
   229
  list $rc $msg
sl@0
   230
} "1 {channel \"$::blob\" wasn't opened for writing}"
sl@0
   231
sl@0
   232
do_test incrblob-3.3 {
sl@0
   233
  set ::blob [db incrblob -readonly blobs v 1]
sl@0
   234
  seek $::blob -40 end
sl@0
   235
  read $::blob 20
sl@0
   236
} "1234567890abcdefghij"
sl@0
   237
do_test incrblob-3.4 {
sl@0
   238
  set rc [catch {
sl@0
   239
    sqlite3_blob_write $::blob 20 "qwertyuioplkjhgfds" 
sl@0
   240
  } msg]
sl@0
   241
  list $rc $msg
sl@0
   242
} {1 SQLITE_READONLY}
sl@0
   243
catch {close $::blob}
sl@0
   244
sl@0
   245
#------------------------------------------------------------------------
sl@0
   246
# incrblob-4.*: 
sl@0
   247
#
sl@0
   248
# Try a couple of error conditions:
sl@0
   249
#
sl@0
   250
#     4.1 - Attempt to open a row that does not exist.
sl@0
   251
#     4.2 - Attempt to open a column that does not exist.
sl@0
   252
#     4.3 - Attempt to open a table that does not exist.
sl@0
   253
#     4.4 - Attempt to open a database that does not exist.
sl@0
   254
#
sl@0
   255
#     4.5 - Attempt to open an integer
sl@0
   256
#     4.6 - Attempt to open a real value
sl@0
   257
#     4.7 - Attempt to open an SQL null
sl@0
   258
#
sl@0
   259
#     4.8 - Attempt to open an indexed column for writing
sl@0
   260
#     4.9 - Attempt to open an indexed column for reading (this works)
sl@0
   261
#
sl@0
   262
#     4.11 - Attempt to open a column of a view.
sl@0
   263
#     4.12 - Attempt to open a column of a virtual table.
sl@0
   264
#
sl@0
   265
do_test incrblob-4.1 {
sl@0
   266
  set rc [catch {
sl@0
   267
    set ::blob [db incrblob blobs v 2]
sl@0
   268
  } msg ] 
sl@0
   269
  list $rc $msg
sl@0
   270
} {1 {no such rowid: 2}}
sl@0
   271
do_test incrblob-4.2 {
sl@0
   272
  set rc [catch {
sl@0
   273
    set ::blob [db incrblob blobs blue 1]
sl@0
   274
  } msg ] 
sl@0
   275
  list $rc $msg
sl@0
   276
} {1 {no such column: "blue"}}
sl@0
   277
do_test incrblob-4.3 {
sl@0
   278
  set rc [catch {
sl@0
   279
    set ::blob [db incrblob nosuchtable blue 1]
sl@0
   280
  } msg ]
sl@0
   281
  list $rc $msg
sl@0
   282
} {1 {no such table: main.nosuchtable}}
sl@0
   283
do_test incrblob-4.4 {
sl@0
   284
  set rc [catch {
sl@0
   285
    set ::blob [db incrblob nosuchdb blobs v 1]
sl@0
   286
  } msg ] 
sl@0
   287
  list $rc $msg
sl@0
   288
} {1 {no such table: nosuchdb.blobs}}
sl@0
   289
sl@0
   290
do_test incrblob-4.5 {
sl@0
   291
  set rc [catch {
sl@0
   292
    set ::blob [db incrblob blobs i 1]
sl@0
   293
  } msg ] 
sl@0
   294
  list $rc $msg
sl@0
   295
} {1 {cannot open value of type integer}}
sl@0
   296
do_test incrblob-4.6 {
sl@0
   297
  execsql {
sl@0
   298
    INSERT INTO blobs(k, v, i) VALUES(123, 567.765, NULL);
sl@0
   299
  }
sl@0
   300
  set rc [catch {
sl@0
   301
    set ::blob [db incrblob blobs v 2]
sl@0
   302
  } msg ] 
sl@0
   303
  list $rc $msg
sl@0
   304
} {1 {cannot open value of type real}}
sl@0
   305
do_test incrblob-4.7 {
sl@0
   306
  set rc [catch {
sl@0
   307
    set ::blob [db incrblob blobs i 2]
sl@0
   308
  } msg ] 
sl@0
   309
  list $rc $msg
sl@0
   310
} {1 {cannot open value of type null}}
sl@0
   311
sl@0
   312
do_test incrblob-4.8 {
sl@0
   313
  execsql {
sl@0
   314
    INSERT INTO blobs(k, v, i) VALUES(X'010203040506070809', 'hello', 'world');
sl@0
   315
  }
sl@0
   316
  set rc [catch {
sl@0
   317
    set ::blob [db incrblob blobs k 3]
sl@0
   318
  } msg ] 
sl@0
   319
  list $rc $msg
sl@0
   320
} {1 {cannot open indexed column for writing}}
sl@0
   321
sl@0
   322
do_test incrblob-4.9.1 {
sl@0
   323
  set rc [catch {
sl@0
   324
    set ::blob [db incrblob -readonly blobs k 3]
sl@0
   325
  } msg]
sl@0
   326
} {0}
sl@0
   327
do_test incrblob-4.9.2 {
sl@0
   328
  binary scan [read $::blob] c* c
sl@0
   329
  close $::blob
sl@0
   330
  set c
sl@0
   331
} {1 2 3 4 5 6 7 8 9}
sl@0
   332
sl@0
   333
do_test incrblob-4.10 {
sl@0
   334
  set ::blob [db incrblob -readonly blobs k 3]
sl@0
   335
  set rc [catch { sqlite3_blob_read $::blob 10 100 } msg]
sl@0
   336
  list $rc $msg
sl@0
   337
} {1 SQLITE_ERROR}
sl@0
   338
do_test incrblob-4.10.2 {
sl@0
   339
  close $::blob
sl@0
   340
} {}
sl@0
   341
sl@0
   342
ifcapable view {
sl@0
   343
  do_test incrblob-4.11 {
sl@0
   344
    execsql { CREATE VIEW blobs_view AS SELECT k, v, i FROM blobs }
sl@0
   345
    set rc [catch { db incrblob blobs_view v 3 } msg]
sl@0
   346
    list $rc $msg
sl@0
   347
  } {1 {cannot open view: blobs_view}}
sl@0
   348
}
sl@0
   349
ifcapable vtab {
sl@0
   350
  register_echo_module [sqlite3_connection_pointer db]
sl@0
   351
  do_test incrblob-4.12 {
sl@0
   352
    execsql { CREATE VIRTUAL TABLE blobs_echo USING echo(blobs) }
sl@0
   353
    set rc [catch { db incrblob blobs_echo v 3 } msg]
sl@0
   354
    list $rc $msg
sl@0
   355
  } {1 {cannot open virtual table: blobs_echo}}
sl@0
   356
}
sl@0
   357
sl@0
   358
sl@0
   359
#------------------------------------------------------------------------
sl@0
   360
# incrblob-5.*: 
sl@0
   361
#
sl@0
   362
#     Test that opening a blob in an attached database works.
sl@0
   363
#
sl@0
   364
ifcapable attach {
sl@0
   365
  do_test incrblob-5.1 {
sl@0
   366
    file delete -force test2.db test2.db-journal
sl@0
   367
    set ::size [expr [file size [info script]]]
sl@0
   368
    execsql {
sl@0
   369
      ATTACH 'test2.db' AS aux;
sl@0
   370
      CREATE TABLE aux.files(name, text);
sl@0
   371
      INSERT INTO aux.files VALUES('this one', zeroblob($::size));
sl@0
   372
    }
sl@0
   373
    set fd  [db incrblob aux files text 1]
sl@0
   374
    fconfigure $fd -translation binary
sl@0
   375
    set fd2 [open [info script]]
sl@0
   376
    fconfigure $fd2 -translation binary
sl@0
   377
    puts -nonewline $fd [read $fd2]
sl@0
   378
    close $fd
sl@0
   379
    close $fd2
sl@0
   380
    set ::text [db one {select text from aux.files}]
sl@0
   381
    string length $::text
sl@0
   382
  } [file size [info script]]
sl@0
   383
  do_test incrblob-5.2 {
sl@0
   384
    set fd2 [open [info script]]
sl@0
   385
    fconfigure $fd2 -translation binary
sl@0
   386
    set ::data [read $fd2]
sl@0
   387
    close $fd2
sl@0
   388
    set ::data
sl@0
   389
  } $::text
sl@0
   390
}
sl@0
   391
sl@0
   392
# free memory
sl@0
   393
unset -nocomplain ::data
sl@0
   394
unset -nocomplain ::text
sl@0
   395
sl@0
   396
#------------------------------------------------------------------------
sl@0
   397
# incrblob-6.*: 
sl@0
   398
#
sl@0
   399
#     Test that opening a blob for write-access is impossible if
sl@0
   400
#     another connection has the database RESERVED lock.
sl@0
   401
#
sl@0
   402
#     Then test that blob writes that take place inside of a
sl@0
   403
#     transaction are not visible to external connections until
sl@0
   404
#     after the transaction is commited and the blob channel 
sl@0
   405
#     closed.
sl@0
   406
#
sl@0
   407
sqlite3_soft_heap_limit 0
sl@0
   408
do_test incrblob-6.1 {
sl@0
   409
  sqlite3 db2 test.db
sl@0
   410
  execsql {
sl@0
   411
    BEGIN;
sl@0
   412
    INSERT INTO blobs(k, v, i) VALUES('a', 'different', 'connection');
sl@0
   413
  } db2
sl@0
   414
} {}
sl@0
   415
do_test incrblob-6.2 {
sl@0
   416
  execsql {
sl@0
   417
    SELECT rowid FROM blobs
sl@0
   418
  }
sl@0
   419
} {1 2 3}
sl@0
   420
do_test incrblob-6.3 {
sl@0
   421
  set rc [catch {
sl@0
   422
    db incrblob blobs v 1
sl@0
   423
  } msg]
sl@0
   424
  list $rc $msg
sl@0
   425
} {1 {database is locked}}
sl@0
   426
do_test incrblob-6.4 {
sl@0
   427
  set rc [catch {
sl@0
   428
    db incrblob blobs v 3
sl@0
   429
  } msg]
sl@0
   430
  list $rc $msg
sl@0
   431
} {1 {database is locked}}
sl@0
   432
do_test incrblob-6.5 {
sl@0
   433
  set ::blob [db incrblob -readonly blobs v 3]
sl@0
   434
  read $::blob
sl@0
   435
} {hello}
sl@0
   436
do_test incrblob-6.6 {
sl@0
   437
  close $::blob
sl@0
   438
} {}
sl@0
   439
sl@0
   440
do_test incrblob-6.7 {
sl@0
   441
  set ::blob [db2 incrblob blobs i 4]
sl@0
   442
  gets $::blob
sl@0
   443
} {connection}
sl@0
   444
do_test incrblob-6.8 {
sl@0
   445
  tell $::blob
sl@0
   446
} {10}
sl@0
   447
do_test incrblob-6.9 {
sl@0
   448
  seek $::blob 0
sl@0
   449
  puts -nonewline $::blob "invocation"
sl@0
   450
  flush $::blob
sl@0
   451
} {}
sl@0
   452
sl@0
   453
# At this point rollback or commit should be illegal (because 
sl@0
   454
# there is an open blob channel).
sl@0
   455
do_test incrblob-6.10 {
sl@0
   456
  catchsql {
sl@0
   457
    ROLLBACK;
sl@0
   458
  } db2
sl@0
   459
} {1 {cannot rollback transaction - SQL statements in progress}}
sl@0
   460
do_test incrblob-6.11 {
sl@0
   461
  catchsql {
sl@0
   462
    COMMIT;
sl@0
   463
  } db2
sl@0
   464
} {1 {cannot commit transaction - SQL statements in progress}}
sl@0
   465
sl@0
   466
do_test incrblob-6.12 {
sl@0
   467
  execsql {
sl@0
   468
    SELECT * FROM blobs WHERE rowid = 4;
sl@0
   469
  }
sl@0
   470
} {}
sl@0
   471
do_test incrblob-6.13 {
sl@0
   472
  close $::blob
sl@0
   473
  execsql {
sl@0
   474
    COMMIT;
sl@0
   475
  } db2
sl@0
   476
} {}
sl@0
   477
do_test incrblob-6.14 {
sl@0
   478
  execsql {
sl@0
   479
    SELECT * FROM blobs WHERE rowid = 4;
sl@0
   480
  }
sl@0
   481
} {a different invocation}
sl@0
   482
db2 close
sl@0
   483
sqlite3_soft_heap_limit $soft_limit
sl@0
   484
sl@0
   485
#-----------------------------------------------------------------------
sl@0
   486
# The following tests verify the behaviour of the incremental IO
sl@0
   487
# APIs in the following cases:
sl@0
   488
#
sl@0
   489
#     7.1 A row that containing an open blob is modified.
sl@0
   490
#
sl@0
   491
#     7.2 A CREATE TABLE requires that an overflow page that is part
sl@0
   492
#         of an open blob is moved.
sl@0
   493
#
sl@0
   494
#     7.3 An INCREMENTAL VACUUM moves an overflow page that is part
sl@0
   495
#         of an open blob.
sl@0
   496
#
sl@0
   497
# In the first case above, correct behaviour is for all subsequent
sl@0
   498
# read/write operations on the blob-handle to return SQLITE_ABORT.
sl@0
   499
# More accurately, blob-handles are invalidated whenever the table
sl@0
   500
# they belong to is written to.
sl@0
   501
#
sl@0
   502
# The second two cases have no external effect. They are testing
sl@0
   503
# that the internal cache of overflow page numbers is correctly
sl@0
   504
# invalidated.
sl@0
   505
#
sl@0
   506
do_test incrblob-7.1.0 {
sl@0
   507
  execsql {
sl@0
   508
    BEGIN;
sl@0
   509
    DROP TABLE blobs;
sl@0
   510
    CREATE TABLE t1 (a, b, c, d BLOB);
sl@0
   511
    INSERT INTO t1(a, b, c, d) VALUES(1, 2, 3, 4);
sl@0
   512
    COMMIT;
sl@0
   513
  }
sl@0
   514
} {}
sl@0
   515
sl@0
   516
foreach {tn arg} {1 "" 2 -readonly} {
sl@0
   517
sl@0
   518
  execsql {
sl@0
   519
    UPDATE t1 SET d = zeroblob(10000);
sl@0
   520
  }
sl@0
   521
sl@0
   522
  do_test incrblob-7.1.$tn.1 {
sl@0
   523
    set ::b [eval db incrblob $arg t1 d 1]
sl@0
   524
    binary scan [sqlite3_blob_read $::b 5000 5] c* c
sl@0
   525
    set c
sl@0
   526
  } {0 0 0 0 0}
sl@0
   527
  do_test incrblob-7.1.$tn.2 {
sl@0
   528
    execsql {
sl@0
   529
      UPDATE t1 SET d = 15;
sl@0
   530
    }
sl@0
   531
  } {}
sl@0
   532
  do_test incrblob-7.1.$tn.3 {
sl@0
   533
    set rc [catch { sqlite3_blob_read $::b 5000 5 } msg]
sl@0
   534
    list $rc $msg
sl@0
   535
  } {1 SQLITE_ABORT}
sl@0
   536
  do_test incrblob-7.1.$tn.4 {
sl@0
   537
    execsql {
sl@0
   538
      SELECT d FROM t1;
sl@0
   539
    }
sl@0
   540
  } {15}
sl@0
   541
  do_test incrblob-7.1.$tn.5 {
sl@0
   542
    set rc [catch { close $::b } msg]
sl@0
   543
    list $rc $msg
sl@0
   544
  } {0 {}}
sl@0
   545
  do_test incrblob-7.1.$tn.6 {
sl@0
   546
    execsql {
sl@0
   547
      SELECT d FROM t1;
sl@0
   548
    }
sl@0
   549
  } {15}
sl@0
   550
sl@0
   551
}
sl@0
   552
sl@0
   553
set fd [open [info script]]
sl@0
   554
fconfigure $fd -translation binary
sl@0
   555
set ::data [read $fd 14000]
sl@0
   556
close $fd
sl@0
   557
sl@0
   558
db close
sl@0
   559
file delete -force test.db test.db-journal
sl@0
   560
sqlite3 db test.db
sl@0
   561
sl@0
   562
do_test incrblob-7.2.1 {
sl@0
   563
  execsql {
sl@0
   564
    PRAGMA auto_vacuum = "incremental";
sl@0
   565
    CREATE TABLE t1(a INTEGER PRIMARY KEY, b);        -- root@page3
sl@0
   566
    INSERT INTO t1 VALUES(123, $::data);
sl@0
   567
  }
sl@0
   568
  set ::b [db incrblob -readonly t1 b 123]
sl@0
   569
  fconfigure $::b -translation binary
sl@0
   570
  read $::b
sl@0
   571
} $::data
sl@0
   572
do_test incrblob-7.2.2 {
sl@0
   573
  execsql {
sl@0
   574
    CREATE TABLE t2(a INTEGER PRIMARY KEY, b);        -- root@page4
sl@0
   575
  }
sl@0
   576
  seek $::b 0
sl@0
   577
  read $::b
sl@0
   578
} $::data
sl@0
   579
do_test incrblob-7.2.3 {
sl@0
   580
  close $::b
sl@0
   581
  execsql {
sl@0
   582
    SELECT rootpage FROM sqlite_master;
sl@0
   583
  }
sl@0
   584
} {3 4}
sl@0
   585
sl@0
   586
set ::otherdata "[string range $::data 0 1000][string range $::data 1001 end]"
sl@0
   587
do_test incrblob-7.3.1 {
sl@0
   588
  execsql {
sl@0
   589
    INSERT INTO t2 VALUES(456, $::otherdata);
sl@0
   590
  }
sl@0
   591
  set ::b [db incrblob -readonly t2 b 456]
sl@0
   592
  fconfigure $::b -translation binary
sl@0
   593
  read $::b
sl@0
   594
} $::otherdata
sl@0
   595
do_test incrblob-7.3.2 {
sl@0
   596
  expr [file size test.db]/1024
sl@0
   597
} 30
sl@0
   598
do_test incrblob-7.3.3 {
sl@0
   599
  execsql {
sl@0
   600
    DELETE FROM t1 WHERE a = 123;
sl@0
   601
    PRAGMA INCREMENTAL_VACUUM(0);
sl@0
   602
  }
sl@0
   603
  seek $::b 0
sl@0
   604
  read $::b
sl@0
   605
} $::otherdata
sl@0
   606
sl@0
   607
# Attempt to write on a read-only blob.  Make sure the error code
sl@0
   608
# gets set.  Ticket #2464.
sl@0
   609
#
sl@0
   610
do_test incrblob-7.4 {
sl@0
   611
  set rc [catch {sqlite3_blob_write $::b 10 HELLO} msg]
sl@0
   612
  lappend rc $msg
sl@0
   613
} {1 SQLITE_READONLY}
sl@0
   614
do_test incrblob-7.5 {
sl@0
   615
  sqlite3_errcode db
sl@0
   616
} {SQLITE_READONLY}
sl@0
   617
do_test incrblob-7.6 {
sl@0
   618
  sqlite3_errmsg db
sl@0
   619
} {attempt to write a readonly database}
sl@0
   620
sl@0
   621
finish_test