coding_tools/
completion.rs1use std::sync::Arc;
16
17use veks_completion::{CommandTree, Node, ValueProvider};
18
19use crate::{cli, rules};
20
21#[derive(Clone, Copy)]
23enum Field {
24 Id,
25 Tag,
26 Def,
27}
28
29fn store_values(field: Field) -> Vec<String> {
33 let Ok(cwd) = std::env::current_dir() else {
34 return Vec::new();
35 };
36 let Some(root) = rules::discover_root(&cwd) else {
37 return Vec::new();
38 };
39 let Ok(text) = std::fs::read_to_string(rules::store_path(&root)) else {
40 return Vec::new();
41 };
42 let Ok(store) = rules::parse_store(&text) else {
43 return Vec::new();
44 };
45 match field {
46 Field::Id => store.rules.iter().map(|r| r.id.clone()).collect(),
47 Field::Tag => {
48 let mut tags: Vec<String> =
49 store.rules.iter().flat_map(|r| r.tags.iter().cloned()).collect();
50 tags.sort();
51 tags.dedup();
52 tags
53 }
54 Field::Def => store.defs.keys().cloned().collect(),
55 }
56}
57
58fn store_provider(field: Field) -> ValueProvider {
60 Arc::new(move |partial: &str, _: &[&str]| {
61 store_values(field).into_iter().filter(|v| v.starts_with(partial)).collect()
62 })
63}
64
65fn set_provider(values: Vec<String>) -> ValueProvider {
67 Arc::new(move |partial: &str, _: &[&str]| {
68 values.iter().filter(|v| v.starts_with(partial)).cloned().collect()
69 })
70}
71
72fn with_store_providers(sub: &str, node: Node) -> Node {
75 match sub {
76 "check" => node
77 .with_value_provider("--id", store_provider(Field::Id))
78 .with_value_provider("--tag", store_provider(Field::Tag)),
79 "rules" => node
80 .with_value_provider("--promote", store_provider(Field::Id))
81 .with_value_provider("--remove", store_provider(Field::Id))
82 .with_value_provider("--tag", store_provider(Field::Tag))
83 .with_value_provider("--def", store_provider(Field::Def)),
84 _ => node,
85 }
86}
87
88pub fn command_tree() -> CommandTree {
92 let mut tree = CommandTree::new("ct");
93 for (name, grammar) in cli::grammars() {
94 let sub = name.strip_prefix("ct-").unwrap_or(name);
95 let value_flags: Vec<String> = grammar
96 .flags
97 .iter()
98 .filter(|f| f.kind != "boolean")
99 .map(|f| format!("--{}", f.name))
100 .collect();
101 let bool_flags: Vec<String> = grammar
102 .flags
103 .iter()
104 .filter(|f| f.kind == "boolean")
105 .map(|f| format!("--{}", f.name))
106 .collect();
107 let vf: Vec<&str> = value_flags.iter().map(String::as_str).collect();
108 let bf: Vec<&str> = bool_flags.iter().map(String::as_str).collect();
109
110 let mut node = Node::leaf_with_flags(&vf, &bf);
111 for f in grammar.flags.iter().filter(|f| f.kind != "boolean" && !f.values.is_empty()) {
112 node = node.with_value_provider(&format!("--{}", f.name), set_provider(f.values.clone()));
113 }
114 node = with_store_providers(sub, node);
115 tree = tree.command(sub, node);
116 }
117 let shells = ["bash", "zsh", "fish"].iter().map(|s| s.to_string()).collect();
120 tree = tree.command(
121 "completions",
122 Node::leaf(&[]).with_positional_provider(set_provider(shells)),
123 );
124 tree
125}