spec_ai/spec_ai_config/config/
cache.rs1use anyhow::{Context, Result};
2use serde_json;
3
4use super::AppConfig;
5use crate::spec_ai_config::persistence::Persistence;
6
7const CONFIG_CACHE_KEY: &str = "effective_config";
8const POLICIES_CACHE_KEY: &str = "effective_policies";
9
10pub struct ConfigCache {
12 persistence: Persistence,
13}
14
15impl ConfigCache {
16 pub fn new(persistence: Persistence) -> Self {
18 Self { persistence }
19 }
20
21 pub fn store_effective_config(&self, config: &AppConfig) -> Result<()> {
23 let value = serde_json::to_value(config).context("serializing config to JSON")?;
24
25 self.persistence
26 .policy_upsert(CONFIG_CACHE_KEY, &value)
27 .context("storing effective config in cache")
28 }
29
30 pub fn load_effective_config(&self) -> Result<Option<AppConfig>> {
32 if let Some(entry) = self.persistence.policy_get(CONFIG_CACHE_KEY)? {
33 let config: AppConfig =
34 serde_json::from_value(entry.value).context("deserializing cached config")?;
35 Ok(Some(config))
36 } else {
37 Ok(None)
38 }
39 }
40
41 pub fn store_effective_policies(&self, policies: &serde_json::Value) -> Result<()> {
43 self.persistence
44 .policy_upsert(POLICIES_CACHE_KEY, policies)
45 .context("storing effective policies in cache")
46 }
47
48 pub fn load_effective_policies(&self) -> Result<Option<serde_json::Value>> {
50 if let Some(entry) = self.persistence.policy_get(POLICIES_CACHE_KEY)? {
51 Ok(Some(entry.value))
52 } else {
53 Ok(None)
54 }
55 }
56
57 pub fn has_config_changed(&self, current: &AppConfig) -> Result<bool> {
60 if let Some(cached) = self.load_effective_config()? {
61 let current_json =
63 serde_json::to_value(current).context("serializing current config")?;
64 let cached_json = serde_json::to_value(&cached).context("serializing cached config")?;
65
66 Ok(current_json != cached_json)
67 } else {
68 Ok(true)
70 }
71 }
72
73 pub fn diff_summary(&self, current: &AppConfig) -> Result<Vec<String>> {
75 let mut changes = Vec::new();
76
77 if let Some(cached) = self.load_effective_config()? {
78 if current.model.provider != cached.model.provider {
80 changes.push(format!(
81 "Model provider: {} -> {}",
82 cached.model.provider, current.model.provider
83 ));
84 }
85
86 if current.model.temperature != cached.model.temperature {
87 changes.push(format!(
88 "Temperature: {} -> {}",
89 cached.model.temperature, current.model.temperature
90 ));
91 }
92
93 if current.logging.level != cached.logging.level {
94 changes.push(format!(
95 "Logging level: {} -> {}",
96 cached.logging.level, current.logging.level
97 ));
98 }
99
100 if current.database.path != cached.database.path {
101 changes.push(format!(
102 "Database path: {} -> {}",
103 cached.database.path.display(),
104 current.database.path.display()
105 ));
106 }
107
108 if current.agents.len() != cached.agents.len() {
109 changes.push(format!(
110 "Number of agents: {} -> {}",
111 cached.agents.len(),
112 current.agents.len()
113 ));
114 }
115
116 if current.default_agent != cached.default_agent {
117 changes.push(format!(
118 "Default agent: {:?} -> {:?}",
119 cached.default_agent, current.default_agent
120 ));
121 }
122 } else {
123 changes.push("No cached config found (first run or cache cleared)".to_string());
124 }
125
126 Ok(changes)
127 }
128
129 pub fn clear(&self) -> Result<()> {
131 self.persistence
133 .policy_upsert(CONFIG_CACHE_KEY, &serde_json::Value::Null)?;
134 self.persistence
135 .policy_upsert(POLICIES_CACHE_KEY, &serde_json::Value::Null)?;
136 Ok(())
137 }
138}
139
140#[cfg(test)]
141mod tests {
142 use super::*;
143 use tempfile::TempDir;
144
145 fn create_test_config() -> AppConfig {
146 use crate::spec_ai_config::config::{
147 AudioConfig, AuthConfig, DatabaseConfig, LoggingConfig, ModelConfig, PluginConfig,
148 SafetyConfig, SyncConfig, UiConfig,
149 };
150 use std::collections::HashMap;
151 use std::path::PathBuf;
152
153 AppConfig {
154 database: DatabaseConfig {
155 path: PathBuf::from("/tmp/test.db"),
156 },
157 model: ModelConfig {
158 provider: "test".to_string(),
159 model_name: None,
160 code_model: None,
161 embeddings_model: None,
162 api_key_source: None,
163 temperature: 0.5,
164 },
165 ui: UiConfig {
166 prompt: "> ".to_string(),
167 theme: "default".to_string(),
168 },
169 logging: LoggingConfig {
170 level: "info".to_string(),
171 },
172 audio: AudioConfig::default(),
173 mesh: crate::spec_ai_config::config::MeshConfig::default(),
174 plugins: PluginConfig::default(),
175 sync: SyncConfig::default(),
176 auth: AuthConfig::default(),
177 safety: SafetyConfig::default(),
178 approval: Default::default(),
179 agents: HashMap::new(),
180 default_agent: None,
181 }
182 }
183
184 #[test]
185 fn test_store_and_load_config() {
186 let temp_dir = TempDir::new().unwrap();
187 let db_path = temp_dir.path().join("test.duckdb");
188 let persistence = Persistence::new(&db_path).unwrap();
189 let cache = ConfigCache::new(persistence);
190
191 let config = create_test_config();
192
193 cache.store_effective_config(&config).unwrap();
195
196 let loaded = cache.load_effective_config().unwrap();
198 assert!(loaded.is_some());
199
200 let loaded_config = loaded.unwrap();
201 assert_eq!(loaded_config.model.provider, "test");
202 assert_eq!(loaded_config.model.temperature, 0.5);
203 }
204
205 #[test]
206 fn test_load_nonexistent_config() {
207 let temp_dir = TempDir::new().unwrap();
208 let db_path = temp_dir.path().join("test.duckdb");
209 let persistence = Persistence::new(&db_path).unwrap();
210 let cache = ConfigCache::new(persistence);
211
212 let loaded = cache.load_effective_config().unwrap();
213 assert!(loaded.is_none());
214 }
215
216 #[test]
217 fn test_store_and_load_policies() {
218 let temp_dir = TempDir::new().unwrap();
219 let db_path = temp_dir.path().join("test.duckdb");
220 let persistence = Persistence::new(&db_path).unwrap();
221 let cache = ConfigCache::new(persistence);
222
223 let policies = serde_json::json!({
224 "allow": ["tool1", "tool2"],
225 "deny": ["tool3"]
226 });
227
228 cache.store_effective_policies(&policies).unwrap();
230
231 let loaded = cache.load_effective_policies().unwrap();
233 assert!(loaded.is_some());
234 assert_eq!(loaded.unwrap(), policies);
235 }
236
237 #[test]
238 fn test_has_config_changed() {
239 let temp_dir = TempDir::new().unwrap();
240 let db_path = temp_dir.path().join("test.duckdb");
241 let persistence = Persistence::new(&db_path).unwrap();
242 let cache = ConfigCache::new(persistence);
243
244 let config1 = create_test_config();
245
246 assert!(cache.has_config_changed(&config1).unwrap());
248
249 cache.store_effective_config(&config1).unwrap();
251
252 assert!(!cache.has_config_changed(&config1).unwrap());
254
255 let mut config2 = config1.clone();
257 config2.model.temperature = 0.9;
258
259 assert!(cache.has_config_changed(&config2).unwrap());
261 }
262
263 #[test]
264 fn test_diff_summary() {
265 let temp_dir = TempDir::new().unwrap();
266 let db_path = temp_dir.path().join("test.duckdb");
267 let persistence = Persistence::new(&db_path).unwrap();
268 let cache = ConfigCache::new(persistence);
269
270 let mut config1 = create_test_config();
271 cache.store_effective_config(&config1).unwrap();
272
273 config1.model.provider = "new_provider".to_string();
275 config1.model.temperature = 0.9;
276
277 let diff = cache.diff_summary(&config1).unwrap();
278 assert!(diff.len() >= 2);
279 assert!(diff.iter().any(|s| s.contains("Model provider")));
280 assert!(diff.iter().any(|s| s.contains("Temperature")));
281 }
282
283 #[test]
284 fn test_clear_cache() {
285 let temp_dir = TempDir::new().unwrap();
286 let db_path = temp_dir.path().join("test.duckdb");
287 let persistence = Persistence::new(&db_path).unwrap();
288 let cache = ConfigCache::new(persistence);
289
290 let config = create_test_config();
291 let policies = serde_json::json!({"test": "value"});
292
293 cache.store_effective_config(&config).unwrap();
295 cache.store_effective_policies(&policies).unwrap();
296
297 assert!(cache.load_effective_config().unwrap().is_some());
299 assert!(cache.load_effective_policies().unwrap().is_some());
300
301 cache.clear().unwrap();
303
304 let _ = cache.load_effective_config();
308 }
309
310 #[test]
311 fn test_idempotent_store() {
312 let temp_dir = TempDir::new().unwrap();
313 let db_path = temp_dir.path().join("test.duckdb");
314 let persistence = Persistence::new(&db_path).unwrap();
315 let cache = ConfigCache::new(persistence);
316
317 let config = create_test_config();
318
319 cache.store_effective_config(&config).unwrap();
321 cache.store_effective_config(&config).unwrap();
322 cache.store_effective_config(&config).unwrap();
323
324 let loaded = cache.load_effective_config().unwrap();
326 assert!(loaded.is_some());
327 assert_eq!(loaded.unwrap().model.provider, "test");
328 }
329}