A python key-value file database
Project description
Introduction
Booklet is a pure python key-value file database. It allows for multiple serializers for both the keys and values. The API is designed to use all of the same python dictionary methods python programmers are used to in addition to the typical dbm methods.
Installation
Install via pip:
pip install booklet
Or conda:
conda install -c mullenkamp booklet
I’ll probably put it on conda-forge once I feel like it’s up to an appropriate standard…
Serialization
Both the keys and values stored in Booklet must be bytes when written to disk. This is the default when “open” is called. Booklet allows for various serializers to be used for taking input keys and values and converting them to bytes. There are many in-built serializers. Check the booklet.available_serializers list for what’s available. Some serializers require additional packages to be installed (e.g. orjson, zstd, etc). If you want to serialize to json, then it is highly recommended to use orjson as it is substantially faster than the standard json python module. If in-built serializers are assigned at initial file creation, then they will be saved on future reading and writing on the same file (i.e. they don’t need to be passed after the first time). Setting a serializer to None will not do any serializing, and the input must be bytes. The user can also pass custom serializers to the key_serializer and value_serializer parameters. These must have “dumps” and “loads” static methods. This allows the user to chain a serializer and a compressor together if desired. Custom serializers must be passed for writing and reading as they are not stored in the booklet file.
import booklet
print(booklet.available_serializers)
Usage
The docstrings have a lot of info about the classes and methods. Files should be opened with the booklet.open function. Read the docstrings of the open function for more details.
Write data using the context manager
import booklet
with booklet.open('test.blt', 'n', value_serializer='pickle', key_serializer='str') as db:
db['test_key'] = ['one', 2, 'three', 4]
Read data using the context manager
with booklet.open('test.blt', 'r') as db:
test_data = db['test_key']
Notice that you don’t need to pass serializer parameters when reading (and additional writing) when in-built serializers are used. Booklet stores this info on the initial file creation.
In most cases, the user should use python’s context manager “with” when reading and writing data. This will ensure data is properly written and (optionally) locks are released on the file. If the context manager is not used, then the user must be sure to run the db.sync() at the end of a series of writes to ensure the data has been fully written to disk.
Write data without using the context manager
import booklet
db = booklet.open('test.blt', 'n', value_serializer='pickle', key_serializer='str')
db['test_key'] = ['one', 2, 'three', 4]
db['2nd_test_key'] = ['five', 6, 'seven', 8]
db.sync()
db.close()
Read data without using the context manager
db = booklet.open('test.blt', 'r')
test_data1 = db['test_key']
test_data2 = db['2nd_test_key']
db.close()
Custom serializers
import orjson
class Orjson:
def dumps(obj):
return orjson.dumps(obj, option=orjson.OPT_NON_STR_KEYS | orjson.OPT_OMIT_MICROSECONDS | orjson.OPT_SERIALIZE_NUMPY)
def loads(obj):
return orjson.loads(obj)
with booklet.open('test.blt', 'n', value_serializer=Orjson, key_serializer='str') as db:
db['test_key'] = ['one', 2, 'three', 4]
The Orjson class is actually already built into the package. You can pass the string ‘orjson’ to either serializer parameters to use the above serializer. This is just an example of a serializer.
Here’s another example with compression.
import orjson
import zstandard as zstd
class OrjsonZstd:
def dumps(obj):
return zstd.compress(orjson.dumps(obj, option=orjson.OPT_NON_STR_KEYS | orjson.OPT_OMIT_MICROSECONDS | orjson.OPT_SERIALIZE_NUMPY))
def loads(obj):
return orjson.loads(zstd.decompress(obj))
with booklet.open('test.blt', 'n', value_serializer=OrjsonZstd, key_serializer='str') as db:
db['big_test'] = list(range(1000000))
with booklet.open('test.blt', 'r', value_serializer=OrjsonZstd) as db:
big_test_data = db['big_test']
The open flag follows the standard dbm options:
Value |
Meaning |
---|---|
'r' |
Open existing database for reading only (default) |
'w' |
Open existing database for reading and writing |
'c' |
Open database for reading and writing, creating it if it doesn’t exist |
'n' |
Always create a new, empty database, open for reading and writing |
TODO
I need to write a lot more tests for the functionality.
I would like to have the ability to prune files (i.e. remove old stale data from the file to shorten the file length). Unfortunately, the current file structure makes it extremely difficult to perform. A future version might have a different structure to support this better, but at the moment this kind of functionality is very minor. If a pruned file is desired, you can simply iterate through all of the keys and values to create a new file.
Benchmarks
From my initial tests, the performance is comparable to other very fast key-value databases (e.g. gdbm, lmdb). Proper benchmarks will be coming soon…
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file booklet-0.1.1.tar.gz
.
File metadata
- Download URL: booklet-0.1.1.tar.gz
- Upload date:
- Size: 15.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: python-httpx/0.26.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 8b76b05dd753a32f847df69db04994d96a77b7ec8030693593fc8314e87989be |
|
MD5 | e1a8bdc7cda885c369418452b015b15f |
|
BLAKE2b-256 | 639e02fa1f360fc5e2134bee5728fd33edd624c76fcb7db92d37b1e21e360805 |
File details
Details for the file booklet-0.1.1-py2.py3-none-any.whl
.
File metadata
- Download URL: booklet-0.1.1-py2.py3-none-any.whl
- Upload date:
- Size: 17.5 kB
- Tags: Python 2, Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: python-httpx/0.26.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | f2ec7458f1c08efc632fba1d8205ab3197fc6a1174b7d1528e87f1b4d5aeb556 |
|
MD5 | 627c775abf395ff476079d18b17c453a |
|
BLAKE2b-256 | 42b0bcb5533a99cfa606bf016c828c476f64c421a978bfea2a99a79ef77c5d37 |