#[cfg(test)]
use std::collections::HashMap;
#[cfg(test)]
use std::sync::Arc;
#[cfg(test)]
use std::time::{Duration, Instant};
#[cfg(test)]
use tokio::sync::RwLock;
#[cfg(test)]
#[allow(dead_code)]
pub struct MockBackend {
name: &'static str,
score: u8,
persistent: bool,
data: Arc<RwLock<HashMap<String, (Vec<u8>, Option<Instant>)>>>,
}
#[cfg(test)]
impl MockBackend {
pub fn new(name: &'static str, score: u8, persistent: bool) -> Self {
Self {
name,
score,
persistent,
data: Arc::new(RwLock::new(HashMap::new())),
}
}
}
#[cfg(test)]
impl crate::backend::BackendScore for MockBackend {
fn score(&self) -> u8 {
self.score
}
fn is_persistent(&self) -> bool {
self.persistent
}
fn backend_name(&self) -> &'static str {
self.name
}
}
#[cfg(test)]
#[async_trait::async_trait]
impl crate::backend::CacheReader for MockBackend {
async fn get(&self, key: &str) -> crate::error::Result<Option<Vec<u8>>> {
let now = Instant::now();
let mut data = self.data.write().await;
if let Some((_v, expires_at)) = data.get(key) {
if let Some(exp) = expires_at {
if *exp <= now {
data.remove(key);
return Ok(None);
}
}
return Ok(Some(data.get(key).unwrap().0.clone()));
}
Ok(None)
}
async fn exists(&self, key: &str) -> crate::error::Result<bool> {
let now = Instant::now();
let mut data = self.data.write().await;
if let Some((_v, expires_at)) = data.get(key) {
if let Some(exp) = expires_at {
if *exp <= now {
data.remove(key);
return Ok(false);
}
}
return Ok(true);
}
Ok(false)
}
async fn ttl(&self, key: &str) -> crate::error::Result<Option<Duration>> {
let now = Instant::now();
let data = self.data.read().await;
if let Some((_v, expires_at)) = data.get(key) {
if let Some(exp) = expires_at {
return Ok(exp.checked_duration_since(now));
}
}
Ok(None)
}
async fn len(&self) -> crate::error::Result<u64> {
let data = self.data.read().await;
Ok(data.len() as u64)
}
async fn is_empty(&self) -> crate::error::Result<bool> {
let data = self.data.read().await;
Ok(data.is_empty())
}
async fn capacity(&self) -> crate::error::Result<u64> {
Ok(0)
}
async fn stats(&self) -> crate::error::Result<HashMap<String, String>> {
let mut stats = HashMap::new();
stats.insert("type".to_string(), self.name.to_string());
Ok(stats)
}
}
#[cfg(test)]
#[async_trait::async_trait]
impl crate::backend::CacheWriter for MockBackend {
async fn set(&self, key: &str, value: Vec<u8>, ttl: Option<Duration>) -> crate::error::Result<()> {
let mut data = self.data.write().await;
let expires_at = ttl.map(|d| Instant::now() + d);
data.insert(key.to_string(), (value, expires_at));
Ok(())
}
async fn delete(&self, key: &str) -> crate::error::Result<()> {
let mut data = self.data.write().await;
data.remove(key);
Ok(())
}
async fn clear(&self) -> crate::error::Result<()> {
let mut data = self.data.write().await;
data.clear();
Ok(())
}
async fn expire(&self, key: &str, ttl: Duration) -> crate::error::Result<bool> {
let mut data = self.data.write().await;
if data.contains_key(key) {
let entry = data.get_mut(key).unwrap();
entry.1 = Some(Instant::now() + ttl);
Ok(true)
} else {
Ok(false)
}
}
}
#[cfg(test)]
#[async_trait::async_trait]
impl crate::backend::CacheConnector for MockBackend {
async fn health_check(&self) -> crate::error::Result<()> {
Ok(())
}
async fn shutdown(&self) {}
fn backend_kind(&self) -> crate::backend::interface::BackendKind {
crate::backend::interface::BackendKind::Mock
}
}
#[cfg(test)]
mod mock_tests {
use super::*;
use crate::backend::{BackendScore, CacheConnector, CacheReader, CacheWriter};
#[tokio::test]
async fn test_mock_backend_new() {
let backend = MockBackend::new("test", 50, false);
assert_eq!(BackendScore::score(&backend), 50);
assert_eq!(BackendScore::backend_name(&backend), "test");
assert!(!BackendScore::is_persistent(&backend));
}
#[tokio::test]
async fn test_mock_backend_set_get() {
let backend = MockBackend::new("test", 50, false);
CacheWriter::set(&backend, "key", b"value".to_vec(), None)
.await
.unwrap();
let result = CacheReader::get(&backend, "key").await.unwrap();
assert_eq!(result, Some(b"value".to_vec()));
}
#[tokio::test]
async fn test_mock_backend_delete() {
let backend = MockBackend::new("test", 50, false);
CacheWriter::set(&backend, "key", b"value".to_vec(), None)
.await
.unwrap();
CacheWriter::delete(&backend, "key").await.unwrap();
assert!(CacheReader::get(&backend, "key").await.unwrap().is_none());
}
#[tokio::test]
async fn test_mock_backend_clear() {
let backend = MockBackend::new("test", 50, false);
CacheWriter::set(&backend, "k1", b"v1".to_vec(), None).await.unwrap();
CacheWriter::clear(&backend).await.unwrap();
assert!(CacheReader::is_empty(&backend).await.unwrap());
}
#[tokio::test]
async fn test_mock_backend_exists() {
let backend = MockBackend::new("test", 50, false);
assert!(!CacheReader::exists(&backend, "key").await.unwrap());
CacheWriter::set(&backend, "key", b"value".to_vec(), None)
.await
.unwrap();
assert!(CacheReader::exists(&backend, "key").await.unwrap());
}
#[tokio::test]
async fn test_mock_backend_len() {
let backend = MockBackend::new("test", 50, false);
assert_eq!(CacheReader::len(&backend).await.unwrap(), 0);
CacheWriter::set(&backend, "k1", b"v1".to_vec(), None).await.unwrap();
assert_eq!(CacheReader::len(&backend).await.unwrap(), 1);
}
#[tokio::test]
async fn test_mock_backend_stats() {
let backend = MockBackend::new("test", 50, false);
let stats = CacheReader::stats(&backend).await.unwrap();
assert_eq!(stats.get("type"), Some(&"test".to_string()));
}
#[tokio::test]
async fn test_mock_backend_health_check() {
let backend = MockBackend::new("test", 50, false);
assert!(CacheConnector::health_check(&backend).await.is_ok());
}
#[tokio::test]
async fn test_mock_backend_shutdown() {
let backend = MockBackend::new("test", 50, false);
CacheConnector::shutdown(&backend).await;
}
#[test]
fn test_mock_backend_kind() {
let backend = MockBackend::new("test", 50, false);
assert_eq!(
CacheConnector::backend_kind(&backend),
crate::backend::interface::BackendKind::Mock
);
}
#[tokio::test]
async fn test_mock_backend_persistent() {
let backend = MockBackend::new("test", 50, true);
assert!(BackendScore::is_persistent(&backend));
}
#[tokio::test]
async fn test_mock_set_with_ttl_expires_after_timeout() {
let backend = MockBackend::new("test", 50, false);
backend
.set("k", b"v".to_vec(), Some(Duration::from_millis(50)))
.await
.unwrap();
assert_eq!(backend.get("k").await.unwrap(), Some(b"v".to_vec()));
tokio::time::sleep(Duration::from_millis(100)).await;
assert_eq!(backend.get("k").await.unwrap(), None);
}
#[tokio::test]
async fn test_mock_set_without_ttl_never_expires() {
let backend = MockBackend::new("test", 50, false);
backend.set("k", b"v".to_vec(), None).await.unwrap();
tokio::time::sleep(Duration::from_millis(100)).await;
assert_eq!(backend.get("k").await.unwrap(), Some(b"v".to_vec()));
}
#[tokio::test]
async fn test_mock_ttl_returns_remaining() {
let backend = MockBackend::new("test", 50, false);
backend
.set("k", b"v".to_vec(), Some(Duration::from_secs(60)))
.await
.unwrap();
let ttl = backend.ttl("k").await.unwrap().expect("ttl should be Some");
assert!(
ttl > Duration::from_secs(58),
"ttl={} should be > 58s",
ttl.as_secs_f64()
);
assert!(
ttl <= Duration::from_secs(60),
"ttl={} should be <= 60s",
ttl.as_secs_f64()
);
}
#[tokio::test]
async fn test_mock_ttl_returns_none_for_missing_key() {
let backend = MockBackend::new("test", 50, false);
assert_eq!(backend.ttl("missing").await.unwrap(), None);
}
#[tokio::test]
async fn test_mock_ttl_returns_none_for_no_ttl_key() {
let backend = MockBackend::new("test", 50, false);
backend.set("k", b"v".to_vec(), None).await.unwrap();
assert_eq!(backend.ttl("k").await.unwrap(), None);
}
#[tokio::test]
async fn test_mock_expire_extends_ttl() {
let backend = MockBackend::new("test", 50, false);
backend
.set("k", b"v".to_vec(), Some(Duration::from_secs(60)))
.await
.unwrap();
let ok = backend.expire("k", Duration::from_secs(120)).await.unwrap();
assert!(ok, "expire on existing key should return true");
let ttl = backend
.ttl("k")
.await
.unwrap()
.expect("ttl should be Some after expire");
assert!(
ttl > Duration::from_secs(118),
"ttl={} should be > 118s",
ttl.as_secs_f64()
);
}
#[tokio::test]
async fn test_mock_expire_missing_key_returns_false() {
let backend = MockBackend::new("test", 50, false);
let ok = backend.expire("missing", Duration::from_secs(60)).await.unwrap();
assert!(!ok, "expire on missing key should return false");
}
#[tokio::test]
async fn test_mock_lazy_cleanup_removes_expired_entry() {
let backend = MockBackend::new("test", 50, false);
backend
.set("k", b"v".to_vec(), Some(Duration::from_millis(50)))
.await
.unwrap();
tokio::time::sleep(Duration::from_millis(100)).await;
let _ = backend.get("k").await.unwrap();
let data = backend.data.read().await;
assert!(!data.contains_key("k"), "expired entry should be lazily removed");
}
}