1use std::collections::HashSet;
7
8use super::config::TeamConfig;
9use super::hierarchy::{self, MemberInstance};
10
11#[derive(Debug, Clone)]
13pub struct MemberChange {
14 pub name: String,
15 pub member: MemberInstance,
16}
17
18#[derive(Debug, Clone)]
20pub struct TopologyDiff {
21 pub added: Vec<MemberChange>,
23 pub removed: Vec<MemberChange>,
25 pub unchanged: Vec<String>,
27}
28
29impl TopologyDiff {
30 pub fn is_empty(&self) -> bool {
32 self.added.is_empty() && self.removed.is_empty()
33 }
34
35 pub fn change_count(&self) -> usize {
37 self.added.len() + self.removed.len()
38 }
39}
40
41pub 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
50pub 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#[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 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 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 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}