use std::collections::HashMap;
use std::sync::atomic::{AtomicU64, Ordering};
use std::sync::{Arc, Mutex, OnceLock};
use crate::Raster;
pub const RASTER_MEMORY_PREFIX: &str = "memory://raster/";
static NEXT_ID: AtomicU64 = AtomicU64::new(1);
static RASTER_STORE: OnceLock<Mutex<HashMap<String, Arc<Raster>>>> = OnceLock::new();
fn store() -> &'static Mutex<HashMap<String, Arc<Raster>>> {
RASTER_STORE.get_or_init(|| Mutex::new(HashMap::new()))
}
pub fn raster_is_memory_path(path: &str) -> bool {
path.starts_with(RASTER_MEMORY_PREFIX)
}
pub fn raster_path_to_id(path: &str) -> Option<&str> {
path.strip_prefix(RASTER_MEMORY_PREFIX)
}
pub fn make_raster_memory_path(id: &str) -> String {
format!("{RASTER_MEMORY_PREFIX}{id}")
}
pub fn put_raster(raster: Raster) -> String {
let id = NEXT_ID.fetch_add(1, Ordering::Relaxed).to_string();
if let Ok(mut map) = store().lock() {
map.insert(id.clone(), Arc::new(raster));
}
id
}
pub fn put_raster_arc(raster: Arc<Raster>) -> String {
let id = NEXT_ID.fetch_add(1, Ordering::Relaxed).to_string();
if let Ok(mut map) = store().lock() {
map.insert(id.clone(), raster);
}
id
}
pub fn get_raster_arc_by_id(id: &str) -> Option<Arc<Raster>> {
store()
.lock()
.ok()
.and_then(|map| map.get(id).cloned())
}
pub fn get_raster_arc_by_path(path: &str) -> Option<Arc<Raster>> {
raster_path_to_id(path).and_then(get_raster_arc_by_id)
}
pub fn get_raster_by_id(id: &str) -> Option<Raster> {
get_raster_arc_by_id(id).map(|r| (*r).clone())
}
pub fn replace_raster_by_id(id: &str, raster: Raster) -> bool {
store()
.lock()
.map(|mut map| map.insert(id.to_string(), Arc::new(raster)).is_some())
.unwrap_or(false)
}
pub fn replace_raster_by_path(path: &str, raster: Raster) -> bool {
raster_path_to_id(path)
.map(|id| replace_raster_by_id(id, raster))
.unwrap_or(false)
}
pub fn remove_raster_by_id(id: &str) -> Option<Raster> {
store()
.lock()
.ok()
.and_then(|mut map| map.remove(id))
.map(|r| Arc::try_unwrap(r).unwrap_or_else(|shared| (*shared).clone()))
}
pub fn remove_raster_by_path(path: &str) -> Option<Raster> {
raster_path_to_id(path).and_then(remove_raster_by_id)
}
pub fn clear_rasters() -> usize {
store()
.lock()
.map(|mut map| {
let count = map.len();
map.clear();
count
})
.unwrap_or(0)
}
pub fn raster_count() -> usize {
store().lock().map(|map| map.len()).unwrap_or(0)
}
pub fn raster_store_bytes() -> usize {
store()
.lock()
.map(|map| {
map.values()
.map(|r| r.data.len() * r.data_type.size_bytes())
.sum()
})
.unwrap_or(0)
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{DataType, RasterConfig};
use std::sync::{Mutex, MutexGuard, OnceLock};
fn memory_store_test_guard() -> MutexGuard<'static, ()> {
static TEST_LOCK: OnceLock<Mutex<()>> = OnceLock::new();
TEST_LOCK
.get_or_init(|| Mutex::new(()))
.lock()
.expect("memory_store test lock poisoned")
}
fn make_test_raster(value: f64) -> Raster {
let cfg = RasterConfig {
cols: 2,
rows: 2,
cell_size: 1.0,
nodata: -9999.0,
data_type: DataType::F32,
..Default::default()
};
Raster::from_data(cfg, vec![value, value + 1.0, value + 2.0, value + 3.0]).unwrap()
}
#[test]
fn remove_raster_by_id_removes_only_target_entry() {
let _guard = memory_store_test_guard();
clear_rasters();
let id1 = put_raster(make_test_raster(1.0));
let id2 = put_raster(make_test_raster(10.0));
assert!(get_raster_by_id(&id1).is_some());
assert!(get_raster_by_id(&id2).is_some());
let removed = remove_raster_by_id(&id1).expect("raster should be removed by id");
assert_eq!(removed.get(0, 0, 0), 1.0);
assert!(get_raster_by_id(&id1).is_none());
assert!(get_raster_by_id(&id2).is_some());
clear_rasters();
}
#[test]
fn remove_raster_by_path_and_clear_rasters_work() {
let _guard = memory_store_test_guard();
clear_rasters();
let id1 = put_raster(make_test_raster(5.0));
let id2 = put_raster(make_test_raster(20.0));
let path1 = make_raster_memory_path(&id1);
let removed = remove_raster_by_path(&path1).expect("raster should be removed by path");
assert_eq!(removed.get(0, 0, 0), 5.0);
assert!(get_raster_by_id(&id1).is_none());
assert!(get_raster_by_id(&id2).is_some());
assert!(raster_count() >= 1);
let cleared = clear_rasters();
assert!(cleared >= 1);
assert!(get_raster_by_id(&id2).is_none());
}
}