diff options
| author | 魏曹先生 <1992414357@qq.com> | 2026-05-07 22:14:36 +0800 |
|---|---|---|
| committer | 魏曹先生 <1992414357@qq.com> | 2026-05-07 22:14:36 +0800 |
| commit | 3a58dc5cea258fb81e8443496f9dac309d11448c (patch) | |
| tree | edfba815a65b6b23da67c2c309ed0a96082fd812 /mingling_macros/src/dispatch_tree_gen.rs | |
| parent | 6703ccb0fc3aca7ba4bdb9083f199fe0c1b66bd9 (diff) | |
Add compile-time dispatch tree generation for O(len) command routing
Diffstat (limited to 'mingling_macros/src/dispatch_tree_gen.rs')
| -rw-r--r-- | mingling_macros/src/dispatch_tree_gen.rs | 195 |
1 files changed, 195 insertions, 0 deletions
diff --git a/mingling_macros/src/dispatch_tree_gen.rs b/mingling_macros/src/dispatch_tree_gen.rs new file mode 100644 index 0000000..840fe15 --- /dev/null +++ b/mingling_macros/src/dispatch_tree_gen.rs @@ -0,0 +1,195 @@ +//! Dispatch Tree Generation +//! +//! This module generates the dispatch tree code for the `dispatch_tree` feature. +//! It builds a compact, hardcoded match tree at compile time to achieve O(len) +//! command dispatch. +//! +//! # Algorithm +//! +//! For each depth, group nodes by the character at that depth. +//! - If a group has only one node: emit `starts_with` check for the full name. +//! - If a group has multiple nodes: emit a `match raw_chars.nth(depth)` arm and recurse. +//! - At the leaf: call `Dispatcher::begin` on the matched dispatcher. +//! +//! Names are matched with a trailing space (e.g. "hello ") to ensure exact boundary. + +use proc_macro2::TokenStream; +use quote::quote; +use std::collections::BTreeMap; + +/// Generate the `get_nodes()` function body for a ProgramCollect impl. +pub fn gen_get_nodes(entries: &[(String, String, String)]) -> TokenStream { + let mut node_entries = Vec::new(); + + for (node_name, _disp_type, _entry_name) in entries { + let static_name_str = format!("__internal_dispatcher_{}", node_name.replace('.', "_")); + let static_ident = syn::Ident::new(&static_name_str, proc_macro2::Span::call_site()); + + let node_display_name = node_name.replace('.', " "); + let node_display_lit = syn::LitStr::new(&node_display_name, proc_macro2::Span::call_site()); + + node_entries.push(quote! { + (#node_display_lit.to_string(), & #static_ident) + }); + } + + quote! { + fn get_nodes() -> Vec<(String, &'static (dyn ::mingling::Dispatcher<Self::Enum> + Send + Sync))> { + vec![ + #(#node_entries),* + ] + } + } +} + +/// Generate the `dispatch_args_trie()` function body for a ProgramCollect impl. +/// +/// Builds a hardcoded match tree: at each depth, group nodes by character. +/// Single-node groups use `starts_with`; multi-node groups recurse with `nth()` match. +pub fn gen_dispatch_args_trie(entries: &[(String, String, String)]) -> TokenStream { + // Prepare (display_name, disp_type) pairs. + // display_name = node_name.replace('.', " ") + let nodes: Vec<(String, String)> = entries + .iter() + .map(|(name, disp, _)| (name.replace('.', " "), disp.clone())) + .collect(); + + let dispatch_body = build_dispatch_body(&nodes, 0); + + quote! { + fn dispatch_args_trie( + raw: &Vec<String>, + ) -> Result<::mingling::AnyOutput<Self::Enum>, ::mingling::error::ProgramInternalExecuteError> + { + let raw_string = format!("{} ", raw.join(" ")); + let raw_str = raw_string.as_str(); + let mut raw_chars = raw_str.chars(); + #dispatch_body + } + } +} + +/// Recursively build the trie match body. +/// +/// `nodes`: slice of (display_name, disp_type) for commands that share the same prefix so far. +/// `depth`: The character index currently being matched. +/// +/// Returns a `TokenStream` representing the match block at this depth. +fn build_dispatch_body(nodes: &[(String, String)], depth: usize) -> TokenStream { + if nodes.is_empty() { + return quote! { + return Ok(Self::build_dispatcher_not_found(raw.clone())); + }; + } + + // Group by character at `depth` + // Nodes that end exactly at this depth (name is closed – rare but possible, e.g. "hell") + // are collected into `exact_nodes`. All others go into `groups[char]`. + let mut groups: BTreeMap<char, Vec<(String, String)>> = BTreeMap::new(); + let mut exact_nodes: Vec<(String, String)> = Vec::new(); + + for (name, disp_type) in nodes { + if let Some(ch) = name.chars().nth(depth) { + groups + .entry(ch) + .or_default() + .push((name.clone(), disp_type.clone())); + } else { + exact_nodes.push((name.clone(), disp_type.clone())); + } + } + + // Build a dispatch arm for a single node via `starts_with` + let make_starts_with_arm = |name: &str, disp_type: &str| -> TokenStream { + let name_space = format!("{} ", name); + let name_lit = syn::LitStr::new(&name_space, proc_macro2::Span::call_site()); + let disp_ident = syn::Ident::new(disp_type, proc_macro2::Span::call_site()); + quote! { + if let Some(stripped) = raw_str.strip_prefix(#name_lit) { + let __cp = <#disp_ident as ::mingling::Dispatcher<Self::Enum>>::begin( + &#disp_ident::default(), + stripped + .split_whitespace() + .map(String::from) + .collect::<Vec<String>>(), + ); + return match __cp { + ::mingling::ChainProcess::Ok(any_output) => Ok(any_output.0), + ::mingling::ChainProcess::Err(chain_process_error) => { + Err(chain_process_error.into()) + } + }; + } + } + }; + + // Build match arms + let mut arms = Vec::new(); + + for (&ch, sub_nodes) in &groups { + let ch_char = ch; + + if sub_nodes.len() == 1 { + // Only one candidate – use `starts_with` directly at this depth. + let (name, disp_type) = &sub_nodes[0]; + let arm = make_starts_with_arm(name, disp_type); + arms.push(quote! { + Some(#ch_char) => { + #arm + return Ok(Self::build_dispatcher_not_found(raw.clone())); + } + }); + } else { + // Multiple candidates – recurse deeper + let sub_body = build_dispatch_body(sub_nodes, depth + 1); + arms.push(quote! { + Some(#ch_char) => { + #sub_body + } + }); + } + } + + // Exact-match nodes at this depth + // These are names that are a prefix of other names (e.g. "hell" when "hello" exists). + // They are tried first via `starts_with`, then fall through to `raw_chars.nth(depth)` match. + let exact_checks: Vec<TokenStream> = exact_nodes + .iter() + .map(|(name, disp_type)| make_starts_with_arm(name, disp_type)) + .collect(); + + // Assemble + // If there are exact nodes, first check starts_with, then do matching. + if !exact_checks.is_empty() && !groups.is_empty() { + // Exact nodes + deeper groups: do starts_with checks, then match on nth(depth) + let match_body = quote! { + match raw_chars.nth(0) { + #(#arms)* + _ => return Ok(Self::build_dispatcher_not_found(raw.clone())), + } + }; + quote! { + #(#exact_checks)* + #match_body + } + } else if !exact_checks.is_empty() { + // Only exact nodes, no deeper groups + quote! { + #(#exact_checks)* + return Ok(Self::build_dispatcher_not_found(raw.clone())); + } + } else if arms.is_empty() { + // Only fallback (shouldn't happen if nodes is non-empty) + quote! { + return Ok(Self::build_dispatcher_not_found(raw.clone())); + } + } else { + // Only group arms + quote! { + match raw_chars.nth(0) { + #(#arms)* + _ => return Ok(Self::build_dispatcher_not_found(raw.clone())), + } + } + } +} |
