summaryrefslogtreecommitdiff
path: root/plugins/Dbx_kyoto/src/kyotocabinet/kcdb.h
diff options
context:
space:
mode:
authorGeorge Hazan <george.hazan@gmail.com>2015-04-02 17:28:07 +0000
committerGeorge Hazan <george.hazan@gmail.com>2015-04-02 17:28:07 +0000
commit2e511ab1b1ff3d78c695874e3b28ff4ce7680cc8 (patch)
tree9c3588c82da7ad3e326f51d899800ad183f0d826 /plugins/Dbx_kyoto/src/kyotocabinet/kcdb.h
parent0f73f1572a03e5bae2664c1b2bb2cd18a1e33fca (diff)
kyotocabinet based db driver
first version that compiles DO NOT USE IT, dragons live there git-svn-id: http://svn.miranda-ng.org/main/trunk@12580 1316c22d-e87f-b044-9b9b-93d7a3e3ba9c
Diffstat (limited to 'plugins/Dbx_kyoto/src/kyotocabinet/kcdb.h')
-rw-r--r--plugins/Dbx_kyoto/src/kyotocabinet/kcdb.h2520
1 files changed, 2520 insertions, 0 deletions
diff --git a/plugins/Dbx_kyoto/src/kyotocabinet/kcdb.h b/plugins/Dbx_kyoto/src/kyotocabinet/kcdb.h
new file mode 100644
index 0000000000..8abffb4c6b
--- /dev/null
+++ b/plugins/Dbx_kyoto/src/kyotocabinet/kcdb.h
@@ -0,0 +1,2520 @@
+/*************************************************************************************************
+ * Database interface
+ * 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 <http://www.gnu.org/licenses/>.
+ *************************************************************************************************/
+
+
+#ifndef _KCDB_H // duplication check
+#define _KCDB_H
+
+#include <kccommon.h>
+#include <kcutil.h>
+#include <kcthread.h>
+#include <kcfile.h>
+#include <kccompress.h>
+#include <kccompare.h>
+#include <kcmap.h>
+
+#define KCDBSSMAGICDATA "KCSS\n" ///< The magic data of the snapshot file
+
+namespace kyotocabinet { // common namespace
+
+
+/**
+ * Interface of database abstraction.
+ * @note This class is an abstract class to prescribe the interface of record access.
+ */
+class DB {
+ public:
+ /**
+ * Interface to access a record.
+ */
+ class Visitor {
+ public:
+ /** Special pointer for no operation. */
+ static const char* const NOP;
+ /** Special pointer to remove the record. */
+ static const char* const REMOVE;
+ /**
+ * Destructor.
+ */
+ virtual ~Visitor() {
+ _assert_(true);
+ }
+ /**
+ * Visit a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @return If it is the pointer to a region, the value is replaced by the content. If it
+ * is Visitor::NOP, nothing is modified. If it is Visitor::REMOVE, the record is removed.
+ */
+ virtual const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ && sp);
+ return NOP;
+ }
+ /**
+ * Visit a empty record space.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @return If it is the pointer to a region, the value is replaced by the content. If it
+ * is Visitor::NOP or Visitor::REMOVE, nothing is modified.
+ */
+ virtual const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && sp);
+ return NOP;
+ }
+ /**
+ * Preprocess the main operations.
+ */
+ virtual void visit_before() {
+ _assert_(true);
+ }
+ /**
+ * Postprocess the main operations.
+ */
+ virtual void visit_after() {
+ _assert_(true);
+ }
+ };
+ /**
+ * Interface of cursor to indicate a record.
+ */
+ class Cursor {
+ public:
+ /**
+ * Destructor.
+ */
+ virtual ~Cursor() {
+ _assert_(true);
+ }
+ /**
+ * Accept a visitor to the current record.
+ * @param visitor a visitor object.
+ * @param writable true for writable operation, or false for read-only operation.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return true on success, or false on failure.
+ * @note The operation for each record is performed atomically and other threads accessing
+ * the same record are blocked. To avoid deadlock, any explicit database operation must not
+ * be performed in this function.
+ */
+ virtual bool accept(Visitor* visitor, bool writable = true, bool step = false) = 0;
+ /**
+ * Set the value of the current record.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return true on success, or false on failure.
+ */
+ virtual bool set_value(const char* vbuf, size_t vsiz, bool step = false) = 0;
+ /**
+ * Set the value of the current record.
+ * @note Equal to the original Cursor::set_value method except that the parameter is
+ * std::string.
+ */
+ virtual bool set_value_str(const std::string& value, bool step = false) = 0;
+ /**
+ * Remove the current record.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, false is returned. The cursor is moved to the
+ * next record implicitly.
+ */
+ virtual bool remove() = 0;
+ /**
+ * Get the key of the current record.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return the pointer to the key region of the current record, or NULL on failure.
+ * @note If the cursor is invalidated, NULL is returned. Because an additional zero
+ * code is appended at the end of the region of the return value, the return value can be
+ * treated as a C-style string. Because the region of the return value is allocated with the
+ * the new[] operator, it should be released with the delete[] operator when it is no longer
+ * in use.
+ */
+ virtual char* get_key(size_t* sp, bool step = false) = 0;
+ /**
+ * Get the key of the current record.
+ * @note Equal to the original Cursor::get_key method except that a parameter is a string to
+ * contain the result and the return value is bool for success.
+ */
+ virtual bool get_key(std::string* key, bool step = false) = 0;
+ /**
+ * Get the value of the current record.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return the pointer to the value region of the current record, or NULL on failure.
+ * @note If the cursor is invalidated, NULL is returned. Because an additional zero
+ * code is appended at the end of the region of the return value, the return value can be
+ * treated as a C-style string. Because the region of the return value is allocated with the
+ * the new[] operator, it should be released with the delete[] operator when it is no longer
+ * in use.
+ */
+ virtual char* get_value(size_t* sp, bool step = false) = 0;
+ /**
+ * Get the value of the current record.
+ * @note Equal to the original Cursor::get_value method except that a parameter is a string
+ * to contain the result and the return value is bool for success.
+ */
+ virtual bool get_value(std::string* value, bool step = false) = 0;
+ /**
+ * Get a pair of the key and the value of the current record.
+ * @param ksp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @param vbp the pointer to the variable into which the pointer to the value region is
+ * assigned.
+ * @param vsp the pointer to the variable into which the size of the value region is
+ * assigned.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return the pointer to the key region, or NULL on failure.
+ * @note If the cursor is invalidated, NULL is returned. Because an additional zero code is
+ * appended at the end of each region of the key and the value, each region can be treated
+ * as a C-style string. The return value should be deleted explicitly by the caller with
+ * the detele[] operator.
+ */
+ virtual char* get(size_t* ksp, const char** vbp, size_t* vsp, bool step = false) = 0;
+ /**
+ * Get a pair of the key and the value of the current record.
+ * @note Equal to the original Cursor::get method except that parameters are strings
+ * to contain the result and the return value is bool for success.
+ */
+ virtual bool get(std::string* key, std::string* value, bool step = false) = 0;
+ /**
+ * Jump the cursor to the first record for forward scan.
+ * @return true on success, or false on failure.
+ */
+ virtual bool jump() = 0;
+ /**
+ * Jump the cursor to a record for forward scan.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @return true on success, or false on failure.
+ */
+ virtual bool jump(const char* kbuf, size_t ksiz) = 0;
+ /**
+ * Jump the cursor to a record for forward scan.
+ * @note Equal to the original Cursor::jump method except that the parameter is std::string.
+ */
+ virtual bool jump(const std::string& key) = 0;
+ /**
+ * Jump the cursor to the last record for backward scan.
+ * @return true on success, or false on failure.
+ * @note This method is dedicated to tree databases. Some database types, especially hash
+ * databases, will provide a dummy implementation.
+ */
+ virtual bool jump_back() = 0;
+ /**
+ * Jump the cursor to a record for backward scan.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @return true on success, or false on failure.
+ * @note This method is dedicated to tree databases. Some database types, especially hash
+ * databases, will provide a dummy implementation.
+ */
+ virtual bool jump_back(const char* kbuf, size_t ksiz) = 0;
+ /**
+ * Jump the cursor to a record for backward scan.
+ * @note Equal to the original Cursor::jump_back method except that the parameter is
+ * std::string.
+ */
+ virtual bool jump_back(const std::string& key) = 0;
+ /**
+ * Step the cursor to the next record.
+ * @return true on success, or false on failure.
+ */
+ virtual bool step() = 0;
+ /**
+ * Step the cursor to the previous record.
+ * @return true on success, or false on failure.
+ * @note This method is dedicated to tree databases. Some database types, especially hash
+ * databases, will provide a dummy implementation.
+ */
+ virtual bool step_back() = 0;
+ /**
+ * Get the database object.
+ * @return the database object.
+ */
+ virtual DB* db() = 0;
+ };
+ /**
+ * Destructor.
+ */
+ virtual ~DB() {
+ _assert_(true);
+ }
+ /**
+ * Accept a visitor to a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param visitor a visitor object.
+ * @param writable true for writable operation, or false for read-only operation.
+ * @return true on success, or false on failure.
+ * @note The operation for each record is performed atomically and other threads accessing the
+ * same record are blocked. To avoid deadlock, any explicit database operation must not be
+ * performed in this function.
+ */
+ virtual bool accept(const char* kbuf, size_t ksiz, Visitor* visitor, bool writable = true) = 0;
+ /**
+ * Set the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, a new record is created. If the corresponding
+ * record exists, the value is overwritten.
+ */
+ virtual bool set(const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) = 0;
+ /**
+ * Set the value of a record.
+ * @note Equal to the original DB::set method except that the parameters are std::string.
+ */
+ virtual bool set(const std::string& key, const std::string& value) = 0;
+ /**
+ * Add a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, a new record is created. If the corresponding
+ * record exists, the record is not modified and false is returned.
+ */
+ virtual bool add(const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) = 0;
+ /**
+ * Set the value of a record.
+ * @note Equal to the original DB::add method except that the parameters are std::string.
+ */
+ virtual bool add(const std::string& key, const std::string& value) = 0;
+ /**
+ * Replace the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, no new record is created and false is returned.
+ * If the corresponding record exists, the value is modified.
+ */
+ virtual bool replace(const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) = 0;
+ /**
+ * Replace the value of a record.
+ * @note Equal to the original DB::replace method except that the parameters are std::string.
+ */
+ virtual bool replace(const std::string& key, const std::string& value) = 0;
+ /**
+ * Append the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, a new record is created. If the corresponding
+ * record exists, the given value is appended at the end of the existing value.
+ */
+ virtual bool append(const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) = 0;
+ /**
+ * Set the value of a record.
+ * @note Equal to the original DB::append method except that the parameters are std::string.
+ */
+ virtual bool append(const std::string& key, const std::string& value) = 0;
+ /**
+ * Add a number to the numeric integer value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param num the additional number.
+ * @param orig the origin number if no record corresponds to the key. If it is INT64MIN and
+ * no record corresponds, this function fails. If it is INT64MAX, the value is set as the
+ * additional number regardless of the current value.
+ * @return the result value, or kyotocabinet::INT64MIN on failure.
+ * @note The value is serialized as an 8-byte binary integer in big-endian order, not a decimal
+ * string. If existing value is not 8-byte, this function fails.
+ */
+ virtual int64_t increment(const char* kbuf, size_t ksiz, int64_t num, int64_t orig = 0) = 0;
+ /**
+ * Add a number to the numeric integer value of a record.
+ * @note Equal to the original DB::increment method except that the parameter is std::string.
+ */
+ virtual int64_t increment(const std::string& key, int64_t num, int64_t orig = 0) = 0;
+ /**
+ * Add a number to the numeric double value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param num the additional number.
+ * @param orig the origin number if no record corresponds to the key. If it is negative
+ * infinity and no record corresponds, this function fails. If it is positive infinity, the
+ * value is set as the additional number regardless of the current value.
+ * @return the result value, or Not-a-number on failure.
+ * @note The value is serialized as an 16-byte binary fixed-point number in big-endian order,
+ * not a decimal string. If existing value is not 16-byte, this function fails.
+ */
+ virtual double increment_double(const char* kbuf, size_t ksiz, double num,
+ double orig = 0) = 0;
+ /**
+ * Add a number to the numeric double value of a record.
+ * @note Equal to the original DB::increment_double method except that the parameter is
+ * std::string.
+ */
+ virtual double increment_double(const std::string& key, double num, double orig = 0) = 0;
+ /**
+ * Perform compare-and-swap.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param ovbuf the pointer to the old value region. NULL means that no record corresponds.
+ * @param ovsiz the size of the old value region.
+ * @param nvbuf the pointer to the new value region. NULL means that the record is removed.
+ * @param nvsiz the size of new old value region.
+ * @return true on success, or false on failure.
+ */
+ virtual bool cas(const char* kbuf, size_t ksiz,
+ const char* ovbuf, size_t ovsiz, const char* nvbuf, size_t nvsiz) = 0;
+ /**
+ * Perform compare-and-swap.
+ * @note Equal to the original DB::cas method except that the parameters are std::string.
+ */
+ virtual bool cas(const std::string& key,
+ const std::string& ovalue, const std::string& nvalue) = 0;
+ /**
+ * Remove a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, false is returned.
+ */
+ virtual bool remove(const char* kbuf, size_t ksiz) = 0;
+ /**
+ * Remove a record.
+ * @note Equal to the original DB::remove method except that the parameter is std::string.
+ */
+ virtual bool remove(const std::string& key) = 0;
+ /**
+ * Retrieve the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @return the pointer to the value region of the corresponding record, or NULL on failure.
+ * @note If no record corresponds to the key, NULL is returned. Because an additional zero
+ * code is appended at the end of the region of the return value, the return value can be
+ * treated as a C-style string. Because the region of the return value is allocated with the
+ * the new[] operator, it should be released with the delete[] operator when it is no longer
+ * in use.
+ */
+ virtual char* get(const char* kbuf, size_t ksiz, size_t* sp) = 0;
+ /**
+ * Retrieve the value of a record.
+ * @note Equal to the original DB::get method except that the first parameters is the key
+ * string and the second parameter is a string to contain the result and the return value is
+ * bool for success.
+ */
+ virtual bool get(const std::string& key, std::string* value) = 0;
+ /**
+ * Retrieve the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the buffer into which the value of the corresponding record is
+ * written.
+ * @param max the size of the buffer.
+ * @return the size of the value, or -1 on failure.
+ */
+ virtual int32_t get(const char* kbuf, size_t ksiz, char* vbuf, size_t max) = 0;
+ /**
+ * Check the existence of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @return the size of the value, or -1 on failure.
+ */
+ virtual int32_t check(const char* kbuf, size_t ksiz) = 0;
+ /**
+ * Check the existence of a record.
+ * @note Equal to the original DB::check method except that the parameter is std::string.
+ */
+ virtual int32_t check(const std::string& key) = 0;
+ /**
+ * Remove all records.
+ * @return true on success, or false on failure.
+ */
+ virtual bool clear() = 0;
+ /**
+ * Get the number of records.
+ * @return the number of records, or -1 on failure.
+ */
+ virtual int64_t count() = 0;
+ /**
+ * Create a cursor object.
+ * @return the return value is the created cursor object.
+ * @note Because the object of the return value is allocated by the constructor, it should be
+ * released with the delete operator when it is no longer in use.
+ */
+ virtual Cursor* cursor() = 0;
+};
+
+
+/**
+ * Basic implementation of database.
+ * @note This class is an abstract class to prescribe the interface of file operations and
+ * provide mix-in methods. This class can be inherited but overwriting methods is forbidden.
+ * Before every database operation, it is necessary to call the BasicDB::open method in order to
+ * open a database file and connect the database object to it. To avoid data missing or
+ * corruption, it is important to close every database file by the BasicDB::close method when the
+ * database is no longer in use. It is forbidden for multible database objects in a process to
+ * open the same database at the same time. It is forbidden to share a database object with
+ * child processes.
+ */
+class BasicDB : public DB {
+ public:
+ class Cursor;
+ class Error;
+ class ProgressChecker;
+ class FileProcessor;
+ class Logger;
+ class MetaTrigger;
+ private:
+ /** The size of the IO buffer. */
+ static const size_t IOBUFSIZ = 8192;
+ public:
+ /**
+ * Database types.
+ */
+ enum Type {
+ TYPEVOID = 0x00, ///< void database
+ TYPEPHASH = 0x10, ///< prototype hash database
+ TYPEPTREE = 0x11, ///< prototype tree database
+ TYPESTASH = 0x18, ///< stash database
+ TYPECACHE = 0x20, ///< cache hash database
+ TYPEGRASS = 0x21, ///< cache tree database
+ TYPEHASH = 0x30, ///< file hash database
+ TYPETREE = 0x31, ///< file tree database
+ TYPEDIR = 0x40, ///< directory hash database
+ TYPEFOREST = 0x41, ///< directory tree database
+ TYPETEXT = 0x50, ///< plain text database
+ TYPEMISC = 0x80 ///< miscellaneous database
+ };
+ /**
+ * Interface of cursor to indicate a record.
+ */
+ class Cursor : public DB::Cursor {
+ public:
+ /**
+ * Destructor.
+ */
+ virtual ~Cursor() {
+ _assert_(true);
+ }
+ /**
+ * Set the value of the current record.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return true on success, or false on failure.
+ */
+ bool set_value(const char* vbuf, size_t vsiz, bool step = false) {
+ _assert_(vbuf && vsiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(const char* vbuf, size_t vsiz) :
+ vbuf_(vbuf), vsiz_(vsiz), ok_(false) {}
+ bool ok() const {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ ok_ = true;
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ const char* vbuf_;
+ size_t vsiz_;
+ bool ok_;
+ };
+ VisitorImpl visitor(vbuf, vsiz);
+ if (!accept(&visitor, true, step)) return false;
+ if (!visitor.ok()) return false;
+ return true;
+ }
+ /**
+ * Set the value of the current record.
+ * @note Equal to the original Cursor::set_value method except that the parameter is
+ * std::string.
+ */
+ bool set_value_str(const std::string& value, bool step = false) {
+ _assert_(true);
+ return set_value(value.c_str(), value.size(), step);
+ }
+ /**
+ * Remove the current record.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, false is returned. The cursor is moved to the
+ * next record implicitly.
+ */
+ bool remove() {
+ _assert_(true);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : ok_(false) {}
+ bool ok() const {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ ok_ = true;
+ return REMOVE;
+ }
+ bool ok_;
+ };
+ VisitorImpl visitor;
+ if (!accept(&visitor, true, false)) return false;
+ if (!visitor.ok()) return false;
+ return true;
+ }
+ /**
+ * Get the key of the current record.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return the pointer to the key region of the current record, or NULL on failure.
+ * @note If the cursor is invalidated, NULL is returned. Because an additional zero
+ * code is appended at the end of the region of the return value, the return value can be
+ * treated as a C-style string. Because the region of the return value is allocated with the
+ * the new[] operator, it should be released with the delete[] operator when it is no longer
+ * in use.
+ */
+ char* get_key(size_t* sp, bool step = false) {
+ _assert_(sp);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : kbuf_(NULL), ksiz_(0) {}
+ char* pop(size_t* sp) {
+ *sp = ksiz_;
+ return kbuf_;
+ }
+ void clear() {
+ delete[] kbuf_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ kbuf_ = new char[ksiz+1];
+ std::memcpy(kbuf_, kbuf, ksiz);
+ kbuf_[ksiz] = '\0';
+ ksiz_ = ksiz;
+ return NOP;
+ }
+ char* kbuf_;
+ size_t ksiz_;
+ };
+ VisitorImpl visitor;
+ if (!accept(&visitor, false, step)) {
+ visitor.clear();
+ *sp = 0;
+ return NULL;
+ }
+ size_t ksiz;
+ char* kbuf = visitor.pop(&ksiz);
+ if (!kbuf) {
+ *sp = 0;
+ return NULL;
+ }
+ *sp = ksiz;
+ return kbuf;
+ }
+ /**
+ * Get the key of the current record.
+ * @note Equal to the original Cursor::get_key method except that a parameter is a string to
+ * contain the result and the return value is bool for success.
+ */
+ bool get_key(std::string* key, bool step = false) {
+ _assert_(key);
+ size_t ksiz;
+ char* kbuf = get_key(&ksiz, step);
+ if (!kbuf) return false;
+ key->clear();
+ key->append(kbuf, ksiz);
+ delete[] kbuf;
+ return true;
+ }
+ /**
+ * Get the value of the current record.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return the pointer to the value region of the current record, or NULL on failure.
+ * @note If the cursor is invalidated, NULL is returned. Because an additional zero
+ * code is appended at the end of the region of the return value, the return value can be
+ * treated as a C-style string. Because the region of the return value is allocated with the
+ * the new[] operator, it should be released with the delete[] operator when it is no longer
+ * in use.
+ */
+ char* get_value(size_t* sp, bool step = false) {
+ _assert_(sp);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : vbuf_(NULL), vsiz_(0) {}
+ char* pop(size_t* sp) {
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ void clear() {
+ delete[] vbuf_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ vbuf_ = new char[vsiz+1];
+ std::memcpy(vbuf_, vbuf, vsiz);
+ vbuf_[vsiz] = '\0';
+ vsiz_ = vsiz;
+ return NOP;
+ }
+ char* vbuf_;
+ size_t vsiz_;
+ };
+ VisitorImpl visitor;
+ if (!accept(&visitor, false, step)) {
+ visitor.clear();
+ *sp = 0;
+ return NULL;
+ }
+ size_t vsiz;
+ char* vbuf = visitor.pop(&vsiz);
+ if (!vbuf) {
+ *sp = 0;
+ return NULL;
+ }
+ *sp = vsiz;
+ return vbuf;
+ }
+ /**
+ * Get the value of the current record.
+ * @note Equal to the original Cursor::get_value method except that a parameter is a string
+ * to contain the result and the return value is bool for success.
+ */
+ bool get_value(std::string* value, bool step = false) {
+ _assert_(value);
+ size_t vsiz;
+ char* vbuf = get_value(&vsiz, step);
+ if (!vbuf) return false;
+ value->clear();
+ value->append(vbuf, vsiz);
+ delete[] vbuf;
+ return true;
+ }
+ /**
+ * Get a pair of the key and the value of the current record.
+ * @param ksp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @param vbp the pointer to the variable into which the pointer to the value region is
+ * assigned.
+ * @param vsp the pointer to the variable into which the size of the value region is
+ * assigned.
+ * @param step true to move the cursor to the next record, or false for no move.
+ * @return the pointer to the key region, or NULL on failure.
+ * @note If the cursor is invalidated, NULL is returned. Because an additional zero code is
+ * appended at the end of each region of the key and the value, each region can be treated
+ * as a C-style string. The return value should be deleted explicitly by the caller with
+ * the detele[] operator.
+ */
+ char* get(size_t* ksp, const char** vbp, size_t* vsp, bool step = false) {
+ _assert_(ksp && vbp && vsp);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : kbuf_(NULL), ksiz_(0), vbuf_(NULL), vsiz_(0) {}
+ char* pop(size_t* ksp, const char** vbp, size_t* vsp) {
+ *ksp = ksiz_;
+ *vbp = vbuf_;
+ *vsp = vsiz_;
+ return kbuf_;
+ }
+ void clear() {
+ delete[] kbuf_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ size_t rsiz = ksiz + 1 + vsiz + 1;
+ kbuf_ = new char[rsiz];
+ std::memcpy(kbuf_, kbuf, ksiz);
+ kbuf_[ksiz] = '\0';
+ ksiz_ = ksiz;
+ vbuf_ = kbuf_ + ksiz + 1;
+ std::memcpy(vbuf_, vbuf, vsiz);
+ vbuf_[vsiz] = '\0';
+ vsiz_ = vsiz;
+ return NOP;
+ }
+ char* kbuf_;
+ size_t ksiz_;
+ char* vbuf_;
+ size_t vsiz_;
+ };
+ VisitorImpl visitor;
+ if (!accept(&visitor, false, step)) {
+ visitor.clear();
+ *ksp = 0;
+ *vbp = NULL;
+ *vsp = 0;
+ return NULL;
+ }
+ return visitor.pop(ksp, vbp, vsp);
+ }
+ /**
+ * Get a pair of the key and the value of the current record.
+ * @note Equal to the original Cursor::get method except that parameters are strings
+ * to contain the result and the return value is bool for success.
+ */
+ bool get(std::string* key, std::string* value, bool step = false) {
+ _assert_(key && value);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(std::string* key, std::string* value) :
+ key_(key), value_(value), ok_(false) {}
+ bool ok() {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ key_->clear();
+ key_->append(kbuf, ksiz);
+ value_->clear();
+ value_->append(vbuf, vsiz);
+ ok_ = true;
+ return NOP;
+ }
+ std::string* key_;
+ std::string* value_;
+ bool ok_;
+ };
+ VisitorImpl visitor(key, value);
+ if (!accept(&visitor, false, step)) return false;
+ return visitor.ok();
+ }
+ /**
+ * Get a pair of the key and the value of the current record and remove it atomically.
+ * @param ksp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @param vbp the pointer to the variable into which the pointer to the value region is
+ * assigned.
+ * @param vsp the pointer to the variable into which the size of the value region is
+ * assigned.
+ * @return the pointer to the key region, or NULL on failure.
+ * @note If the cursor is invalidated, NULL is returned. Because an additional zero code is
+ * appended at the end of each region of the key and the value, each region can be treated
+ * as a C-style string. The return value should be deleted explicitly by the caller with
+ * the detele[] operator. The cursor is moved to the next record implicitly.
+ */
+ char* seize(size_t* ksp, const char** vbp, size_t* vsp) {
+ _assert_(ksp && vbp && vsp);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : kbuf_(NULL), ksiz_(0), vbuf_(NULL), vsiz_(0) {}
+ char* pop(size_t* ksp, const char** vbp, size_t* vsp) {
+ *ksp = ksiz_;
+ *vbp = vbuf_;
+ *vsp = vsiz_;
+ return kbuf_;
+ }
+ void clear() {
+ delete[] kbuf_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ size_t rsiz = ksiz + 1 + vsiz + 1;
+ kbuf_ = new char[rsiz];
+ std::memcpy(kbuf_, kbuf, ksiz);
+ kbuf_[ksiz] = '\0';
+ ksiz_ = ksiz;
+ vbuf_ = kbuf_ + ksiz + 1;
+ std::memcpy(vbuf_, vbuf, vsiz);
+ vbuf_[vsiz] = '\0';
+ vsiz_ = vsiz;
+ return REMOVE;
+ }
+ char* kbuf_;
+ size_t ksiz_;
+ char* vbuf_;
+ size_t vsiz_;
+ };
+ VisitorImpl visitor;
+ if (!accept(&visitor, true, false)) {
+ visitor.clear();
+ *ksp = 0;
+ *vbp = NULL;
+ *vsp = 0;
+ return NULL;
+ }
+ return visitor.pop(ksp, vbp, vsp);
+ }
+ /**
+ * Get a pair of the key and the value of the current record and remove it atomically.
+ * @note Equal to the original Cursor::seize method except that parameters are strings
+ * to contain the result and the return value is bool for success.
+ */
+ bool seize(std::string* key, std::string* value) {
+ _assert_(key && value);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(std::string* key, std::string* value) :
+ key_(key), value_(value), ok_(false) {}
+ bool ok() {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ key_->clear();
+ key_->append(kbuf, ksiz);
+ value_->clear();
+ value_->append(vbuf, vsiz);
+ ok_ = true;
+ return REMOVE;
+ }
+ std::string* key_;
+ std::string* value_;
+ bool ok_;
+ };
+ VisitorImpl visitor(key, value);
+ if (!accept(&visitor, true, false)) return false;
+ return visitor.ok();
+ }
+ /**
+ * Get the database object.
+ * @return the database object.
+ */
+ virtual BasicDB* db() = 0;
+ /**
+ * Get the last happened error.
+ * @return the last happened error.
+ */
+ Error error() {
+ _assert_(true);
+ return db()->error();
+ }
+ };
+ /**
+ * Error data.
+ */
+ class Error {
+ public:
+ /**
+ * Error codes.
+ */
+ enum Code {
+ SUCCESS, ///< success
+ NOIMPL, ///< not implemented
+ INVALID, ///< invalid operation
+ NOREPOS, ///< no repository
+ NOPERM, ///< no permission
+ BROKEN, ///< broken file
+ DUPREC, ///< record duplication
+ NOREC, ///< no record
+ LOGIC, ///< logical inconsistency
+ SYSTEM, ///< system error
+ MISC = 15 ///< miscellaneous error
+ };
+ /**
+ * Default constructor.
+ */
+ explicit Error() : code_(SUCCESS), message_("no error") {
+ _assert_(true);
+ }
+ /**
+ * Copy constructor.
+ * @param src the source object.
+ */
+ Error(const Error& src) : code_(src.code_), message_(src.message_) {
+ _assert_(true);
+ }
+ /**
+ * Constructor.
+ * @param code an error code.
+ * @param message a supplement message.
+ */
+ explicit Error(Code code, const char* message) : code_(code), message_(message) {
+ _assert_(message);
+ }
+ /**
+ * Destructor.
+ */
+ ~Error() {
+ _assert_(true);
+ }
+ /**
+ * Set the error information.
+ * @param code an error code.
+ * @param message a supplement message.
+ */
+ void set(Code code, const char* message) {
+ _assert_(message);
+ code_ = code;
+ message_ = message;
+ }
+ /**
+ * Get the error code.
+ * @return the error code.
+ */
+ Code code() const {
+ _assert_(true);
+ return code_;
+ }
+ /**
+ * Get the readable string of the code.
+ * @return the readable string of the code.
+ */
+ const char* name() const {
+ _assert_(true);
+ return codename(code_);
+ }
+ /**
+ * Get the supplement message.
+ * @return the supplement message.
+ */
+ const char* message() const {
+ _assert_(true);
+ return message_;
+ }
+ /**
+ * Get the readable string of an error code.
+ * @param code the error code.
+ * @return the readable string of the error code.
+ */
+ static const char* codename(Code code) {
+ _assert_(true);
+ switch (code) {
+ case SUCCESS: return "success";
+ case NOIMPL: return "not implemented";
+ case INVALID: return "invalid operation";
+ case NOREPOS: return "no repository";
+ case NOPERM: return "no permission";
+ case BROKEN: return "broken file";
+ case DUPREC: return "record duplication";
+ case NOREC: return "no record";
+ case LOGIC: return "logical inconsistency";
+ case SYSTEM: return "system error";
+ default: break;
+ }
+ return "miscellaneous error";
+ }
+ /**
+ * Assignment operator from the self type.
+ * @param right the right operand.
+ * @return the reference to itself.
+ */
+ Error& operator =(const Error& right) {
+ _assert_(true);
+ if (&right == this) return *this;
+ code_ = right.code_;
+ message_ = right.message_;
+ return *this;
+ }
+ /**
+ * Cast operator to integer.
+ * @return the error code.
+ */
+ operator int32_t() const {
+ return code_;
+ }
+ private:
+ /** The error code. */
+ Code code_;
+ /** The supplement message. */
+ const char* message_;
+ };
+ /**
+ * Interface to check progress status of long process.
+ */
+ class ProgressChecker {
+ public:
+ /**
+ * Destructor.
+ */
+ virtual ~ProgressChecker() {
+ _assert_(true);
+ }
+ /**
+ * Check the progress status.
+ * @param name the name of the process.
+ * @param message a supplement message.
+ * @param curcnt the count of the current step of the progress, or -1 if not applicable.
+ * @param allcnt the estimation count of all steps of the progress, or -1 if not applicable.
+ * @return true to continue the process, or false to stop the process.
+ */
+ virtual bool check(const char* name, const char* message,
+ int64_t curcnt, int64_t allcnt) = 0;
+ };
+ /**
+ * Interface to process the database file.
+ */
+ class FileProcessor {
+ public:
+ /**
+ * Destructor.
+ */
+ virtual ~FileProcessor() {
+ _assert_(true);
+ }
+ /**
+ * Process the database file.
+ * @param path the path of the database file.
+ * @param count the number of records. A negative value means omission.
+ * @param size the size of the available region. A negative value means omission.
+ * @return true on success, or false on failure.
+ */
+ virtual bool process(const std::string& path, int64_t count, int64_t size) = 0;
+ };
+ /**
+ * Interface to log internal information and errors.
+ */
+ class Logger {
+ public:
+ /**
+ * Event kinds.
+ */
+ enum Kind {
+ DEBUG = 1 << 0, ///< debugging
+ INFO = 1 << 1, ///< normal information
+ WARN = 1 << 2, ///< warning
+ ERROR = 1 << 3 ///< error
+ };
+ /**
+ * Destructor.
+ */
+ virtual ~Logger() {
+ _assert_(true);
+ }
+ /**
+ * Process a log message.
+ * @param file the file name of the program source code.
+ * @param line the line number of the program source code.
+ * @param func the function name of the program source code.
+ * @param kind the kind of the event. Logger::DEBUG for debugging, Logger::INFO for normal
+ * information, Logger::WARN for warning, and Logger::ERROR for fatal error.
+ * @param message the supplement message.
+ */
+ virtual void log(const char* file, int32_t line, const char* func, Kind kind,
+ const char* message) = 0;
+ };
+ /**
+ * Interface to trigger meta database operations.
+ */
+ class MetaTrigger {
+ public:
+ /**
+ * Event kinds.
+ */
+ enum Kind {
+ OPEN, ///< opening
+ CLOSE, ///< closing
+ CLEAR, ///< clearing
+ ITERATE, ///< iteration
+ SYNCHRONIZE, ///< synchronization
+ OCCUPY, ///< occupation
+ BEGINTRAN, ///< beginning transaction
+ COMMITTRAN, ///< committing transaction
+ ABORTTRAN, ///< aborting transaction
+ MISC = 15 ///< miscellaneous operation
+ };
+ /**
+ * Destructor.
+ */
+ virtual ~MetaTrigger() {
+ _assert_(true);
+ }
+ /**
+ * Trigger a meta database operation.
+ * @param kind the kind of the event. MetaTrigger::OPEN for opening, MetaTrigger::CLOSE for
+ * closing, MetaTrigger::CLEAR for clearing, MetaTrigger::ITERATE for iteration,
+ * MetaTrigger::SYNCHRONIZE for synchronization, MetaTrigger::OCCUPY for occupation,
+ * MetaTrigger::BEGINTRAN for beginning transaction, MetaTrigger::COMMITTRAN for committing
+ * transaction, MetaTrigger::ABORTTRAN for aborting transaction, and MetaTrigger::MISC for
+ * miscellaneous operations.
+ * @param message the supplement message.
+ */
+ virtual void trigger(Kind kind, const char* message) = 0;
+ };
+ /**
+ * Open modes.
+ */
+ enum OpenMode {
+ OREADER = 1 << 0, ///< open as a reader
+ OWRITER = 1 << 1, ///< open as a writer
+ OCREATE = 1 << 2, ///< writer creating
+ OTRUNCATE = 1 << 3, ///< writer truncating
+ OAUTOTRAN = 1 << 4, ///< auto transaction
+ OAUTOSYNC = 1 << 5, ///< auto synchronization
+ ONOLOCK = 1 << 6, ///< open without locking
+ OTRYLOCK = 1 << 7, ///< lock without blocking
+ ONOREPAIR = 1 << 8 ///< open without auto repair
+ };
+ /**
+ * Destructor.
+ * @note If the database is not closed, it is closed implicitly.
+ */
+ virtual ~BasicDB() {
+ _assert_(true);
+ }
+ /**
+ * Get the last happened error.
+ * @return the last happened error.
+ */
+ virtual Error error() const = 0;
+ /**
+ * Set the error information.
+ * @param file the file name of the program source code.
+ * @param line the line number of the program source code.
+ * @param func the function name of the program source code.
+ * @param code an error code.
+ * @param message a supplement message.
+ */
+ virtual void set_error(const char* file, int32_t line, const char* func,
+ Error::Code code, const char* message) = 0;
+ /**
+ * Open a database file.
+ * @param path the path of a database file.
+ * @param mode the connection mode. BasicDB::OWRITER as a writer, BasicDB::OREADER as a
+ * reader. The following may be added to the writer mode by bitwise-or: BasicDB::OCREATE,
+ * which means it creates a new database if the file does not exist, BasicDB::OTRUNCATE, which
+ * means it creates a new database regardless if the file exists, BasicDB::OAUTOTRAN, which
+ * means each updating operation is performed in implicit transaction, BasicDB::OAUTOSYNC,
+ * which means each updating operation is followed by implicit synchronization with the file
+ * system. The following may be added to both of the reader mode and the writer mode by
+ * bitwise-or: BasicDB::ONOLOCK, which means it opens the database file without file locking,
+ * BasicDB::OTRYLOCK, which means locking is performed without blocking, File::ONOREPAIR, which
+ * means the database file is not repaired implicitly even if file destruction is detected.
+ * @return true on success, or false on failure.
+ * @note Every opened database must be closed by the BasicDB::close method when it is no longer
+ * in use. It is not allowed for two or more database objects in the same process to keep
+ * their connections to the same database file at the same time.
+ */
+ virtual bool open(const std::string& path, uint32_t mode = OWRITER | OCREATE) = 0;
+ /**
+ * Close the database file.
+ * @return true on success, or false on failure.
+ */
+ virtual bool close() = 0;
+ /**
+ * Accept a visitor to multiple records at once.
+ * @param keys specifies a string vector of the keys.
+ * @param visitor a visitor object.
+ * @param writable true for writable operation, or false for read-only operation.
+ * @return true on success, or false on failure.
+ * @note The operations for specified records are performed atomically and other threads
+ * accessing the same records are blocked. To avoid deadlock, any explicit database operation
+ * must not be performed in this function.
+ */
+ virtual bool accept_bulk(const std::vector<std::string>& keys, Visitor* visitor,
+ bool writable = true) = 0;
+ /**
+ * Iterate to accept a visitor for each record.
+ * @param visitor a visitor object.
+ * @param writable true for writable operation, or false for read-only operation.
+ * @param checker a progress checker object. If it is NULL, no checking is performed.
+ * @return true on success, or false on failure.
+ * @note The whole iteration is performed atomically and other threads are blocked. To avoid
+ * deadlock, any explicit database operation must not be performed in this function.
+ */
+ virtual bool iterate(Visitor *visitor, bool writable = true,
+ ProgressChecker* checker = NULL) = 0;
+ /**
+ * Scan each record in parallel.
+ * @param visitor a visitor object.
+ * @param thnum the number of worker threads.
+ * @param checker a progress checker object. If it is NULL, no checking is performed.
+ * @return true on success, or false on failure.
+ * @note This function is for reading records and not for updating ones. The return value of
+ * the visitor is just ignored. To avoid deadlock, any explicit database operation must not
+ * be performed in this function.
+ */
+ virtual bool scan_parallel(Visitor *visitor, size_t thnum,
+ ProgressChecker* checker = NULL) = 0;
+ /**
+ * Synchronize updated contents with the file and the device.
+ * @param hard true for physical synchronization with the device, or false for logical
+ * synchronization with the file system.
+ * @param proc a postprocessor object. If it is NULL, no postprocessing is performed.
+ * @param checker a progress checker object. If it is NULL, no checking is performed.
+ * @return true on success, or false on failure.
+ * @note The operation of the postprocessor is performed atomically and other threads accessing
+ * the same record are blocked. To avoid deadlock, any explicit database operation must not
+ * be performed in this function.
+ */
+ virtual bool synchronize(bool hard = false, FileProcessor* proc = NULL,
+ ProgressChecker* checker = NULL) = 0;
+ /**
+ * Occupy database by locking and do something meanwhile.
+ * @param writable true to use writer lock, or false to use reader lock.
+ * @param proc a processor object. If it is NULL, no processing is performed.
+ * @return true on success, or false on failure.
+ * @note The operation of the processor is performed atomically and other threads accessing
+ * the same record are blocked. To avoid deadlock, any explicit database operation must not
+ * be performed in this function.
+ */
+ virtual bool occupy(bool writable = true, FileProcessor* proc = NULL) = 0;
+ /**
+ * Create a copy of the database file.
+ * @param dest the path of the destination file.
+ * @param checker a progress checker object. If it is NULL, no checking is performed.
+ * @return true on success, or false on failure.
+ */
+ bool copy(const std::string& dest, ProgressChecker* checker = NULL) {
+ _assert_(true);
+ class FileProcessorImpl : public FileProcessor {
+ public:
+ explicit FileProcessorImpl(const std::string& dest, ProgressChecker* checker,
+ BasicDB* db) :
+ dest_(dest), checker_(checker), db_(db) {}
+ private:
+ bool process(const std::string& path, int64_t count, int64_t size) {
+ File::Status sbuf;
+ if (!File::status(path, &sbuf)) return false;
+ if (sbuf.isdir) {
+ if (!File::make_directory(dest_)) return false;
+ bool err = false;
+ DirStream dir;
+ if (dir.open(path)) {
+ if (checker_ && !checker_->check("copy", "beginning", 0, -1)) {
+ db_->set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ }
+ std::string name;
+ int64_t curcnt = 0;
+ while (!err && dir.read(&name)) {
+ const std::string& spath = path + File::PATHCHR + name;
+ const std::string& dpath = dest_ + File::PATHCHR + name;
+ int64_t dsiz;
+ char* dbuf = File::read_file(spath, &dsiz);
+ if (dbuf) {
+ if (!File::write_file(dpath, dbuf, dsiz)) err = true;
+ delete[] dbuf;
+ } else {
+ err = true;
+ }
+ curcnt++;
+ if (checker_ && !checker_->check("copy", "processing", curcnt, -1)) {
+ db_->set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ break;
+ }
+ }
+ if (checker_ && !checker_->check("copy", "ending", -1, -1)) {
+ db_->set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ }
+ if (!dir.close()) err = true;
+ } else {
+ err = true;
+ }
+ return !err;
+ }
+ std::ofstream ofs;
+ ofs.open(dest_.c_str(),
+ std::ios_base::out | std::ios_base::binary | std::ios_base::trunc);
+ if (!ofs) return false;
+ bool err = false;
+ std::ifstream ifs;
+ ifs.open(path.c_str(), std::ios_base::in | std::ios_base::binary);
+ if (checker_ && !checker_->check("copy", "beginning", 0, size)) {
+ db_->set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ }
+ if (ifs) {
+ char buf[IOBUFSIZ];
+ int64_t curcnt = 0;
+ while (!err && !ifs.eof()) {
+ size_t n = ifs.read(buf, sizeof(buf)).gcount();
+ if (n > 0) {
+ ofs.write(buf, n);
+ if (!ofs) {
+ err = true;
+ break;
+ }
+ }
+ curcnt += n;
+ if (checker_ && !checker_->check("copy", "processing", curcnt, size)) {
+ db_->set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ break;
+ }
+ }
+ ifs.close();
+ if (ifs.bad()) err = true;
+ } else {
+ err = true;
+ }
+ if (checker_ && !checker_->check("copy", "ending", -1, size)) {
+ db_->set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ }
+ ofs.close();
+ if (!ofs) err = true;
+ return !err;
+ }
+ const std::string& dest_;
+ ProgressChecker* checker_;
+ BasicDB* db_;
+ };
+ FileProcessorImpl proc(dest, checker, this);
+ return synchronize(false, &proc, checker);
+ }
+ /**
+ * Begin transaction.
+ * @param hard true for physical synchronization with the device, or false for logical
+ * synchronization with the file system.
+ * @return true on success, or false on failure.
+ */
+ virtual bool begin_transaction(bool hard = false) = 0;
+ /**
+ * Try to begin transaction.
+ * @param hard true for physical synchronization with the device, or false for logical
+ * synchronization with the file system.
+ * @return true on success, or false on failure.
+ */
+ virtual bool begin_transaction_try(bool hard = false) = 0;
+ /**
+ * End transaction.
+ * @param commit true to commit the transaction, or false to abort the transaction.
+ * @return true on success, or false on failure.
+ */
+ virtual bool end_transaction(bool commit = true) = 0;
+ /**
+ * Get the size of the database file.
+ * @return the size of the database file in bytes, or -1 on failure.
+ */
+ virtual int64_t size() = 0;
+ /**
+ * Get the path of the database file.
+ * @return the path of the database file, or an empty string on failure.
+ */
+ virtual std::string path() = 0;
+ /**
+ * Get the miscellaneous status information.
+ * @param strmap a string map to contain the result.
+ * @return true on success, or false on failure.
+ */
+ virtual bool status(std::map<std::string, std::string>* strmap) = 0;
+ /**
+ * Set the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, a new record is created. If the corresponding
+ * record exists, the value is overwritten.
+ */
+ bool set(const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(const char* vbuf, size_t vsiz) : vbuf_(vbuf), vsiz_(vsiz) {}
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ const char* vbuf_;
+ size_t vsiz_;
+ };
+ VisitorImpl visitor(vbuf, vsiz);
+ if (!accept(kbuf, ksiz, &visitor, true)) return false;
+ return true;
+ }
+ /**
+ * Set the value of a record.
+ * @note Equal to the original DB::set method except that the parameters are std::string.
+ */
+ bool set(const std::string& key, const std::string& value) {
+ _assert_(true);
+ return set(key.c_str(), key.size(), value.c_str(), value.size());
+ }
+ /**
+ * Add a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, a new record is created. If the corresponding
+ * record exists, the record is not modified and false is returned.
+ */
+ bool add(const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(const char* vbuf, size_t vsiz) :
+ vbuf_(vbuf), vsiz_(vsiz), ok_(false) {}
+ bool ok() const {
+ return ok_;
+ }
+ private:
+ const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
+ ok_ = true;
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ const char* vbuf_;
+ size_t vsiz_;
+ bool ok_;
+ };
+ VisitorImpl visitor(vbuf, vsiz);
+ if (!accept(kbuf, ksiz, &visitor, true)) return false;
+ if (!visitor.ok()) {
+ set_error(_KCCODELINE_, Error::DUPREC, "record duplication");
+ return false;
+ }
+ return true;
+ }
+ /**
+ * Set the value of a record.
+ * @note Equal to the original DB::add method except that the parameters are std::string.
+ */
+ bool add(const std::string& key, const std::string& value) {
+ _assert_(true);
+ return add(key.c_str(), key.size(), value.c_str(), value.size());
+ }
+ /**
+ * Replace the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, no new record is created and false is returned.
+ * If the corresponding record exists, the value is modified.
+ */
+ bool replace(const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(const char* vbuf, size_t vsiz) :
+ vbuf_(vbuf), vsiz_(vsiz), ok_(false) {}
+ bool ok() const {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ ok_ = true;
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ const char* vbuf_;
+ size_t vsiz_;
+ bool ok_;
+ };
+ VisitorImpl visitor(vbuf, vsiz);
+ if (!accept(kbuf, ksiz, &visitor, true)) return false;
+ if (!visitor.ok()) {
+ set_error(_KCCODELINE_, Error::NOREC, "no record");
+ return false;
+ }
+ return true;
+ }
+ /**
+ * Replace the value of a record.
+ * @note Equal to the original DB::replace method except that the parameters are std::string.
+ */
+ bool replace(const std::string& key, const std::string& value) {
+ _assert_(true);
+ return replace(key.c_str(), key.size(), value.c_str(), value.size());
+ }
+ /**
+ * Append the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the value region.
+ * @param vsiz the size of the value region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, a new record is created. If the corresponding
+ * record exists, the given value is appended at the end of the existing value.
+ */
+ bool append(const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(const char* vbuf, size_t vsiz) :
+ vbuf_(vbuf), vsiz_(vsiz), nbuf_(NULL) {}
+ ~VisitorImpl() {
+ if (nbuf_) delete[] nbuf_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ size_t nsiz = vsiz + vsiz_;
+ nbuf_ = new char[nsiz];
+ std::memcpy(nbuf_, vbuf, vsiz);
+ std::memcpy(nbuf_ + vsiz, vbuf_, vsiz_);
+ *sp = nsiz;
+ return nbuf_;
+ }
+ const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ const char* vbuf_;
+ size_t vsiz_;
+ char* nbuf_;
+ };
+ VisitorImpl visitor(vbuf, vsiz);
+ if (!accept(kbuf, ksiz, &visitor, true)) return false;
+ return true;
+ }
+ /**
+ * Set the value of a record.
+ * @note Equal to the original DB::append method except that the parameters are std::string.
+ */
+ bool append(const std::string& key, const std::string& value) {
+ _assert_(true);
+ return append(key.c_str(), key.size(), value.c_str(), value.size());
+ }
+ /**
+ * Add a number to the numeric value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param num the additional number.
+ * @param orig the origin number if no record corresponds to the key. If it is INT64MIN and
+ * no record corresponds, this function fails. If it is INT64MAX, the value is set as the
+ * additional number regardless of the current value.
+ * @return the result value, or kyotocabinet::INT64MIN on failure.
+ * @note The value is serialized as an 8-byte binary integer in big-endian order, not a decimal
+ * string. If existing value is not 8-byte, this function fails.
+ */
+ int64_t increment(const char* kbuf, size_t ksiz, int64_t num, int64_t orig = 0) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(int64_t num, int64_t orig) : num_(num), orig_(orig), big_(0) {}
+ int64_t num() {
+ return num_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ if (vsiz != sizeof(num_)) {
+ num_ = INT64MIN;
+ return NOP;
+ }
+ int64_t onum;
+ if (orig_ == INT64MAX) {
+ onum = 0;
+ } else {
+ std::memcpy(&onum, vbuf, vsiz);
+ onum = ntoh64(onum);
+ if (num_ == 0) {
+ num_ = onum;
+ return NOP;
+ }
+ }
+ num_ += onum;
+ big_ = hton64(num_);
+ *sp = sizeof(big_);
+ return (const char*)&big_;
+ }
+ const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
+ if (orig_ == INT64MIN) {
+ num_ = INT64MIN;
+ return NOP;
+ }
+ if (orig_ != INT64MAX) num_ += orig_;
+ big_ = hton64(num_);
+ *sp = sizeof(big_);
+ return (const char*)&big_;
+ }
+ int64_t num_;
+ int64_t orig_;
+ uint64_t big_;
+ };
+ VisitorImpl visitor(num, orig);
+ if (!accept(kbuf, ksiz, &visitor, num != 0 || orig != INT64MIN)) return INT64MIN;
+ num = visitor.num();
+ if (num == INT64MIN) {
+ set_error(_KCCODELINE_, Error::LOGIC, "logical inconsistency");
+ return num;
+ }
+ return num;
+ }
+ /**
+ * Add a number to the numeric value of a record.
+ * @note Equal to the original DB::increment method except that the parameter is std::string.
+ */
+ int64_t increment(const std::string& key, int64_t num, int64_t orig = 0) {
+ _assert_(true);
+ return increment(key.c_str(), key.size(), num, orig);
+ }
+ /**
+ * Add a number to the numeric double value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param num the additional number.
+ * @param orig the origin number if no record corresponds to the key. If it is negative
+ * infinity and no record corresponds, this function fails. If it is positive infinity, the
+ * value is set as the additional number regardless of the current value.
+ * @return the result value, or Not-a-number on failure.
+ * @note The value is serialized as an 16-byte binary fixed-point number in big-endian order,
+ * not a decimal string. If existing value is not 16-byte, this function fails.
+ */
+ double increment_double(const char* kbuf, size_t ksiz, double num, double orig = 0) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(double num, double orig) :
+ DECUNIT(1000000000000000LL), num_(num), orig_(orig), buf_() {}
+ double num() {
+ return num_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ if (vsiz != sizeof(buf_)) {
+ num_ = nan();
+ return NOP;
+ }
+ int64_t linteg, lfract;
+ if (chkinf(orig_) && orig_ >= 0) {
+ linteg = 0;
+ lfract = 0;
+ } else {
+ std::memcpy(&linteg, vbuf, sizeof(linteg));
+ linteg = ntoh64(linteg);
+ std::memcpy(&lfract, vbuf + sizeof(linteg), sizeof(lfract));
+ lfract = ntoh64(lfract);
+ }
+ if (lfract == INT64MIN && linteg == INT64MIN) {
+ num_ = nan();
+ return NOP;
+ } else if (linteg == INT64MAX) {
+ num_ = HUGE_VAL;
+ return NOP;
+ } else if (linteg == INT64MIN) {
+ num_ = -HUGE_VAL;
+ return NOP;
+ }
+ if (num_ == 0.0 && !(chkinf(orig_) && orig_ >= 0)) {
+ num_ = linteg + (double)lfract / DECUNIT;
+ return NOP;
+ }
+ long double dinteg;
+ long double dfract = std::modfl(num_, &dinteg);
+ if (chknan(dinteg)) {
+ linteg = INT64MIN;
+ lfract = INT64MIN;
+ num_ = nan();
+ } else if (chkinf(dinteg)) {
+ linteg = dinteg > 0 ? INT64MAX : INT64MIN;
+ lfract = 0;
+ num_ = dinteg;
+ } else {
+ linteg += (int64_t)dinteg;
+ lfract += (int64_t)(dfract * DECUNIT);
+ if (lfract >= DECUNIT) {
+ linteg += 1;
+ lfract -= DECUNIT;
+ }
+ num_ = linteg + (double)lfract / DECUNIT;
+ }
+ linteg = hton64(linteg);
+ std::memcpy(buf_, &linteg, sizeof(linteg));
+ lfract = hton64(lfract);
+ std::memcpy(buf_ + sizeof(linteg), &lfract, sizeof(lfract));
+ *sp = sizeof(buf_);
+ return buf_;
+ }
+ const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
+ if (chknan(orig_) || (chkinf(orig_) && orig_ < 0)) {
+ num_ = nan();
+ return NOP;
+ }
+ if (!chkinf(orig_)) num_ += orig_;
+ long double dinteg;
+ long double dfract = std::modfl(num_, &dinteg);
+ int64_t linteg, lfract;
+ if (chknan(dinteg)) {
+ linteg = INT64MIN;
+ lfract = INT64MIN;
+ } else if (chkinf(dinteg)) {
+ linteg = dinteg > 0 ? INT64MAX : INT64MIN;
+ lfract = 0;
+ } else {
+ linteg = (int64_t)dinteg;
+ lfract = (int64_t)(dfract * DECUNIT);
+ }
+ linteg = hton64(linteg);
+ std::memcpy(buf_, &linteg, sizeof(linteg));
+ lfract = hton64(lfract);
+ std::memcpy(buf_ + sizeof(linteg), &lfract, sizeof(lfract));
+ *sp = sizeof(buf_);
+ return buf_;
+ }
+ const int64_t DECUNIT;
+ double num_;
+ double orig_;
+ char buf_[sizeof(int64_t)*2];
+ };
+ VisitorImpl visitor(num, orig);
+ if (!accept(kbuf, ksiz, &visitor, true)) return nan();
+ num = visitor.num();
+ if (chknan(num)) {
+ set_error(_KCCODELINE_, Error::LOGIC, "logical inconsistency");
+ return nan();
+ }
+ return num;
+ }
+ /**
+ * Add a number to the numeric double value of a record.
+ * @note Equal to the original DB::increment_double method except that the parameter is
+ * std::string.
+ */
+ double increment_double(const std::string& key, double num, double orig) {
+ _assert_(true);
+ return increment_double(key.c_str(), key.size(), num, orig);
+ }
+ /**
+ * Perform compare-and-swap.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param ovbuf the pointer to the old value region. NULL means that no record corresponds.
+ * @param ovsiz the size of the old value region.
+ * @param nvbuf the pointer to the new value region. NULL means that the record is removed.
+ * @param nvsiz the size of new old value region.
+ * @return true on success, or false on failure.
+ */
+ bool cas(const char* kbuf, size_t ksiz,
+ const char* ovbuf, size_t ovsiz, const char* nvbuf, size_t nvsiz) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(const char* ovbuf, size_t ovsiz, const char* nvbuf, size_t nvsiz) :
+ ovbuf_(ovbuf), ovsiz_(ovsiz), nvbuf_(nvbuf), nvsiz_(nvsiz), ok_(false) {}
+ bool ok() const {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ if (!ovbuf_ || vsiz != ovsiz_ || std::memcmp(vbuf, ovbuf_, vsiz)) return NOP;
+ ok_ = true;
+ if (!nvbuf_) return REMOVE;
+ *sp = nvsiz_;
+ return nvbuf_;
+ }
+ const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
+ if (ovbuf_) return NOP;
+ ok_ = true;
+ if (!nvbuf_) return NOP;
+ *sp = nvsiz_;
+ return nvbuf_;
+ }
+ const char* ovbuf_;
+ size_t ovsiz_;
+ const char* nvbuf_;
+ size_t nvsiz_;
+ bool ok_;
+ };
+ VisitorImpl visitor(ovbuf, ovsiz, nvbuf, nvsiz);
+ if (!accept(kbuf, ksiz, &visitor, true)) return false;
+ if (!visitor.ok()) {
+ set_error(_KCCODELINE_, Error::LOGIC, "status conflict");
+ return false;
+ }
+ return true;
+ }
+ /**
+ * Perform compare-and-swap.
+ * @note Equal to the original DB::cas method except that the parameters are std::string.
+ */
+ bool cas(const std::string& key,
+ const std::string& ovalue, const std::string& nvalue) {
+ _assert_(true);
+ return cas(key.c_str(), key.size(),
+ ovalue.c_str(), ovalue.size(), nvalue.c_str(), nvalue.size());
+ }
+ /**
+ * Remove a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @return true on success, or false on failure.
+ * @note If no record corresponds to the key, false is returned.
+ */
+ bool remove(const char* kbuf, size_t ksiz) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : ok_(false) {}
+ bool ok() const {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ ok_ = true;
+ return REMOVE;
+ }
+ bool ok_;
+ };
+ VisitorImpl visitor;
+ if (!accept(kbuf, ksiz, &visitor, true)) return false;
+ if (!visitor.ok()) {
+ set_error(_KCCODELINE_, Error::NOREC, "no record");
+ return false;
+ }
+ return true;
+ }
+ /**
+ * Remove a record.
+ * @note Equal to the original DB::remove method except that the parameter is std::string.
+ */
+ bool remove(const std::string& key) {
+ _assert_(true);
+ return remove(key.c_str(), key.size());
+ }
+ /**
+ * Retrieve the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @return the pointer to the value region of the corresponding record, or NULL on failure.
+ * @note If no record corresponds to the key, NULL is returned. Because an additional zero
+ * code is appended at the end of the region of the return value, the return value can be
+ * treated as a C-style string. Because the region of the return value is allocated with the
+ * the new[] operator, it should be released with the delete[] operator when it is no longer
+ * in use.
+ */
+ char* get(const char* kbuf, size_t ksiz, size_t* sp) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && sp);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : vbuf_(NULL), vsiz_(0) {}
+ char* pop(size_t* sp) {
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ vbuf_ = new char[vsiz+1];
+ std::memcpy(vbuf_, vbuf, vsiz);
+ vbuf_[vsiz] = '\0';
+ vsiz_ = vsiz;
+ return NOP;
+ }
+ char* vbuf_;
+ size_t vsiz_;
+ };
+ VisitorImpl visitor;
+ if (!accept(kbuf, ksiz, &visitor, false)) {
+ *sp = 0;
+ return NULL;
+ }
+ size_t vsiz;
+ char* vbuf = visitor.pop(&vsiz);
+ if (!vbuf) {
+ set_error(_KCCODELINE_, Error::NOREC, "no record");
+ *sp = 0;
+ return NULL;
+ }
+ *sp = vsiz;
+ return vbuf;
+ }
+ /**
+ * Retrieve the value of a record.
+ * @note Equal to the original DB::get method except that the first parameters is the key
+ * string and the second parameter is a string to contain the result and the return value is
+ * bool for success.
+ */
+ bool get(const std::string& key, std::string* value) {
+ _assert_(value);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(std::string* value) : value_(value), ok_(false) {}
+ bool ok() {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ value_->clear();
+ value_->append(vbuf, vsiz);
+ ok_ = true;
+ return NOP;
+ }
+ std::string* value_;
+ bool ok_;
+ };
+ VisitorImpl visitor(value);
+ if (!accept(key.data(), key.size(), &visitor, false)) return false;
+ if (!visitor.ok()) {
+ set_error(_KCCODELINE_, Error::NOREC, "no record");
+ return false;
+ }
+ return true;
+ }
+ /**
+ * Retrieve the value of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param vbuf the pointer to the buffer into which the value of the corresponding record is
+ * written.
+ * @param max the size of the buffer.
+ * @return the size of the value, or -1 on failure.
+ */
+ int32_t get(const char* kbuf, size_t ksiz, char* vbuf, size_t max) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && vbuf);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(char* vbuf, size_t max) : vbuf_(vbuf), max_(max), vsiz_(-1) {}
+ int32_t vsiz() {
+ return vsiz_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ vsiz_ = vsiz;
+ size_t max = vsiz < max_ ? vsiz : max_;
+ std::memcpy(vbuf_, vbuf, max);
+ return NOP;
+ }
+ char* vbuf_;
+ size_t max_;
+ int32_t vsiz_;
+ };
+ VisitorImpl visitor(vbuf, max);
+ if (!accept(kbuf, ksiz, &visitor, false)) return -1;
+ int32_t vsiz = visitor.vsiz();
+ if (vsiz < 0) {
+ set_error(_KCCODELINE_, Error::NOREC, "no record");
+ return -1;
+ }
+ return vsiz;
+ }
+ /**
+ * Check the existence of a record.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @return the size of the value, or -1 on failure.
+ */
+ int32_t check(const char* kbuf, size_t ksiz) {
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : vsiz_(-1) {}
+ int32_t vsiz() {
+ return vsiz_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ vsiz_ = vsiz;
+ return NOP;
+ }
+ size_t vsiz_;
+ };
+ VisitorImpl visitor;
+ if (!accept(kbuf, ksiz, &visitor, false)) return -1;
+ int32_t vsiz = visitor.vsiz();
+ if (vsiz < 0) {
+ set_error(_KCCODELINE_, Error::NOREC, "no record");
+ return -1;
+ }
+ return vsiz;
+ }
+ /**
+ * Check the existence of a record.
+ * @note Equal to the original DB::check method except that the parameter is std::string.
+ */
+ int32_t check(const std::string& key) {
+ return check(key.data(), key.size());
+ }
+ /**
+ * Retrieve the value of a record and remove it atomically.
+ * @param kbuf the pointer to the key region.
+ * @param ksiz the size of the key region.
+ * @param sp the pointer to the variable into which the size of the region of the return
+ * value is assigned.
+ * @return the pointer to the value region of the corresponding record, or NULL on failure.
+ * @note If no record corresponds to the key, NULL is returned. Because an additional zero
+ * code is appended at the end of the region of the return value, the return value can be
+ * treated as a C-style string. Because the region of the return value is allocated with the
+ * the new[] operator, it should be released with the delete[] operator when it is no longer
+ * in use.
+ */
+ char* seize(const char* kbuf, size_t ksiz, size_t* sp) {
+ _assert_(kbuf && ksiz <= MEMMAXSIZ && sp);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : vbuf_(NULL), vsiz_(0) {}
+ char* pop(size_t* sp) {
+ *sp = vsiz_;
+ return vbuf_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ vbuf_ = new char[vsiz+1];
+ std::memcpy(vbuf_, vbuf, vsiz);
+ vbuf_[vsiz] = '\0';
+ vsiz_ = vsiz;
+ return REMOVE;
+ }
+ char* vbuf_;
+ size_t vsiz_;
+ };
+ VisitorImpl visitor;
+ if (!accept(kbuf, ksiz, &visitor, true)) {
+ *sp = 0;
+ return NULL;
+ }
+ size_t vsiz;
+ char* vbuf = visitor.pop(&vsiz);
+ if (!vbuf) {
+ set_error(_KCCODELINE_, Error::NOREC, "no record");
+ *sp = 0;
+ return NULL;
+ }
+ *sp = vsiz;
+ return vbuf;
+ }
+ /**
+ * Retrieve the value of a record and remove it atomically.
+ * @note Equal to the original DB::seize method except that the first parameters is the key
+ * string and the second parameter is a string to contain the result and the return value is
+ * bool for success.
+ */
+ bool seize(const std::string& key, std::string* value) {
+ _assert_(value);
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(std::string* value) : value_(value), ok_(false) {}
+ bool ok() {
+ return ok_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ value_->clear();
+ value_->append(vbuf, vsiz);
+ ok_ = true;
+ return REMOVE;
+ }
+ std::string* value_;
+ bool ok_;
+ };
+ VisitorImpl visitor(value);
+ if (!accept(key.data(), key.size(), &visitor, true)) return false;
+ if (!visitor.ok()) {
+ set_error(_KCCODELINE_, Error::NOREC, "no record");
+ return false;
+ }
+ return true;
+ }
+ /**
+ * Store records at once.
+ * @param recs the records to store.
+ * @param atomic true to perform all operations atomically, or false for non-atomic operations.
+ * @return the number of stored records, or -1 on failure.
+ */
+ int64_t set_bulk(const std::map<std::string, std::string>& recs, bool atomic = true) {
+ _assert_(true);
+ if (atomic) {
+ std::vector<std::string> keys;
+ keys.reserve(recs.size());
+ std::map<std::string, std::string>::const_iterator rit = recs.begin();
+ std::map<std::string, std::string>::const_iterator ritend = recs.end();
+ while (rit != ritend) {
+ keys.push_back(rit->first);
+ ++rit;
+ }
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(const std::map<std::string, std::string>& recs) : recs_(recs) {}
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ std::map<std::string, std::string>::const_iterator rit =
+ recs_.find(std::string(kbuf, ksiz));
+ if (rit == recs_.end()) return NOP;
+ *sp = rit->second.size();
+ return rit->second.data();
+ }
+ const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) {
+ std::map<std::string, std::string>::const_iterator rit =
+ recs_.find(std::string(kbuf, ksiz));
+ if (rit == recs_.end()) return NOP;
+ *sp = rit->second.size();
+ return rit->second.data();
+ }
+ const std::map<std::string, std::string>& recs_;
+ };
+ VisitorImpl visitor(recs);
+ if (!accept_bulk(keys, &visitor, true)) return -1;
+ return keys.size();
+ }
+ std::map<std::string, std::string>::const_iterator rit = recs.begin();
+ std::map<std::string, std::string>::const_iterator ritend = recs.end();
+ while (rit != ritend) {
+ if (!set(rit->first.data(), rit->first.size(), rit->second.data(), rit->second.size()))
+ return -1;
+ ++rit;
+ }
+ return recs.size();
+ }
+ /**
+ * Remove records at once.
+ * @param keys the keys of the records to remove.
+ * @param atomic true to perform all operations atomically, or false for non-atomic operations.
+ * @return the number of removed records, or -1 on failure.
+ */
+ int64_t remove_bulk(const std::vector<std::string>& keys, bool atomic = true) {
+ _assert_(true);
+ if (atomic) {
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl() : cnt_(0) {}
+ int64_t cnt() const {
+ return cnt_;
+ }
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ cnt_++;
+ return REMOVE;
+ }
+ int64_t cnt_;
+ };
+ VisitorImpl visitor;
+ if (!accept_bulk(keys, &visitor, true)) return -1;
+ return visitor.cnt();
+ }
+ int64_t cnt = 0;
+ std::vector<std::string>::const_iterator kit = keys.begin();
+ std::vector<std::string>::const_iterator kitend = keys.end();
+ while (kit != kitend) {
+ if (remove(kit->data(), kit->size())) {
+ cnt++;
+ } else if (error() != Error::NOREC) {
+ return -1;
+ }
+ ++kit;
+ }
+ return cnt;
+ }
+ /**
+ * Retrieve records at once.
+ * @param keys the keys of the records to retrieve.
+ * @param recs a string map to contain the retrieved records.
+ * @param atomic true to perform all operations atomically, or false for non-atomic operations.
+ * @return the number of retrieved records, or -1 on failure.
+ */
+ int64_t get_bulk(const std::vector<std::string>& keys,
+ std::map<std::string, std::string>* recs, bool atomic = true) {
+ _assert_(recs);
+ if (atomic) {
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(std::map<std::string, std::string>* recs) : recs_(recs) {}
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ (*recs_)[std::string(kbuf, ksiz)] = std::string(vbuf, vsiz);
+ return NOP;
+ }
+ std::map<std::string, std::string>* recs_;
+ };
+ VisitorImpl visitor(recs);
+ if (!accept_bulk(keys, &visitor, false)) return -1;
+ return recs->size();
+ }
+ std::vector<std::string>::const_iterator kit = keys.begin();
+ std::vector<std::string>::const_iterator kitend = keys.end();
+ while (kit != kitend) {
+ size_t vsiz;
+ const char* vbuf = get(kit->data(), kit->size(), &vsiz);
+ if (vbuf) {
+ (*recs)[*kit] = std::string(vbuf, vsiz);
+ delete[] vbuf;
+ } else if (error() != Error::NOREC) {
+ return -1;
+ }
+ ++kit;
+ }
+ return recs->size();
+ }
+ /**
+ * Dump records into a data stream.
+ * @param dest the destination stream.
+ * @param checker a progress checker object. If it is NULL, no checking is performed.
+ * @return true on success, or false on failure.
+ */
+ bool dump_snapshot(std::ostream* dest, ProgressChecker* checker = NULL) {
+ _assert_(dest);
+ if (dest->fail()) {
+ set_error(_KCCODELINE_, Error::INVALID, "invalid stream");
+ return false;
+ }
+ class VisitorImpl : public Visitor {
+ public:
+ explicit VisitorImpl(std::ostream* dest) : dest_(dest), stack_() {}
+ private:
+ const char* visit_full(const char* kbuf, size_t ksiz,
+ const char* vbuf, size_t vsiz, size_t* sp) {
+ char* wp = stack_;
+ *(wp++) = 0x00;
+ wp += writevarnum(wp, ksiz);
+ wp += writevarnum(wp, vsiz);
+ dest_->write(stack_, wp - stack_);
+ dest_->write(kbuf, ksiz);
+ dest_->write(vbuf, vsiz);
+ return NOP;
+ }
+ std::ostream* dest_;
+ char stack_[NUMBUFSIZ*2];
+ };
+ VisitorImpl visitor(dest);
+ bool err = false;
+ dest->write(KCDBSSMAGICDATA, sizeof(KCDBSSMAGICDATA));
+ if (iterate(&visitor, false, checker)) {
+ unsigned char c = 0xff;
+ dest->write((char*)&c, 1);
+ if (dest->fail()) {
+ set_error(_KCCODELINE_, Error::SYSTEM, "stream output error");
+ err = true;
+ }
+ } else {
+ err = true;
+ }
+ return !err;
+ }
+ /**
+ * Dump records into a file.
+ * @param dest the path of the destination file.
+ * @param checker a progress checker object. If it is NULL, no checking is performed.
+ * @return true on success, or false on failure.
+ */
+ bool dump_snapshot(const std::string& dest, ProgressChecker* checker = NULL) {
+ _assert_(true);
+ std::ofstream ofs;
+ ofs.open(dest.c_str(), std::ios_base::out | std::ios_base::binary | std::ios_base::trunc);
+ if (!ofs) {
+ set_error(_KCCODELINE_, Error::NOREPOS, "open failed");
+ return false;
+ }
+ bool err = false;
+ if (!dump_snapshot(&ofs, checker)) err = true;
+ ofs.close();
+ if (!ofs) {
+ set_error(_KCCODELINE_, Error::SYSTEM, "close failed");
+ err = true;
+ }
+ return !err;
+ }
+ /**
+ * Load records from a data stream.
+ * @param src the source stream.
+ * @param checker a progress checker object. If it is NULL, no checking is performed.
+ * @return true on success, or false on failure.
+ */
+ bool load_snapshot(std::istream* src, ProgressChecker* checker = NULL) {
+ _assert_(src);
+ if (src->fail()) {
+ set_error(_KCCODELINE_, Error::INVALID, "invalid stream");
+ return false;
+ }
+ char buf[IOBUFSIZ];
+ src->read(buf, sizeof(KCDBSSMAGICDATA));
+ if (src->fail()) {
+ set_error(_KCCODELINE_, Error::SYSTEM, "stream input error");
+ return false;
+ }
+ if (std::memcmp(buf, KCDBSSMAGICDATA, sizeof(KCDBSSMAGICDATA))) {
+ set_error(_KCCODELINE_, Error::INVALID, "invalid magic data of input stream");
+ return false;
+ }
+ bool err = false;
+ if (checker && !checker->check("load_snapshot", "beginning", 0, -1)) {
+ set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ }
+ int64_t curcnt = 0;
+ while (!err) {
+ int32_t c = src->get();
+ if (src->fail()) {
+ set_error(_KCCODELINE_, Error::SYSTEM, "stream input error");
+ err = true;
+ break;
+ }
+ if (c == 0xff) break;
+ if (c == 0x00) {
+ size_t ksiz = 0;
+ do {
+ c = src->get();
+ ksiz = (ksiz << 7) + (c & 0x7f);
+ } while (c >= 0x80);
+ size_t vsiz = 0;
+ do {
+ c = src->get();
+ vsiz = (vsiz << 7) + (c & 0x7f);
+ } while (c >= 0x80);
+ size_t rsiz = ksiz + vsiz;
+ char* rbuf = rsiz > sizeof(buf) ? new char[rsiz] : buf;
+ src->read(rbuf, ksiz + vsiz);
+ if (src->fail()) {
+ set_error(_KCCODELINE_, Error::SYSTEM, "stream input error");
+ err = true;
+ if (rbuf != buf) delete[] rbuf;
+ break;
+ }
+ if (!set(rbuf, ksiz, rbuf + ksiz, vsiz)) {
+ err = true;
+ if (rbuf != buf) delete[] rbuf;
+ break;
+ }
+ if (rbuf != buf) delete[] rbuf;
+ } else {
+ set_error(_KCCODELINE_, Error::INVALID, "invalid magic data of input stream");
+ err = true;
+ break;
+ }
+ curcnt++;
+ if (checker && !checker->check("load_snapshot", "processing", curcnt, -1)) {
+ set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ break;
+ }
+ }
+ if (checker && !checker->check("load_snapshot", "ending", -1, -1)) {
+ set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
+ err = true;
+ }
+ return !err;
+ }
+ /**
+ * Load records from a file.
+ * @param src the path of the source file.
+ * @param checker a progress checker object. If it is NULL, no checking is performed.
+ * @return true on success, or false on failure.
+ */
+ bool load_snapshot(const std::string& src, ProgressChecker* checker = NULL) {
+ _assert_(true);
+ std::ifstream ifs;
+ ifs.open(src.c_str(), std::ios_base::in | std::ios_base::binary);
+ if (!ifs) {
+ set_error(_KCCODELINE_, Error::NOREPOS, "open failed");
+ return false;
+ }
+ bool err = false;
+ if (!load_snapshot(&ifs, checker)) err = true;
+ ifs.close();
+ if (ifs.bad()) {
+ set_error(_KCCODELINE_, Error::SYSTEM, "close failed");
+ return false;
+ }
+ return !err;
+ }
+ /**
+ * Create a cursor object.
+ * @return the return value is the created cursor object.
+ * @note Because the object of the return value is allocated by the constructor, it should be
+ * released with the delete operator when it is no longer in use.
+ */
+ virtual Cursor* cursor() = 0;
+ /**
+ * Write a log message.
+ * @param file the file name of the program source code.
+ * @param line the line number of the program source code.
+ * @param func the function name of the program source code.
+ * @param kind the kind of the event. Logger::DEBUG for debugging, Logger::INFO for normal
+ * information, Logger::WARN for warning, and Logger::ERROR for fatal error.
+ * @param message the supplement message.
+ */
+ virtual void log(const char* file, int32_t line, const char* func, Logger::Kind kind,
+ const char* message) = 0;
+ /**
+ * Set the internal logger.
+ * @param logger the logger object.
+ * @param kinds kinds of logged messages by bitwise-or: Logger::DEBUG for debugging,
+ * Logger::INFO for normal information, Logger::WARN for warning, and Logger::ERROR for fatal
+ * error.
+ * @return true on success, or false on failure.
+ */
+ virtual bool tune_logger(Logger* logger, uint32_t kinds = Logger::WARN | Logger::ERROR) = 0;
+ /**
+ * Set the internal meta operation trigger.
+ * @param trigger the trigger object.
+ * @return true on success, or false on failure.
+ */
+ virtual bool tune_meta_trigger(MetaTrigger* trigger) = 0;
+ /**
+ * Get the class name of a database type.
+ * @param type the database type.
+ * @return the string of the type name.
+ */
+ static const char* typecname(uint32_t type) {
+ _assert_(true);
+ switch (type) {
+ case TYPEVOID: return "void";
+ case TYPEPHASH: return "ProtoHashDB";
+ case TYPEPTREE: return "ProtoTreeDB";
+ case TYPESTASH: return "StashDB";
+ case TYPECACHE: return "CacheDB";
+ case TYPEGRASS: return "GrassDB";
+ case TYPEHASH: return "HashDB";
+ case TYPETREE: return "TreeDB";
+ case TYPEDIR: return "DirDB";
+ case TYPEFOREST: return "ForestDB";
+ case TYPEMISC: return "misc";
+ }
+ return "unknown";
+ }
+ /**
+ * Get the description string of a database type.
+ * @param type the database type.
+ * @return the string of the type name.
+ */
+ static const char* typestring(uint32_t type) {
+ _assert_(true);
+ switch (type) {
+ case TYPEVOID: return "void";
+ case TYPEPHASH: return "prototype hash database";
+ case TYPEPTREE: return "prototype tree database";
+ case TYPESTASH: return "stash database";
+ case TYPECACHE: return "cache hash database";
+ case TYPEGRASS: return "cache tree database";
+ case TYPEHASH: return "file hash database";
+ case TYPETREE: return "file tree database";
+ case TYPEDIR: return "directory hash database";
+ case TYPEFOREST: return "directory tree database";
+ case TYPEMISC: return "miscellaneous database";
+ }
+ return "unknown";
+ }
+};
+
+
+} // common namespace
+
+#endif // duplication check
+
+// END OF FILE