An experiment on persistent data-streaming
Currently, working in the foundation of the storage layer.
Checkout the Roadmap for feature-specific details.
- Learn and have fun™️
- Implement a Kinesis-like streaming-service
- Single binary
- Easy to Host, Run & Operate (have you tried to run Kafka yourself?)
- Kubernetes friendly
The main component of the whole system is the commit-log, an abstraction manages reads and writes to the log by implementing an immutable, append-only, file-backed sequence of "records", or chunks of data/events that are transmited from producers to consumers.
Records can be written to the log, always appending the last record over and over.
e.g.:
current cursor
segment 0 ^
|-------------------------------|
| record 0 | record 1 | ... | --> time
|-------------------------------|
In order to manage and scale read and writes, the commit-log split groups of records into Segments, managing to write to a single segment until it reaches a certain, specified size.
Each time a record is written, the segment is trusted to have enough space for the given buffer, then the record is written to the current segment, and the pointer is updated.
More info in the commit_log/src/lib.rs
file.
A Segment is a tuple abstraction to manage the Index and Log files.
Every Segment is composed of a log-file and an index, e.g.:
00000000000011812312.log
00000000000011812312.idx
The role of the segment is to manage writes to the logfile and ensure the entries can be read later on by doing lookups in the index.
On every write, the segment writes an entry to the index with the record's position and size, in the log-file, for later use.
The segment also manages the size of the log file, preventing it from being written once it reaches the specified.
When a segment is full, the commit log makes sure to rotate to a new one, closing the old one.
See how it looks like on disk (on a high-level):
current cursor
segment 0 ^
|-------------------------------| |
| record 0 | record 1 | ... | segment 1 (current) |
|-------------------------------|-----------------------------| --> time
| record 2 | record 3 | ... |
|-----------------------------|
Under the hood is a bit more complex, the management of writing to the file to disk is of the Segments', as well as managing the Index file.
More info in the commit_log/src/segment.rs
and commit_log/src/segment/index.rs and log.rs
files.
The log file is a varied-size sequence of bytes that is storing the content of the records produced by the producers. However, the log itself doesn't have any mechanism for recovery of such records. That's responsibility of the index.
Once initialized, the log-file is truncated to reach the desired value and reserve both memory and disk space, the same for the index.
current cursor
^
|-------------------------------|
| record 0 | record 1 | ... |----> time
|-------------------------------|
Neither reads nor writes to the index are directly triggering disk-level actions.
Both operations are being intermediated by a memory-mapping buffers, managed by the OS.
More info in the commit_log/src/segment/log.rs
file.
The role of the index is to provide pointers to records in the log file. Each entry of the index is 20 bytes long, 10 bytes are used for the offset address of the record in the log file, the other 10 bytes for the size of the record.
e.g.:
current cursor
^
|-------------------------------|
| offset-size | offset-size |...|----> time
|-------------------------------|
There is no separator, it's position-based.
e.g.:
00000001000000000020
---------------------
offset | size
* 000000010 -> offset
* 000000020 -> size
Neither reads nor writes to the index are directly triggering disk-level actions.
Both operations are being intermediated by a memory-mapping buffers, managed by the OS.
More info in the commit_log/src/segment/index.rs
file.
These are preliminar and poorly collected results, yet it looks interesting:
Storage (Tests are completely offline, no network¹ ...)
Segment size: 20MB
Index size: 10MB
5 GB worth records written in 37.667706s
5 GB worth cold records read in 1.384433s
5 GB worth warm records read in 1.266053s
Per-segment²:
- ~130 MB/s on write
- ~3.7 GB/s on cold read (while loading into memory pages)
- ~4.2 GB/s on warm read (already loaded into memory pages)
Notes:
- ¹ - Offline - no network overhead taken into account, network will be a big player on the overhead. However, the focus now is storage.
- ² - Per-segment performance, in a comparinson with kinesis/kafka that would be the per-shard value. If you were to have 10 shards, you could achieve 10x that, limited by external factors, HD/CPU/...
Setup:
OS: macOS Mojave 10.14.4 (18E226)
CPU: 2,5 GHz Intel Core i7
RAM: 16 GB 2133 MHz LPDDR3
HD: 256 GB SSD Storage
- Settle for a name
- Storage
- Core CommitLog
- Segments
- Write to segments
- Write across segments
- Read from segment
- Read across segments
- Index
- Better API for reading
- Extract logfile from segment
- Memory Mapped IO (Performance test before/after)
- Write
- Index
- Segment
- Refactor segment split to take index into account
- Read
- Index
- Segment
- Write
- Restore state from disk - Ref: (https://github.com/zowens/commitlog/blob/master/src/file_set.rs#L17-L98)
- Record Struct
- End-to-end use of records
- Segments
- Topics/Streams (probably should come up with a better name)
- Partitions/Shards
- ?
- Partitions/Shards
- Core CommitLog
- Networking/API
- Decide simple inicial protocol
- Implement Basic Socket Communication
- Implement TCP/HTTP?
- Write to streams over the network
- zero-cost copy (OS sendfile)
- Replication
- More to come here...
- Operational
- Configuration
- CLI Basics
- Configuration
- CI/Tooling
- Setup Travis
- Build with Linux/macOS
- Docker
- Alpine
- Setup Travis
- Testing
- Benchmarks
- Write
- Read
- Setup Integration Tests
- Abstract Test Code
- Benchmarks
- Documentation
- Document decisions
- Document data structures / types
- Improve README
- How to Run
- Document decisions
Available make commands
make build
- Builds the application with cargomake build_release
- Builds the application with cargo, with release optimizationsmake docker_test_watcher
- Runs funzzy on linux over docker-composemake docs
- Generate the GitHub Markdown docs (At the moment only mermaid)make format
- Formats the code according to cargomake help
- Lists the available commands. Add a comment with '##' to describe a command.make install
- Builds a release version and installs to your cago bin pathmake run
- Runs the newly builtmake test
- Tests all features
- Talks
- Blogposts
- Building a Distributed Log from Scratch, Part 1: Storage Mechanics
- Building a Distributed Log from Scratch, Part 2: Data Replication
- Building a Distributed Log from Scratch, Part 3: Scaling Message Delivery
- Building a Distributed Log from Scratch, Part 4: Trade-Offs and Lessons Learned
- Building a Distributed Log from Scratch, Part 5: Sketching a New System
- The Log: What every software engineer should know about real-time data's unifying abstraction
- How Kafka's Storage Internals Work
- Code
- travisjeffery/Jocko - Distributed commit log service in Go
- zowens/commitlog - Append-only commit log library for Rust
- liftbridge-io/liftbridge - Lightweight, fault-tolerant message streams