Skip to main content

oxios_kernel/
config.rs

1#![allow(missing_docs)]
2//! Configuration loading from TOML files.
3//!
4//! Configuration is stored at `~/.oxios/config.toml` and controls
5//! kernel, gateway, and execution settings.
6
7use cron::Schedule;
8use serde::{Deserialize, Serialize};
9use std::str::FromStr;
10
11use crate::email::{SmtpProvider, SmtpTls};
12use crate::scheduler::Priority;
13
14/// Cron scheduler configuration.
15#[derive(Debug, Clone, Deserialize, Serialize)]
16pub struct CronConfig {
17    /// Enable the cron scheduler.
18    #[serde(default)]
19    pub enabled: bool,
20    /// Tick interval in seconds.
21    #[serde(default = "default_tick_interval")]
22    pub tick_interval_secs: u64,
23    /// Inline job definitions from config.toml.
24    #[serde(default)]
25    pub jobs: std::collections::HashMap<String, InlineCronJob>,
26}
27
28impl Default for CronConfig {
29    fn default() -> Self {
30        Self {
31            enabled: false,
32            tick_interval_secs: default_tick_interval(),
33            jobs: std::collections::HashMap::new(),
34        }
35    }
36}
37
38fn default_tick_interval() -> u64 {
39    60
40}
41
42/// Inline cron job definition in config.toml.
43#[derive(Debug, Clone, Deserialize, Serialize)]
44pub struct InlineCronJob {
45    /// Cron expression (e.g. "0 */6 * * *").
46    pub schedule: String,
47    /// Goal description for the agent.
48    pub goal: String,
49    /// Constraints on agent behavior.
50    #[serde(default)]
51    pub constraints: Vec<String>,
52    /// Criteria that must be met for the job to be considered successful.
53    #[serde(default)]
54    pub acceptance_criteria: Vec<String>,
55    /// Toolchain preset name.
56    #[serde(default = "default_toolchain_inline")]
57    pub toolchain: String,
58    /// Job priority.
59    #[serde(default)]
60    pub priority: Priority,
61    /// Whether the job is active.
62    #[serde(default = "default_true_inline")]
63    pub enabled: bool,
64}
65
66fn default_toolchain_inline() -> String {
67    "default".into()
68}
69
70fn default_true_inline() -> bool {
71    true
72}
73
74/// Memory system configuration.
75#[derive(Debug, Clone, Serialize, Deserialize)]
76pub struct MemoryConfig {
77    /// Enable the memory system.
78    #[serde(default = "default_true")]
79    pub enabled: bool,
80    /// Maximum memories returned by recall.
81    #[serde(default = "default_max_recall")]
82    pub max_recall: usize,
83    /// Auto-summarize sessions on completion.
84    #[serde(default = "default_true")]
85    pub auto_summarize: bool,
86    /// Capture compaction summaries as conversation memory.
87    #[serde(default = "default_true")]
88    pub capture_compaction: bool,
89    /// Memory retention in days (0 = unlimited).
90    #[serde(default)]
91    pub retention_days: u32,
92    /// Enable embedding cache.
93    #[serde(default = "default_true")]
94    pub cache_enabled: bool,
95    /// Embedding cache TTL in seconds.
96    #[serde(default = "default_cache_ttl")]
97    pub cache_ttl_secs: u64,
98    /// Maximum embedding cache entries.
99    #[serde(default = "default_cache_max_entries")]
100    pub cache_max_entries: usize,
101    /// Consolidation configuration (RFC-008).
102    #[serde(default)]
103    pub consolidation: ConsolidationConfig,
104    /// SQLite memory storage configuration (RFC-012).
105    #[serde(default)]
106    pub sqlite: SqliteMemoryConfig,
107    /// Embedding provider configuration (RFC-012).
108    #[serde(default)]
109    pub embedding: EmbeddingConfig,
110    /// Learning configuration (RFC-012 Phase 4: SONA).
111    #[serde(default)]
112    pub learning: LearningConfig,
113    /// Knowledge dream configuration (RFC-022).
114    #[serde(default)]
115    pub knowledge_dream: crate::knowledge_dream::KnowledgeDreamConfig,
116    /// AutoMemoryBridge configuration (RFC-012 Phase 7: SQLite ↔ MEMORY.md sync).
117    #[serde(default)]
118    pub bridge: MemoryBridgeConfig,
119}
120
121fn default_true() -> bool {
122    true
123}
124
125fn default_max_recall() -> usize {
126    10
127}
128
129fn default_cache_ttl() -> u64 {
130    3600 // 1 hour
131}
132
133fn default_cache_max_entries() -> usize {
134    10000
135}
136
137impl Default for MemoryConfig {
138    fn default() -> Self {
139        Self {
140            enabled: true,
141            max_recall: 10,
142            auto_summarize: true,
143            capture_compaction: true,
144            retention_days: 0,
145            cache_enabled: true,
146            cache_ttl_secs: 3600,
147            cache_max_entries: 10000,
148            consolidation: ConsolidationConfig::default(),
149            sqlite: SqliteMemoryConfig::default(),
150            embedding: EmbeddingConfig::default(),
151            learning: LearningConfig::default(),
152            knowledge_dream: crate::knowledge_dream::KnowledgeDreamConfig::default(),
153            bridge: MemoryBridgeConfig::default(),
154        }
155    }
156}
157
158// ---------------------------------------------------------------------------
159// SqliteMemoryConfig (RFC-012: SQLite Memory Storage)
160// ---------------------------------------------------------------------------
161
162/// SQLite-backed memory storage configuration (RFC-012).
163///
164/// When enabled, memories are stored in a single `memory.db` file with
165/// FTS5 BM25 + sqlite-vec KNN search. Falls back to the existing JSON
166/// + TF-IDF approach when disabled.
167#[derive(Debug, Clone, Serialize, Deserialize)]
168pub struct SqliteMemoryConfig {
169    /// Enable SQLite-backed memory storage.
170    #[serde(default = "default_true")]
171    pub enabled: bool,
172    /// Path to the SQLite database file.
173    /// Empty string means default: `~/.oxios/workspace/memory.db`
174    #[serde(default)]
175    pub path: String,
176    /// Embedding vector dimension.
177    /// Controls the `vec0` virtual table dimension.
178    /// Common values: 128 (fast), 256 (balanced), 768 (full Gemma).
179    #[serde(default = "default_embedding_dim")]
180    pub embedding_dim: usize,
181    /// Enable WAL mode for concurrent reads.
182    #[serde(default = "default_true")]
183    pub wal_mode: bool,
184}
185
186fn default_embedding_dim() -> usize {
187    256
188}
189
190impl Default for SqliteMemoryConfig {
191    fn default() -> Self {
192        Self {
193            enabled: true,
194            path: String::new(),
195            embedding_dim: 256,
196            wal_mode: true,
197        }
198    }
199}
200
201// ---------------------------------------------------------------------------
202// EmbeddingConfig (RFC-012: Embedding Provider)
203// ---------------------------------------------------------------------------
204
205/// Embedding provider configuration (RFC-012).
206///
207/// Controls which embedding model is used for semantic search.
208/// When `embedding-mlx` feature is enabled and `provider = "mlx"`,
209/// uses EmbeddingGemma-300m via MLX on Apple Silicon.
210/// Otherwise falls back to TF-IDF.
211#[derive(Debug, Clone, Serialize, Deserialize)]
212pub struct EmbeddingConfig {
213    /// Embedding provider: "tfidf" (default) or "mlx" (Apple Silicon).
214    #[serde(default = "default_embedding_provider")]
215    pub provider: String,
216    /// Matryoshka dimension: 128, 256, 512, or 768.
217    /// Only used when provider = "mlx".
218    #[serde(default = "default_embedding_dim")]
219    pub dimension: usize,
220    /// Model TTL in seconds. Unloaded after this duration of inactivity.
221    /// Only used when provider = "mlx".
222    #[serde(default = "default_model_ttl")]
223    pub model_ttl_secs: u64,
224}
225
226fn default_embedding_provider() -> String {
227    "gguf".to_string()
228}
229
230fn default_model_ttl() -> u64 {
231    300 // 5 minutes
232}
233
234impl Default for EmbeddingConfig {
235    fn default() -> Self {
236        Self {
237            provider: default_embedding_provider(),
238            dimension: default_embedding_dim(),
239            model_ttl_secs: default_model_ttl(),
240        }
241    }
242}
243
244// ---------------------------------------------------------------------------
245// LearningConfig (RFC-012 Phase 4: SONA)
246// ---------------------------------------------------------------------------
247
248/// Learning engine configuration (RFC-012 Phase 4).
249///
250/// Controls SONA self-learning persistence.
251#[derive(Debug, Clone, Serialize, Deserialize)]
252pub struct LearningConfig {
253    /// Enable the learning subsystem (SONA).
254    #[serde(default = "default_true")]
255    pub enabled: bool,
256    /// SONA operating mode: "realtime", "balanced", "research", "edge".
257    #[serde(default = "default_sona_mode")]
258    pub sona_mode: String,
259    /// Interval between automatic distillation runs (hours).
260    #[serde(default = "default_distill_interval")]
261    pub distill_interval_hours: u64,
262    /// Minimum quality score for auto-promoting patterns to long-term.
263    #[serde(default = "default_auto_promote_quality")]
264    pub auto_promote_quality: f32,
265    /// Minimum usage count before auto-promotion is considered.
266    #[serde(default = "default_auto_promote_min_usage")]
267    pub auto_promote_min_usage: u32,
268}
269
270fn default_sona_mode() -> String {
271    "balanced".to_string()
272}
273
274fn default_distill_interval() -> u64 {
275    6
276}
277
278fn default_auto_promote_quality() -> f32 {
279    0.8
280}
281
282fn default_auto_promote_min_usage() -> u32 {
283    3
284}
285
286impl Default for LearningConfig {
287    fn default() -> Self {
288        Self {
289            enabled: true,
290            sona_mode: default_sona_mode(),
291            distill_interval_hours: default_distill_interval(),
292            auto_promote_quality: default_auto_promote_quality(),
293            auto_promote_min_usage: default_auto_promote_min_usage(),
294        }
295    }
296}
297
298// ---------------------------------------------------------------------------
299// MemoryBridgeConfig (RFC-012 Phase 7: SQLite ↔ MEMORY.md)
300// ---------------------------------------------------------------------------
301
302/// AutoMemoryBridge configuration (RFC-012 Phase 7).
303///
304/// Controls bidirectional sync between SQLite memory store
305/// and external MEMORY.md files.
306#[derive(Debug, Clone, Serialize, Deserialize)]
307pub struct MemoryBridgeConfig {
308    /// Enable bidirectional sync with MEMORY.md.
309    #[serde(default)]
310    pub sync_enabled: bool,
311    /// Sync interval in seconds.
312    #[serde(default = "default_bridge_interval")]
313    pub interval_secs: u64,
314}
315
316fn default_bridge_interval() -> u64 {
317    3600
318}
319
320impl Default for MemoryBridgeConfig {
321    fn default() -> Self {
322        Self {
323            sync_enabled: false,
324            interval_secs: default_bridge_interval(),
325        }
326    }
327}
328
329// ---------------------------------------------------------------------------
330// ConsolidationConfig (RFC-008: Memory Consolidation)
331// ---------------------------------------------------------------------------
332
333/// Memory consolidation configuration (RFC-008).
334/// All values have sensible defaults — users never need to configure these.
335#[derive(Debug, Clone, Serialize, Deserialize)]
336pub struct ConsolidationConfig {
337    /// Preset: "conservative" | "balanced" | "aggressive" | "custom".
338    /// When not "custom", all other fields are overridden by the preset values.
339    /// Call `apply_preset()` once during kernel init to resolve.
340    #[serde(default = "default_preset")]
341    pub preset: String,
342
343    // ── Dream Process ─────────────────────────────────
344    #[serde(default = "default_true")]
345    pub dream_enabled: bool,
346    #[serde(default = "default_dream_interval")]
347    pub dream_interval_hours: u64,
348    #[serde(default = "default_dream_min_sessions")]
349    pub dream_min_sessions: u32,
350
351    // ── Tier Budgets ──────────────────────────────────
352    #[serde(default = "default_hot_max")]
353    pub hot_max_entries: usize,
354    #[serde(default = "default_warm_max")]
355    pub warm_max_entries: usize,
356    #[serde(default = "default_cold_max")]
357    pub cold_max_entries: usize,
358    #[serde(default = "default_hot_token_budget")]
359    pub hot_token_budget: usize,
360
361    // ── Decay ─────────────────────────────────────────
362    #[serde(default = "default_true")]
363    pub decay_enabled: bool,
364    #[serde(default = "default_one")]
365    pub decay_multiplier: f32,
366    #[serde(default = "default_decay_threshold")]
367    pub decay_threshold: f32,
368    #[serde(default = "default_retention_days")]
369    pub retention_days: u32,
370
371    // ── Auto-Protection ───────────────────────────────
372    #[serde(default = "default_true")]
373    pub auto_protection: bool,
374    #[serde(default = "default_protection_low_access")]
375    pub protection_low_access: u32,
376    #[serde(default = "default_protection_medium_access")]
377    pub protection_medium_access: u32,
378    #[serde(default = "default_protection_high_access")]
379    pub protection_high_access: u32,
380    #[serde(default = "default_protection_medium_sessions")]
381    pub protection_medium_sessions: u32,
382    #[serde(default = "default_protection_high_sessions")]
383    pub protection_high_sessions: u32,
384
385    // ── Auto-Classification ───────────────────────────
386    #[serde(default = "default_true")]
387    pub auto_classification: bool,
388    #[serde(default = "default_type_promotion_threshold")]
389    pub type_promotion_repetitions: u32,
390
391    // ── Compaction ────────────────────────────────────
392    #[serde(default = "default_compaction_threshold")]
393    pub compaction_line_threshold: usize,
394    #[serde(default = "default_true")]
395    pub llm_compaction: bool,
396
397    // ── Dream LLM ──────────────────────────────────────
398    /// Optional model for Dream LLM operations (None = rule-based fallback).
399    #[serde(default)]
400    pub dream_model: Option<String>,
401
402    // ── Protection Demotion ────────────────────────────
403    #[serde(default = "default_true")]
404    pub protection_demotion_enabled: bool,
405    #[serde(default = "default_demotion_stale_days")]
406    pub protection_demotion_stale_days: u32,
407    #[serde(default = "default_demotion_max_step")]
408    pub protection_demotion_max_step: u32,
409
410    // ── Proactive Recall ──────────────────────────────
411    #[serde(default = "default_true")]
412    pub proactive_recall: bool,
413    #[serde(default = "default_proactive_limit")]
414    pub proactive_recall_limit: usize,
415    #[serde(default = "default_proactive_threshold")]
416    pub proactive_recall_threshold: f32,
417}
418
419fn default_dream_interval() -> u64 {
420    24
421}
422fn default_dream_min_sessions() -> u32 {
423    5
424}
425fn default_hot_max() -> usize {
426    50
427}
428fn default_warm_max() -> usize {
429    500
430}
431fn default_cold_max() -> usize {
432    10_000
433}
434fn default_hot_token_budget() -> usize {
435    3_000
436}
437fn default_one() -> f32 {
438    1.0
439}
440fn default_decay_threshold() -> f32 {
441    0.05
442}
443fn default_retention_days() -> u32 {
444    90
445}
446fn default_protection_low_access() -> u32 {
447    2
448}
449fn default_protection_medium_access() -> u32 {
450    3
451}
452fn default_protection_high_access() -> u32 {
453    5
454}
455fn default_protection_medium_sessions() -> u32 {
456    2
457}
458fn default_protection_high_sessions() -> u32 {
459    3
460}
461fn default_type_promotion_threshold() -> u32 {
462    3
463}
464fn default_compaction_threshold() -> usize {
465    200
466}
467fn default_proactive_limit() -> usize {
468    5
469}
470fn default_proactive_threshold() -> f32 {
471    0.6
472}
473fn default_demotion_stale_days() -> u32 {
474    30
475}
476fn default_demotion_max_step() -> u32 {
477    1
478}
479
480fn default_preset() -> String {
481    "balanced".into()
482}
483
484impl Default for ConsolidationConfig {
485    fn default() -> Self {
486        Self {
487            preset: default_preset(),
488            dream_enabled: true,
489            dream_interval_hours: 24,
490            dream_min_sessions: 5,
491            hot_max_entries: 50,
492            warm_max_entries: 500,
493            cold_max_entries: 10_000,
494            hot_token_budget: 3_000,
495            decay_enabled: true,
496            decay_multiplier: 1.0,
497            decay_threshold: 0.05,
498            retention_days: 90,
499            auto_protection: true,
500            protection_low_access: 2,
501            protection_medium_access: 3,
502            protection_high_access: 5,
503            protection_medium_sessions: 2,
504            protection_high_sessions: 3,
505            auto_classification: true,
506            type_promotion_repetitions: 3,
507            compaction_line_threshold: 200,
508            llm_compaction: true,
509            dream_model: None,
510            protection_demotion_enabled: true,
511            protection_demotion_stale_days: 30,
512            protection_demotion_max_step: 1,
513            proactive_recall: true,
514            proactive_recall_limit: 5,
515            proactive_recall_threshold: 0.6,
516        }
517    }
518}
519
520impl ConsolidationConfig {
521    /// Apply the preset to all fields.
522    /// Call once during kernel initialization.
523    /// When `preset` is "custom", individual fields are left untouched.
524    pub fn apply_preset(&mut self) {
525        let resolved = match self.preset.as_str() {
526            "conservative" => Self::conservative(),
527            "aggressive" => Self::aggressive(),
528            "custom" => return,
529            _ => Self::default(), // "balanced" 및 알 수 없는 값
530        };
531        *self = resolved;
532    }
533
534    /// Conservative preset: slow decay, long retention, larger capacities.
535    fn conservative() -> Self {
536        Self {
537            preset: "conservative".into(),
538            dream_enabled: true,
539            dream_interval_hours: 48,
540            dream_min_sessions: 10,
541            hot_max_entries: 100,
542            warm_max_entries: 1000,
543            cold_max_entries: 50_000,
544            hot_token_budget: 5_000,
545            decay_enabled: true,
546            decay_multiplier: 0.8,
547            decay_threshold: 0.05,
548            retention_days: 365,
549            auto_protection: true,
550            protection_low_access: 3,
551            protection_medium_access: 5,
552            protection_high_access: 10,
553            protection_medium_sessions: 3,
554            protection_high_sessions: 5,
555            auto_classification: true,
556            type_promotion_repetitions: 5,
557            compaction_line_threshold: 300,
558            llm_compaction: true,
559            dream_model: None,
560            protection_demotion_enabled: true,
561            protection_demotion_stale_days: 90,
562            protection_demotion_max_step: 1,
563            proactive_recall: true,
564            proactive_recall_limit: 8,
565            proactive_recall_threshold: 0.5,
566        }
567    }
568
569    /// Aggressive preset: fast decay, short retention, smaller capacities.
570    fn aggressive() -> Self {
571        Self {
572            preset: "aggressive".into(),
573            dream_enabled: true,
574            dream_interval_hours: 4,
575            dream_min_sessions: 2,
576            hot_max_entries: 20,
577            warm_max_entries: 100,
578            cold_max_entries: 1_000,
579            hot_token_budget: 2_000,
580            decay_enabled: true,
581            decay_multiplier: 1.0,
582            decay_threshold: 0.1,
583            retention_days: 30,
584            auto_protection: true,
585            protection_low_access: 1,
586            protection_medium_access: 2,
587            protection_high_access: 3,
588            protection_medium_sessions: 1,
589            protection_high_sessions: 2,
590            auto_classification: true,
591            type_promotion_repetitions: 2,
592            compaction_line_threshold: 150,
593            llm_compaction: true,
594            dream_model: None,
595            protection_demotion_enabled: true,
596            protection_demotion_stale_days: 14,
597            protection_demotion_max_step: 2,
598            proactive_recall: true,
599            proactive_recall_limit: 3,
600            proactive_recall_threshold: 0.7,
601        }
602    }
603}
604
605/// Channel activation configuration.
606#[derive(Debug, Clone, Deserialize, Serialize, Default)]
607pub struct ChannelsConfig {
608    /// List of channel names to activate on startup.
609    /// Channels are message-only interfaces (CLI, Telegram).
610    #[serde(default)]
611    pub enabled: Vec<String>,
612
613    /// Telegram-specific configuration.
614    #[serde(default)]
615    pub telegram: TelegramChannelConfig,
616}
617
618/// Surface activation configuration.
619///
620/// Surfaces are kernel-connected control interfaces (Web dashboard, future desktop apps).
621/// They have direct kernel access for management, monitoring, and configuration.
622#[derive(Debug, Clone, Deserialize, Serialize)]
623pub struct SurfacesConfig {
624    /// List of surface names to activate on startup.
625    /// Default: ["web"] if the web feature is compiled in.
626    #[serde(default = "default_surfaces_enabled")]
627    pub enabled: Vec<String>,
628}
629
630fn default_surfaces_enabled() -> Vec<String> {
631    vec!["web".to_string()]
632}
633
634impl Default for SurfacesConfig {
635    fn default() -> Self {
636        Self {
637            enabled: default_surfaces_enabled(),
638        }
639    }
640}
641
642/// Telegram channel configuration.
643#[derive(Debug, Clone, Deserialize, Serialize)]
644pub struct TelegramChannelConfig {
645    /// Environment variable name holding the bot token.
646    #[serde(default = "default_telegram_token_env")]
647    pub bot_token_env: String,
648    /// List of allowed Telegram user IDs (empty = allow all).
649    #[serde(default)]
650    pub allowed_users: Vec<i64>,
651    /// Telegram session management settings.
652    #[serde(default)]
653    pub session: TelegramSessionConfig,
654}
655
656fn default_telegram_token_env() -> String {
657    "TELEGRAM_BOT_TOKEN".to_string()
658}
659
660impl Default for TelegramChannelConfig {
661    fn default() -> Self {
662        Self {
663            bot_token_env: default_telegram_token_env(),
664            allowed_users: Vec::new(),
665            session: TelegramSessionConfig::default(),
666        }
667    }
668}
669
670/// LLM engine configuration.
671#[derive(Debug, Clone, Deserialize, Serialize)]
672#[allow(clippy::derivable_impls)]
673pub struct EngineConfig {
674    /// Default model in "provider/model" format.
675    /// Empty string means no model configured — onboarding required.
676    #[serde(default)]
677    pub default_model: String,
678    /// Explicit API key override (highest priority).
679    /// If empty/None, falls back to oxi auth store, then env vars.
680    /// Masked when serialized to API responses.
681    #[serde(default, skip_serializing)]
682    pub api_key: Option<String>,
683    /// Per-provider options for fine-grained control (thinking mode, etc.).
684    /// Passed through to `AgentLoopConfig::provider_options`.
685    #[serde(default)]
686    pub provider_options: Option<oxi_sdk::ProviderOptions>,
687    /// Enable complexity-based model routing.
688    /// When enabled, the engine can route simple tasks to cheaper models
689    /// and complex tasks to more capable ones.
690    #[serde(default)]
691    pub routing_enabled: bool,
692    /// Prefer cost-efficient models when routing.
693    #[serde(default)]
694    pub prefer_cost_efficient: bool,
695    /// Fallback models to try when the primary model fails.
696    #[serde(default)]
697    pub fallback_models: Vec<String>,
698    /// Models excluded from automatic routing.
699    #[serde(default)]
700    pub excluded_models: Vec<String>,
701}
702
703#[allow(clippy::derivable_impls)]
704impl Default for EngineConfig {
705    fn default() -> Self {
706        Self {
707            default_model: String::new(),
708            api_key: None,
709            provider_options: None,
710            routing_enabled: false,
711            prefer_cost_efficient: false,
712            fallback_models: Vec::new(),
713            excluded_models: Vec::new(),
714        }
715    }
716}
717
718/// Daemon mode configuration.
719#[derive(Debug, Clone, Deserialize, Serialize)]
720pub struct DaemonConfig {
721    /// PID file path.
722    #[serde(default = "default_pid_file")]
723    pub pid_file: String,
724    /// Log directory.
725    #[serde(default = "default_daemon_log_dir")]
726    pub log_dir: String,
727}
728
729fn default_pid_file() -> String {
730    dirs::home_dir()
731        .map(|h| format!("{}/.oxios/oxios.pid", h.display()))
732        .unwrap_or_else(|| "./oxios.pid".into())
733}
734
735fn default_daemon_log_dir() -> String {
736    dirs::home_dir()
737        .map(|h| format!("{}/.oxios/logs", h.display()))
738        .unwrap_or_else(|| "./logs".into())
739}
740
741impl Default for DaemonConfig {
742    fn default() -> Self {
743        Self {
744            pid_file: default_pid_file(),
745            log_dir: default_daemon_log_dir(),
746        }
747    }
748}
749
750/// Session management configuration.
751#[derive(Debug, Clone, Deserialize, Serialize)]
752pub struct SessionConfig {
753    /// Maximum number of sessions to retain.
754    /// When exceeded, oldest sessions (by `updated_at`) are pruned.
755    /// Set to 0 for unlimited.
756    #[serde(default = "default_max_sessions")]
757    pub max_sessions: usize,
758
759    /// Time-to-live for sessions in hours.
760    /// Sessions older than this are automatically pruned.
761    /// Set to 0 for unlimited (no TTL-based pruning).
762    #[serde(default = "default_session_ttl_hours")]
763    pub ttl_hours: u64,
764
765    /// Enable automatic session pruning on every session save.
766    #[serde(default = "default_true")]
767    pub auto_prune: bool,
768}
769
770fn default_max_sessions() -> usize {
771    100
772}
773
774fn default_session_ttl_hours() -> u64 {
775    168 // 7 days
776}
777
778impl Default for SessionConfig {
779    fn default() -> Self {
780        Self {
781            max_sessions: default_max_sessions(),
782            ttl_hours: default_session_ttl_hours(),
783            auto_prune: true,
784        }
785    }
786}
787
788/// RFC-025 Phase 5: Mount auto-promotion configuration.
789/// Controls the background scanner that promotes frequently-used paths into
790/// Mounts. See `mount::path_promotion`.
791#[derive(Debug, Clone, Deserialize, Serialize)]
792pub struct MountsConfig {
793    /// Enable the auto-promotion scanner.
794    #[serde(default = "default_true")]
795    pub auto_promote_enabled: bool,
796    /// Minimum distinct touches within the window to trigger promotion.
797    #[serde(default = "default_promote_threshold")]
798    pub auto_promote_threshold: usize,
799    /// How far back to look, in days.
800    #[serde(default = "default_promote_window_days")]
801    pub auto_promote_window_days: i64,
802    /// Seconds between promotion scans (background cadence).
803    #[serde(default = "default_promote_interval_secs")]
804    pub auto_promote_interval_secs: u64,
805}
806
807fn default_promote_threshold() -> usize {
808    3
809}
810
811fn default_promote_window_days() -> i64 {
812    14
813}
814
815fn default_promote_interval_secs() -> u64 {
816    3600 // hourly
817}
818
819impl Default for MountsConfig {
820    fn default() -> Self {
821        Self {
822            auto_promote_enabled: true,
823            auto_promote_threshold: default_promote_threshold(),
824            auto_promote_window_days: default_promote_window_days(),
825            auto_promote_interval_secs: default_promote_interval_secs(),
826        }
827    }
828}
829
830/// Telegram session management configuration.
831#[derive(Debug, Clone, Deserialize, Serialize)]
832pub struct TelegramSessionConfig {
833    /// Automatically rotate to a new session after this many hours of inactivity.
834    /// Set to 0 to disable time-based rotation.
835    #[serde(default = "default_telegram_session_rotation_hours")]
836    pub rotation_hours: u64,
837
838    /// Maximum number of messages per session before auto-rotating.
839    /// Set to 0 for unlimited.
840    #[serde(default = "default_telegram_session_max_messages")]
841    pub max_messages: usize,
842}
843
844fn default_telegram_session_rotation_hours() -> u64 {
845    2 // 2 hours
846}
847
848fn default_telegram_session_max_messages() -> usize {
849    0 // unlimited by default
850}
851
852impl Default for TelegramSessionConfig {
853    fn default() -> Self {
854        Self {
855            rotation_hours: default_telegram_session_rotation_hours(),
856            max_messages: default_telegram_session_max_messages(),
857        }
858    }
859}
860
861/// Top-level Oxios configuration.
862#[derive(Debug, Clone, Deserialize, Serialize, Default)]
863pub struct OxiosConfig {
864    /// Kernel settings.
865    pub kernel: KernelConfig,
866    /// LLM engine settings.
867    #[serde(default)]
868    pub engine: EngineConfig,
869    /// Daemon mode settings.
870    #[serde(default)]
871    pub daemon: DaemonConfig,
872    /// Gateway settings.
873    #[serde(default)]
874    pub gateway: GatewayConfig,
875    /// Scheduler settings (AIOS-inspired task scheduling).
876    #[serde(default)]
877    pub scheduler: SchedulerConfig,
878    /// Orchestrator settings (Ouroboros protocol execution).
879    #[serde(default)]
880    pub orchestrator: OrchestratorConfig,
881    /// Context manager settings (LLM context window management).
882    #[serde(default)]
883    pub context: ContextConfig,
884    /// Security/access control settings.
885    #[serde(default)]
886    pub security: SecurityConfig,
887    /// Persona system settings.
888    #[serde(default)]
889    pub persona: PersonaConfig,
890    /// Memory system settings.
891    #[serde(default)]
892    pub memory: MemoryConfig,
893    /// Cron scheduler settings.
894    #[serde(default)]
895    pub cron: CronConfig,
896    /// MCP server configurations.
897    #[serde(default)]
898    pub mcp: McpConfig,
899    /// Git version control settings.
900    #[serde(default)]
901    pub git: GitConfig,
902    /// Audit trail configuration.
903    #[serde(default)]
904    pub audit: AuditConfig,
905    /// Budget enforcement configuration.
906    #[serde(default)]
907    pub budget: BudgetConfig,
908    /// Exec configuration (host command execution bridge).
909    #[serde(default)]
910    pub exec: ExecConfig,
911    /// Resource monitor configuration.
912    #[serde(default)]
913    pub resource_monitor: ResourceMonitorConfig,
914    /// OpenTelemetry tracing configuration.
915    #[serde(default)]
916    pub otel: OtelConfig,
917    /// Logging configuration.
918    #[serde(default)]
919    pub logging: LoggingConfig,
920    /// Channel activation configuration (message interfaces: CLI, Telegram).
921    #[serde(default)]
922    pub channels: ChannelsConfig,
923    /// Surface activation configuration (control interfaces: Web dashboard).
924    #[serde(default)]
925    pub surfaces: Option<SurfacesConfig>,
926    /// Headless browser configuration.
927    #[serde(default)]
928    pub browser: BrowserConfig,
929    /// Session management configuration.
930    #[serde(default)]
931    pub session: SessionConfig,
932    /// RFC-025: Mount system configuration (auto-promotion scanner).
933    #[serde(default)]
934    pub mounts: MountsConfig,
935    /// ClawHub marketplace configuration.
936    #[serde(default)]
937    pub marketplace: MarketplaceConfig,
938    /// Calendar configuration.
939    #[serde(default)]
940    pub calendar: CalendarConfig,
941    /// Email configuration.
942    #[serde(default)]
943    pub email: EmailConfig,
944    /// Agent history log configuration.
945    #[serde(default)]
946    pub agent_log: AgentLogConfig,
947}
948
949/// Kernel configuration.
950#[derive(Debug, Clone, Deserialize, Serialize)]
951pub struct KernelConfig {
952    /// Path to the workspace directory.
953    #[serde(default = "default_workspace")]
954    pub workspace: String,
955    /// Broadcast capacity for the event bus.
956    #[serde(default = "default_event_bus_capacity")]
957    pub event_bus_capacity: usize,
958    /// Maximum number of concurrent agents.
959    #[serde(default = "default_max_agents")]
960    pub max_agents: usize,
961}
962
963fn default_workspace() -> String {
964    dirs_home().unwrap_or_else(|| ".".into())
965}
966
967fn dirs_home() -> Option<String> {
968    dirs::home_dir().map(|h| format!("{}/.oxios/workspace", h.display()))
969}
970
971fn default_event_bus_capacity() -> usize {
972    256
973}
974
975fn default_max_agents() -> usize {
976    10
977}
978
979impl Default for KernelConfig {
980    fn default() -> Self {
981        Self {
982            workspace: default_workspace(),
983            event_bus_capacity: default_event_bus_capacity(),
984            max_agents: 10,
985        }
986    }
987}
988
989/// Gateway configuration.
990#[derive(Debug, Clone, Deserialize, Serialize)]
991pub struct GatewayConfig {
992    /// Host to bind the gateway to.
993    #[serde(default = "default_gateway_host")]
994    pub host: String,
995    /// Port for the gateway server.
996    #[serde(default = "default_gateway_port")]
997    pub port: u16,
998    /// Expose `/api-docs` (Swagger UI) and `/openapi.json`.
999    ///
1000    /// For safety this is gated to localhost-only binds (127.0.0.0/8, ::1,
1001    /// "localhost"). Setting this to `true` while binding to a public address
1002    /// is a no-op. Default: `false`.
1003    ///
1004    /// Why: Swagger UI + the full OpenAPI schema expand the attack surface
1005    /// (route discovery, parameter names, security scheme details). Local
1006    /// dev typically wants them; production typically does not.
1007    #[serde(default)]
1008    pub expose_api_docs: bool,
1009    /// RFC-024 SP1: ceiling on `send_and_wait` for HTTP request-response
1010    /// matching. The HTTP layer returns 504 Gateway Timeout when the
1011    /// orchestrator does not respond within this duration.
1012    #[serde(default = "default_response_timeout_secs")]
1013    pub response_timeout_secs: u64,
1014    /// RFC-024 SP1: in-memory replay buffer tuning (per channel).
1015    #[serde(default)]
1016    pub reliability: GatewayReliabilityConfig,
1017}
1018
1019/// RFC-024 SP1: in-memory replay buffer tuning.
1020#[derive(Debug, Clone, Serialize, Deserialize)]
1021pub struct GatewayReliabilityConfig {
1022    /// Per-channel replay buffer size. Older messages are evicted when
1023    /// the buffer is full.
1024    #[serde(default = "default_replay_buffer_size")]
1025    pub replay_buffer_size: usize,
1026    /// How long a message stays in the replay buffer.
1027    #[serde(default = "default_replay_ttl_secs")]
1028    pub replay_ttl_secs: u64,
1029}
1030
1031impl Default for GatewayReliabilityConfig {
1032    fn default() -> Self {
1033        Self {
1034            replay_buffer_size: default_replay_buffer_size(),
1035            replay_ttl_secs: default_replay_ttl_secs(),
1036        }
1037    }
1038}
1039
1040fn default_response_timeout_secs() -> u64 {
1041    120
1042}
1043fn default_replay_buffer_size() -> usize {
1044    512
1045}
1046fn default_replay_ttl_secs() -> u64 {
1047    60
1048}
1049
1050impl GatewayConfig {
1051    /// Whether the gateway may expose `/api-docs` and `/openapi.json`.
1052    ///
1053    /// Returns `true` only when both:
1054    /// - `expose_api_docs` is explicitly enabled, AND
1055    /// - the bind address is a loopback address.
1056    pub fn should_expose_api_docs(&self) -> bool {
1057        if !self.expose_api_docs {
1058            return false;
1059        }
1060        let h = self.host.trim();
1061        h == "127.0.0.1" || h == "::1" || h == "localhost" || h.starts_with("127.")
1062    }
1063}
1064
1065/// ClawHub marketplace configuration.
1066#[derive(Debug, Clone, Deserialize, Serialize)]
1067pub struct MarketplaceConfig {
1068    /// Base URL for the ClawHub registry.
1069    /// Defaults to `https://clawhub.ai`.
1070    #[serde(default)]
1071    pub base_url: Option<String>,
1072    /// Whether the marketplace is enabled.
1073    #[serde(default = "default_true")]
1074    pub enabled: bool,
1075    /// Skills.sh (Vercel Labs ecosystem) configuration.
1076    #[serde(default)]
1077    pub skills_sh: SkillsShConfig,
1078}
1079
1080/// Skills.sh registry configuration.
1081#[derive(Debug, Clone, Deserialize, Serialize)]
1082pub struct SkillsShConfig {
1083    /// Base URL for the Skills.sh API.
1084    /// Defaults to `https://skills.sh`.
1085    #[serde(default)]
1086    pub base_url: Option<String>,
1087    /// API key for Skills.sh authentication.
1088    /// Falls back to `SKILLS_SH_TOKEN` env var if not set.
1089    #[serde(default)]
1090    pub api_key: Option<String>,
1091    /// Whether Skills.sh integration is enabled.
1092    #[serde(default = "default_true")]
1093    pub enabled: bool,
1094}
1095
1096impl Default for MarketplaceConfig {
1097    fn default() -> Self {
1098        Self {
1099            base_url: Some("https://clawhub.ai".to_string()),
1100            enabled: true,
1101            skills_sh: SkillsShConfig::default(),
1102        }
1103    }
1104}
1105
1106impl Default for SkillsShConfig {
1107    fn default() -> Self {
1108        Self {
1109            base_url: None,
1110            api_key: None,
1111            enabled: true,
1112        }
1113    }
1114}
1115
1116/// Calendar configuration.
1117#[derive(Debug, Clone, Deserialize, Serialize)]
1118pub struct CalendarConfig {
1119    /// Enable the calendar system.
1120    #[serde(default)]
1121    pub enabled: bool,
1122    /// Default timezone for events.
1123    #[serde(default = "default_calendar_timezone")]
1124    pub timezone: String,
1125    /// Default reminder minutes for new events.
1126    #[serde(default = "default_reminder_minutes")]
1127    pub default_reminder_minutes: Vec<u32>,
1128    /// Alarm dispatch channels.
1129    #[serde(default)]
1130    pub alarm_channels: Vec<String>,
1131    /// Journal sync mode: "on_open", "midnight", "both".
1132    #[serde(default = "default_journal_sync")]
1133    pub journal_sync: String,
1134    /// Show cron jobs on the calendar.
1135    #[serde(default = "default_true")]
1136    pub system_calendar: bool,
1137    /// Days after which old events are archived.
1138    #[serde(default = "default_archive_days")]
1139    pub archive_after_days: u32,
1140}
1141
1142fn default_calendar_timezone() -> String {
1143    "Asia/Seoul".to_string()
1144}
1145
1146fn default_reminder_minutes() -> Vec<u32> {
1147    vec![15]
1148}
1149
1150fn default_journal_sync() -> String {
1151    "on_open".to_string()
1152}
1153
1154fn default_archive_days() -> u32 {
1155    365
1156}
1157
1158impl Default for CalendarConfig {
1159    fn default() -> Self {
1160        Self {
1161            enabled: false,
1162            timezone: default_calendar_timezone(),
1163            default_reminder_minutes: default_reminder_minutes(),
1164            alarm_channels: vec![],
1165            journal_sync: default_journal_sync(),
1166            system_calendar: true,
1167            archive_after_days: default_archive_days(),
1168        }
1169    }
1170}
1171
1172/// Email configuration.
1173///
1174/// Controls SMTP email sending. When enabled, agents gain the `send_email` tool.
1175/// v1 sends to the user's own email only.
1176#[derive(Debug, Clone, Deserialize, Serialize)]
1177pub struct EmailConfig {
1178    /// Enable the email system.
1179    #[serde(default)]
1180    pub enabled: bool,
1181    /// The user's email address (used as both sender and default recipient).
1182    #[serde(default)]
1183    pub my_email: String,
1184    /// SMTP provider preset ("gmail", "icloud", "fastmail", "custom").
1185    #[serde(default = "default_email_provider")]
1186    pub provider: SmtpProvider,
1187    /// SMTP host (auto-filled from provider if empty).
1188    #[serde(default)]
1189    pub host: String,
1190    /// SMTP port (auto-filled from provider if 0).
1191    #[serde(default)]
1192    pub port: u16,
1193    /// TLS mode (auto-filled from provider if None).
1194    #[serde(default)]
1195    pub tls: Option<SmtpTls>,
1196    /// SMTP auth username (defaults to `my_email` if empty).
1197    #[serde(default)]
1198    pub user: String,
1199    /// Credential store key for the SMTP password.
1200    /// Falls back to `OXIOS_EMAIL_PASSWORD` env var.
1201    #[serde(default = "default_email_secret_ref")]
1202    pub secret_ref: String,
1203    /// Maximum emails per hour (rate limit, default: 10).
1204    #[serde(default = "default_rate_limit_emails")]
1205    pub rate_limit_per_hour: usize,
1206}
1207
1208fn default_email_provider() -> SmtpProvider {
1209    SmtpProvider::Gmail
1210}
1211
1212fn default_email_secret_ref() -> String {
1213    "email_smtp".to_string()
1214}
1215
1216fn default_rate_limit_emails() -> usize {
1217    10
1218}
1219
1220impl Default for EmailConfig {
1221    fn default() -> Self {
1222        Self {
1223            enabled: false,
1224            my_email: String::new(),
1225            provider: default_email_provider(),
1226            host: String::new(),
1227            port: 0,
1228            tls: None,
1229            user: String::new(),
1230            secret_ref: default_email_secret_ref(),
1231            rate_limit_per_hour: default_rate_limit_emails(),
1232        }
1233    }
1234}
1235
1236impl EmailConfig {
1237    /// Resolve the effective provider, falling back to Gmail.
1238    pub fn provider(&self) -> SmtpProvider {
1239        self.provider
1240    }
1241}
1242
1243fn default_gateway_host() -> String {
1244    "127.0.0.1".into()
1245}
1246
1247fn default_gateway_port() -> u16 {
1248    4200
1249}
1250
1251impl Default for GatewayConfig {
1252    fn default() -> Self {
1253        Self {
1254            host: default_gateway_host(),
1255            port: default_gateway_port(),
1256            expose_api_docs: false,
1257            response_timeout_secs: default_response_timeout_secs(),
1258            reliability: GatewayReliabilityConfig::default(),
1259        }
1260    }
1261}
1262
1263/// Execution mode for commands.
1264///
1265/// - `Structured`: Binary allowlist + metacharacter blocking (recommended)
1266/// - `Shell`: Raw bash execution (dangerous, requires `allow_shell_mode=true`)
1267#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize, Default)]
1268#[serde(rename_all = "lowercase")]
1269pub enum ExecMode {
1270    /// Structured binary execution with allowlist and metacharacter blocking.
1271    #[default]
1272    Structured,
1273    /// Shell execution via `bash -c`. DANGEROUS — requires explicit enable.
1274    Shell,
1275}
1276
1277/// Execution allowlist behavior mode.
1278#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
1279#[serde(rename_all = "snake_case")]
1280#[derive(Default)]
1281pub enum AllowlistMode {
1282    /// All binaries are permitted (development only).
1283    Permissive,
1284    /// Only binaries in `allowed_commands` may execute.
1285    #[default]
1286    Enforced,
1287}
1288
1289/// Exec configuration.
1290///
1291/// Governs how the kernel dispatches commands for execution.
1292#[derive(Debug, Clone, Deserialize, Serialize)]
1293pub struct ExecConfig {
1294    /// Default execution mode.
1295    #[serde(default)]
1296    pub default_mode: ExecMode,
1297    /// Allow shell mode. DANGEROUS — should be false in production.
1298    #[serde(default = "default_false")]
1299    pub allow_shell_mode: bool,
1300    /// Commands allowed to run on the host.
1301    /// If empty, *all* bare-name commands are permitted (development mode).
1302    #[serde(default)]
1303    pub allowed_commands: Vec<String>,
1304    /// Allowlist enforcement mode.
1305    /// `Permissive` = empty list means all allowed (dev mode).
1306    /// `Enforced` = only listed commands allowed (production).
1307    #[serde(default)]
1308    pub allowlist_mode: AllowlistMode,
1309    /// Default timeout for an exec call in seconds.
1310    #[serde(default = "default_exec_timeout")]
1311    pub default_timeout_secs: u64,
1312    /// Maximum allowed timeout for an exec call in seconds.
1313    #[serde(default = "default_exec_max_timeout")]
1314    pub max_timeout_secs: u64,
1315}
1316
1317fn default_false() -> bool {
1318    false
1319}
1320
1321fn default_exec_timeout() -> u64 {
1322    120
1323}
1324
1325fn default_exec_max_timeout() -> u64 {
1326    600
1327}
1328
1329impl ExecConfig {
1330    /// Check whether a binary / command name is allowed to execute.
1331    ///
1332    /// In `Permissive` mode, returns `true` when `allowed_commands` is empty
1333    /// (all allowed) **or** when the name is present in the allow-list.
1334    ///
1335    /// In `Enforced` mode, only names present in the allow-list are permitted.
1336    pub fn is_binary_allowed(&self, name: &str) -> bool {
1337        match self.allowlist_mode {
1338            AllowlistMode::Permissive => {
1339                self.allowed_commands.is_empty() || self.allowed_commands.iter().any(|c| c == name)
1340            }
1341            AllowlistMode::Enforced => self.allowed_commands.iter().any(|c| c == name),
1342        }
1343    }
1344}
1345
1346impl Default for ExecConfig {
1347    fn default() -> Self {
1348        Self {
1349            default_mode: ExecMode::default(),
1350            allow_shell_mode: default_false(),
1351            allowed_commands: Vec::new(),
1352            allowlist_mode: AllowlistMode::default(),
1353            default_timeout_secs: default_exec_timeout(),
1354            max_timeout_secs: default_exec_max_timeout(),
1355        }
1356    }
1357}
1358
1359/// Scheduler configuration (inspired by AIOS / AgentRM).
1360#[derive(Debug, Clone, Deserialize, Serialize)]
1361pub struct SchedulerConfig {
1362    /// Maximum number of concurrent agent tasks.
1363    #[serde(default = "default_max_concurrent")]
1364    pub max_concurrent: usize,
1365    /// Maximum LLM API calls per minute (rate limiting).
1366    #[serde(default = "default_rate_limit")]
1367    pub rate_limit_per_minute: u32,
1368    /// Timeout in seconds before a running task is considered a zombie.
1369    #[serde(default = "default_zombie_timeout")]
1370    pub zombie_timeout_secs: u64,
1371}
1372
1373fn default_max_concurrent() -> usize {
1374    5
1375}
1376
1377fn default_rate_limit() -> u32 {
1378    60
1379}
1380
1381fn default_zombie_timeout() -> u64 {
1382    300
1383}
1384
1385impl Default for SchedulerConfig {
1386    fn default() -> Self {
1387        Self {
1388            max_concurrent: default_max_concurrent(),
1389            rate_limit_per_minute: default_rate_limit(),
1390            zombie_timeout_secs: default_zombie_timeout(),
1391        }
1392    }
1393}
1394
1395/// Orchestrator configuration (Ouroboros protocol execution).
1396#[derive(Debug, Clone, Deserialize, Serialize)]
1397pub struct OrchestratorConfig {
1398    /// Maximum evolution iterations (0 = evaluate only, no evolution).
1399    /// Default: 3.
1400    #[serde(default = "default_max_evolution_iterations")]
1401    pub max_evolution_iterations: u32,
1402
1403    /// Minimum evaluation score for task to be considered passed (0.0–1.0).
1404    /// Default: 0.8.
1405    #[serde(default = "default_min_evaluation_score")]
1406    pub min_evaluation_score: f64,
1407}
1408
1409fn default_max_evolution_iterations() -> u32 {
1410    3
1411}
1412
1413fn default_min_evaluation_score() -> f64 {
1414    0.8
1415}
1416
1417impl Default for OrchestratorConfig {
1418    fn default() -> Self {
1419        Self {
1420            max_evolution_iterations: default_max_evolution_iterations(),
1421            min_evaluation_score: default_min_evaluation_score(),
1422        }
1423    }
1424}
1425
1426/// Intent engine configuration (RFC-027 unified intent handling).
1427///
1428/// Controls the unified intent engine that replaces the legacy Ouroboros
1429/// five-phase protocol: `assess` → `crystallize` → `execute` → `review` → `retry`.
1430#[derive(Debug, Clone, Serialize, Deserialize)]
1431pub struct IntentConfig {
1432    /// Maximum retry attempts when a Substantial task fails review.
1433    /// Set to 0 to disable retries entirely.
1434    /// Default: 2.
1435    #[serde(default = "default_intent_max_retries")]
1436    pub max_retries: u32,
1437
1438    /// Minimum review score (0.0–1.0) required for a verdict to pass.
1439    /// Reviews below this threshold trigger a retry.
1440    /// Default: 0.7.
1441    #[serde(default = "default_intent_score_threshold")]
1442    pub score_threshold: f64,
1443
1444    /// Maximum clarification rounds before forcing the task to proceed
1445    /// with the system's best-guess understanding.
1446    /// Default: 3.
1447    #[serde(default = "default_intent_max_clarify_rounds")]
1448    pub max_clarify_rounds: u32,
1449
1450    /// Whether to retry Substantial tasks whose review verdict fails.
1451    /// When false, a failing review is reported back to the user directly.
1452    /// Default: true.
1453    #[serde(default = "default_intent_enable_retry")]
1454    pub enable_retry: bool,
1455
1456    /// Optional lightweight model ID for `assess`/`crystallize`/`review` calls.
1457    /// When None, the engine uses the resolver's default model.
1458    /// Default: None.
1459    #[serde(default)]
1460    pub lightweight_model: Option<String>,
1461}
1462
1463fn default_intent_max_retries() -> u32 {
1464    2
1465}
1466
1467fn default_intent_score_threshold() -> f64 {
1468    0.7
1469}
1470
1471fn default_intent_max_clarify_rounds() -> u32 {
1472    3
1473}
1474
1475fn default_intent_enable_retry() -> bool {
1476    true
1477}
1478
1479impl Default for IntentConfig {
1480    fn default() -> Self {
1481        Self {
1482            max_retries: default_intent_max_retries(),
1483            score_threshold: default_intent_score_threshold(),
1484            max_clarify_rounds: default_intent_max_clarify_rounds(),
1485            enable_retry: default_intent_enable_retry(),
1486            lightweight_model: None,
1487        }
1488    }
1489}
1490
1491/// Context manager configuration (inspired by AIOS).
1492#[derive(Debug, Clone, Deserialize, Serialize)]
1493pub struct ContextConfig {
1494    /// Maximum tokens in the active (in-context) tier.
1495    #[serde(default = "default_active_limit")]
1496    pub active_limit_tokens: usize,
1497    /// Maximum entries in the cache tier.
1498    #[serde(default = "default_cache_limit")]
1499    pub cache_limit_entries: usize,
1500}
1501
1502fn default_active_limit() -> usize {
1503    100_000
1504}
1505
1506fn default_cache_limit() -> usize {
1507    50
1508}
1509
1510impl Default for ContextConfig {
1511    fn default() -> Self {
1512        Self {
1513            active_limit_tokens: default_active_limit(),
1514            cache_limit_entries: default_cache_limit(),
1515        }
1516    }
1517}
1518
1519/// Security/access control configuration (inspired by OWASP Agentic AI).
1520#[derive(Debug, Clone, Deserialize, Serialize)]
1521pub struct SecurityConfig {
1522    /// Default allowed tools for agents (least privilege).
1523    #[serde(default = "default_allowed_tools")]
1524    pub allowed_tools: Vec<String>,
1525    /// Whether agents can make network requests by default.
1526    #[serde(default)]
1527    pub network_access: bool,
1528    /// Maximum execution time in seconds for agent tasks.
1529    #[serde(default = "default_max_exec_time")]
1530    pub max_execution_time_secs: u64,
1531    /// Maximum memory in MB for agent tasks.
1532    #[serde(default = "default_max_memory")]
1533    pub max_memory_mb: u64,
1534    /// Whether agents can fork sub-agents by default.
1535    #[serde(default)]
1536    pub can_fork: bool,
1537    /// Maximum audit log entries to retain.
1538    #[serde(default = "default_max_audit")]
1539    pub max_audit_entries: usize,
1540    /// Enable API key authentication.
1541    #[serde(default)]
1542    pub auth_enabled: bool,
1543    /// Allowed CORS origins.
1544    #[serde(default = "default_cors_origins")]
1545    pub cors_origins: Vec<String>,
1546    /// Path for audit log file (optional, enables file-based persistence).
1547    #[serde(default)]
1548    pub audit_log_path: Option<String>,
1549    /// Rate limit for API endpoints (requests per minute).
1550    #[serde(default = "default_rate_limit_per_minute")]
1551    pub rate_limit_per_minute: u32,
1552}
1553
1554fn default_allowed_tools() -> Vec<String> {
1555    vec![
1556        "read".to_string(),
1557        "write".to_string(),
1558        "edit".to_string(),
1559        "bash".to_string(),
1560        "grep".to_string(),
1561        "find".to_string(),
1562        "exec".to_string(),
1563    ]
1564}
1565
1566fn default_max_exec_time() -> u64 {
1567    300
1568}
1569
1570fn default_max_memory() -> u64 {
1571    512
1572}
1573
1574fn default_max_audit() -> usize {
1575    10_000
1576}
1577
1578fn default_rate_limit_per_minute() -> u32 {
1579    120
1580}
1581
1582fn default_cors_origins() -> Vec<String> {
1583    // Browsers treat `localhost` and `127.0.0.1` as distinct origins, so both
1584    // must be allow-listed or cross-origin requests silently fail CORS checks.
1585    // 4200 = backend that also serves the production SPA (same origin).
1586    // 5173 = Vite dev server (`bun dev` in web/).
1587    vec![
1588        "http://localhost:4200".to_string(),
1589        "http://127.0.0.1:4200".to_string(),
1590        "http://localhost:5173".to_string(),
1591        "http://127.0.0.1:5173".to_string(),
1592    ]
1593}
1594
1595impl Default for SecurityConfig {
1596    fn default() -> Self {
1597        Self {
1598            allowed_tools: default_allowed_tools(),
1599            network_access: false,
1600            max_execution_time_secs: default_max_exec_time(),
1601            max_memory_mb: default_max_memory(),
1602            can_fork: false,
1603            max_audit_entries: default_max_audit(),
1604            auth_enabled: false,
1605            cors_origins: default_cors_origins(),
1606            audit_log_path: None,
1607            rate_limit_per_minute: default_rate_limit_per_minute(),
1608        }
1609    }
1610}
1611
1612/// Persona system configuration.
1613#[derive(Debug, Clone, Deserialize, Serialize)]
1614pub struct PersonaConfig {
1615    /// Default persona ID to activate on startup.
1616    #[serde(default)]
1617    pub default_persona_id: Option<String>,
1618    /// Maximum concurrent personas.
1619    #[serde(default = "default_max_concurrent_personas")]
1620    pub max_concurrent_personas: usize,
1621}
1622
1623fn default_max_concurrent_personas() -> usize {
1624    5
1625}
1626
1627impl Default for PersonaConfig {
1628    fn default() -> Self {
1629        Self {
1630            default_persona_id: Some("dev".to_string()),
1631            max_concurrent_personas: default_max_concurrent_personas(),
1632        }
1633    }
1634}
1635
1636/// MCP server configuration loaded from config.toml.
1637///
1638/// Each key is a server name; the value is a table with:
1639/// - `command`: executable to run (e.g. "npx", "python")
1640/// - `args`: arguments array
1641/// - `env`: optional map of environment variables
1642/// - `enabled`: whether to start this server on boot (default: true)
1643#[derive(Debug, Clone, Deserialize, Serialize, Default)]
1644pub struct McpConfig {
1645    /// Map of server-name → server definition.
1646    #[serde(default)]
1647    pub servers: std::collections::HashMap<String, McpServerDef>,
1648}
1649
1650/// A single MCP server definition in config.toml.
1651#[derive(Debug, Clone, Deserialize, Serialize)]
1652pub struct McpServerDef {
1653    /// Command to execute.
1654    pub command: String,
1655    /// Arguments passed to the command.
1656    #[serde(default)]
1657    pub args: Vec<String>,
1658    /// Environment variables.
1659    #[serde(default)]
1660    pub env: std::collections::HashMap<String, String>,
1661    /// Whether this server is enabled (default: true).
1662    #[serde(default = "default_mcp_enabled")]
1663    pub enabled: bool,
1664}
1665
1666fn default_mcp_enabled() -> bool {
1667    true
1668}
1669
1670/// Git version control configuration.
1671#[derive(Debug, Clone, Deserialize, Serialize)]
1672pub struct GitConfig {
1673    /// Enable automatic commits for state changes.
1674    #[serde(default = "default_true")]
1675    pub auto_commit: bool,
1676}
1677
1678impl Default for GitConfig {
1679    fn default() -> Self {
1680        Self { auto_commit: true }
1681    }
1682}
1683
1684/// Audit trail configuration.
1685#[derive(Debug, Clone, Deserialize, Serialize)]
1686pub struct AuditConfig {
1687    /// Maximum audit entries before pruning.
1688    #[serde(default = "default_audit_max_entries")]
1689    pub max_entries: usize,
1690    /// Enable audit trail.
1691    #[serde(default = "default_true")]
1692    pub enabled: bool,
1693}
1694
1695fn default_audit_max_entries() -> usize {
1696    100_000
1697}
1698
1699impl Default for AuditConfig {
1700    fn default() -> Self {
1701        Self {
1702            max_entries: default_audit_max_entries(),
1703            enabled: true,
1704        }
1705    }
1706}
1707
1708/// Budget enforcement configuration.
1709#[derive(Debug, Clone, Deserialize, Serialize)]
1710pub struct BudgetConfig {
1711    /// Default token budget per agent (0 = unlimited).
1712    #[serde(default)]
1713    pub default_token_budget: u64,
1714    /// Default call budget per agent (0 = unlimited).
1715    #[serde(default)]
1716    pub default_calls_budget: u64,
1717    /// Default budget window in seconds.
1718    #[serde(default = "default_budget_window")]
1719    pub default_window_secs: u64,
1720    /// Enable budget enforcement.
1721    #[serde(default = "default_true")]
1722    pub enabled: bool,
1723}
1724
1725fn default_budget_window() -> u64 {
1726    3600
1727}
1728
1729impl Default for BudgetConfig {
1730    fn default() -> Self {
1731        Self {
1732            default_token_budget: 0,
1733            default_calls_budget: 0,
1734            default_window_secs: default_budget_window(),
1735            enabled: true,
1736        }
1737    }
1738}
1739
1740/// Resource monitor configuration.
1741#[derive(Debug, Clone, Deserialize, Serialize)]
1742pub struct ResourceMonitorConfig {
1743    /// Snapshot interval in seconds.
1744    #[serde(default = "default_rm_interval")]
1745    pub interval_secs: u64,
1746    /// Maximum history entries.
1747    #[serde(default = "default_rm_history_max")]
1748    pub history_max: usize,
1749    /// CPU threshold for overload.
1750    #[serde(default = "default_rm_cpu_threshold")]
1751    pub cpu_threshold: f32,
1752    /// Memory threshold for overload (percentage).
1753    #[serde(default = "default_rm_mem_threshold")]
1754    pub memory_threshold: f32,
1755    /// Load average threshold for overload.
1756    #[serde(default = "default_rm_load_threshold")]
1757    pub load_threshold: f32,
1758}
1759
1760fn default_rm_interval() -> u64 {
1761    60
1762}
1763
1764fn default_rm_history_max() -> usize {
1765    60
1766}
1767
1768fn default_rm_cpu_threshold() -> f32 {
1769    90.0
1770}
1771
1772fn default_rm_mem_threshold() -> f32 {
1773    90.0
1774}
1775
1776fn default_rm_load_threshold() -> f32 {
1777    8.0
1778}
1779
1780impl Default for ResourceMonitorConfig {
1781    fn default() -> Self {
1782        Self {
1783            interval_secs: default_rm_interval(),
1784            history_max: default_rm_history_max(),
1785            cpu_threshold: default_rm_cpu_threshold(),
1786            memory_threshold: default_rm_mem_threshold(),
1787            load_threshold: default_rm_load_threshold(),
1788        }
1789    }
1790}
1791
1792/// OpenTelemetry tracing configuration.
1793#[derive(Debug, Clone, Deserialize, Serialize)]
1794pub struct OtelConfig {
1795    /// Enable OTLP export (default: false).
1796    #[serde(default)]
1797    pub enabled: bool,
1798    /// OTLP gRPC endpoint.
1799    #[serde(default = "default_otel_endpoint")]
1800    pub endpoint: String,
1801    /// Service name for traces.
1802    #[serde(default = "default_otel_service_name")]
1803    pub service_name: String,
1804    /// Sampling ratio (0.0 to 1.0).
1805    #[serde(default = "default_otel_sampling_ratio")]
1806    pub sampling_ratio: f64,
1807}
1808
1809fn default_otel_endpoint() -> String {
1810    "http://localhost:4317".into()
1811}
1812
1813fn default_otel_service_name() -> String {
1814    "oxios".into()
1815}
1816
1817fn default_otel_sampling_ratio() -> f64 {
1818    1.0
1819}
1820
1821impl Default for OtelConfig {
1822    fn default() -> Self {
1823        Self {
1824            enabled: false,
1825            endpoint: default_otel_endpoint(),
1826            service_name: default_otel_service_name(),
1827            sampling_ratio: default_otel_sampling_ratio(),
1828        }
1829    }
1830}
1831
1832/// Agent history log configuration.
1833#[derive(Debug, Clone, Serialize, Deserialize)]
1834pub struct AgentLogConfig {
1835    /// Maximum number of agent records to keep (0 = unlimited).
1836    #[serde(default = "default_agent_log_max_entries")]
1837    pub max_entries: usize,
1838    /// TTL for agent records in hours (0 = unlimited).
1839    #[serde(default = "default_agent_log_ttl_hours")]
1840    pub ttl_hours: u64,
1841    /// Max tool_calls per agent to persist (0 = unlimited).
1842    #[serde(default = "default_agent_log_max_tool_calls")]
1843    pub max_tool_calls_per_agent: usize,
1844    /// How many agents to prune per cycle.
1845    #[serde(default = "default_agent_log_prune_batch")]
1846    pub prune_batch_size: usize,
1847    /// Path to the SQLite database file (empty = default).
1848    #[serde(default)]
1849    pub db_path: String,
1850}
1851
1852fn default_agent_log_max_entries() -> usize {
1853    10_000
1854}
1855fn default_agent_log_ttl_hours() -> u64 {
1856    720
1857}
1858fn default_agent_log_max_tool_calls() -> usize {
1859    500
1860}
1861fn default_agent_log_prune_batch() -> usize {
1862    100
1863}
1864
1865impl Default for AgentLogConfig {
1866    fn default() -> Self {
1867        Self {
1868            max_entries: 10_000,
1869            ttl_hours: 720,
1870            max_tool_calls_per_agent: 500,
1871            prune_batch_size: 100,
1872            db_path: String::new(),
1873        }
1874    }
1875}
1876
1877/// Logging configuration.
1878#[derive(Debug, Clone, Deserialize, Serialize)]
1879pub struct LoggingConfig {
1880    /// Log format: "pretty", "json", or "compact".
1881    #[serde(default = "default_log_format")]
1882    pub format: String,
1883    /// Log level override (e.g. "info", "debug"). Falls back to RUST_LOG env var.
1884    #[serde(default)]
1885    pub level: Option<String>,
1886}
1887
1888fn default_log_format() -> String {
1889    "pretty".into()
1890}
1891
1892impl Default for LoggingConfig {
1893    fn default() -> Self {
1894        Self {
1895            format: default_log_format(),
1896            level: None,
1897        }
1898    }
1899}
1900
1901/// Headless browser configuration.
1902///
1903/// Engine configuration. Passes through to `oxi-sdk` browser tools.
1904/// with an `enabled` toggle. The engine config is passed through directly
1905/// to the browser — no field-by-field duplication.
1906#[derive(Debug, Clone, Deserialize, Serialize)]
1907pub struct BrowserConfig {
1908    /// Enable the browser integration.
1909    #[serde(default = "default_browser_enabled")]
1910    pub enabled: bool,
1911
1912    /// Engine configuration — passed to oxi-sdk's `native_browser_tools_with_config()`.
1913    ///
1914    /// All fields have sensible defaults; override only what you need:
1915    ///
1916    /// ```toml
1917    /// [browser.engine]
1918    /// user_agent = "MyBot/1.0"
1919    /// obey_robots = false
1920    /// js_timeout_ms = 10000
1921    /// ```
1922    #[serde(default)]
1923    pub engine: serde_json::Value,
1924}
1925
1926fn default_browser_enabled() -> bool {
1927    true
1928}
1929
1930impl Default for BrowserConfig {
1931    fn default() -> Self {
1932        Self {
1933            enabled: true,
1934            engine: serde_json::json!({}),
1935        }
1936    }
1937}
1938
1939/// Loads configuration from a TOML file.
1940pub fn load_config(path: &std::path::Path) -> anyhow::Result<OxiosConfig> {
1941    let content = std::fs::read_to_string(path)?;
1942    let config: OxiosConfig = toml::from_str(&content)?;
1943    let (errors, warnings) = config.validate();
1944    for w in warnings {
1945        tracing::warn!("config: {}", w);
1946    }
1947    if !errors.is_empty() {
1948        let msg = errors.join("; ");
1949        anyhow::bail!("Configuration validation failed: {msg}");
1950    }
1951    Ok(config)
1952}
1953
1954impl OxiosConfig {
1955    /// Returns the effective API key from the engine config.
1956    pub fn api_key(&self) -> Option<String> {
1957        self.engine.api_key.clone().filter(|k| !k.is_empty())
1958    }
1959
1960    /// Validate configuration values and return a list of warnings.
1961    /// Returns (errors, warnings). Empty errors = valid config.
1962    pub fn validate(&self) -> (Vec<String>, Vec<String>) {
1963        let mut errors = Vec::new();
1964        let mut warnings = Vec::new();
1965
1966        // Kernel validation
1967        if self.kernel.max_agents == 0 {
1968            errors.push("kernel.max_agents must be > 0".into());
1969        }
1970        if self.kernel.workspace.is_empty() {
1971            errors.push("kernel.workspace must not be empty".into());
1972        }
1973
1974        // Gateway validation
1975        if self.gateway.port == 0 {
1976            errors.push("gateway.port must be > 0".into());
1977        }
1978        if self.gateway.port < 1024 && self.gateway.host == "0.0.0.0" {
1979            warnings.push("Running on port <1024 as 0.0.0.0 may require root".into());
1980        }
1981
1982        // Scheduler validation
1983        if self.scheduler.max_concurrent == 0 {
1984            warnings.push("scheduler.max_concurrent is 0 — no tasks will run".into());
1985        }
1986        if self.scheduler.zombie_timeout_secs == 0 {
1987            errors.push("scheduler.zombie_timeout_secs must be > 0".into());
1988        }
1989
1990        // Cron validation
1991        for (name, job) in &self.cron.jobs {
1992            if job.schedule.is_empty() {
1993                errors.push(format!("cron.jobs.{name}: schedule is empty"));
1994            } else {
1995                // Normalize 5-field to 6-field (prepend "0 " for seconds)
1996                let normalized = {
1997                    let fields: Vec<&str> = job.schedule.split_whitespace().collect();
1998                    match fields.len() {
1999                        5 => format!("0 {}", job.schedule),
2000                        _ => job.schedule.clone(),
2001                    }
2002                };
2003                if Schedule::from_str(&normalized).is_err() {
2004                    errors.push(format!(
2005                        "cron.jobs.{}: invalid cron expression '{}'",
2006                        name, job.schedule
2007                    ));
2008                }
2009            }
2010            if job.goal.is_empty() {
2011                errors.push(format!("cron.jobs.{name}: goal is empty"));
2012            }
2013        }
2014
2015        // Security validation
2016        if self.security.max_execution_time_secs == 0 {
2017            warnings.push("security.max_execution_time_secs is 0 — no timeout".into());
2018        }
2019
2020        // Audit validation
2021        if self.audit.max_entries == 0 {
2022            warnings.push("audit.max_entries is 0 — audit will never prune".into());
2023        }
2024
2025        // Budget validation
2026        if self.budget.default_window_secs == 0 {
2027            warnings.push("budget.default_window_secs is 0 — no time window".into());
2028        }
2029
2030        // Gateway field-level validation
2031        if self.gateway.response_timeout_secs == 0 {
2032            errors.push("gateway.response_timeout_secs must be > 0".into());
2033        }
2034
2035        // Engine: warn when an API key is committed to config in plaintext.
2036        // The auth store and env-var fallback are preferred for secret hygiene.
2037        if self.engine.api_key.as_ref().is_some_and(|k| !k.is_empty()) {
2038            warnings.push(
2039                "engine.api_key is set in config — prefer the oxi auth store or env var to avoid storing a secret on disk"
2040                    .into(),
2041            );
2042        }
2043
2044        // MCP server validation: reject empty commands (would spawn a no-op).
2045        for (name, server) in &self.mcp.servers {
2046            if server.command.trim().is_empty() {
2047                errors.push(format!("mcp.servers.{name}: command must not be empty"));
2048            }
2049        }
2050
2051        // Session validation
2052        if self.session.max_sessions == 0 && self.session.ttl_hours == 0 && self.session.auto_prune
2053        {
2054            warnings.push("session: auto_prune is enabled but both max_sessions and ttl_hours are 0 — nothing will be pruned".into());
2055        }
2056
2057        // Exec validation
2058        if self.exec.default_timeout_secs == 0 {
2059            errors.push("exec.default_timeout_secs must be > 0".into());
2060        }
2061        if self.exec.max_timeout_secs == 0 {
2062            errors.push("exec.max_timeout_secs must be > 0".into());
2063        }
2064        if self.exec.default_timeout_secs > self.exec.max_timeout_secs {
2065            errors.push(format!(
2066                "exec.default_timeout_secs ({}) must not exceed max_timeout_secs ({})",
2067                self.exec.default_timeout_secs, self.exec.max_timeout_secs
2068            ));
2069        }
2070
2071        // Resource monitor validation
2072        if self.resource_monitor.cpu_threshold > 100.0 {
2073            errors.push("resource_monitor.cpu_threshold must be <= 100".into());
2074        }
2075        if self.resource_monitor.memory_threshold > 100.0 {
2076            errors.push("resource_monitor.memory_threshold must be <= 100".into());
2077        }
2078
2079        // Channels validation (message interfaces only)
2080        for name in &self.channels.enabled {
2081            let valid = ["cli", "telegram"];
2082            if !valid.contains(&name.as_str()) {
2083                warnings.push(format!("channels.enabled: unknown channel '{name}'"));
2084            }
2085        }
2086        // Warn if 'web' is listed in channels — it should be in surfaces
2087        if self.channels.enabled.iter().any(|c| c == "web") {
2088            warnings.push(
2089                "channels.enabled: 'web' should be listed under [surfaces], not [channels]".into(),
2090            );
2091        }
2092        if self.channels.enabled.iter().any(|c| c == "telegram")
2093            && std::env::var(&self.channels.telegram.bot_token_env).is_err()
2094        {
2095            warnings.push(format!(
2096                "channels.telegram: {} env var not set — telegram channel will fail",
2097                self.channels.telegram.bot_token_env
2098            ));
2099        }
2100
2101        (errors, warnings)
2102    }
2103}
2104
2105/// Expand `~/` in paths to the user's home directory.
2106///
2107/// Shared utility for path expansion across the binary and kernel.
2108///
2109/// Resolution order for the home directory:
2110/// 1. `$HOME` environment variable (preserves existing behavior).
2111/// 2. `dirs::home_dir()` (works in environments where HOME is unset, e.g.
2112///    systemd units, containers, cron jobs).
2113/// 3. If neither is available, the literal path is returned unchanged so the
2114///    caller still gets a usable `PathBuf` rather than a panic — the failure
2115///    will surface as a normal "path not found" downstream.
2116pub fn expand_home(path: &str) -> std::path::PathBuf {
2117    if let Some(rest) = path.strip_prefix("~/") {
2118        if let Ok(home) = std::env::var("HOME") {
2119            return std::path::PathBuf::from(format!("{home}/{rest}"));
2120        }
2121        if let Some(home) = dirs::home_dir() {
2122            return home.join(rest);
2123        }
2124    }
2125    std::path::PathBuf::from(path)
2126}
2127
2128#[cfg(test)]
2129mod tests {
2130    use super::*;
2131
2132    #[test]
2133    fn test_default_config_validates() {
2134        let config = OxiosConfig::default();
2135        let (errors, _warnings) = config.validate();
2136        assert!(
2137            errors.is_empty(),
2138            "Default config should have no errors: {:?}",
2139            errors
2140        );
2141    }
2142
2143    #[test]
2144    fn test_exec_config_default_allowed_commands() {
2145        let config = ExecConfig::default();
2146        // Default is Enforced mode — empty list means NOTHING allowed.
2147        assert!(config.allowed_commands.is_empty());
2148        assert_eq!(config.allowlist_mode, AllowlistMode::Enforced);
2149        assert!(!config.is_binary_allowed("anything"));
2150        assert!(!config.is_binary_allowed("bash"));
2151    }
2152
2153    #[test]
2154    fn test_exec_config_permissive_mode() {
2155        let config = ExecConfig {
2156            allowlist_mode: AllowlistMode::Permissive,
2157            ..Default::default()
2158        };
2159        // Permissive + empty list = all allowed
2160        assert!(config.is_binary_allowed("anything"));
2161        assert!(config.is_binary_allowed("bash"));
2162    }
2163
2164    #[test]
2165    fn test_is_binary_allowed_with_allowlist() {
2166        let config = ExecConfig {
2167            allowed_commands: vec!["git".into(), "echo".into()],
2168            ..Default::default()
2169        };
2170        assert!(config.is_binary_allowed("git"));
2171        assert!(config.is_binary_allowed("echo"));
2172        assert!(!config.is_binary_allowed("bash"));
2173        assert!(!config.is_binary_allowed("rm"));
2174        assert!(!config.is_binary_allowed("sudo"));
2175    }
2176
2177    #[test]
2178    fn test_expand_home() {
2179        // With HOME set.
2180        let home = std::env::var("HOME").unwrap_or_else(|_| "/tmp/testhome".into());
2181        let expanded = expand_home("~/projects/test");
2182        assert_eq!(
2183            expanded.to_str().unwrap(),
2184            format!("{}/projects/test", home)
2185        );
2186
2187        // Non-tilde path should pass through unchanged.
2188        let abs = expand_home("/absolute/path");
2189        assert_eq!(abs, std::path::PathBuf::from("/absolute/path"));
2190
2191        // Just ~ without slash should not expand.
2192        let bare = expand_home("~something");
2193        assert_eq!(bare, std::path::PathBuf::from("~something"));
2194    }
2195
2196    #[test]
2197    fn test_invalid_cron_expression() {
2198        let mut config = OxiosConfig::default();
2199        config.cron.enabled = true;
2200        config.cron.jobs.insert(
2201            "bad-job".to_string(),
2202            InlineCronJob {
2203                schedule: "not a valid cron".to_string(),
2204                goal: "Test goal".to_string(),
2205                constraints: vec![],
2206                acceptance_criteria: vec![],
2207                toolchain: "default".to_string(),
2208                priority: Priority::Normal,
2209                enabled: true,
2210            },
2211        );
2212
2213        let (errors, _warnings) = config.validate();
2214        assert!(
2215            !errors.is_empty(),
2216            "Expected validation error for invalid cron"
2217        );
2218        let has_cron_error = errors.iter().any(|e| e.contains("invalid cron expression"));
2219        assert!(
2220            has_cron_error,
2221            "Expected 'invalid cron expression' error, got: {:?}",
2222            errors
2223        );
2224    }
2225
2226    #[test]
2227    fn test_config_serialization_roundtrip() {
2228        let config = OxiosConfig::default();
2229
2230        // Serialize to TOML string.
2231        let toml_str = toml::to_string(&config).expect("serialization should succeed");
2232
2233        // Deserialize back.
2234        let deserialized: OxiosConfig =
2235            toml::from_str(&toml_str).expect("deserialization should succeed");
2236
2237        // Key fields should match.
2238        assert_eq!(config.kernel.max_agents, deserialized.kernel.max_agents);
2239        assert_eq!(config.kernel.workspace, deserialized.kernel.workspace);
2240        assert_eq!(config.gateway.host, deserialized.gateway.host);
2241        assert_eq!(config.gateway.port, deserialized.gateway.port);
2242        assert_eq!(
2243            config.exec.default_timeout_secs,
2244            deserialized.exec.default_timeout_secs
2245        );
2246        assert_eq!(
2247            config.exec.max_timeout_secs,
2248            deserialized.exec.max_timeout_secs
2249        );
2250    }
2251
2252    #[test]
2253    fn test_exec_timeout_validation() {
2254        let mut config = OxiosConfig::default();
2255        // default_timeout > max_timeout should be an error.
2256        config.exec.default_timeout_secs = 999;
2257        config.exec.max_timeout_secs = 100;
2258        let (errors, _warnings) = config.validate();
2259        let has_error = errors.iter().any(|e| e.contains("must not exceed"));
2260        assert!(
2261            has_error,
2262            "Expected timeout ordering error, got: {:?}",
2263            errors
2264        );
2265    }
2266
2267    #[test]
2268    fn test_zero_max_agents_error() {
2269        let mut config = OxiosConfig::default();
2270        config.kernel.max_agents = 0;
2271        let (errors, _warnings) = config.validate();
2272        assert!(errors.iter().any(|e| e.contains("max_agents must be > 0")));
2273    }
2274
2275    /// Rust Default와 share/default-config.toml 간 핵심 기본값 일치 확인.
2276    /// TOML 템플릿은 "프로덕션 준비" 기본값을 가지며,
2277    /// Rust Default는 "안전한 최소" 기본값을 가질 수 있음.
2278    /// 핵심 스칼라 값(포트, 호스트, max_agents 등)은 반드시 일치해야 함.
2279    #[test]
2280    fn test_default_config_matches_toml() {
2281        let from_rust = OxiosConfig::default();
2282
2283        let toml_str = include_str!("../../../share/default-config.toml");
2284        let from_toml: OxiosConfig =
2285            toml::from_str(toml_str).expect("share/default-config.toml이 유효하지 않습니다");
2286
2287        // 핵심 스칼라 필드 — Rust와 TOML이 반드시 일치해야 함
2288        assert_eq!(
2289            from_rust.kernel.max_agents, from_toml.kernel.max_agents,
2290            "kernel.max_agents 불일치: Rust={}, TOML={}",
2291            from_rust.kernel.max_agents, from_toml.kernel.max_agents
2292        );
2293        assert_eq!(
2294            from_rust.gateway.host, from_toml.gateway.host,
2295            "gateway.host 불일치: Rust={}, TOML={}",
2296            from_rust.gateway.host, from_toml.gateway.host
2297        );
2298        assert_eq!(
2299            from_rust.gateway.port, from_toml.gateway.port,
2300            "gateway.port 불일치: Rust={}, TOML={}",
2301            from_rust.gateway.port, from_toml.gateway.port
2302        );
2303        assert_eq!(
2304            from_rust.kernel.event_bus_capacity, from_toml.kernel.event_bus_capacity,
2305            "kernel.event_bus_capacity 불일치"
2306        );
2307        assert_eq!(
2308            from_rust.scheduler.max_concurrent, from_toml.scheduler.max_concurrent,
2309            "scheduler.max_concurrent 불일치"
2310        );
2311        assert_eq!(
2312            from_rust.memory.consolidation.preset, from_toml.memory.consolidation.preset,
2313            "memory.consolidation.preset 불일치"
2314        );
2315
2316        // TOML 템플릿이 파싱 가능한지 확인
2317        let (_, warnings) = from_toml.validate();
2318        for w in &warnings {
2319            eprintln!("default-config.toml 경고: {}", w);
2320        }
2321    }
2322
2323    /// `gateway.expose_api_docs` is gated to loopback binds for safety.
2324    /// Verifies all four cases: opt-out, opt-in + public, opt-in + loopback.
2325    #[test]
2326    fn test_gateway_should_expose_api_docs() {
2327        // Default: opt-out — never expose.
2328        let cfg = GatewayConfig::default();
2329        assert!(!cfg.should_expose_api_docs());
2330
2331        // Opt-in + public bind (0.0.0.0) — still NOT exposed.
2332        let cfg = GatewayConfig {
2333            host: "0.0.0.0".into(),
2334            port: 4200,
2335            expose_api_docs: true,
2336            ..Default::default()
2337        };
2338        assert!(
2339            !cfg.should_expose_api_docs(),
2340            "public bind must not expose api docs even when opt-in is true"
2341        );
2342
2343        // Opt-in + loopback (127.0.0.1) — exposed.
2344        let cfg = GatewayConfig {
2345            host: "127.0.0.1".into(),
2346            port: 4200,
2347            expose_api_docs: true,
2348            ..Default::default()
2349        };
2350        assert!(cfg.should_expose_api_docs());
2351
2352        // Opt-in + ::1 — exposed.
2353        let cfg = GatewayConfig {
2354            host: "::1".into(),
2355            port: 4200,
2356            expose_api_docs: true,
2357            ..Default::default()
2358        };
2359        assert!(cfg.should_expose_api_docs());
2360
2361        // Opt-in + "localhost" — exposed.
2362        let cfg = GatewayConfig {
2363            host: "localhost".into(),
2364            port: 4200,
2365            expose_api_docs: true,
2366            ..Default::default()
2367        };
2368        assert!(cfg.should_expose_api_docs());
2369
2370        // Opt-out (explicit false) + loopback — NOT exposed.
2371        let cfg = GatewayConfig {
2372            host: "127.0.0.1".into(),
2373            port: 4200,
2374            expose_api_docs: false,
2375            ..Default::default()
2376        };
2377        assert!(!cfg.should_expose_api_docs());
2378    }
2379}