use std::borrow::Cow;
use std::path::PathBuf;
use crate::store::{BlobEntry, BlobStoreInterface, BlobTable, BlobTableInterface, StoreError};
use directories::ProjectDirs;
use log::*;
use percent_encoding::{percent_decode_str, utf8_percent_encode, AsciiSet, CONTROLS};
use tokio::fs;
const ESCAPE_CHARS: &AsciiSet = &CONTROLS
.add(b' ')
.add(b'"')
.add(b'<')
.add(b'>')
.add(b'`')
.add(b'\'')
.add(b'/');
#[derive(Clone, Debug)]
pub struct BlobStoreFS {
basedir: PathBuf,
}
impl BlobStoreFS {
pub async fn new(basedir: PathBuf) -> Result<BlobStoreFS, StoreError> {
if !fs::try_exists(&basedir).await? {
fs::create_dir_all(&basedir).await?;
}
Ok(BlobStoreFS { basedir })
}
pub async fn from_standards(dirname: &str) -> Result<BlobStoreFS, StoreError> {
let standards_dir =
ProjectDirs::from("rs.xmpp", "", dirname).expect("Failed to create project directory");
let basedir = standards_dir.data_dir();
info!("Using {} for data store.", basedir.display());
Self::new(basedir.to_path_buf()).await
}
#[allow(dead_code)]
async fn len(&self, entry: &BlobEntry) -> Result<u64, StoreError> {
let path = self.basedir.join(entry.table).join(&entry.key);
let metadata = fs::metadata(path).await?;
Ok(metadata.len())
}
pub fn sanitize_key(key: &str) -> String {
utf8_percent_encode(key, ESCAPE_CHARS).to_string()
}
}
#[async_trait::async_trait]
impl BlobStoreInterface for BlobStoreFS {
async fn table(&self, table: &BlobTable) -> Result<Box<dyn BlobTableInterface>, StoreError> {
let store = BlobStoreFS::new(self.basedir.join(table)).await?;
Ok(Box::new(store))
}
}
#[async_trait::async_trait]
impl<'a> BlobTableInterface<'a> for BlobStoreFS {
async fn has(&self, key: &str) -> Result<bool, StoreError> {
let key = Self::sanitize_key(key);
let path = self.basedir.join(key);
fs::try_exists(path).await.map_err(|e| StoreError::from(e))
}
async fn get(&self, key: &str) -> Result<Cow<'a, [u8]>, StoreError> {
let key = Self::sanitize_key(key);
let path = self.basedir.join(key);
debug!("Reading data from {}", path.display());
fs::read(path)
.await
.map_err(|e| StoreError::from(e))
.map(|v| Cow::from(v))
}
async fn set(&mut self, key: &str, value: &[u8]) -> Result<(), StoreError> {
let key = Self::sanitize_key(key);
let path = self.basedir.join(key);
debug!("Saving data to {}", path.display());
fs::write(path, value)
.await
.map_err(|e| StoreError::from(e))
}
async fn delete(&mut self, key: &str) -> Result<(), StoreError> {
let key = Self::sanitize_key(key);
let path = self.basedir.join(key);
fs::remove_file(path).await?;
Ok(())
}
async fn delete_all(&mut self) -> Result<(), StoreError> {
fs::remove_dir_all(&self.basedir).await?;
fs::create_dir_all(&self.basedir).await?;
Ok(())
}
async fn list(&self) -> Result<Vec<String>, StoreError> {
let mut l: Vec<String> = vec![];
let mut entries = fs::read_dir(&self.basedir).await?;
while let Some(entry) = entries.next_entry().await? {
let key = percent_decode_str(entry.file_name().to_str().unwrap())
.decode_utf8()
.unwrap()
.to_string();
l.push(key);
}
Ok(l)
}
}
#[cfg(test)]
mod tests {
use crate::store::{BlobEntry, BlobStoreFS, BlobStoreInterface, TABLE_AVATAR};
use std::path::PathBuf;
use temp_dir::TempDir;
#[tokio::test]
async fn blob_store_fs_entry() {
let temp_dir = PathBuf::from(TempDir::new().unwrap().path());
let mut store = BlobStoreFS::new(temp_dir.clone()).await.unwrap();
let entry = BlobEntry::new(TABLE_AVATAR, "foo".to_string());
store.set_in_table(&entry, &vec![1, 2, 3, 4]).await.unwrap();
assert_eq!(
*vec!(1, 2, 3, 4),
*store.get_in_table(&entry).await.unwrap()
);
tokio::fs::remove_dir_all(temp_dir).await.unwrap();
}
#[tokio::test]
async fn blob_store_fs_table() {
let temp_dir = PathBuf::from(TempDir::new().unwrap().path());
println!("{:?}", temp_dir);
let store = BlobStoreFS::new(temp_dir.clone()).await.unwrap();
let mut avatars = store.table(&TABLE_AVATAR).await.unwrap();
avatars.set("foo", &vec![1, 2, 3, 4]).await.unwrap();
assert_eq!(*vec!(1, 2, 3, 4), *avatars.get("foo").await.unwrap());
tokio::fs::remove_dir_all(temp_dir).await.unwrap();
}
#[tokio::test]
async fn blob_store_fs_escape() {
let temp_dir = PathBuf::from(TempDir::new().unwrap().path());
let store = BlobStoreFS::new(temp_dir.clone()).await.unwrap();
let mut avatars = store.table(&TABLE_AVATAR).await.unwrap();
avatars.set("foo", &vec![1, 2, 3, 4]).await.unwrap();
avatars.set("/etc/passwd", &vec![1, 3, 1, 2]).await.unwrap();
let mut expected = vec!["foo".to_string(), "/etc/passwd".to_string()];
expected.sort_unstable();
let mut found = avatars.list().await.unwrap();
found.sort_unstable();
assert_eq!(expected, found);
assert_eq!(
*vec!(1, 3, 1, 2),
*avatars.get("/etc/passwd").await.unwrap()
);
}
}