#![cfg_attr(not(feature = "filesystem"), allow(unused))]
use crate::cache::Cache;
use crate::hash::Hash;
use std::fs::{File, create_dir_all};
use std::io::{self, Write};
use std::path::PathBuf;
use wasmer::{AsEngineRef, DeserializeError, Module, SerializeError};
#[derive(Debug, Clone)]
pub struct FileSystemCache {
path: PathBuf,
ext: Option<String>,
}
#[cfg(feature = "filesystem")]
impl FileSystemCache {
pub fn new<P: Into<PathBuf>>(path: P) -> io::Result<Self> {
let path: PathBuf = path.into();
if path.exists() {
let metadata = path.metadata()?;
if metadata.is_dir() {
if !metadata.permissions().readonly() {
Ok(Self { path, ext: None })
} else {
Err(io::Error::new(
io::ErrorKind::PermissionDenied,
format!("the supplied path is readonly: {}", path.display()),
))
}
} else {
Err(io::Error::new(
io::ErrorKind::PermissionDenied,
format!(
"the supplied path already points to a file: {}",
path.display()
),
))
}
} else {
let res = create_dir_all(&path);
if res.is_err() {
Err(io::Error::other(format!(
"failed to create cache directory: {}",
path.display()
)))
} else {
Ok(Self { path, ext: None })
}
}
}
pub fn set_cache_extension(&mut self, ext: Option<impl ToString>) {
self.ext = ext.map(|ext| ext.to_string());
}
}
#[cfg(feature = "filesystem")]
impl Cache for FileSystemCache {
type DeserializeError = DeserializeError;
type SerializeError = SerializeError;
unsafe fn load(
&self,
engine: &impl AsEngineRef,
key: Hash,
) -> Result<Module, Self::DeserializeError> {
let filename = if let Some(ref ext) = self.ext {
format!("{key}.{ext}")
} else {
key.to_string()
};
let path = self.path.join(filename);
let ret = unsafe { Module::deserialize_from_file(engine, path.clone()) };
if ret.is_err() {
let _ = std::fs::remove_file(path);
}
ret
}
fn store(&mut self, key: Hash, module: &Module) -> Result<(), Self::SerializeError> {
let filename = if let Some(ref ext) = self.ext {
format!("{key}.{ext}")
} else {
key.to_string()
};
let path = self.path.join(filename);
let mut file = File::create(path)?;
let buffer = module.serialize()?;
file.write_all(&buffer)?;
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_fs_cache() {
let dir = tempfile::tempdir().unwrap();
let mut cache = FileSystemCache::new(dir.path()).unwrap();
let engine = wasmer::Engine::default();
let bytes = include_bytes!("../../wasix/tests/envvar.wasm");
let module = Module::from_binary(&engine, bytes).unwrap();
let key = Hash::generate(bytes);
cache.store(key, &module).unwrap();
let _restored = unsafe { cache.load(&engine, key).unwrap() };
}
}