1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
//! Embedded-store configuration. Builder-style — every knob has a sane
//! default so `Config::default()` works for the simplest use case
//! (in-memory, no persistence, background TTL reaper).
use std::path::PathBuf;
use std::time::Duration;
pub use kevy_persist::Fsync as AppendFsync;
pub use kevy_store::EvictionPolicy;
/// How the active TTL reaper runs.
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum TtlReaperMode {
/// Spawn a background thread that ticks at the configured interval
/// (default 100 ms / 10 Hz, matching Redis's `hz=10`). Default.
Background,
/// Caller-driven via [`crate::Store::tick`]. Required for WASM
/// targets (no threads) and single-threaded apps that don't want a
/// background worker.
Manual,
}
/// Embedded-store config. Build by chaining `with_*` methods on
/// [`Config::default`].
#[derive(Debug, Clone)]
pub struct Config {
/// Soft memory ceiling in bytes. `0` (default) = unlimited.
pub maxmemory: u64,
/// Eviction policy when over `maxmemory`. Default `NoEviction`.
pub eviction_policy: EvictionPolicy,
/// Persistence directory. `None` = pure in-memory (no AOF, no snapshot).
pub data_dir: Option<PathBuf>,
/// AOF on/off when `data_dir` is set. Defaults to `true` (on) when
/// `with_persist` was called; ignored if `data_dir` is `None`.
pub aof: bool,
/// AOF fsync policy. Default `EverySec` (matches Redis: ≤ 1 s loss).
pub appendfsync: AppendFsync,
/// Snapshot file name inside `data_dir`. Default `"dump-0.rdb"`.
pub snapshot_filename: String,
/// AOF file name inside `data_dir`. Default `"aof-0.aof"`.
pub aof_filename: String,
/// TTL reaper mode. Default `Background`.
pub ttl_reaper: TtlReaperMode,
/// Reaper tick interval. Default 100 ms (10 Hz).
pub reaper_interval: Duration,
/// `tick_expire` samples per round. Default 20 (matches Redis).
pub reaper_samples: usize,
/// Max sample rounds per tick. Default 16.
pub reaper_max_rounds: u32,
/// Auto-`BGREWRITEAOF` trigger: rewrite when the live AOF has grown by at
/// least this percent over its size at the previous rewrite. `0` disables
/// (call [`crate::Store::rewrite_aof`] manually). Default `100` (Redis).
pub auto_aof_rewrite_pct: u32,
/// Floor below which auto-rewrite is skipped. Default `64 MiB` (Redis).
pub auto_aof_rewrite_min_size: u64,
/// Optional push-style metric callback (replay / rewrite events). Default
/// `None`. Set via [`Self::with_metric_sink`]; not part of `Debug` output.
pub(crate) metric_sink: Option<crate::metric::MetricSink>,
/// Keyspace shard count (`hash(key) % shards`), each a fully independent
/// lock + keyspace + AOF (shared-nothing) — concurrent access scales across
/// cores. **Default `1`** (single shard = the original single-lock /
/// single-`aof-0.aof` layout, zero migration). Set `> 1` via
/// [`Self::with_shards`]; the first open with `> 1` re-shards an existing
/// single AOF into per-shard files.
pub shards: usize,
}
impl Default for Config {
fn default() -> Self {
Self {
maxmemory: 0,
eviction_policy: EvictionPolicy::NoEviction,
data_dir: None,
aof: true,
appendfsync: AppendFsync::EverySec,
snapshot_filename: String::from("dump-0.rdb"),
aof_filename: String::from("aof-0.aof"),
ttl_reaper: TtlReaperMode::Background,
reaper_interval: Duration::from_millis(100),
reaper_samples: 20,
reaper_max_rounds: 16,
auto_aof_rewrite_pct: 100,
auto_aof_rewrite_min_size: 64 * 1024 * 1024,
metric_sink: None,
shards: 1,
}
}
}
impl Config {
/// Enable persistence under `dir` — snapshot file + AOF land inside.
/// AOF defaults on; turn it off with [`Self::without_aof`] for pure
/// snapshot-only durability.
pub fn with_persist(mut self, dir: impl Into<PathBuf>) -> Self {
self.data_dir = Some(dir.into());
self
}
/// Disable the AOF (snapshot-only persistence — explicit `save_snapshot`
/// calls are the only way data survives restart).
pub fn without_aof(mut self) -> Self {
self.aof = false;
self
}
/// Soft memory ceiling in bytes. `0` keeps the default (unlimited).
pub fn with_max_memory(mut self, bytes: u64) -> Self {
self.maxmemory = bytes;
self
}
/// Eviction policy when over [`Self::with_max_memory`].
pub fn with_eviction(mut self, policy: EvictionPolicy) -> Self {
self.eviction_policy = policy;
self
}
/// AOF fsync policy. Default [`AppendFsync::EverySec`].
pub fn with_appendfsync(mut self, fsync: AppendFsync) -> Self {
self.appendfsync = fsync;
self
}
/// Auto-`BGREWRITEAOF` thresholds: rewrite once the AOF has grown `pct`
/// percent past its size at the last rewrite AND is at least `min_size`
/// bytes. In `Background` reaper mode the check runs on the reaper tick;
/// in `Manual` mode it runs when you call [`crate::Store::tick`]. Pass
/// `pct = 0` to disable auto-rewrite (you can still call
/// [`crate::Store::rewrite_aof`] yourself). Defaults: 100 % / 64 MiB.
pub fn with_auto_aof_rewrite(mut self, pct: u32, min_size: u64) -> Self {
self.auto_aof_rewrite_pct = pct;
self.auto_aof_rewrite_min_size = min_size;
self
}
/// Shard the keyspace into `n` shared-nothing partitions (`hash(key) % n`),
/// each with its own lock + keyspace + AOF, so concurrent access scales
/// across cores. `n` clamps to ≥ 1; `1` (default) is the original
/// single-shard layout. Going from a single-AOF store to `n > 1`
/// re-shards the existing `aof-0.aof` into `aof-0..aof-{n-1}` on the next
/// open (the old file is backed up to `aof-0.aof.premigration.<ts>` first).
/// Pub/sub is process-wide (handled on shard 0), not sharded.
pub fn with_shards(mut self, n: usize) -> Self {
self.shards = n.max(1);
self
}
/// Register a push-style metric callback. It receives a [`crate::KevyMetric`] for
/// each AOF replay (startup) and AOF rewrite (compaction) — wire it to
/// Prometheus / a log line / a counter. The callback runs synchronously on
/// the emitting thread (reaper thread for background rewrites), so keep it
/// fast and non-blocking. Replaces any previously-set sink.
pub fn with_metric_sink(
mut self,
sink: impl Fn(crate::KevyMetric) + Send + Sync + 'static,
) -> Self {
self.metric_sink = Some(crate::metric::MetricSink::new(sink));
self
}
/// Caller-driven TTL reaping — disables the background thread.
/// Required for WASM (no threads available). Call
/// [`crate::Store::tick`] yourself from your event loop.
pub fn with_ttl_reaper_manual(mut self) -> Self {
self.ttl_reaper = TtlReaperMode::Manual;
self
}
/// Override the background reaper interval. Default 100 ms.
pub fn with_reaper_interval(mut self, iv: Duration) -> Self {
self.reaper_interval = iv;
self
}
/// Override the snapshot file name inside `data_dir`.
pub fn with_snapshot_filename(mut self, name: impl Into<String>) -> Self {
self.snapshot_filename = name.into();
self
}
/// Override the AOF file name inside `data_dir`.
pub fn with_aof_filename(mut self, name: impl Into<String>) -> Self {
self.aof_filename = name.into();
self
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn default_is_pure_in_memory() {
let c = Config::default();
assert_eq!(c.maxmemory, 0);
assert!(c.data_dir.is_none());
assert_eq!(c.ttl_reaper, TtlReaperMode::Background);
assert!(c.aof);
}
#[test]
fn builder_chains() {
let c = Config::default()
.with_persist("/tmp/foo")
.with_max_memory(1024)
.with_eviction(EvictionPolicy::AllKeysLru)
.with_ttl_reaper_manual()
.with_appendfsync(AppendFsync::Always);
assert_eq!(c.data_dir.as_deref(), Some(std::path::Path::new("/tmp/foo")));
assert_eq!(c.maxmemory, 1024);
assert_eq!(c.eviction_policy, EvictionPolicy::AllKeysLru);
assert_eq!(c.ttl_reaper, TtlReaperMode::Manual);
}
#[test]
fn without_aof_disables_logging_path() {
let c = Config::default().with_persist("/tmp/foo").without_aof();
assert!(c.data_dir.is_some());
assert!(!c.aof);
}
}