kvp.rs

  1use gpui::App;
  2use sqlez_macros::sql;
  3use util::ResultExt as _;
  4
  5use crate::{define_connection, query, write_and_log};
  6
  7define_connection!(pub static ref KEY_VALUE_STORE: KeyValueStore<()> =
  8    &[sql!(
  9        CREATE TABLE IF NOT EXISTS kv_store(
 10            key TEXT PRIMARY KEY,
 11            value TEXT NOT NULL
 12        ) STRICT;
 13    )];
 14);
 15
 16pub trait Dismissable {
 17    const KEY: &'static str;
 18
 19    fn dismissed() -> bool {
 20        KEY_VALUE_STORE
 21            .read_kvp(Self::KEY)
 22            .log_err()
 23            .is_some_and(|s| s.is_some())
 24    }
 25
 26    fn set_dismissed(is_dismissed: bool, cx: &mut App) {
 27        write_and_log(cx, move || async move {
 28            if is_dismissed {
 29                KEY_VALUE_STORE
 30                    .write_kvp(Self::KEY.into(), "1".into())
 31                    .await
 32            } else {
 33                KEY_VALUE_STORE.delete_kvp(Self::KEY.into()).await
 34            }
 35        })
 36    }
 37}
 38
 39impl KeyValueStore {
 40    query! {
 41        pub fn read_kvp(key: &str) -> Result<Option<String>> {
 42            SELECT value FROM kv_store WHERE key = (?)
 43        }
 44    }
 45
 46    pub async fn write_kvp(&self, key: String, value: String) -> anyhow::Result<()> {
 47        log::debug!("Writing key-value pair for key {key}");
 48        self.write_kvp_inner(key, value).await
 49    }
 50
 51    query! {
 52        async fn write_kvp_inner(key: String, value: String) -> Result<()> {
 53            INSERT OR REPLACE INTO kv_store(key, value) VALUES ((?), (?))
 54        }
 55    }
 56
 57    query! {
 58        pub async fn delete_kvp(key: String) -> Result<()> {
 59            DELETE FROM kv_store WHERE key = (?)
 60        }
 61    }
 62}
 63
 64#[cfg(test)]
 65mod tests {
 66    use crate::kvp::KeyValueStore;
 67
 68    #[gpui::test]
 69    async fn test_kvp() {
 70        let db = KeyValueStore::open_test_db("test_kvp").await;
 71
 72        assert_eq!(db.read_kvp("key-1").unwrap(), None);
 73
 74        db.write_kvp("key-1".to_string(), "one".to_string())
 75            .await
 76            .unwrap();
 77        assert_eq!(db.read_kvp("key-1").unwrap(), Some("one".to_string()));
 78
 79        db.write_kvp("key-1".to_string(), "one-2".to_string())
 80            .await
 81            .unwrap();
 82        assert_eq!(db.read_kvp("key-1").unwrap(), Some("one-2".to_string()));
 83
 84        db.write_kvp("key-2".to_string(), "two".to_string())
 85            .await
 86            .unwrap();
 87        assert_eq!(db.read_kvp("key-2").unwrap(), Some("two".to_string()));
 88
 89        db.delete_kvp("key-1".to_string()).await.unwrap();
 90        assert_eq!(db.read_kvp("key-1").unwrap(), None);
 91    }
 92}
 93
 94define_connection!(pub static ref GLOBAL_KEY_VALUE_STORE: GlobalKeyValueStore<()> =
 95    &[sql!(
 96        CREATE TABLE IF NOT EXISTS kv_store(
 97            key TEXT PRIMARY KEY,
 98            value TEXT NOT NULL
 99        ) STRICT;
100    )];
101    global
102);
103
104impl GlobalKeyValueStore {
105    query! {
106        pub fn read_kvp(key: &str) -> Result<Option<String>> {
107            SELECT value FROM kv_store WHERE key = (?)
108        }
109    }
110
111    pub async fn write_kvp(&self, key: String, value: String) -> anyhow::Result<()> {
112        log::debug!("Writing global key-value pair for key {key}");
113        self.write_kvp_inner(key, value).await
114    }
115
116    query! {
117        async fn write_kvp_inner(key: String, value: String) -> Result<()> {
118            INSERT OR REPLACE INTO kv_store(key, value) VALUES ((?), (?))
119        }
120    }
121
122    query! {
123        pub async fn delete_kvp(key: String) -> Result<()> {
124            DELETE FROM kv_store WHERE key = (?)
125        }
126    }
127}