Server: domain (entities, value objects, ports), protocol (postcard wire types), application (config service, data projection), adapters (config-memory, tcp-server), bootstrap (composition root with fake data). Client: client-domain (layout engine, render tree, HAL ports), client-application (message handling, repaint commands), adapters (tcp-client, display-terminal), client-desktop (end-to-end working). ESP32: client-esp32 firmware with ILI9341 display over SPI, WiFi networking. Display test verified on hardware — landscape orientation, text rendering works. 60 workspace tests, all passing.
120 lines
4.2 KiB
Rust
120 lines
4.2 KiB
Rust
use std::collections::HashMap;
|
|
use std::sync::RwLock;
|
|
use domain::{
|
|
ConfigRepository,
|
|
DataSource, DataSourceId, Layout, LayoutPreset, LayoutPresetId,
|
|
WidgetConfig, WidgetId,
|
|
};
|
|
|
|
#[derive(Debug)]
|
|
pub enum MemoryConfigError {
|
|
LockPoisoned,
|
|
}
|
|
|
|
impl std::fmt::Display for MemoryConfigError {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
MemoryConfigError::LockPoisoned => write!(f, "lock poisoned"),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct MemoryConfigStore {
|
|
widgets: RwLock<HashMap<WidgetId, WidgetConfig>>,
|
|
data_sources: RwLock<HashMap<DataSourceId, DataSource>>,
|
|
layout: RwLock<Option<Layout>>,
|
|
presets: RwLock<HashMap<LayoutPresetId, LayoutPreset>>,
|
|
}
|
|
|
|
impl MemoryConfigStore {
|
|
pub fn new() -> Self {
|
|
Self {
|
|
widgets: RwLock::new(HashMap::new()),
|
|
data_sources: RwLock::new(HashMap::new()),
|
|
layout: RwLock::new(None),
|
|
presets: RwLock::new(HashMap::new()),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ConfigRepository for MemoryConfigStore {
|
|
type Error = MemoryConfigError;
|
|
|
|
async fn get_widget(&self, id: WidgetId) -> Result<Option<WidgetConfig>, Self::Error> {
|
|
let guard = self.widgets.read().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
Ok(guard.get(&id).cloned())
|
|
}
|
|
|
|
async fn list_widgets(&self) -> Result<Vec<WidgetConfig>, Self::Error> {
|
|
let guard = self.widgets.read().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
Ok(guard.values().cloned().collect())
|
|
}
|
|
|
|
async fn save_widget(&self, config: &WidgetConfig) -> Result<(), Self::Error> {
|
|
let mut guard = self.widgets.write().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
guard.insert(config.id, config.clone());
|
|
Ok(())
|
|
}
|
|
|
|
async fn delete_widget(&self, id: WidgetId) -> Result<(), Self::Error> {
|
|
let mut guard = self.widgets.write().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
guard.remove(&id);
|
|
Ok(())
|
|
}
|
|
|
|
async fn get_data_source(&self, id: DataSourceId) -> Result<Option<DataSource>, Self::Error> {
|
|
let guard = self.data_sources.read().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
Ok(guard.get(&id).cloned())
|
|
}
|
|
|
|
async fn list_data_sources(&self) -> Result<Vec<DataSource>, Self::Error> {
|
|
let guard = self.data_sources.read().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
Ok(guard.values().cloned().collect())
|
|
}
|
|
|
|
async fn save_data_source(&self, source: &DataSource) -> Result<(), Self::Error> {
|
|
let mut guard = self.data_sources.write().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
guard.insert(source.id, source.clone());
|
|
Ok(())
|
|
}
|
|
|
|
async fn delete_data_source(&self, id: DataSourceId) -> Result<(), Self::Error> {
|
|
let mut guard = self.data_sources.write().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
guard.remove(&id);
|
|
Ok(())
|
|
}
|
|
|
|
async fn get_layout(&self) -> Result<Option<Layout>, Self::Error> {
|
|
let guard = self.layout.read().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
Ok(guard.clone())
|
|
}
|
|
|
|
async fn save_layout(&self, layout: &Layout) -> Result<(), Self::Error> {
|
|
let mut guard = self.layout.write().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
*guard = Some(layout.clone());
|
|
Ok(())
|
|
}
|
|
|
|
async fn get_preset(&self, id: LayoutPresetId) -> Result<Option<LayoutPreset>, Self::Error> {
|
|
let guard = self.presets.read().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
Ok(guard.get(&id).cloned())
|
|
}
|
|
|
|
async fn list_presets(&self) -> Result<Vec<LayoutPreset>, Self::Error> {
|
|
let guard = self.presets.read().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
Ok(guard.values().cloned().collect())
|
|
}
|
|
|
|
async fn save_preset(&self, preset: &LayoutPreset) -> Result<(), Self::Error> {
|
|
let mut guard = self.presets.write().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
guard.insert(preset.id, preset.clone());
|
|
Ok(())
|
|
}
|
|
|
|
async fn delete_preset(&self, id: LayoutPresetId) -> Result<(), Self::Error> {
|
|
let mut guard = self.presets.write().map_err(|_| MemoryConfigError::LockPoisoned)?;
|
|
guard.remove(&id);
|
|
Ok(())
|
|
}
|
|
}
|