Skip to main content

batty_cli/team/
config_diff.rs

1//! Diff two team configurations to produce a topology change set.
2//!
3//! Used by both `batty scale` (CLI-side) and daemon hot-reload to determine
4//! which agents need to be added, removed, or left unchanged.
5
6use std::collections::HashSet;
7
8use super::config::TeamConfig;
9use super::hierarchy::{self, MemberInstance};
10
11/// A single member that changed between two configurations.
12#[derive(Debug, Clone)]
13pub struct MemberChange {
14    pub name: String,
15    pub member: MemberInstance,
16}
17
18/// The result of diffing two resolved team topologies.
19#[derive(Debug, Clone)]
20pub struct TopologyDiff {
21    /// Members present in new config but not in old.
22    pub added: Vec<MemberChange>,
23    /// Members present in old config but not in new.
24    pub removed: Vec<MemberChange>,
25    /// Members present in both configs (by name).
26    pub unchanged: Vec<String>,
27}
28
29impl TopologyDiff {
30    /// True when no members were added or removed.
31    pub fn is_empty(&self) -> bool {
32        self.added.is_empty() && self.removed.is_empty()
33    }
34
35    /// Total number of changes.
36    pub fn change_count(&self) -> usize {
37        self.added.len() + self.removed.len()
38    }
39}
40
41/// Compute the topology diff between two team configurations.
42///
43/// Resolves both configs into member instance lists and compares by name.
44pub fn diff_configs(old: &TeamConfig, new: &TeamConfig) -> anyhow::Result<TopologyDiff> {
45    let old_members = hierarchy::resolve_hierarchy(old)?;
46    let new_members = hierarchy::resolve_hierarchy(new)?;
47    Ok(diff_members(&old_members, &new_members))
48}
49
50/// Compute the topology diff between two resolved member lists.
51pub fn diff_members(old: &[MemberInstance], new: &[MemberInstance]) -> TopologyDiff {
52    let old_names: HashSet<&str> = old.iter().map(|m| m.name.as_str()).collect();
53    let new_names: HashSet<&str> = new.iter().map(|m| m.name.as_str()).collect();
54
55    let added: Vec<MemberChange> = new
56        .iter()
57        .filter(|m| !old_names.contains(m.name.as_str()))
58        .map(|m| MemberChange {
59            name: m.name.clone(),
60            member: m.clone(),
61        })
62        .collect();
63
64    let removed: Vec<MemberChange> = old
65        .iter()
66        .filter(|m| !new_names.contains(m.name.as_str()))
67        .map(|m| MemberChange {
68            name: m.name.clone(),
69            member: m.clone(),
70        })
71        .collect();
72
73    let unchanged: Vec<String> = old
74        .iter()
75        .filter(|m| new_names.contains(m.name.as_str()))
76        .map(|m| m.name.clone())
77        .collect();
78
79    TopologyDiff {
80        added,
81        removed,
82        unchanged,
83    }
84}
85
86// ---------------------------------------------------------------------------
87// Tests
88// ---------------------------------------------------------------------------
89
90#[cfg(test)]
91mod tests {
92    use super::*;
93    use crate::team::config::{RoleDef, RoleType};
94
95    fn minimal_config(engineer_instances: u32, manager_instances: u32) -> TeamConfig {
96        TeamConfig {
97            name: "test".into(),
98            agent: None,
99            workflow_mode: Default::default(),
100            board: Default::default(),
101            standup: Default::default(),
102            automation: Default::default(),
103            automation_sender: None,
104            external_senders: vec![],
105            orchestrator_pane: false,
106            orchestrator_position: Default::default(),
107            layout: None,
108            workflow_policy: Default::default(),
109            cost: Default::default(),
110            grafana: Default::default(),
111            use_shim: true,
112            use_sdk_mode: false,
113            auto_respawn_on_crash: false,
114            shim_health_check_interval_secs: 60,
115            shim_health_timeout_secs: 120,
116            shim_shutdown_timeout_secs: 30,
117            shim_working_state_timeout_secs: 1800,
118            pending_queue_max_age_secs: 600,
119            event_log_max_bytes: 10 * 1024 * 1024,
120            retro_min_duration_secs: 60,
121            roles: vec![
122                RoleDef {
123                    name: "architect".into(),
124                    role_type: RoleType::Architect,
125                    agent: None,
126                    instances: 1,
127                    prompt: None,
128                    talks_to: vec![],
129                    channel: None,
130                    channel_config: None,
131                    nudge_interval_secs: None,
132                    receives_standup: None,
133                    standup_interval_secs: None,
134                    owns: vec![],
135                    use_worktrees: false,
136                },
137                RoleDef {
138                    name: "manager".into(),
139                    role_type: RoleType::Manager,
140                    agent: None,
141                    instances: manager_instances,
142                    prompt: None,
143                    talks_to: vec![],
144                    channel: None,
145                    channel_config: None,
146                    nudge_interval_secs: None,
147                    receives_standup: None,
148                    standup_interval_secs: None,
149                    owns: vec![],
150                    use_worktrees: false,
151                },
152                RoleDef {
153                    name: "engineer".into(),
154                    role_type: RoleType::Engineer,
155                    agent: None,
156                    instances: engineer_instances,
157                    prompt: None,
158                    talks_to: vec![],
159                    channel: None,
160                    channel_config: None,
161                    nudge_interval_secs: None,
162                    receives_standup: None,
163                    standup_interval_secs: None,
164                    owns: vec![],
165                    use_worktrees: true,
166                },
167            ],
168        }
169    }
170
171    #[test]
172    fn no_change_produces_empty_diff() {
173        let config = minimal_config(3, 1);
174        let diff = diff_configs(&config, &config).unwrap();
175        assert!(diff.is_empty());
176        assert_eq!(diff.change_count(), 0);
177        // architect + manager + 3 engineers = 5 unchanged
178        assert_eq!(diff.unchanged.len(), 5);
179    }
180
181    #[test]
182    fn scale_up_engineers_shows_added() {
183        let old = minimal_config(2, 1);
184        let new = minimal_config(4, 1);
185        let diff = diff_configs(&old, &new).unwrap();
186        assert_eq!(diff.added.len(), 2);
187        assert!(diff.removed.is_empty());
188        // Check the added names
189        let added_names: HashSet<&str> = diff.added.iter().map(|c| c.name.as_str()).collect();
190        assert!(added_names.contains("eng-1-3"));
191        assert!(added_names.contains("eng-1-4"));
192    }
193
194    #[test]
195    fn scale_down_engineers_shows_removed() {
196        let old = minimal_config(4, 1);
197        let new = minimal_config(2, 1);
198        let diff = diff_configs(&old, &new).unwrap();
199        assert!(diff.added.is_empty());
200        assert_eq!(diff.removed.len(), 2);
201        let removed_names: HashSet<&str> = diff.removed.iter().map(|c| c.name.as_str()).collect();
202        assert!(removed_names.contains("eng-1-3"));
203        assert!(removed_names.contains("eng-1-4"));
204    }
205
206    #[test]
207    fn add_manager_shows_added_manager_and_engineers() {
208        let old = minimal_config(2, 1);
209        let new = minimal_config(2, 2);
210        let diff = diff_configs(&old, &new).unwrap();
211        // Adding a second manager creates manager-2 + eng-2-1, eng-2-2
212        // And renames architect→architect, manager→manager-1, manager-2 new
213        // Actually hierarchy naming: with instances=2, managers become manager-1, manager-2
214        // With instances=1, manager stays as "manager"
215        // So old has: architect, manager, eng-1-1, eng-1-2
216        // New has: architect, manager-1, manager-2, eng-1-1, eng-1-2, eng-2-1, eng-2-2
217        // Diff: removed manager, added manager-1, manager-2, eng-2-1, eng-2-2
218        assert!(!diff.added.is_empty());
219        let added_names: HashSet<&str> = diff.added.iter().map(|c| c.name.as_str()).collect();
220        assert!(added_names.contains("manager-2"));
221    }
222
223    #[test]
224    fn diff_members_direct() {
225        let old = vec![
226            MemberInstance {
227                name: "architect".into(),
228                role_name: "architect".into(),
229                role_type: RoleType::Architect,
230                agent: Some("claude".into()),
231                prompt: None,
232                reports_to: None,
233                use_worktrees: false,
234            },
235            MemberInstance {
236                name: "eng-1-1".into(),
237                role_name: "engineer".into(),
238                role_type: RoleType::Engineer,
239                agent: Some("claude".into()),
240                prompt: None,
241                reports_to: Some("manager".into()),
242                use_worktrees: true,
243            },
244        ];
245        let new = vec![
246            MemberInstance {
247                name: "architect".into(),
248                role_name: "architect".into(),
249                role_type: RoleType::Architect,
250                agent: Some("claude".into()),
251                prompt: None,
252                reports_to: None,
253                use_worktrees: false,
254            },
255            MemberInstance {
256                name: "eng-1-1".into(),
257                role_name: "engineer".into(),
258                role_type: RoleType::Engineer,
259                agent: Some("claude".into()),
260                prompt: None,
261                reports_to: Some("manager".into()),
262                use_worktrees: true,
263            },
264            MemberInstance {
265                name: "eng-1-2".into(),
266                role_name: "engineer".into(),
267                role_type: RoleType::Engineer,
268                agent: Some("claude".into()),
269                prompt: None,
270                reports_to: Some("manager".into()),
271                use_worktrees: true,
272            },
273        ];
274        let diff = diff_members(&old, &new);
275        assert_eq!(diff.added.len(), 1);
276        assert_eq!(diff.added[0].name, "eng-1-2");
277        assert!(diff.removed.is_empty());
278        assert_eq!(diff.unchanged.len(), 2);
279    }
280}