- Added a C binding for LevelDB. May be useful as a stable ABI that can be used by programs that keep leveldb in a shared library, or for JNI API. - Replaced SQLite's readseq benchmark to a more efficient version. SQLite readseq speeds increased by about a factor of 2x from the previous version. Also updated benchmark page to reflect readseq speed up. git-svn-id: https://leveldb.googlecode.com/svn/trunk@46 62dab493-f737-651d-591e-8d6aee1b9529main
@ -0,0 +1,453 @@ | |||||
// Copyright (c) 2011 The LevelDB Authors. All rights reserved. | |||||
// Use of this source code is governed by a BSD-style license that can be | |||||
// found in the LICENSE file. See the AUTHORS file for names of contributors. | |||||
#include "leveldb/c.h" | |||||
#include <stdlib.h> | |||||
#include <unistd.h> | |||||
#include "leveldb/cache.h" | |||||
#include "leveldb/comparator.h" | |||||
#include "leveldb/db.h" | |||||
#include "leveldb/env.h" | |||||
#include "leveldb/iterator.h" | |||||
#include "leveldb/options.h" | |||||
#include "leveldb/status.h" | |||||
#include "leveldb/write_batch.h" | |||||
namespace leveldb { | |||||
extern "C" { | |||||
struct leveldb_t { DB* rep; }; | |||||
struct leveldb_iterator_t { Iterator* rep; }; | |||||
struct leveldb_writebatch_t { WriteBatch rep; }; | |||||
struct leveldb_snapshot_t { const Snapshot* rep; }; | |||||
struct leveldb_readoptions_t { ReadOptions rep; }; | |||||
struct leveldb_writeoptions_t { WriteOptions rep; }; | |||||
struct leveldb_options_t { Options rep; }; | |||||
struct leveldb_cache_t { Cache* rep; }; | |||||
struct leveldb_seqfile_t { SequentialFile* rep; }; | |||||
struct leveldb_randomfile_t { RandomAccessFile* rep; }; | |||||
struct leveldb_writablefile_t { WritableFile* rep; }; | |||||
struct leveldb_logger_t { Logger* rep; }; | |||||
struct leveldb_filelock_t { FileLock* rep; }; | |||||
struct leveldb_comparator_t : public Comparator { | |||||
void* state_; | |||||
void (*destructor_)(void*); | |||||
int (*compare_)( | |||||
void*, | |||||
const char* a, size_t alen, | |||||
const char* b, size_t blen); | |||||
const char* (*name_)(void*); | |||||
virtual ~leveldb_comparator_t() { | |||||
(*destructor_)(state_); | |||||
} | |||||
virtual int Compare(const Slice& a, const Slice& b) const { | |||||
return (*compare_)(state_, a.data(), a.size(), b.data(), b.size()); | |||||
} | |||||
virtual const char* Name() const { | |||||
return (*name_)(state_); | |||||
} | |||||
// No-ops since the C binding does not support key shortening methods. | |||||
virtual void FindShortestSeparator(std::string*, const Slice&) const { } | |||||
virtual void FindShortSuccessor(std::string* key) const { } | |||||
}; | |||||
struct leveldb_env_t { | |||||
Env* rep; | |||||
bool is_default; | |||||
}; | |||||
static bool SaveError(char** errptr, const Status& s) { | |||||
assert(errptr != NULL); | |||||
if (s.ok()) { | |||||
return false; | |||||
} else if (*errptr == NULL) { | |||||
*errptr = strdup(s.ToString().c_str()); | |||||
} else { | |||||
// TODO(sanjay): Merge with existing error? | |||||
free(*errptr); | |||||
*errptr = strdup(s.ToString().c_str()); | |||||
} | |||||
return true; | |||||
} | |||||
static char* CopyString(const std::string& str) { | |||||
char* result = reinterpret_cast<char*>(malloc(sizeof(char) * str.size())); | |||||
memcpy(result, str.data(), sizeof(char) * str.size()); | |||||
return result; | |||||
} | |||||
leveldb_t* leveldb_open( | |||||
const leveldb_options_t* options, | |||||
const char* name, | |||||
char** errptr) { | |||||
DB* db; | |||||
if (SaveError(errptr, DB::Open(options->rep, std::string(name), &db))) { | |||||
return NULL; | |||||
} | |||||
leveldb_t* result = new leveldb_t; | |||||
result->rep = db; | |||||
return result; | |||||
} | |||||
void leveldb_close(leveldb_t* db) { | |||||
delete db->rep; | |||||
delete db; | |||||
} | |||||
void leveldb_put( | |||||
leveldb_t* db, | |||||
const leveldb_writeoptions_t* options, | |||||
const char* key, size_t keylen, | |||||
const char* val, size_t vallen, | |||||
char** errptr) { | |||||
SaveError(errptr, | |||||
db->rep->Put(options->rep, Slice(key, keylen), Slice(val, vallen))); | |||||
} | |||||
void leveldb_delete( | |||||
leveldb_t* db, | |||||
const leveldb_writeoptions_t* options, | |||||
const char* key, size_t keylen, | |||||
char** errptr) { | |||||
SaveError(errptr, db->rep->Delete(options->rep, Slice(key, keylen))); | |||||
} | |||||
void leveldb_write( | |||||
leveldb_t* db, | |||||
const leveldb_writeoptions_t* options, | |||||
leveldb_writebatch_t* batch, | |||||
char** errptr) { | |||||
SaveError(errptr, db->rep->Write(options->rep, &batch->rep)); | |||||
} | |||||
char* leveldb_get( | |||||
leveldb_t* db, | |||||
const leveldb_readoptions_t* options, | |||||
const char* key, size_t keylen, | |||||
size_t* vallen, | |||||
char** errptr) { | |||||
char* result = NULL; | |||||
std::string tmp; | |||||
Status s = db->rep->Get(options->rep, Slice(key, keylen), &tmp); | |||||
if (s.ok()) { | |||||
*vallen = tmp.size(); | |||||
result = CopyString(tmp); | |||||
} else { | |||||
*vallen = 0; | |||||
if (!s.IsNotFound()) { | |||||
SaveError(errptr, s); | |||||
} | |||||
} | |||||
return result; | |||||
} | |||||
leveldb_iterator_t* leveldb_create_iterator( | |||||
leveldb_t* db, | |||||
const leveldb_readoptions_t* options) { | |||||
leveldb_iterator_t* result = new leveldb_iterator_t; | |||||
result->rep = db->rep->NewIterator(options->rep); | |||||
return result; | |||||
} | |||||
const leveldb_snapshot_t* leveldb_create_snapshot( | |||||
leveldb_t* db) { | |||||
leveldb_snapshot_t* result = new leveldb_snapshot_t; | |||||
result->rep = db->rep->GetSnapshot(); | |||||
return result; | |||||
} | |||||
void leveldb_release_snapshot( | |||||
leveldb_t* db, | |||||
const leveldb_snapshot_t* snapshot) { | |||||
db->rep->ReleaseSnapshot(snapshot->rep); | |||||
delete snapshot; | |||||
} | |||||
const char* leveldb_property_value( | |||||
leveldb_t* db, | |||||
const char* propname) { | |||||
std::string tmp; | |||||
if (db->rep->GetProperty(Slice(propname), &tmp)) { | |||||
return CopyString(tmp); | |||||
} else { | |||||
return NULL; | |||||
} | |||||
} | |||||
void leveldb_approximate_sizes( | |||||
leveldb_t* db, | |||||
int num_ranges, | |||||
const char* const* range_start_key, const size_t* range_start_key_len, | |||||
const char* const* range_limit_key, const size_t* range_limit_key_len, | |||||
uint64_t* sizes) { | |||||
Range* ranges = new Range[num_ranges]; | |||||
for (int i = 0; i < num_ranges; i++) { | |||||
ranges[i].start = Slice(range_start_key[i], range_start_key_len[i]); | |||||
ranges[i].limit = Slice(range_limit_key[i], range_limit_key_len[i]); | |||||
} | |||||
db->rep->GetApproximateSizes(ranges, num_ranges, sizes); | |||||
delete[] ranges; | |||||
} | |||||
void leveldb_destroy_db( | |||||
const leveldb_options_t* options, | |||||
const char* name, | |||||
char** errptr) { | |||||
SaveError(errptr, DestroyDB(name, options->rep)); | |||||
} | |||||
void leveldb_repair_db( | |||||
const leveldb_options_t* options, | |||||
const char* name, | |||||
char** errptr) { | |||||
SaveError(errptr, RepairDB(name, options->rep)); | |||||
} | |||||
void leveldb_iter_destroy(leveldb_iterator_t* iter) { | |||||
delete iter->rep; | |||||
delete iter; | |||||
} | |||||
unsigned char leveldb_iter_valid(const leveldb_iterator_t* iter) { | |||||
return iter->rep->Valid(); | |||||
} | |||||
void leveldb_iter_seek_to_first(leveldb_iterator_t* iter) { | |||||
iter->rep->SeekToFirst(); | |||||
} | |||||
void leveldb_iter_seek_to_last(leveldb_iterator_t* iter) { | |||||
iter->rep->SeekToLast(); | |||||
} | |||||
void leveldb_iter_seek(leveldb_iterator_t* iter, const char* k, size_t klen) { | |||||
iter->rep->Seek(Slice(k, klen)); | |||||
} | |||||
void leveldb_iter_next(leveldb_iterator_t* iter) { | |||||
iter->rep->Next(); | |||||
} | |||||
void leveldb_iter_prev(leveldb_iterator_t* iter) { | |||||
iter->rep->Prev(); | |||||
} | |||||
const char* leveldb_iter_key(const leveldb_iterator_t* iter, size_t* klen) { | |||||
Slice s = iter->rep->key(); | |||||
*klen = s.size(); | |||||
return s.data(); | |||||
} | |||||
const char* leveldb_iter_value(const leveldb_iterator_t* iter, size_t* vlen) { | |||||
Slice s = iter->rep->value(); | |||||
*vlen = s.size(); | |||||
return s.data(); | |||||
} | |||||
void leveldb_iter_get_error(const leveldb_iterator_t* iter, char** errptr) { | |||||
SaveError(errptr, iter->rep->status()); | |||||
} | |||||
leveldb_writebatch_t* leveldb_writebatch_create() { | |||||
return new leveldb_writebatch_t; | |||||
} | |||||
void leveldb_writebatch_destroy(leveldb_writebatch_t* b) { | |||||
delete b; | |||||
} | |||||
void leveldb_writebatch_clear(leveldb_writebatch_t* b) { | |||||
b->rep.Clear(); | |||||
} | |||||
void leveldb_writebatch_put( | |||||
leveldb_writebatch_t* b, | |||||
const char* key, size_t klen, | |||||
const char* val, size_t vlen) { | |||||
b->rep.Put(Slice(key, klen), Slice(val, vlen)); | |||||
} | |||||
void leveldb_writebatch_delete( | |||||
leveldb_writebatch_t* b, | |||||
const char* key, size_t klen) { | |||||
b->rep.Delete(Slice(key, klen)); | |||||
} | |||||
void leveldb_writebatch_iterate( | |||||
leveldb_writebatch_t* b, | |||||
void* state, | |||||
void (*put)(void*, const char* k, size_t klen, const char* v, size_t vlen), | |||||
void (*deleted)(void*, const char* k, size_t klen)) { | |||||
class H : public WriteBatch::Handler { | |||||
public: | |||||
void* state_; | |||||
void (*put_)(void*, const char* k, size_t klen, const char* v, size_t vlen); | |||||
void (*deleted_)(void*, const char* k, size_t klen); | |||||
virtual void Put(const Slice& key, const Slice& value) { | |||||
(*put_)(state_, key.data(), key.size(), value.data(), value.size()); | |||||
} | |||||
virtual void Delete(const Slice& key) { | |||||
(*deleted_)(state_, key.data(), key.size()); | |||||
} | |||||
}; | |||||
H handler; | |||||
handler.state_ = state; | |||||
handler.put_ = put; | |||||
handler.deleted_ = deleted; | |||||
b->rep.Iterate(&handler); | |||||
} | |||||
leveldb_options_t* leveldb_options_create() { | |||||
return new leveldb_options_t; | |||||
} | |||||
void leveldb_options_destroy(leveldb_options_t* options) { | |||||
delete options; | |||||
} | |||||
void leveldb_options_set_comparator( | |||||
leveldb_options_t* opt, | |||||
leveldb_comparator_t* cmp) { | |||||
opt->rep.comparator = cmp; | |||||
} | |||||
void leveldb_options_set_create_if_missing( | |||||
leveldb_options_t* opt, unsigned char v) { | |||||
opt->rep.create_if_missing = v; | |||||
} | |||||
void leveldb_options_set_error_if_exists( | |||||
leveldb_options_t* opt, unsigned char v) { | |||||
opt->rep.error_if_exists = v; | |||||
} | |||||
void leveldb_options_set_paranoid_checks( | |||||
leveldb_options_t* opt, unsigned char v) { | |||||
opt->rep.paranoid_checks = v; | |||||
} | |||||
void leveldb_options_set_env(leveldb_options_t* opt, leveldb_env_t* env) { | |||||
opt->rep.env = (env ? env->rep : NULL); | |||||
} | |||||
void leveldb_options_set_info_log(leveldb_options_t* opt, leveldb_logger_t* l) { | |||||
opt->rep.info_log = (l ? l->rep : NULL); | |||||
} | |||||
void leveldb_options_set_write_buffer_size(leveldb_options_t* opt, size_t s) { | |||||
opt->rep.write_buffer_size = s; | |||||
} | |||||
void leveldb_options_set_max_open_files(leveldb_options_t* opt, int n) { | |||||
opt->rep.max_open_files = n; | |||||
} | |||||
void leveldb_options_set_cache(leveldb_options_t* opt, leveldb_cache_t* c) { | |||||
opt->rep.block_cache = c->rep; | |||||
} | |||||
void leveldb_options_set_block_size(leveldb_options_t* opt, size_t s) { | |||||
opt->rep.block_size = s; | |||||
} | |||||
void leveldb_options_set_block_restart_interval(leveldb_options_t* opt, int n) { | |||||
opt->rep.block_restart_interval = n; | |||||
} | |||||
void leveldb_options_set_compression(leveldb_options_t* opt, int t) { | |||||
opt->rep.compression = static_cast<CompressionType>(t); | |||||
} | |||||
leveldb_comparator_t* leveldb_comparator_create( | |||||
void* state, | |||||
void (*destructor)(void*), | |||||
int (*compare)( | |||||
void*, | |||||
const char* a, size_t alen, | |||||
const char* b, size_t blen), | |||||
const char* (*name)(void*)) { | |||||
leveldb_comparator_t* result = new leveldb_comparator_t; | |||||
result->state_ = state; | |||||
result->destructor_ = destructor; | |||||
result->compare_ = compare; | |||||
result->name_ = name; | |||||
return result; | |||||
} | |||||
void leveldb_comparator_destroy(leveldb_comparator_t* cmp) { | |||||
delete cmp; | |||||
} | |||||
leveldb_readoptions_t* leveldb_readoptions_create() { | |||||
return new leveldb_readoptions_t; | |||||
} | |||||
void leveldb_readoptions_destroy(leveldb_readoptions_t* opt) { | |||||
delete opt; | |||||
} | |||||
void leveldb_readoptions_set_verify_checksums( | |||||
leveldb_readoptions_t* opt, | |||||
unsigned char v) { | |||||
opt->rep.verify_checksums = v; | |||||
} | |||||
void leveldb_readoptions_set_fill_cache( | |||||
leveldb_readoptions_t* opt, unsigned char v) { | |||||
opt->rep.fill_cache = v; | |||||
} | |||||
void leveldb_readoptions_set_snapshot( | |||||
leveldb_readoptions_t* opt, | |||||
const leveldb_snapshot_t* snap) { | |||||
opt->rep.snapshot = (snap ? snap->rep : NULL); | |||||
} | |||||
leveldb_writeoptions_t* leveldb_writeoptions_create() { | |||||
return new leveldb_writeoptions_t; | |||||
} | |||||
void leveldb_writeoptions_destroy(leveldb_writeoptions_t* opt) { | |||||
delete opt; | |||||
} | |||||
void leveldb_writeoptions_set_sync( | |||||
leveldb_writeoptions_t* opt, unsigned char v) { | |||||
opt->rep.sync = v; | |||||
} | |||||
leveldb_cache_t* leveldb_cache_create_lru(size_t capacity) { | |||||
leveldb_cache_t* c = new leveldb_cache_t; | |||||
c->rep = NewLRUCache(capacity); | |||||
return c; | |||||
} | |||||
void leveldb_cache_destroy(leveldb_cache_t* cache) { | |||||
delete cache->rep; | |||||
delete cache; | |||||
} | |||||
leveldb_env_t* leveldb_create_default_env() { | |||||
leveldb_env_t* result = new leveldb_env_t; | |||||
result->rep = Env::Default(); | |||||
result->is_default = true; | |||||
return result; | |||||
} | |||||
void leveldb_env_destroy(leveldb_env_t* env) { | |||||
if (!env->is_default) delete env->rep; | |||||
delete env; | |||||
} | |||||
} // end extern "C" | |||||
} |
@ -0,0 +1,295 @@ | |||||
/* Copyright (c) 2011 The LevelDB Authors. All rights reserved. | |||||
Use of this source code is governed by a BSD-style license that can be | |||||
found in the LICENSE file. See the AUTHORS file for names of contributors. */ | |||||
#include "leveldb/c.h" | |||||
#include <stddef.h> | |||||
#include <stdio.h> | |||||
#include <stdlib.h> | |||||
#include <string.h> | |||||
#include <sys/types.h> | |||||
#include <unistd.h> | |||||
const char* phase = ""; | |||||
static char dbname[200]; | |||||
static void StartPhase(const char* name) { | |||||
fprintf(stderr, "=== Test %s\n", name); | |||||
phase = name; | |||||
} | |||||
#define CheckNoError(err) \ | |||||
if ((err) != NULL) { \ | |||||
fprintf(stderr, "%s:%d: %s: %s\n", __FILE__, __LINE__, phase, (err)); \ | |||||
abort(); \ | |||||
} | |||||
#define CheckCondition(cond) \ | |||||
if (!(cond)) { \ | |||||
fprintf(stderr, "%s:%d: %s: %s\n", __FILE__, __LINE__, phase, #cond); \ | |||||
abort(); \ | |||||
} | |||||
static void CheckEqual(const char* expected, const char* v, size_t n) { | |||||
if (expected == NULL && v == NULL) { | |||||
// ok | |||||
} else if (expected != NULL && v != NULL && n == strlen(expected) && | |||||
memcmp(expected, v, n) == 0) { | |||||
// ok | |||||
return; | |||||
} else { | |||||
fprintf(stderr, "%s: expected '%s', got '%s'\n", | |||||
phase, | |||||
(expected ? expected : "(null)"), | |||||
(v ? v : "(null")); | |||||
abort(); | |||||
} | |||||
} | |||||
static void Free(char** ptr) { | |||||
if (*ptr) { | |||||
free(*ptr); | |||||
*ptr = NULL; | |||||
} | |||||
} | |||||
static void CheckGet( | |||||
leveldb_t* db, | |||||
const leveldb_readoptions_t* options, | |||||
const char* key, | |||||
const char* expected) { | |||||
char* err = NULL; | |||||
size_t val_len; | |||||
char* val; | |||||
val = leveldb_get(db, options, key, strlen(key), &val_len, &err); | |||||
CheckNoError(err); | |||||
CheckEqual(expected, val, val_len); | |||||
Free(&val); | |||||
} | |||||
static void CheckIter(leveldb_iterator_t* iter, | |||||
const char* key, const char* val) { | |||||
size_t len; | |||||
const char* str; | |||||
str = leveldb_iter_key(iter, &len); | |||||
CheckEqual(key, str, len); | |||||
str = leveldb_iter_value(iter, &len); | |||||
CheckEqual(val, str, len); | |||||
} | |||||
// Callback from leveldb_writebatch_iterate() | |||||
static void CheckPut(void* ptr, | |||||
const char* k, size_t klen, | |||||
const char* v, size_t vlen) { | |||||
int* state = (int*) ptr; | |||||
CheckCondition(*state < 2); | |||||
switch (*state) { | |||||
case 0: | |||||
CheckEqual("bar", k, klen); | |||||
CheckEqual("b", v, vlen); | |||||
break; | |||||
case 1: | |||||
CheckEqual("box", k, klen); | |||||
CheckEqual("c", v, vlen); | |||||
break; | |||||
} | |||||
(*state)++; | |||||
} | |||||
// Callback from leveldb_writebatch_iterate() | |||||
static void CheckDel(void* ptr, const char* k, size_t klen) { | |||||
int* state = (int*) ptr; | |||||
CheckCondition(*state == 2); | |||||
CheckEqual("bar", k, klen); | |||||
(*state)++; | |||||
} | |||||
static void CmpDestroy(void* arg) { } | |||||
static int CmpCompare(void* arg, const char* a, size_t alen, | |||||
const char* b, size_t blen) { | |||||
int n = (alen < blen) ? alen : blen; | |||||
int r = memcmp(a, b, n); | |||||
if (r == 0) { | |||||
if (alen < blen) r = -1; | |||||
else if (alen > blen) r = +1; | |||||
} | |||||
return r; | |||||
} | |||||
static const char* CmpName(void* arg) { | |||||
return "foo"; | |||||
} | |||||
int main(int argc, char** argv) { | |||||
leveldb_t* db; | |||||
leveldb_comparator_t* cmp; | |||||
leveldb_cache_t* cache; | |||||
leveldb_env_t* env; | |||||
leveldb_options_t* options; | |||||
leveldb_readoptions_t* roptions; | |||||
leveldb_writeoptions_t* woptions; | |||||
char* err = NULL; | |||||
snprintf(dbname, sizeof(dbname), "/tmp/leveldb_c_test-%d", | |||||
((int) geteuid())); | |||||
StartPhase("create_objects"); | |||||
cmp = leveldb_comparator_create(NULL, CmpDestroy, CmpCompare, CmpName); | |||||
env = leveldb_create_default_env(); | |||||
cache = leveldb_cache_create_lru(100000); | |||||
options = leveldb_options_create(); | |||||
leveldb_options_set_comparator(options, cmp); | |||||
leveldb_options_set_error_if_exists(options, 1); | |||||
leveldb_options_set_cache(options, cache); | |||||
leveldb_options_set_env(options, env); | |||||
leveldb_options_set_info_log(options, NULL); | |||||
leveldb_options_set_write_buffer_size(options, 100000); | |||||
leveldb_options_set_paranoid_checks(options, 1); | |||||
leveldb_options_set_max_open_files(options, 10); | |||||
leveldb_options_set_block_size(options, 1024); | |||||
leveldb_options_set_block_restart_interval(options, 8); | |||||
leveldb_options_set_compression(options, leveldb_no_compression); | |||||
roptions = leveldb_readoptions_create(); | |||||
leveldb_readoptions_set_verify_checksums(roptions, 1); | |||||
leveldb_readoptions_set_fill_cache(roptions, 0); | |||||
woptions = leveldb_writeoptions_create(); | |||||
leveldb_writeoptions_set_sync(woptions, 1); | |||||
StartPhase("destroy"); | |||||
leveldb_destroy_db(options, dbname, &err); | |||||
Free(&err); | |||||
StartPhase("open_error"); | |||||
db = leveldb_open(options, dbname, &err); | |||||
CheckCondition(err != NULL); | |||||
Free(&err); | |||||
StartPhase("open"); | |||||
leveldb_options_set_create_if_missing(options, 1); | |||||
db = leveldb_open(options, dbname, &err); | |||||
CheckNoError(err); | |||||
CheckGet(db, roptions, "foo", NULL); | |||||
StartPhase("put"); | |||||
leveldb_put(db, woptions, "foo", 3, "hello", 5, &err); | |||||
CheckNoError(err); | |||||
CheckGet(db, roptions, "foo", "hello"); | |||||
StartPhase("writebatch"); | |||||
{ | |||||
leveldb_writebatch_t* wb = leveldb_writebatch_create(); | |||||
leveldb_writebatch_put(wb, "foo", 3, "a", 1); | |||||
leveldb_writebatch_clear(wb); | |||||
leveldb_writebatch_put(wb, "bar", 3, "b", 1); | |||||
leveldb_writebatch_put(wb, "box", 3, "c", 1); | |||||
leveldb_writebatch_delete(wb, "bar", 3); | |||||
leveldb_write(db, woptions, wb, &err); | |||||
CheckNoError(err); | |||||
CheckGet(db, roptions, "foo", "hello"); | |||||
CheckGet(db, roptions, "bar", NULL); | |||||
CheckGet(db, roptions, "box", "c"); | |||||
int pos = 0; | |||||
leveldb_writebatch_iterate(wb, &pos, CheckPut, CheckDel); | |||||
CheckCondition(pos == 3); | |||||
leveldb_writebatch_destroy(wb); | |||||
} | |||||
StartPhase("iter"); | |||||
{ | |||||
leveldb_iterator_t* iter = leveldb_create_iterator(db, roptions); | |||||
CheckCondition(!leveldb_iter_valid(iter)); | |||||
leveldb_iter_seek_to_first(iter); | |||||
CheckCondition(leveldb_iter_valid(iter)); | |||||
CheckIter(iter, "box", "c"); | |||||
leveldb_iter_next(iter); | |||||
CheckIter(iter, "foo", "hello"); | |||||
leveldb_iter_prev(iter); | |||||
CheckIter(iter, "box", "c"); | |||||
leveldb_iter_prev(iter); | |||||
CheckCondition(!leveldb_iter_valid(iter)); | |||||
leveldb_iter_seek_to_last(iter); | |||||
CheckIter(iter, "foo", "hello"); | |||||
leveldb_iter_seek(iter, "b", 1); | |||||
CheckIter(iter, "box", "c"); | |||||
leveldb_iter_get_error(iter, &err); | |||||
CheckNoError(err); | |||||
leveldb_iter_destroy(iter); | |||||
} | |||||
StartPhase("approximate_sizes"); | |||||
{ | |||||
int i; | |||||
int n = 20000; | |||||
char keybuf[100]; | |||||
char valbuf[100]; | |||||
uint64_t sizes[2]; | |||||
const char* start[2] = { "a", "k00000000000000010000" }; | |||||
size_t start_len[2] = { 1, 21 }; | |||||
const char* limit[2] = { "k00000000000000010000", "z" }; | |||||
size_t limit_len[2] = { 21, 1 }; | |||||
leveldb_writeoptions_set_sync(woptions, 0); | |||||
for (i = 0; i < n; i++) { | |||||
snprintf(keybuf, sizeof(keybuf), "k%020d", i); | |||||
snprintf(valbuf, sizeof(valbuf), "v%020d", i); | |||||
leveldb_put(db, woptions, keybuf, strlen(keybuf), valbuf, strlen(valbuf), | |||||
&err); | |||||
CheckNoError(err); | |||||
} | |||||
leveldb_approximate_sizes(db, 2, start, start_len, limit, limit_len, sizes); | |||||
CheckCondition(sizes[0] > 0); | |||||
CheckCondition(sizes[1] > 0); | |||||
} | |||||
StartPhase("property"); | |||||
{ | |||||
char* prop = leveldb_property_value(db, "nosuchprop"); | |||||
CheckCondition(prop == NULL); | |||||
prop = leveldb_property_value(db, "leveldb.stats"); | |||||
CheckCondition(prop != NULL); | |||||
Free(&prop); | |||||
} | |||||
StartPhase("snapshot"); | |||||
{ | |||||
const leveldb_snapshot_t* snap; | |||||
snap = leveldb_create_snapshot(db); | |||||
leveldb_delete(db, woptions, "foo", 3, &err); | |||||
CheckNoError(err); | |||||
leveldb_readoptions_set_snapshot(roptions, snap); | |||||
CheckGet(db, roptions, "foo", "hello"); | |||||
leveldb_readoptions_set_snapshot(roptions, NULL); | |||||
CheckGet(db, roptions, "foo", NULL); | |||||
leveldb_release_snapshot(db, snap); | |||||
} | |||||
StartPhase("repair"); | |||||
{ | |||||
leveldb_close(db); | |||||
leveldb_options_set_create_if_missing(options, 0); | |||||
leveldb_options_set_error_if_exists(options, 0); | |||||
leveldb_repair_db(options, dbname, &err); | |||||
CheckNoError(err); | |||||
db = leveldb_open(options, dbname, &err); | |||||
CheckNoError(err); | |||||
CheckGet(db, roptions, "foo", NULL); | |||||
CheckGet(db, roptions, "bar", NULL); | |||||
CheckGet(db, roptions, "box", "c"); | |||||
} | |||||
StartPhase("cleanup"); | |||||
leveldb_close(db); | |||||
leveldb_options_destroy(options); | |||||
leveldb_readoptions_destroy(roptions); | |||||
leveldb_writeoptions_destroy(woptions); | |||||
leveldb_cache_destroy(cache); | |||||
leveldb_comparator_destroy(cmp); | |||||
leveldb_env_destroy(env); | |||||
fprintf(stderr, "PASS\n"); | |||||
return 0; | |||||
} |
@ -0,0 +1,246 @@ | |||||
/* Copyright (c) 2011 The LevelDB Authors. All rights reserved. | |||||
Use of this source code is governed by a BSD-style license that can be | |||||
found in the LICENSE file. See the AUTHORS file for names of contributors. | |||||
C bindings for leveldb. May be useful as a stable ABI that can be | |||||
used by programs that keep leveldb in a shared library, or for | |||||
a JNI api. | |||||
Does not support: | |||||
. getters for the option types | |||||
. custom comparators that implement key shortening | |||||
. capturing post-write-snapshot | |||||
. custom iter, db, env, cache implementations using just the C bindings | |||||
Some conventions: | |||||
(1) We expose just opaque struct pointers and functions to clients. | |||||
This allows us to change internal representations without having to | |||||
recompile clients. | |||||
(2) For simplicity, there is no equivalent to the Slice type. Instead, | |||||
the caller has to pass the pointer and length as separate | |||||
arguments. | |||||
(3) Errors are represented by a null-terminated c string. NULL | |||||
means no error. All operations that can raise an error are passed | |||||
a "char** errptr" as the last argument. One of the following must | |||||
be true on entry: | |||||
*errptr == NULL | |||||
*errptr points to a malloc()ed null-terminated error message | |||||
On success, a leveldb routine leaves *errptr unchanged. | |||||
On failure, leveldb frees the old value of *errptr and | |||||
set *errptr to a malloc()ed error message. | |||||
(4) Bools have the type unsigned char (0 == false; rest == true) | |||||
(5) All of the pointer arguments must be non-NULL. | |||||
*/ | |||||
#ifndef STORAGE_LEVELDB_INCLUDE_C_H_ | |||||
#define STORAGE_LEVELDB_INCLUDE_C_H_ | |||||
#ifdef __cplusplus | |||||
extern "C" { | |||||
#endif | |||||
#include <stdarg.h> | |||||
#include <stddef.h> | |||||
#include <stdint.h> | |||||
/* Exported types */ | |||||
typedef struct leveldb_t leveldb_t; | |||||
typedef struct leveldb_cache_t leveldb_cache_t; | |||||
typedef struct leveldb_comparator_t leveldb_comparator_t; | |||||
typedef struct leveldb_env_t leveldb_env_t; | |||||
typedef struct leveldb_filelock_t leveldb_filelock_t; | |||||
typedef struct leveldb_iterator_t leveldb_iterator_t; | |||||
typedef struct leveldb_logger_t leveldb_logger_t; | |||||
typedef struct leveldb_options_t leveldb_options_t; | |||||
typedef struct leveldb_randomfile_t leveldb_randomfile_t; | |||||
typedef struct leveldb_readoptions_t leveldb_readoptions_t; | |||||
typedef struct leveldb_seqfile_t leveldb_seqfile_t; | |||||
typedef struct leveldb_snapshot_t leveldb_snapshot_t; | |||||
typedef struct leveldb_writablefile_t leveldb_writablefile_t; | |||||
typedef struct leveldb_writebatch_t leveldb_writebatch_t; | |||||
typedef struct leveldb_writeoptions_t leveldb_writeoptions_t; | |||||
/* DB operations */ | |||||
extern leveldb_t* leveldb_open( | |||||
const leveldb_options_t* options, | |||||
const char* name, | |||||
char** errptr); | |||||
extern void leveldb_close(leveldb_t* db); | |||||
extern void leveldb_put( | |||||
leveldb_t* db, | |||||
const leveldb_writeoptions_t* options, | |||||
const char* key, size_t keylen, | |||||
const char* val, size_t vallen, | |||||
char** errptr); | |||||
extern void leveldb_delete( | |||||
leveldb_t* db, | |||||
const leveldb_writeoptions_t* options, | |||||
const char* key, size_t keylen, | |||||
char** errptr); | |||||
extern void leveldb_write( | |||||
leveldb_t* db, | |||||
const leveldb_writeoptions_t* options, | |||||
leveldb_writebatch_t* batch, | |||||
char** errptr); | |||||
/* Returns NULL if not found. A malloc()ed array otherwise. | |||||
Stores the length of the array in *vallen. */ | |||||
extern char* leveldb_get( | |||||
leveldb_t* db, | |||||
const leveldb_readoptions_t* options, | |||||
const char* key, size_t keylen, | |||||
size_t* vallen, | |||||
char** errptr); | |||||
extern leveldb_iterator_t* leveldb_create_iterator( | |||||
leveldb_t* db, | |||||
const leveldb_readoptions_t* options); | |||||
extern const leveldb_snapshot_t* leveldb_create_snapshot( | |||||
leveldb_t* db); | |||||
extern void leveldb_release_snapshot( | |||||
leveldb_t* db, | |||||
const leveldb_snapshot_t* snapshot); | |||||
/* Returns NULL if property name is unknown. | |||||
Else returns a pointer to a malloc()-ed null-terminated value. */ | |||||
extern char* leveldb_property_value( | |||||
leveldb_t* db, | |||||
const char* propname); | |||||
extern void leveldb_approximate_sizes( | |||||
leveldb_t* db, | |||||
int num_ranges, | |||||
const char* const* range_start_key, const size_t* range_start_key_len, | |||||
const char* const* range_limit_key, const size_t* range_limit_key_len, | |||||
uint64_t* sizes); | |||||
/* Management operations */ | |||||
extern void leveldb_destroy_db( | |||||
const leveldb_options_t* options, | |||||
const char* name, | |||||
char** errptr); | |||||
extern void leveldb_repair_db( | |||||
const leveldb_options_t* options, | |||||
const char* name, | |||||
char** errptr); | |||||
/* Iterator */ | |||||
extern void leveldb_iter_destroy(leveldb_iterator_t*); | |||||
extern unsigned char leveldb_iter_valid(const leveldb_iterator_t*); | |||||
extern void leveldb_iter_seek_to_first(leveldb_iterator_t*); | |||||
extern void leveldb_iter_seek_to_last(leveldb_iterator_t*); | |||||
extern void leveldb_iter_seek(leveldb_iterator_t*, const char* k, size_t klen); | |||||
extern void leveldb_iter_next(leveldb_iterator_t*); | |||||
extern void leveldb_iter_prev(leveldb_iterator_t*); | |||||
extern const char* leveldb_iter_key(const leveldb_iterator_t*, size_t* klen); | |||||
extern const char* leveldb_iter_value(const leveldb_iterator_t*, size_t* vlen); | |||||
extern void leveldb_iter_get_error(const leveldb_iterator_t*, char** errptr); | |||||
/* Write batch */ | |||||
extern leveldb_writebatch_t* leveldb_writebatch_create(); | |||||
extern void leveldb_writebatch_destroy(leveldb_writebatch_t*); | |||||
extern void leveldb_writebatch_clear(leveldb_writebatch_t*); | |||||
extern void leveldb_writebatch_put( | |||||
leveldb_writebatch_t*, | |||||
const char* key, size_t klen, | |||||
const char* val, size_t vlen); | |||||
extern void leveldb_writebatch_delete( | |||||
leveldb_writebatch_t*, | |||||
const char* key, size_t klen); | |||||
extern void leveldb_writebatch_iterate( | |||||
leveldb_writebatch_t*, | |||||
void* state, | |||||
void (*put)(void*, const char* k, size_t klen, const char* v, size_t vlen), | |||||
void (*deleted)(void*, const char* k, size_t klen)); | |||||
/* Options */ | |||||
extern leveldb_options_t* leveldb_options_create(); | |||||
extern void leveldb_options_destroy(leveldb_options_t*); | |||||
extern void leveldb_options_set_comparator( | |||||
leveldb_options_t*, | |||||
leveldb_comparator_t*); | |||||
extern void leveldb_options_set_create_if_missing( | |||||
leveldb_options_t*, unsigned char); | |||||
extern void leveldb_options_set_error_if_exists( | |||||
leveldb_options_t*, unsigned char); | |||||
extern void leveldb_options_set_paranoid_checks( | |||||
leveldb_options_t*, unsigned char); | |||||
extern void leveldb_options_set_env(leveldb_options_t*, leveldb_env_t*); | |||||
extern void leveldb_options_set_info_log(leveldb_options_t*, leveldb_logger_t*); | |||||
extern void leveldb_options_set_write_buffer_size(leveldb_options_t*, size_t); | |||||
extern void leveldb_options_set_max_open_files(leveldb_options_t*, int); | |||||
extern void leveldb_options_set_cache(leveldb_options_t*, leveldb_cache_t*); | |||||
extern void leveldb_options_set_block_size(leveldb_options_t*, size_t); | |||||
extern void leveldb_options_set_block_restart_interval(leveldb_options_t*, int); | |||||
enum { | |||||
leveldb_no_compression = 0, | |||||
leveldb_snappy_compression = 1 | |||||
}; | |||||
extern void leveldb_options_set_compression(leveldb_options_t*, int); | |||||
/* Comparator */ | |||||
extern leveldb_comparator_t* leveldb_comparator_create( | |||||
void* state, | |||||
void (*destructor)(void*), | |||||
int (*compare)( | |||||
void*, | |||||
const char* a, size_t alen, | |||||
const char* b, size_t blen), | |||||
const char* (*name)(void*)); | |||||
extern void leveldb_comparator_destroy(leveldb_comparator_t*); | |||||
/* Read options */ | |||||
extern leveldb_readoptions_t* leveldb_readoptions_create(); | |||||
extern void leveldb_readoptions_destroy(leveldb_readoptions_t*); | |||||
extern void leveldb_readoptions_set_verify_checksums( | |||||
leveldb_readoptions_t*, | |||||
unsigned char); | |||||
extern void leveldb_readoptions_set_fill_cache( | |||||
leveldb_readoptions_t*, unsigned char); | |||||
extern void leveldb_readoptions_set_snapshot( | |||||
leveldb_readoptions_t*, | |||||
const leveldb_snapshot_t*); | |||||
/* Write options */ | |||||
extern leveldb_writeoptions_t* leveldb_writeoptions_create(); | |||||
extern void leveldb_writeoptions_destroy(leveldb_writeoptions_t*); | |||||
extern void leveldb_writeoptions_set_sync( | |||||
leveldb_writeoptions_t*, unsigned char); | |||||
/* Cache */ | |||||
extern leveldb_cache_t* leveldb_cache_create_lru(size_t capacity); | |||||
extern void leveldb_cache_destroy(leveldb_cache_t* cache); | |||||
/* Env */ | |||||
extern leveldb_env_t* leveldb_create_default_env(); | |||||
extern void leveldb_env_destroy(leveldb_env_t*); | |||||
#ifdef __cplusplus | |||||
} /* end extern "C" */ | |||||
#endif | |||||
#endif /* STORAGE_LEVELDB_INCLUDE_C_H_ */ |