2024-08-05 16:10:25 +02:00
|
|
|
//! SQLite-backed key/value store, primarily intended as a config file (few writes).
|
|
|
|
//!
|
2024-08-06 18:05:26 +02:00
|
|
|
//! ```ignore
|
2024-08-05 16:10:25 +02:00
|
|
|
//! // Open a new or existing DB (no distinction made.)
|
|
|
|
//! let cfg = KeyValueStore::new("config.sqlite")?;
|
2024-08-06 18:05:26 +02:00
|
|
|
//! ```
|
|
|
|
//!
|
|
|
|
//! ```
|
|
|
|
//! # let cfg = KeyValueStore::new_dummy()?;
|
2024-08-05 16:10:25 +02:00
|
|
|
//! // Ensure defaults exist. (Do this early, this function panics on error.)
|
|
|
|
//! cfg.default( "foo", "bar" );
|
|
|
|
//! cfg.default( "baz", "quux" );
|
|
|
|
//! // Normal use after that.
|
|
|
|
//! let foo = cfg.get( "foo" ).unwrap();
|
|
|
|
//! let baz = &cfg["baz"]; // shorthand that LEAKS the string and panics on error
|
|
|
|
//! let asdf = cfg.get( "asdf" ).unwrap_or_default( "abc" );
|
|
|
|
//! cfg.set( "fnord", "23" )?;
|
|
|
|
//! cfg.delete( "foo" )?;
|
|
|
|
//! // If any writes failed, this flag will be set and the data got logged to stderr.
|
|
|
|
//! let all_ok = !cfg.has_errors();
|
|
|
|
//! ```
|
|
|
|
|
2024-08-06 18:05:26 +02:00
|
|
|
use colored::Colorize;
|
|
|
|
use rusqlite::{params, Connection, Result};
|
2024-07-27 15:16:44 +02:00
|
|
|
use std::cell::Cell;
|
|
|
|
use std::collections::HashSet;
|
2024-08-02 15:17:13 +02:00
|
|
|
use std::ops::Index;
|
2024-07-21 00:44:36 +02:00
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
/// Simple SQLite-backed key/value store.
|
|
|
|
///
|
|
|
|
/// All failing writes will log a message, any errors will set `has_errors`.
|
|
|
|
///
|
|
|
|
/// There's `Index` support for convenience (`&kvstore["foo"]`),
|
|
|
|
/// but **it will leak data.**
|
|
|
|
/// (That's perfectly fine when used for getting data once for the duration of
|
|
|
|
/// the program, just use `.get()` for stuff that changes.)
|
2024-07-21 00:44:36 +02:00
|
|
|
pub struct KeyValueStore {
|
|
|
|
conn: Connection,
|
2024-07-27 15:16:44 +02:00
|
|
|
has_write_errors: Cell<bool>,
|
2024-07-21 00:44:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl KeyValueStore {
|
2024-07-27 15:16:44 +02:00
|
|
|
/// Open or create a DB with a `kv_store` table storing all tuples.
|
2024-07-21 00:44:36 +02:00
|
|
|
pub fn new(db_path: &str) -> Result<Self> {
|
|
|
|
let conn = Connection::open(db_path)?;
|
2024-08-06 18:05:26 +02:00
|
|
|
KeyValueStore::setup(conn)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Mostly for tests, make a temporary DB in memory.
|
|
|
|
pub fn new_dummy() -> Result<Self> {
|
|
|
|
let conn = Connection::open_in_memory()?;
|
|
|
|
KeyValueStore::setup(conn)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Actual constructor.
|
|
|
|
fn setup(conn: Connection) -> Result<Self> {
|
2024-07-21 00:44:36 +02:00
|
|
|
conn.execute(
|
|
|
|
"CREATE TABLE IF NOT EXISTS kv_store (
|
|
|
|
key TEXT PRIMARY KEY,
|
|
|
|
value TEXT NOT NULL
|
|
|
|
)",
|
|
|
|
[],
|
|
|
|
)?;
|
2024-08-02 15:17:13 +02:00
|
|
|
Ok(Self { conn, has_write_errors: Cell::new(false) })
|
2024-07-21 00:44:36 +02:00
|
|
|
}
|
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
/// Report if any write errors occurred.
|
|
|
|
pub fn has_errors(&self) -> bool {
|
|
|
|
self.has_write_errors.get()
|
2024-07-21 00:44:36 +02:00
|
|
|
}
|
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
/// Get value at `key`. (Use `.unwrap_or(default)` & friends to work with the Result.)
|
|
|
|
pub fn get(&self, key: &str) -> Result<String> {
|
2024-07-21 00:44:36 +02:00
|
|
|
let mut stmt = self.conn.prepare("SELECT value FROM kv_store WHERE key = ?1")?;
|
|
|
|
let mut rows = stmt.query(params![key])?;
|
|
|
|
|
|
|
|
if let Some(row) = rows.next()? {
|
2024-07-27 15:16:44 +02:00
|
|
|
let result = row.get(0)?;
|
|
|
|
Ok(result)
|
2024-07-21 00:44:36 +02:00
|
|
|
} else {
|
2024-07-27 15:16:44 +02:00
|
|
|
Err(rusqlite::Error::QueryReturnedNoRows)
|
2024-07-21 00:44:36 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
/// If `key` is absent from the DB, write `key`/`value` pair (else leave unchanged).
|
|
|
|
///
|
|
|
|
/// Will panic on error. (The idea is to call this a bunch of times at boot,
|
|
|
|
/// to populate the DB if missing / newly created, and not use it after.)
|
|
|
|
pub fn default(&self, key: &str, value: &str) {
|
2024-08-02 15:17:13 +02:00
|
|
|
self.conn
|
|
|
|
.execute(
|
|
|
|
"INSERT INTO kv_store (key, value) VALUES (?1, ?2)
|
2024-07-27 15:16:44 +02:00
|
|
|
ON CONFLICT(key) DO NOTHING",
|
2024-08-02 15:17:13 +02:00
|
|
|
params![key, value],
|
|
|
|
)
|
|
|
|
.expect(&format!("Failed to write default at key: {}", key));
|
2024-07-21 00:44:36 +02:00
|
|
|
}
|
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
/// Write a `key`/`value` pair to the DB.
|
|
|
|
///
|
|
|
|
/// Errors log the failed write and set `has_errors`.
|
|
|
|
pub fn set(&self, key: &str, value: &str) -> Result<()> {
|
|
|
|
match self.conn.execute(
|
|
|
|
"INSERT INTO kv_store (key, value) VALUES (?1, ?2)
|
|
|
|
ON CONFLICT(key) DO UPDATE SET value = excluded.value",
|
|
|
|
params![key, value],
|
|
|
|
) {
|
|
|
|
Ok(_) => Ok(()),
|
|
|
|
Err(e) => {
|
|
|
|
eprintln!("Failed DB write: ({}, {})", key, value);
|
|
|
|
self.has_write_errors.set(true);
|
|
|
|
Err(e)
|
2024-08-02 15:17:13 +02:00
|
|
|
},
|
2024-07-27 15:16:44 +02:00
|
|
|
}
|
|
|
|
}
|
2024-07-21 00:44:36 +02:00
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
/// Delete the key-value pair associated with `key`.
|
|
|
|
///
|
|
|
|
/// Errors log the failed write and set `has_errors`.
|
|
|
|
pub fn delete(&self, key: &str) -> Result<()> {
|
|
|
|
match self.conn.execute("DELETE FROM kv_store WHERE key = ?1", params![key]) {
|
|
|
|
Ok(_) => Ok(()),
|
|
|
|
Err(e) => {
|
|
|
|
eprintln!("Failed DB write: ({}, NULL)", key);
|
|
|
|
self.has_write_errors.set(true);
|
|
|
|
Err(e)
|
2024-08-02 15:17:13 +02:00
|
|
|
},
|
2024-07-27 15:16:44 +02:00
|
|
|
}
|
2024-07-21 00:44:36 +02:00
|
|
|
}
|
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
/// Dump the current DB contents to stderr.
|
|
|
|
///
|
|
|
|
/// For all keys listed in `redacted_keys`, no value is printed, only the presence is reported.
|
|
|
|
pub fn dump_redacting(&self, redacted_keys: &[&str]) -> Result<()> {
|
|
|
|
let exclude_set: HashSet<&str> = redacted_keys.iter().cloned().collect();
|
|
|
|
let mut stmt = self.conn.prepare("SELECT key, value FROM kv_store")?;
|
|
|
|
let mut rows = stmt.query(params![])?;
|
|
|
|
while let Some(row) = rows.next()? {
|
|
|
|
let key: String = row.get(0)?;
|
|
|
|
if exclude_set.contains(key.as_str()) {
|
2024-08-05 19:57:12 +02:00
|
|
|
eprintln!("{}{}", key, " = REDACTED".blue());
|
2024-07-27 15:16:44 +02:00
|
|
|
} else {
|
|
|
|
let value: String = row.get(1)?;
|
2024-08-05 19:57:12 +02:00
|
|
|
eprintln!("{}{}{}", key, " = ".blue(), value.blue());
|
2024-07-27 15:16:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
2024-07-21 00:44:36 +02:00
|
|
|
}
|
2024-07-27 15:16:44 +02:00
|
|
|
}
|
2024-07-21 00:44:36 +02:00
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
impl Index<&str> for KeyValueStore {
|
|
|
|
type Output = str;
|
2024-07-21 00:44:36 +02:00
|
|
|
|
2024-07-27 15:16:44 +02:00
|
|
|
fn index(&self, key: &str) -> &Self::Output {
|
|
|
|
Box::leak(Box::new(self.get(key).unwrap())).as_str()
|
|
|
|
}
|
2024-07-21 00:44:36 +02:00
|
|
|
}
|