/AugustDB

Key/value store backed by LSM Tree architecture.

Primary LanguageElixirMIT LicenseMIT

AugustDB

A key/value store backed by LSM Trees and SSTables.

This project is a work in progress 🚧 and is being developed primarily to suit the author's personal study goals 🎓. But if you find something in here that helps you learn, that's great too!

Initial design

All writes are first written to a commit log, protecting against crashes.

Newly written values are stored in a memtable backed by :gb_trees.

We define a binary SSTable format.

We use phoenix to expose a REST API (PUT, GET, DEL) for creating, reading, updating, and deleting (for now) string resources using Content-Type: application/json. Support for application/octet-stream is forthcoming.

Webservice operation

To start the Phoenix server in dev mode:

  • Install dependencies with mix deps.get
  • Install Node.js dependencies with npm install inside the assets directory
  • Start Phoenix endpoint with mix phx.server

Now you can visit localhost:4000 from your browser.

To run in prod mode with some optimizations, you first need to follow the Phoenix deployment instructions to generate a secret, etc. Then:

PORT=4001 MIX_ENV=prod mix phx.server

Making HTTP calls

Create or update a record

Using simple JSON string:

curl -X PUT  -H 'Content-Type: application/json' -d '"meh meh"'  http://localhost:4000/api/values/1

Using form data:

curl -X PUT -d value='meh meh'  http://localhost:4000/api/values/1

Get a record

Returns Content-Type: application/json.

curl http://localhost:4000/api/values/1

Delete a record

curl -X DELETE http://localhost:4000/api/values/1

SSTable Format

A Sorted String Table contains zero or more gzipped key/value chunks.

GZipped key/value chunks

A gzip key/value chunk follows this binary specification:

  1. Four bytes: length of the gzipped chunk
  2. Variable length: gzipped chunk of key/value pairs, with tombstones.

Unzipped key/value chunks

Once unzipped, each key/value chunk contains zero or more key/value records. Each record describes its own length. Some keys may point to tombstones.

Value records

  1. Four bytes: Length of key
  2. Four bytes: Length of value
  3. Variable length: Raw key, not escaped
  4. Variable length: Raw value, not escaped

Tombstone records

  1. Four bytes: Length of key in bytes
  2. Four bytes: 2^32 - 1 to indicate tombstone
  3. Variable length: Raw key, not escaped

Example in hex

One can imagine a simple, uncompressed binary representation of keys to values:

  • hey: now
  • no: yes
  • one: TOMBSTONE
  • three: four

binary-sstable-hex

In practice you won't ever see this sequence on disk, because the records will be gzipped.

Managing Commit Log Files

On app startup, all outstanding commit logs are replayed and flushed from memtable to SSTable, then deleted. During the course of normal operation, a new commit log is created every time the memtable is flushed, and new writes are routed to that file. The old commit log can be removed right away.

Read the design ticket for more information.

Inspiration

Kleppmann: Designing Data-Intensive Applications gives a fantastic summary of local-node operation for data stores using SSTable, followed by detail on strategies for replication and partitioning. Check it out!

I sourced https://github.com/tamas-soos/expoll/blob/master/lib/ex_poll_web/views/poll_view.ex as a preliminary example for working with Phoenix.

Dev environment

You should enable multi-time warp mode during development in the REPL.

ELIXIR_ERL_OPTIONS="+C multi_time_warp" iex -S mix

Generating docs

You can follow the CLI instructions for ExDoc:

ex_doc "AugustDB" "0.0.0" "_build/dev/lib/august_db/ebin"
 -m "AugustDbWeb.ValueController" -u "https://github.com/Terkwood/AugustDB"

🔮 The Glorious Future: a distributed system

First implement a local key-value store that uses a memtable, SSTables, and a commit log. Then implement a replicating data store which syncs via gossip. Then implement partitioning using vnodes. See the issue tracker.