1use std::sync::Arc;
5
6use crate::config::{
7 Config, DebugConfig, DocumentConfig, GraphConfig, LearningConfig, OrchestrationConfig,
8 PersonaConfig, SecurityConfig, TimeoutConfig,
9};
10use crate::vault::Secret;
11
12pub const CONTEXT_BUDGET_RESERVE_RATIO: f32 = 0.20;
16
17#[derive(Clone)]
46#[allow(clippy::struct_excessive_bools)] pub struct AgentSessionConfig {
48 pub max_tool_iterations: usize,
50 pub max_tool_retries: usize,
51 pub max_retry_duration_secs: u64,
52 pub retry_base_ms: u64,
53 pub retry_max_ms: u64,
54 pub parameter_reformat_provider: String,
55 pub tool_repeat_threshold: usize,
56 pub tool_summarization: bool,
57 pub tool_call_cutoff: usize,
58 pub max_tool_calls_per_session: Option<u32>,
59 pub overflow_config: zeph_tools::OverflowConfig,
60 pub permission_policy: zeph_tools::PermissionPolicy,
61
62 pub model_name: String,
64 pub embed_model: String,
65
66 pub semantic_cache_enabled: bool,
68 pub semantic_cache_threshold: f32,
69 pub semantic_cache_max_candidates: u32,
70
71 pub budget_tokens: usize,
73 pub soft_compaction_threshold: f32,
74 pub hard_compaction_threshold: f32,
75 pub compaction_preserve_tail: usize,
76 pub compaction_cooldown_turns: u8,
77 pub prune_protect_tokens: usize,
78 pub redact_credentials: bool,
79
80 pub security: SecurityConfig,
82 pub timeouts: TimeoutConfig,
83
84 pub learning: LearningConfig,
86 pub document_config: DocumentConfig,
87 pub graph_config: GraphConfig,
88 pub persona_config: PersonaConfig,
89 pub trajectory_config: crate::config::TrajectoryConfig,
90 pub category_config: crate::config::CategoryConfig,
91 pub reasoning_config: zeph_config::ReasoningConfig,
92 pub memcot_config: zeph_config::MemCotConfig,
93 pub tree_config: crate::config::TreeConfig,
94 pub microcompact_config: crate::config::MicrocompactConfig,
95 pub autodream_config: crate::config::AutoDreamConfig,
96 pub magic_docs_config: crate::config::MagicDocsConfig,
97 pub acon_config: zeph_config::AconConfig,
99 pub arc_config: zeph_config::ArcCompactionConfig,
101 pub anomaly_config: zeph_tools::AnomalyConfig,
102 pub result_cache_config: zeph_tools::ResultCacheConfig,
103 pub utility_config: zeph_tools::UtilityScoringConfig,
104 pub orchestration_config: OrchestrationConfig,
105 pub debug_config: DebugConfig,
106 pub server_compaction: bool,
107
108 pub budget_hint_enabled: bool,
110
111 pub recap: zeph_config::RecapConfig,
113
114 pub loop_min_interval_secs: u64,
116
117 pub goal_config: zeph_config::GoalConfig,
119
120 pub secrets: Arc<[(String, Secret)]>,
126
127 pub fidelity_config: Option<zeph_config::FidelityConfig>,
129}
130
131impl AgentSessionConfig {
132 #[must_use]
137 pub fn from_config(config: &Config, budget_tokens: usize) -> Self {
138 Self {
139 max_tool_iterations: config.agent.max_tool_iterations,
140 max_tool_retries: config.tools.retry.max_attempts,
141 max_retry_duration_secs: config.tools.retry.budget_secs,
142 retry_base_ms: config.tools.retry.base_ms,
143 retry_max_ms: config.tools.retry.max_ms,
144 parameter_reformat_provider: config.tools.retry.parameter_reformat_provider.to_string(),
145 tool_repeat_threshold: config.agent.tool_repeat_threshold,
146 tool_summarization: config.tools.summarize_output,
147 tool_call_cutoff: config.memory.tool_call_cutoff,
148 max_tool_calls_per_session: config.tools.max_tool_calls_per_session,
149 overflow_config: config.tools.overflow.clone(),
150 permission_policy: zeph_tools::build_permission_policy(
151 &config.tools,
152 config.security.autonomy_level,
153 ),
154 model_name: config.llm.effective_model().to_owned(),
155 embed_model: crate::provider_factory::stable_skill_embedding_model(config),
156 semantic_cache_enabled: config.llm.semantic_cache_enabled,
157 semantic_cache_threshold: config.llm.semantic_cache_threshold,
158 semantic_cache_max_candidates: config.llm.semantic_cache_max_candidates,
159 budget_tokens,
160 soft_compaction_threshold: config.memory.soft_compaction_threshold,
161 hard_compaction_threshold: config.memory.hard_compaction_threshold,
162 compaction_preserve_tail: config.memory.compaction_preserve_tail,
163 compaction_cooldown_turns: config.memory.compaction_cooldown_turns,
164 prune_protect_tokens: config.memory.prune_protect_tokens,
165 redact_credentials: config.memory.redact_credentials,
166 security: config.security.clone(),
167 timeouts: config.timeouts,
168 learning: config.skills.learning.clone(),
169 document_config: config.memory.documents.clone(),
170 graph_config: config.memory.graph.clone(),
171 persona_config: config.memory.persona.clone(),
172 trajectory_config: config.memory.trajectory.clone(),
173 category_config: config.memory.category.clone(),
174 reasoning_config: config.memory.reasoning.clone(),
175 memcot_config: config.memory.memcot.clone(),
176 tree_config: config.memory.tree.clone(),
177 microcompact_config: config.memory.microcompact.clone(),
178 autodream_config: config.memory.autodream.clone(),
179 magic_docs_config: config.magic_docs.clone(),
180 acon_config: config.memory.compression.acon.clone(),
181 arc_config: config.memory.compression.arc.clone(),
182 anomaly_config: config.tools.anomaly.clone(),
183 result_cache_config: config.tools.result_cache.clone(),
184 utility_config: config.tools.utility.clone(),
185 orchestration_config: config.orchestration.clone(),
186 debug_config: config.debug.clone(),
187 server_compaction: config.llm.providers.iter().any(|e| e.server_compaction),
188 budget_hint_enabled: config.agent.budget_hint_enabled,
189 secrets: config
190 .secrets
191 .custom
192 .iter()
193 .map(|(k, v)| (k.clone(), Secret::new(v.expose().to_owned())))
194 .collect::<Vec<_>>()
195 .into(),
196 recap: config.session.recap.clone(),
197 loop_min_interval_secs: config.cli.loop_.min_interval_secs,
198 goal_config: config.goals.clone(),
199 fidelity_config: {
200 let fc = config.memory.fidelity.clone();
201 if let Some(ref cfg) = fc
202 && let Err(e) = cfg.validate()
203 {
204 tracing::warn!("fidelity config invalid, scoring disabled: {e}");
205 }
206 fc
207 },
208 }
209 }
210}
211
212#[cfg(test)]
213mod tests {
214 use super::*;
215
216 #[test]
217 fn from_config_maps_all_fields() {
218 let config = Config::default();
219 let budget = 100_000;
220 let sc = AgentSessionConfig::from_config(&config, budget);
221
222 assert_eq!(sc.max_tool_iterations, config.agent.max_tool_iterations);
223 assert_eq!(sc.max_tool_retries, config.tools.retry.max_attempts);
224 assert_eq!(sc.max_retry_duration_secs, config.tools.retry.budget_secs);
225 assert_eq!(sc.retry_base_ms, config.tools.retry.base_ms);
226 assert_eq!(sc.retry_max_ms, config.tools.retry.max_ms);
227 assert_eq!(
228 sc.parameter_reformat_provider,
229 config.tools.retry.parameter_reformat_provider.as_str()
230 );
231 assert_eq!(sc.tool_repeat_threshold, config.agent.tool_repeat_threshold);
232 assert_eq!(sc.tool_summarization, config.tools.summarize_output);
233 assert_eq!(sc.tool_call_cutoff, config.memory.tool_call_cutoff);
234 assert_eq!(sc.model_name, config.llm.effective_model());
235 assert_eq!(
236 sc.embed_model,
237 crate::provider_factory::stable_skill_embedding_model(&config)
238 );
239 assert_eq!(sc.semantic_cache_enabled, config.llm.semantic_cache_enabled);
240 assert!(
241 (sc.semantic_cache_threshold - config.llm.semantic_cache_threshold).abs()
242 < f32::EPSILON
243 );
244 assert_eq!(
245 sc.semantic_cache_max_candidates,
246 config.llm.semantic_cache_max_candidates
247 );
248 assert_eq!(sc.budget_tokens, budget);
249 assert!(
250 (sc.soft_compaction_threshold - config.memory.soft_compaction_threshold).abs()
251 < f32::EPSILON
252 );
253 assert!(
254 (sc.hard_compaction_threshold - config.memory.hard_compaction_threshold).abs()
255 < f32::EPSILON
256 );
257 assert_eq!(
258 sc.compaction_preserve_tail,
259 config.memory.compaction_preserve_tail
260 );
261 assert_eq!(
262 sc.compaction_cooldown_turns,
263 config.memory.compaction_cooldown_turns
264 );
265 assert_eq!(sc.prune_protect_tokens, config.memory.prune_protect_tokens);
266 assert_eq!(sc.redact_credentials, config.memory.redact_credentials);
267 assert_eq!(sc.graph_config.enabled, config.memory.graph.enabled);
268 assert_eq!(
269 sc.orchestration_config.enabled,
270 config.orchestration.enabled
271 );
272 assert_eq!(
273 sc.orchestration_config.max_tasks,
274 config.orchestration.max_tasks
275 );
276 assert_eq!(sc.anomaly_config.enabled, config.tools.anomaly.enabled);
277 assert_eq!(
278 sc.result_cache_config.enabled,
279 config.tools.result_cache.enabled
280 );
281 assert_eq!(
282 sc.result_cache_config.ttl_secs,
283 config.tools.result_cache.ttl_secs
284 );
285 assert_eq!(sc.debug_config.enabled, config.debug.enabled);
286 assert_eq!(
287 sc.document_config.rag_enabled,
288 config.memory.documents.rag_enabled
289 );
290 assert_eq!(
291 sc.overflow_config.threshold,
292 config.tools.overflow.threshold
293 );
294 assert_eq!(
295 sc.permission_policy.autonomy_level(),
296 config.security.autonomy_level
297 );
298 assert_eq!(sc.security.autonomy_level, config.security.autonomy_level);
299 assert_eq!(sc.timeouts.llm_seconds, config.timeouts.llm_seconds);
300 assert_eq!(sc.learning.enabled, config.skills.learning.enabled);
301 assert_eq!(
302 sc.server_compaction,
303 config.llm.providers.iter().any(|e| e.server_compaction)
304 );
305 assert_eq!(sc.secrets.len(), config.secrets.custom.len());
306 assert_eq!(sc.recap.on_resume, config.session.recap.on_resume);
307 assert_eq!(sc.recap.max_tokens, config.session.recap.max_tokens);
308 assert_eq!(sc.goal_config.enabled, config.goals.enabled);
309 assert_eq!(sc.goal_config.max_text_chars, config.goals.max_text_chars);
310 assert_eq!(
311 sc.fidelity_config.is_some(),
312 config.memory.fidelity.is_some()
313 );
314 }
315}