diff options
author | Vika <vika@fireburn.ru> | 2021-09-26 01:35:00 +0300 |
---|---|---|
committer | Vika <vika@fireburn.ru> | 2021-09-26 02:06:55 +0300 |
commit | 572435f7c8e1d613983309eca268c3f87ec5f00f (patch) | |
tree | e0c134c994379e037928f4e20e95f5fbc0bfc0da /src/database/file/mod.rs | |
parent | 13bcfb013c4a5ac5ea15c7ebe04f243431165c03 (diff) | |
download | kittybox-572435f7c8e1d613983309eca268c3f87ec5f00f.tar.zst |
Added file updates and fixed a bug with truncated JSON files
There was a bug where `File::write()` would not write the entire buffer, and this condition was left unchecked by the code. All `File::write()` calls are now replaced with `File::write_all()` which ensures the whole buffer is written to the backing file. Additionally added a smoke check for the file updates. It is in no way comprehensive nor it is able to catch all the possible failures but it's a good way of testing the functionality without way too much hassle.
Diffstat (limited to 'src/database/file/mod.rs')
-rw-r--r-- | src/database/file/mod.rs | 103 |
1 files changed, 100 insertions, 3 deletions
diff --git a/src/database/file/mod.rs b/src/database/file/mod.rs index 36300fc..ff5ad13 100644 --- a/src/database/file/mod.rs +++ b/src/database/file/mod.rs @@ -1,4 +1,3 @@ -//pub mod async_file_ext; use async_std::fs::{File, OpenOptions}; use async_std::io::{ErrorKind as IOErrorKind, BufReader}; use async_std::io::prelude::*; @@ -8,6 +7,7 @@ use crate::database::{ErrorKind, Result, Storage, StorageError}; use fd_lock::RwLock; use log::debug; use std::path::{Path, PathBuf}; +use std::collections::HashMap; impl From<std::io::Error> for StorageError { fn from(source: std::io::Error) -> Self { @@ -35,6 +35,79 @@ fn url_to_path(root: &Path, url: &str) -> PathBuf { path } +fn modify_post(post: &serde_json::Value, update: &serde_json::Value) -> Result<serde_json::Value> { + let mut add_keys: HashMap<String, serde_json::Value> = HashMap::new(); + let mut remove_keys: Vec<String> = vec![]; + let mut remove_values: HashMap<String, Vec<serde_json::Value>> = HashMap::new(); + let mut post = post.clone(); + + if let Some(delete) = update["delete"].as_array() { + remove_keys.extend(delete.iter().filter_map(|v| v.as_str()).map(|v| v.to_string())); + } else if let Some(delete) = update["delete"].as_object() { + for (k, v) in delete { + if let Some(v) = v.as_array() { + remove_values.entry(k.to_string()).or_default().extend(v.clone()); + } else { + return Err(StorageError::new(ErrorKind::BadRequest, "Malformed update object")); + } + } + } + if let Some(add) = update["add"].as_object() { + for (k, v) in add { + if v.is_array() { + add_keys.insert(k.to_string(), v.clone()); + } else { + return Err(StorageError::new(ErrorKind::BadRequest, "Malformed update object")); + } + } + } + if let Some(replace) = update["replace"].as_object() { + for (k, v) in replace { + remove_keys.push(k.to_string()); + add_keys.insert(k.to_string(), v.clone()); + } + } + + for k in remove_keys { + post["properties"].as_object_mut().unwrap().remove(&k); + } + for (k, v) in remove_values { + let k = &k; + let props; + if k == "children" { + props = &mut post; + } else { + props = &mut post["properties"]; + } + v.iter().for_each(|v| { + if let Some(vec) = props[k].as_array_mut() { + if let Some(index) = vec.iter().position(|w| w == v) { + vec.remove(index); + } + } + }); + } + for (k, v) in add_keys { + let props; + if k == "children" { + props = &mut post; + } else { + props = &mut post["properties"]; + } + let k = &k; + if let Some(prop) = props[k].as_array_mut() { + if k == "children" { + v.as_array().unwrap().iter().cloned().rev().for_each(|v| prop.insert(0, v)); + } else { + prop.extend(v.as_array().unwrap().iter().cloned()); + } + } else { + post["properties"][k] = v + } + } + Ok(post) +} + #[derive(Clone)] pub struct FileStorage { root_dir: PathBuf, @@ -113,7 +186,8 @@ impl Storage for FileStorage { let mut lock = get_lockable_file(f).await; let mut guard = lock.write()?; - (*guard).write(post.to_string().as_bytes()).await?; + (*guard).write_all(post.to_string().as_bytes()).await?; + (*guard).flush().await?; drop(guard); if post["properties"]["url"].is_array() { @@ -139,7 +213,30 @@ impl Storage for FileStorage { } async fn update_post<'a>(&self, url: &'a str, update: serde_json::Value) -> Result<()> { - todo!() + let path = url_to_path(&self.root_dir, url); + let f = OpenOptions::new() + .write(true) + .read(true) + .truncate(false) + .open(&path) + .await?; + + let mut lock = get_lockable_file(f).await; + let mut guard = lock.write()?; + + let mut content = String::new(); + guard.read_to_string(&mut content).await?; + let json: serde_json::Value = serde_json::from_str(&content)?; + // Apply the editing algorithms + let new_json = modify_post(&json, &update)?; + + (*guard).set_len(0).await?; + (*guard).seek(std::io::SeekFrom::Start(0)).await?; + (*guard).write_all(new_json.to_string().as_bytes()).await?; + (*guard).flush().await?; + drop(guard); + // TODO check if URLs changed between old and new JSON + Ok(()) } async fn get_channels( |