sbdb/lib/sbdb/db.rb

157 lines
3.3 KiB
Ruby

require 'bdb'
require 'sbdb/cursor'
module SBDB
TYPES = []
class DB
UNKNOWN = Bdb::Db::UNKNOWN
BTREE = Bdb::Db::BTREE
HASH = Bdb::Db::HASH
QUEUE = Bdb::Db::QUEUE
ARRAY = RECNO = Bdb::Db::RECNO
RDONLY = READLONY = Bdb::DB_RDONLY
CONSUME = Bdb::DB_CONSUME
CONSUME_WAIT = Bdb::DB_CONSUME_WAIT
attr_reader :home
attr_accessor :txn
include Enumerable
def bdb_object() @db end
def sync() @db.sync end
def close( flg = nil) @db.close flg || 0 end
def cursor( &exe) Cursor.new self, &exe end
def at key, txn = nil
@db.get _txn(txn), key.nil? ? nil : key.to_s, nil, 0
rescue Bdb::KeyEmpty
return nil
end
alias [] at
def put key, val, txn = nil
if val.nil?
@db.del _txn(txn), key.to_s, 0
else
@db.put _txn(txn), key.nil? ? nil : key.to_s, val.to_s, 0
end
end
def []= key, val
put key, val
end
def delete key, txn = nil
@db.del _txn(txn), key.to_s
end
alias del delete
class << self
def new *ps, &exe
ret = obj = super( *ps)
begin ret = exe.call obj
ensure
SBDB::raise_barrier &obj.method(:sync)
SBDB::raise_barrier &obj.method(:close)
end if exe
ret
end
alias open new
end
def _txn txn
txn ||= @txn
txn && txn.bdb_object
end
def transaction flg = nil, &exe
block_given? ? home.transaction( flg, &exe) : home.transaction( flg)
end
# Arguments:
# * file
# * name
# * type
# * flags
# * mode
# * env
# or: **file**, **opts**. *opts* must be a *::Hash* with keys like above, excluded *file*.
def initialize file, *args
opts = ::Hash === args.last ? args.pop : {}
opts = {:name => args[0], :type => args[1], :flags => args[2], :mode => args[3], :env => args[4]}.update opts
#type = BTREE if type == UNKNOWN and (flags & CREATE) == CREATE
@home, @db = opts[:env], opts[:env] ? opts[:env].bdb_object.db : Bdb::Db.new
opts[:type] = TYPES.index(self.class) || UNKNOWN
@db.re_len = opts[:re_len] if opts[:re_len]
@txn = nil
txn = opts[:txn] # First is the global txn, second only for open.
begin
@db.open txn && txn.bdb_object, file, opts[:name], opts[:type], opts[:flags] || 0, opts[:mode] || 0
rescue Object => exc
close
raise exc
end
end
def each key = nil, val = nil, &exe
cursor {|c| c.each key, val, &exe }
end
def reverse key = nil, val = nil, &exe
cursor {|c| c.reverse key, val, &exe }
end
def to_hash key = nil, val = nil
ht = {}
each key, val, &ht.method( :[]=)
ht
end
def truncate txn = nil
@db.truncate _txn(txn)
end
end
class Unknown < DB
def self.new file, *ps, &exe
dbt = super( file, *ps) {|db| db.bdb_object.get_type }
TYPES[dbt] ? TYPES[dbt].new( file, *ps, &exe) : super( file, *ps, &exe)
end
end
class Btree < DB
end
TYPES[DB::BTREE] = Btree
class Hash < DB
end
TYPES[DB::HASH] = Hash
module Arrayisch
def [] key
super [key].pack('I')
end
def []= key, val
super [key].pack('I'), val
end
def push val, txn = nil
@db.put _txn(txn), "\0\0\0\0", val, Bdb::DB_APPEND
end
end
class Recno < DB
include Arrayisch
end
Array = Recno
TYPES[DB::RECNO] = Recno
class Queue < DB
include Arrayisch
def unshift txn = nil
@db.get _txn(txn), "\0\0\0\0", nil, Bdb::DB_CONSUME
end
end
TYPES[DB::QUEUE] = Queue
end