/*************************************************************************************************
* C language binding
* Copyright (C) 2009-2012 FAL Labs
* This file is part of Kyoto Cabinet.
* This program is free software: you can redistribute it and/or modify it under the terms of
* the GNU General Public License as published by the Free Software Foundation, either version
* 3 of the License, or any later version.
* This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the GNU General Public License for more details.
* You should have received a copy of the GNU General Public License along with this program.
* If not, see .
*************************************************************************************************/
#include "kcpolydb.h"
#include "kcdbext.h"
#include "kclangc.h"
#include "myconf.h"
using namespace kyotocabinet;
extern "C" {
/** The package version. */
const char* const KCVERSION = VERSION;
/** Special pointer for no operation by the visiting function. */
const char* const KCVISNOP = DB::Visitor::NOP;
/** Special pointer to remove the record by the visiting function. */
const char* const KCVISREMOVE = DB::Visitor::REMOVE;
/**
* Allocate a region on memory.
*/
void* kcmalloc(size_t size) {
_assert_(size > 0 && size <= MEMMAXSIZ);
return new char[size];
}
/**
* Release a region allocated in the library.
*/
void kcfree(void* ptr) {
_assert_(true);
delete[] (char*)ptr;
}
/**
* Get the time of day in seconds.
*/
double kctime(void) {
_assert_(true);
return kyotocabinet::time();
}
/**
* Convert a string to an integer.
*/
int64_t kcatoi(const char* str) {
_assert_(str);
return kyotocabinet::atoi(str);
}
/**
* Convert a string with a metric prefix to an integer.
*/
int64_t kcatoix(const char* str) {
_assert_(str);
return kyotocabinet::atoix(str);
}
/**
* Convert a string to a real number.
*/
double kcatof(const char* str) {
_assert_(str);
return kyotocabinet::atof(str);
}
/**
* Get the hash value by MurMur hashing.
*/
uint64_t kchashmurmur(const void* buf, size_t size) {
_assert_(buf && size <= MEMMAXSIZ);
return kyotocabinet::hashmurmur(buf, size);
}
/**
* Get the hash value by FNV hashing.
*/
uint64_t kchashfnv(const void* buf, size_t size) {
_assert_(buf && size <= MEMMAXSIZ);
return kyotocabinet::hashfnv(buf, size);
}
/**
* Calculate the levenshtein distance of two regions.
*/
size_t kclevdist(const void* abuf, size_t asiz, const void* bbuf, size_t bsiz, int32_t utf) {
_assert_(abuf && asiz <= MEMMAXSIZ && bbuf && bsiz <= MEMMAXSIZ);
size_t dist;
if (utf) {
uint32_t astack[128];
uint32_t* aary = asiz > sizeof(astack) / sizeof(*astack) ? new uint32_t[asiz] : astack;
size_t anum;
strutftoucs((const char*)abuf, asiz, aary, &anum);
uint32_t bstack[128];
uint32_t* bary = bsiz > sizeof(bstack) / sizeof(*bstack) ? new uint32_t[bsiz] : bstack;
size_t bnum;
strutftoucs((const char*)bbuf, bsiz, bary, &bnum);
dist = strucsdist(aary, anum, bary, bnum);
if (bary != bstack) delete[] bary;
if (aary != astack) delete[] aary;
} else {
dist = memdist(abuf, asiz, bbuf, bsiz);
}
return dist;
}
/**
* Get the quiet Not-a-Number value.
*/
double kcnan() {
_assert_(true);
return kyotocabinet::nan();
}
/**
* Get the positive infinity value.
*/
double kcinf() {
_assert_(true);
return kyotocabinet::inf();
}
/**
* Check a number is a Not-a-Number value.
*/
int32_t kcchknan(double num) {
_assert_(true);
return kyotocabinet::chknan(num);
}
/**
* Check a number is an infinity value.
*/
int32_t kcchkinf(double num) {
_assert_(true);
return kyotocabinet::chkinf(num);
}
/**
* Get the readable string of an error code.
*/
const char* kcecodename(int32_t code) {
_assert_(true);
return BasicDB::Error::codename((BasicDB::Error::Code)code);
}
/**
* Create a database object.
*/
KCDB* kcdbnew(void) {
_assert_(true);
return (KCDB*)new PolyDB;
}
/**
* Destroy a database object.
*/
void kcdbdel(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
delete pdb;
}
/**
* Open a database file.
*/
int32_t kcdbopen(KCDB* db, const char* path, uint32_t mode) {
_assert_(db && path);
PolyDB* pdb = (PolyDB*)db;
return pdb->open(path, mode);
}
/**
* Close the database file.
*/
int32_t kcdbclose(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->close();
}
/**
* Get the code of the last happened error.
*/
int32_t kcdbecode(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->error().code();
}
/**
* Get the supplement message of the last happened error.
*/
const char* kcdbemsg(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->error().message();
}
/**
* Accept a visitor to a record.
*/
int32_t kcdbaccept(KCDB* db, const char* kbuf, size_t ksiz,
KCVISITFULL fullproc, KCVISITEMPTY emptyproc, void* opq, int32_t writable) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
class VisitorImpl : public DB::Visitor {
public:
explicit VisitorImpl(KCVISITFULL fullproc, KCVISITEMPTY emptyproc, void* opq) :
fullproc_(fullproc), emptyproc_(emptyproc), opq_(opq) {}
const char* visit_full(const char* kbuf, size_t ksiz,
const char* vbuf, size_t vsiz, size_t* sp) {
if (!fullproc_) return NOP;
return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_);
}
const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
if (!emptyproc_) return NOP;
return emptyproc_(kbuf, ksiz, sp, opq_);
}
private:
KCVISITFULL fullproc_;
KCVISITEMPTY emptyproc_;
void* opq_;
};
VisitorImpl visitor(fullproc, emptyproc, opq);
return pdb->accept(kbuf, ksiz, &visitor, writable);
}
/**
* Accept a visitor to multiple records at once.
*/
int32_t kcdbacceptbulk(KCDB* db, const KCSTR* keys, size_t knum,
KCVISITFULL fullproc, KCVISITEMPTY emptyproc,
void* opq, int32_t writable) {
_assert_(db && keys && knum <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
std::vector xkeys;
xkeys.reserve(knum);
for (size_t i = 0; i < knum; i++) {
xkeys.push_back(std::string(keys[i].buf, keys[i].size));
}
class VisitorImpl : public DB::Visitor {
public:
explicit VisitorImpl(KCVISITFULL fullproc, KCVISITEMPTY emptyproc, void* opq) :
fullproc_(fullproc), emptyproc_(emptyproc), opq_(opq) {}
const char* visit_full(const char* kbuf, size_t ksiz,
const char* vbuf, size_t vsiz, size_t* sp) {
if (!fullproc_) return NOP;
return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_);
}
const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
if (!emptyproc_) return NOP;
return emptyproc_(kbuf, ksiz, sp, opq_);
}
private:
KCVISITFULL fullproc_;
KCVISITEMPTY emptyproc_;
void* opq_;
};
VisitorImpl visitor(fullproc, emptyproc, opq);
return pdb->accept_bulk(xkeys, &visitor, writable);
}
/**
* Iterate to accept a visitor for each record.
*/
int32_t kcdbiterate(KCDB* db, KCVISITFULL fullproc, void* opq, int32_t writable) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
class VisitorImpl : public DB::Visitor {
public:
explicit VisitorImpl(KCVISITFULL fullproc, void* opq) : fullproc_(fullproc), opq_(opq) {}
const char* visit_full(const char* kbuf, size_t ksiz,
const char* vbuf, size_t vsiz, size_t* sp) {
if (!fullproc_) return NOP;
return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_);
}
private:
KCVISITFULL fullproc_;
void* opq_;
};
VisitorImpl visitor(fullproc, opq);
return pdb->iterate(&visitor, writable);
}
/**
* Scan each record in parallel.
*/
int32_t kcdbscanpara(KCDB* db, KCVISITFULL fullproc, void* opq, size_t thnum) {
_assert_(db && thnum <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
class VisitorImpl : public DB::Visitor {
public:
explicit VisitorImpl(KCVISITFULL fullproc, void* opq) : fullproc_(fullproc), opq_(opq) {}
const char* visit_full(const char* kbuf, size_t ksiz,
const char* vbuf, size_t vsiz, size_t* sp) {
if (!fullproc_) return NOP;
return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_);
}
private:
KCVISITFULL fullproc_;
void* opq_;
};
VisitorImpl visitor(fullproc, opq);
return pdb->scan_parallel(&visitor, thnum);
}
/**
* Set the value of a record.
*/
int32_t kcdbset(KCDB* db, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->set(kbuf, ksiz, vbuf, vsiz);
}
/**
* Add a record.
*/
int32_t kcdbadd(KCDB* db, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->add(kbuf, ksiz, vbuf, vsiz);
}
/**
* Replace the value of a record.
*/
int32_t kcdbreplace(KCDB* db, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->replace(kbuf, ksiz, vbuf, vsiz);
}
/**
* Append the value of a record.
*/
int32_t kcdbappend(KCDB* db, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->append(kbuf, ksiz, vbuf, vsiz);
}
/**
* Add a number to the numeric value of a record.
*/
int64_t kcdbincrint(KCDB* db, const char* kbuf, size_t ksiz, int64_t num, int64_t orig) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->increment(kbuf, ksiz, num, orig);
}
/**
* Add a number to the numeric value of a record.
*/
double kcdbincrdouble(KCDB* db, const char* kbuf, size_t ksiz, double num, double orig) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->increment_double(kbuf, ksiz, num, orig);
}
/**
* Perform compare-and-swap.
*/
int32_t kcdbcas(KCDB* db, const char* kbuf, size_t ksiz,
const char* ovbuf, size_t ovsiz, const char* nvbuf, size_t nvsiz) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->cas(kbuf, ksiz, ovbuf, ovsiz, nvbuf, nvsiz);
}
/**
* Remove a record.
*/
int32_t kcdbremove(KCDB* db, const char* kbuf, size_t ksiz) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->remove(kbuf, ksiz);
}
/**
* Retrieve the value of a record.
*/
char* kcdbget(KCDB* db, const char* kbuf, size_t ksiz, size_t* sp) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ && sp);
PolyDB* pdb = (PolyDB*)db;
return pdb->get(kbuf, ksiz, sp);
}
/**
* Check the existence of a record.
*/
int32_t kcdbcheck(KCDB* db, const char* kbuf, size_t ksiz) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->check(kbuf, ksiz);
}
/**
* Retrieve the value of a record.
*/
int32_t kcdbgetbuf(KCDB* db, const char* kbuf, size_t ksiz, char* vbuf, size_t max) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf);
PolyDB* pdb = (PolyDB*)db;
return pdb->get(kbuf, ksiz, vbuf, max);
}
/**
* Retrieve the value of a record and remove it atomically.
*/
char* kcdbseize(KCDB* db, const char* kbuf, size_t ksiz, size_t* sp) {
_assert_(db && kbuf && ksiz <= MEMMAXSIZ && sp);
PolyDB* pdb = (PolyDB*)db;
return pdb->seize(kbuf, ksiz, sp);
}
/**
* Store records at once.
*/
int64_t kcdbsetbulk(KCDB* db, const KCREC* recs, size_t rnum, int32_t atomic) {
_assert_(db && recs && rnum <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
std::map xrecs;
for (size_t i = 0; i < rnum; i++) {
const KCREC* rec = recs + i;
xrecs[std::string(rec->key.buf, rec->key.size)] =
std::string(rec->value.buf, rec->value.size);
}
return pdb->set_bulk(xrecs, atomic);
}
/**
* Remove records at once.
*/
int64_t kcdbremovebulk(KCDB* db, const KCSTR* keys, size_t knum, int32_t atomic) {
_assert_(db && keys && knum <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
std::vector xkeys;
xkeys.reserve(knum);
for (size_t i = 0; i < knum; i++) {
const KCSTR* key = keys + i;
xkeys.push_back(std::string(key->buf, key->size));
}
return pdb->remove_bulk(xkeys, atomic);
}
/**
* Retrieve records at once.
*/
int64_t kcdbgetbulk(KCDB* db, const KCSTR* keys, size_t knum, KCREC* recs, int32_t atomic) {
_assert_(db && keys && knum <= MEMMAXSIZ && recs);
PolyDB* pdb = (PolyDB*)db;
std::vector xkeys;
xkeys.reserve(knum);
for (size_t i = 0; i < knum; i++) {
const KCSTR* key = keys + i;
xkeys.push_back(std::string(key->buf, key->size));
}
std::map xrecs;
if (pdb->get_bulk(xkeys, &xrecs, atomic) < 0) return -1;
std::map::iterator it = xrecs.begin();
std::map::iterator itend = xrecs.end();
size_t ridx = 0;
while (ridx < knum && it != itend) {
size_t ksiz = it->first.size();
char* kbuf = new char[ksiz+1];
std::memcpy(kbuf, it->first.data(), ksiz);
kbuf[ksiz] = '\0';
size_t vsiz = it->second.size();
char* vbuf = new char[vsiz+1];
std::memcpy(vbuf, it->second.data(), vsiz);
vbuf[vsiz] = '\0';
KCREC* rec = recs + (ridx++);
rec->key.buf = kbuf;
rec->key.size = ksiz;
rec->value.buf = vbuf;
rec->value.size = vsiz;
++it;
}
return ridx;
}
/**
* Synchronize updated contents with the file and the device.
*/
int32_t kcdbsync(KCDB* db, int32_t hard, KCFILEPROC proc, void* opq) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
class FileProcessorImpl : public BasicDB::FileProcessor {
public:
explicit FileProcessorImpl(KCFILEPROC proc, void* opq) : proc_(proc), opq_(opq) {}
bool process(const std::string& path, int64_t count, int64_t size) {
if (!proc_) return true;
return proc_(path.c_str(), count, size, opq_);
}
private:
KCFILEPROC proc_;
void* opq_;
};
FileProcessorImpl myproc(proc, opq);
return pdb->synchronize(hard, &myproc);
}
/**
* Occupy database by locking and do something meanwhile.
*/
int32_t kcdboccupy(KCDB* db, int32_t writable, KCFILEPROC proc, void* opq) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
class FileProcessorImpl : public BasicDB::FileProcessor {
public:
explicit FileProcessorImpl(KCFILEPROC proc, void* opq) : proc_(proc), opq_(opq) {}
bool process(const std::string& path, int64_t count, int64_t size) {
if (!proc_) return true;
return proc_(path.c_str(), count, size, opq_);
}
private:
KCFILEPROC proc_;
void* opq_;
};
FileProcessorImpl myproc(proc, opq);
return pdb->occupy(writable, &myproc);
}
/**
* Create a copy of the database file.
*/
int32_t kcdbcopy(KCDB* db, const char* dest) {
_assert_(db && dest);
PolyDB* pdb = (PolyDB*)db;
return pdb->copy(dest);
}
/**
* Begin transaction.
*/
int32_t kcdbbegintran(KCDB* db, int32_t hard) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->begin_transaction(hard);
}
/**
* Try to begin transaction.
*/
int32_t kcdbbegintrantry(KCDB* db, int32_t hard) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->begin_transaction_try(hard);
}
/**
* End transaction.
*/
int32_t kcdbendtran(KCDB* db, int32_t commit) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->end_transaction(commit);
}
/**
* Remove all records.
*/
int32_t kcdbclear(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->clear();
}
/**
* Dump records into a file.
*/
int32_t kcdbdumpsnap(KCDB* db, const char* dest) {
_assert_(db && dest);
PolyDB* pdb = (PolyDB*)db;
return pdb->dump_snapshot(dest);
}
/**
* Load records from a file.
*/
int32_t kcdbloadsnap(KCDB* db, const char* src) {
_assert_(db && src);
PolyDB* pdb = (PolyDB*)db;
return pdb->load_snapshot(src);
}
/**
* Get the number of records.
*/
int64_t kcdbcount(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->count();
}
/**
* Get the size of the database file.
*/
int64_t kcdbsize(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return pdb->size();
}
/**
* Get the path of the database file.
*/
char* kcdbpath(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
std::string path = pdb->path();
size_t psiz = path.size();
char* pbuf = new char[psiz+1];
std::memcpy(pbuf, path.c_str(), psiz + 1);
return pbuf;
}
/**
* Get the miscellaneous status information.
*/
char* kcdbstatus(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
std::map status;
if (!pdb->status(&status)) return NULL;
std::ostringstream obuf;
std::map::iterator it = status.begin();
std::map::iterator itend = status.end();
while (it != itend) {
obuf << it->first << "\t" << it->second << "\n";
++it;
}
std::string sstr = obuf.str();
size_t ssiz = sstr.size();
char* sbuf = new char[ssiz+1];
std::memcpy(sbuf, sstr.c_str(), ssiz + 1);
return sbuf;
}
/**
* Get keys matching a prefix string.
*/
int64_t kcdbmatchprefix(KCDB* db, const char* prefix, char** strary, size_t max) {
_assert_(db && prefix && strary && max <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
std::vector strvec;
if (pdb->match_prefix(prefix, &strvec, max) == -1) return -1;
int64_t cnt = 0;
std::vector::iterator it = strvec.begin();
std::vector::iterator itend = strvec.end();
while (it != itend) {
size_t ksiz = it->size();
char* kbuf = new char[ksiz+1];
std::memcpy(kbuf, it->data(), ksiz);
kbuf[ksiz] = '\0';
strary[cnt++] = kbuf;
++it;
}
return cnt;
}
/**
* Get keys matching a regular expression string.
*/
int64_t kcdbmatchregex(KCDB* db, const char* regex, char** strary, size_t max) {
_assert_(db && regex && strary && max <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
std::vector strvec;
if (pdb->match_regex(regex, &strvec, max) == -1) return -1;
int64_t cnt = 0;
std::vector::iterator it = strvec.begin();
std::vector::iterator itend = strvec.end();
while (it != itend) {
size_t ksiz = it->size();
char* kbuf = new char[ksiz+1];
std::memcpy(kbuf, it->data(), ksiz);
kbuf[ksiz] = '\0';
strary[cnt++] = kbuf;
++it;
}
return cnt;
}
/**
* Get keys similar to a string in terms of the levenshtein distance.
*/
int64_t kcdbmatchsimilar(KCDB* db, const char* origin, uint32_t range, int32_t utf,
char** strary, size_t max) {
_assert_(db && origin && strary && max <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
std::vector strvec;
if (pdb->match_similar(origin, range, utf, &strvec, max) == -1) return -1;
int64_t cnt = 0;
std::vector::iterator it = strvec.begin();
std::vector::iterator itend = strvec.end();
while (it != itend) {
size_t ksiz = it->size();
char* kbuf = new char[ksiz+1];
std::memcpy(kbuf, it->data(), ksiz);
kbuf[ksiz] = '\0';
strary[cnt++] = kbuf;
++it;
}
return cnt;
}
/**
* Merge records from other databases.
*/
int32_t kcdbmerge(KCDB* db, KCDB** srcary, size_t srcnum, uint32_t mode) {
_assert_(db && srcary && srcnum <= MEMMAXSIZ);
PolyDB* pdb = (PolyDB*)db;
return pdb->merge((BasicDB**)srcary, srcnum, (PolyDB::MergeMode)mode);
}
/**
* Create a cursor object.
*/
KCCUR* kcdbcursor(KCDB* db) {
_assert_(db);
PolyDB* pdb = (PolyDB*)db;
return (KCCUR*)pdb->cursor();
}
/**
* Destroy a cursor object.
*/
void kccurdel(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
delete pcur;
}
/**
* Accept a visitor to the current record.
*/
int32_t kccuraccept(KCCUR* cur, KCVISITFULL fullproc, void* opq,
int32_t writable, int32_t step) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
class VisitorImpl : public DB::Visitor {
public:
explicit VisitorImpl(KCVISITFULL fullproc, void* opq) : fullproc_(fullproc), opq_(opq) {}
const char* visit_full(const char* kbuf, size_t ksiz,
const char* vbuf, size_t vsiz, size_t* sp) {
if (!fullproc_) return NOP;
return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_);
}
private:
KCVISITFULL fullproc_;
void* opq_;
};
VisitorImpl visitor(fullproc, opq);
return pcur->accept(&visitor, writable, step);
}
/**
* Set the value of the current record.
*/
int32_t kccursetvalue(KCCUR* cur, const char* vbuf, size_t vsiz, int32_t step) {
_assert_(cur && vbuf && vsiz <= MEMMAXSIZ);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->set_value(vbuf, vsiz, step);
}
/**
* Remove the current record.
*/
int32_t kccurremove(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->remove();
}
/**
* Get the key of the current record.
*/
char* kccurgetkey(KCCUR* cur, size_t* sp, int32_t step) {
_assert_(cur && sp);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->get_key(sp, step);
}
/**
* Get the value of the current record.
*/
char* kccurgetvalue(KCCUR* cur, size_t* sp, int32_t step) {
_assert_(cur && sp);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->get_value(sp, step);
}
/**
* Get a pair of the key and the value of the current record.
*/
char* kccurget(KCCUR* cur, size_t* ksp, const char** vbp, size_t* vsp, int32_t step) {
_assert_(cur && ksp && vbp && vsp);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->get(ksp, vbp, vsp, step);
}
/**
* Get a pair of the key and the value of the current record and remove it atomically.
*/
char* kccurseize(KCCUR* cur, size_t* ksp, const char** vbp, size_t* vsp) {
_assert_(cur && ksp && vbp && vsp);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->seize(ksp, vbp, vsp);
}
/**
* Jump the cursor to the first record.
*/
int32_t kccurjump(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->jump();
}
/**
* Jump the cursor to a record.
*/
int32_t kccurjumpkey(KCCUR* cur, const char* kbuf, size_t ksiz) {
_assert_(cur && kbuf && ksiz <= MEMMAXSIZ);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->jump(kbuf, ksiz);
}
/**
* Jump the cursor to the last record for backward scan.
*/
int32_t kccurjumpback(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->jump_back();
}
/**
* Jump the cursor to a record for backward scan.
*/
int32_t kccurjumpbackkey(KCCUR* cur, const char* kbuf, size_t ksiz) {
_assert_(cur && kbuf && ksiz <= MEMMAXSIZ);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->jump_back(kbuf, ksiz);
}
/**
* Step the cursor to the next record.
*/
int32_t kccurstep(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->step();
}
/**
* Step the cursor to the previous record.
*/
int32_t kccurstepback(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->step_back();
}
/**
* Get the database object.
*/
KCDB* kccurdb(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return (KCDB*)pcur->db();
}
/**
* Get the code of the last happened error.
*/
int32_t kccurecode(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->error().code();
}
/**
* Get the supplement message of the last happened error.
*/
const char* kccuremsg(KCCUR* cur) {
_assert_(cur);
PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur;
return pcur->error().message();
}
/**
* Create an index database object.
*/
KCIDX* kcidxnew(void) {
_assert_(true);
return (KCIDX*)new IndexDB;
}
/**
* Destroy a database object.
*/
void kcidxdel(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
delete idb;
}
/**
* Open a database file.
*/
int32_t kcidxopen(KCIDX* idx, const char* path, uint32_t mode) {
_assert_(idx && path);
IndexDB* idb = (IndexDB*)idx;
return idb->open(path, mode);
}
/**
* Close the database file.
*/
int32_t kcidxclose(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
return idb->close();
}
/**
* Get the code of the last happened error.
*/
int32_t kcidxecode(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
return idb->error().code();
}
/**
* Get the supplement message of the last happened error.
*/
const char* kcidxemsg(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
return idb->error().message();
}
/**
* Set the value of a record.
*/
int32_t kcidxset(KCIDX* idx, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(idx && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
IndexDB* idb = (IndexDB*)idx;
return idb->set(kbuf, ksiz, vbuf, vsiz);
}
/**
* Add a record.
*/
int32_t kcidxadd(KCIDX* idx, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(idx && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
IndexDB* idb = (IndexDB*)idx;
return idb->add(kbuf, ksiz, vbuf, vsiz);
}
/**
* Replace the value of a record.
*/
int32_t kcidxreplace(KCIDX* idx, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(idx && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
IndexDB* idb = (IndexDB*)idx;
return idb->replace(kbuf, ksiz, vbuf, vsiz);
}
/**
* Append the value of a record.
*/
int32_t kcidxappend(KCIDX* idx, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(idx && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
IndexDB* idb = (IndexDB*)idx;
return idb->append(kbuf, ksiz, vbuf, vsiz);
}
/**
* Remove a record.
*/
int32_t kcidxremove(KCIDX* idx, const char* kbuf, size_t ksiz) {
_assert_(idx && kbuf && ksiz <= MEMMAXSIZ);
IndexDB* idb = (IndexDB*)idx;
return idb->remove(kbuf, ksiz);
}
/**
* Retrieve the value of a record.
*/
char* kcidxget(KCIDX* idx, const char* kbuf, size_t ksiz, size_t* sp) {
_assert_(idx && kbuf && ksiz <= MEMMAXSIZ && sp);
IndexDB* idb = (IndexDB*)idx;
return idb->get(kbuf, ksiz, sp);
}
/**
* Synchronize updated contents with the file and the device.
*/
int32_t kcidxsync(KCIDX* idx, int32_t hard, KCFILEPROC proc, void* opq) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
class FileProcessorImpl : public BasicDB::FileProcessor {
public:
explicit FileProcessorImpl(KCFILEPROC proc, void* opq) : proc_(proc), opq_(opq) {}
bool process(const std::string& path, int64_t count, int64_t size) {
if (!proc_) return true;
return proc_(path.c_str(), count, size, opq_);
}
private:
KCFILEPROC proc_;
void* opq_;
};
FileProcessorImpl myproc(proc, opq);
return idb->synchronize(hard, &myproc);
}
/**
* Remove all records.
*/
int32_t kcidxclear(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
return idb->clear();
}
/**
* Get the number of records.
*/
int64_t kcidxcount(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
return idb->count();
}
/**
* Get the size of the database file.
*/
int64_t kcidxsize(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
return idb->size();
}
/**
* Get the path of the database file.
*/
char* kcidxpath(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
std::string path = idb->path();
size_t psiz = path.size();
char* pbuf = new char[psiz+1];
std::memcpy(pbuf, path.c_str(), psiz + 1);
return pbuf;
}
/**
* Get the miscellaneous status information.
*/
char* kcidxstatus(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
std::map status;
if (!idb->status(&status)) return NULL;
std::ostringstream obuf;
std::map::iterator it = status.begin();
std::map::iterator itend = status.end();
while (it != itend) {
obuf << it->first << "\t" << it->second << "\n";
++it;
}
std::string sstr = obuf.str();
size_t ssiz = sstr.size();
char* sbuf = new char[ssiz+1];
std::memcpy(sbuf, sstr.c_str(), ssiz + 1);
return sbuf;
}
/**
* Reveal the inner database object.
*/
KCDB* kcidxrevealinnerdb(KCIDX* idx) {
_assert_(idx);
IndexDB* idb = (IndexDB*)idx;
return (KCDB*)idb->reveal_inner_db();
}
/**
* Create a string hash map object.
*/
KCMAP* kcmapnew(size_t bnum) {
_assert_(true);
return (KCMAP*)new TinyHashMap(bnum);
}
/**
* Destroy a map object.
*/
void kcmapdel(KCMAP* map) {
_assert_(map);
TinyHashMap* thm = (TinyHashMap*)map;
delete thm;
}
/**
* Set the value of a record.
*/
void kcmapset(KCMAP* map, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(map && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
TinyHashMap* thm = (TinyHashMap*)map;
thm->set(kbuf, ksiz, vbuf, vsiz);
}
/**
* Add a record.
*/
int32_t kcmapadd(KCMAP* map, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(map && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
TinyHashMap* thm = (TinyHashMap*)map;
return thm->add(kbuf, ksiz, vbuf, vsiz);
}
/**
* Replace the value of a record.
*/
int32_t kcmapreplace(KCMAP* map, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(map && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
TinyHashMap* thm = (TinyHashMap*)map;
return thm->replace(kbuf, ksiz, vbuf, vsiz);
}
/**
* Append the value of a record.
*/
void kcmapappend(KCMAP* map, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
_assert_(map && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
TinyHashMap* thm = (TinyHashMap*)map;
thm->append(kbuf, ksiz, vbuf, vsiz);
}
/**
* Remove a record.
*/
int32_t kcmapremove(KCMAP* map, const char* kbuf, size_t ksiz) {
_assert_(map && kbuf && ksiz <= MEMMAXSIZ);
TinyHashMap* thm = (TinyHashMap*)map;
return thm->remove(kbuf, ksiz);
}
/**
* Retrieve the value of a record.
*/
const char* kcmapget(KCMAP* map, const char* kbuf, size_t ksiz, size_t* sp) {
_assert_(map && kbuf && ksiz <= MEMMAXSIZ && sp);
TinyHashMap* thm = (TinyHashMap*)map;
return thm->get(kbuf, ksiz, sp);
}
/**
* Remove all records.
*/
void kcmapclear(KCMAP* map) {
_assert_(map);
TinyHashMap* thm = (TinyHashMap*)map;
thm->clear();
}
/**
* Get the number of records.
*/
size_t kcmapcount(KCMAP* map) {
_assert_(map);
TinyHashMap* thm = (TinyHashMap*)map;
return thm->count();
}
/**
* Create a string hash map iterator object.
*/
KCMAPITER* kcmapiterator(KCMAP* map) {
_assert_(map);
TinyHashMap* thm = (TinyHashMap*)map;
return (KCMAPITER*)new TinyHashMap::Iterator(thm);
}
/**
* Destroy an iterator object.
*/
void kcmapiterdel(KCMAPITER* iter) {
_assert_(iter);
TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter;
delete thmi;
}
/**
* Get the key of the current record.
*/
const char* kcmapitergetkey(KCMAPITER* iter, size_t* sp) {
_assert_(iter && sp);
TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter;
return thmi->get_key(sp);
}
/**
* Get the value of the current record.
*/
const char* kcmapitergetvalue(KCMAPITER* iter, size_t* sp) {
_assert_(iter && sp);
TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter;
return thmi->get_value(sp);
}
/**
* Get a pair of the key and the value of the current record.
*/
const char* kcmapiterget(KCMAPITER* iter, size_t* ksp, const char** vbp, size_t* vsp) {
_assert_(iter && ksp && vbp && vsp);
TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter;
return thmi->get(ksp, vbp, vsp);
}
/**
* Step the cursor to the next record.
*/
void kcmapiterstep(KCMAPITER* iter) {
_assert_(iter);
TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter;
return thmi->step();
}
/**
* Create a string hash map sorter object.
*/
KCMAPSORT* kcmapsorter(KCMAP* map) {
_assert_(map);
TinyHashMap* thm = (TinyHashMap*)map;
return (KCMAPSORT*)new TinyHashMap::Sorter(thm);
}
/**
* Destroy an sorter object.
*/
void kcmapsortdel(KCMAPSORT* sort) {
_assert_(sort);
TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort;
delete thms;
}
/**
* Get the key of the current record.
*/
const char* kcmapsortgetkey(KCMAPSORT* sort, size_t* sp) {
_assert_(sort && sp);
TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort;
return thms->get_key(sp);
}
/**
* Get the value of the current record.
*/
const char* kcmapsortgetvalue(KCMAPSORT* sort, size_t* sp) {
_assert_(sort && sp);
TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort;
return thms->get_value(sp);
}
/**
* Get a pair of the key and the value of the current record.
*/
const char* kcmapsortget(KCMAPSORT* sort, size_t* ksp, const char** vbp, size_t* vsp) {
_assert_(sort && ksp && vbp && vsp);
TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort;
return thms->get(ksp, vbp, vsp);
}
/**
* Step the cursor to the next record.
*/
void kcmapsortstep(KCMAPSORT* sort) {
_assert_(sort);
TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort;
return thms->step();
}
/**
* Create a string array list object.
*/
KCLIST* kclistnew() {
_assert_(true);
return (KCLIST*)new TinyArrayList();
}
/**
* Destroy a list object.
*/
void kclistdel(KCLIST* list) {
_assert_(list);
TinyArrayList* tal = (TinyArrayList*)list;
delete tal;
}
/**
* Insert a record at the bottom of the list.
*/
void kclistpush(KCLIST* list, const char* buf, size_t size) {
_assert_(list && buf && size <= MEMMAXSIZ);
TinyArrayList* tal = (TinyArrayList*)list;
tal->push(buf, size);
}
/**
* Remove a record at the bottom of the list.
*/
int32_t kclistpop(KCLIST* list) {
_assert_(list);
TinyArrayList* tal = (TinyArrayList*)list;
return tal->pop();
}
/**
* Insert a record at the top of the list.
*/
void kclistunshift(KCLIST* list, const char* buf, size_t size) {
_assert_(list && buf && size <= MEMMAXSIZ);
TinyArrayList* tal = (TinyArrayList*)list;
tal->unshift(buf, size);
}
/**
* Remove a record at the top of the list.
*/
int32_t kclistshift(KCLIST* list) {
_assert_(list);
TinyArrayList* tal = (TinyArrayList*)list;
return tal->shift();
}
/**
* Insert a record at the position of the given index of the list.
*/
void kclistinsert(KCLIST* list, const char* buf, size_t size, size_t idx) {
_assert_(list && buf && size <= MEMMAXSIZ);
TinyArrayList* tal = (TinyArrayList*)list;
tal->insert(buf, size, idx);
}
/**
* Remove a record at the position of the given index of the list.
*/
void kclistremove(KCLIST* list, size_t idx) {
_assert_(list);
TinyArrayList* tal = (TinyArrayList*)list;
return tal->remove(idx);
}
/**
* Retrieve a record at the position of the given index of the list.
*/
const char* kclistget(KCLIST* list, size_t idx, size_t* sp) {
_assert_(list && sp);
TinyArrayList* tal = (TinyArrayList*)list;
return tal->get(idx, sp);
}
/**
* Remove all records.
*/
void kclistclear(KCLIST* list) {
_assert_(list);
TinyArrayList* tal = (TinyArrayList*)list;
tal->clear();
}
/**
* Get the number of records.
*/
size_t kclistcount(KCLIST* list) {
_assert_(list);
TinyArrayList* tal = (TinyArrayList*)list;
return tal->count();
}
}
// END OF FILE