[t:4063], add a test

git-svn-id: file:///svn/toku/tokudb@37569 c7de825b-a66e-492c-adef-691d508d4ae1
This commit is contained in:
Zardosht Kasheff 2013-04-16 23:59:58 -04:00 committed by Yoni Fogel
parent 68da600038
commit 61478b295e

View file

@ -0,0 +1,154 @@
// This test verifies that queries that have a provisional delete at the end of a basement node work.
// The issue is that when we read off the end of a basement node, the next basement node may not be available memory, so we
// need to release the ydb lock and try again. This test verifies that this scenario works by having many deletes
// and a small cachetable.
#include "test.h"
const int envflags = DB_INIT_MPOOL|DB_CREATE|DB_THREAD |DB_INIT_LOCK|DB_INIT_LOG|DB_INIT_TXN|DB_PRIVATE;
int test_main (int argc, char * const argv[]) {
parse_args(argc, argv);
int r;
r = system("rm -rf " ENVDIR);
CKERR(r);
toku_os_mkdir(ENVDIR, S_IRWXU+S_IRWXG+S_IRWXO);
DB_ENV *env;
r = db_env_create(&env, 0); CKERR(r);
env->set_errfile(env, stderr);
// set a cachetable size of 10K
u_int32_t cachesize = 10*1024;
r = env->set_cachesize(env, 0, cachesize, 1); CKERR(r);
r = env->set_default_bt_compare(env, int64_dbt_cmp); CKERR(r);
r = env->open(env, ENVDIR, envflags, S_IRWXU+S_IRWXG+S_IRWXO); CKERR(r);
DB *db;
{
DB_TXN *txna;
r = env->txn_begin(env, NULL, &txna, 0); CKERR(r);
r = db_create(&db, env, 0); CKERR(r);
r = db->set_pagesize(db, 4096);
CKERR(r);
r = db->set_readpagesize(db, 1024);
CKERR(r);
r = db->open(db, txna, "foo.db", NULL, DB_BTREE, DB_CREATE, 0666); CKERR(r);
r = txna->commit(txna, 0); CKERR(r);
}
if (verbose) printf("starting insertion of even elements\n");
//
// now insert a bunch of elements
//
DB_TXN* txn;
r = env->txn_begin(env, NULL, &txn, 0);
CKERR(r);
for (u_int32_t i = 0; i < cachesize; i++) {
DBT key,val;
u_int64_t key_data = 2*i;
u_int64_t val_data = 4*i;
r = db->put(
db,
txn,
dbt_init(&key, &key_data, sizeof(key_data)),
dbt_init(&val, &val_data, sizeof(val_data)),
0
);
CKERR(r);
}
r = txn->commit(txn, 0);
CKERR(r);
// this transaction will read all even keys inserted above
DB_TXN* txn_first = NULL;
r = env->txn_begin(env, NULL, &txn_first, DB_TXN_SNAPSHOT);
CKERR(r);
if (verbose) printf("starting insertion of odd elements and deletion of even elements\n");
r = env->txn_begin(env, NULL, &txn, 0);
CKERR(r);
for (u_int32_t i = 0; i < cachesize; i++) {
//
// insert odd values, and delete even values
//
DBT key,val;
u_int64_t key_data = 2*i+1;
u_int64_t val_data = 4*i+2;
dbt_init(&key, &key_data, sizeof(key_data));
dbt_init(&val, &val_data, sizeof(val_data));
r = db->put(
db,
txn,
&key,
&val,
0
);
CKERR(r);
key_data = 2*i;
r = db->del(db, txn, &key, DB_DELETE_ANY);
CKERR(r);
}
r = txn->commit(txn, 0);
CKERR(r);
// this transaction will read all odd keys inserted in the second round
DB_TXN* txn_second = NULL;
r = env->txn_begin(env, NULL, &txn_second, DB_TXN_SNAPSHOT);
CKERR(r);
DBC* cursor_first = NULL;
DBC* cursor_second = NULL;
r = db->cursor(db, txn_first, &cursor_first, 0);
CKERR(r);
r = db->cursor(db, txn_second, &cursor_second, 0);
CKERR(r);
DBT key, val;
memset(&key, 0, sizeof(key));
memset(&val, 0, sizeof(val));
if (verbose) printf("starting cursor first query\n");
// now let's do the cursor reads and verify that all the data is read properly
for (u_int32_t i = 0; i < cachesize; i++) {
r = cursor_first->c_get(cursor_first, &key, &val, DB_NEXT);
CKERR(r);
assert(key.size == 8);
assert(val.size == 8);
assert(*(uint64_t *)key.data == 2*i);
assert(*(uint64_t *)val.data == 4*i);
}
r = cursor_first->c_get(cursor_first, &key, &val, DB_NEXT);
CKERR2(r, DB_NOTFOUND);
if (verbose) printf("starting cursor second query\n");
// now let's do the cursor reads and verify that all the data is read properly
for (u_int32_t i = 0; i < cachesize; i++) {
r = cursor_second->c_get(cursor_second, &key, &val, DB_NEXT);
CKERR(r);
assert(key.size == 8);
assert(val.size == 8);
assert(*(uint64_t *)key.data == 2*i+1);
assert(*(uint64_t *)val.data == 4*i+2);
}
r = cursor_second->c_get(cursor_second, &key, &val, DB_NEXT);
CKERR2(r, DB_NOTFOUND);
if (verbose) printf("cleaning up\n");
r = cursor_first->c_close(cursor_first);
CKERR(r);
r = cursor_second->c_close(cursor_second);
CKERR(r);
r = txn_first->commit(txn_first,0);
CKERR(r);
r = txn_second->commit(txn_second,0);
CKERR(r);
r = db->close(db, 0);
CKERR(r);
r = env->close(env, 0);
CKERR(r);
return 0;
}