1999-12-21 03:04:37 +00:00
/* a test program for tdb - the trivial database */
2006-10-20 09:55:47 +00:00
# include "replace.h"
# include "tdb.h"
# include "system/filesys.h"
# include "system/time.h"
# include <gdbm.h>
1999-12-21 03:04:37 +00:00
# define DELETE_PROB 7
# define STORE_PROB 5
2006-10-20 09:55:47 +00:00
static struct tdb_context * db ;
1999-12-21 03:04:37 +00:00
static GDBM_FILE gdbm ;
struct timeval tp1 , tp2 ;
2006-10-20 09:55:47 +00:00
static void _start_timer ( void )
1999-12-21 03:04:37 +00:00
{
gettimeofday ( & tp1 , NULL ) ;
}
2006-10-20 09:55:47 +00:00
static double _end_timer ( void )
1999-12-21 03:04:37 +00:00
{
gettimeofday ( & tp2 , NULL ) ;
2013-03-18 14:39:04 +01:00
return ( ( tp2 . tv_sec - tp1 . tv_sec ) +
1999-12-21 03:04:37 +00:00
( tp2 . tv_usec - tp1 . tv_usec ) * 1.0e-6 ) ;
}
2006-10-20 09:55:47 +00:00
static void fatal ( const char * why )
1999-12-21 03:04:37 +00:00
{
perror ( why ) ;
exit ( 1 ) ;
}
2006-10-20 09:55:47 +00:00
# ifdef PRINTF_ATTRIBUTE
static void tdb_log ( struct tdb_context * tdb , int level , const char * format , . . . ) PRINTF_ATTRIBUTE ( 3 , 4 ) ;
# endif
static void tdb_log ( struct tdb_context * tdb , int level , const char * format , . . . )
2001-05-28 13:29:06 +00:00
{
va_list ap ;
2013-03-18 14:39:04 +01:00
2001-05-28 13:29:06 +00:00
va_start ( ap , format ) ;
vfprintf ( stdout , format , ap ) ;
va_end ( ap ) ;
2001-09-06 05:59:32 +00:00
fflush ( stdout ) ;
2001-05-28 13:29:06 +00:00
}
1999-12-21 03:04:37 +00:00
static void compare_db ( void )
{
TDB_DATA d , key , nextkey ;
datum gd , gkey , gnextkey ;
key = tdb_firstkey ( db ) ;
while ( key . dptr ) {
d = tdb_fetch ( db , key ) ;
gkey . dptr = key . dptr ;
gkey . dsize = key . dsize ;
gd = gdbm_fetch ( gdbm , gkey ) ;
if ( ! gd . dptr ) fatal ( " key not in gdbm " ) ;
if ( gd . dsize ! = d . dsize ) fatal ( " data sizes differ " ) ;
if ( memcmp ( gd . dptr , d . dptr , d . dsize ) ) {
fatal ( " data differs " ) ;
}
nextkey = tdb_nextkey ( db , key ) ;
free ( key . dptr ) ;
free ( d . dptr ) ;
free ( gd . dptr ) ;
key = nextkey ;
}
gkey = gdbm_firstkey ( gdbm ) ;
while ( gkey . dptr ) {
gd = gdbm_fetch ( gdbm , gkey ) ;
key . dptr = gkey . dptr ;
key . dsize = gkey . dsize ;
d = tdb_fetch ( db , key ) ;
if ( ! d . dptr ) fatal ( " key not in db " ) ;
if ( d . dsize ! = gd . dsize ) fatal ( " data sizes differ " ) ;
if ( memcmp ( d . dptr , gd . dptr , gd . dsize ) ) {
fatal ( " data differs " ) ;
}
gnextkey = gdbm_nextkey ( gdbm , gkey ) ;
free ( gkey . dptr ) ;
free ( gd . dptr ) ;
free ( d . dptr ) ;
gkey = gnextkey ;
}
}
static char * randbuf ( int len )
{
char * buf ;
int i ;
buf = ( char * ) malloc ( len + 1 ) ;
for ( i = 0 ; i < len ; i + + ) {
buf [ i ] = ' a ' + ( rand ( ) % 26 ) ;
}
buf [ i ] = 0 ;
return buf ;
}
static void addrec_db ( void )
{
int klen , dlen ;
char * k , * d ;
TDB_DATA key , data ;
klen = 1 + ( rand ( ) % 4 ) ;
dlen = 1 + ( rand ( ) % 100 ) ;
k = randbuf ( klen ) ;
d = randbuf ( dlen ) ;
key . dptr = k ;
key . dsize = klen + 1 ;
data . dptr = d ;
data . dsize = dlen + 1 ;
if ( rand ( ) % DELETE_PROB = = 0 ) {
tdb_delete ( db , key ) ;
} else if ( rand ( ) % STORE_PROB = = 0 ) {
if ( tdb_store ( db , key , data , TDB_REPLACE ) ! = 0 ) {
fatal ( " tdb_store failed " ) ;
}
} else {
data = tdb_fetch ( db , key ) ;
if ( data . dptr ) free ( data . dptr ) ;
}
free ( k ) ;
free ( d ) ;
}
static void addrec_gdbm ( void )
{
int klen , dlen ;
char * k , * d ;
datum key , data ;
klen = 1 + ( rand ( ) % 4 ) ;
dlen = 1 + ( rand ( ) % 100 ) ;
k = randbuf ( klen ) ;
d = randbuf ( dlen ) ;
key . dptr = k ;
key . dsize = klen + 1 ;
data . dptr = d ;
data . dsize = dlen + 1 ;
if ( rand ( ) % DELETE_PROB = = 0 ) {
gdbm_delete ( gdbm , key ) ;
} else if ( rand ( ) % STORE_PROB = = 0 ) {
if ( gdbm_store ( gdbm , key , data , GDBM_REPLACE ) ! = 0 ) {
fatal ( " gdbm_store failed " ) ;
}
} else {
data = gdbm_fetch ( gdbm , key ) ;
if ( data . dptr ) free ( data . dptr ) ;
}
free ( k ) ;
free ( d ) ;
}
2006-10-20 09:55:47 +00:00
static int traverse_fn ( struct tdb_context * tdb , TDB_DATA key , TDB_DATA dbuf , void * state )
1999-12-21 03:04:37 +00:00
{
#if 0
printf ( " [%s] [%s] \n " , key . dptr , dbuf . dptr ) ;
# endif
2001-12-11 08:24:36 +00:00
tdb_delete ( tdb , key ) ;
1999-12-21 03:04:37 +00:00
return 0 ;
}
2001-12-11 08:24:36 +00:00
static void merge_test ( void )
2000-12-05 03:12:57 +00:00
{
2001-07-04 04:39:52 +00:00
int i ;
2000-12-05 03:12:57 +00:00
char keys [ 5 ] [ 2 ] ;
2006-10-20 09:55:47 +00:00
char tdata [ ] = " test " ;
2001-07-04 04:39:52 +00:00
TDB_DATA key , data ;
2013-03-18 14:39:04 +01:00
2000-12-05 03:12:57 +00:00
for ( i = 0 ; i < 5 ; i + + ) {
2006-10-20 09:55:47 +00:00
snprintf ( keys [ i ] , 2 , " %d " , i ) ;
2001-07-04 04:39:52 +00:00
key . dptr = keys [ i ] ;
key . dsize = 2 ;
2013-03-18 14:39:04 +01:00
2006-10-20 09:55:47 +00:00
data . dptr = tdata ;
2001-07-04 04:39:52 +00:00
data . dsize = 4 ;
2013-03-18 14:39:04 +01:00
2001-07-04 04:39:52 +00:00
if ( tdb_store ( db , key , data , TDB_REPLACE ) ! = 0 ) {
fatal ( " tdb_store failed " ) ;
}
2000-12-05 03:12:57 +00:00
}
key . dptr = keys [ 0 ] ;
tdb_delete ( db , key ) ;
key . dptr = keys [ 4 ] ;
tdb_delete ( db , key ) ;
key . dptr = keys [ 2 ] ;
tdb_delete ( db , key ) ;
key . dptr = keys [ 1 ] ;
tdb_delete ( db , key ) ;
key . dptr = keys [ 3 ] ;
tdb_delete ( db , key ) ;
}
2007-11-06 22:57:35 -08:00
2010-12-24 11:54:03 +01:00
static char * test_path ( const char * filename )
{
const char * prefix = getenv ( " TEST_DATA_PREFIX " ) ;
if ( prefix ) {
char * path = NULL ;
int ret ;
ret = asprintf ( & path , " %s/%s " , prefix , filename ) ;
if ( ret = = - 1 ) {
return NULL ;
}
return path ;
}
return strdup ( filename ) ;
}
2006-10-20 09:55:47 +00:00
int main ( int argc , const char * argv [ ] )
1999-12-21 03:04:37 +00:00
{
int i , seed = 0 ;
2000-01-02 23:00:27 +00:00
int loops = 10000 ;
2006-11-30 03:05:55 +00:00
int num_entries ;
2010-12-24 11:54:03 +01:00
char test_gdbm [ 1 ] = " test.gdbm " ;
char * test_tdb ;
1999-12-21 03:04:37 +00:00
2010-12-24 11:54:03 +01:00
test_gdbm [ 0 ] = test_path ( " test.gdbm " ) ;
test_tdb = test_path ( " test.tdb " ) ;
1999-12-21 03:04:37 +00:00
2010-12-24 11:54:03 +01:00
unlink ( test_gdbm [ 0 ] ) ;
db = tdb_open ( test_tdb , 0 , TDB_CLEAR_IF_FIRST ,
2001-12-04 11:41:12 +00:00
O_RDWR | O_CREAT | O_TRUNC , 0600 ) ;
2013-03-18 14:39:04 +01:00
gdbm = gdbm_open ( test_gdbm , 512 , GDBM_WRITER | GDBM_NEWDB | GDBM_FAST ,
1999-12-21 03:04:37 +00:00
0600 , NULL ) ;
if ( ! db | | ! gdbm ) {
fatal ( " db open failed " ) ;
}
# if 1
srand ( seed ) ;
2006-10-20 09:55:47 +00:00
_start_timer ( ) ;
1999-12-21 03:04:37 +00:00
for ( i = 0 ; i < loops ; i + + ) addrec_gdbm ( ) ;
2006-10-20 09:55:47 +00:00
printf ( " gdbm got %.2f ops/sec \n " , i / _end_timer ( ) ) ;
1999-12-21 03:04:37 +00:00
# endif
2000-12-05 03:12:57 +00:00
merge_test ( ) ;
1999-12-21 03:04:37 +00:00
srand ( seed ) ;
2006-10-20 09:55:47 +00:00
_start_timer ( ) ;
1999-12-21 03:04:37 +00:00
for ( i = 0 ; i < loops ; i + + ) addrec_db ( ) ;
2006-10-20 09:55:47 +00:00
printf ( " tdb got %.2f ops/sec \n " , i / _end_timer ( ) ) ;
1999-12-21 03:04:37 +00:00
2006-11-30 03:05:55 +00:00
if ( tdb_validate_freelist ( db , & num_entries ) = = - 1 ) {
printf ( " tdb freelist is corrupt \n " ) ;
} else {
printf ( " tdb freelist is good (%d entries) \n " , num_entries ) ;
}
1999-12-21 03:04:37 +00:00
compare_db ( ) ;
1) added void* state argument to tdb_traverse. guess what! there were
two places i found where it was appropriate to _use_ that third argument,
in locking.c and brlock.c! there was a static traverse_function and
i removed the static variable, typecast it to a void*, passed it to
tdb_traverse and re-cast it back to the traverse_function inside the
tdb_traverse function. this makes the use of tdb_traverse() reentrant,
which is never going to happen, i know, i just don't like to see
statics lying about when there's no need for them.
as i had to do in samba-tng, all uses of tdb_traverse modified to take
the new void* state argument.
2) disabled rpcclient: referring people to use SAMBA_TNG rpcclient.
i don't know how the other samba team members would react if i deleted
rpcclient from cvs main. damn, that code's so old, it's unreal.
20 rpcclient commands, instead of about 70 in SAMBA_TNG.
(This used to be commit 49d7f0afbc1c5425d53019e234d54ddf205c8e9a)
2000-02-04 04:59:31 +00:00
printf ( " traversed %d records \n " , tdb_traverse ( db , traverse_fn , NULL ) ) ;
printf ( " traversed %d records \n " , tdb_traverse ( db , traverse_fn , NULL ) ) ;
1999-12-21 03:04:37 +00:00
tdb_close ( db ) ;
gdbm_close ( gdbm ) ;
2010-12-24 11:54:03 +01:00
free ( test_gdbm [ 0 ] ) ;
free ( test_tdb ) ;
1999-12-21 03:04:37 +00:00
return 0 ;
}