use arc_swap::ArcSwap;
use notify::{Event, RecommendedWatcher, RecursiveMode, Watcher};
use serde::{Deserialize, Serialize};
use std::collections::HashMap;
use std::fs;
use std::path::{Path, PathBuf};
use std::sync::Arc;
use toon_format::{decode_default, encode_default, ToonError};
use tracing::{debug, error, info, warn};
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
pub struct ToonAgentConfig {
pub name: String,
#[serde(default = "default_version")]
pub version: String,
pub model: String,
#[serde(default)]
pub system_prompt: Option<String>,
#[serde(default)]
pub tools: Vec<String>,
#[serde(default = "default_max_tool_iterations")]
pub max_tool_iterations: usize,
#[serde(default)]
pub parallel_tools: bool,
#[serde(flatten)]
pub extra: HashMap<String, serde_json::Value>,
}
fn default_version() -> String {
"0.1.0".to_string()
}
fn default_max_tool_iterations() -> usize {
10
}
impl ToonAgentConfig {
pub fn new(name: impl Into<String>, model: impl Into<String>) -> Self {
Self {
name: name.into(),
version: default_version(),
model: model.into(),
system_prompt: None,
tools: Vec::new(),
max_tool_iterations: default_max_tool_iterations(),
parallel_tools: false,
extra: HashMap::new(),
}
}
pub fn with_system_prompt(mut self, prompt: impl Into<String>) -> Self {
self.system_prompt = Some(prompt.into());
self
}
pub fn with_tools(mut self, tools: Vec<String>) -> Self {
self.tools = tools;
self
}
pub fn to_toon(&self) -> Result<String, ToonConfigError> {
encode_default(self).map_err(ToonConfigError::from)
}
pub fn from_toon(toon: &str) -> Result<Self, ToonConfigError> {
decode_default(toon).map_err(ToonConfigError::from)
}
}
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
pub struct ToonModelConfig {
pub name: String,
pub provider: String,
pub model: String,
#[serde(default = "default_temperature")]
pub temperature: f32,
#[serde(default = "default_max_tokens")]
pub max_tokens: u32,
#[serde(default)]
pub top_p: Option<f32>,
#[serde(default)]
pub frequency_penalty: Option<f32>,
#[serde(default)]
pub presence_penalty: Option<f32>,
}
fn default_temperature() -> f32 {
0.7
}
fn default_max_tokens() -> u32 {
512
}
impl ToonModelConfig {
pub fn new(
name: impl Into<String>,
provider: impl Into<String>,
model: impl Into<String>,
) -> Self {
Self {
name: name.into(),
provider: provider.into(),
model: model.into(),
temperature: default_temperature(),
max_tokens: default_max_tokens(),
top_p: None,
frequency_penalty: None,
presence_penalty: None,
}
}
pub fn to_toon(&self) -> Result<String, ToonConfigError> {
encode_default(self).map_err(ToonConfigError::from)
}
pub fn from_toon(toon: &str) -> Result<Self, ToonConfigError> {
decode_default(toon).map_err(ToonConfigError::from)
}
}
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
pub struct ToonToolConfig {
pub name: String,
#[serde(default = "default_true")]
pub enabled: bool,
#[serde(default)]
pub description: Option<String>,
#[serde(default = "default_timeout")]
pub timeout_secs: u64,
#[serde(flatten)]
pub extra: HashMap<String, serde_json::Value>,
}
fn default_true() -> bool {
true
}
fn default_timeout() -> u64 {
30
}
impl ToonToolConfig {
pub fn new(name: impl Into<String>) -> Self {
Self {
name: name.into(),
enabled: default_true(),
description: None,
timeout_secs: default_timeout(),
extra: HashMap::new(),
}
}
pub fn to_toon(&self) -> Result<String, ToonConfigError> {
encode_default(self).map_err(ToonConfigError::from)
}
pub fn from_toon(toon: &str) -> Result<Self, ToonConfigError> {
decode_default(toon).map_err(ToonConfigError::from)
}
}
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
pub struct ToonWorkflowConfig {
pub name: String,
pub entry_agent: String,
#[serde(default)]
pub fallback_agent: Option<String>,
#[serde(default = "default_max_depth")]
pub max_depth: u8,
#[serde(default = "default_max_iterations")]
pub max_iterations: u8,
#[serde(default)]
pub parallel_subagents: bool,
}
fn default_max_depth() -> u8 {
3
}
fn default_max_iterations() -> u8 {
5
}
impl ToonWorkflowConfig {
pub fn new(name: impl Into<String>, entry_agent: impl Into<String>) -> Self {
Self {
name: name.into(),
entry_agent: entry_agent.into(),
fallback_agent: None,
max_depth: default_max_depth(),
max_iterations: default_max_iterations(),
parallel_subagents: false,
}
}
pub fn to_toon(&self) -> Result<String, ToonConfigError> {
encode_default(self).map_err(ToonConfigError::from)
}
pub fn from_toon(toon: &str) -> Result<Self, ToonConfigError> {
decode_default(toon).map_err(ToonConfigError::from)
}
}
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
pub struct ToonMcpConfig {
pub name: String,
#[serde(default = "default_true")]
pub enabled: bool,
#[serde(default)]
pub command: Option<String>,
#[serde(default)]
pub args: Vec<String>,
#[serde(default)]
pub env: HashMap<String, String>,
#[serde(default = "default_timeout")]
pub timeout_secs: u64,
}
impl ToonMcpConfig {
pub fn new(name: impl Into<String>, command: impl Into<String>) -> Self {
Self {
name: name.into(),
enabled: default_true(),
command: Some(command.into()),
args: Vec::new(),
env: HashMap::new(),
timeout_secs: default_timeout(),
}
}
pub fn to_toon(&self) -> Result<String, ToonConfigError> {
encode_default(self).map_err(ToonConfigError::from)
}
pub fn from_toon(toon: &str) -> Result<Self, ToonConfigError> {
decode_default(toon).map_err(ToonConfigError::from)
}
}
#[derive(Debug, Clone, Default)]
pub struct DynamicConfig {
pub agents: HashMap<String, ToonAgentConfig>,
pub models: HashMap<String, ToonModelConfig>,
pub tools: HashMap<String, ToonToolConfig>,
pub workflows: HashMap<String, ToonWorkflowConfig>,
pub mcps: HashMap<String, ToonMcpConfig>,
}
impl DynamicConfig {
pub fn load(
agents_dir: &Path,
models_dir: &Path,
tools_dir: &Path,
workflows_dir: &Path,
mcps_dir: &Path,
) -> Result<Self, ToonConfigError> {
let agents = load_configs_from_dir::<ToonAgentConfig>(agents_dir, "agents")?;
let models = load_configs_from_dir::<ToonModelConfig>(models_dir, "models")?;
let tools = load_configs_from_dir::<ToonToolConfig>(tools_dir, "tools")?;
let workflows = load_configs_from_dir::<ToonWorkflowConfig>(workflows_dir, "workflows")?;
let mcps = load_configs_from_dir::<ToonMcpConfig>(mcps_dir, "mcps")?;
info!(
"Loaded dynamic config: {} agents, {} models, {} tools, {} workflows, {} mcps",
agents.len(),
models.len(),
tools.len(),
workflows.len(),
mcps.len()
);
Ok(Self {
agents,
models,
tools,
workflows,
mcps,
})
}
pub fn get_agent(&self, name: &str) -> Option<&ToonAgentConfig> {
self.agents.get(name)
}
pub fn get_model(&self, name: &str) -> Option<&ToonModelConfig> {
self.models.get(name)
}
pub fn get_tool(&self, name: &str) -> Option<&ToonToolConfig> {
self.tools.get(name)
}
pub fn get_workflow(&self, name: &str) -> Option<&ToonWorkflowConfig> {
self.workflows.get(name)
}
pub fn get_mcp(&self, name: &str) -> Option<&ToonMcpConfig> {
self.mcps.get(name)
}
pub fn agent_names(&self) -> Vec<&str> {
self.agents.keys().map(|s| s.as_str()).collect()
}
pub fn model_names(&self) -> Vec<&str> {
self.models.keys().map(|s| s.as_str()).collect()
}
pub fn tool_names(&self) -> Vec<&str> {
self.tools.keys().map(|s| s.as_str()).collect()
}
pub fn workflow_names(&self) -> Vec<&str> {
self.workflows.keys().map(|s| s.as_str()).collect()
}
pub fn mcp_names(&self) -> Vec<&str> {
self.mcps.keys().map(|s| s.as_str()).collect()
}
pub fn validate(&self) -> Result<Vec<ConfigWarning>, ToonConfigError> {
let mut warnings = Vec::new();
for (agent_name, agent) in &self.agents {
if !self.models.contains_key(&agent.model) {
return Err(ToonConfigError::Validation(format!(
"Agent '{}' references unknown model '{}'",
agent_name, agent.model
)));
}
for tool_name in &agent.tools {
if !self.tools.contains_key(tool_name) {
return Err(ToonConfigError::Validation(format!(
"Agent '{}' references unknown tool '{}'",
agent_name, tool_name
)));
}
}
}
for (workflow_name, workflow) in &self.workflows {
if !self.agents.contains_key(&workflow.entry_agent) {
return Err(ToonConfigError::Validation(format!(
"Workflow '{}' references unknown entry agent '{}'",
workflow_name, workflow.entry_agent
)));
}
if let Some(ref fallback) = workflow.fallback_agent {
if !self.agents.contains_key(fallback) {
return Err(ToonConfigError::Validation(format!(
"Workflow '{}' references unknown fallback agent '{}'",
workflow_name, fallback
)));
}
}
}
let used_models: std::collections::HashSet<_> =
self.agents.values().map(|a| &a.model).collect();
for model_name in self.models.keys() {
if !used_models.contains(model_name) {
warnings.push(ConfigWarning {
kind: WarningKind::UnusedModel,
message: format!("Model '{}' is not used by any agent", model_name),
});
}
}
let used_tools: std::collections::HashSet<_> =
self.agents.values().flat_map(|a| a.tools.iter()).collect();
for tool_name in self.tools.keys() {
if !used_tools.contains(tool_name) {
warnings.push(ConfigWarning {
kind: WarningKind::UnusedTool,
message: format!("Tool '{}' is not used by any agent", tool_name),
});
}
}
Ok(warnings)
}
}
pub trait HasName {
fn name(&self) -> &str;
}
impl HasName for ToonAgentConfig {
fn name(&self) -> &str {
&self.name
}
}
impl HasName for ToonModelConfig {
fn name(&self) -> &str {
&self.name
}
}
impl HasName for ToonToolConfig {
fn name(&self) -> &str {
&self.name
}
}
impl HasName for ToonWorkflowConfig {
fn name(&self) -> &str {
&self.name
}
}
impl HasName for ToonMcpConfig {
fn name(&self) -> &str {
&self.name
}
}
fn load_configs_from_dir<T>(
dir: &Path,
config_type: &str,
) -> Result<HashMap<String, T>, ToonConfigError>
where
T: for<'de> Deserialize<'de> + HasName,
{
let mut configs = HashMap::new();
if !dir.exists() {
debug!("Config directory does not exist: {:?}", dir);
return Ok(configs);
}
let entries = fs::read_dir(dir).map_err(|e| {
ToonConfigError::Io(std::io::Error::new(
e.kind(),
format!("Failed to read {} directory {:?}: {}", config_type, dir, e),
))
})?;
for entry in entries {
let entry = entry.map_err(ToonConfigError::Io)?;
let path = entry.path();
if path.extension().and_then(|e| e.to_str()) != Some("toon") {
continue;
}
match load_toon_file::<T>(&path) {
Ok(config) => {
let name = config.name().to_string();
debug!("Loaded {} config: {}", config_type, name);
configs.insert(name, config);
}
Err(e) => {
warn!("Failed to load {} from {:?}: {}", config_type, path, e);
}
}
}
Ok(configs)
}
fn load_toon_file<T>(path: &Path) -> Result<T, ToonConfigError>
where
T: for<'de> Deserialize<'de>,
{
let content = fs::read_to_string(path).map_err(|e| {
ToonConfigError::Io(std::io::Error::new(
e.kind(),
format!("Failed to read {:?}: {}", path, e),
))
})?;
decode_default(&content)
.map_err(|e| ToonConfigError::Parse(format!("Failed to parse {:?}: {}", path, e)))
}
#[derive(Debug, thiserror::Error)]
pub enum ToonConfigError {
#[error("IO error: {0}")]
Io(#[from] std::io::Error),
#[error("TOON parse error: {0}")]
Parse(String),
#[error("Validation error: {0}")]
Validation(String),
#[error("Watch error: {0}")]
Watch(#[from] notify::Error),
}
impl From<ToonError> for ToonConfigError {
fn from(e: ToonError) -> Self {
ToonConfigError::Parse(e.to_string())
}
}
#[derive(Debug, Clone)]
pub struct ConfigWarning {
pub kind: WarningKind,
pub message: String,
}
#[derive(Debug, Clone, PartialEq)]
pub enum WarningKind {
UnusedModel,
UnusedTool,
UnusedAgent,
UnusedWorkflow,
UnusedMcp,
}
impl std::fmt::Display for ConfigWarning {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.message)
}
}
pub struct DynamicConfigManager {
config: Arc<ArcSwap<DynamicConfig>>,
agents_dir: PathBuf,
models_dir: PathBuf,
tools_dir: PathBuf,
workflows_dir: PathBuf,
mcps_dir: PathBuf,
_watcher: Option<RecommendedWatcher>,
version_tx: Arc<std::sync::Mutex<Option<tokio::sync::mpsc::UnboundedSender<Vec<ToonAgentConfig>>>>>,
}
impl DynamicConfigManager {
pub fn from_config(
config: &crate::utils::toml_config::AresConfig,
) -> Result<Self, ToonConfigError> {
let agents_dir = PathBuf::from(&config.config.agents_dir);
let models_dir = PathBuf::from(&config.config.models_dir);
let tools_dir = PathBuf::from(&config.config.tools_dir);
let workflows_dir = PathBuf::from(&config.config.workflows_dir);
let mcps_dir = PathBuf::from(&config.config.mcps_dir);
Self::new(
agents_dir,
models_dir,
tools_dir,
workflows_dir,
mcps_dir,
true, )
}
pub fn new(
agents_dir: PathBuf,
models_dir: PathBuf,
tools_dir: PathBuf,
workflows_dir: PathBuf,
mcps_dir: PathBuf,
hot_reload: bool,
) -> Result<Self, ToonConfigError> {
let initial_config = DynamicConfig::load(
&agents_dir,
&models_dir,
&tools_dir,
&workflows_dir,
&mcps_dir,
)?;
let config = Arc::new(ArcSwap::from_pointee(initial_config));
let version_tx: Arc<std::sync::Mutex<Option<tokio::sync::mpsc::UnboundedSender<Vec<ToonAgentConfig>>>>> =
Arc::new(std::sync::Mutex::new(None));
let watcher = if hot_reload {
Some(Self::setup_watcher(
config.clone(),
agents_dir.clone(),
models_dir.clone(),
tools_dir.clone(),
workflows_dir.clone(),
mcps_dir.clone(),
version_tx.clone(),
)?)
} else {
None
};
Ok(Self {
config,
agents_dir,
models_dir,
tools_dir,
workflows_dir,
mcps_dir,
_watcher: watcher,
version_tx,
})
}
pub fn set_version_tx(&self, tx: tokio::sync::mpsc::UnboundedSender<Vec<ToonAgentConfig>>) {
if let Ok(mut guard) = self.version_tx.lock() {
*guard = Some(tx);
}
}
fn setup_watcher(
config: Arc<ArcSwap<DynamicConfig>>,
agents_dir: PathBuf,
models_dir: PathBuf,
tools_dir: PathBuf,
workflows_dir: PathBuf,
mcps_dir: PathBuf,
version_tx: Arc<std::sync::Mutex<Option<tokio::sync::mpsc::UnboundedSender<Vec<ToonAgentConfig>>>>>,
) -> Result<RecommendedWatcher, ToonConfigError> {
let agents_dir_clone = agents_dir.clone();
let models_dir_clone = models_dir.clone();
let tools_dir_clone = tools_dir.clone();
let workflows_dir_clone = workflows_dir.clone();
let mcps_dir_clone = mcps_dir.clone();
let mut watcher = notify::recommended_watcher(move |res: Result<Event, notify::Error>| {
match res {
Ok(event) => {
if matches!(
event.kind,
notify::EventKind::Create(_)
| notify::EventKind::Modify(_)
| notify::EventKind::Remove(_)
) {
info!("Config change detected, reloading...");
match DynamicConfig::load(
&agents_dir_clone,
&models_dir_clone,
&tools_dir_clone,
&workflows_dir_clone,
&mcps_dir_clone,
) {
Ok(new_config) => {
match new_config.validate() {
Ok(warnings) => {
for warning in warnings {
warn!("Config warning: {}", warning);
}
let agents: Vec<ToonAgentConfig> =
new_config.agents.values().cloned().collect();
if let Ok(guard) = version_tx.lock() {
if let Some(tx) = guard.as_ref() {
let _ = tx.send(agents);
}
}
config.store(Arc::new(new_config));
info!("Config reloaded successfully");
}
Err(e) => {
error!(
"Config validation failed, keeping old config: {}",
e
);
}
}
}
Err(e) => {
error!("Failed to reload config: {}", e);
}
}
}
}
Err(e) => {
error!("Watch error: {:?}", e);
}
}
})?;
for dir in [
&agents_dir,
&models_dir,
&tools_dir,
&workflows_dir,
&mcps_dir,
] {
if dir.exists() {
watcher.watch(dir, RecursiveMode::Recursive)?;
debug!("Watching directory: {:?}", dir);
}
}
Ok(watcher)
}
pub fn config(&self) -> arc_swap::Guard<Arc<DynamicConfig>> {
self.config.load()
}
pub fn agent(&self, name: &str) -> Option<ToonAgentConfig> {
self.config.load().get_agent(name).cloned()
}
pub fn model(&self, name: &str) -> Option<ToonModelConfig> {
self.config.load().get_model(name).cloned()
}
pub fn tool(&self, name: &str) -> Option<ToonToolConfig> {
self.config.load().get_tool(name).cloned()
}
pub fn workflow(&self, name: &str) -> Option<ToonWorkflowConfig> {
self.config.load().get_workflow(name).cloned()
}
pub fn mcp(&self, name: &str) -> Option<ToonMcpConfig> {
self.config.load().get_mcp(name).cloned()
}
pub fn agents(&self) -> Vec<ToonAgentConfig> {
self.config.load().agents.values().cloned().collect()
}
pub fn models(&self) -> Vec<ToonModelConfig> {
self.config.load().models.values().cloned().collect()
}
pub fn tools(&self) -> Vec<ToonToolConfig> {
self.config.load().tools.values().cloned().collect()
}
pub fn workflows(&self) -> Vec<ToonWorkflowConfig> {
self.config.load().workflows.values().cloned().collect()
}
pub fn mcps(&self) -> Vec<ToonMcpConfig> {
self.config.load().mcps.values().cloned().collect()
}
pub fn agent_names(&self) -> Vec<String> {
self.config
.load()
.agent_names()
.into_iter()
.map(String::from)
.collect()
}
pub fn model_names(&self) -> Vec<String> {
self.config
.load()
.model_names()
.into_iter()
.map(String::from)
.collect()
}
pub fn tool_names(&self) -> Vec<String> {
self.config
.load()
.tool_names()
.into_iter()
.map(String::from)
.collect()
}
pub fn workflow_names(&self) -> Vec<String> {
self.config
.load()
.workflow_names()
.into_iter()
.map(String::from)
.collect()
}
pub fn mcp_names(&self) -> Vec<String> {
self.config
.load()
.mcp_names()
.into_iter()
.map(String::from)
.collect()
}
pub fn upsert_agent(&self, agent: ToonAgentConfig) {
let current = self.config.load();
let mut new_agents = current.agents.clone();
new_agents.insert(agent.name.clone(), agent);
let new_config = DynamicConfig {
agents: new_agents,
models: current.models.clone(),
tools: current.tools.clone(),
workflows: current.workflows.clone(),
mcps: current.mcps.clone(),
};
self.config.store(Arc::new(new_config));
}
pub fn reload(&self) -> Result<Vec<ConfigWarning>, ToonConfigError> {
let new_config = DynamicConfig::load(
&self.agents_dir,
&self.models_dir,
&self.tools_dir,
&self.workflows_dir,
&self.mcps_dir,
)?;
let warnings = new_config.validate()?;
self.config.store(Arc::new(new_config));
Ok(warnings)
}
}
#[cfg(test)]
mod tests {
use super::*;
use tempfile::TempDir;
#[test]
fn test_agent_config_roundtrip() {
let agent = ToonAgentConfig::new("test-agent", "fast")
.with_system_prompt("You are a test agent.")
.with_tools(vec!["calculator".to_string(), "web_search".to_string()]);
let toon = agent.to_toon().expect("Failed to encode");
let decoded = ToonAgentConfig::from_toon(&toon).expect("Failed to decode");
assert_eq!(agent.name, decoded.name);
assert_eq!(agent.model, decoded.model);
assert_eq!(agent.system_prompt, decoded.system_prompt);
assert_eq!(agent.tools, decoded.tools);
}
#[test]
fn test_model_config_roundtrip() {
let model = ToonModelConfig::new("fast", "ollama-local", "ministral-3:3b");
let toon = model.to_toon().expect("Failed to encode");
let decoded = ToonModelConfig::from_toon(&toon).expect("Failed to decode");
assert_eq!(model.name, decoded.name);
assert_eq!(model.provider, decoded.provider);
assert_eq!(model.model, decoded.model);
assert_eq!(model.temperature, decoded.temperature);
assert_eq!(model.max_tokens, decoded.max_tokens);
}
#[test]
fn test_tool_config_roundtrip() {
let mut tool = ToonToolConfig::new("calculator");
tool.description = Some("Performs arithmetic operations".to_string());
tool.timeout_secs = 10;
let toon = tool.to_toon().expect("Failed to encode");
let decoded = ToonToolConfig::from_toon(&toon).expect("Failed to decode");
assert_eq!(tool.name, decoded.name);
assert_eq!(tool.enabled, decoded.enabled);
assert_eq!(tool.description, decoded.description);
assert_eq!(tool.timeout_secs, decoded.timeout_secs);
}
#[test]
fn test_workflow_config_roundtrip() {
let mut workflow = ToonWorkflowConfig::new("default", "router");
workflow.fallback_agent = Some("orchestrator".to_string());
workflow.max_depth = 3;
workflow.max_iterations = 5;
let toon = workflow.to_toon().expect("Failed to encode");
let decoded = ToonWorkflowConfig::from_toon(&toon).expect("Failed to decode");
assert_eq!(workflow.name, decoded.name);
assert_eq!(workflow.entry_agent, decoded.entry_agent);
assert_eq!(workflow.fallback_agent, decoded.fallback_agent);
assert_eq!(workflow.max_depth, decoded.max_depth);
assert_eq!(workflow.max_iterations, decoded.max_iterations);
}
#[test]
fn test_mcp_config_roundtrip() {
let mut mcp = ToonMcpConfig::new("filesystem", "npx");
mcp.args = vec![
"-y".to_string(),
"@modelcontextprotocol/server-filesystem".to_string(),
"/home".to_string(),
"/tmp".to_string(),
];
mcp.env
.insert("NODE_ENV".to_string(), "production".to_string());
mcp.timeout_secs = 30;
let toon = mcp.to_toon().expect("Failed to encode");
let decoded = ToonMcpConfig::from_toon(&toon).expect("Failed to decode");
assert_eq!(mcp.name, decoded.name);
assert_eq!(mcp.command, decoded.command);
assert_eq!(mcp.args, decoded.args);
assert_eq!(mcp.env, decoded.env);
assert_eq!(mcp.timeout_secs, decoded.timeout_secs);
}
#[test]
fn test_load_configs_from_dir() {
let temp_dir = TempDir::new().expect("Failed to create temp dir");
let agents_dir = temp_dir.path().join("agents");
fs::create_dir_all(&agents_dir).expect("Failed to create agents dir");
let agent_content = r#"name: test-agent
model: fast
max_tool_iterations: 5
parallel_tools: false
tools[0]:
system_prompt: Test agent prompt"#;
fs::write(agents_dir.join("test-agent.toon"), agent_content)
.expect("Failed to write agent file");
let agents = load_configs_from_dir::<ToonAgentConfig>(&agents_dir, "agents")
.expect("Failed to load agents");
assert_eq!(agents.len(), 1);
let agent = agents.get("test-agent").expect("Agent not found");
assert_eq!(agent.name, "test-agent");
assert_eq!(agent.model, "fast");
assert_eq!(agent.max_tool_iterations, 5);
}
#[test]
fn test_dynamic_config_validation() {
let mut config = DynamicConfig::default();
config.models.insert(
"fast".to_string(),
ToonModelConfig::new("fast", "ollama-local", "ministral-3:3b"),
);
config
.tools
.insert("calculator".to_string(), ToonToolConfig::new("calculator"));
let mut agent = ToonAgentConfig::new("router", "fast");
agent.tools = vec!["calculator".to_string()];
config.agents.insert("router".to_string(), agent);
config.workflows.insert(
"default".to_string(),
ToonWorkflowConfig::new("default", "router"),
);
let warnings = config.validate().expect("Validation failed");
assert!(warnings.is_empty());
}
#[test]
fn test_dynamic_config_validation_missing_model() {
let mut config = DynamicConfig::default();
let agent = ToonAgentConfig::new("router", "non-existent-model");
config.agents.insert("router".to_string(), agent);
let result = config.validate();
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains("unknown model"));
}
#[test]
fn test_dynamic_config_validation_missing_tool() {
let mut config = DynamicConfig::default();
config.models.insert(
"fast".to_string(),
ToonModelConfig::new("fast", "ollama-local", "ministral-3:3b"),
);
let mut agent = ToonAgentConfig::new("router", "fast");
agent.tools = vec!["non-existent-tool".to_string()];
config.agents.insert("router".to_string(), agent);
let result = config.validate();
assert!(result.is_err());
assert!(result.unwrap_err().to_string().contains("unknown tool"));
}
#[test]
fn test_parse_agent_from_toon_string() {
let toon = r#"name: router
model: fast
max_tool_iterations: 1
parallel_tools: false
tools[0]:
system_prompt: You are a routing agent."#;
let agent = ToonAgentConfig::from_toon(toon).expect("Failed to parse");
assert_eq!(agent.name, "router");
assert_eq!(agent.model, "fast");
assert_eq!(agent.max_tool_iterations, 1);
assert!(!agent.parallel_tools);
assert!(agent.tools.is_empty());
}
#[test]
fn test_parse_model_from_toon_string() {
let toon = r#"name: fast
provider: ollama-local
model: ministral-3:3b
temperature: 0.7
max_tokens: 256"#;
let model = ToonModelConfig::from_toon(toon).expect("Failed to parse");
assert_eq!(model.name, "fast");
assert_eq!(model.provider, "ollama-local");
assert_eq!(model.model, "ministral-3:3b");
assert!((model.temperature - 0.7).abs() < 0.01);
assert_eq!(model.max_tokens, 256);
}
}