Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/preetam/vlmap

Versioned, ordered, skip list map in C
https://github.com/preetam/vlmap

Last synced: 4 days ago
JSON representation

Versioned, ordered, skip list map in C

Awesome Lists containing this project

README

        

vlmap
=====
vlmap is a versioned, ordered map built on a skip list.

Keys and values are `uint8_t`s, and the keys are sorted in lexicographical order.

Since it is versioned, snapshot reads are possible.

This is **not** guaranteed to be thread-safe, so it would be best to do writes serially.
You should be able to do concurrent reads as long as the versions being used aren't cleaned.

"Cleaning" basically means removing elements that have been logically deleted. This is an O(n log n)
operation since we have to iterate through the entire list, which is O(n), and remove
elements from the skip list, which is O(log n).

License
----
MIT

API
---
The header file has a short description above each function:
```c
// vlmap_create returns a pointer to a new vlmap.
vlmap*
vlmap_create();

// vlmap_destroy destroys a vlmap and all of
// its nodes.
void
vlmap_destroy(vlmap* m);

// vlmap_insert inserts a key-value pair into the map at a certain version.
int
vlmap_insert(vlmap* m, uint64_t version, uint8_t* key, int keylength,
uint8_t* value, int valuelength);

// vlmap_remove logically removes a node from the map, if it matches
// the given key, at a certain version. Returns 0 on success.
int
vlmap_remove(vlmap* m, uint64_t version, uint8_t* key, int keylength);

// vlmap_get gets a key-value at a certain version.
// On success, 0 is returned and value and valuelength are filled
// with the respective data.
// **It's the caller's responsibility to free the value**.
// On error, value and valuelength will not be modified.
int
vlmap_get(vlmap* m, uint64_t version, uint8_t* key, int keylength,
uint8_t** value, int* valuelength);

// vlmap_version returns the current version of the map.
uint64_t
vlmap_version(vlmap* m);

// vlmap_version_increment increments the current version.
void
vlmap_version_increment(vlmap* m);

// vlmap_prints a map, along with its skip list levels,
// at a certain version. This is useful for debugging.
void
vlmap_print(vlmap* m, int version);

// vlmap_clean removes any node in the map
// older than version.
void
vlmap_clean(vlmap* m, uint64_t version);

// vlmap_iterator_create creates a new iterator,
// which must later be destroyed with vlmap_iterator_destroy,
// at a version. This allows for snapshot range reads.
//
// A start key and an end key are required, but one can
// use `\x00' and `\xff' as the start and end, respectively,
// to essentially read from the entire range.
//
// [startkey, endkey)
// The start key is inclusive, but the end key
// is exclusive. Getting a range from `a' to `f' will
// not include a key matching `f'.
// Appending a `\xff' to the end key should give
// the range [startkey, endkey].
vlmap_iterator*
vlmap_iterator_create(vlmap* m, uint64_t version,
uint8_t* startkey, int startkeylen,
uint8_t* endkey, int endkeylen);

// vlmap_iterator_destroy frees the space allocated
// to a vlmap_iterator.
void
vlmap_iterator_destroy(vlmap_iterator* i);

// vlmap_iterator_get_key returns the key of the current
// location of the iterator.
// This returns 0 on success, and key must be freed by
// the caller.
int
vlmap_iterator_get_key(vlmap_iterator* i, uint8_t** key, int* keylength);

// vlmap_iterator_get_value works the same way as calling
// vlmap_get on the current node of the iterator.
// This returns 0 on success, and key must be freed by
// the caller.
int
vlmap_iterator_get_value(vlmap_iterator* i, uint8_t** value, int* valuelength);

// vlmap_iterator_next moves to the next node in the map.
// Map elements not in the current snapshot are ignored.
// Returns NULL when out of range, so it's important to
// store the original pointer to the iterator to free it
// using vlmap_iterator_destroy.
vlmap_iterator*
vlmap_iterator_next(vlmap_iterator* i);

// vlmap_iterator_remove removes the current node in the
// map and moves to the next node.
vlmap_iterator*
vlmap_iterator_remove(vlmap_iterator* i);
```