2022-12-02 21:37:56 +05:00
|
|
|
# zkv
|
|
|
|
|
|
|
|
Simple key-value store for single-user applications.
|
|
|
|
|
|
|
|
## Pros
|
|
|
|
|
2022-12-11 21:33:51 +05:00
|
|
|
* Simple two file structure (data file and index file)
|
2022-12-03 12:41:11 +05:00
|
|
|
* Internal Zstandard compression by [klauspost/compress/zstd](https://github.com/klauspost/compress/tree/master/zstd)
|
2022-12-02 21:37:56 +05:00
|
|
|
* Threadsafe operations through `sync.RWMutex`
|
|
|
|
|
|
|
|
## Cons
|
|
|
|
|
2022-12-10 22:00:42 +05:00
|
|
|
* Index stored in memory (`map[key hash (28 bytes)]file offset (int64)`)
|
2022-12-11 21:33:51 +05:00
|
|
|
* No transaction system
|
|
|
|
* Index file is fully rewrited on every store commit
|
2022-12-02 21:37:56 +05:00
|
|
|
* No way to recover disk space from deleted records
|
|
|
|
* Write/Delete operations block Read and each other operations
|
|
|
|
|
|
|
|
## Usage
|
|
|
|
|
|
|
|
Create or open existing file:
|
|
|
|
|
|
|
|
```go
|
2022-12-09 20:05:57 +05:00
|
|
|
db, err := zkv.Open("path to file")
|
2022-12-02 21:37:56 +05:00
|
|
|
```
|
|
|
|
|
|
|
|
Data operations:
|
|
|
|
|
|
|
|
```go
|
|
|
|
// Write data
|
|
|
|
err = db.Set(key, value) // key and value can be any of type
|
|
|
|
|
|
|
|
// Read data
|
|
|
|
var value ValueType
|
2022-12-03 20:57:50 +05:00
|
|
|
err = db.Get(key, &value)
|
2022-12-02 21:37:56 +05:00
|
|
|
|
|
|
|
// Delete data
|
|
|
|
err = db.Delete(key)
|
|
|
|
```
|
|
|
|
|
2022-12-03 20:57:50 +05:00
|
|
|
Other methods:
|
|
|
|
|
|
|
|
```go
|
|
|
|
// Flush data to disk
|
|
|
|
err = db.Flush()
|
2022-12-05 21:26:54 +05:00
|
|
|
|
|
|
|
// Backup data to another file
|
|
|
|
err = db.Backup("new/file/path")
|
2022-12-03 20:57:50 +05:00
|
|
|
```
|
|
|
|
|
2022-12-10 21:39:24 +05:00
|
|
|
## Store options
|
|
|
|
|
|
|
|
```go
|
|
|
|
type Options struct {
|
|
|
|
// Maximum number of concurrent reads
|
|
|
|
MaxParallelReads int
|
|
|
|
|
|
|
|
// Compression level
|
|
|
|
CompressionLevel zstd.EncoderLevel
|
|
|
|
|
|
|
|
// Memory write buffer size in bytes
|
|
|
|
MemoryBufferSize int
|
|
|
|
|
|
|
|
// Disk write buffer size in bytes
|
|
|
|
DiskBufferSize int
|
|
|
|
}
|
|
|
|
|
|
|
|
```
|
|
|
|
|
2022-12-02 21:37:56 +05:00
|
|
|
## File structure
|
|
|
|
|
|
|
|
Record is `encoding/gob` structure:
|
|
|
|
|
|
|
|
| Field | Description | Size |
|
|
|
|
| ---------- | ---------------------------------- | -------- |
|
|
|
|
| Type | Record type | uint8 |
|
|
|
|
| KeyHash | Key hash | 28 bytes |
|
|
|
|
| ValueBytes | Value gob-encoded bytes | variable |
|
|
|
|
|
|
|
|
File is log stuctured list of commands:
|
|
|
|
|
|
|
|
| Field | Description | Size |
|
|
|
|
| -------| ------------------------ | -------- |
|
|
|
|
| Length | Record body bytes length | int64 |
|
|
|
|
| Body | Gob-encoded record | variable |
|
2022-12-03 21:03:27 +05:00
|
|
|
|
2022-12-11 21:33:51 +05:00
|
|
|
Index file is simple gob-encoded map:
|
|
|
|
|
|
|
|
```go
|
|
|
|
map[string]struct {
|
|
|
|
BlockOffset int64
|
|
|
|
RecordOffset int64
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
where map key is data key hash and value - data offset in data file.
|
|
|
|
|
2022-12-10 22:00:08 +05:00
|
|
|
## Resource consumption
|
|
|
|
|
|
|
|
Store requirements:
|
|
|
|
|
|
|
|
* around 300 Mb of RAM per 1 million of keys
|
|
|
|
* around 34 Mb of disk space for index file per 1 million of keys
|
|
|
|
|
2022-12-03 21:03:27 +05:00
|
|
|
## TODO
|
|
|
|
|
2022-12-07 21:06:36 +05:00
|
|
|
- [ ] Add recovery previous state of store file on write error
|
2022-12-11 21:33:51 +05:00
|
|
|
- [ ] Add method for index rebuild
|