mirror of
https://github.com/MariaDB/server.git
synced 2025-01-27 01:04:19 +01:00
155e78f014
BitKeeper/deleted/.del-ex_access.wpj~3df6ae8c99bf7c5f: Delete: bdb/build_vxworks/ex_access/ex_access.wpj BitKeeper/deleted/.del-ex_btrec.wpj~a7622f1c6f432dc6: Delete: bdb/build_vxworks/ex_btrec/ex_btrec.wpj BitKeeper/deleted/.del-ex_dbclient.wpj~7345440f3b204cdd: Delete: bdb/build_vxworks/ex_dbclient/ex_dbclient.wpj BitKeeper/deleted/.del-ex_env.wpj~fbe1ab10b04e8b74: Delete: bdb/build_vxworks/ex_env/ex_env.wpj BitKeeper/deleted/.del-ex_mpool.wpj~4479cfd5c45f327d: Delete: bdb/build_vxworks/ex_mpool/ex_mpool.wpj BitKeeper/deleted/.del-ex_tpcb.wpj~f78093006e14bf41: Delete: bdb/build_vxworks/ex_tpcb/ex_tpcb.wpj BitKeeper/deleted/.del-db_buildall.dsp~bd749ff6da11682: Delete: bdb/build_win32/db_buildall.dsp BitKeeper/deleted/.del-cxx_app.cpp~ad8df8e0791011ed: Delete: bdb/cxx/cxx_app.cpp BitKeeper/deleted/.del-cxx_log.cpp~a50ff3118fe06952: Delete: bdb/cxx/cxx_log.cpp BitKeeper/deleted/.del-cxx_table.cpp~ecd751e79b055556: Delete: bdb/cxx/cxx_table.cpp BitKeeper/deleted/.del-namemap.txt~796a3acd3885d8fd: Delete: bdb/cxx/namemap.txt BitKeeper/deleted/.del-Design.fileop~3ca4da68f1727373: Delete: bdb/db/Design.fileop BitKeeper/deleted/.del-db185_int.h~61bee3736e7959ef: Delete: bdb/db185/db185_int.h BitKeeper/deleted/.del-acconfig.h~411e8854d67ad8b5: Delete: bdb/dist/acconfig.h BitKeeper/deleted/.del-mutex.m4~a13383cde18a64e1: Delete: bdb/dist/aclocal/mutex.m4 BitKeeper/deleted/.del-options.m4~b9d0ca637213750a: Delete: bdb/dist/aclocal/options.m4 BitKeeper/deleted/.del-programs.m4~3ce7890b47732b30: Delete: bdb/dist/aclocal/programs.m4 BitKeeper/deleted/.del-tcl.m4~f944e2db93c3b6db: Delete: bdb/dist/aclocal/tcl.m4 BitKeeper/deleted/.del-types.m4~59cae158c9a32cff: Delete: bdb/dist/aclocal/types.m4 BitKeeper/deleted/.del-script~d38f6d3a4f159cb4: Delete: bdb/dist/build/script BitKeeper/deleted/.del-configure.in~ac795a92c8fe049c: Delete: bdb/dist/configure.in BitKeeper/deleted/.del-ltconfig~66bbd007d8024af: Delete: bdb/dist/ltconfig BitKeeper/deleted/.del-rec_ctemp~a28554362534f00a: Delete: bdb/dist/rec_ctemp BitKeeper/deleted/.del-s_tcl~2ffe4326459fcd9f: Delete: bdb/dist/s_tcl BitKeeper/deleted/.del-.IGNORE_ME~d8148b08fa7d5d15: Delete: bdb/dist/template/.IGNORE_ME BitKeeper/deleted/.del-btree.h~179f2aefec1753d: Delete: bdb/include/btree.h BitKeeper/deleted/.del-cxx_int.h~6b649c04766508f8: Delete: bdb/include/cxx_int.h BitKeeper/deleted/.del-db.src~6b433ae615b16a8d: Delete: bdb/include/db.src BitKeeper/deleted/.del-db_185.h~ad8b373d9391d35c: Delete: bdb/include/db_185.h BitKeeper/deleted/.del-db_am.h~a714912b6b75932f: Delete: bdb/include/db_am.h BitKeeper/deleted/.del-db_cxx.h~fcafadf45f5d19e9: Delete: bdb/include/db_cxx.h BitKeeper/deleted/.del-db_dispatch.h~6844f20f7eb46904: Delete: bdb/include/db_dispatch.h BitKeeper/deleted/.del-db_int.src~419a3f48b6a01da7: Delete: bdb/include/db_int.src BitKeeper/deleted/.del-db_join.h~76f9747a42c3399a: Delete: bdb/include/db_join.h BitKeeper/deleted/.del-db_page.h~e302ca3a4db3abdc: Delete: bdb/include/db_page.h BitKeeper/deleted/.del-db_server_int.h~e1d20b6ba3bca1ab: Delete: bdb/include/db_server_int.h BitKeeper/deleted/.del-db_shash.h~5fbf2d696fac90f3: Delete: bdb/include/db_shash.h BitKeeper/deleted/.del-db_swap.h~1e60887550864a59: Delete: bdb/include/db_swap.h BitKeeper/deleted/.del-db_upgrade.h~c644eee73701fc8d: Delete: bdb/include/db_upgrade.h BitKeeper/deleted/.del-db_verify.h~b8d6c297c61f342e: Delete: bdb/include/db_verify.h BitKeeper/deleted/.del-debug.h~dc2b4f2cf27ccebc: Delete: bdb/include/debug.h BitKeeper/deleted/.del-hash.h~2aaa548b28882dfb: Delete: bdb/include/hash.h BitKeeper/deleted/.del-lock.h~a761c1b7de57b77f: Delete: bdb/include/lock.h BitKeeper/deleted/.del-log.h~ff20184238e35e4d: Delete: bdb/include/log.h BitKeeper/deleted/.del-mp.h~7e317597622f3411: Delete: bdb/include/mp.h BitKeeper/deleted/.del-mutex.h~d3ae7a2977a68137: Delete: bdb/include/mutex.h BitKeeper/deleted/.del-os.h~91867cc8757cd0e3: Delete: bdb/include/os.h BitKeeper/deleted/.del-os_jump.h~e1b939fa5151d4be: Delete: bdb/include/os_jump.h BitKeeper/deleted/.del-qam.h~6fad0c1b5723d597: Delete: bdb/include/qam.h BitKeeper/deleted/.del-queue.h~4c72c0826c123d5: Delete: bdb/include/queue.h BitKeeper/deleted/.del-region.h~513fe04d977ca0fc: Delete: bdb/include/region.h BitKeeper/deleted/.del-shqueue.h~525fc3e6c2025c36: Delete: bdb/include/shqueue.h BitKeeper/deleted/.del-tcl_db.h~c536fd61a844f23f: Delete: bdb/include/tcl_db.h BitKeeper/deleted/.del-txn.h~c8d94b221ec147e4: Delete: bdb/include/txn.h BitKeeper/deleted/.del-xa.h~ecc466493aae9d9a: Delete: bdb/include/xa.h BitKeeper/deleted/.del-DbRecoveryInit.java~756b52601a0b9023: Delete: bdb/java/src/com/sleepycat/db/DbRecoveryInit.java BitKeeper/deleted/.del-DbTxnRecover.java~74607cba7ab89d6d: Delete: bdb/java/src/com/sleepycat/db/DbTxnRecover.java BitKeeper/deleted/.del-lock_conflict.c~fc5e0f14cf597a2b: Delete: bdb/lock/lock_conflict.c BitKeeper/deleted/.del-log.src~53ac9e7b5cb023f2: Delete: bdb/log/log.src BitKeeper/deleted/.del-log_findckp.c~24287f008916e81f: Delete: bdb/log/log_findckp.c BitKeeper/deleted/.del-log_rec.c~d51711f2cac09297: Delete: bdb/log/log_rec.c BitKeeper/deleted/.del-log_register.c~b40bb4efac75ca15: Delete: bdb/log/log_register.c BitKeeper/deleted/.del-Design~b3d0f179f2767b: Delete: bdb/mp/Design BitKeeper/deleted/.del-os_finit.c~95dbefc6fe79b26c: Delete: bdb/os/os_finit.c BitKeeper/deleted/.del-os_abs.c~df95d1e7db81924: Delete: bdb/os_vxworks/os_abs.c BitKeeper/deleted/.del-os_finit.c~803b484bdb9d0122: Delete: bdb/os_vxworks/os_finit.c BitKeeper/deleted/.del-os_map.c~3a6d7926398b76d3: Delete: bdb/os_vxworks/os_map.c BitKeeper/deleted/.del-os_finit.c~19a227c6d3c78ad: Delete: bdb/os_win32/os_finit.c BitKeeper/deleted/.del-log-corruption.patch~1cf2ecc7c6408d5d: Delete: bdb/patches/log-corruption.patch BitKeeper/deleted/.del-Btree.pm~af6d0c5eaed4a98e: Delete: bdb/perl.BerkeleyDB/BerkeleyDB/Btree.pm BitKeeper/deleted/.del-BerkeleyDB.pm~7244036d4482643: Delete: bdb/perl.BerkeleyDB/BerkeleyDB.pm BitKeeper/deleted/.del-BerkeleyDB.pod~e7b18fd6132448e3: Delete: bdb/perl.BerkeleyDB/BerkeleyDB.pod BitKeeper/deleted/.del-Hash.pm~10292a26c06a5c95: Delete: bdb/perl.BerkeleyDB/BerkeleyDB/Hash.pm BitKeeper/deleted/.del-BerkeleyDB.pod.P~79f76a1495eda203: Delete: bdb/perl.BerkeleyDB/BerkeleyDB.pod.P BitKeeper/deleted/.del-BerkeleyDB.xs~80c99afbd98e392c: Delete: bdb/perl.BerkeleyDB/BerkeleyDB.xs BitKeeper/deleted/.del-Changes~729c1891efa60de9: Delete: bdb/perl.BerkeleyDB/Changes BitKeeper/deleted/.del-MANIFEST~63a1e34aecf157a0: Delete: bdb/perl.BerkeleyDB/MANIFEST BitKeeper/deleted/.del-Makefile.PL~c68797707d8df87a: Delete: bdb/perl.BerkeleyDB/Makefile.PL BitKeeper/deleted/.del-README~5f2f579b1a241407: Delete: bdb/perl.BerkeleyDB/README BitKeeper/deleted/.del-Todo~dca3c66c193adda9: Delete: bdb/perl.BerkeleyDB/Todo BitKeeper/deleted/.del-config.in~ae81681e450e0999: Delete: bdb/perl.BerkeleyDB/config.in BitKeeper/deleted/.del-dbinfo~28ad67d83be4f68e: Delete: bdb/perl.BerkeleyDB/dbinfo BitKeeper/deleted/.del-mkconsts~543ab60669c7a04e: Delete: bdb/perl.BerkeleyDB/mkconsts BitKeeper/deleted/.del-mkpod~182c0ca54e439afb: Delete: bdb/perl.BerkeleyDB/mkpod BitKeeper/deleted/.del-5.004~e008cb5a48805543: Delete: bdb/perl.BerkeleyDB/patches/5.004 BitKeeper/deleted/.del-irix_6_5.pl~61662bb08afcdec8: Delete: bdb/perl.BerkeleyDB/hints/irix_6_5.pl BitKeeper/deleted/.del-solaris.pl~6771e7182394e152: Delete: bdb/perl.BerkeleyDB/hints/solaris.pl BitKeeper/deleted/.del-typemap~783b8f5295b05f3d: Delete: bdb/perl.BerkeleyDB/typemap BitKeeper/deleted/.del-5.004_01~6081ce2fff7b0bc: Delete: bdb/perl.BerkeleyDB/patches/5.004_01 BitKeeper/deleted/.del-5.004_02~87214eac35ad9e6: Delete: bdb/perl.BerkeleyDB/patches/5.004_02 BitKeeper/deleted/.del-5.004_03~9a672becec7cb40f: Delete: bdb/perl.BerkeleyDB/patches/5.004_03 BitKeeper/deleted/.del-5.004_04~e326cb51af09d154: Delete: bdb/perl.BerkeleyDB/patches/5.004_04 BitKeeper/deleted/.del-5.004_05~7ab457a1e41a92fe: Delete: bdb/perl.BerkeleyDB/patches/5.004_05 BitKeeper/deleted/.del-5.005~f9e2d59b5964cd4b: Delete: bdb/perl.BerkeleyDB/patches/5.005 BitKeeper/deleted/.del-5.005_01~3eb9fb7b5842ea8e: Delete: bdb/perl.BerkeleyDB/patches/5.005_01 BitKeeper/deleted/.del-5.005_02~67477ce0bef717cb: Delete: bdb/perl.BerkeleyDB/patches/5.005_02 BitKeeper/deleted/.del-5.005_03~c4c29a1fb21e290a: Delete: bdb/perl.BerkeleyDB/patches/5.005_03 BitKeeper/deleted/.del-5.6.0~e1fb9897d124ee22: Delete: bdb/perl.BerkeleyDB/patches/5.6.0 BitKeeper/deleted/.del-btree.t~e4a1a3c675ddc406: Delete: bdb/perl.BerkeleyDB/t/btree.t BitKeeper/deleted/.del-db-3.0.t~d2c60991d84558f2: Delete: bdb/perl.BerkeleyDB/t/db-3.0.t BitKeeper/deleted/.del-db-3.1.t~6ee88cd13f55e018: Delete: bdb/perl.BerkeleyDB/t/db-3.1.t BitKeeper/deleted/.del-db-3.2.t~f73b6461f98fd1cf: Delete: bdb/perl.BerkeleyDB/t/db-3.2.t BitKeeper/deleted/.del-destroy.t~cc6a2ae1980a2ecd: Delete: bdb/perl.BerkeleyDB/t/destroy.t BitKeeper/deleted/.del-env.t~a8604a4499c4bd07: Delete: bdb/perl.BerkeleyDB/t/env.t BitKeeper/deleted/.del-examples.t~2571b77c3cc75574: Delete: bdb/perl.BerkeleyDB/t/examples.t BitKeeper/deleted/.del-examples.t.T~8228bdd75ac78b88: Delete: bdb/perl.BerkeleyDB/t/examples.t.T BitKeeper/deleted/.del-examples3.t.T~66a186897a87026d: Delete: bdb/perl.BerkeleyDB/t/examples3.t.T BitKeeper/deleted/.del-examples3.t~fe3822ba2f2d7f83: Delete: bdb/perl.BerkeleyDB/t/examples3.t BitKeeper/deleted/.del-filter.t~f87b045c1b708637: Delete: bdb/perl.BerkeleyDB/t/filter.t BitKeeper/deleted/.del-hash.t~616bfb4d644de3a3: Delete: bdb/perl.BerkeleyDB/t/hash.t BitKeeper/deleted/.del-join.t~29fc39f74a83ca22: Delete: bdb/perl.BerkeleyDB/t/join.t BitKeeper/deleted/.del-mldbm.t~31f5015341eea040: Delete: bdb/perl.BerkeleyDB/t/mldbm.t BitKeeper/deleted/.del-queue.t~8f338034ce44a641: Delete: bdb/perl.BerkeleyDB/t/queue.t BitKeeper/deleted/.del-recno.t~d4ddbd3743add63e: Delete: bdb/perl.BerkeleyDB/t/recno.t BitKeeper/deleted/.del-strict.t~6885cdd2ea71ca2d: Delete: bdb/perl.BerkeleyDB/t/strict.t BitKeeper/deleted/.del-subdb.t~aab62a5d5864c603: Delete: bdb/perl.BerkeleyDB/t/subdb.t BitKeeper/deleted/.del-txn.t~65033b8558ae1216: Delete: bdb/perl.BerkeleyDB/t/txn.t BitKeeper/deleted/.del-unknown.t~f3710458682665e1: Delete: bdb/perl.BerkeleyDB/t/unknown.t BitKeeper/deleted/.del-Changes~436f74a5c414c65b: Delete: bdb/perl.DB_File/Changes BitKeeper/deleted/.del-DB_File.pm~ae0951c6c7665a82: Delete: bdb/perl.DB_File/DB_File.pm BitKeeper/deleted/.del-DB_File.xs~89e49a0b5556f1d8: Delete: bdb/perl.DB_File/DB_File.xs BitKeeper/deleted/.del-DB_File_BS~290fad5dbbb87069: Delete: bdb/perl.DB_File/DB_File_BS BitKeeper/deleted/.del-MANIFEST~90ee581572bdd4ac: Delete: bdb/perl.DB_File/MANIFEST BitKeeper/deleted/.del-Makefile.PL~ac0567bb5a377e38: Delete: bdb/perl.DB_File/Makefile.PL BitKeeper/deleted/.del-README~77e924a5a9bae6b3: Delete: bdb/perl.DB_File/README BitKeeper/deleted/.del-config.in~ab4c2792b86a810b: Delete: bdb/perl.DB_File/config.in BitKeeper/deleted/.del-dbinfo~461c43b30fab2cb: Delete: bdb/perl.DB_File/dbinfo BitKeeper/deleted/.del-dynixptx.pl~50dcddfae25d17e9: Delete: bdb/perl.DB_File/hints/dynixptx.pl BitKeeper/deleted/.del-typemap~55cffb3288a9e587: Delete: bdb/perl.DB_File/typemap BitKeeper/deleted/.del-version.c~a4df0e646f8b3975: Delete: bdb/perl.DB_File/version.c BitKeeper/deleted/.del-5.004_01~d6830d0082702af7: Delete: bdb/perl.DB_File/patches/5.004_01 BitKeeper/deleted/.del-5.004_02~78b082dc80c91031: Delete: bdb/perl.DB_File/patches/5.004_02 BitKeeper/deleted/.del-5.004~4411ec2e3c9e008b: Delete: bdb/perl.DB_File/patches/5.004 BitKeeper/deleted/.del-sco.pl~1e795fe14fe4dcfe: Delete: bdb/perl.DB_File/hints/sco.pl BitKeeper/deleted/.del-5.004_03~33f274648b160d95: Delete: bdb/perl.DB_File/patches/5.004_03 BitKeeper/deleted/.del-5.004_04~8f3d1b3cf18bb20a: Delete: bdb/perl.DB_File/patches/5.004_04 BitKeeper/deleted/.del-5.004_05~9c0f02e7331e142: Delete: bdb/perl.DB_File/patches/5.004_05 BitKeeper/deleted/.del-5.005~c2108cb2e3c8d951: Delete: bdb/perl.DB_File/patches/5.005 BitKeeper/deleted/.del-5.005_01~3b45e9673afc4cfa: Delete: bdb/perl.DB_File/patches/5.005_01 BitKeeper/deleted/.del-5.005_02~9fe5766bb02a4522: Delete: bdb/perl.DB_File/patches/5.005_02 BitKeeper/deleted/.del-5.005_03~ffa1c38c19ae72ea: Delete: bdb/perl.DB_File/patches/5.005_03 BitKeeper/deleted/.del-5.6.0~373be3a5ce47be85: Delete: bdb/perl.DB_File/patches/5.6.0 BitKeeper/deleted/.del-db-btree.t~3231595a1c241eb3: Delete: bdb/perl.DB_File/t/db-btree.t BitKeeper/deleted/.del-db-hash.t~7c4ad0c795c7fad2: Delete: bdb/perl.DB_File/t/db-hash.t BitKeeper/deleted/.del-db-recno.t~6c2d3d80b9ba4a50: Delete: bdb/perl.DB_File/t/db-recno.t BitKeeper/deleted/.del-db_server.sed~cdb00ebcd48a64e2: Delete: bdb/rpc_server/db_server.sed BitKeeper/deleted/.del-db_server_proc.c~d46c8f409c3747f4: Delete: bdb/rpc_server/db_server_proc.c BitKeeper/deleted/.del-db_server_svc.sed~3f5e59f334fa4607: Delete: bdb/rpc_server/db_server_svc.sed BitKeeper/deleted/.del-db_server_util.c~a809f3a4629acda: Delete: bdb/rpc_server/db_server_util.c BitKeeper/deleted/.del-log.tcl~ff1b41f1355b97d7: Delete: bdb/test/log.tcl BitKeeper/deleted/.del-mpool.tcl~b0df4dc1b04db26c: Delete: bdb/test/mpool.tcl BitKeeper/deleted/.del-mutex.tcl~52fd5c73a150565: Delete: bdb/test/mutex.tcl BitKeeper/deleted/.del-txn.tcl~c4ff071550b5446e: Delete: bdb/test/txn.tcl BitKeeper/deleted/.del-README~e800a12a5392010a: Delete: bdb/test/upgrade/README BitKeeper/deleted/.del-pack-2.6.6.pl~89d5076d758d3e98: Delete: bdb/test/upgrade/generate-2.X/pack-2.6.6.pl BitKeeper/deleted/.del-test-2.6.patch~4a52dc83d447547b: Delete: bdb/test/upgrade/generate-2.X/test-2.6.patch
2291 lines
62 KiB
Perl
2291 lines
62 KiB
Perl
# DB_File.pm -- Perl 5 interface to Berkeley DB
|
|
#
|
|
# written by Paul Marquess (Paul.Marquess@btinternet.com)
|
|
# last modified 1st September 2002
|
|
# version 1.805
|
|
#
|
|
# Copyright (c) 1995-2002 Paul Marquess. All rights reserved.
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the same terms as Perl itself.
|
|
|
|
|
|
package DB_File::HASHINFO ;
|
|
|
|
require 5.00404;
|
|
|
|
use warnings;
|
|
use strict;
|
|
use Carp;
|
|
require Tie::Hash;
|
|
@DB_File::HASHINFO::ISA = qw(Tie::Hash);
|
|
|
|
sub new
|
|
{
|
|
my $pkg = shift ;
|
|
my %x ;
|
|
tie %x, $pkg ;
|
|
bless \%x, $pkg ;
|
|
}
|
|
|
|
|
|
sub TIEHASH
|
|
{
|
|
my $pkg = shift ;
|
|
|
|
bless { VALID => {
|
|
bsize => 1,
|
|
ffactor => 1,
|
|
nelem => 1,
|
|
cachesize => 1,
|
|
hash => 2,
|
|
lorder => 1,
|
|
},
|
|
GOT => {}
|
|
}, $pkg ;
|
|
}
|
|
|
|
|
|
sub FETCH
|
|
{
|
|
my $self = shift ;
|
|
my $key = shift ;
|
|
|
|
return $self->{GOT}{$key} if exists $self->{VALID}{$key} ;
|
|
|
|
my $pkg = ref $self ;
|
|
croak "${pkg}::FETCH - Unknown element '$key'" ;
|
|
}
|
|
|
|
|
|
sub STORE
|
|
{
|
|
my $self = shift ;
|
|
my $key = shift ;
|
|
my $value = shift ;
|
|
|
|
my $type = $self->{VALID}{$key};
|
|
|
|
if ( $type )
|
|
{
|
|
croak "Key '$key' not associated with a code reference"
|
|
if $type == 2 && !ref $value && ref $value ne 'CODE';
|
|
$self->{GOT}{$key} = $value ;
|
|
return ;
|
|
}
|
|
|
|
my $pkg = ref $self ;
|
|
croak "${pkg}::STORE - Unknown element '$key'" ;
|
|
}
|
|
|
|
sub DELETE
|
|
{
|
|
my $self = shift ;
|
|
my $key = shift ;
|
|
|
|
if ( exists $self->{VALID}{$key} )
|
|
{
|
|
delete $self->{GOT}{$key} ;
|
|
return ;
|
|
}
|
|
|
|
my $pkg = ref $self ;
|
|
croak "DB_File::HASHINFO::DELETE - Unknown element '$key'" ;
|
|
}
|
|
|
|
sub EXISTS
|
|
{
|
|
my $self = shift ;
|
|
my $key = shift ;
|
|
|
|
exists $self->{VALID}{$key} ;
|
|
}
|
|
|
|
sub NotHere
|
|
{
|
|
my $self = shift ;
|
|
my $method = shift ;
|
|
|
|
croak ref($self) . " does not define the method ${method}" ;
|
|
}
|
|
|
|
sub FIRSTKEY { my $self = shift ; $self->NotHere("FIRSTKEY") }
|
|
sub NEXTKEY { my $self = shift ; $self->NotHere("NEXTKEY") }
|
|
sub CLEAR { my $self = shift ; $self->NotHere("CLEAR") }
|
|
|
|
package DB_File::RECNOINFO ;
|
|
|
|
use warnings;
|
|
use strict ;
|
|
|
|
@DB_File::RECNOINFO::ISA = qw(DB_File::HASHINFO) ;
|
|
|
|
sub TIEHASH
|
|
{
|
|
my $pkg = shift ;
|
|
|
|
bless { VALID => { map {$_, 1}
|
|
qw( bval cachesize psize flags lorder reclen bfname )
|
|
},
|
|
GOT => {},
|
|
}, $pkg ;
|
|
}
|
|
|
|
package DB_File::BTREEINFO ;
|
|
|
|
use warnings;
|
|
use strict ;
|
|
|
|
@DB_File::BTREEINFO::ISA = qw(DB_File::HASHINFO) ;
|
|
|
|
sub TIEHASH
|
|
{
|
|
my $pkg = shift ;
|
|
|
|
bless { VALID => {
|
|
flags => 1,
|
|
cachesize => 1,
|
|
maxkeypage => 1,
|
|
minkeypage => 1,
|
|
psize => 1,
|
|
compare => 2,
|
|
prefix => 2,
|
|
lorder => 1,
|
|
},
|
|
GOT => {},
|
|
}, $pkg ;
|
|
}
|
|
|
|
|
|
package DB_File ;
|
|
|
|
use warnings;
|
|
use strict;
|
|
our ($VERSION, @ISA, @EXPORT, $AUTOLOAD, $DB_BTREE, $DB_HASH, $DB_RECNO);
|
|
our ($db_version, $use_XSLoader, $splice_end_array);
|
|
use Carp;
|
|
|
|
|
|
$VERSION = "1.805" ;
|
|
|
|
{
|
|
local $SIG{__WARN__} = sub {$splice_end_array = "@_";};
|
|
my @a =(1); splice(@a, 3);
|
|
$splice_end_array =
|
|
($splice_end_array =~ /^splice\(\) offset past end of array at /);
|
|
}
|
|
|
|
#typedef enum { DB_BTREE, DB_HASH, DB_RECNO } DBTYPE;
|
|
$DB_BTREE = new DB_File::BTREEINFO ;
|
|
$DB_HASH = new DB_File::HASHINFO ;
|
|
$DB_RECNO = new DB_File::RECNOINFO ;
|
|
|
|
require Tie::Hash;
|
|
require Exporter;
|
|
use AutoLoader;
|
|
BEGIN {
|
|
$use_XSLoader = 1 ;
|
|
{ local $SIG{__DIE__} ; eval { require XSLoader } ; }
|
|
|
|
if ($@) {
|
|
$use_XSLoader = 0 ;
|
|
require DynaLoader;
|
|
@ISA = qw(DynaLoader);
|
|
}
|
|
}
|
|
|
|
push @ISA, qw(Tie::Hash Exporter);
|
|
@EXPORT = qw(
|
|
$DB_BTREE $DB_HASH $DB_RECNO
|
|
|
|
BTREEMAGIC
|
|
BTREEVERSION
|
|
DB_LOCK
|
|
DB_SHMEM
|
|
DB_TXN
|
|
HASHMAGIC
|
|
HASHVERSION
|
|
MAX_PAGE_NUMBER
|
|
MAX_PAGE_OFFSET
|
|
MAX_REC_NUMBER
|
|
RET_ERROR
|
|
RET_SPECIAL
|
|
RET_SUCCESS
|
|
R_CURSOR
|
|
R_DUP
|
|
R_FIRST
|
|
R_FIXEDLEN
|
|
R_IAFTER
|
|
R_IBEFORE
|
|
R_LAST
|
|
R_NEXT
|
|
R_NOKEY
|
|
R_NOOVERWRITE
|
|
R_PREV
|
|
R_RECNOSYNC
|
|
R_SETCURSOR
|
|
R_SNAPSHOT
|
|
__R_UNUSED
|
|
|
|
);
|
|
|
|
sub AUTOLOAD {
|
|
my($constname);
|
|
($constname = $AUTOLOAD) =~ s/.*:://;
|
|
my ($error, $val) = constant($constname);
|
|
Carp::croak $error if $error;
|
|
no strict 'refs';
|
|
*{$AUTOLOAD} = sub { $val };
|
|
goto &{$AUTOLOAD};
|
|
}
|
|
|
|
|
|
eval {
|
|
# Make all Fcntl O_XXX constants available for importing
|
|
require Fcntl;
|
|
my @O = grep /^O_/, @Fcntl::EXPORT;
|
|
Fcntl->import(@O); # first we import what we want to export
|
|
push(@EXPORT, @O);
|
|
};
|
|
|
|
if ($use_XSLoader)
|
|
{ XSLoader::load("DB_File", $VERSION)}
|
|
else
|
|
{ bootstrap DB_File $VERSION }
|
|
|
|
# Preloaded methods go here. Autoload methods go after __END__, and are
|
|
# processed by the autosplit program.
|
|
|
|
sub tie_hash_or_array
|
|
{
|
|
my (@arg) = @_ ;
|
|
my $tieHASH = ( (caller(1))[3] =~ /TIEHASH/ ) ;
|
|
|
|
$arg[4] = tied %{ $arg[4] }
|
|
if @arg >= 5 && ref $arg[4] && $arg[4] =~ /=HASH/ && tied %{ $arg[4] } ;
|
|
|
|
$arg[2] = O_CREAT()|O_RDWR() if @arg >=3 && ! defined $arg[2];
|
|
$arg[3] = 0666 if @arg >=4 && ! defined $arg[3];
|
|
|
|
# make recno in Berkeley DB version 2 work like recno in version 1.
|
|
if ($db_version > 1 and defined $arg[4] and $arg[4] =~ /RECNO/ and
|
|
$arg[1] and ! -e $arg[1]) {
|
|
open(FH, ">$arg[1]") or return undef ;
|
|
close FH ;
|
|
chmod $arg[3] ? $arg[3] : 0666 , $arg[1] ;
|
|
}
|
|
|
|
DoTie_($tieHASH, @arg) ;
|
|
}
|
|
|
|
sub TIEHASH
|
|
{
|
|
tie_hash_or_array(@_) ;
|
|
}
|
|
|
|
sub TIEARRAY
|
|
{
|
|
tie_hash_or_array(@_) ;
|
|
}
|
|
|
|
sub CLEAR
|
|
{
|
|
my $self = shift;
|
|
my $key = 0 ;
|
|
my $value = "" ;
|
|
my $status = $self->seq($key, $value, R_FIRST());
|
|
my @keys;
|
|
|
|
while ($status == 0) {
|
|
push @keys, $key;
|
|
$status = $self->seq($key, $value, R_NEXT());
|
|
}
|
|
foreach $key (reverse @keys) {
|
|
my $s = $self->del($key);
|
|
}
|
|
}
|
|
|
|
sub EXTEND { }
|
|
|
|
sub STORESIZE
|
|
{
|
|
my $self = shift;
|
|
my $length = shift ;
|
|
my $current_length = $self->length() ;
|
|
|
|
if ($length < $current_length) {
|
|
my $key ;
|
|
for ($key = $current_length - 1 ; $key >= $length ; -- $key)
|
|
{ $self->del($key) }
|
|
}
|
|
elsif ($length > $current_length) {
|
|
$self->put($length-1, "") ;
|
|
}
|
|
}
|
|
|
|
|
|
sub SPLICE
|
|
{
|
|
my $self = shift;
|
|
my $offset = shift;
|
|
if (not defined $offset) {
|
|
warnings::warnif('uninitialized', 'Use of uninitialized value in splice');
|
|
$offset = 0;
|
|
}
|
|
|
|
my $length = @_ ? shift : 0;
|
|
# Carping about definedness comes _after_ the OFFSET sanity check.
|
|
# This is so we get the same error messages as Perl's splice().
|
|
#
|
|
|
|
my @list = @_;
|
|
|
|
my $size = $self->FETCHSIZE();
|
|
|
|
# 'If OFFSET is negative then it start that far from the end of
|
|
# the array.'
|
|
#
|
|
if ($offset < 0) {
|
|
my $new_offset = $size + $offset;
|
|
if ($new_offset < 0) {
|
|
die "Modification of non-creatable array value attempted, "
|
|
. "subscript $offset";
|
|
}
|
|
$offset = $new_offset;
|
|
}
|
|
|
|
if (not defined $length) {
|
|
warnings::warnif('uninitialized', 'Use of uninitialized value in splice');
|
|
$length = 0;
|
|
}
|
|
|
|
if ($offset > $size) {
|
|
$offset = $size;
|
|
warnings::warnif('misc', 'splice() offset past end of array')
|
|
if $splice_end_array;
|
|
}
|
|
|
|
# 'If LENGTH is omitted, removes everything from OFFSET onward.'
|
|
if (not defined $length) {
|
|
$length = $size - $offset;
|
|
}
|
|
|
|
# 'If LENGTH is negative, leave that many elements off the end of
|
|
# the array.'
|
|
#
|
|
if ($length < 0) {
|
|
$length = $size - $offset + $length;
|
|
|
|
if ($length < 0) {
|
|
# The user must have specified a length bigger than the
|
|
# length of the array passed in. But perl's splice()
|
|
# doesn't catch this, it just behaves as for length=0.
|
|
#
|
|
$length = 0;
|
|
}
|
|
}
|
|
|
|
if ($length > $size - $offset) {
|
|
$length = $size - $offset;
|
|
}
|
|
|
|
# $num_elems holds the current number of elements in the database.
|
|
my $num_elems = $size;
|
|
|
|
# 'Removes the elements designated by OFFSET and LENGTH from an
|
|
# array,'...
|
|
#
|
|
my @removed = ();
|
|
foreach (0 .. $length - 1) {
|
|
my $old;
|
|
my $status = $self->get($offset, $old);
|
|
if ($status != 0) {
|
|
my $msg = "error from Berkeley DB on get($offset, \$old)";
|
|
if ($status == 1) {
|
|
$msg .= ' (no such element?)';
|
|
}
|
|
else {
|
|
$msg .= ": error status $status";
|
|
if (defined $! and $! ne '') {
|
|
$msg .= ", message $!";
|
|
}
|
|
}
|
|
die $msg;
|
|
}
|
|
push @removed, $old;
|
|
|
|
$status = $self->del($offset);
|
|
if ($status != 0) {
|
|
my $msg = "error from Berkeley DB on del($offset)";
|
|
if ($status == 1) {
|
|
$msg .= ' (no such element?)';
|
|
}
|
|
else {
|
|
$msg .= ": error status $status";
|
|
if (defined $! and $! ne '') {
|
|
$msg .= ", message $!";
|
|
}
|
|
}
|
|
die $msg;
|
|
}
|
|
|
|
-- $num_elems;
|
|
}
|
|
|
|
# ...'and replaces them with the elements of LIST, if any.'
|
|
my $pos = $offset;
|
|
while (defined (my $elem = shift @list)) {
|
|
my $old_pos = $pos;
|
|
my $status;
|
|
if ($pos >= $num_elems) {
|
|
$status = $self->put($pos, $elem);
|
|
}
|
|
else {
|
|
$status = $self->put($pos, $elem, $self->R_IBEFORE);
|
|
}
|
|
|
|
if ($status != 0) {
|
|
my $msg = "error from Berkeley DB on put($pos, $elem, ...)";
|
|
if ($status == 1) {
|
|
$msg .= ' (no such element?)';
|
|
}
|
|
else {
|
|
$msg .= ", error status $status";
|
|
if (defined $! and $! ne '') {
|
|
$msg .= ", message $!";
|
|
}
|
|
}
|
|
die $msg;
|
|
}
|
|
|
|
die "pos unexpectedly changed from $old_pos to $pos with R_IBEFORE"
|
|
if $old_pos != $pos;
|
|
|
|
++ $pos;
|
|
++ $num_elems;
|
|
}
|
|
|
|
if (wantarray) {
|
|
# 'In list context, returns the elements removed from the
|
|
# array.'
|
|
#
|
|
return @removed;
|
|
}
|
|
elsif (defined wantarray and not wantarray) {
|
|
# 'In scalar context, returns the last element removed, or
|
|
# undef if no elements are removed.'
|
|
#
|
|
if (@removed) {
|
|
my $last = pop @removed;
|
|
return "$last";
|
|
}
|
|
else {
|
|
return undef;
|
|
}
|
|
}
|
|
elsif (not defined wantarray) {
|
|
# Void context
|
|
}
|
|
else { die }
|
|
}
|
|
sub ::DB_File::splice { &SPLICE }
|
|
|
|
sub find_dup
|
|
{
|
|
croak "Usage: \$db->find_dup(key,value)\n"
|
|
unless @_ == 3 ;
|
|
|
|
my $db = shift ;
|
|
my ($origkey, $value_wanted) = @_ ;
|
|
my ($key, $value) = ($origkey, 0);
|
|
my ($status) = 0 ;
|
|
|
|
for ($status = $db->seq($key, $value, R_CURSOR() ) ;
|
|
$status == 0 ;
|
|
$status = $db->seq($key, $value, R_NEXT() ) ) {
|
|
|
|
return 0 if $key eq $origkey and $value eq $value_wanted ;
|
|
}
|
|
|
|
return $status ;
|
|
}
|
|
|
|
sub del_dup
|
|
{
|
|
croak "Usage: \$db->del_dup(key,value)\n"
|
|
unless @_ == 3 ;
|
|
|
|
my $db = shift ;
|
|
my ($key, $value) = @_ ;
|
|
my ($status) = $db->find_dup($key, $value) ;
|
|
return $status if $status != 0 ;
|
|
|
|
$status = $db->del($key, R_CURSOR() ) ;
|
|
return $status ;
|
|
}
|
|
|
|
sub get_dup
|
|
{
|
|
croak "Usage: \$db->get_dup(key [,flag])\n"
|
|
unless @_ == 2 or @_ == 3 ;
|
|
|
|
my $db = shift ;
|
|
my $key = shift ;
|
|
my $flag = shift ;
|
|
my $value = 0 ;
|
|
my $origkey = $key ;
|
|
my $wantarray = wantarray ;
|
|
my %values = () ;
|
|
my @values = () ;
|
|
my $counter = 0 ;
|
|
my $status = 0 ;
|
|
|
|
# iterate through the database until either EOF ($status == 0)
|
|
# or a different key is encountered ($key ne $origkey).
|
|
for ($status = $db->seq($key, $value, R_CURSOR()) ;
|
|
$status == 0 and $key eq $origkey ;
|
|
$status = $db->seq($key, $value, R_NEXT()) ) {
|
|
|
|
# save the value or count number of matches
|
|
if ($wantarray) {
|
|
if ($flag)
|
|
{ ++ $values{$value} }
|
|
else
|
|
{ push (@values, $value) }
|
|
}
|
|
else
|
|
{ ++ $counter }
|
|
|
|
}
|
|
|
|
return ($wantarray ? ($flag ? %values : @values) : $counter) ;
|
|
}
|
|
|
|
|
|
1;
|
|
__END__
|
|
|
|
=head1 NAME
|
|
|
|
DB_File - Perl5 access to Berkeley DB version 1.x
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
use DB_File;
|
|
|
|
[$X =] tie %hash, 'DB_File', [$filename, $flags, $mode, $DB_HASH] ;
|
|
[$X =] tie %hash, 'DB_File', $filename, $flags, $mode, $DB_BTREE ;
|
|
[$X =] tie @array, 'DB_File', $filename, $flags, $mode, $DB_RECNO ;
|
|
|
|
$status = $X->del($key [, $flags]) ;
|
|
$status = $X->put($key, $value [, $flags]) ;
|
|
$status = $X->get($key, $value [, $flags]) ;
|
|
$status = $X->seq($key, $value, $flags) ;
|
|
$status = $X->sync([$flags]) ;
|
|
$status = $X->fd ;
|
|
|
|
# BTREE only
|
|
$count = $X->get_dup($key) ;
|
|
@list = $X->get_dup($key) ;
|
|
%list = $X->get_dup($key, 1) ;
|
|
$status = $X->find_dup($key, $value) ;
|
|
$status = $X->del_dup($key, $value) ;
|
|
|
|
# RECNO only
|
|
$a = $X->length;
|
|
$a = $X->pop ;
|
|
$X->push(list);
|
|
$a = $X->shift;
|
|
$X->unshift(list);
|
|
@r = $X->splice(offset, length, elements);
|
|
|
|
# DBM Filters
|
|
$old_filter = $db->filter_store_key ( sub { ... } ) ;
|
|
$old_filter = $db->filter_store_value( sub { ... } ) ;
|
|
$old_filter = $db->filter_fetch_key ( sub { ... } ) ;
|
|
$old_filter = $db->filter_fetch_value( sub { ... } ) ;
|
|
|
|
untie %hash ;
|
|
untie @array ;
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
B<DB_File> is a module which allows Perl programs to make use of the
|
|
facilities provided by Berkeley DB version 1.x (if you have a newer
|
|
version of DB, see L<Using DB_File with Berkeley DB version 2 or greater>).
|
|
It is assumed that you have a copy of the Berkeley DB manual pages at
|
|
hand when reading this documentation. The interface defined here
|
|
mirrors the Berkeley DB interface closely.
|
|
|
|
Berkeley DB is a C library which provides a consistent interface to a
|
|
number of database formats. B<DB_File> provides an interface to all
|
|
three of the database types currently supported by Berkeley DB.
|
|
|
|
The file types are:
|
|
|
|
=over 5
|
|
|
|
=item B<DB_HASH>
|
|
|
|
This database type allows arbitrary key/value pairs to be stored in data
|
|
files. This is equivalent to the functionality provided by other
|
|
hashing packages like DBM, NDBM, ODBM, GDBM, and SDBM. Remember though,
|
|
the files created using DB_HASH are not compatible with any of the
|
|
other packages mentioned.
|
|
|
|
A default hashing algorithm, which will be adequate for most
|
|
applications, is built into Berkeley DB. If you do need to use your own
|
|
hashing algorithm it is possible to write your own in Perl and have
|
|
B<DB_File> use it instead.
|
|
|
|
=item B<DB_BTREE>
|
|
|
|
The btree format allows arbitrary key/value pairs to be stored in a
|
|
sorted, balanced binary tree.
|
|
|
|
As with the DB_HASH format, it is possible to provide a user defined
|
|
Perl routine to perform the comparison of keys. By default, though, the
|
|
keys are stored in lexical order.
|
|
|
|
=item B<DB_RECNO>
|
|
|
|
DB_RECNO allows both fixed-length and variable-length flat text files
|
|
to be manipulated using the same key/value pair interface as in DB_HASH
|
|
and DB_BTREE. In this case the key will consist of a record (line)
|
|
number.
|
|
|
|
=back
|
|
|
|
=head2 Using DB_File with Berkeley DB version 2 or greater
|
|
|
|
Although B<DB_File> is intended to be used with Berkeley DB version 1,
|
|
it can also be used with version 2, 3 or 4. In this case the interface is
|
|
limited to the functionality provided by Berkeley DB 1.x. Anywhere the
|
|
version 2 or greater interface differs, B<DB_File> arranges for it to work
|
|
like version 1. This feature allows B<DB_File> scripts that were built
|
|
with version 1 to be migrated to version 2 or greater without any changes.
|
|
|
|
If you want to make use of the new features available in Berkeley DB
|
|
2.x or greater, use the Perl module B<BerkeleyDB> instead.
|
|
|
|
B<Note:> The database file format has changed multiple times in Berkeley
|
|
DB version 2, 3 and 4. If you cannot recreate your databases, you
|
|
must dump any existing databases with either the C<db_dump> or the
|
|
C<db_dump185> utility that comes with Berkeley DB.
|
|
Once you have rebuilt DB_File to use Berkeley DB version 2 or greater,
|
|
your databases can be recreated using C<db_load>. Refer to the Berkeley DB
|
|
documentation for further details.
|
|
|
|
Please read L<"COPYRIGHT"> before using version 2.x or greater of Berkeley
|
|
DB with DB_File.
|
|
|
|
=head2 Interface to Berkeley DB
|
|
|
|
B<DB_File> allows access to Berkeley DB files using the tie() mechanism
|
|
in Perl 5 (for full details, see L<perlfunc/tie()>). This facility
|
|
allows B<DB_File> to access Berkeley DB files using either an
|
|
associative array (for DB_HASH & DB_BTREE file types) or an ordinary
|
|
array (for the DB_RECNO file type).
|
|
|
|
In addition to the tie() interface, it is also possible to access most
|
|
of the functions provided in the Berkeley DB API directly.
|
|
See L<THE API INTERFACE>.
|
|
|
|
=head2 Opening a Berkeley DB Database File
|
|
|
|
Berkeley DB uses the function dbopen() to open or create a database.
|
|
Here is the C prototype for dbopen():
|
|
|
|
DB*
|
|
dbopen (const char * file, int flags, int mode,
|
|
DBTYPE type, const void * openinfo)
|
|
|
|
The parameter C<type> is an enumeration which specifies which of the 3
|
|
interface methods (DB_HASH, DB_BTREE or DB_RECNO) is to be used.
|
|
Depending on which of these is actually chosen, the final parameter,
|
|
I<openinfo> points to a data structure which allows tailoring of the
|
|
specific interface method.
|
|
|
|
This interface is handled slightly differently in B<DB_File>. Here is
|
|
an equivalent call using B<DB_File>:
|
|
|
|
tie %array, 'DB_File', $filename, $flags, $mode, $DB_HASH ;
|
|
|
|
The C<filename>, C<flags> and C<mode> parameters are the direct
|
|
equivalent of their dbopen() counterparts. The final parameter $DB_HASH
|
|
performs the function of both the C<type> and C<openinfo> parameters in
|
|
dbopen().
|
|
|
|
In the example above $DB_HASH is actually a pre-defined reference to a
|
|
hash object. B<DB_File> has three of these pre-defined references.
|
|
Apart from $DB_HASH, there is also $DB_BTREE and $DB_RECNO.
|
|
|
|
The keys allowed in each of these pre-defined references is limited to
|
|
the names used in the equivalent C structure. So, for example, the
|
|
$DB_HASH reference will only allow keys called C<bsize>, C<cachesize>,
|
|
C<ffactor>, C<hash>, C<lorder> and C<nelem>.
|
|
|
|
To change one of these elements, just assign to it like this:
|
|
|
|
$DB_HASH->{'cachesize'} = 10000 ;
|
|
|
|
The three predefined variables $DB_HASH, $DB_BTREE and $DB_RECNO are
|
|
usually adequate for most applications. If you do need to create extra
|
|
instances of these objects, constructors are available for each file
|
|
type.
|
|
|
|
Here are examples of the constructors and the valid options available
|
|
for DB_HASH, DB_BTREE and DB_RECNO respectively.
|
|
|
|
$a = new DB_File::HASHINFO ;
|
|
$a->{'bsize'} ;
|
|
$a->{'cachesize'} ;
|
|
$a->{'ffactor'};
|
|
$a->{'hash'} ;
|
|
$a->{'lorder'} ;
|
|
$a->{'nelem'} ;
|
|
|
|
$b = new DB_File::BTREEINFO ;
|
|
$b->{'flags'} ;
|
|
$b->{'cachesize'} ;
|
|
$b->{'maxkeypage'} ;
|
|
$b->{'minkeypage'} ;
|
|
$b->{'psize'} ;
|
|
$b->{'compare'} ;
|
|
$b->{'prefix'} ;
|
|
$b->{'lorder'} ;
|
|
|
|
$c = new DB_File::RECNOINFO ;
|
|
$c->{'bval'} ;
|
|
$c->{'cachesize'} ;
|
|
$c->{'psize'} ;
|
|
$c->{'flags'} ;
|
|
$c->{'lorder'} ;
|
|
$c->{'reclen'} ;
|
|
$c->{'bfname'} ;
|
|
|
|
The values stored in the hashes above are mostly the direct equivalent
|
|
of their C counterpart. Like their C counterparts, all are set to a
|
|
default values - that means you don't have to set I<all> of the
|
|
values when you only want to change one. Here is an example:
|
|
|
|
$a = new DB_File::HASHINFO ;
|
|
$a->{'cachesize'} = 12345 ;
|
|
tie %y, 'DB_File', "filename", $flags, 0777, $a ;
|
|
|
|
A few of the options need extra discussion here. When used, the C
|
|
equivalent of the keys C<hash>, C<compare> and C<prefix> store pointers
|
|
to C functions. In B<DB_File> these keys are used to store references
|
|
to Perl subs. Below are templates for each of the subs:
|
|
|
|
sub hash
|
|
{
|
|
my ($data) = @_ ;
|
|
...
|
|
# return the hash value for $data
|
|
return $hash ;
|
|
}
|
|
|
|
sub compare
|
|
{
|
|
my ($key, $key2) = @_ ;
|
|
...
|
|
# return 0 if $key1 eq $key2
|
|
# -1 if $key1 lt $key2
|
|
# 1 if $key1 gt $key2
|
|
return (-1 , 0 or 1) ;
|
|
}
|
|
|
|
sub prefix
|
|
{
|
|
my ($key, $key2) = @_ ;
|
|
...
|
|
# return number of bytes of $key2 which are
|
|
# necessary to determine that it is greater than $key1
|
|
return $bytes ;
|
|
}
|
|
|
|
See L<Changing the BTREE sort order> for an example of using the
|
|
C<compare> template.
|
|
|
|
If you are using the DB_RECNO interface and you intend making use of
|
|
C<bval>, you should check out L<The 'bval' Option>.
|
|
|
|
=head2 Default Parameters
|
|
|
|
It is possible to omit some or all of the final 4 parameters in the
|
|
call to C<tie> and let them take default values. As DB_HASH is the most
|
|
common file format used, the call:
|
|
|
|
tie %A, "DB_File", "filename" ;
|
|
|
|
is equivalent to:
|
|
|
|
tie %A, "DB_File", "filename", O_CREAT|O_RDWR, 0666, $DB_HASH ;
|
|
|
|
It is also possible to omit the filename parameter as well, so the
|
|
call:
|
|
|
|
tie %A, "DB_File" ;
|
|
|
|
is equivalent to:
|
|
|
|
tie %A, "DB_File", undef, O_CREAT|O_RDWR, 0666, $DB_HASH ;
|
|
|
|
See L<In Memory Databases> for a discussion on the use of C<undef>
|
|
in place of a filename.
|
|
|
|
=head2 In Memory Databases
|
|
|
|
Berkeley DB allows the creation of in-memory databases by using NULL
|
|
(that is, a C<(char *)0> in C) in place of the filename. B<DB_File>
|
|
uses C<undef> instead of NULL to provide this functionality.
|
|
|
|
=head1 DB_HASH
|
|
|
|
The DB_HASH file format is probably the most commonly used of the three
|
|
file formats that B<DB_File> supports. It is also very straightforward
|
|
to use.
|
|
|
|
=head2 A Simple Example
|
|
|
|
This example shows how to create a database, add key/value pairs to the
|
|
database, delete keys/value pairs and finally how to enumerate the
|
|
contents of the database.
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
our (%h, $k, $v) ;
|
|
|
|
unlink "fruit" ;
|
|
tie %h, "DB_File", "fruit", O_RDWR|O_CREAT, 0666, $DB_HASH
|
|
or die "Cannot open file 'fruit': $!\n";
|
|
|
|
# Add a few key/value pairs to the file
|
|
$h{"apple"} = "red" ;
|
|
$h{"orange"} = "orange" ;
|
|
$h{"banana"} = "yellow" ;
|
|
$h{"tomato"} = "red" ;
|
|
|
|
# Check for existence of a key
|
|
print "Banana Exists\n\n" if $h{"banana"} ;
|
|
|
|
# Delete a key/value pair.
|
|
delete $h{"apple"} ;
|
|
|
|
# print the contents of the file
|
|
while (($k, $v) = each %h)
|
|
{ print "$k -> $v\n" }
|
|
|
|
untie %h ;
|
|
|
|
here is the output:
|
|
|
|
Banana Exists
|
|
|
|
orange -> orange
|
|
tomato -> red
|
|
banana -> yellow
|
|
|
|
Note that the like ordinary associative arrays, the order of the keys
|
|
retrieved is in an apparently random order.
|
|
|
|
=head1 DB_BTREE
|
|
|
|
The DB_BTREE format is useful when you want to store data in a given
|
|
order. By default the keys will be stored in lexical order, but as you
|
|
will see from the example shown in the next section, it is very easy to
|
|
define your own sorting function.
|
|
|
|
=head2 Changing the BTREE sort order
|
|
|
|
This script shows how to override the default sorting algorithm that
|
|
BTREE uses. Instead of using the normal lexical ordering, a case
|
|
insensitive compare function will be used.
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
|
|
my %h ;
|
|
|
|
sub Compare
|
|
{
|
|
my ($key1, $key2) = @_ ;
|
|
"\L$key1" cmp "\L$key2" ;
|
|
}
|
|
|
|
# specify the Perl sub that will do the comparison
|
|
$DB_BTREE->{'compare'} = \&Compare ;
|
|
|
|
unlink "tree" ;
|
|
tie %h, "DB_File", "tree", O_RDWR|O_CREAT, 0666, $DB_BTREE
|
|
or die "Cannot open file 'tree': $!\n" ;
|
|
|
|
# Add a key/value pair to the file
|
|
$h{'Wall'} = 'Larry' ;
|
|
$h{'Smith'} = 'John' ;
|
|
$h{'mouse'} = 'mickey' ;
|
|
$h{'duck'} = 'donald' ;
|
|
|
|
# Delete
|
|
delete $h{"duck"} ;
|
|
|
|
# Cycle through the keys printing them in order.
|
|
# Note it is not necessary to sort the keys as
|
|
# the btree will have kept them in order automatically.
|
|
foreach (keys %h)
|
|
{ print "$_\n" }
|
|
|
|
untie %h ;
|
|
|
|
Here is the output from the code above.
|
|
|
|
mouse
|
|
Smith
|
|
Wall
|
|
|
|
There are a few point to bear in mind if you want to change the
|
|
ordering in a BTREE database:
|
|
|
|
=over 5
|
|
|
|
=item 1.
|
|
|
|
The new compare function must be specified when you create the database.
|
|
|
|
=item 2.
|
|
|
|
You cannot change the ordering once the database has been created. Thus
|
|
you must use the same compare function every time you access the
|
|
database.
|
|
|
|
=item 3
|
|
|
|
Duplicate keys are entirely defined by the comparison function.
|
|
In the case-insensitive example above, the keys: 'KEY' and 'key'
|
|
would be considered duplicates, and assigning to the second one
|
|
would overwrite the first. If duplicates are allowed for (with the
|
|
R_DUPS flag discussed below), only a single copy of duplicate keys
|
|
is stored in the database --- so (again with example above) assigning
|
|
three values to the keys: 'KEY', 'Key', and 'key' would leave just
|
|
the first key: 'KEY' in the database with three values. For some
|
|
situations this results in information loss, so care should be taken
|
|
to provide fully qualified comparison functions when necessary.
|
|
For example, the above comparison routine could be modified to
|
|
additionally compare case-sensitively if two keys are equal in the
|
|
case insensitive comparison:
|
|
|
|
sub compare {
|
|
my($key1, $key2) = @_;
|
|
lc $key1 cmp lc $key2 ||
|
|
$key1 cmp $key2;
|
|
}
|
|
|
|
And now you will only have duplicates when the keys themselves
|
|
are truly the same. (note: in versions of the db library prior to
|
|
about November 1996, such duplicate keys were retained so it was
|
|
possible to recover the original keys in sets of keys that
|
|
compared as equal).
|
|
|
|
|
|
=back
|
|
|
|
=head2 Handling Duplicate Keys
|
|
|
|
The BTREE file type optionally allows a single key to be associated
|
|
with an arbitrary number of values. This option is enabled by setting
|
|
the flags element of C<$DB_BTREE> to R_DUP when creating the database.
|
|
|
|
There are some difficulties in using the tied hash interface if you
|
|
want to manipulate a BTREE database with duplicate keys. Consider this
|
|
code:
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
|
|
my ($filename, %h) ;
|
|
|
|
$filename = "tree" ;
|
|
unlink $filename ;
|
|
|
|
# Enable duplicate records
|
|
$DB_BTREE->{'flags'} = R_DUP ;
|
|
|
|
tie %h, "DB_File", $filename, O_RDWR|O_CREAT, 0666, $DB_BTREE
|
|
or die "Cannot open $filename: $!\n";
|
|
|
|
# Add some key/value pairs to the file
|
|
$h{'Wall'} = 'Larry' ;
|
|
$h{'Wall'} = 'Brick' ; # Note the duplicate key
|
|
$h{'Wall'} = 'Brick' ; # Note the duplicate key and value
|
|
$h{'Smith'} = 'John' ;
|
|
$h{'mouse'} = 'mickey' ;
|
|
|
|
# iterate through the associative array
|
|
# and print each key/value pair.
|
|
foreach (sort keys %h)
|
|
{ print "$_ -> $h{$_}\n" }
|
|
|
|
untie %h ;
|
|
|
|
Here is the output:
|
|
|
|
Smith -> John
|
|
Wall -> Larry
|
|
Wall -> Larry
|
|
Wall -> Larry
|
|
mouse -> mickey
|
|
|
|
As you can see 3 records have been successfully created with key C<Wall>
|
|
- the only thing is, when they are retrieved from the database they
|
|
I<seem> to have the same value, namely C<Larry>. The problem is caused
|
|
by the way that the associative array interface works. Basically, when
|
|
the associative array interface is used to fetch the value associated
|
|
with a given key, it will only ever retrieve the first value.
|
|
|
|
Although it may not be immediately obvious from the code above, the
|
|
associative array interface can be used to write values with duplicate
|
|
keys, but it cannot be used to read them back from the database.
|
|
|
|
The way to get around this problem is to use the Berkeley DB API method
|
|
called C<seq>. This method allows sequential access to key/value
|
|
pairs. See L<THE API INTERFACE> for details of both the C<seq> method
|
|
and the API in general.
|
|
|
|
Here is the script above rewritten using the C<seq> API method.
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
|
|
my ($filename, $x, %h, $status, $key, $value) ;
|
|
|
|
$filename = "tree" ;
|
|
unlink $filename ;
|
|
|
|
# Enable duplicate records
|
|
$DB_BTREE->{'flags'} = R_DUP ;
|
|
|
|
$x = tie %h, "DB_File", $filename, O_RDWR|O_CREAT, 0666, $DB_BTREE
|
|
or die "Cannot open $filename: $!\n";
|
|
|
|
# Add some key/value pairs to the file
|
|
$h{'Wall'} = 'Larry' ;
|
|
$h{'Wall'} = 'Brick' ; # Note the duplicate key
|
|
$h{'Wall'} = 'Brick' ; # Note the duplicate key and value
|
|
$h{'Smith'} = 'John' ;
|
|
$h{'mouse'} = 'mickey' ;
|
|
|
|
# iterate through the btree using seq
|
|
# and print each key/value pair.
|
|
$key = $value = 0 ;
|
|
for ($status = $x->seq($key, $value, R_FIRST) ;
|
|
$status == 0 ;
|
|
$status = $x->seq($key, $value, R_NEXT) )
|
|
{ print "$key -> $value\n" }
|
|
|
|
undef $x ;
|
|
untie %h ;
|
|
|
|
that prints:
|
|
|
|
Smith -> John
|
|
Wall -> Brick
|
|
Wall -> Brick
|
|
Wall -> Larry
|
|
mouse -> mickey
|
|
|
|
This time we have got all the key/value pairs, including the multiple
|
|
values associated with the key C<Wall>.
|
|
|
|
To make life easier when dealing with duplicate keys, B<DB_File> comes with
|
|
a few utility methods.
|
|
|
|
=head2 The get_dup() Method
|
|
|
|
The C<get_dup> method assists in
|
|
reading duplicate values from BTREE databases. The method can take the
|
|
following forms:
|
|
|
|
$count = $x->get_dup($key) ;
|
|
@list = $x->get_dup($key) ;
|
|
%list = $x->get_dup($key, 1) ;
|
|
|
|
In a scalar context the method returns the number of values associated
|
|
with the key, C<$key>.
|
|
|
|
In list context, it returns all the values which match C<$key>. Note
|
|
that the values will be returned in an apparently random order.
|
|
|
|
In list context, if the second parameter is present and evaluates
|
|
TRUE, the method returns an associative array. The keys of the
|
|
associative array correspond to the values that matched in the BTREE
|
|
and the values of the array are a count of the number of times that
|
|
particular value occurred in the BTREE.
|
|
|
|
So assuming the database created above, we can use C<get_dup> like
|
|
this:
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
|
|
my ($filename, $x, %h) ;
|
|
|
|
$filename = "tree" ;
|
|
|
|
# Enable duplicate records
|
|
$DB_BTREE->{'flags'} = R_DUP ;
|
|
|
|
$x = tie %h, "DB_File", $filename, O_RDWR|O_CREAT, 0666, $DB_BTREE
|
|
or die "Cannot open $filename: $!\n";
|
|
|
|
my $cnt = $x->get_dup("Wall") ;
|
|
print "Wall occurred $cnt times\n" ;
|
|
|
|
my %hash = $x->get_dup("Wall", 1) ;
|
|
print "Larry is there\n" if $hash{'Larry'} ;
|
|
print "There are $hash{'Brick'} Brick Walls\n" ;
|
|
|
|
my @list = sort $x->get_dup("Wall") ;
|
|
print "Wall => [@list]\n" ;
|
|
|
|
@list = $x->get_dup("Smith") ;
|
|
print "Smith => [@list]\n" ;
|
|
|
|
@list = $x->get_dup("Dog") ;
|
|
print "Dog => [@list]\n" ;
|
|
|
|
|
|
and it will print:
|
|
|
|
Wall occurred 3 times
|
|
Larry is there
|
|
There are 2 Brick Walls
|
|
Wall => [Brick Brick Larry]
|
|
Smith => [John]
|
|
Dog => []
|
|
|
|
=head2 The find_dup() Method
|
|
|
|
$status = $X->find_dup($key, $value) ;
|
|
|
|
This method checks for the existence of a specific key/value pair. If the
|
|
pair exists, the cursor is left pointing to the pair and the method
|
|
returns 0. Otherwise the method returns a non-zero value.
|
|
|
|
Assuming the database from the previous example:
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
|
|
my ($filename, $x, %h, $found) ;
|
|
|
|
$filename = "tree" ;
|
|
|
|
# Enable duplicate records
|
|
$DB_BTREE->{'flags'} = R_DUP ;
|
|
|
|
$x = tie %h, "DB_File", $filename, O_RDWR|O_CREAT, 0666, $DB_BTREE
|
|
or die "Cannot open $filename: $!\n";
|
|
|
|
$found = ( $x->find_dup("Wall", "Larry") == 0 ? "" : "not") ;
|
|
print "Larry Wall is $found there\n" ;
|
|
|
|
$found = ( $x->find_dup("Wall", "Harry") == 0 ? "" : "not") ;
|
|
print "Harry Wall is $found there\n" ;
|
|
|
|
undef $x ;
|
|
untie %h ;
|
|
|
|
prints this
|
|
|
|
Larry Wall is there
|
|
Harry Wall is not there
|
|
|
|
|
|
=head2 The del_dup() Method
|
|
|
|
$status = $X->del_dup($key, $value) ;
|
|
|
|
This method deletes a specific key/value pair. It returns
|
|
0 if they exist and have been deleted successfully.
|
|
Otherwise the method returns a non-zero value.
|
|
|
|
Again assuming the existence of the C<tree> database
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
|
|
my ($filename, $x, %h, $found) ;
|
|
|
|
$filename = "tree" ;
|
|
|
|
# Enable duplicate records
|
|
$DB_BTREE->{'flags'} = R_DUP ;
|
|
|
|
$x = tie %h, "DB_File", $filename, O_RDWR|O_CREAT, 0666, $DB_BTREE
|
|
or die "Cannot open $filename: $!\n";
|
|
|
|
$x->del_dup("Wall", "Larry") ;
|
|
|
|
$found = ( $x->find_dup("Wall", "Larry") == 0 ? "" : "not") ;
|
|
print "Larry Wall is $found there\n" ;
|
|
|
|
undef $x ;
|
|
untie %h ;
|
|
|
|
prints this
|
|
|
|
Larry Wall is not there
|
|
|
|
=head2 Matching Partial Keys
|
|
|
|
The BTREE interface has a feature which allows partial keys to be
|
|
matched. This functionality is I<only> available when the C<seq> method
|
|
is used along with the R_CURSOR flag.
|
|
|
|
$x->seq($key, $value, R_CURSOR) ;
|
|
|
|
Here is the relevant quote from the dbopen man page where it defines
|
|
the use of the R_CURSOR flag with seq:
|
|
|
|
Note, for the DB_BTREE access method, the returned key is not
|
|
necessarily an exact match for the specified key. The returned key
|
|
is the smallest key greater than or equal to the specified key,
|
|
permitting partial key matches and range searches.
|
|
|
|
In the example script below, the C<match> sub uses this feature to find
|
|
and print the first matching key/value pair given a partial key.
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
use Fcntl ;
|
|
|
|
my ($filename, $x, %h, $st, $key, $value) ;
|
|
|
|
sub match
|
|
{
|
|
my $key = shift ;
|
|
my $value = 0;
|
|
my $orig_key = $key ;
|
|
$x->seq($key, $value, R_CURSOR) ;
|
|
print "$orig_key\t-> $key\t-> $value\n" ;
|
|
}
|
|
|
|
$filename = "tree" ;
|
|
unlink $filename ;
|
|
|
|
$x = tie %h, "DB_File", $filename, O_RDWR|O_CREAT, 0666, $DB_BTREE
|
|
or die "Cannot open $filename: $!\n";
|
|
|
|
# Add some key/value pairs to the file
|
|
$h{'mouse'} = 'mickey' ;
|
|
$h{'Wall'} = 'Larry' ;
|
|
$h{'Walls'} = 'Brick' ;
|
|
$h{'Smith'} = 'John' ;
|
|
|
|
|
|
$key = $value = 0 ;
|
|
print "IN ORDER\n" ;
|
|
for ($st = $x->seq($key, $value, R_FIRST) ;
|
|
$st == 0 ;
|
|
$st = $x->seq($key, $value, R_NEXT) )
|
|
|
|
{ print "$key -> $value\n" }
|
|
|
|
print "\nPARTIAL MATCH\n" ;
|
|
|
|
match "Wa" ;
|
|
match "A" ;
|
|
match "a" ;
|
|
|
|
undef $x ;
|
|
untie %h ;
|
|
|
|
Here is the output:
|
|
|
|
IN ORDER
|
|
Smith -> John
|
|
Wall -> Larry
|
|
Walls -> Brick
|
|
mouse -> mickey
|
|
|
|
PARTIAL MATCH
|
|
Wa -> Wall -> Larry
|
|
A -> Smith -> John
|
|
a -> mouse -> mickey
|
|
|
|
=head1 DB_RECNO
|
|
|
|
DB_RECNO provides an interface to flat text files. Both variable and
|
|
fixed length records are supported.
|
|
|
|
In order to make RECNO more compatible with Perl, the array offset for
|
|
all RECNO arrays begins at 0 rather than 1 as in Berkeley DB.
|
|
|
|
As with normal Perl arrays, a RECNO array can be accessed using
|
|
negative indexes. The index -1 refers to the last element of the array,
|
|
-2 the second last, and so on. Attempting to access an element before
|
|
the start of the array will raise a fatal run-time error.
|
|
|
|
=head2 The 'bval' Option
|
|
|
|
The operation of the bval option warrants some discussion. Here is the
|
|
definition of bval from the Berkeley DB 1.85 recno manual page:
|
|
|
|
The delimiting byte to be used to mark the end of a
|
|
record for variable-length records, and the pad charac-
|
|
ter for fixed-length records. If no value is speci-
|
|
fied, newlines (``\n'') are used to mark the end of
|
|
variable-length records and fixed-length records are
|
|
padded with spaces.
|
|
|
|
The second sentence is wrong. In actual fact bval will only default to
|
|
C<"\n"> when the openinfo parameter in dbopen is NULL. If a non-NULL
|
|
openinfo parameter is used at all, the value that happens to be in bval
|
|
will be used. That means you always have to specify bval when making
|
|
use of any of the options in the openinfo parameter. This documentation
|
|
error will be fixed in the next release of Berkeley DB.
|
|
|
|
That clarifies the situation with regards Berkeley DB itself. What
|
|
about B<DB_File>? Well, the behavior defined in the quote above is
|
|
quite useful, so B<DB_File> conforms to it.
|
|
|
|
That means that you can specify other options (e.g. cachesize) and
|
|
still have bval default to C<"\n"> for variable length records, and
|
|
space for fixed length records.
|
|
|
|
Also note that the bval option only allows you to specify a single byte
|
|
as a delimeter.
|
|
|
|
=head2 A Simple Example
|
|
|
|
Here is a simple example that uses RECNO (if you are using a version
|
|
of Perl earlier than 5.004_57 this example won't work -- see
|
|
L<Extra RECNO Methods> for a workaround).
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
|
|
my $filename = "text" ;
|
|
unlink $filename ;
|
|
|
|
my @h ;
|
|
tie @h, "DB_File", $filename, O_RDWR|O_CREAT, 0666, $DB_RECNO
|
|
or die "Cannot open file 'text': $!\n" ;
|
|
|
|
# Add a few key/value pairs to the file
|
|
$h[0] = "orange" ;
|
|
$h[1] = "blue" ;
|
|
$h[2] = "yellow" ;
|
|
|
|
push @h, "green", "black" ;
|
|
|
|
my $elements = scalar @h ;
|
|
print "The array contains $elements entries\n" ;
|
|
|
|
my $last = pop @h ;
|
|
print "popped $last\n" ;
|
|
|
|
unshift @h, "white" ;
|
|
my $first = shift @h ;
|
|
print "shifted $first\n" ;
|
|
|
|
# Check for existence of a key
|
|
print "Element 1 Exists with value $h[1]\n" if $h[1] ;
|
|
|
|
# use a negative index
|
|
print "The last element is $h[-1]\n" ;
|
|
print "The 2nd last element is $h[-2]\n" ;
|
|
|
|
untie @h ;
|
|
|
|
Here is the output from the script:
|
|
|
|
The array contains 5 entries
|
|
popped black
|
|
shifted white
|
|
Element 1 Exists with value blue
|
|
The last element is green
|
|
The 2nd last element is yellow
|
|
|
|
=head2 Extra RECNO Methods
|
|
|
|
If you are using a version of Perl earlier than 5.004_57, the tied
|
|
array interface is quite limited. In the example script above
|
|
C<push>, C<pop>, C<shift>, C<unshift>
|
|
or determining the array length will not work with a tied array.
|
|
|
|
To make the interface more useful for older versions of Perl, a number
|
|
of methods are supplied with B<DB_File> to simulate the missing array
|
|
operations. All these methods are accessed via the object returned from
|
|
the tie call.
|
|
|
|
Here are the methods:
|
|
|
|
=over 5
|
|
|
|
=item B<$X-E<gt>push(list) ;>
|
|
|
|
Pushes the elements of C<list> to the end of the array.
|
|
|
|
=item B<$value = $X-E<gt>pop ;>
|
|
|
|
Removes and returns the last element of the array.
|
|
|
|
=item B<$X-E<gt>shift>
|
|
|
|
Removes and returns the first element of the array.
|
|
|
|
=item B<$X-E<gt>unshift(list) ;>
|
|
|
|
Pushes the elements of C<list> to the start of the array.
|
|
|
|
=item B<$X-E<gt>length>
|
|
|
|
Returns the number of elements in the array.
|
|
|
|
=item B<$X-E<gt>splice(offset, length, elements);>
|
|
|
|
Returns a splice of the the array.
|
|
|
|
=back
|
|
|
|
=head2 Another Example
|
|
|
|
Here is a more complete example that makes use of some of the methods
|
|
described above. It also makes use of the API interface directly (see
|
|
L<THE API INTERFACE>).
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
my (@h, $H, $file, $i) ;
|
|
use DB_File ;
|
|
use Fcntl ;
|
|
|
|
$file = "text" ;
|
|
|
|
unlink $file ;
|
|
|
|
$H = tie @h, "DB_File", $file, O_RDWR|O_CREAT, 0666, $DB_RECNO
|
|
or die "Cannot open file $file: $!\n" ;
|
|
|
|
# first create a text file to play with
|
|
$h[0] = "zero" ;
|
|
$h[1] = "one" ;
|
|
$h[2] = "two" ;
|
|
$h[3] = "three" ;
|
|
$h[4] = "four" ;
|
|
|
|
|
|
# Print the records in order.
|
|
#
|
|
# The length method is needed here because evaluating a tied
|
|
# array in a scalar context does not return the number of
|
|
# elements in the array.
|
|
|
|
print "\nORIGINAL\n" ;
|
|
foreach $i (0 .. $H->length - 1) {
|
|
print "$i: $h[$i]\n" ;
|
|
}
|
|
|
|
# use the push & pop methods
|
|
$a = $H->pop ;
|
|
$H->push("last") ;
|
|
print "\nThe last record was [$a]\n" ;
|
|
|
|
# and the shift & unshift methods
|
|
$a = $H->shift ;
|
|
$H->unshift("first") ;
|
|
print "The first record was [$a]\n" ;
|
|
|
|
# Use the API to add a new record after record 2.
|
|
$i = 2 ;
|
|
$H->put($i, "Newbie", R_IAFTER) ;
|
|
|
|
# and a new record before record 1.
|
|
$i = 1 ;
|
|
$H->put($i, "New One", R_IBEFORE) ;
|
|
|
|
# delete record 3
|
|
$H->del(3) ;
|
|
|
|
# now print the records in reverse order
|
|
print "\nREVERSE\n" ;
|
|
for ($i = $H->length - 1 ; $i >= 0 ; -- $i)
|
|
{ print "$i: $h[$i]\n" }
|
|
|
|
# same again, but use the API functions instead
|
|
print "\nREVERSE again\n" ;
|
|
my ($s, $k, $v) = (0, 0, 0) ;
|
|
for ($s = $H->seq($k, $v, R_LAST) ;
|
|
$s == 0 ;
|
|
$s = $H->seq($k, $v, R_PREV))
|
|
{ print "$k: $v\n" }
|
|
|
|
undef $H ;
|
|
untie @h ;
|
|
|
|
and this is what it outputs:
|
|
|
|
ORIGINAL
|
|
0: zero
|
|
1: one
|
|
2: two
|
|
3: three
|
|
4: four
|
|
|
|
The last record was [four]
|
|
The first record was [zero]
|
|
|
|
REVERSE
|
|
5: last
|
|
4: three
|
|
3: Newbie
|
|
2: one
|
|
1: New One
|
|
0: first
|
|
|
|
REVERSE again
|
|
5: last
|
|
4: three
|
|
3: Newbie
|
|
2: one
|
|
1: New One
|
|
0: first
|
|
|
|
Notes:
|
|
|
|
=over 5
|
|
|
|
=item 1.
|
|
|
|
Rather than iterating through the array, C<@h> like this:
|
|
|
|
foreach $i (@h)
|
|
|
|
it is necessary to use either this:
|
|
|
|
foreach $i (0 .. $H->length - 1)
|
|
|
|
or this:
|
|
|
|
for ($a = $H->get($k, $v, R_FIRST) ;
|
|
$a == 0 ;
|
|
$a = $H->get($k, $v, R_NEXT) )
|
|
|
|
=item 2.
|
|
|
|
Notice that both times the C<put> method was used the record index was
|
|
specified using a variable, C<$i>, rather than the literal value
|
|
itself. This is because C<put> will return the record number of the
|
|
inserted line via that parameter.
|
|
|
|
=back
|
|
|
|
=head1 THE API INTERFACE
|
|
|
|
As well as accessing Berkeley DB using a tied hash or array, it is also
|
|
possible to make direct use of most of the API functions defined in the
|
|
Berkeley DB documentation.
|
|
|
|
To do this you need to store a copy of the object returned from the tie.
|
|
|
|
$db = tie %hash, "DB_File", "filename" ;
|
|
|
|
Once you have done that, you can access the Berkeley DB API functions
|
|
as B<DB_File> methods directly like this:
|
|
|
|
$db->put($key, $value, R_NOOVERWRITE) ;
|
|
|
|
B<Important:> If you have saved a copy of the object returned from
|
|
C<tie>, the underlying database file will I<not> be closed until both
|
|
the tied variable is untied and all copies of the saved object are
|
|
destroyed.
|
|
|
|
use DB_File ;
|
|
$db = tie %hash, "DB_File", "filename"
|
|
or die "Cannot tie filename: $!" ;
|
|
...
|
|
undef $db ;
|
|
untie %hash ;
|
|
|
|
See L<The untie() Gotcha> for more details.
|
|
|
|
All the functions defined in L<dbopen> are available except for
|
|
close() and dbopen() itself. The B<DB_File> method interface to the
|
|
supported functions have been implemented to mirror the way Berkeley DB
|
|
works whenever possible. In particular note that:
|
|
|
|
=over 5
|
|
|
|
=item *
|
|
|
|
The methods return a status value. All return 0 on success.
|
|
All return -1 to signify an error and set C<$!> to the exact
|
|
error code. The return code 1 generally (but not always) means that the
|
|
key specified did not exist in the database.
|
|
|
|
Other return codes are defined. See below and in the Berkeley DB
|
|
documentation for details. The Berkeley DB documentation should be used
|
|
as the definitive source.
|
|
|
|
=item *
|
|
|
|
Whenever a Berkeley DB function returns data via one of its parameters,
|
|
the equivalent B<DB_File> method does exactly the same.
|
|
|
|
=item *
|
|
|
|
If you are careful, it is possible to mix API calls with the tied
|
|
hash/array interface in the same piece of code. Although only a few of
|
|
the methods used to implement the tied interface currently make use of
|
|
the cursor, you should always assume that the cursor has been changed
|
|
any time the tied hash/array interface is used. As an example, this
|
|
code will probably not do what you expect:
|
|
|
|
$X = tie %x, 'DB_File', $filename, O_RDWR|O_CREAT, 0777, $DB_BTREE
|
|
or die "Cannot tie $filename: $!" ;
|
|
|
|
# Get the first key/value pair and set the cursor
|
|
$X->seq($key, $value, R_FIRST) ;
|
|
|
|
# this line will modify the cursor
|
|
$count = scalar keys %x ;
|
|
|
|
# Get the second key/value pair.
|
|
# oops, it didn't, it got the last key/value pair!
|
|
$X->seq($key, $value, R_NEXT) ;
|
|
|
|
The code above can be rearranged to get around the problem, like this:
|
|
|
|
$X = tie %x, 'DB_File', $filename, O_RDWR|O_CREAT, 0777, $DB_BTREE
|
|
or die "Cannot tie $filename: $!" ;
|
|
|
|
# this line will modify the cursor
|
|
$count = scalar keys %x ;
|
|
|
|
# Get the first key/value pair and set the cursor
|
|
$X->seq($key, $value, R_FIRST) ;
|
|
|
|
# Get the second key/value pair.
|
|
# worked this time.
|
|
$X->seq($key, $value, R_NEXT) ;
|
|
|
|
=back
|
|
|
|
All the constants defined in L<dbopen> for use in the flags parameters
|
|
in the methods defined below are also available. Refer to the Berkeley
|
|
DB documentation for the precise meaning of the flags values.
|
|
|
|
Below is a list of the methods available.
|
|
|
|
=over 5
|
|
|
|
=item B<$status = $X-E<gt>get($key, $value [, $flags]) ;>
|
|
|
|
Given a key (C<$key>) this method reads the value associated with it
|
|
from the database. The value read from the database is returned in the
|
|
C<$value> parameter.
|
|
|
|
If the key does not exist the method returns 1.
|
|
|
|
No flags are currently defined for this method.
|
|
|
|
=item B<$status = $X-E<gt>put($key, $value [, $flags]) ;>
|
|
|
|
Stores the key/value pair in the database.
|
|
|
|
If you use either the R_IAFTER or R_IBEFORE flags, the C<$key> parameter
|
|
will have the record number of the inserted key/value pair set.
|
|
|
|
Valid flags are R_CURSOR, R_IAFTER, R_IBEFORE, R_NOOVERWRITE and
|
|
R_SETCURSOR.
|
|
|
|
=item B<$status = $X-E<gt>del($key [, $flags]) ;>
|
|
|
|
Removes all key/value pairs with key C<$key> from the database.
|
|
|
|
A return code of 1 means that the requested key was not in the
|
|
database.
|
|
|
|
R_CURSOR is the only valid flag at present.
|
|
|
|
=item B<$status = $X-E<gt>fd ;>
|
|
|
|
Returns the file descriptor for the underlying database.
|
|
|
|
See L<Locking: The Trouble with fd> for an explanation for why you should
|
|
not use C<fd> to lock your database.
|
|
|
|
=item B<$status = $X-E<gt>seq($key, $value, $flags) ;>
|
|
|
|
This interface allows sequential retrieval from the database. See
|
|
L<dbopen> for full details.
|
|
|
|
Both the C<$key> and C<$value> parameters will be set to the key/value
|
|
pair read from the database.
|
|
|
|
The flags parameter is mandatory. The valid flag values are R_CURSOR,
|
|
R_FIRST, R_LAST, R_NEXT and R_PREV.
|
|
|
|
=item B<$status = $X-E<gt>sync([$flags]) ;>
|
|
|
|
Flushes any cached buffers to disk.
|
|
|
|
R_RECNOSYNC is the only valid flag at present.
|
|
|
|
=back
|
|
|
|
=head1 DBM FILTERS
|
|
|
|
A DBM Filter is a piece of code that is be used when you I<always>
|
|
want to make the same transformation to all keys and/or values in a
|
|
DBM database.
|
|
|
|
There are four methods associated with DBM Filters. All work identically,
|
|
and each is used to install (or uninstall) a single DBM Filter. Each
|
|
expects a single parameter, namely a reference to a sub. The only
|
|
difference between them is the place that the filter is installed.
|
|
|
|
To summarise:
|
|
|
|
=over 5
|
|
|
|
=item B<filter_store_key>
|
|
|
|
If a filter has been installed with this method, it will be invoked
|
|
every time you write a key to a DBM database.
|
|
|
|
=item B<filter_store_value>
|
|
|
|
If a filter has been installed with this method, it will be invoked
|
|
every time you write a value to a DBM database.
|
|
|
|
|
|
=item B<filter_fetch_key>
|
|
|
|
If a filter has been installed with this method, it will be invoked
|
|
every time you read a key from a DBM database.
|
|
|
|
=item B<filter_fetch_value>
|
|
|
|
If a filter has been installed with this method, it will be invoked
|
|
every time you read a value from a DBM database.
|
|
|
|
=back
|
|
|
|
You can use any combination of the methods, from none, to all four.
|
|
|
|
All filter methods return the existing filter, if present, or C<undef>
|
|
in not.
|
|
|
|
To delete a filter pass C<undef> to it.
|
|
|
|
=head2 The Filter
|
|
|
|
When each filter is called by Perl, a local copy of C<$_> will contain
|
|
the key or value to be filtered. Filtering is achieved by modifying
|
|
the contents of C<$_>. The return code from the filter is ignored.
|
|
|
|
=head2 An Example -- the NULL termination problem.
|
|
|
|
Consider the following scenario. You have a DBM database
|
|
that you need to share with a third-party C application. The C application
|
|
assumes that I<all> keys and values are NULL terminated. Unfortunately
|
|
when Perl writes to DBM databases it doesn't use NULL termination, so
|
|
your Perl application will have to manage NULL termination itself. When
|
|
you write to the database you will have to use something like this:
|
|
|
|
$hash{"$key\0"} = "$value\0" ;
|
|
|
|
Similarly the NULL needs to be taken into account when you are considering
|
|
the length of existing keys/values.
|
|
|
|
It would be much better if you could ignore the NULL terminations issue
|
|
in the main application code and have a mechanism that automatically
|
|
added the terminating NULL to all keys and values whenever you write to
|
|
the database and have them removed when you read from the database. As I'm
|
|
sure you have already guessed, this is a problem that DBM Filters can
|
|
fix very easily.
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
|
|
my %hash ;
|
|
my $filename = "/tmp/filt" ;
|
|
unlink $filename ;
|
|
|
|
my $db = tie %hash, 'DB_File', $filename, O_CREAT|O_RDWR, 0666, $DB_HASH
|
|
or die "Cannot open $filename: $!\n" ;
|
|
|
|
# Install DBM Filters
|
|
$db->filter_fetch_key ( sub { s/\0$// } ) ;
|
|
$db->filter_store_key ( sub { $_ .= "\0" } ) ;
|
|
$db->filter_fetch_value( sub { s/\0$// } ) ;
|
|
$db->filter_store_value( sub { $_ .= "\0" } ) ;
|
|
|
|
$hash{"abc"} = "def" ;
|
|
my $a = $hash{"ABC"} ;
|
|
# ...
|
|
undef $db ;
|
|
untie %hash ;
|
|
|
|
Hopefully the contents of each of the filters should be
|
|
self-explanatory. Both "fetch" filters remove the terminating NULL,
|
|
and both "store" filters add a terminating NULL.
|
|
|
|
|
|
=head2 Another Example -- Key is a C int.
|
|
|
|
Here is another real-life example. By default, whenever Perl writes to
|
|
a DBM database it always writes the key and value as strings. So when
|
|
you use this:
|
|
|
|
$hash{12345} = "soemthing" ;
|
|
|
|
the key 12345 will get stored in the DBM database as the 5 byte string
|
|
"12345". If you actually want the key to be stored in the DBM database
|
|
as a C int, you will have to use C<pack> when writing, and C<unpack>
|
|
when reading.
|
|
|
|
Here is a DBM Filter that does it:
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
my %hash ;
|
|
my $filename = "/tmp/filt" ;
|
|
unlink $filename ;
|
|
|
|
|
|
my $db = tie %hash, 'DB_File', $filename, O_CREAT|O_RDWR, 0666, $DB_HASH
|
|
or die "Cannot open $filename: $!\n" ;
|
|
|
|
$db->filter_fetch_key ( sub { $_ = unpack("i", $_) } ) ;
|
|
$db->filter_store_key ( sub { $_ = pack ("i", $_) } ) ;
|
|
$hash{123} = "def" ;
|
|
# ...
|
|
undef $db ;
|
|
untie %hash ;
|
|
|
|
This time only two filters have been used -- we only need to manipulate
|
|
the contents of the key, so it wasn't necessary to install any value
|
|
filters.
|
|
|
|
=head1 HINTS AND TIPS
|
|
|
|
|
|
=head2 Locking: The Trouble with fd
|
|
|
|
Until version 1.72 of this module, the recommended technique for locking
|
|
B<DB_File> databases was to flock the filehandle returned from the "fd"
|
|
function. Unfortunately this technique has been shown to be fundamentally
|
|
flawed (Kudos to David Harris for tracking this down). Use it at your own
|
|
peril!
|
|
|
|
The locking technique went like this.
|
|
|
|
$db = tie(%db, 'DB_File', '/tmp/foo.db', O_CREAT|O_RDWR, 0666)
|
|
|| die "dbcreat /tmp/foo.db $!";
|
|
$fd = $db->fd;
|
|
open(DB_FH, "+<&=$fd") || die "dup $!";
|
|
flock (DB_FH, LOCK_EX) || die "flock: $!";
|
|
...
|
|
$db{"Tom"} = "Jerry" ;
|
|
...
|
|
flock(DB_FH, LOCK_UN);
|
|
undef $db;
|
|
untie %db;
|
|
close(DB_FH);
|
|
|
|
In simple terms, this is what happens:
|
|
|
|
=over 5
|
|
|
|
=item 1.
|
|
|
|
Use "tie" to open the database.
|
|
|
|
=item 2.
|
|
|
|
Lock the database with fd & flock.
|
|
|
|
=item 3.
|
|
|
|
Read & Write to the database.
|
|
|
|
=item 4.
|
|
|
|
Unlock and close the database.
|
|
|
|
=back
|
|
|
|
Here is the crux of the problem. A side-effect of opening the B<DB_File>
|
|
database in step 2 is that an initial block from the database will get
|
|
read from disk and cached in memory.
|
|
|
|
To see why this is a problem, consider what can happen when two processes,
|
|
say "A" and "B", both want to update the same B<DB_File> database
|
|
using the locking steps outlined above. Assume process "A" has already
|
|
opened the database and has a write lock, but it hasn't actually updated
|
|
the database yet (it has finished step 2, but not started step 3 yet). Now
|
|
process "B" tries to open the same database - step 1 will succeed,
|
|
but it will block on step 2 until process "A" releases the lock. The
|
|
important thing to notice here is that at this point in time both
|
|
processes will have cached identical initial blocks from the database.
|
|
|
|
Now process "A" updates the database and happens to change some of the
|
|
data held in the initial buffer. Process "A" terminates, flushing
|
|
all cached data to disk and releasing the database lock. At this point
|
|
the database on disk will correctly reflect the changes made by process
|
|
"A".
|
|
|
|
With the lock released, process "B" can now continue. It also updates the
|
|
database and unfortunately it too modifies the data that was in its
|
|
initial buffer. Once that data gets flushed to disk it will overwrite
|
|
some/all of the changes process "A" made to the database.
|
|
|
|
The result of this scenario is at best a database that doesn't contain
|
|
what you expect. At worst the database will corrupt.
|
|
|
|
The above won't happen every time competing process update the same
|
|
B<DB_File> database, but it does illustrate why the technique should
|
|
not be used.
|
|
|
|
=head2 Safe ways to lock a database
|
|
|
|
Starting with version 2.x, Berkeley DB has internal support for locking.
|
|
The companion module to this one, B<BerkeleyDB>, provides an interface
|
|
to this locking functionality. If you are serious about locking
|
|
Berkeley DB databases, I strongly recommend using B<BerkeleyDB>.
|
|
|
|
If using B<BerkeleyDB> isn't an option, there are a number of modules
|
|
available on CPAN that can be used to implement locking. Each one
|
|
implements locking differently and has different goals in mind. It is
|
|
therefore worth knowing the difference, so that you can pick the right
|
|
one for your application. Here are the three locking wrappers:
|
|
|
|
=over 5
|
|
|
|
=item B<Tie::DB_Lock>
|
|
|
|
A B<DB_File> wrapper which creates copies of the database file for
|
|
read access, so that you have a kind of a multiversioning concurrent read
|
|
system. However, updates are still serial. Use for databases where reads
|
|
may be lengthy and consistency problems may occur.
|
|
|
|
=item B<Tie::DB_LockFile>
|
|
|
|
A B<DB_File> wrapper that has the ability to lock and unlock the database
|
|
while it is being used. Avoids the tie-before-flock problem by simply
|
|
re-tie-ing the database when you get or drop a lock. Because of the
|
|
flexibility in dropping and re-acquiring the lock in the middle of a
|
|
session, this can be massaged into a system that will work with long
|
|
updates and/or reads if the application follows the hints in the POD
|
|
documentation.
|
|
|
|
=item B<DB_File::Lock>
|
|
|
|
An extremely lightweight B<DB_File> wrapper that simply flocks a lockfile
|
|
before tie-ing the database and drops the lock after the untie. Allows
|
|
one to use the same lockfile for multiple databases to avoid deadlock
|
|
problems, if desired. Use for databases where updates are reads are
|
|
quick and simple flock locking semantics are enough.
|
|
|
|
=back
|
|
|
|
=head2 Sharing Databases With C Applications
|
|
|
|
There is no technical reason why a Berkeley DB database cannot be
|
|
shared by both a Perl and a C application.
|
|
|
|
The vast majority of problems that are reported in this area boil down
|
|
to the fact that C strings are NULL terminated, whilst Perl strings are
|
|
not. See L<DBM FILTERS> for a generic way to work around this problem.
|
|
|
|
Here is a real example. Netscape 2.0 keeps a record of the locations you
|
|
visit along with the time you last visited them in a DB_HASH database.
|
|
This is usually stored in the file F<~/.netscape/history.db>. The key
|
|
field in the database is the location string and the value field is the
|
|
time the location was last visited stored as a 4 byte binary value.
|
|
|
|
If you haven't already guessed, the location string is stored with a
|
|
terminating NULL. This means you need to be careful when accessing the
|
|
database.
|
|
|
|
Here is a snippet of code that is loosely based on Tom Christiansen's
|
|
I<ggh> script (available from your nearest CPAN archive in
|
|
F<authors/id/TOMC/scripts/nshist.gz>).
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
use Fcntl ;
|
|
|
|
my ($dotdir, $HISTORY, %hist_db, $href, $binary_time, $date) ;
|
|
$dotdir = $ENV{HOME} || $ENV{LOGNAME};
|
|
|
|
$HISTORY = "$dotdir/.netscape/history.db";
|
|
|
|
tie %hist_db, 'DB_File', $HISTORY
|
|
or die "Cannot open $HISTORY: $!\n" ;;
|
|
|
|
# Dump the complete database
|
|
while ( ($href, $binary_time) = each %hist_db ) {
|
|
|
|
# remove the terminating NULL
|
|
$href =~ s/\x00$// ;
|
|
|
|
# convert the binary time into a user friendly string
|
|
$date = localtime unpack("V", $binary_time);
|
|
print "$date $href\n" ;
|
|
}
|
|
|
|
# check for the existence of a specific key
|
|
# remember to add the NULL
|
|
if ( $binary_time = $hist_db{"http://mox.perl.com/\x00"} ) {
|
|
$date = localtime unpack("V", $binary_time) ;
|
|
print "Last visited mox.perl.com on $date\n" ;
|
|
}
|
|
else {
|
|
print "Never visited mox.perl.com\n"
|
|
}
|
|
|
|
untie %hist_db ;
|
|
|
|
=head2 The untie() Gotcha
|
|
|
|
If you make use of the Berkeley DB API, it is I<very> strongly
|
|
recommended that you read L<perltie/The untie Gotcha>.
|
|
|
|
Even if you don't currently make use of the API interface, it is still
|
|
worth reading it.
|
|
|
|
Here is an example which illustrates the problem from a B<DB_File>
|
|
perspective:
|
|
|
|
use DB_File ;
|
|
use Fcntl ;
|
|
|
|
my %x ;
|
|
my $X ;
|
|
|
|
$X = tie %x, 'DB_File', 'tst.fil' , O_RDWR|O_TRUNC
|
|
or die "Cannot tie first time: $!" ;
|
|
|
|
$x{123} = 456 ;
|
|
|
|
untie %x ;
|
|
|
|
tie %x, 'DB_File', 'tst.fil' , O_RDWR|O_CREAT
|
|
or die "Cannot tie second time: $!" ;
|
|
|
|
untie %x ;
|
|
|
|
When run, the script will produce this error message:
|
|
|
|
Cannot tie second time: Invalid argument at bad.file line 14.
|
|
|
|
Although the error message above refers to the second tie() statement
|
|
in the script, the source of the problem is really with the untie()
|
|
statement that precedes it.
|
|
|
|
Having read L<perltie> you will probably have already guessed that the
|
|
error is caused by the extra copy of the tied object stored in C<$X>.
|
|
If you haven't, then the problem boils down to the fact that the
|
|
B<DB_File> destructor, DESTROY, will not be called until I<all>
|
|
references to the tied object are destroyed. Both the tied variable,
|
|
C<%x>, and C<$X> above hold a reference to the object. The call to
|
|
untie() will destroy the first, but C<$X> still holds a valid
|
|
reference, so the destructor will not get called and the database file
|
|
F<tst.fil> will remain open. The fact that Berkeley DB then reports the
|
|
attempt to open a database that is already open via the catch-all
|
|
"Invalid argument" doesn't help.
|
|
|
|
If you run the script with the C<-w> flag the error message becomes:
|
|
|
|
untie attempted while 1 inner references still exist at bad.file line 12.
|
|
Cannot tie second time: Invalid argument at bad.file line 14.
|
|
|
|
which pinpoints the real problem. Finally the script can now be
|
|
modified to fix the original problem by destroying the API object
|
|
before the untie:
|
|
|
|
...
|
|
$x{123} = 456 ;
|
|
|
|
undef $X ;
|
|
untie %x ;
|
|
|
|
$X = tie %x, 'DB_File', 'tst.fil' , O_RDWR|O_CREAT
|
|
...
|
|
|
|
|
|
=head1 COMMON QUESTIONS
|
|
|
|
=head2 Why is there Perl source in my database?
|
|
|
|
If you look at the contents of a database file created by DB_File,
|
|
there can sometimes be part of a Perl script included in it.
|
|
|
|
This happens because Berkeley DB uses dynamic memory to allocate
|
|
buffers which will subsequently be written to the database file. Being
|
|
dynamic, the memory could have been used for anything before DB
|
|
malloced it. As Berkeley DB doesn't clear the memory once it has been
|
|
allocated, the unused portions will contain random junk. In the case
|
|
where a Perl script gets written to the database, the random junk will
|
|
correspond to an area of dynamic memory that happened to be used during
|
|
the compilation of the script.
|
|
|
|
Unless you don't like the possibility of there being part of your Perl
|
|
scripts embedded in a database file, this is nothing to worry about.
|
|
|
|
=head2 How do I store complex data structures with DB_File?
|
|
|
|
Although B<DB_File> cannot do this directly, there is a module which
|
|
can layer transparently over B<DB_File> to accomplish this feat.
|
|
|
|
Check out the MLDBM module, available on CPAN in the directory
|
|
F<modules/by-module/MLDBM>.
|
|
|
|
=head2 What does "Invalid Argument" mean?
|
|
|
|
You will get this error message when one of the parameters in the
|
|
C<tie> call is wrong. Unfortunately there are quite a few parameters to
|
|
get wrong, so it can be difficult to figure out which one it is.
|
|
|
|
Here are a couple of possibilities:
|
|
|
|
=over 5
|
|
|
|
=item 1.
|
|
|
|
Attempting to reopen a database without closing it.
|
|
|
|
=item 2.
|
|
|
|
Using the O_WRONLY flag.
|
|
|
|
=back
|
|
|
|
=head2 What does "Bareword 'DB_File' not allowed" mean?
|
|
|
|
You will encounter this particular error message when you have the
|
|
C<strict 'subs'> pragma (or the full strict pragma) in your script.
|
|
Consider this script:
|
|
|
|
use warnings ;
|
|
use strict ;
|
|
use DB_File ;
|
|
my %x ;
|
|
tie %x, DB_File, "filename" ;
|
|
|
|
Running it produces the error in question:
|
|
|
|
Bareword "DB_File" not allowed while "strict subs" in use
|
|
|
|
To get around the error, place the word C<DB_File> in either single or
|
|
double quotes, like this:
|
|
|
|
tie %x, "DB_File", "filename" ;
|
|
|
|
Although it might seem like a real pain, it is really worth the effort
|
|
of having a C<use strict> in all your scripts.
|
|
|
|
=head1 REFERENCES
|
|
|
|
Articles that are either about B<DB_File> or make use of it.
|
|
|
|
=over 5
|
|
|
|
=item 1.
|
|
|
|
I<Full-Text Searching in Perl>, Tim Kientzle (tkientzle@ddj.com),
|
|
Dr. Dobb's Journal, Issue 295, January 1999, pp 34-41
|
|
|
|
=back
|
|
|
|
=head1 HISTORY
|
|
|
|
Moved to the Changes file.
|
|
|
|
=head1 BUGS
|
|
|
|
Some older versions of Berkeley DB had problems with fixed length
|
|
records using the RECNO file format. This problem has been fixed since
|
|
version 1.85 of Berkeley DB.
|
|
|
|
I am sure there are bugs in the code. If you do find any, or can
|
|
suggest any enhancements, I would welcome your comments.
|
|
|
|
=head1 AVAILABILITY
|
|
|
|
B<DB_File> comes with the standard Perl source distribution. Look in
|
|
the directory F<ext/DB_File>. Given the amount of time between releases
|
|
of Perl the version that ships with Perl is quite likely to be out of
|
|
date, so the most recent version can always be found on CPAN (see
|
|
L<perlmod/CPAN> for details), in the directory
|
|
F<modules/by-module/DB_File>.
|
|
|
|
This version of B<DB_File> will work with either version 1.x, 2.x or
|
|
3.x of Berkeley DB, but is limited to the functionality provided by
|
|
version 1.
|
|
|
|
The official web site for Berkeley DB is F<http://www.sleepycat.com>.
|
|
All versions of Berkeley DB are available there.
|
|
|
|
Alternatively, Berkeley DB version 1 is available at your nearest CPAN
|
|
archive in F<src/misc/db.1.85.tar.gz>.
|
|
|
|
If you are running IRIX, then get Berkeley DB version 1 from
|
|
F<http://reality.sgi.com/ariel>. It has the patches necessary to
|
|
compile properly on IRIX 5.3.
|
|
|
|
=head1 COPYRIGHT
|
|
|
|
Copyright (c) 1995-2002 Paul Marquess. All rights reserved. This program
|
|
is free software; you can redistribute it and/or modify it under the
|
|
same terms as Perl itself.
|
|
|
|
Although B<DB_File> is covered by the Perl license, the library it
|
|
makes use of, namely Berkeley DB, is not. Berkeley DB has its own
|
|
copyright and its own license. Please take the time to read it.
|
|
|
|
Here are are few words taken from the Berkeley DB FAQ (at
|
|
F<http://www.sleepycat.com>) regarding the license:
|
|
|
|
Do I have to license DB to use it in Perl scripts?
|
|
|
|
No. The Berkeley DB license requires that software that uses
|
|
Berkeley DB be freely redistributable. In the case of Perl, that
|
|
software is Perl, and not your scripts. Any Perl scripts that you
|
|
write are your property, including scripts that make use of
|
|
Berkeley DB. Neither the Perl license nor the Berkeley DB license
|
|
place any restriction on what you may do with them.
|
|
|
|
If you are in any doubt about the license situation, contact either the
|
|
Berkeley DB authors or the author of DB_File. See L<"AUTHOR"> for details.
|
|
|
|
|
|
=head1 SEE ALSO
|
|
|
|
L<perl(1)>, L<dbopen(3)>, L<hash(3)>, L<recno(3)>, L<btree(3)>,
|
|
L<dbmfilter>
|
|
|
|
=head1 AUTHOR
|
|
|
|
The DB_File interface was written by Paul Marquess
|
|
E<lt>Paul.Marquess@btinternet.comE<gt>.
|
|
Questions about the DB system itself may be addressed to
|
|
E<lt>db@sleepycat.com<gt>.
|
|
|
|
=cut
|