libdb/test/tcl/test096.tcl

399 lines
13 KiB
Tcl
Raw Permalink Normal View History

# See the file LICENSE for redistribution information.
#
2012-11-14 20:13:24 +00:00
# Copyright (c) 1999, 2012 Oracle and/or its affiliates. All rights reserved.
#
# $Id$
#
# TEST test096
# TEST Db->truncate test.
# TEST For all methods:
# TEST Test that truncate empties an existing database.
# TEST Test that truncate-write in an aborted txn doesn't
# TEST change the original contents.
# TEST Test that truncate-write in a committed txn does
# TEST overwrite the original contents.
# TEST For btree and hash, do the same in a database with offpage dups.
proc test096 { method {pagesize 512} {nentries 1000} {ndups 19} args} {
global fixed_len
global alphabet
source ./include.tcl
set orig_tdir $testdir
set orig_fixed_len $fixed_len
set args [convert_args $method $args]
set encargs ""
set args [split_encargs $args encargs]
set omethod [convert_method $method]
if { [is_partitioned $args] == 1 } {
set nodump 1
} else {
set nodump 0
}
puts "Test096: $method db truncate method test"
set pgindex [lsearch -exact $args "-pagesize"]
if { $pgindex != -1 } {
puts "Test096: Skipping for specific pagesizes"
return
}
# Create the database and open the dictionary
set eindex [lsearch -exact $args "-env"]
set testfile test096.db
if { $eindex != -1 } {
incr eindex
set env [lindex $args $eindex]
set txnenv [is_txnenv $env]
if { $txnenv == 0 } {
puts "Environment w/o txns specified; skipping."
return
}
if { $nentries == 1000 } {
set nentries 100
}
reduce_dups nentries ndups
set testdir [get_home $env]
set closeenv 0
} else {
env_cleanup $testdir
# We need an env for exclusive-use testing. Since we are
# using txns, we need at least 1 lock per record for queue.
# We need more locks for heap, because of the aux databases.
set lockmax [expr $nentries * 3]
if { [is_heap $method] } {
set lockmax [expr $lockmax * 3]
}
set env [eval {berkdb_env -create -home $testdir \
-lock_max_locks $lockmax -lock_max_objects $lockmax \
-pagesize $pagesize -txn} $encargs]
error_check_good env_create [is_valid_env $env] TRUE
set closeenv 1
}
set t1 $testdir/t1
puts "\tTest096.a: Create database with $nentries entries"
set db [eval {berkdb_open -create -auto_commit \
-env $env $omethod -mode 0644} $args $testfile]
error_check_good db_open [is_valid_db $db] TRUE
t96_populate $db $omethod $env $nentries
error_check_good dbclose [$db close] 0
puts "\tTest096.b: Truncate database"
set dbtr [eval {berkdb_open -create -auto_commit \
-env $env $omethod -mode 0644} $args $testfile]
error_check_good db_open [is_valid_db $dbtr] TRUE
set ret [$dbtr truncate]
error_check_good dbtrunc $ret $nentries
error_check_good db_close [$dbtr close] 0
set db [eval {berkdb_open -env $env} $args $testfile]
error_check_good dbopen [is_valid_db $db] TRUE
set number [number_of_entries $db $method]
error_check_good number_of_entries $number 0
error_check_good dbclose [$db close] 0
error_check_good dbverify [verify_dir $testdir "\tTest096.c: " 0 0 $nodump] 0
# Remove and recreate database.
puts "\tTest096.d: Recreate database with $nentries entries"
set db [eval {berkdb_open -create -auto_commit \
-env $env $omethod -mode 0644} $args {$testfile}]
error_check_good db_open [is_valid_db $db] TRUE
t96_populate $db $omethod $env $nentries
error_check_good dbclose [$db close] 0
puts "\tTest096.e: Truncate and write in a txn, then abort"
txn_truncate $env $omethod $args $testfile $nentries abort 1
set db [eval {berkdb_open -env $env} $args $testfile]
error_check_good dbopen [is_valid_db $db] TRUE
# Database should have original contents since both the truncate
# and the write were aborted
set number [number_of_entries $db $method]
error_check_good number_of_entries $number $nentries
error_check_good dbclose [$db close] 0
error_check_good dbverify [verify_dir $testdir "\tTest096.f: " 0 0 $nodump] 0
puts "\tTest096.g: Truncate and write in a txn, then commit"
txn_truncate $env $omethod $args $testfile $nentries commit 1
set db [eval {berkdb_open -env $env} $args $testfile]
error_check_good dbopen [is_valid_db $db] TRUE
# Database should contain only the new items
set number [number_of_entries $db $method]
error_check_good number_of_entries $number [expr $nentries / 2]
error_check_good dbclose [$db close] 0
error_check_good dbverify [verify_dir $testdir "\tTest096.h: " 0 0 $nodump] 0
puts "\tTest096.i: Check proper handling of overflow pages."
# Large keys and data compared to page size guarantee
# overflow pages, or split records in heap.
if { [is_fixed_length $method] == 1 } {
puts "Skipping overflow test for fixed-length method."
} else {
set overflowfile overflow096.db
set data [repeat $alphabet 600]
set db [eval {berkdb_open -create -auto_commit -pagesize 512 \
-env $env $omethod -mode 0644} $args $overflowfile]
error_check_good db_open [is_valid_db $db] TRUE
set noverflows 100
for { set i 1 } { $i <= $noverflows } { incr i } {
set ret [eval {$db put} \
$i [chop_data $method "$i$data"]]
}
# Hash reports pages of type P_OVERFLOW as "big pages", other
# access methods as "overflow pages". Heap doesn't use
# P_OVERFLOW pages.
if { [is_hash $method] == 1 } {
set bigpages [stat_field $db stat "Number of big pages"]
error_check_good stat:bigpages [expr $bigpages > 0] 1
} elseif { [is_heap $method] == 0 } {
set overflow [stat_field $db stat "Overflow pages"]
error_check_good stat:overflow [expr $overflow > 0] 1
}
error_check_good overflow_truncate [$db truncate] $noverflows
error_check_good overflow_close [$db close] 0
}
# Remove database and create a new one with unsorted dups.
# Skip the rest of the test for methods not supporting dups
# and for compression, which does not support unsorted dups.
#
if { [is_record_based $method] == 1 || \
[is_compressed $args] == 1 || \
[is_rbtree $method] == 1 } {
puts "Skipping remainder of test096."
if { $closeenv == 1 } {
error_check_good envclose [$env close] 0
}
return
}
set ret [berkdb dbremove -env $env -auto_commit $testfile]
set ret [berkdb dbremove -env $env -auto_commit $overflowfile]
puts "\tTest096.j: Create $nentries entries with $ndups duplicates"
set db [eval {berkdb_open -pagesize $pagesize -dup -auto_commit \
-create -env $env $omethod -mode 0644} $args $testfile]
error_check_good db_open [is_valid_db $db] TRUE
t96_populate $db $omethod $env $nentries $ndups
set dlist ""
for { set i 1 } {$i <= $ndups} {incr i} {
lappend dlist $i
}
set t [$env txn]
error_check_good txn [is_valid_txn $t $env] TRUE
set txn "-txn $t"
dup_check $db $txn $t1 $dlist
error_check_good txn [$t commit] 0
# Make sure there are duplicates.
puts "\tTest096.k: Verify off page duplicates status"
set duplicate [stat_field $db stat "Duplicate pages"]
error_check_good stat:offpage_dups [expr $duplicate > 0] 1
set recs [expr $ndups * $nentries]
error_check_good dbclose [$db close] 0
puts "\tTest096.l: Truncate database in a txn then abort"
txn_truncate $env $omethod $args $testfile $recs abort
set db [eval {berkdb_open -auto_commit -env $env} $args $testfile]
error_check_good dbopen [is_valid_db $db] TRUE
set number [number_of_entries $db $method]
error_check_good number_of_entries $number $recs
error_check_good dbclose [$db close] 0
puts "\tTest096.m: Truncate database in a txn then commit"
txn_truncate $env $omethod $args $testfile $recs commit
set db [eval {berkdb_open -auto_commit -env $env} $args {$testfile}]
error_check_good dbopen [is_valid_db $db] TRUE
set number [number_of_entries $db $method]
error_check_good number_of_entries $number 0
error_check_good dbclose [$db close] 0
set testdir [get_home $env]
error_check_good dbverify \
[verify_dir $testdir "\tTest096.n: " 0 0 $nodump] 0
# Remove database, and create a new one with dups. Test
# truncate + write within a transaction.
puts "\tTest096.o: Create $nentries entries with $ndups duplicates"
set ret [berkdb dbremove -env $env -auto_commit $testfile]
set db [eval {berkdb_open -pagesize $pagesize -dup -auto_commit \
-create -env $env $omethod -mode 0644} $args $testfile]
error_check_good db_open [is_valid_db $db] TRUE
t96_populate $db $omethod $env $nentries $ndups
set dlist ""
for { set i 1 } {$i <= $ndups} {incr i} {
lappend dlist $i
}
set t [$env txn]
error_check_good txn [is_valid_txn $t $env] TRUE
set txn "-txn $t"
dup_check $db $txn $t1 $dlist
error_check_good txn [$t commit] 0
puts "\tTest096.p: Verify off page duplicates status"
set duplicate [stat_field $db stat "Duplicate pages"]
error_check_good stat:offpage [expr $duplicate > 0] 1
set recs [expr $ndups * $nentries]
error_check_good dbclose [$db close] 0
puts "\tTest096.q: Truncate and write in a txn, then abort"
txn_truncate $env $omethod $args $testfile $recs abort 1
set db [eval {berkdb_open -auto_commit -env $env} $args $testfile]
error_check_good dbopen [is_valid_db $db] TRUE
set number [number_of_entries $db $method]
error_check_good number_of_entries $number $recs
error_check_good dbclose [$db close] 0
puts "\tTest096.r: Truncate and write in a txn, then commit"
txn_truncate $env $omethod $args $testfile $recs commit 1
set db [eval {berkdb_open -auto_commit -env $env} $args {$testfile}]
error_check_good dbopen [is_valid_db $db] TRUE
set number [number_of_entries $db $method]
error_check_good number_of_entries $number [expr $recs / 2]
error_check_good dbclose [$db close] 0
puts "\tTest096.s: Check overflow pages with dups."
set ndups 3
set db [eval {berkdb_open -create -auto_commit -pagesize 512 \
-env $env $omethod -dup -mode 0644} $args $overflowfile]
error_check_good db_open [is_valid_db $db] TRUE
for { set i 1 } { $i <= $noverflows } { incr i } {
for { set j 0 } { $j < $ndups } { incr j } {
set ret [eval {$db put} \
$i [chop_data $method "$i.$j$data"]]
}
}
# Hash reports pages of type P_OVERFLOW as "big pages", other
# access methods as "overflow pages".
if { [is_hash $method] == 1 } {
set bigpages [stat_field $db stat "Number of big pages"]
error_check_good stat:bigpages [expr $bigpages > 0] 1
} else {
set overflow [stat_field $db stat "Overflow pages"]
error_check_good stat:overflow [expr $overflow > 0] 1
}
set nentries [expr $noverflows * $ndups]
error_check_good overflow_truncate [$db truncate] $nentries
error_check_good overflow_close [$db close] 0
set testdir [get_home $env]
error_check_good dbverify [verify_dir $testdir "\tTest096.t: " 0 0 $nodump] 0
if { $closeenv == 1 } {
error_check_good envclose [$env close] 0
}
set testdir $orig_tdir
}
proc t96_populate {db method env nentries {ndups 1}} {
source ./include.tcl
set did [open $dict]
set count 0
set txn ""
set pflags ""
set gflags ""
if { [is_record_based $method] == 1 } {
append gflags "-recno"
}
while { [gets $did str] != -1 && $count < $nentries } {
if { [is_record_based $method] == 1 } {
set key [expr $count + 1]
} else {
set key $str
}
if { $ndups > 1 } {
for { set i 1 } { $i <= $ndups } { incr i } {
set datastr $i:$str
set t [$env txn]
error_check_good txn [is_valid_txn $t $env] TRUE
set txn "-txn $t"
set ret [eval {$db put} $txn $pflags \
{$key [chop_data $method $datastr]}]
error_check_good put $ret 0
error_check_good txn [$t commit] 0
}
} else {
set datastr [reverse $str]
set t [$env txn]
error_check_good txn [is_valid_txn $t $env] TRUE
set txn "-txn $t"
set ret [eval {$db put} \
$txn $pflags {$key [chop_data $method $datastr]}]
error_check_good put $ret 0
error_check_good txn [$t commit] 0
}
set ret [eval {$db get} $gflags {$key}]
error_check_good $key:dbget [llength $ret] $ndups
incr count
}
close $did
}
proc number_of_entries { db method } {
if { [is_record_based $method] == 1 } {
set dbc [$db cursor]
set last [$dbc get -last]
if {[llength $last] == 0} {
set number 0
} else {
set number [lindex [lindex $last 0] 0]
}
} else {
set ret [$db get -glob *]
set number [llength $ret]
}
return $number
}
# Open database. Truncate in a transaction, optionally with a write
# included in the transaction as well, then abort or commit. Close database.
proc txn_truncate { env method args testfile nentries op {write 0}} {
set db [eval {berkdb_open -create -auto_commit \
-env $env $method -mode 0644} $args $testfile]
error_check_good db_open [is_valid_db $db] TRUE
set txn [$env txn]
error_check_good txnbegin [is_valid_txn $txn $env] TRUE
set ret [$db truncate -txn $txn]
error_check_good dbtrunc $ret $nentries
if { $write == 1 } {
for {set i 1} {$i <= [expr $nentries / 2]} {incr i} {
set ret [eval {$db put} -txn $txn \
{$i [chop_data $method "aaaaaaaaaa"]}]
error_check_good write $ret 0
}
}
error_check_good txn$op [$txn $op] 0
error_check_good db_close [$db close] 0
}