1use crate::config::schema::{ArgumentType, CommandDefinition, CommandsConfig};
38use colored::Colorize;
39
40pub trait HelpFormatter {
76 fn format_app(&self, config: &CommandsConfig) -> String;
80
81 fn format_command(&self, config: &CommandsConfig, command: &str) -> String;
87}
88
89#[derive(Debug, Default)]
114pub struct DefaultHelpFormatter;
115
116impl DefaultHelpFormatter {
117 pub fn new() -> Self {
119 Self
120 }
121
122 fn type_label(t: ArgumentType) -> &'static str {
128 t.as_str()
129 }
130
131 fn pad(s: &str, width: usize) -> String {
133 format!("{:<width$}", s, width = width)
134 }
135
136 fn find_command<'a>(config: &'a CommandsConfig, name: &str) -> Option<&'a CommandDefinition> {
138 config
139 .commands
140 .iter()
141 .find(|cmd| cmd.name == name || cmd.aliases.iter().any(|a| a == name))
142 }
143
144 fn format_arguments(cmd: &CommandDefinition) -> String {
146 if cmd.arguments.is_empty() {
147 return String::new();
148 }
149
150 let col_width = cmd
152 .arguments
153 .iter()
154 .map(|a| a.name.len())
155 .max()
156 .unwrap_or(0)
157 + 4; let mut out = format!("\n{}\n", "ARGUMENTS:".bold());
160 for arg in &cmd.arguments {
161 let req = if arg.required { "required" } else { "optional" };
162 let label = format!("({}, {req})", Self::type_label(arg.arg_type));
163 out.push_str(&format!(
164 " {} {} {}\n",
165 Self::pad(&arg.name, col_width).green(),
166 label.dimmed(),
167 arg.description
168 ));
169 }
170 out
171 }
172
173 fn format_options(cmd: &CommandDefinition) -> String {
175 if cmd.options.is_empty() {
176 return String::new();
177 }
178
179 let flags: Vec<String> = cmd
181 .options
182 .iter()
183 .map(|opt| {
184 let short = opt
185 .short
186 .as_deref()
187 .map(|s| format!("-{s}"))
188 .unwrap_or_default();
189 let long = opt
190 .long
191 .as_deref()
192 .map(|l| format!("--{l}"))
193 .unwrap_or_default();
194 match (short.is_empty(), long.is_empty()) {
195 (false, false) => format!("{short}, {long}"),
196 (false, true) => short,
197 (true, false) => long,
198 (true, true) => opt.name.clone(),
199 }
200 })
201 .collect();
202
203 let col_width = flags.iter().map(|f| f.len()).max().unwrap_or(0) + 4;
204
205 let mut out = format!("\n{}\n", "OPTIONS:".bold());
206 for (opt, flag) in cmd.options.iter().zip(flags.iter()) {
207 let type_label = format!("({})", Self::type_label(opt.option_type));
208 let default_note = opt
209 .default
210 .as_deref()
211 .map(|d| format!(" [default: {d}]"))
212 .unwrap_or_default();
213 out.push_str(&format!(
214 " {} {} {}{}\n",
215 Self::pad(flag, col_width).yellow(),
216 type_label.dimmed(),
217 opt.description,
218 default_note.dimmed()
219 ));
220 }
221 out
222 }
223
224 fn format_aliases(cmd: &CommandDefinition) -> String {
226 if cmd.aliases.is_empty() {
227 return String::new();
228 }
229 format!(
230 "\n{}\n {}\n",
231 "ALIASES:".bold(),
232 cmd.aliases.join(", ").italic()
233 )
234 }
235
236 fn usage_args(cmd: &CommandDefinition) -> String {
238 let args: String = cmd
239 .arguments
240 .iter()
241 .map(|a| {
242 if a.required {
243 format!("<{}>", a.name)
244 } else {
245 format!("[{}]", a.name)
246 }
247 })
248 .collect::<Vec<_>>()
249 .join(" ");
250
251 let opts = if cmd.options.is_empty() {
252 String::new()
253 } else {
254 " [options]".to_string()
255 };
256
257 format!("{args}{opts}")
258 }
259}
260
261impl HelpFormatter for DefaultHelpFormatter {
262 fn format_app(&self, config: &CommandsConfig) -> String {
279 let mut out = String::new();
280
281 out.push_str(&format!(
283 "{} {}\n",
284 config.metadata.prompt.bold().cyan(),
285 config.metadata.version.dimmed()
286 ));
287
288 out.push('\n');
290 out.push_str(&format!("{}\n", "USAGE:".bold()));
291 out.push_str(&format!(
292 " {} {} [arguments] [options]\n",
293 config.metadata.prompt,
294 "<command>".green()
295 ));
296
297 if !config.commands.is_empty() {
299 out.push('\n');
300 out.push_str(&format!("{}\n", "COMMANDS:".bold()));
301
302 let col_width = config
303 .commands
304 .iter()
305 .map(|c| c.name.len())
306 .max()
307 .unwrap_or(0)
308 + 4;
309
310 for cmd in &config.commands {
311 out.push_str(&format!(
312 " {} {}\n",
313 Self::pad(&cmd.name, col_width).green(),
314 cmd.description
315 ));
316 }
317 }
318
319 out.push('\n');
321 out.push_str(&format!(
322 "{} '{}' {}\n",
323 "Run".dimmed(),
324 format!("{} --help <command>", config.metadata.prompt).italic(),
325 "for more information on a command.".dimmed()
326 ));
327
328 out
329 }
330
331 fn format_command(&self, config: &CommandsConfig, command: &str) -> String {
353 let Some(cmd) = Self::find_command(config, command) else {
354 let available = config
356 .commands
357 .iter()
358 .map(|c| c.name.as_str())
359 .collect::<Vec<_>>()
360 .join(", ");
361 return format!(
362 "{} '{}'\n\nAvailable commands: {}\n",
363 "Unknown command:".red().bold(),
364 command,
365 available
366 );
367 };
368
369 let mut out = String::new();
370
371 out.push_str(&format!(
373 "{} — {}\n",
374 cmd.name.bold().cyan(),
375 cmd.description
376 ));
377
378 out.push('\n');
380 out.push_str(&format!("{}\n", "USAGE:".bold()));
381 out.push_str(&format!(
382 " {} {}\n",
383 cmd.name.green(),
384 Self::usage_args(cmd)
385 ));
386
387 out.push_str(&Self::format_arguments(cmd));
389 out.push_str(&Self::format_options(cmd));
390 out.push_str(&Self::format_aliases(cmd));
391
392 out
393 }
394}
395
396#[cfg(test)]
401mod tests {
402 use super::*;
403 use crate::config::schema::{
404 ArgumentDefinition, ArgumentType, CommandDefinition, Metadata, OptionDefinition,
405 };
406
407 fn no_color() {
409 colored::control::set_override(false);
410 }
411
412 fn make_config() -> CommandsConfig {
417 CommandsConfig {
418 metadata: Metadata {
419 version: "1.0.0".to_string(),
420 prompt: "myapp".to_string(),
421 prompt_suffix: " > ".to_string(),
422 },
423 commands: vec![
424 CommandDefinition {
425 name: "hello".to_string(),
426 aliases: vec!["hi".to_string(), "hey".to_string()],
427 description: "Say hello to someone".to_string(),
428 required: false,
429 arguments: vec![ArgumentDefinition {
430 name: "name".to_string(),
431 arg_type: ArgumentType::String,
432 required: true,
433 description: "Name to greet".to_string(),
434 validation: vec![],
435 }],
436 options: vec![OptionDefinition {
437 name: "loud".to_string(),
438 short: Some("l".to_string()),
439 long: Some("loud".to_string()),
440 option_type: ArgumentType::Bool,
441 required: false,
442 default: None,
443 description: "Use uppercase".to_string(),
444 choices: vec![],
445 }],
446 implementation: "hello_handler".to_string(),
447 },
448 CommandDefinition {
449 name: "process".to_string(),
450 aliases: vec![],
451 description: "Process data files".to_string(),
452 required: true,
453 arguments: vec![],
454 options: vec![],
455 implementation: "process_handler".to_string(),
456 },
457 ],
458 global_options: vec![],
459 }
460 }
461
462 fn make_formatter() -> DefaultHelpFormatter {
463 DefaultHelpFormatter::new()
464 }
465
466 #[test]
471 fn test_new_and_default_are_equivalent() {
472 let _a = DefaultHelpFormatter::new();
474 let _b = DefaultHelpFormatter::default();
475 }
476
477 #[test]
482 fn test_format_app_contains_prompt_and_version() {
483 no_color();
484 let config = make_config();
485 let out = make_formatter().format_app(&config);
486
487 assert!(out.contains("myapp"), "should contain prompt");
488 assert!(out.contains("1.0.0"), "should contain version");
489 }
490
491 #[test]
492 fn test_format_app_contains_all_commands() {
493 no_color();
494 let config = make_config();
495 let out = make_formatter().format_app(&config);
496
497 assert!(out.contains("hello"), "should list command 'hello'");
498 assert!(out.contains("process"), "should list command 'process'");
499 assert!(
500 out.contains("Say hello to someone"),
501 "should include description"
502 );
503 }
504
505 #[test]
506 fn test_format_app_contains_usage_and_footer() {
507 no_color();
508 let config = make_config();
509 let out = make_formatter().format_app(&config);
510
511 assert!(out.contains("USAGE:"), "should have USAGE section");
512 assert!(out.contains("COMMANDS:"), "should have COMMANDS section");
513 assert!(
514 out.contains("--help <command>"),
515 "should hint at per-command help"
516 );
517 }
518
519 #[test]
520 fn test_format_app_empty_commands() {
521 no_color();
522 let mut config = make_config();
523 config.commands.clear();
524 let out = make_formatter().format_app(&config);
525
526 assert!(out.contains("myapp"));
528 assert!(!out.contains("COMMANDS:"));
529 }
530
531 #[test]
536 fn test_format_command_by_name() {
537 no_color();
538 let config = make_config();
539 let out = make_formatter().format_command(&config, "hello");
540
541 assert!(out.contains("hello"), "should contain command name");
542 assert!(
543 out.contains("Say hello to someone"),
544 "should contain description"
545 );
546 }
547
548 #[test]
549 fn test_format_command_by_alias() {
550 no_color();
551 let config = make_config();
552 let out = make_formatter().format_command(&config, "hi");
554
555 assert!(out.contains("hello"));
557 assert!(out.contains("Say hello to someone"));
558 }
559
560 #[test]
561 fn test_format_command_shows_arguments() {
562 no_color();
563 let config = make_config();
564 let out = make_formatter().format_command(&config, "hello");
565
566 assert!(out.contains("ARGUMENTS:"), "should have ARGUMENTS section");
567 assert!(out.contains("name"), "should list argument name");
568 assert!(out.contains("string"), "should show argument type");
569 assert!(out.contains("required"), "should show required status");
570 assert!(out.contains("Name to greet"), "should show description");
571 }
572
573 #[test]
574 fn test_format_command_shows_options() {
575 no_color();
576 let config = make_config();
577 let out = make_formatter().format_command(&config, "hello");
578
579 assert!(out.contains("OPTIONS:"), "should have OPTIONS section");
580 assert!(out.contains("-l"), "should show short flag");
581 assert!(out.contains("--loud"), "should show long flag");
582 assert!(
583 out.contains("Use uppercase"),
584 "should show option description"
585 );
586 }
587
588 #[test]
589 fn test_format_command_shows_aliases() {
590 no_color();
591 let config = make_config();
592 let out = make_formatter().format_command(&config, "hello");
593
594 assert!(out.contains("ALIASES:"), "should have ALIASES section");
595 assert!(out.contains("hi"), "should list alias 'hi'");
596 assert!(out.contains("hey"), "should list alias 'hey'");
597 }
598
599 #[test]
600 fn test_format_command_no_aliases_section_when_empty() {
601 no_color();
602 let config = make_config();
603 let out = make_formatter().format_command(&config, "process");
605
606 assert!(!out.contains("ALIASES:"), "should omit ALIASES section");
607 }
608
609 #[test]
610 fn test_format_command_no_arguments_section_when_empty() {
611 no_color();
612 let config = make_config();
613 let out = make_formatter().format_command(&config, "process");
614
615 assert!(!out.contains("ARGUMENTS:"), "should omit ARGUMENTS section");
616 }
617
618 #[test]
619 fn test_format_command_no_options_section_when_empty() {
620 no_color();
621 let config = make_config();
622 let out = make_formatter().format_command(&config, "process");
623
624 assert!(!out.contains("OPTIONS:"), "should omit OPTIONS section");
625 }
626
627 #[test]
632 fn test_format_command_unknown_returns_error_string() {
633 no_color();
634 let config = make_config();
635 let out = make_formatter().format_command(&config, "nonexistent");
636
637 assert!(
638 out.contains("Unknown command"),
639 "should signal unknown command"
640 );
641 assert!(
642 out.contains("nonexistent"),
643 "should echo the unknown name back"
644 );
645 }
646
647 #[test]
648 fn test_format_command_unknown_lists_available() {
649 no_color();
650 let config = make_config();
651 let out = make_formatter().format_command(&config, "nonexistent");
652
653 assert!(
655 out.contains("hello"),
656 "should list available command 'hello'"
657 );
658 assert!(
659 out.contains("process"),
660 "should list available command 'process'"
661 );
662 }
663
664 #[test]
669 fn test_trait_is_dyn_compatible() {
670 no_color();
671 let formatter: Box<dyn HelpFormatter> = Box::new(DefaultHelpFormatter::new());
673 let config = make_config();
674 let _ = formatter.format_app(&config);
675 }
676
677 #[test]
682 fn test_format_command_shows_default_value() {
683 no_color();
684 let mut config = make_config();
685 config.commands[0].options[0].default = Some("false".to_string());
687 let out = make_formatter().format_command(&config, "hello");
688
689 assert!(out.contains("false"), "should show default value");
690 }
691
692 struct MinimalFormatter;
697
698 impl HelpFormatter for MinimalFormatter {
699 fn format_app(&self, config: &CommandsConfig) -> String {
700 config.metadata.prompt.clone()
701 }
702 fn format_command(&self, _config: &CommandsConfig, command: &str) -> String {
703 command.to_string()
704 }
705 }
706
707 #[test]
708 fn test_custom_formatter_via_trait_object() {
709 let config = make_config();
710 let f: Box<dyn HelpFormatter> = Box::new(MinimalFormatter);
711
712 assert_eq!(f.format_app(&config), "myapp");
713 assert_eq!(f.format_command(&config, "hello"), "hello");
714 }
715}