1use std::io::Write;
7
8use serde::Serialize;
9use termcolor::{Color, ColorChoice, ColorSpec, StandardStream, WriteColor};
10
11use crate::diagnostic::Diagnostic;
12use crate::sync::SyncReport;
13use crate::sync::apply::{ActionOutcome, ActionTaken};
14use crate::types::managed_cmd;
15
16pub fn use_color() -> bool {
20 std::env::var_os("NO_COLOR").is_none()
21}
22
23fn color_choice() -> ColorChoice {
24 if use_color() {
25 ColorChoice::Auto
26 } else {
27 ColorChoice::Never
28 }
29}
30
31#[derive(Debug, Serialize)]
33pub struct ListEntry {
34 pub source: String,
35 pub item: String,
36 pub kind: String,
37 pub version: String,
38 pub status: String,
39}
40
41#[derive(Debug, Serialize)]
43pub struct CatalogEntry {
44 pub name: String,
45 pub description: String,
46 pub kind: String,
47 #[serde(skip_serializing_if = "Option::is_none")]
48 pub variants: Option<String>,
49}
50
51pub fn print_catalog(
53 agents: &[CatalogEntry],
54 skills: &[CatalogEntry],
55 bootstrap: &[CatalogEntry],
56 kind_filter: Option<&str>,
57) {
58 let show_agents =
59 kind_filter.is_none() || kind_filter == Some("agents") || kind_filter == Some("agent");
60 let show_skills =
61 kind_filter.is_none() || kind_filter == Some("skills") || kind_filter == Some("skill");
62 let show_bootstrap = kind_filter.is_none()
63 || kind_filter == Some("bootstrap")
64 || kind_filter == Some("bootstrap-doc");
65
66 if show_agents && !agents.is_empty() {
67 println!("AGENTS");
68 for entry in agents {
69 let variant_suffix = entry
70 .variants
71 .as_ref()
72 .map(|variants| format!(" [variants: {variants}]"))
73 .unwrap_or_default();
74 if entry.description.is_empty() {
75 println!("- {}{}", entry.name, variant_suffix);
76 } else {
77 println!("- {}{}: {}", entry.name, variant_suffix, entry.description);
78 }
79 }
80 }
81
82 if show_agents && !agents.is_empty() && show_skills && !skills.is_empty() {
83 println!();
84 }
85
86 if show_skills && !skills.is_empty() {
87 println!("SKILLS");
88 for entry in skills {
89 let variant_suffix = entry
90 .variants
91 .as_ref()
92 .map(|variants| format!(" [variants: {variants}]"))
93 .unwrap_or_default();
94 if entry.description.is_empty() {
95 println!("- {}{}", entry.name, variant_suffix);
96 } else {
97 println!("- {}{}: {}", entry.name, variant_suffix, entry.description);
98 }
99 }
100 }
101
102 if ((show_agents && !agents.is_empty()) || (show_skills && !skills.is_empty()))
103 && show_bootstrap
104 && !bootstrap.is_empty()
105 {
106 println!();
107 }
108
109 if show_bootstrap && !bootstrap.is_empty() {
110 println!("BOOTSTRAP");
111 for entry in bootstrap {
112 if entry.description.is_empty() {
113 println!("- {}", entry.name);
114 } else {
115 println!("- {}: {}", entry.name, entry.description);
116 }
117 }
118 }
119
120 if (show_agents
121 && agents.is_empty()
122 && show_skills
123 && skills.is_empty()
124 && show_bootstrap
125 && bootstrap.is_empty())
126 || (show_agents && !show_skills && agents.is_empty())
127 || (show_skills && !show_agents && !show_bootstrap && skills.is_empty())
128 || (show_bootstrap && !show_agents && !show_skills && bootstrap.is_empty())
129 {
130 println!(" no managed items");
131 }
132}
133
134pub fn print_sync_report(report: &SyncReport, json: bool, no_upgrade_hint: bool) {
136 if json {
137 print_sync_report_json(report);
138 } else {
139 print_sync_report_human(report, no_upgrade_hint);
140 }
141}
142
143fn is_dry_run(report: &SyncReport) -> bool {
146 report.dry_run
147}
148
149fn print_sync_report_json(report: &SyncReport) {
150 println!("{}", sync_report_json(report));
151}
152
153pub fn sync_report_json(report: &SyncReport) -> serde_json::Value {
154 #[derive(Serialize)]
155 struct JsonTargetOutcome {
156 name: String,
157 synced: usize,
158 removed: usize,
159 errors: Vec<String>,
160 }
161
162 #[derive(Serialize)]
163 struct JsonReport {
164 ok: bool,
165 dry_run: bool,
166 installed: usize,
167 updated: usize,
168 removed: usize,
169 conflicts: usize,
170 kept: usize,
171 skipped: usize,
172 native_emitted: usize,
173 native_removed: usize,
174 upgrades_available: usize,
175 targets: Vec<JsonTargetOutcome>,
176 diagnostics: Vec<Diagnostic>,
177 }
178
179 let mut installed = 0;
180 let mut updated = 0;
181 let mut removed = 0;
182 let mut conflicts = 0;
183 let mut kept = 0;
184 let mut skipped = 0;
185
186 for outcome in &report.applied.outcomes {
187 match outcome.action {
188 ActionTaken::Installed => installed += 1,
189 ActionTaken::Updated => updated += 1,
190 ActionTaken::Merged => updated += 1,
191 ActionTaken::Conflicted => conflicts += 1,
192 ActionTaken::Removed => removed += 1,
193 ActionTaken::Kept => kept += 1,
194 ActionTaken::Skipped => skipped += 1,
195 }
196 }
197
198 for outcome in &report.pruned {
199 if matches!(outcome.action, ActionTaken::Removed) {
200 removed += 1;
201 }
202 }
203
204 let targets = report
205 .target_outcomes
206 .iter()
207 .map(|outcome| JsonTargetOutcome {
208 name: outcome.target.clone(),
209 synced: outcome.items_synced,
210 removed: outcome.items_removed,
211 errors: outcome.errors.clone(),
212 })
213 .collect();
214
215 serde_json::to_value(JsonReport {
216 ok: conflicts == 0,
217 dry_run: report.dry_run,
218 installed,
219 updated,
220 removed,
221 conflicts,
222 kept,
223 skipped,
224 native_emitted: report.native_emitted.len(),
225 native_removed: report.native_removed.len(),
226 upgrades_available: report.upgrades_available,
227 targets,
228 diagnostics: report.diagnostics.clone(),
229 })
230 .unwrap_or_else(|_| serde_json::json!({}))
231}
232
233fn print_sync_report_human(report: &SyncReport, no_upgrade_hint: bool) {
234 let mut stdout = StandardStream::stdout(color_choice());
235
236 let mut installed = 0usize;
237 let mut updated = 0usize;
238 let mut removed = 0usize;
239 let mut conflicts = 0usize;
240 let mut kept = 0usize;
241
242 for outcome in &report.applied.outcomes {
244 match outcome.action {
245 ActionTaken::Installed => {
246 installed += 1;
247 print_action_line(&mut stdout, "+", Color::Green, outcome);
248 }
249 ActionTaken::Updated | ActionTaken::Merged => {
250 updated += 1;
251 print_action_line(&mut stdout, "~", Color::Yellow, outcome);
252 }
253 ActionTaken::Conflicted => {
254 conflicts += 1;
255 print_action_line(&mut stdout, "!", Color::Red, outcome);
256 }
257 ActionTaken::Removed => {
258 removed += 1;
259 print_action_line(&mut stdout, "-", Color::Red, outcome);
260 }
261 ActionTaken::Kept => {
262 kept += 1;
263 }
264 ActionTaken::Skipped => {}
265 }
266 }
267
268 for outcome in &report.pruned {
269 if matches!(outcome.action, ActionTaken::Removed) {
270 removed += 1;
271 print_action_line(&mut stdout, "-", Color::Red, outcome);
272 }
273 }
274
275 let native_emitted = report.native_emitted.len();
276 let native_removed = report.native_removed.len();
277 for (target_root, dest_path) in &report.native_removed {
278 print_native_line(&mut stdout, "-", Color::Red, target_root, dest_path);
279 }
280 for (target_root, dest_path) in &report.native_emitted {
281 print_native_line(&mut stdout, "+", Color::Green, target_root, dest_path);
282 }
283
284 let _ = writeln!(stdout);
286 let dry = is_dry_run(report);
287 if installed > 0 {
288 if dry {
289 let _ = writeln!(stdout, " would install {installed} new items");
290 } else {
291 let _ = writeln!(stdout, " installed {installed} new items");
292 }
293 }
294 if updated > 0 {
295 if dry {
296 let _ = writeln!(stdout, " would update {updated} items");
297 } else {
298 let _ = writeln!(stdout, " updated {updated} items");
299 }
300 }
301 if removed > 0 {
302 if dry {
303 let _ = writeln!(stdout, " would remove {removed} orphans");
304 } else {
305 let _ = writeln!(stdout, " removed {removed} orphans");
306 }
307 }
308 if native_emitted > 0 {
309 if dry {
310 let _ = writeln!(stdout, " would emit {native_emitted} native agents");
311 } else {
312 let _ = writeln!(stdout, " emitted {native_emitted} native agents");
313 }
314 }
315 if native_removed > 0 {
316 if dry {
317 let _ = writeln!(stdout, " would remove {native_removed} native agents");
318 } else {
319 let _ = writeln!(stdout, " removed {native_removed} native agents");
320 }
321 }
322 if kept > 0 {
323 let _ = writeln!(stdout, " kept {kept} locally modified");
324 }
325 if conflicts > 0 {
326 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(Color::Red)));
327 let _ = writeln!(
328 stdout,
329 " conflicts {conflicts} files (run `{cmd}` after fixing)",
330 cmd = managed_cmd("mars resolve"),
331 );
332 let _ = stdout.reset();
333 }
334
335 if installed == 0
336 && updated == 0
337 && removed == 0
338 && conflicts == 0
339 && kept == 0
340 && native_emitted == 0
341 && native_removed == 0
342 {
343 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(Color::Green)));
344 let _ = writeln!(stdout, " already up to date");
345 let _ = stdout.reset();
346 }
347
348 let mut stderr = StandardStream::stderr(color_choice());
350 for diag in &report.diagnostics {
351 let color = match diag.level {
352 crate::diagnostic::DiagnosticLevel::Error => Color::Red,
353 crate::diagnostic::DiagnosticLevel::Warning => Color::Yellow,
354 crate::diagnostic::DiagnosticLevel::Info => Color::Cyan,
355 };
356 let _ = stderr.set_color(ColorSpec::new().set_fg(Some(color)));
357 let _ = writeln!(stderr, " {diag}");
358 let _ = stderr.reset();
359 }
360
361 if report.upgrades_available > 0 && !report.dry_run && !no_upgrade_hint {
362 let noun = if report.upgrades_available == 1 {
363 "upgrade"
364 } else {
365 "upgrades"
366 };
367 let _ = stderr.set_color(ColorSpec::new().set_fg(Some(Color::Cyan)));
368 let _ = writeln!(
369 stderr,
370 " ℹ {} {noun} available — run `{cmd}` to update",
371 report.upgrades_available,
372 cmd = managed_cmd("mars upgrade"),
373 );
374 let _ = stderr.reset();
375 }
376}
377
378fn print_action_line(
379 stdout: &mut StandardStream,
380 prefix: &str,
381 color: Color,
382 outcome: &ActionOutcome,
383) {
384 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(color)));
385 let _ = write!(stdout, " {prefix} ");
386 let _ = stdout.reset();
387 let _ = writeln!(stdout, "{} ({})", outcome.dest_path, outcome.item_id.kind);
388}
389
390fn print_native_line(
391 stdout: &mut StandardStream,
392 prefix: &str,
393 color: Color,
394 target_root: &str,
395 dest_path: &str,
396) {
397 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(color)));
398 let _ = write!(stdout, " {prefix} ");
399 let _ = stdout.reset();
400 let _ = writeln!(stdout, "{target_root}/{dest_path} (native agent)");
401}
402
403pub fn print_list(entries: &[ListEntry], json: bool) {
405 if json {
406 println!("{}", serde_json::to_string(entries).unwrap_or_default());
407 } else {
408 print_list_human(entries);
409 }
410}
411
412fn print_list_human(entries: &[ListEntry]) {
413 if entries.is_empty() {
414 println!(" no managed items");
415 return;
416 }
417
418 let source_w = entries
420 .iter()
421 .map(|e| e.source.len())
422 .max()
423 .unwrap_or(6)
424 .max(6);
425 let item_w = entries
426 .iter()
427 .map(|e| e.item.len())
428 .max()
429 .unwrap_or(4)
430 .max(4);
431 let version_w = entries
432 .iter()
433 .map(|e| e.version.len())
434 .max()
435 .unwrap_or(7)
436 .max(7);
437
438 println!(
440 "{:<source_w$} {:<item_w$} {:<version_w$} STATUS",
441 "SOURCE", "ITEM", "VERSION"
442 );
443
444 let mut stdout = StandardStream::stdout(color_choice());
445 for entry in entries {
446 let _ = write!(
447 stdout,
448 "{:<source_w$} {:<item_w$} {:<version_w$} ",
449 entry.source, entry.item, entry.version
450 );
451 let color = match entry.status.as_str() {
452 "ok" => Color::Green,
453 "modified" => Color::Yellow,
454 "conflicted" => Color::Red,
455 _ => Color::White,
456 };
457 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(color)));
458 let _ = writeln!(stdout, "{}", entry.status);
459 let _ = stdout.reset();
460 }
461}
462
463pub fn print_doctor(errors: &[String], warnings: &[String], json: bool) {
465 if json {
466 #[derive(Serialize)]
467 struct DoctorReport {
468 ok: bool,
469 errors: Vec<String>,
470 warnings: Vec<String>,
471 }
472 let report = DoctorReport {
473 ok: errors.is_empty(),
474 errors: errors.to_vec(),
475 warnings: warnings.to_vec(),
476 };
477 println!("{}", serde_json::to_string(&report).unwrap_or_default());
478 } else {
479 let mut stdout = StandardStream::stdout(color_choice());
480 if errors.is_empty() && warnings.is_empty() {
481 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(Color::Green)));
482 let _ = writeln!(stdout, " all checks passed");
483 let _ = stdout.reset();
484 } else {
485 for warning in warnings {
486 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(Color::Yellow)));
487 let _ = write!(stdout, " ⚠ ");
488 let _ = stdout.reset();
489 let _ = writeln!(stdout, "{warning}");
490 }
491
492 for error in errors {
493 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(Color::Red)));
494 let _ = write!(stdout, " ✗ ");
495 let _ = stdout.reset();
496 let _ = writeln!(stdout, "{error}");
497 }
498 let _ = writeln!(stdout);
499 if !warnings.is_empty() {
500 let _ = writeln!(stdout, " {} warning(s)", warnings.len());
501 }
502 if !errors.is_empty() {
503 let _ = writeln!(stdout, " {} error(s)", errors.len());
504 }
505 }
506 }
507}
508
509pub fn print_json<T: Serialize>(value: &T) {
511 println!("{}", serde_json::to_string(value).unwrap_or_default());
512}
513
514pub fn print_success(msg: &str) {
516 let mut stdout = StandardStream::stdout(color_choice());
517 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(Color::Green)));
518 let _ = write!(stdout, " ✓ ");
519 let _ = stdout.reset();
520 let _ = writeln!(stdout, "{msg}");
521}
522
523pub fn print_warn(msg: &str) {
525 let mut stdout = StandardStream::stdout(color_choice());
526 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(Color::Yellow)));
527 let _ = write!(stdout, " ⚠ ");
528 let _ = stdout.reset();
529 let _ = writeln!(stdout, "{msg}");
530}
531
532pub fn print_error(msg: &str) {
534 let mut stdout = StandardStream::stdout(color_choice());
535 let _ = stdout.set_color(ColorSpec::new().set_fg(Some(Color::Red)));
536 let _ = write!(stdout, " ✗ ");
537 let _ = stdout.reset();
538 let _ = writeln!(stdout, "{msg}");
539}
540
541pub fn print_info(msg: &str) {
543 println!(" {msg}");
544}