add support for set_btree_compare and multi-key secondary indexes
This commit is contained in:
parent
3d4a54eb05
commit
c5e7eb6c73
2
Rakefile
2
Rakefile
|
@ -22,7 +22,7 @@ end
|
||||||
|
|
||||||
desc "Run tests"
|
desc "Run tests"
|
||||||
Rake::TestTask.new("test") do |t|
|
Rake::TestTask.new("test") do |t|
|
||||||
t.libs << ["test", "ext"]
|
t.libs.concat ["test", "ext"]
|
||||||
t.pattern = 'test/*_test.rb'
|
t.pattern = 'test/*_test.rb'
|
||||||
t.verbose = true
|
t.verbose = true
|
||||||
t.warning = true
|
t.warning = true
|
||||||
|
|
145
ext/bdb.c
145
ext/bdb.c
|
@ -30,7 +30,7 @@ VALUE cTxnStat; /* Transaction Status class */
|
||||||
VALUE cTxnStatActive; /* Active Transaction Status class */
|
VALUE cTxnStatActive; /* Active Transaction Status class */
|
||||||
VALUE eDbError;
|
VALUE eDbError;
|
||||||
|
|
||||||
static ID fv_call, fv_err_new,fv_err_code,fv_err_msg;
|
static ID fv_call,fv_uniq,fv_err_new,fv_err_code,fv_err_msg;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Document-class: Bdb::DbError
|
* Document-class: Bdb::DbError
|
||||||
|
@ -178,6 +178,7 @@ VALUE db_init_aux(VALUE obj,t_envh * eh)
|
||||||
dbh->self=obj;
|
dbh->self=obj;
|
||||||
dbh->env=eh;
|
dbh->env=eh;
|
||||||
dbh->aproc=Qnil;
|
dbh->aproc=Qnil;
|
||||||
|
dbh->sproc=Qnil;
|
||||||
memset(&(dbh->filename),0,FNLEN+1);
|
memset(&(dbh->filename),0,FNLEN+1);
|
||||||
|
|
||||||
dbh->adbc=Qnil;
|
dbh->adbc=Qnil;
|
||||||
|
@ -264,6 +265,7 @@ VALUE db_open(VALUE obj, VALUE vtxn, VALUE vdisk_file,
|
||||||
if ( ! NIL_P(dbh->adbc) )
|
if ( ! NIL_P(dbh->adbc) )
|
||||||
raise_error(0,"db handle already opened");
|
raise_error(0,"db handle already opened");
|
||||||
|
|
||||||
|
dbh->db->app_private=dbh;
|
||||||
rv = dbh->db->open(dbh->db,txn?txn->txn:NULL,
|
rv = dbh->db->open(dbh->db,txn?txn->txn:NULL,
|
||||||
StringValueCStr(vdisk_file),
|
StringValueCStr(vdisk_file),
|
||||||
logical_db,
|
logical_db,
|
||||||
|
@ -511,6 +513,7 @@ VALUE db_close(VALUE obj, VALUE vflags)
|
||||||
rv = dbh->db->close(dbh->db,flags);
|
rv = dbh->db->close(dbh->db,flags);
|
||||||
dbh->db=NULL;
|
dbh->db=NULL;
|
||||||
dbh->aproc=Qnil;
|
dbh->aproc=Qnil;
|
||||||
|
dbh->sproc=Qnil;
|
||||||
if ( dbh->env ) {
|
if ( dbh->env ) {
|
||||||
if ( RTEST(ruby_debug) )
|
if ( RTEST(ruby_debug) )
|
||||||
rb_warning("%s/%d %s 0x%x",__FILE__,__LINE__,"db_close! removing",obj);
|
rb_warning("%s/%d %s 0x%x",__FILE__,__LINE__,"db_close! removing",obj);
|
||||||
|
@ -1087,7 +1090,18 @@ VALUE db_del(VALUE obj, VALUE vtxn, VALUE vkey, VALUE vflags)
|
||||||
return Qtrue;
|
return Qtrue;
|
||||||
}
|
}
|
||||||
|
|
||||||
t_dbh *dbassoc[LMAXFD];
|
void assoc_key(DBT* result, VALUE obj) {
|
||||||
|
VALUE str=StringValue(obj);
|
||||||
|
|
||||||
|
#ifdef DEBUG_DB
|
||||||
|
fprintf(stderr,"assoc_key %*s", RSTRING_LEN(str),RSTRING_PTR(str));
|
||||||
|
#endif
|
||||||
|
|
||||||
|
result->data=RSTRING_PTR(str);
|
||||||
|
result->size=RSTRING_LEN(str);
|
||||||
|
result->flags=LMEMFLAG;
|
||||||
|
}
|
||||||
|
|
||||||
VALUE
|
VALUE
|
||||||
assoc_callback2(VALUE *args)
|
assoc_callback2(VALUE *args)
|
||||||
{
|
{
|
||||||
|
@ -1098,37 +1112,53 @@ int assoc_callback(DB *secdb,const DBT* key, const DBT* data, DBT* result)
|
||||||
{
|
{
|
||||||
t_dbh *dbh;
|
t_dbh *dbh;
|
||||||
VALUE proc;
|
VALUE proc;
|
||||||
int fdp,status;
|
int status;
|
||||||
VALUE retv;
|
VALUE retv;
|
||||||
VALUE args[4];
|
VALUE args[4];
|
||||||
|
VALUE keys;
|
||||||
|
int i;
|
||||||
|
|
||||||
memset(result,0,sizeof(DBT));
|
memset(result,0,sizeof(DBT));
|
||||||
secdb->fd(secdb,&fdp);
|
dbh=secdb->app_private;
|
||||||
dbh=dbassoc[fdp];
|
|
||||||
|
|
||||||
args[0]=dbh->aproc;
|
args[0]=dbh->aproc;
|
||||||
args[1]=dbh->self;
|
args[1]=dbh->self;
|
||||||
args[2]=rb_str_new(key->data,key->size);
|
args[2]=rb_str_new(key->data,key->size);
|
||||||
args[3]=rb_str_new(data->data,data->size);
|
args[3]=rb_str_new(data->data,data->size);
|
||||||
|
|
||||||
|
#ifdef DEBUG_DB
|
||||||
|
fprintf(stderr,"assoc_data %*s", data->size, data->data);
|
||||||
|
#endif
|
||||||
|
|
||||||
retv=rb_protect((VALUE(*)_((VALUE)))assoc_callback2,(VALUE)args,&status);
|
retv=rb_protect((VALUE(*)_((VALUE)))assoc_callback2,(VALUE)args,&status);
|
||||||
|
|
||||||
if (status) return 99999;
|
if (status) return 99999;
|
||||||
if ( NIL_P(retv) )
|
if ( NIL_P(retv) )
|
||||||
return DB_DONOTINDEX;
|
return DB_DONOTINDEX;
|
||||||
|
|
||||||
if ( TYPE(retv) != T_STRING )
|
keys = rb_check_array_type(retv);
|
||||||
rb_warning("return from assoc callback not a string!");
|
if (!NIL_P(keys)) {
|
||||||
|
keys = rb_funcall(keys,fv_uniq,0); /* secondary keys must be uniq */
|
||||||
|
switch(RARRAY(keys)->len) {
|
||||||
|
case 0:
|
||||||
|
return DB_DONOTINDEX;
|
||||||
|
case 1:
|
||||||
|
retv=RARRAY(keys)->ptr[0];
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
result->size=RARRAY(keys)->len;
|
||||||
|
result->flags=DB_DBT_MULTIPLE | DB_DBT_APPMALLOC;
|
||||||
|
result->data=malloc(result->size * sizeof(DBT));
|
||||||
|
memset(result->data,0,result->size * sizeof(DBT));
|
||||||
|
|
||||||
StringValue(retv);
|
for (i=0; i<result->size; i++) {
|
||||||
#ifdef DEBUG_DB
|
assoc_key(result->data + i*sizeof(DBT), (VALUE)RARRAY(keys)->ptr[i]);
|
||||||
fprintf(stderr,"assoc_key %*s for %*s\n",
|
}
|
||||||
RSTRING_LEN(retv),RSTRING_PTR(retv),
|
return 0;
|
||||||
data->size,data->data);
|
}
|
||||||
#endif
|
}
|
||||||
result->data=RSTRING_PTR(retv);
|
|
||||||
result->size=RSTRING_LEN(retv);
|
assoc_key(result, retv);
|
||||||
result->flags=LMEMFLAG;
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1138,14 +1168,7 @@ int assoc_callback(DB *secdb,const DBT* key, const DBT* data, DBT* result)
|
||||||
*
|
*
|
||||||
* associate a secondary index(database) with this (primary)
|
* associate a secondary index(database) with this (primary)
|
||||||
* database. The proc can be nil if the database is only opened
|
* database. The proc can be nil if the database is only opened
|
||||||
* DB_RDONLY. Only +one+ proc can be assigned to a given database
|
* DB_RDONLY.
|
||||||
* according to the file-descriptor number of the secondary within
|
|
||||||
* a single ruby process. This is typcially not an issue since
|
|
||||||
* the proc should always be the same (it would corrupt the
|
|
||||||
* secondary otherwise). But this does limit use of nil if the
|
|
||||||
* db is openend DB_RDONLY and writeable in the same process.
|
|
||||||
* (although, opening the same db more than once has not been
|
|
||||||
* tested)
|
|
||||||
*
|
*
|
||||||
* call back proc has signature:
|
* call back proc has signature:
|
||||||
* proc(secdb,key,value)
|
* proc(secdb,key,value)
|
||||||
|
@ -1190,17 +1213,12 @@ VALUE db_associate(VALUE obj, VALUE vtxn, VALUE osecdb,
|
||||||
raise_error(0, "db_associate proc required");
|
raise_error(0, "db_associate proc required");
|
||||||
}
|
}
|
||||||
|
|
||||||
sdbh->db->fd(sdbh->db,&fdp);
|
|
||||||
sdbh->aproc=cb_proc;
|
sdbh->aproc=cb_proc;
|
||||||
|
|
||||||
/* No register is needed since this is just a way to
|
|
||||||
* get back to a real object
|
|
||||||
*/
|
|
||||||
dbassoc[fdp]=sdbh;
|
|
||||||
rv=pdbh->db->associate(pdbh->db,txn?txn->txn:NULL,sdbh->db,assoc_callback,flags);
|
rv=pdbh->db->associate(pdbh->db,txn?txn->txn:NULL,sdbh->db,assoc_callback,flags);
|
||||||
#ifdef DEBUG_DB
|
#ifdef DEBUG_DB
|
||||||
fprintf(stderr,"file is %d\n",fdp);
|
fprintf(stderr,"file is %d\n",fdp);
|
||||||
fprintf(stderr,"assoc done 0x%x 0x%x\n",sdbh,dbassoc[fdp]);
|
fprintf(stderr,"assoc done 0x%x\n",sdbh);
|
||||||
#endif
|
#endif
|
||||||
if (rv != 0) {
|
if (rv != 0) {
|
||||||
raise_error(rv, "db_associate failure: %s",db_strerror(rv));
|
raise_error(rv, "db_associate failure: %s",db_strerror(rv));
|
||||||
|
@ -1208,6 +1226,71 @@ VALUE db_associate(VALUE obj, VALUE vtxn, VALUE osecdb,
|
||||||
return Qtrue;
|
return Qtrue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VALUE
|
||||||
|
bt_compare_callback2(VALUE *args)
|
||||||
|
{
|
||||||
|
return rb_funcall(args[0],fv_call,3,args[1],args[2],args[3]);
|
||||||
|
}
|
||||||
|
|
||||||
|
int bt_compare_callback(DB *db, const DBT* key1, const DBT* key2)
|
||||||
|
{
|
||||||
|
t_dbh *dbh;
|
||||||
|
VALUE proc;
|
||||||
|
int cmp;
|
||||||
|
VALUE retv;
|
||||||
|
|
||||||
|
dbh=db->app_private;
|
||||||
|
|
||||||
|
/* Shouldn't catch exceptions in the callback, because bad sort data will corrupt the BTree.*/
|
||||||
|
retv=rb_funcall(dbh->sproc,fv_call,3,dbh->self,
|
||||||
|
rb_str_new(key1->data,key1->size),rb_str_new(key2->data,key2->size));
|
||||||
|
|
||||||
|
if (!FIXNUM_P(retv))
|
||||||
|
rb_raise(rb_eTypeError,"btree comparison should return Fixnum");
|
||||||
|
|
||||||
|
cmp=FIX2INT(retv);
|
||||||
|
|
||||||
|
#ifdef DEBUG_DB
|
||||||
|
fprintf(stderr,"bt_compare %*s <=> %*s: %d", key1->size, key1->data, key2->size, key2->data, cmp);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return cmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* call-seq:
|
||||||
|
* db.set_bt_compare(proc)
|
||||||
|
*
|
||||||
|
* set the btree key comparison function to the callback proc.
|
||||||
|
*
|
||||||
|
* callback proc has signature:
|
||||||
|
* proc(db,key1,key2)
|
||||||
|
*/
|
||||||
|
VALUE db_set_bt_compare(VALUE obj, VALUE cb_proc)
|
||||||
|
{
|
||||||
|
t_dbh *dbh;
|
||||||
|
int rv;
|
||||||
|
|
||||||
|
Data_Get_Struct(obj,t_dbh,dbh);
|
||||||
|
if (!dbh->db)
|
||||||
|
raise(0, "db is closed");
|
||||||
|
|
||||||
|
if ( rb_obj_is_instance_of(cb_proc,rb_cProc) != Qtrue ) {
|
||||||
|
raise_error(0, "db_associate proc required");
|
||||||
|
}
|
||||||
|
|
||||||
|
dbh->sproc=cb_proc;
|
||||||
|
rv=dbh->db->set_bt_compare(dbh->db,bt_compare_callback);
|
||||||
|
|
||||||
|
#ifdef DEBUG_DB
|
||||||
|
fprintf(stderr,"set_bt_compare done 0x%x\n",dbh);
|
||||||
|
#endif
|
||||||
|
if (rv != 0) {
|
||||||
|
raise_error(rv, "db_set_bt_compare failure: %s",db_strerror(rv));
|
||||||
|
}
|
||||||
|
return Qtrue;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* call-seq:
|
* call-seq:
|
||||||
* db.cursor(txn,flags)
|
* db.cursor(txn,flags)
|
||||||
|
@ -2762,6 +2845,7 @@ VALUE txn_set_timeout(VALUE obj, VALUE vtimeout, VALUE vflags)
|
||||||
|
|
||||||
void Init_bdb() {
|
void Init_bdb() {
|
||||||
fv_call=rb_intern("call");
|
fv_call=rb_intern("call");
|
||||||
|
fv_uniq=rb_intern("uniq");
|
||||||
fv_err_new=rb_intern("new");
|
fv_err_new=rb_intern("new");
|
||||||
fv_err_code=rb_intern("@code");
|
fv_err_code=rb_intern("@code");
|
||||||
fv_err_msg=rb_intern("@message");
|
fv_err_msg=rb_intern("@message");
|
||||||
|
@ -2790,6 +2874,7 @@ void Init_bdb() {
|
||||||
rb_define_method(cDb,"del",db_del,3);
|
rb_define_method(cDb,"del",db_del,3);
|
||||||
rb_define_method(cDb,"cursor",db_cursor,2);
|
rb_define_method(cDb,"cursor",db_cursor,2);
|
||||||
rb_define_method(cDb,"associate",db_associate,4);
|
rb_define_method(cDb,"associate",db_associate,4);
|
||||||
|
rb_define_method(cDb,"set_btree_compare",db_set_bt_compare,1);
|
||||||
rb_define_method(cDb,"flags=",db_flags_set,1);
|
rb_define_method(cDb,"flags=",db_flags_set,1);
|
||||||
rb_define_method(cDb,"flags",db_flags_get,0);
|
rb_define_method(cDb,"flags",db_flags_get,0);
|
||||||
rb_define_method(cDb,"open",db_open,6);
|
rb_define_method(cDb,"open",db_open,6);
|
||||||
|
|
|
@ -52,6 +52,8 @@ typedef struct s_dbh {
|
||||||
DB *db;
|
DB *db;
|
||||||
int db_opened;
|
int db_opened;
|
||||||
VALUE aproc;
|
VALUE aproc;
|
||||||
|
VALUE sproc; /* key sorting callback */
|
||||||
|
|
||||||
t_envh *env; /* Parent environment, NULL if not opened from one */
|
t_envh *env; /* Parent environment, NULL if not opened from one */
|
||||||
VALUE adbc; /* Ruby array holding opened cursor */
|
VALUE adbc; /* Ruby array holding opened cursor */
|
||||||
char filename[FNLEN+1];
|
char filename[FNLEN+1];
|
||||||
|
|
|
@ -63,6 +63,42 @@ class CursorTest < Test::Unit::TestCase
|
||||||
assert_equal 1, @cursor.count
|
assert_equal 1, @cursor.count
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_get_all_in_order
|
||||||
|
all = []
|
||||||
|
while pair = @cursor.get(nil, nil, Bdb::DB_NEXT)
|
||||||
|
all << pair.first
|
||||||
|
end
|
||||||
|
assert_equal (0..9).collect {|i| i.to_s}, all
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_get_all_with_set_btree_compare
|
||||||
|
@db1 = Bdb::Db.new
|
||||||
|
@db1.set_btree_compare(proc {|db, key1, key2| key2 <=> key1})
|
||||||
|
@db1.open(nil, File.join(File.dirname(__FILE__), 'tmp', 'test1.db'), nil, Bdb::Db::BTREE, Bdb::DB_CREATE, 0)
|
||||||
|
10.times { |i| @db1.put(nil, i.to_s, "data-#{i}", 0)}
|
||||||
|
@cursor1 = @db1.cursor(nil, 0)
|
||||||
|
|
||||||
|
all = []
|
||||||
|
while pair = @cursor1.get(nil, nil, Bdb::DB_NEXT)
|
||||||
|
all << pair.first
|
||||||
|
end
|
||||||
|
assert_equal (0..9).collect {|i| i.to_s}.reverse, all
|
||||||
|
@cursor1.close
|
||||||
|
@db1.close(0)
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_btree_compare_raises_if_fixnum_not_returned
|
||||||
|
@db1 = Bdb::Db.new
|
||||||
|
@db1.set_btree_compare(proc {|db, key1, key2| key1})
|
||||||
|
@db1.open(nil, File.join(File.dirname(__FILE__), 'tmp', 'test1.db'), nil, Bdb::Db::BTREE, Bdb::DB_CREATE, 0)
|
||||||
|
|
||||||
|
assert_raises(TypeError) do
|
||||||
|
@db1.put(nil, "no", "way", 0)
|
||||||
|
@db1.put(nil, "ho", "say", 0)
|
||||||
|
end
|
||||||
|
@db1.close(Bdb::DB_NOSYNC)
|
||||||
|
end
|
||||||
|
|
||||||
def test_join
|
def test_join
|
||||||
@personnel_db = Bdb::Db.new
|
@personnel_db = Bdb::Db.new
|
||||||
@personnel_db.open(nil, File.join(File.dirname(__FILE__), 'tmp', 'personnel_db.db'), nil, Bdb::Db::HASH, Bdb::DB_CREATE, 0)
|
@personnel_db.open(nil, File.join(File.dirname(__FILE__), 'tmp', 'personnel_db.db'), nil, Bdb::Db::HASH, Bdb::DB_CREATE, 0)
|
||||||
|
|
|
@ -55,6 +55,30 @@ class DbTest < Test::Unit::TestCase
|
||||||
@db1.close(0)
|
@db1.close(0)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_associate_with_multiple_keys
|
||||||
|
@db1 = Bdb::Db.new
|
||||||
|
@db1.flags = Bdb::DB_DUPSORT
|
||||||
|
@db1.open(nil, File.join(File.dirname(__FILE__), 'tmp', 'test1.db'), nil, Bdb::Db::HASH, Bdb::DB_CREATE, 0)
|
||||||
|
|
||||||
|
@db.associate(nil, @db1, 0, proc { |sdb, key, data| key.split('-') })
|
||||||
|
|
||||||
|
@db.put(nil, '1234-5678', 'data', 0)
|
||||||
|
@db.put(nil, '8765-4321', 'atad', 0)
|
||||||
|
|
||||||
|
result = @db.get(nil, '1234-5678', nil, 0)
|
||||||
|
assert_equal 'data', result
|
||||||
|
result = @db1.get(nil, '5678', nil, 0)
|
||||||
|
assert_equal 'data', result
|
||||||
|
result = @db1.get(nil, '1234', nil, 0)
|
||||||
|
assert_equal 'data', result
|
||||||
|
result = @db1.get(nil, '8765', nil, 0)
|
||||||
|
assert_equal 'atad', result
|
||||||
|
result = @db1.get(nil, '4321', nil, 0)
|
||||||
|
assert_equal 'atad', result
|
||||||
|
|
||||||
|
@db1.close(0)
|
||||||
|
end
|
||||||
|
|
||||||
def test_aset_and_aget
|
def test_aset_and_aget
|
||||||
@db['key'] = 'data'
|
@db['key'] = 'data'
|
||||||
result = @db.get(nil, 'key', nil, 0)
|
result = @db.get(nil, 'key', nil, 0)
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
require "test/unit"
|
require 'test/unit'
|
||||||
require 'fileutils'
|
require 'fileutils'
|
||||||
require "bdb"
|
require File.dirname(__FILE__) + '/../ext/bdb'
|
||||||
|
|
||||||
class Test::Unit::TestCase
|
class Test::Unit::TestCase
|
||||||
include FileUtils
|
include FileUtils
|
||||||
|
|
Loading…
Reference in a new issue