# Chord: Sync, Manage, and Harmonize Contexts 🎵
Welcome to **Chord** — a flexible and powerful Elixir library designed to simplify context management and delta tracking in your distributed or real-time applications.
## 🎯 Why Chord?
When you need a solution for real-time state synchronization, partial updates, and efficient cleanup, Chord strikes the perfect note! Here’s what makes Chord special:
- **Seamless State Sync**: Keep your clients up-to-date with full context or delta-based updates.
- **Customizable Backend**: Use ETS, Redis, or your own backend implementation.
- **Flexible Delta Formatting**: Define how your updates are structured.
- **Periodic Cleanup**: Automatically clear stale contexts or deltas.
- **Developer-Friendly APIs**: Simple, consistent, and easy-to-use APIs.
- **Context Export and Restore**: Export contexts to or restore them from external providers.
- **Partial Updates**: Apply updates to specific fields within a context.
---
## 🚀 Getting Started
### 1️⃣ Install the Library
Add Chord to your Mix dependencies:
```elixir
def deps do
[
{:chord, "~> 0.1.0"}
]
end
```
Run:
```bash
mix deps.get
```
### 2️⃣ Configure Chord
Add your desired configuration in `config/config.exs`:
```elixir
config :chord,
backend: Chord.Backend.ETS, # Choose your backend (Redis, ETS, etc.)
context_auto_delete: false, # Enable or disable auto-deletion of old contexts
context_ttl: :timer.hours(6), # Time-to-live for contexts
delta_ttl: :timer.hours(24), # Time-to-live for deltas
delta_threshold: 100, # Number of deltas to retain
delta_formatter: Chord.Delta.Formatter.Default, # Format for deltas
time_provider: Chord.Utils.Time, # Time provider for consistent timestamps
export_callback: nil, # Optional: Define a callback for exporting contexts
context_external_provider: nil # Optional: Define a function for fetching external contexts
```
---
## 🎹 How to Use Chord
### Setting a Context
```elixir
{:ok, result} = Chord.set_context("user:123", %{status: "online"})
IO.inspect(result, label: "Context Set")
```
### Updating a Context
```elixir
{:ok, result} = Chord.update_context("user:123", %{status: "away"})
IO.inspect(result, label: "Context Updated")
```
### Synchronizing State
```elixir
case Chord.sync_context("user:123", nil) do
{:full_context, context} -> IO.inspect(context, label: "Full Context")
{:delta, delta} -> IO.inspect(delta, label: "Delta Update")
{:no_change, version} -> IO.puts("No changes for version #{version}")
end
```
### Exporting a Context
```elixir
:ok = Chord.export_context("user:123")
```
### Restoring a Context
```elixir
{:ok, restored_context} = Chord.restore_context("user:123")
IO.inspect(restored_context, label: "Restored Context")
```
### Cleanup Operations
Run periodic cleanup to remove stale data:
```elixir
Chord.cleanup(limit: 50)
```
### Managing the Cleanup Server
Start and manage the Cleanup Server for automated periodic cleanup:
```elixir
{:ok, _pid} = Chord.start_cleanup_server(interval: :timer.minutes(30))
Chord.update_cleanup_interval(:timer.minutes(60))
Chord.update_cleanup_backend_opts(limit: 100)
Chord.stop_cleanup_server()
```
---
## 🛠️ Customization
### Backends
Chord supports multiple backends out-of-the-box:
- **ETS** (In-Memory)
- **Redis** (Distributed)
You can implement your own backend by adhering to the `Chord.Backend.Behaviour`.
### Delta Formatters
Customize how deltas are structured by implementing the `Chord.Delta.Formatter` behaviour.
## ⚡ Features at a Glance
| Feature | Description |
|--------------------------|--------------------------------------------------|
| **Real-Time Sync** | Delta-based and full-context synchronization. |
| **Customizable Backends**| Redis, ETS, or your own custom backend. |
| **Periodic Cleanup** | Automatically remove stale data. |
| **Partial Updates** | Update only specific fields in a context. |
| **Delta Tracking** | Efficiently track and retrieve state changes. |
| **Context Export** | Export context to external storage. |
| **Context Restore** | Restore context from external providers. |
---
## 🕰 Benchmark Results
Chord has been benchmarked using Redis and ETS backends under both stateless and stateful architectures. Below are the results for various scenarios:
### Redis Benchmark Results
#### With Input Data
| Scenario | IPS | Avg. Time | Deviation | Median Time | 99th Percentile |
|------------------------------------------------------|--------|-----------|-----------|-------------|-----------------|
| Stateless - Single Context (50 participants) | 62.79 | 15.93 ms | ±18.03% | 15.88 ms | 22.99 ms |
| Stateful - Single Context (50 participants) | 7.69 | 130.07 ms | ±51.34% | 102.08 ms | 270.40 ms |
| Stateless - Multiple Contexts (100 contexts) | 2.08 | 481.38 ms | ±3.48% | 474.73 ms | 511.20 ms |
| Stateful - Multiple Contexts (100 contexts) | 1.85 | 541.19 ms | ±3.51% | 541.24 ms | 566.32 ms |
**Comparison:**
- Stateless - Single Context: 62.79x faster than Stateful - Multiple Contexts.
### ETS Benchmark Results
#### With Input Data
| Scenario | IPS | Avg. Time | Deviation | Median Time | 99th Percentile |
|------------------------------------------------------|--------|-----------|-----------|-------------|-----------------|
| Stateless - Single Context (50 participants) | 151.19 | 6.61 ms | ±20.47% | 6.46 ms | 14.43 ms |
| Stateful - Single Context (50 participants) | 29.56 | 33.82 ms | ±13.06% | 36.05 ms | 45.56 ms |
| Stateless - Multiple Contexts (100 contexts) | 3.51 | 284.85 ms | ±14.28% | 279.17 ms | 405.34 ms |
| Stateful - Multiple Contexts (100 contexts) | 3.73 | 268.05 ms | ±23.67% | 237.10 ms | 374.59 ms |
**Comparison:**
- Stateless - Single Context: 43.07x faster than Stateful - Multiple Contexts.
### Device Information
| Property | Value |
|----------------------------|--------------------------|
| **Operating System** | macOS |
| **CPU Information** | Apple M4 Pro |
| **Number of Cores** | 12 |
| **Available Memory** | 24 GB |
| **Elixir Version** | 1.17.3 |
| **Erlang Version** | 27.1.2 |
| **JIT Enabled** | True |
**Benchmark Suite Configuration:**
- **Warmup:** 2 seconds
- **Execution Time:** 5 seconds
- **Parallel:** 1
- **Inputs:** Data
---
## 🧰 Contributing
Contributions from the community are welcome to make Chord even better! Whether it's fixing bugs, improving documentation, or adding new features, your help is greatly appreciated.
### How to Contribute
1. Fork the repository.
2. Create a new branch for your changes.
3. Make your changes and test them thoroughly.
4. Submit a pull request with a clear description of your changes.
Feel free to open issues for discussion or if you need help. Together, we can build something amazing!
---
## 🛡️ Testing
Chord comes with a robust suite of tests to ensure reliability. Run tests with:
```bash
mix test
```
---
🎵 *"Let Chord orchestrate your state management with precision and elegance."*