|
|
@ -6,32 +6,36 @@ pub use error::Error;
|
|
|
|
use log::warn;
|
|
|
|
use log::warn;
|
|
|
|
use serde::Serialize;
|
|
|
|
use serde::Serialize;
|
|
|
|
use serde_json::Value as JsonValue;
|
|
|
|
use serde_json::Value as JsonValue;
|
|
|
|
use std::{collections::HashMap, path::PathBuf, sync::Mutex};
|
|
|
|
use std::{
|
|
|
|
|
|
|
|
collections::HashMap,
|
|
|
|
|
|
|
|
path::{Path, PathBuf},
|
|
|
|
|
|
|
|
sync::Mutex,
|
|
|
|
|
|
|
|
};
|
|
|
|
pub use store::{Store, StoreBuilder};
|
|
|
|
pub use store::{Store, StoreBuilder};
|
|
|
|
use tauri::{
|
|
|
|
use tauri::{
|
|
|
|
plugin::{self, TauriPlugin},
|
|
|
|
plugin::{self, TauriPlugin},
|
|
|
|
AppHandle, Manager, RunEvent, Runtime, State, Window,
|
|
|
|
AppHandle, Manager, RunEvent, Runtime, State,
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
mod error;
|
|
|
|
mod error;
|
|
|
|
mod store;
|
|
|
|
mod store;
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Serialize, Clone)]
|
|
|
|
#[derive(Serialize, Clone)]
|
|
|
|
struct ChangePayload {
|
|
|
|
struct ChangePayload<'a> {
|
|
|
|
path: PathBuf,
|
|
|
|
path: &'a Path,
|
|
|
|
key: String,
|
|
|
|
key: &'a str,
|
|
|
|
value: JsonValue,
|
|
|
|
value: &'a JsonValue,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
#[derive(Default)]
|
|
|
|
struct StoreCollection {
|
|
|
|
pub struct StoreCollection<R: Runtime> {
|
|
|
|
stores: Mutex<HashMap<PathBuf, Store>>,
|
|
|
|
stores: Mutex<HashMap<PathBuf, Store<R>>>,
|
|
|
|
frozen: bool,
|
|
|
|
frozen: bool,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn with_store<R: Runtime, T, F: FnOnce(&mut Store) -> Result<T, Error>>(
|
|
|
|
pub fn with_store<R: Runtime, T, F: FnOnce(&mut Store<R>) -> Result<T, Error>>(
|
|
|
|
app: &AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
collection: State<'_, StoreCollection>,
|
|
|
|
collection: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
f: F,
|
|
|
|
f: F,
|
|
|
|
) -> Result<T, Error> {
|
|
|
|
) -> Result<T, Error> {
|
|
|
@ -41,9 +45,9 @@ fn with_store<R: Runtime, T, F: FnOnce(&mut Store) -> Result<T, Error>>(
|
|
|
|
if collection.frozen {
|
|
|
|
if collection.frozen {
|
|
|
|
return Err(Error::NotFound(path));
|
|
|
|
return Err(Error::NotFound(path));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let mut store = StoreBuilder::new(path.clone()).build();
|
|
|
|
let mut store = StoreBuilder::new(app, path.clone()).build();
|
|
|
|
// ignore loading errors, just use the default
|
|
|
|
// ignore loading errors, just use the default
|
|
|
|
if let Err(err) = store.load(app) {
|
|
|
|
if let Err(err) = store.load() {
|
|
|
|
warn!(
|
|
|
|
warn!(
|
|
|
|
"Failed to load store {:?} from disk: {}. Falling back to default values.",
|
|
|
|
"Failed to load store {:?} from disk: {}. Falling back to default values.",
|
|
|
|
path, err
|
|
|
|
path, err
|
|
|
@ -60,196 +64,132 @@ fn with_store<R: Runtime, T, F: FnOnce(&mut Store) -> Result<T, Error>>(
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn set<R: Runtime>(
|
|
|
|
async fn set<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
window: Window<R>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
key: String,
|
|
|
|
key: String,
|
|
|
|
value: JsonValue,
|
|
|
|
value: JsonValue,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
with_store(&app, stores, path.clone(), |store| {
|
|
|
|
with_store(app, stores, path, |store| store.set(key, value))
|
|
|
|
store.cache.insert(key.clone(), value.clone());
|
|
|
|
|
|
|
|
let _ = window.emit("store://change", ChangePayload { path, key, value });
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn get<R: Runtime>(
|
|
|
|
async fn get<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
key: String,
|
|
|
|
key: String,
|
|
|
|
) -> Result<Option<JsonValue>, Error> {
|
|
|
|
) -> Result<Option<JsonValue>, Error> {
|
|
|
|
with_store(&app, stores, path, |store| {
|
|
|
|
with_store(app, stores, path, |store| Ok(store.get(key).cloned()))
|
|
|
|
Ok(store.cache.get(&key).cloned())
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn has<R: Runtime>(
|
|
|
|
async fn has<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
key: String,
|
|
|
|
key: String,
|
|
|
|
) -> Result<bool, Error> {
|
|
|
|
) -> Result<bool, Error> {
|
|
|
|
with_store(&app, stores, path, |store| {
|
|
|
|
with_store(app, stores, path, |store| Ok(store.has(key)))
|
|
|
|
Ok(store.cache.contains_key(&key))
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn delete<R: Runtime>(
|
|
|
|
async fn delete<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
window: Window<R>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
key: String,
|
|
|
|
key: String,
|
|
|
|
) -> Result<bool, Error> {
|
|
|
|
) -> Result<bool, Error> {
|
|
|
|
with_store(&app, stores, path.clone(), |store| {
|
|
|
|
with_store(app, stores, path, |store| store.delete(key))
|
|
|
|
let flag = store.cache.remove(&key).is_some();
|
|
|
|
|
|
|
|
if flag {
|
|
|
|
|
|
|
|
let _ = window.emit(
|
|
|
|
|
|
|
|
"store://change",
|
|
|
|
|
|
|
|
ChangePayload {
|
|
|
|
|
|
|
|
path,
|
|
|
|
|
|
|
|
key,
|
|
|
|
|
|
|
|
value: JsonValue::Null,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(flag)
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn clear<R: Runtime>(
|
|
|
|
async fn clear<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
window: Window<R>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
with_store(&app, stores, path.clone(), |store| {
|
|
|
|
with_store(app, stores, path, |store| store.clear())
|
|
|
|
let keys = store.cache.keys().cloned().collect::<Vec<String>>();
|
|
|
|
|
|
|
|
store.cache.clear();
|
|
|
|
|
|
|
|
for key in keys {
|
|
|
|
|
|
|
|
let _ = window.emit(
|
|
|
|
|
|
|
|
"store://change",
|
|
|
|
|
|
|
|
ChangePayload {
|
|
|
|
|
|
|
|
path: path.clone(),
|
|
|
|
|
|
|
|
key,
|
|
|
|
|
|
|
|
value: JsonValue::Null,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn reset<R: Runtime>(
|
|
|
|
async fn reset<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
window: Window<R>,
|
|
|
|
collection: State<'_, StoreCollection<R>>,
|
|
|
|
collection: State<'_, StoreCollection>,
|
|
|
|
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
let has_defaults = collection
|
|
|
|
with_store(app, collection, path, |store| store.reset())
|
|
|
|
.stores
|
|
|
|
|
|
|
|
.lock()
|
|
|
|
|
|
|
|
.expect("mutex poisoned")
|
|
|
|
|
|
|
|
.get(&path)
|
|
|
|
|
|
|
|
.map(|store| store.defaults.is_some());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if Some(true) == has_defaults {
|
|
|
|
|
|
|
|
with_store(&app, collection, path.clone(), |store| {
|
|
|
|
|
|
|
|
if let Some(defaults) = &store.defaults {
|
|
|
|
|
|
|
|
for (key, value) in &store.cache {
|
|
|
|
|
|
|
|
if defaults.get(key) != Some(value) {
|
|
|
|
|
|
|
|
let _ = window.emit(
|
|
|
|
|
|
|
|
"store://change",
|
|
|
|
|
|
|
|
ChangePayload {
|
|
|
|
|
|
|
|
path: path.clone(),
|
|
|
|
|
|
|
|
key: key.clone(),
|
|
|
|
|
|
|
|
value: defaults.get(key).cloned().unwrap_or(JsonValue::Null),
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
store.cache = defaults.clone();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
clear(app, window, collection, path).await
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn keys<R: Runtime>(
|
|
|
|
async fn keys<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
) -> Result<Vec<String>, Error> {
|
|
|
|
) -> Result<Vec<String>, Error> {
|
|
|
|
with_store(&app, stores, path, |store| {
|
|
|
|
with_store(app, stores, path, |store| {
|
|
|
|
Ok(store.cache.keys().cloned().collect())
|
|
|
|
Ok(store.keys().cloned().collect())
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn values<R: Runtime>(
|
|
|
|
async fn values<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
) -> Result<Vec<JsonValue>, Error> {
|
|
|
|
) -> Result<Vec<JsonValue>, Error> {
|
|
|
|
with_store(&app, stores, path, |store| {
|
|
|
|
with_store(app, stores, path, |store| {
|
|
|
|
Ok(store.cache.values().cloned().collect())
|
|
|
|
Ok(store.values().cloned().collect())
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn entries<R: Runtime>(
|
|
|
|
async fn entries<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
) -> Result<Vec<(String, JsonValue)>, Error> {
|
|
|
|
) -> Result<Vec<(String, JsonValue)>, Error> {
|
|
|
|
with_store(&app, stores, path, |store| {
|
|
|
|
with_store(app, stores, path, |store| {
|
|
|
|
Ok(store.cache.clone().into_iter().collect())
|
|
|
|
Ok(store
|
|
|
|
|
|
|
|
.entries()
|
|
|
|
|
|
|
|
.map(|(k, v)| (k.to_owned(), v.to_owned()))
|
|
|
|
|
|
|
|
.collect())
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn length<R: Runtime>(
|
|
|
|
async fn length<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
) -> Result<usize, Error> {
|
|
|
|
) -> Result<usize, Error> {
|
|
|
|
with_store(&app, stores, path, |store| Ok(store.cache.len()))
|
|
|
|
with_store(app, stores, path, |store| Ok(store.len()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn load<R: Runtime>(
|
|
|
|
async fn load<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
with_store(&app, stores, path, |store| store.load(&app))
|
|
|
|
with_store(app, stores, path, |store| store.load())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tauri::command]
|
|
|
|
#[tauri::command]
|
|
|
|
async fn save<R: Runtime>(
|
|
|
|
async fn save<R: Runtime>(
|
|
|
|
app: AppHandle<R>,
|
|
|
|
app: AppHandle<R>,
|
|
|
|
stores: State<'_, StoreCollection>,
|
|
|
|
stores: State<'_, StoreCollection<R>>,
|
|
|
|
path: PathBuf,
|
|
|
|
path: PathBuf,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
with_store(&app, stores, path, |store| store.save(&app))
|
|
|
|
with_store(app, stores, path, |store| store.save())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct Builder {
|
|
|
|
pub struct Builder<R: Runtime> {
|
|
|
|
stores: HashMap<PathBuf, Store>,
|
|
|
|
stores: HashMap<PathBuf, Store<R>>,
|
|
|
|
frozen: bool,
|
|
|
|
frozen: bool,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
impl Builder {
|
|
|
|
impl<R: Runtime> Builder<R> {
|
|
|
|
/// Registers a store with the plugin.
|
|
|
|
/// Registers a store with the plugin.
|
|
|
|
///
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
/// # Examples
|
|
|
@ -265,7 +205,7 @@ impl Builder {
|
|
|
|
/// # Ok(())
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
/// ```
|
|
|
|
pub fn store(mut self, store: Store) -> Self {
|
|
|
|
pub fn store(mut self, store: Store<R>) -> Self {
|
|
|
|
self.stores.insert(store.path.clone(), store);
|
|
|
|
self.stores.insert(store.path.clone(), store);
|
|
|
|
self
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -285,7 +225,7 @@ impl Builder {
|
|
|
|
/// # Ok(())
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
/// ```
|
|
|
|
pub fn stores<T: IntoIterator<Item = Store>>(mut self, stores: T) -> Self {
|
|
|
|
pub fn stores<T: IntoIterator<Item = Store<R>>>(mut self, stores: T) -> Self {
|
|
|
|
self.stores = stores
|
|
|
|
self.stores = stores
|
|
|
|
.into_iter()
|
|
|
|
.into_iter()
|
|
|
|
.map(|store| (store.path.clone(), store))
|
|
|
|
.map(|store| (store.path.clone(), store))
|
|
|
@ -331,7 +271,7 @@ impl Builder {
|
|
|
|
/// # Ok(())
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
/// ```
|
|
|
|
pub fn build<R: Runtime>(mut self) -> TauriPlugin<R> {
|
|
|
|
pub fn build(mut self) -> TauriPlugin<R> {
|
|
|
|
plugin::Builder::new("store")
|
|
|
|
plugin::Builder::new("store")
|
|
|
|
.invoke_handler(tauri::generate_handler![
|
|
|
|
.invoke_handler(tauri::generate_handler![
|
|
|
|
set, get, has, delete, clear, reset, keys, values, length, entries, load, save
|
|
|
|
set, get, has, delete, clear, reset, keys, values, length, entries, load, save
|
|
|
@ -339,7 +279,7 @@ impl Builder {
|
|
|
|
.setup(move |app_handle| {
|
|
|
|
.setup(move |app_handle| {
|
|
|
|
for (path, store) in self.stores.iter_mut() {
|
|
|
|
for (path, store) in self.stores.iter_mut() {
|
|
|
|
// ignore loading errors, just use the default
|
|
|
|
// ignore loading errors, just use the default
|
|
|
|
if let Err(err) = store.load(app_handle) {
|
|
|
|
if let Err(err) = store.load() {
|
|
|
|
warn!(
|
|
|
|
warn!(
|
|
|
|
"Failed to load store {:?} from disk: {}. Falling back to default values.",
|
|
|
|
"Failed to load store {:?} from disk: {}. Falling back to default values.",
|
|
|
|
path, err
|
|
|
|
path, err
|
|
|
@ -356,10 +296,10 @@ impl Builder {
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.on_event(|app_handle, event| {
|
|
|
|
.on_event(|app_handle, event| {
|
|
|
|
if let RunEvent::Exit = event {
|
|
|
|
if let RunEvent::Exit = event {
|
|
|
|
let collection = app_handle.state::<StoreCollection>();
|
|
|
|
let collection = app_handle.state::<StoreCollection<R>>();
|
|
|
|
|
|
|
|
|
|
|
|
for store in collection.stores.lock().expect("mutex poisoned").values() {
|
|
|
|
for store in collection.stores.lock().expect("mutex poisoned").values() {
|
|
|
|
if let Err(err) = store.save(app_handle) {
|
|
|
|
if let Err(err) = store.save() {
|
|
|
|
eprintln!("failed to save store {:?} with error {:?}", store.path, err);
|
|
|
|
eprintln!("failed to save store {:?} with error {:?}", store.path, err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|