Skip to main content

hematite/agent/
routing.rs

1use super::conversation::WorkflowMode;
2
3#[derive(Clone, Copy, Debug, Eq, PartialEq)]
4pub enum QueryIntentClass {
5    ProductTruth,
6    RuntimeDiagnosis,
7    RepoArchitecture,
8    Toolchain,
9    Capability,
10    Implementation,
11    Research,
12    Unknown,
13}
14
15#[derive(Clone, Copy, Debug, Eq, PartialEq)]
16pub enum DirectAnswerKind {
17    About,
18    LanguageCapability,
19    UnsafeWorkflowPressure,
20    SessionMemory,
21    RecoveryRecipes,
22    McpLifecycle,
23    AuthorizationPolicy,
24    ToolClasses,
25    ToolRegistryOwnership,
26    SessionResetSemantics,
27    ProductSurface,
28    ReasoningSplit,
29    Identity,
30    WorkflowModes,
31    GemmaNative,
32    GemmaNativeSettings,
33    VerifyProfiles,
34    Toolchain,
35    HostInspection,
36    ArchitectSessionResetPlan,
37    Help,
38}
39
40#[derive(Clone, Copy, Debug)]
41pub struct QueryIntent {
42    pub primary_class: QueryIntentClass,
43    pub direct_answer: Option<DirectAnswerKind>,
44    pub grounded_trace_mode: bool,
45    pub capability_mode: bool,
46    pub capability_needs_repo: bool,
47    pub toolchain_mode: bool,
48    pub host_inspection_mode: bool,
49    pub maintainer_workflow_mode: bool,
50    pub workspace_workflow_mode: bool,
51    pub architecture_overview_mode: bool,
52    pub sovereign_mode: bool,
53    pub surgical_filesystem_mode: bool,
54    pub scaffold_mode: bool,
55}
56
57fn contains_any(haystack: &str, needles: &[&str]) -> bool {
58    needles.iter().any(|needle| haystack.contains(needle))
59}
60
61fn contains_all(haystack: &str, needles: &[&str]) -> bool {
62    needles.iter().all(|needle| haystack.contains(needle))
63}
64
65const CODE_KEYWORDS: &[&str] = &[
66    ".rs",
67    ".js",
68    ".ts",
69    ".py",
70    ".go",
71    ".c",
72    ".cpp",
73    ".h",
74    ".hpp",
75    ".css",
76    ".html",
77    ".json",
78    ".toml",
79    ".yaml",
80    ".yml",
81    ".md",
82    ".sh",
83    ".ps1",
84    ".sql",
85    "rust",
86    "python",
87    "javascript",
88    "typescript",
89    "golang",
90    "react",
91    "svelte",
92    "vue",
93    "nextjs",
94    "node",
95    "npm",
96    "cargo",
97    "pip",
98    "logic",
99    "refactor",
100    "implementation",
101    "styles",
102    "script",
103];
104
105static CODE_KW_AC: std::sync::OnceLock<aho_corasick::AhoCorasick> = std::sync::OnceLock::new();
106
107fn code_kw_ac() -> &'static aho_corasick::AhoCorasick {
108    CODE_KW_AC
109        .get_or_init(|| aho_corasick::AhoCorasick::new(CODE_KEYWORDS).expect("valid patterns"))
110}
111
112fn mentions_reset_commands(lower: &str) -> bool {
113    contains_all(lower, &["/clear", "/new", "/forget"])
114}
115
116fn mentions_stable_product_surface(lower: &str) -> bool {
117    contains_any(
118        lower,
119        &[
120            "stable product-surface question",
121            "stable product surface question",
122            "stable product-surface questions",
123            "stable product surface questions",
124        ],
125    )
126}
127
128fn mentions_product_truth_routing(lower: &str) -> bool {
129    let asks_decision_policy = contains_any(
130        lower,
131        &[
132            "how hematite decides",
133            "how does hematite decide",
134            "decides whether",
135            "decide whether",
136        ],
137    );
138    let asks_direct_vs_inspect_split = contains_any(
139        lower,
140        &[
141            "answered as stable product truth",
142            "stable product truth",
143            "stable product behavior",
144            "answer directly",
145            "direct answer",
146            "inspect the repository",
147            "inspect repository",
148            "repository implementation",
149            "repo implementation",
150        ],
151    );
152    asks_decision_policy && asks_direct_vs_inspect_split
153}
154
155fn mentions_broad_system_walkthrough(lower: &str) -> bool {
156    let asks_walkthrough = contains_any(
157        lower,
158        &[
159            "walk me through",
160            "walk through",
161            "how hematite is wired",
162            "understand how hematite is wired",
163            "major runtime pieces",
164            "normal message moves",
165            "moves from the tui to the model and back",
166        ],
167    );
168    let asks_multiple_runtime_areas = contains_any(
169        lower,
170        &[
171            "session recovery",
172            "tool policy",
173            "mcp state",
174            "mcp policy",
175            "files own the major runtime pieces",
176            "which files own",
177            "where session recovery",
178            "where tool policy",
179            "where mcp state",
180        ],
181    );
182    asks_walkthrough && asks_multiple_runtime_areas
183}
184
185fn mentions_research_query(lower: &str) -> bool {
186    contains_any(
187        lower,
188        &[
189            "search for",
190            "lookup",
191            "look up",
192            "google",
193            "find info",
194            "find information",
195            "what are the latest",
196            "who is",
197            "who are",
198            "who was",
199            "what is",
200            "what was",
201            "who's",
202            "current version of",
203            "history of",
204            "what happened with",
205            "tell me about",
206            "tell me about the new",
207        ],
208    )
209}
210
211fn mentions_codebase_keywords(lower: &str) -> bool {
212    contains_any(
213        lower,
214        &[
215            "this repo",
216            "the repo",
217            "this project",
218            "the project",
219            "in the code",
220            "in my code",
221            "this codebase",
222            "the codebase",
223            "function",
224            "module",
225            "file",
226            "struct",
227            "enum",
228            "impl",
229            "trait",
230            "crate",
231            "logic",
232            "implementation",
233            "wiring",
234            "handles ",
235            "defined",
236            "located",
237        ],
238    )
239}
240
241fn mentions_capability_question(lower: &str) -> bool {
242    contains_any(
243        lower,
244        &[
245            "what is your purpose",
246            "what's your purpose",
247            "what are you for",
248            "what is your job",
249            "what's your job",
250            "what can you do",
251            "what are you capable",
252            "can you make projects",
253            "can you build projects",
254            "do you know other coding languages",
255            "other coding languages",
256            "what languages",
257            "can you use the internet",
258            "internet research capabilities",
259            "what tools do you have",
260        ],
261    )
262}
263
264fn mentions_creator_question(lower: &str) -> bool {
265    contains_any(
266        lower,
267        &[
268            "who is ocean bennett",
269            "who's ocean bennett",
270            "tell me about ocean bennett",
271            "who created you",
272            "who built you",
273            "who made you",
274            "who developed you",
275            "who engineered you",
276            "who engineered your architecture",
277            "who created hematite",
278            "who built hematite",
279            "who developed hematite",
280            "who engineered hematite",
281            "who maintains hematite",
282            "who authored hematite",
283            "who is the author",
284            "who wrote this",
285            "who made this app",
286        ],
287    )
288}
289
290fn capability_question_requires_repo_inspection(lower: &str) -> bool {
291    contains_any(
292        lower,
293        &[
294            "this repo",
295            "this repository",
296            "codebase",
297            "which files",
298            "implementation",
299            "in this project",
300        ],
301    )
302}
303
304/// Returns true for conversational, advisory, or declarative turns that should not
305/// trigger a blanket inspect_host(summary) call. Covers:
306///   - Advisory/opinion questions: "would more ram be nice?", "should I upgrade?"
307///   - Opinion assertions: "i think the gpu is fine"
308///   - Hypotheticals: "what if I had more ram", "if i upgraded the gpu"
309///   - Conversational acknowledgments: "makes sense", "so the cpu is fine", "ok so"
310///   - Positive/negative statements that aren't asking for new data
311///
312/// Does NOT block specific diagnostic routes — those fire before this catch-all guard.
313fn is_conversational_advisory(lower: &str) -> bool {
314    // ── Advisory openers — seeking opinion or recommendation, not data ──────────
315    let starts_advisory = lower.starts_with("would ")
316        || lower.starts_with("could ")
317        || lower.starts_with("should ")
318        || lower.starts_with("is that ")
319        || lower.starts_with("was that ")
320        || lower.starts_with("do you think")
321        || lower.starts_with("what do you think")
322        || lower.starts_with("does that ")
323        || lower.starts_with("is it worth")
324        || lower.starts_with("would it ");
325
326    // ── Opinion / belief assertions — not requesting fresh data ─────────────────
327    let opinion_opener = (lower.starts_with("i think ")
328        || lower.starts_with("i believe ")
329        || lower.starts_with("i know ")
330        || lower.starts_with("i guess ")
331        || lower.starts_with("i see,")
332        || lower.starts_with("i see ")
333        || lower.starts_with("i feel like"))
334        && !lower.trim_end().ends_with('?');
335
336    // ── Hypotheticals — not asking about current machine state ──────────────────
337    let hypothetical = lower.starts_with("what if ")
338        || lower.starts_with("if i ")
339        || lower.starts_with("if i'd ")
340        || lower.starts_with("say i ")
341        || lower.starts_with("suppose ");
342
343    // ── Conversational acknowledgments / pivots without a follow-up question ────
344    let no_question = !lower.trim_end().ends_with('?');
345    let no_imperative = !lower.contains("what is ")
346        && !lower.contains("what are ")
347        && !lower.contains("how do ")
348        && !lower.contains("how much ")
349        && !lower.contains("how many ")
350        && !lower.contains("show me")
351        && !lower.contains("tell me")
352        && !lower.contains("check ");
353    let acknowledgment = (lower.starts_with("makes sense")
354        || lower.starts_with("that makes sense")
355        || lower.starts_with("ok so ")
356        || lower.starts_with("right so ")
357        || lower.starts_with("so the ")
358        || lower.starts_with("so it ")
359        || lower.starts_with("so my ")
360        || lower.starts_with("ah ")
361        || lower.starts_with("got it")
362        || lower.starts_with("ok, ")
363        || lower.starts_with("everything "))
364        && no_question
365        && no_imperative;
366
367    // ── Confirmation-seeking tail — "right?", "correct?" ────────────────────────
368    let ends_confirmation = lower.trim_end_matches(['?', ' ']).ends_with("right")
369        || lower.trim_end_matches(['?', ' ']).ends_with("correct")
370        || lower.ends_with("right?")
371        || lower.ends_with("yeah?");
372
373    // ── Advisory tail vocabulary ─────────────────────────────────────────────────
374    let advisory_tail = lower.contains(" be nice")
375        || lower.contains(" be worth")
376        || lower.contains(" be helpful")
377        || lower.contains(" be useful")
378        || lower.contains(" be better")
379        || lower.contains(" be good")
380        || lower.contains(" help with")
381        || lower.contains("offload")
382        || lower.contains("upgrade");
383
384    starts_advisory
385        || opinion_opener
386        || hypothetical
387        || acknowledgment
388        || ends_confirmation && advisory_tail
389}
390
391fn mentions_host_inspection_question(lower: &str) -> bool {
392    let host_scope = lower.split_whitespace().any(|w| {
393        let w = w.trim_matches(|c: char| !c.is_alphanumeric());
394        matches!(
395            w,
396            "path"
397                | "pip"
398                | "winget"
399                | "choco"
400                | "scoop"
401                | "network"
402                | "adapter"
403                | "dns"
404                | "gateway"
405                | "wifi"
406                | "ethernet"
407                | "service"
408                | "services"
409                | "daemon"
410                | "process"
411                | "processes"
412                | "ram"
413                | "cpu"
414                | "gpu"
415                | "vram"
416                | "nvidia"
417                | "memory"
418                | "machine"
419                | "computer"
420                | "firewall"
421                | "vpn"
422                | "proxy"
423                | "internet"
424                | "online"
425                | "connectivity"
426                | "uptime"
427                | "reboot"
428                | "silicon"
429                | "throttle"
430                | "throttled"
431                | "throttling"
432                | "clocks"
433                | "mhz"
434                | "health"
435                | "report"
436                | "bitlocker"
437                | "rdp"
438                | "vss"
439                | "pagefile"
440                | "swap"
441                | "printer"
442                | "audio"
443                | "sound"
444                | "speaker"
445                | "speakers"
446                | "microphone"
447                | "mic"
448                | "bluetooth"
449                | "pairing"
450                | "headset"
451                | "headphones"
452                | "camera"
453                | "webcam"
454                | "msi"
455                | "msiexec"
456                | "onedrive"
457                | "indexer"
458                | "ntp"
459                | "w32tm"
460                | "winrm"
461                | "psremoting"
462                | "slat"
463                | "error"
464                | "warning"
465                | "event"
466                | "log"
467                | "throughput"
468                | "registry"
469                | "share"
470                | "mbps"
471                | "ad"
472                | "sid"
473                | "vm"
474                | "hyper-v"
475                | "hyperv"
476                | "dhcp"
477                | "lease"
478                | "login"
479                | "disk"
480                | "drive"
481                | "backup"
482                | "ssd"
483                | "hdd"
484                | "nvme"
485                | "encryption"
486        )
487    }) || contains_any(
488        lower,
489        &[
490            "package manager",
491            "environment doctor",
492            "ip address",
493            "ipconfig",
494            "task manager",
495            "developer tools",
496            "toolchains",
497            "local development",
498            "tcp connection",
499            "active connection",
500            "traceroute",
501            "tracert",
502            "dns cache",
503            "arp table",
504            "route table",
505            "routing table",
506            "default gateway",
507            "power plan",
508            "windows feature",
509            "optional feature",
510            "microsoft store",
511            "app installer",
512            "search index",
513            "windows search",
514            "monitor resolution",
515            "display config",
516            "refresh rate",
517            "sign in",
518            "hard drive",
519            "task scheduler",
520        ],
521    );
522
523    let host_action = lower.split_whitespace().any(|w| {
524        let w = w.trim_matches(|c: char| !c.is_alphanumeric());
525        matches!(
526            w,
527            "inspect"
528                | "count"
529                | "summarize"
530                | "analyze"
531                | "missing"
532                | "ready"
533                | "resolve"
534                | "troubleshoot"
535                | "show"
536                | "find"
537                | "list"
538                | "audit"
539                | "test"
540                | "check"
541                | "currently"
542                | "status"
543                | "stats"
544                | "vitals"
545                | "telemetry"
546                | "looking"
547        )
548    }) || contains_any(lower, &["tell me", "how big", "show me"]);
549
550    // Some words are self-sufficient diagnostic state indicators: asking "is my GPU
551    // throttled?" implicitly asks to inspect whether throttling is happening.
552    let self_sufficient_state =
553        lower.contains("throttl") || lower.contains("overheat") || lower.contains("bottleneck");
554
555    host_scope && (host_action || self_sufficient_state)
556}
557
558pub fn preferred_host_inspection_topic(user_input: &str) -> Option<&'static str> {
559    let lower = user_input.to_lowercase();
560    let asks_fix_plan = (lower.contains("fix")
561        || lower.contains("repair")
562        || lower.contains("resolve")
563        || lower.contains("troubleshoot"))
564        && (lower.contains("cargo")
565            || lower.contains("path")
566            || lower.contains("package manager")
567            || lower.contains("toolchain")
568            || lower.contains("port ")
569            || lower.contains("already in use")
570            || lower.contains("lm studio")
571            || lower.contains("localhost:1234")
572            || lower.contains("embedding model")
573            || lower.contains("no coding model loaded"));
574    let asks_path = lower.contains("path entries")
575        || lower.contains("raw path")
576        || lower.contains("path issue")
577        || lower.contains("missing from path")
578        || lower.contains("not in path")
579        || lower.contains("path variable")
580        || (lower.contains("path") && (lower.contains("show") || lower.contains("what is")));
581    let asks_gpo = lower.contains("gpo")
582        || lower.contains("group polic")
583        || lower.contains("gpresult")
584        || lower.contains("applied policy")
585        || lower.contains("active policies")
586        || lower.contains("what policies")
587        || lower.contains("policy objects")
588        || lower.contains("policy applied")
589        || (lower.contains("policies") && lower.contains("applied"))
590        || (lower.contains("policies") && lower.contains("effect"))
591        || (lower.contains("policy") && lower.contains("in effect"));
592    let asks_certificates = lower.contains("cert")
593        || lower.contains("ssl")
594        || lower.contains("client cert")
595        || lower.contains("expiring cert")
596        || lower.contains("tls certificate")
597        || lower.contains("x509")
598        || lower.contains("x.509")
599        || lower.contains(".pfx")
600        || lower.contains(".p12")
601        || lower.contains(".pem")
602        || lower.contains("pkcs")
603        || lower.contains("trust store")
604        || lower.contains("certificate store")
605        || lower.contains("certificate expir")
606        || lower.contains("untrusted cert")
607        || (lower.contains("tls")
608            && (lower.contains("check")
609                || lower.contains("inspect")
610                || lower.contains("status")
611                || lower.contains("valid")));
612    let asks_integrity = lower.contains("integrity")
613        || lower.contains("sfc")
614        || lower.contains("dism")
615        || lower.contains("corrupt")
616        || lower.contains("os health")
617        || lower.contains("system file")
618        || (lower.contains("windows") && lower.contains("damaged"))
619        || (lower.contains("check") && lower.contains("system") && lower.contains("file"));
620    let asks_user_accounts = (lower.contains("user account")
621        && !lower.contains("user account control"))
622        || lower.contains("local user")
623        || lower.contains("local group")
624        || lower.contains("get-localuser")
625        || lower.contains("get-localgroup")
626        || lower.contains("get-localgroupmember")
627        || lower.contains("who am i")
628        || lower.contains("logged in as")
629        || lower.contains("logged in user")
630        || lower.contains("logged on user")
631        || lower.contains("admin group")
632        || lower.contains("administrators group")
633        || lower.contains("local admin")
634        || lower.contains("who has admin")
635        || lower.contains("running as admin")
636        || lower.contains("is this elevated")
637        || lower.contains("net user")
638        || lower.contains("net localgroup")
639        || lower.contains("who has admin rights")
640        || lower.contains("list all users")
641        || lower.contains("list users")
642        || lower.contains("what accounts")
643        || (lower.contains("accounts") && lower.contains("admin"));
644    let asks_ad_user = lower.contains("ad user")
645        || lower.contains("domain user")
646        || (lower.contains("user") && (lower.contains("sid") || lower.contains("membership")));
647    let asks_mdm = lower.contains("mdm")
648        || lower.contains("intune")
649        || lower.contains("autopilot")
650        || lower.contains("device enrollment")
651        || lower.contains("enrolled in")
652        || lower.contains("mdm enrollment")
653        || lower.contains("device management")
654        || lower.contains("managed device")
655        || lower.contains("azure ad join")
656        || lower.contains("aad join")
657        || (lower.contains("enrolled") && lower.contains("device"))
658        || (lower.contains("enroll") && lower.contains("device"))
659        || (lower.contains("microsoft") && lower.contains("endpoint"));
660    let asks_hyperv = lower.contains("hyper-v")
661        || lower.contains("hyperv")
662        || lower.contains("hyper v")
663        || lower.contains("list vm")
664        || lower.contains("list vms")
665        || lower.contains("running vms")
666        || lower.contains("virtual machines")
667        || lower.contains("virtual machine")
668        || (lower.contains("vm")
669            && !lower.contains("nvme")
670            && (lower.contains("running")
671                || lower.contains("status")
672                || lower.contains("health")
673                || lower.contains("checkpoint")
674                || lower.contains("snapshot")
675                || lower.contains("switch")
676                || lower.contains("memory")
677                || lower.contains("ram")))
678        || lower.contains("vmms")
679        || lower.contains("vmmem");
680    let asks_storage_spaces = lower.contains("storage space")
681        || lower.contains("storage pool")
682        || lower.contains("storage pools")
683        || lower.contains("virtual disk")
684        || lower.contains("virtual disks")
685        || lower.contains("windows raid")
686        || lower.contains("disk pool")
687        || lower.contains("resiliency")
688        || (lower.contains("storage") && lower.contains("pool"))
689        || (lower.contains("mdadm")
690            || lower.contains("software raid")
691            || lower.contains("md array"));
692    let asks_defender_quarantine = lower.contains("quarantine")
693        || lower.contains("threat history")
694        || lower.contains("malware history")
695        || lower.contains("defender history")
696        || lower.contains("detected threat")
697        || lower.contains("detected virus")
698        || lower.contains("defender found")
699        || lower.contains("defender detected")
700        || lower.contains("defender find")
701        || lower.contains("virus found")
702        || lower.contains("threats found")
703        || lower.contains("threat detected")
704        || (lower.contains("defender")
705            && (lower.contains("malware") || lower.contains("virus") || lower.contains("threat")))
706        || (lower.contains("defender")
707            && lower.contains("scan")
708            && (lower.contains("result") || lower.contains("history") || lower.contains("found")));
709    let asks_event_query = lower.contains("event id")
710        || lower.contains("event log query")
711        || lower.contains("event_id")
712        || lower.contains("eventid")
713        || lower.contains("search event")
714        || lower.contains("query event")
715        || lower.contains("find event")
716        || lower.contains("filter event")
717        || (lower.contains("event") && lower.contains("4625"))
718        || (lower.contains("event") && lower.contains("7034"))
719        || (lower.contains("event") && lower.contains("7031"))
720        || (lower.contains("event") && lower.contains("4648"))
721        || (lower.contains("event") && lower.contains("41"))
722        || (lower.contains("event")
723            && (lower.contains("last hour")
724                || lower.contains("last 24")
725                || lower.contains("past hour")
726                || lower.contains("today")))
727        || ((lower.contains("event log")
728            || lower.contains("system log")
729            || lower.contains("application log")
730            || lower.contains("security log"))
731            && (lower.contains("last ")
732                || lower.contains("past ")
733                || lower.contains("today")
734                || lower.contains("hour")
735                || lower.contains("hours"))
736            && (lower.contains("error")
737                || lower.contains("errors")
738                || lower.contains("warning")
739                || lower.contains("warnings")
740                || lower.contains("critical")))
741        || lower.contains("failed logon event")
742        || lower.contains("failed login event")
743        || lower.contains("application error event")
744        || lower.contains("crash event")
745        || lower.contains("service crash event");
746    let asks_ip_config =
747        lower.contains("ipconfig") && (lower.contains("all") || lower.contains("detailed"));
748    let asks_domain = lower.contains("domain")
749        || lower.contains("active directory")
750        || lower.contains("ad join")
751        || lower.contains("workgroup");
752    let asks_device_health = lower.contains("device health")
753        || lower.contains("hardware error")
754        || lower.contains("malfunctioning")
755        || lower.contains("yellow bang")
756        || lower.contains("hardware failing")
757        || lower.contains("device manager")
758        || lower.contains("unknown device")
759        || lower.contains("code 43")
760        || lower.contains("code 10")
761        || lower.contains("code 28")
762        || lower.contains("pnp device")
763        || lower.contains("exclamation mark in device")
764        || (lower.contains("device") && lower.contains("error code"))
765        || (lower.contains("device") && lower.contains("not recognized"))
766        || (lower.contains("device") && lower.contains("broken"))
767        || (lower.contains("device") && lower.contains("not working"))
768        || (lower.contains("device") && lower.contains("stopped working"))
769        || (lower.contains("hardware") && lower.contains("broken"));
770    let asks_drivers =
771        lower.contains("driver") || lower.contains("kmod") || lower.contains("kernel module");
772    let asks_audio = lower.contains("no sound")
773        || lower.contains("audio service")
774        || lower.contains("windows audio")
775        || lower.contains("speaker")
776        || lower.contains("speakers")
777        || lower.contains("microphone")
778        || lower.contains(" mic ")
779        || lower.starts_with("mic ")
780        || lower.contains("mic not")
781        || lower.contains("headset")
782        || lower.contains("headphones")
783        || lower.contains("playback device")
784        || lower.contains("recording device")
785        || lower.contains("audio endpoint")
786        || lower.contains("audioendpointbuilder")
787        || lower.contains("can't hear")
788        || lower.contains("cannot hear")
789        || lower.contains("cant hear")
790        || lower.contains("no audio")
791        || ((lower.contains("audio") || lower.contains("sound"))
792            && (lower.contains("device")
793                || lower.contains("driver")
794                || lower.contains("service")
795                || lower.contains("working")
796                || lower.contains("broken")
797                || lower.contains("input")
798                || lower.contains("output")
799                || lower.contains("crackling")
800                || lower.contains("mute")
801                || lower.contains("muted")
802                || lower.contains("volume")
803                || lower.contains("speaker")
804                || lower.contains("microphone")))
805            && !lower.contains("audio file")
806            && !lower.contains("voice engine");
807    let asks_bluetooth = lower.contains("bluetooth")
808        || lower.contains("pairing")
809        || lower.contains("paired device")
810        || lower.contains("paired devices")
811        || lower.contains("bthserv")
812        || lower.contains("bthavctpsvc")
813        || lower.contains("btagservice")
814        || lower.contains("bluetoothuserservice")
815        || lower.contains("wireless headset")
816        || lower.contains("wireless earbuds")
817        || ((lower.contains("headset") || lower.contains("headphones"))
818            && (lower.contains("disconnect")
819                || lower.contains("pair")
820                || lower.contains("reconnect")
821                || lower.contains("bluetooth")))
822        || ((lower.contains("won't") || lower.contains("cannot") || lower.contains("can't"))
823            && (lower.contains("pair") || lower.contains("connect"))
824            && lower.contains("bluetooth"));
825    let asks_camera = lower.contains("camera")
826        || lower.contains("webcam")
827        || lower.contains("web cam")
828        || (lower.contains("app") && lower.contains("can't see") && lower.contains("camera"))
829        || (lower.contains("camera") && lower.contains("permission"))
830        || (lower.contains("camera") && lower.contains("privacy"))
831        || (lower.contains("camera") && lower.contains("not working"))
832        || (lower.contains("camera") && lower.contains("missing"))
833        || lower.contains("camera_privacy");
834    let asks_sign_in = lower.contains("windows hello")
835        || (lower.contains("hello") && lower.contains("not working"))
836        || (lower.contains("pin")
837            && (lower.contains("broken")
838                || lower.contains("not working")
839                || lower.contains("forgot")))
840        || (lower.contains("can't sign in")
841            || lower.contains("cannot sign in")
842            || lower.contains("cant sign in"))
843        || (lower.contains("sign") && lower.contains("in") && lower.contains("issue"))
844        || lower.contains("logon failure")
845        || lower.contains("credential provider")
846        || lower.contains("biometric service")
847        || (lower.contains("profile") && lower.contains("corrupt"))
848        || lower.contains("wbiosrvc")
849        || lower.contains("login screen stuck")
850        || lower.contains("stuck on login")
851        || lower.contains("stuck at login")
852        || lower.contains("login loop")
853        || lower.contains("sign-in loop")
854        || lower.contains("sign in loop")
855        || lower.contains("reboot loop on login")
856        || (lower.contains("can't log in") && !lower.contains("vpn") && !lower.contains("ssh"))
857        || (lower.contains("cannot log in") && !lower.contains("vpn") && !lower.contains("ssh"))
858        || lower.contains("can't login")
859        || lower.contains("cannot login")
860        || lower.contains("cant login")
861        || lower.contains("login failed")
862        || lower.contains("login failure")
863        || (lower.contains("login") && lower.contains("not working"))
864        || (lower.contains("login") && lower.contains("problem"))
865        || (lower.contains("login") && lower.contains("status"))
866        || lower.contains("sign-in status")
867        || lower.contains("sign in status");
868    let asks_identity_auth = lower.contains("web account manager")
869        || lower.contains("token broker")
870        || lower.contains("tokenbroker")
871        || lower.contains("aad broker")
872        || lower.contains("broker plugin")
873        || lower.contains("identity broker")
874        || lower.contains("microsoft 365 sign-in")
875        || lower.contains("microsoft 365 signin")
876        || lower.contains("office sign-in")
877        || lower.contains("office signin")
878        || lower.contains("workplace join")
879        || lower.contains("device registration")
880        || lower.contains("device registered")
881        || lower.contains("entra")
882        || lower.contains("azure ad")
883        || lower.contains("azuread")
884        || lower.contains("azure ad prt")
885        || lower.contains("azureadprt")
886        || lower.contains("wamdefaultset")
887        || lower.contains("single sign-on")
888        || lower.contains("organizational account")
889        || lower.contains("corporate account")
890        || (lower.contains("azure") && lower.contains("registered"))
891        || ((lower.contains("outlook")
892            || lower.contains("teams")
893            || lower.contains("onedrive")
894            || lower.contains("office")
895            || lower.contains("microsoft 365"))
896            && (lower.contains("sign in")
897                || lower.contains("signin")
898                || lower.contains("signed in")
899                || lower.contains("signed out")
900                || lower.contains("keeps asking")
901                || lower.contains("keep asking")
902                || lower.contains("authentication")
903                || lower.contains("auth")
904                || lower.contains("token")
905                || lower.contains("work account")
906                || lower.contains("school account")
907                || lower.contains("account mismatch")));
908    let asks_installer_health = lower.contains("installer health")
909        || lower.contains("installer broken")
910        || lower.contains("msiexec")
911        || lower.contains("msi installer")
912        || lower.contains("windows installer")
913        || lower.contains("app installer")
914        || lower.contains("desktopappinstaller")
915        || lower.contains("microsoft store")
916        || lower.contains("winget broken")
917        || (lower.contains("can't install")
918            && (lower.contains("app") || lower.contains("apps") || lower.contains("program")))
919        || (lower.contains("cannot install")
920            && (lower.contains("app") || lower.contains("apps") || lower.contains("program")))
921        || (lower.contains("cant install")
922            && (lower.contains("app") || lower.contains("apps") || lower.contains("program")))
923        || (lower.contains("unable to install")
924            && (lower.contains("app") || lower.contains("program") || lower.contains("software")))
925        || ((lower.contains("install") || lower.contains("installer"))
926            && (lower.contains("fail")
927                || lower.contains("failing")
928                || lower.contains("broken")
929                || lower.contains("stuck")
930                || lower.contains("hanging")
931                || lower.contains("error"))
932            && !lower.contains("windows update"));
933    let asks_onedrive = lower.contains("onedrive")
934        || lower.contains("one drive")
935        || lower.contains("files on-demand")
936        || lower.contains("known folder backup")
937        || lower.contains("known folder move")
938        || lower.contains("kfm")
939        || lower.contains("sharepoint sync")
940        || lower.contains("sync root")
941        || ((lower.contains("desktop")
942            || lower.contains("documents")
943            || lower.contains("pictures"))
944            && lower.contains("backup")
945            && (lower.contains("onedrive") || lower.contains("cloud") || lower.contains("sync")))
946        || ((lower.contains("desktop")
947            || lower.contains("documents")
948            || lower.contains("pictures"))
949            && lower.contains("sync")
950            && (lower.contains("onedrive")
951                || lower.contains("sharepoint")
952                || lower.contains("cloud")));
953    let asks_browser_health = lower.contains("browser health")
954        || lower.contains("webview2")
955        || lower.contains("default browser")
956        || ((lower.contains("browser")
957            || lower.contains("chrome")
958            || lower.contains("edge")
959            || lower.contains("firefox"))
960            && (lower.contains("slow")
961                || lower.contains("sluggish")
962                || lower.contains("lag")
963                || lower.contains("crash")
964                || lower.contains("crashing")
965                || lower.contains("hang")
966                || lower.contains("frozen")
967                || lower.contains("freeze")
968                || lower.contains("broken")
969                || lower.contains("not opening")
970                || lower.contains("won't open")
971                || lower.contains("cannot open")
972                || lower.contains("unresponsive")
973                || lower.contains("not starting")
974                || lower.contains("not loading")
975                || lower.contains("extension")
976                || lower.contains("extensions")
977                || lower.contains("proxy")
978                || lower.contains("policy")))
979        || ((lower.contains("links") || lower.contains("link"))
980            && (lower.contains("open wrong")
981                || lower.contains("opens wrong")
982                || lower.contains("wrong browser")
983                || lower.contains("wrong app")
984                || lower.contains("default browser")))
985        || ((lower.contains("website") || lower.contains("websites") || lower.contains("web app"))
986            && (lower.contains("browser")
987                || lower.contains("chrome")
988                || lower.contains("edge")
989                || lower.contains("firefox"))
990            && (lower.contains("load")
991                || lower.contains("broken")
992                || lower.contains("slow")
993                || lower.contains("proxy")
994                || lower.contains("policy")));
995    let asks_outlook = lower.contains("outlook")
996        || lower.contains("ms outlook")
997        || lower.contains("microsoft outlook")
998        || (lower.contains("ost") && lower.contains("mail"))
999        || (lower.contains("pst") && lower.contains("mail"))
1000        || (lower.contains("add-in") && lower.contains("mail"))
1001        || (lower.contains("addin") && lower.contains("outlook"))
1002        || (lower.contains("email client")
1003            && (lower.contains("slow")
1004                || lower.contains("crash")
1005                || lower.contains("broken")
1006                || lower.contains("hanging")))
1007        || (lower.contains("mail profile") && lower.contains("corrupt"));
1008    let not_nic_teaming = !lower.contains("nic teaming")
1009        && !lower.contains("nic-teaming")
1010        && !lower.contains("link aggregation")
1011        && !lower.contains("lbfo");
1012    let asks_teams = (lower.contains("teams") && not_nic_teaming)
1013        || lower.contains("ms teams")
1014        || lower.contains("microsoft teams")
1015        || (lower.contains("teams cache") && lower.contains("clear"))
1016        || (lower.contains("teams")
1017            && not_nic_teaming
1018            && lower.contains("sign-in")
1019            && lower.contains("broken"))
1020        || (lower.contains("teams")
1021            && not_nic_teaming
1022            && lower.contains("device")
1023            && (lower.contains("audio")
1024                || lower.contains("video")
1025                || lower.contains("camera")
1026                || lower.contains("microphone")));
1027    let asks_windows_backup = lower.contains("file history")
1028        || lower.contains("windows backup")
1029        || lower.contains("wbadmin")
1030        || lower.contains("system restore")
1031        || lower.contains("restore point")
1032        || lower.contains("restore points")
1033        || lower.contains("backed up")
1034        || lower.contains("being backed")
1035        || (lower.contains("backup")
1036            && !lower.contains("onedrive")
1037            && !lower.contains("one drive")
1038            && (lower.contains("backup drive")
1039                || lower.contains("backup disk")
1040                || lower.contains("configured")
1041                || lower.contains("schedule")
1042                || lower.contains("last backup")
1043                || lower.contains("backup health")
1044                || lower.contains("backup status")
1045                || lower.contains("backup running")
1046                || lower.contains("broken")
1047                || lower.contains("failed")
1048                || lower.contains("running")
1049                || lower.contains("enabled")
1050                || lower.contains("working")
1051                || lower.contains("set up")))
1052        || (lower.contains("recovery")
1053            && (lower.contains("backup")
1054                || lower.contains("restore")
1055                || lower.contains("posture")))
1056        || lower.contains("known folder move")
1057        || lower.contains("known folder backup");
1058    let asks_search_index = (lower.contains("search")
1059        && (lower.contains("broken")
1060            || lower.contains("not working")
1061            || lower.contains("slow")
1062            || lower.contains("indexing")
1063            || lower.contains("index")))
1064        || lower.contains("wsearch")
1065        || lower.contains("windows search")
1066        || lower.contains("search index")
1067        || lower.contains("indexer")
1068        || (lower.contains("search") && lower.contains("stuck"))
1069        || (lower.contains("search") && lower.contains("results") && lower.contains("show"));
1070    let asks_display_config = lower.contains("monitor")
1071        || lower.contains("display")
1072        || lower.contains("resolution")
1073        || lower.contains("refresh rate")
1074        || lower.contains("refresh hz")
1075        || lower.contains("screen config")
1076        || lower.contains("dpi")
1077        || lower.contains("scaling")
1078        || lower.contains("hdmi")
1079        || lower.contains("displayport")
1080        || lower.contains("how many screens")
1081        || lower.contains("multi-monitor")
1082        || lower.contains("second screen")
1083        || lower.contains("external display");
1084    let asks_data_audit = lower.contains("data audit")
1085        || lower.contains("audit data")
1086        || lower.contains("csv schema")
1087        || lower.contains("data schema")
1088        || lower.contains("inspect file")
1089        || lower.contains("profile data")
1090        || lower.contains("data distribution")
1091        || (lower.contains("audit")
1092            && (lower.contains("csv")
1093                || lower.contains("json")
1094                || lower.contains("file")
1095                || lower.contains("data")))
1096        || (lower.contains("schema")
1097            && (lower.contains("csv") || lower.contains("json") || lower.contains("data")));
1098    let asks_ntp = lower.contains("ntp")
1099        || lower.contains("time sync")
1100        || lower.contains("clock sync")
1101        || lower.contains("w32tm")
1102        || lower.contains("clock drift")
1103        || lower.contains("system clock")
1104        || lower.contains("time server")
1105        || lower.contains("time zone")
1106        || lower.contains("timezone")
1107        || lower.contains("wrong timezone")
1108        || (lower.contains("time") && lower.contains("drift"))
1109        || (lower.contains("clock") && lower.contains("wrong"))
1110        || (lower.contains("time") && lower.contains("wrong"))
1111        || (lower.contains("clock") && lower.contains("off"))
1112        || (lower.contains("time") && lower.contains("off") && lower.contains("sync"))
1113        || lower.contains("system time")
1114        || (lower.contains("time") && lower.contains("accurate"))
1115        || (lower.contains("time") && lower.contains("correct"));
1116    let asks_cpu_power = lower.contains("turbo boost")
1117        || lower.contains("cpu frequency")
1118        || lower.contains("cpu freq")
1119        || lower.contains("processor frequency")
1120        || lower.contains("cpu clock")
1121        || lower.contains("cpu speed")
1122        || lower.contains("processor speed")
1123        || lower.contains("cpu stuck")
1124        || lower.contains("cpu slow")
1125        || lower.contains("power plan")
1126        || lower.contains("cpu power")
1127        || lower.contains("processor state")
1128        || (lower.contains("cpu") && lower.contains("slow"))
1129        || (lower.contains("cpu") && lower.contains("underclocking"))
1130        || (lower.contains("boost") && lower.contains("disabled"))
1131        || (lower.contains("processor") && lower.contains("slow"))
1132        || (lower.contains("processor") && lower.contains("running slow"))
1133        || lower.contains("processor running at");
1134    let asks_credentials = lower.contains("credential manager")
1135        || lower.contains("credential store")
1136        || lower.contains("saved password")
1137        || lower.contains("stored credential")
1138        || lower.contains("saved credential")
1139        || lower.contains("credential vault")
1140        || lower.contains("cmdkey")
1141        || (lower.contains("credential") && lower.contains("list"))
1142        || (lower.contains("password") && lower.contains("vault"))
1143        || (lower.contains("windows") && lower.contains("credential"))
1144        || (lower.contains("credential")
1145            && (lower.contains("clear")
1146                || lower.contains("cached")
1147                || lower.contains("view")
1148                || lower.contains("delete")
1149                || lower.contains("remove")));
1150    let asks_tpm = lower.contains("tpm")
1151        || lower.contains("secure boot")
1152        || lower.contains("secureboot")
1153        || lower.contains("trusted platform module")
1154        || lower.contains("firmware security")
1155        || lower.contains("uefi security")
1156        || lower.contains("uefi mode")
1157        || lower.contains("uefi enabled")
1158        || lower.contains("uefi settings")
1159        || lower.contains("legacy bios")
1160        || lower.contains("uefi bios")
1161        || (lower.contains("uefi")
1162            && (lower.contains("boot")
1163                || lower.contains("secure")
1164                || lower.contains("status")
1165                || lower.contains("check")))
1166        || (lower.contains("bitlocker") && lower.contains("chip"))
1167        || (lower.contains("windows 11") && lower.contains("tpm"));
1168    let asks_dhcp = lower.contains("dhcp lease")
1169        || lower.contains("lease expires")
1170        || lower.contains("lease obtained")
1171        || lower.contains("dhcp server")
1172        || lower.contains("ip lease")
1173        || lower.contains("lease time")
1174        || lower.contains("lease renew")
1175        || lower.contains("renew lease")
1176        || (lower.contains("dhcp")
1177            && (lower.contains("detail")
1178                || lower.contains("info")
1179                || lower.contains("check")
1180                || lower.contains("show")))
1181        || (lower.contains("ip") && lower.contains("lease"));
1182    let asks_mtu = lower.contains("mtu")
1183        || lower.contains("path mtu")
1184        || lower.contains("pmtu")
1185        || lower.contains("jumbo frame") && lower.contains("test")
1186        || lower.contains("frame size")
1187        || lower.contains("mtu discovery")
1188        || lower.contains("fragmentation")
1189        || (lower.contains("packet") && lower.contains("size") && lower.contains("max"))
1190        || (lower.contains("vpn") && lower.contains("mtu"))
1191        || (lower.contains("mtu") && lower.contains("check"));
1192    let asks_latency = (lower
1193        .split_whitespace()
1194        .any(|w| w.trim_matches(|c: char| !c.is_alphanumeric()) == "ping"))
1195        || lower.contains("latency")
1196        || lower.contains("packet loss")
1197        || lower.contains("rtt")
1198        || lower.contains("round trip")
1199        || lower.contains("reachability")
1200        || lower.contains("ping test")
1201        || (lower.contains("network") && lower.contains("slow"))
1202        || (lower.contains("internet") && lower.contains("slow"))
1203        || (lower.contains("connection") && lower.contains("slow"))
1204        || (lower.contains("high") && lower.contains("latency"))
1205        || lower.contains("network lag")
1206        || lower.contains("jitter");
1207    let asks_network_adapter = lower.contains("nic settings")
1208        || lower.contains("nic offload")
1209        || lower.contains("adapter settings")
1210        || lower.contains("adapter offload")
1211        || lower.contains("jumbo frame")
1212        || lower.contains("rss setting")
1213        || lower.contains("tcp offload")
1214        || lower.contains("lso")
1215        || lower.contains("checksum offload")
1216        || lower.contains("wake on lan")
1217        || lower.contains("wake-on-lan")
1218        || lower.contains("wol")
1219        || lower.contains("nic advanced")
1220        || lower.contains("adapter error")
1221        || lower.contains("duplex mismatch")
1222        || lower.contains("ethernet not")
1223        || lower.contains("ethernet port")
1224        || lower.contains("ethernet cable")
1225        || lower.contains("wired connection not")
1226        || lower.contains("wired network not")
1227        || lower.contains("nic not working")
1228        || lower.contains("network adapter not")
1229        || lower.contains("network card")
1230        || lower.contains("link speed")
1231        || lower.contains("network adapter settings")
1232        || (lower.contains("nic")
1233            && (lower.contains("driver")
1234                || lower.contains("setting")
1235                || lower.contains("error")
1236                || lower.contains("config")));
1237    let asks_ipv6 = lower.contains("ipv6")
1238        || lower.contains("slaac")
1239        || lower.contains("dhcpv6")
1240        || lower.contains("ipv6 address")
1241        || lower.contains("ipv6 prefix")
1242        || lower.contains("ipv6 gateway")
1243        || lower.contains("ipv6 config")
1244        || lower.contains("privacy extension")
1245        || lower.contains("global unicast")
1246        || lower.contains("link-local address")
1247        || (lower.contains("ipv6")
1248            && (lower.contains("check") || lower.contains("show") || lower.contains("status")));
1249    let asks_tcp_params = lower.contains("tcp autotuning")
1250        || lower.contains("tcp auto-tuning")
1251        || lower.contains("tcp window scaling")
1252        || lower.contains("tcp congestion")
1253        || lower.contains("congestion algorithm")
1254        || lower.contains("congestion provider")
1255        || lower.contains("tcp settings")
1256        || lower.contains("tcp parameters")
1257        || lower.contains("tcp tuning")
1258        || lower.contains("tcp chimney")
1259        || lower.contains("tcp offload")
1260        || lower.contains("ecn")
1261        || lower.contains("rwin")
1262        || lower.contains("receive window")
1263        || lower.contains("dynamic port range")
1264        || lower.contains("tcp window")
1265        || (lower.contains("tcp")
1266            && (lower.contains("slow")
1267                || lower.contains("throughput")
1268                || lower.contains("performance")
1269                || lower.contains("config")
1270                || lower.contains("speed")
1271                || lower.contains("window size")));
1272    let asks_wlan_profiles = lower.contains("saved wifi")
1273        || lower.contains("saved wireless")
1274        || lower.contains("wifi profile")
1275        || lower.contains("wlan profile")
1276        || lower.contains("wireless profile")
1277        || lower.contains("saved network")
1278        || lower.contains("known network")
1279        || lower.contains("netsh wlan")
1280        || (lower.contains("wifi")
1281            && (lower.contains("security")
1282                || lower.contains("audit")
1283                || lower.contains("wep")
1284                || lower.contains("saved")
1285                || lower.contains("remember")
1286                || lower.contains("password")))
1287        || (lower.contains("wireless")
1288            && (lower.contains("profile")
1289                || lower.contains("saved")
1290                || lower.contains("audit")
1291                || lower.contains("remember")));
1292    let asks_ipsec = lower.contains("ipsec")
1293        || lower.contains("ip sec")
1294        || lower.contains("ipsec sa")
1295        || lower.contains("security association")
1296        || lower.contains("ike ")
1297        || lower.contains("ikev2")
1298        || lower.contains("ike tunnel")
1299        || lower.contains("ipsec tunnel")
1300        || lower.contains("ipsec policy")
1301        || lower.contains("ipsec rule")
1302        || lower.contains("policy agent")
1303        || lower.contains("xfrm")
1304        || (lower.contains("ipsec")
1305            && (lower.contains("check") || lower.contains("active") || lower.contains("status")));
1306    let asks_netbios = lower.contains("netbios")
1307        || lower.contains("nbtstat")
1308        || lower.contains("wins server")
1309        || lower.contains("wins address")
1310        || lower.contains("netbios name")
1311        || lower.contains("netbios over tcp")
1312        || lower.contains("nbns")
1313        || (lower.contains("wins")
1314            && (lower.contains("server") || lower.contains("config") || lower.contains("check")));
1315    let asks_nic_teaming = lower.contains("nic team")
1316        || lower.contains("nic teaming")
1317        || lower.contains("network team")
1318        || lower.contains("lacp")
1319        || lower.contains("link aggregation")
1320        || lower.contains("bonding")
1321        || lower.contains("bond interface")
1322        || lower.contains("lbfo")
1323        || (lower.contains("team")
1324            && (lower.contains("nic") || lower.contains("adapter") || lower.contains("network")))
1325        || (lower.contains("bond")
1326            && (lower.contains("adapter") || lower.contains("interface") || lower.contains("nic")));
1327    let asks_snmp = lower.contains("snmp")
1328        || lower.contains("snmp agent")
1329        || lower.contains("snmp trap")
1330        || lower.contains("community string")
1331        || lower.contains("community name")
1332        || lower.contains("snmp service")
1333        || lower.contains("snmpd");
1334    let asks_port_test = lower.contains("port test")
1335        || lower.contains("test port")
1336        || lower.contains("port check")
1337        || lower.contains("check port")
1338        || lower.contains("port reachab")
1339        || lower.contains("can i reach")
1340        || lower.contains("is port")
1341        || lower.contains("tcp test")
1342        || lower.contains("test-netconnection")
1343        || lower.contains("test connection")
1344        || (lower.contains("port")
1345            && (lower.contains("open")
1346                || lower.contains("closed")
1347                || lower.contains("blocked")
1348                || lower.contains("reachable")))
1349        || (lower.contains("reach") && lower.contains("port"));
1350    let asks_network_profile = lower.contains("network profile")
1351        || lower.contains("network location")
1352        || lower.contains("network category")
1353        || lower.contains("public network")
1354        || lower.contains("private network")
1355        || lower.contains("domain network")
1356        || lower.contains("net profile")
1357        || (lower.contains("network") && lower.contains("location"))
1358        || (lower.contains("firewall") && lower.contains("profile") && lower.contains("network"));
1359    let asks_dns_lookup = lower.contains("dns lookup")
1360        || lower.contains("dns record")
1361        || lower.contains("nslookup")
1362        || lower.contains("resolve-dnsname")
1363        || lower.contains("gethostaddresses")
1364        || lower.contains("gethostentry")
1365        || lower.contains("[system.net.dns]")
1366        || lower.contains("net.dns]")
1367        || lower.contains("look up ")
1368        || lower.contains("look up the")
1369        || lower.contains("resolve ")
1370        || lower.contains("mx record")
1371        || lower.contains("srv record")
1372        || lower.contains("txt record")
1373        || lower.contains("a record")
1374        || lower.contains("aaaa record")
1375        || lower.contains("cname record")
1376        || lower.contains(" dig ")
1377        || lower.starts_with("host ")
1378        || (lower.contains("what") && lower.contains("ip") && lower.contains("for"))
1379        || (lower.contains("ip address") && lower.contains(" of "))
1380        || (lower.contains("resolve")
1381            && (lower.contains("hostname") || lower.contains("domain") || lower.contains("name")))
1382        || (lower.contains("lookup")
1383            && (lower.contains("domain") || lower.contains("host") || lower.contains("name")));
1384    let asks_peripherals = lower.contains("peripheral")
1385        || lower.contains("usb")
1386        || lower.contains("keyboard")
1387        || lower.contains("mouse")
1388        || lower.contains("pointer")
1389        || lower.contains("monitor")
1390        || lower.contains("input device")
1391        || lower.contains("connected hardware");
1392    let asks_sessions = lower.contains("session")
1393        || lower.contains("login")
1394        || lower.contains("who is on")
1395        || lower.contains("active user")
1396        || lower.contains("who is logged on")
1397        || lower.contains("who is logged in")
1398        || lower.contains("logged on users")
1399        || lower.contains("logged in users")
1400        || lower.contains("current users")
1401        || lower.contains("query session")
1402        || lower.contains("qwinsta")
1403        || lower.contains("connected users")
1404        || lower.contains("user sessions")
1405        || lower.contains("terminal session")
1406        || (lower.contains("who") && lower.contains("logged"))
1407        || (lower.contains("who")
1408            && lower.contains("using")
1409            && (lower.contains("computer") || lower.contains("machine")));
1410    let asks_virtualization = lower.contains("virtualization")
1411        || lower.contains("hypervisor")
1412        || lower.contains("vt-x")
1413        || lower.contains("slat")
1414        || lower.contains("v-p")
1415        || lower.contains("nested virt")
1416        || lower.contains("cpu model")
1417        || lower.contains("ram size")
1418        || lower.contains("hardware spec")
1419        || lower.contains("hardware dna")
1420        || lower.contains("hardware info")
1421        || lower.contains("bios version")
1422        || lower.contains("motherboard")
1423        || lower.contains("how much ram")
1424        || lower.contains("what processor")
1425        || lower.contains("what cpu")
1426        || (lower.contains("what hardware") && lower.contains("have"))
1427        || (lower.contains("hardware") && lower.contains("inventory"));
1428    let asks_startup = lower.contains("startup")
1429        || lower.contains("boot program")
1430        || lower.contains("autorun")
1431        || lower.contains("run at boot")
1432        || lower.contains("startup program")
1433        || lower.contains("startup app")
1434        || lower.contains("startup list")
1435        || lower.contains("startup item")
1436        || lower.contains("starts with windows")
1437        || lower.contains("start with windows")
1438        || lower.contains("launch at startup")
1439        || lower.contains("launch on startup")
1440        || lower.contains("open at startup")
1441        || lower.contains("open on boot")
1442        || lower.contains("runs on boot")
1443        || lower.contains("run at login")
1444        || lower.contains("msconfig")
1445        || lower.contains("login item")
1446        || lower.contains("autostart")
1447        || (lower.contains("disable") && lower.contains("startup"))
1448        || (lower.contains("what") && lower.contains("start") && lower.contains("boot"))
1449        || (lower.contains("load") && lower.contains("boot"))
1450        || (lower.contains("load") && lower.contains("startup") && !lower.contains("reload"));
1451    let asks_env_doctor = lower.contains("env doctor")
1452        || lower.contains("environment doctor")
1453        || lower.contains("package manager")
1454        || lower.contains("package managers")
1455        || lower.contains("shims")
1456        || lower.contains("path drift")
1457        || lower.contains("environment is broken")
1458        || lower.contains("env is broken")
1459        || (lower.contains("dev machine") && lower.contains("off"))
1460        || (lower.contains("environment") && lower.contains("sane"));
1461    let asks_lan_discovery = lower.contains("upnp")
1462        || lower.contains("ssdp")
1463        || lower.contains("mdns")
1464        || lower.contains("bonjour")
1465        || lower.contains("llmnr")
1466        || lower.contains("network neighborhood")
1467        || lower.contains("device discovery")
1468        || lower.contains("local discovery")
1469        || lower.contains("discover local devices")
1470        || lower.contains("discover devices")
1471        || lower.contains("browse computers")
1472        || (lower.contains("local network")
1473            && (lower.contains("discover")
1474                || lower.contains("discovery")
1475                || lower.contains("neighborhood")
1476                || lower.contains("device")
1477                || lower.contains("devices")
1478                || lower.contains("aware of")))
1479        || ((lower.contains("netbios") || lower.contains("smb visibility"))
1480            && !lower.contains("active directory"))
1481        || ((lower.contains("nas")
1482            || lower.contains("printer")
1483            || lower.contains("device")
1484            || lower.contains("computer")
1485            || lower.contains("pc"))
1486            && ((lower.contains("can't") && lower.contains("see"))
1487                || (lower.contains("cannot") && lower.contains("see"))
1488                || (lower.contains("cant") && lower.contains("see"))
1489                || lower.contains("can't see")
1490                || lower.contains("cannot see")
1491                || lower.contains("cant see")
1492                || lower.contains("not visible")
1493                || lower.contains("not showing up")
1494                || lower.contains("not show up")
1495                || lower.contains("discover"))
1496            && (lower.contains("network")
1497                || lower.contains("lan")
1498                || lower.contains("local")
1499                || lower.contains("neighborhood")));
1500    let asks_network = (((lower.contains("network") && !lower.contains("active directory"))
1501        && !lower.contains("stat")
1502        && !lower.contains("share")
1503        && !lower.contains("throughput"))
1504        || lower.contains("adapter")
1505        || lower.contains("ip address")
1506        || lower.contains("ipconfig")
1507        || lower.contains("ipv4")
1508        || lower.contains("ipv6")
1509        || lower.contains("subnet")
1510        || lower.contains("dns server")
1511        || lower.contains("nameserver")
1512        || lower.contains("wifi")
1513        || lower.contains("wireless")
1514        || lower.contains("ethernet")
1515        || lower.contains("lan"))
1516        && !asks_ad_user;
1517    let asks_services = lower.contains("service")
1518        || lower.contains("services")
1519        || lower.contains("daemon")
1520        || lower.contains("startup type")
1521        || lower.contains("background service")
1522        || lower.contains("windows service")
1523        || lower.contains("systemctl")
1524        || lower.contains("get-service");
1525    let asks_processes = lower.contains("process")
1526        || lower.contains("processes")
1527        || lower.contains("task manager")
1528        || lower.contains("what is running")
1529        || lower.contains("what's running")
1530        || lower.contains("using my ram")
1531        || lower.contains("using ram")
1532        || lower.contains("using my cpu")
1533        || lower.contains("using the cpu")
1534        || lower.contains("top memory")
1535        || lower.contains("top ram")
1536        || lower.contains("high memory")
1537        || lower.contains("resource-heavy processes")
1538        || lower.contains("heavy hitters")
1539        || lower.contains("cpu hog")
1540        || lower.contains("memory hog")
1541        || lower.contains("ram hog")
1542        || lower.contains("hogging cpu")
1543        || lower.contains("hogging ram")
1544        || lower.contains("hogging memory")
1545        || lower.contains("eating up cpu")
1546        || lower.contains("eating up ram")
1547        || lower.contains("eating up memory")
1548        || lower.contains("eating my cpu")
1549        || lower.contains("eating my ram")
1550        || lower.contains("eating my memory")
1551        || (lower.contains("hogging")
1552            && (lower.contains("cpu") || lower.contains("ram") || lower.contains("memory")))
1553        || (lower.contains("eating up")
1554            && (lower.contains("cpu") || lower.contains("ram") || lower.contains("memory")))
1555        || (lower.contains("using the most")
1556            && (lower.contains("cpu") || lower.contains("ram") || lower.contains("memory")))
1557        || (lower.contains("most cpu")
1558            || lower.contains("most ram")
1559            || lower.contains("most memory"))
1560        || (lower.contains("hitting")
1561            && (lower.contains("cpu") || lower.contains("ram") || lower.contains("disk")));
1562    let asks_toolchains = lower.contains("developer tools")
1563        || lower.contains("toolchains")
1564        || lower.contains("toolchain not found")
1565        || lower.contains("toolchain missing")
1566        || (lower.contains("installed") && lower.contains("version"))
1567        || (lower.contains("detect") && lower.contains("version"));
1568    let asks_permissions = lower.contains("permission")
1569        || lower.contains("access control")
1570        || lower.contains("get-acl")
1571        || lower.contains("acl ")
1572        || lower.contains("icacls")
1573        || lower.contains("takeown")
1574        || lower.contains("ntfs permission")
1575        || (lower.contains("who has") && lower.contains("access"));
1576    let asks_login_history = lower.contains("login history")
1577        || lower.contains("logon history")
1578        || lower.contains("who logged in")
1579        || lower.contains("recent logon")
1580        || lower.contains("failed logon")
1581        || lower.contains("event id 4624")
1582        || lower.contains("eventid 4624");
1583    let asks_registry_audit = lower.contains("registry audit")
1584        || lower.contains("persistence")
1585        || lower.contains("debugger hijack")
1586        || lower.contains("ifeo")
1587        || lower.contains("winlogon shell")
1588        || lower.contains("bootexecute")
1589        || lower.contains("reg query")
1590        || lower.contains("regedit")
1591        || lower.contains("sticky keys")
1592        || lower.contains("sethc.exe");
1593    let asks_share_access = lower.contains("share access")
1594        || lower.contains("unc path")
1595        || lower.contains("smbshare")
1596        || lower.contains("net share")
1597        || lower.contains("net use")
1598        || lower.contains("\\\\")
1599        || lower.contains("share is reachable")
1600        || lower.contains("reachable share")
1601        || (lower.contains("network share")
1602            && (lower.contains("reach") || lower.contains("access") || lower.contains("test")))
1603        || (lower.contains("network drive") && !lower.contains("network drives"))
1604        || lower.contains("mapped drive")
1605        || lower.contains("shared folder");
1606    let asks_thermal = lower.contains("thermal")
1607        || (lower.contains("throttl") && !lower.contains("gpu"))
1608        || lower.contains("overheat")
1609        || lower.contains("too hot")
1610        || lower.contains("running hot")
1611        || lower.contains("laptop hot")
1612        || lower.contains("pc getting hot")
1613        || lower.contains("getting hot")
1614        || lower.contains("temperature high")
1615        || lower.contains("cpu temp")
1616        || lower.contains("cpu temperature")
1617        || lower.contains("temp sensor")
1618        || lower.contains("check temps")
1619        || lower.contains("fan loud")
1620        || lower.contains("fan noise")
1621        || lower.contains("fan running")
1622        || lower.contains("fans running")
1623        || lower.contains("fan spinning")
1624        || lower.contains("fans spinning")
1625        || lower.contains("loud fan")
1626        || lower.contains("fan always on")
1627        || (lower.contains("fan") && lower.contains("always on"))
1628        || lower.contains("fan constantly")
1629        || lower.contains("fan at max")
1630        || lower.contains("fan at 100")
1631        || (lower.contains("temperature")
1632            && (lower.contains("cpu")
1633                || lower.contains("gpu")
1634                || lower.contains("system")
1635                || lower.contains("sensor")
1636                || lower.contains("check")
1637                || lower.contains("monitor")));
1638    let asks_overclocker = lower.contains("overclocker")
1639        || lower.contains("nvidia stats")
1640        || lower.contains("silicon health")
1641        || lower.contains("mhz")
1642        || ((lower.contains("voltage") || lower.contains("volts"))
1643            && (lower.contains("gpu")
1644                || lower.contains("cpu")
1645                || lower.contains("nvidia")
1646                || lower.contains("silicon")))
1647        || (lower.contains("gpu")
1648            && (lower.contains("throttl")
1649                || lower.contains("bottleneck")
1650                || lower.contains("clock")
1651                || lower.contains("fan")
1652                || lower.contains("power draw")
1653                || lower.contains("frequency")
1654                || lower.contains("overheating")
1655                || lower.contains("usage")
1656                || lower.contains("utilization")
1657                || lower.contains("performance")));
1658    let asks_hardware = lower.contains("cpu model")
1659        || lower.contains("ram size")
1660        || lower.contains("hardware spec")
1661        || (lower.contains("what hardware") && lower.contains("have"))
1662        || (lower.contains("gpu") && (lower.contains("what") || lower.contains("show")))
1663        || lower.contains("motherboard")
1664        || lower.contains("bios version")
1665        || lower.contains("graphics card")
1666        || lower.contains("video card")
1667        || lower.contains("system information")
1668        || lower.contains("system info")
1669        || lower.contains("system specs")
1670        || lower.contains("computer spec")
1671        || lower.contains("display adapter")
1672        || (lower.contains("how much") && lower.contains("ram"))
1673        || (lower.contains("what") && lower.contains("processor"))
1674        || (lower.contains("what") && lower.contains("cpu") && !lower.contains("using"));
1675    let asks_activation = lower.contains("activation")
1676        || lower.contains("activated")
1677        || lower.contains("not activated")
1678        || lower.contains("product key")
1679        || lower.contains("license expired")
1680        || lower.contains("slmgr")
1681        || lower.contains("license status")
1682        || lower.contains("is windows genuine")
1683        || lower.contains("licensed")
1684        || lower.contains("unlicensed")
1685        || (lower.contains("activate") && lower.contains("window"));
1686    let asks_patch_history = lower.contains("patch history")
1687        || lower.contains("hotfix")
1688        || lower.contains("kb history")
1689        || lower.contains("installed updates")
1690        || lower.contains("security patch")
1691        || (lower.contains("update") && lower.contains("applied"));
1692    let asks_ports = lower.contains("listening on port")
1693        || lower.contains("listening port")
1694        || lower.contains("open port")
1695        || lower.contains("port 3000")
1696        || lower.contains("listening on ")
1697        || lower.contains("what ports are")
1698        || lower.contains("what port is")
1699        || lower.contains("exposed")
1700        || lower.contains("what is listening")
1701        || (lower.contains("listening") && lower.contains("port"));
1702    let asks_repo_doctor = lower.contains("repo doctor")
1703        || lower.contains("repository doctor")
1704        || lower.contains("workspace health")
1705        || lower.contains("repo health")
1706        || lower.contains("workspace sanity")
1707        || (lower.contains("git state")
1708            && (lower.contains("release artifacts")
1709                || lower.contains("build markers")
1710                || lower.contains("hematite memory")));
1711    let asks_directory = lower.contains("directory")
1712        || lower.contains("folder")
1713        || lower.contains("how big")
1714        || lower.contains("biggest");
1715
1716    let asks_mutation_intent = (lower.contains("make")
1717        || lower.contains("create")
1718        || lower.contains("mkdir")
1719        || lower.contains("organize")
1720        || lower.contains("edit")
1721        || lower.contains("write")
1722        || lower.contains("save")
1723        || lower.contains("update")
1724        || lower.contains("change")
1725        || lower.contains("fix")
1726        || lower.contains("implement")
1727        || lower.contains("refactor"))
1728        && (lower.contains("folder")
1729            || lower.contains("directory")
1730            || lower.split_whitespace().any(|w| {
1731                let w = w.trim_matches(|c: char| !c.is_alphanumeric());
1732                w == "file"
1733                    || w == "files"
1734                    || w == "code"
1735                    || w == "script"
1736                    || w == "css"
1737                    || w == "js"
1738                    || w == "html"
1739                    || w == "ts"
1740                    || w == "rust"
1741                    || w == "json"
1742                    || w == "logic"
1743            })
1744            || lower.contains("code")
1745            || lower.contains("desktop")
1746            || lower.contains("logic")
1747            || lower.contains("css")
1748            || lower.contains("styles")
1749            || lower.contains("script")
1750            || code_kw_ac().find(&lower).is_some());
1751    let asks_broad_readiness = lower.contains("local development")
1752        || lower.contains("ready for local development")
1753        || (lower.contains("machine") && lower.contains("ready"))
1754        || (lower.contains("computer") && lower.contains("ready"));
1755    let asks_os_config = lower.contains("firewall")
1756        || lower.contains("power plan")
1757        || lower.contains("power settings")
1758        || lower.contains("powercfg")
1759        || lower.contains("uptime")
1760        || lower.contains("boot time")
1761        || lower.contains("last boot")
1762        || lower.contains("windows version")
1763        || lower.contains("what version of windows")
1764        || lower.contains("os version")
1765        || lower.contains("build number")
1766        || lower.contains("windows build")
1767        || lower.contains("edition of windows")
1768        || lower.contains("which windows")
1769        || (lower.contains("windows")
1770            && (lower.contains("10 or 11") || lower.contains("11 or 10")));
1771    let asks_health_report = lower.contains("health report")
1772        || lower.contains("system health")
1773        || (lower.contains("how") && lower.contains("machine") && lower.contains("doing"))
1774        || (lower.contains("status") && lower.contains("report") && !lower.contains("git"));
1775    let asks_updates = lower.contains("up to date")
1776        || lower.contains("windows update")
1777        || lower.contains("pending update")
1778        || lower.contains("update available")
1779        || lower.contains("check for update")
1780        || lower.contains("latest update")
1781        || (lower.contains("update")
1782            && (lower.contains("my pc")
1783                || lower.contains("my computer")
1784                || lower.contains("my machine")))
1785        || lower.contains("check updates")
1786        || lower.contains("update windows")
1787        || lower.contains("windows needs update")
1788        || (lower.contains("windows") && lower.contains("out of date"));
1789    let asks_security = lower.contains("antivirus")
1790        || lower.contains("defender")
1791        || lower.contains("virus protection")
1792        || lower.contains("malware")
1793        || lower.contains("windows security")
1794        || lower.contains("windows activated")
1795        || lower.contains("activation status")
1796        || (lower.contains("protected") && (lower.contains("pc") || lower.contains("computer")))
1797        || (lower.contains("security")
1798            && !lower.contains("git")
1799            && !lower.contains("ssh")
1800            && !lower.contains("token"));
1801    let asks_pending_reboot = lower.contains("need to restart")
1802        || lower.contains("need to reboot")
1803        || lower.contains("requires restart")
1804        || lower.contains("requires a reboot")
1805        || lower.contains("reboot required")
1806        || lower.contains("restart required")
1807        || lower.contains("pending restart")
1808        || lower.contains("pending reboot")
1809        || (lower.contains("restart")
1810            && (lower.contains("waiting")
1811                || lower.contains("queued")
1812                || lower.contains("required")))
1813        || (lower.contains("reboot") && lower.contains("required"))
1814        || (lower.contains("reboot") && lower.contains("pending"))
1815        || (lower.contains("restart") && lower.contains("pending"))
1816        || (lower.contains("have to") && (lower.contains("restart") || lower.contains("reboot")))
1817        || (lower.contains("do i need to")
1818            && (lower.contains("restart") || lower.contains("reboot")));
1819    let asks_disk_health = lower.contains("disk health")
1820        || lower.contains("drive health")
1821        || lower.contains("hard drive dying")
1822        || lower.contains("smart status")
1823        || lower.contains("drive failing")
1824        || lower.contains("drive fail")
1825        || lower.contains("ssd health")
1826        || lower.contains("nvme health")
1827        || lower.contains("hard drive status")
1828        || (lower.contains("drive") && lower.contains("status") && !lower.contains("backup"))
1829        || (lower.contains("dying") && (lower.contains("drive") || lower.contains("disk")))
1830        || (lower.contains("healthy")
1831            && (lower.contains("drive")
1832                || lower.contains("disk")
1833                || lower.contains("ssd")
1834                || lower.contains("hdd")))
1835        || lower.contains("bad sector")
1836        || lower.contains("smart data")
1837        || (lower.contains("disk") && lower.contains("fail"));
1838    let asks_battery = lower.contains("battery")
1839        || lower.contains("battery life")
1840        || lower.contains("battery health")
1841        || lower.contains("battery wear")
1842        || lower.contains("charge level")
1843        || lower.contains("charge percentage")
1844        || lower.contains("current charge")
1845        || lower.contains("charge status")
1846        || lower.contains("how long until")
1847        || (lower.contains("dying") && lower.contains("batter"));
1848    let asks_app_crashes = lower.contains("application crash")
1849        || lower.contains("application error")
1850        || lower.contains("application hang")
1851        || lower.contains("app hang")
1852        || lower.contains("faulting application")
1853        || lower.contains("faulting module")
1854        || lower.contains("exception code")
1855        || lower.contains("windows error reporting")
1856        || lower.contains("wer report")
1857        || lower.contains("which app crashed")
1858        || lower.contains("what app crashed")
1859        || lower.contains("what crashed")
1860        || lower.contains("app crash history")
1861        || lower.contains("application crash log")
1862        || lower.contains("apps crashing")
1863        || lower.contains("apps have been crashing")
1864        || lower.contains("applications crashing")
1865        || lower.contains("applications have been crashing")
1866        || lower.contains("what applications crashed")
1867        || lower.contains("which applications crashed")
1868        || lower.contains("what applications have been crashing")
1869        || lower.contains("which applications have been crashing")
1870        || (lower.contains("applications") && lower.contains("crashing"))
1871        || (lower.contains("apps") && lower.contains("crashing"))
1872        || (lower.contains("crash") && lower.contains("program"))
1873        || (lower.contains("crash")
1874            && (lower.contains("chrome")
1875                || lower.contains("edge")
1876                || lower.contains("firefox")
1877                || lower.contains("discord")
1878                || lower.contains("steam")
1879                || lower.contains("office")
1880                || lower.contains("word")
1881                || lower.contains("excel")
1882                || lower.contains("photoshop")));
1883    let asks_recent_crashes = lower.contains("crash")
1884        || lower.contains("bsod")
1885        || lower.contains("blue screen")
1886        || lower.contains("why did my pc restart")
1887        || lower.contains("unexpected restart")
1888        || lower.contains("sudden restart")
1889        || lower.contains("keep restarting")
1890        || lower.contains("keeps restarting")
1891        || lower.contains("restarts randomly")
1892        || lower.contains("random restart")
1893        || lower.contains("random reboot")
1894        || lower.contains("kernel panic")
1895        || (lower.contains("restart") && lower.contains("itself"))
1896        || (lower.contains("restart") && lower.contains("by itself"));
1897    let asks_log_check = lower.contains("event log")
1898        || lower.contains("windows log")
1899        || lower.contains("system log")
1900        || lower.contains("error log")
1901        || lower.contains("recent errors")
1902        || lower.contains("recent warnings")
1903        || lower.contains("recent events")
1904        || lower.contains("event viewer")
1905        || lower.contains("journald")
1906        || lower.contains("journal log")
1907        || lower.contains("show me warnings")
1908        || (lower.contains("log") && lower.contains("error"))
1909        || (lower.contains("log") && lower.contains("warning"))
1910        || (lower.contains("show me") && lower.contains("error"))
1911        || (lower.contains("show me") && lower.contains("warning"))
1912        || (lower.contains("what errors") && lower.contains("log"));
1913    let asks_scheduled_tasks = lower.contains("scheduled task")
1914        || lower.contains("scheduled tasks")
1915        || lower.contains("task scheduler")
1916        || lower.contains("what runs on a timer")
1917        || lower.contains("what runs at")
1918        || lower.contains("cron job")
1919        || lower.contains("background task")
1920        || lower.contains("scheduled job")
1921        || lower.contains("runs automatically")
1922        || lower.contains("running automatically")
1923        || lower.contains("auto-run task")
1924        || (lower.contains("background") && lower.contains("run"))
1925        || (lower.contains("periodic") && lower.contains("run"))
1926        || (lower.contains("what") && lower.contains("schedule"));
1927    let asks_dev_conflicts = lower.contains("dev conflict")
1928        || lower.contains("environment conflict")
1929        || lower.contains("toolchain conflict")
1930        || lower.contains("version conflict")
1931        || lower.contains("path conflict")
1932        || lower.contains("duplicate path")
1933        || lower.contains("package manager conflict")
1934        || lower.contains("nvm conflict")
1935        || lower.contains("pyenv conflict")
1936        || (lower.contains("python") && lower.contains("wrong version"))
1937        || (lower.contains("node") && lower.contains("wrong version"))
1938        || lower.contains("conda shadow")
1939        || lower.contains("dev environment clean");
1940    let asks_disk_benchmark = lower.contains("benchmark")
1941        || lower.contains("stress test")
1942        || lower.contains("load test")
1943        || lower.contains("intensity report")
1944        || lower.contains("io intensity")
1945        || lower.contains("disk intensity")
1946        || lower.contains("thrash")
1947        || lower.contains("latency report");
1948    let asks_storage_deep = lower.contains("where did my")
1949        && (lower.contains("space") || lower.contains("disk") || lower.contains("storage"))
1950        || lower.contains("what is taking up")
1951        || lower.contains("what is using my disk")
1952        || lower.contains("what is eating my disk")
1953        || lower.contains("biggest folders")
1954        || lower.contains("largest folders")
1955        || lower.contains("largest directories")
1956        || lower.contains("what is filling")
1957        || lower.contains("storage breakdown")
1958        || lower.contains("disk breakdown")
1959        || lower.contains("find large files")
1960        || lower.contains("find big files")
1961        || lower.contains("storage deep")
1962        || lower.contains("deep storage")
1963        || lower.contains("storage analysis")
1964        || lower.contains("disk analysis")
1965        || (lower.contains("clean up")
1966            && (lower.contains("disk")
1967                || lower.contains("drive")
1968                || lower.contains("space")
1969                || lower.contains("storage")))
1970        || (lower.contains("clean") && lower.contains("c drive"))
1971        || (lower.contains("analyze") && (lower.contains("storage") || lower.contains("disk")));
1972    let asks_storage = lower.contains("storage")
1973        || lower.contains("disk space")
1974        || lower.contains("drive capacity")
1975        || lower.contains("free space")
1976        || lower.contains("how much space")
1977        || lower.contains("space left")
1978        || lower.contains("running out of space")
1979        || lower.contains("i/o pressure")
1980        || lower.contains("disk usage")
1981        || lower.contains("disk usage")
1982        || lower.contains("how much disk")
1983        || lower.contains("how full")
1984        || lower.contains("cache size")
1985        || (lower.contains("drive") && lower.contains("usage"))
1986        || (lower.contains("drives") && lower.contains("usage"))
1987        || (lower.contains("where") && lower.contains("space") && lower.contains("go"))
1988        || ((lower.contains("disk") || lower.contains("drive")) && lower.contains("full"))
1989        || lower.contains("out of space");
1990    let asks_resource_load = lower.contains("resource load")
1991        || lower.contains("system load")
1992        || lower.contains("performance")
1993        || lower.contains("utilization")
1994        || lower.contains("usage report")
1995        || lower.contains("performance report")
1996        || lower.contains("what is my load")
1997        || lower.contains("current load")
1998        || lower.contains("why is it slow")
1999        || lower.contains("why is it laggy")
2000        || lower.contains("memory pressure")
2001        || lower.contains("memory load")
2002        || lower.contains("process overhead")
2003        || lower.contains("slow")
2004        || lower.contains("lag")
2005        || lower.contains("sluggish")
2006        || lower.contains("hang")
2007        || lower.contains("unresponsive")
2008        || lower.contains("frozen")
2009        || lower.contains("freezing up")
2010        || lower.contains("computer freeze")
2011        || lower.contains("is it working hard")
2012        || lower.contains("high cpu")
2013        || lower.contains("high ram")
2014        || lower.contains("cpu load")
2015        || lower.contains("heavy hitters")
2016        || (lower.contains("resource") && lower.contains("usage"));
2017
2018    let asks_connectivity = lower.contains("internet")
2019        || lower.contains("online")
2020        || lower.contains("connectivity")
2021        || lower.contains("am i connected")
2022        || lower.contains("ping google")
2023        || lower.contains("reach the internet")
2024        || lower.contains("internet access")
2025        || lower.contains("no internet")
2026        || lower.contains("internet down")
2027        || lower.starts_with("ping ")
2028        || lower.contains(" ping ")
2029        || (lower.contains("check") && lower.contains("connection"))
2030        || (lower.contains("dns") && (lower.contains("resolv") || lower.contains("working")))
2031        || lower.contains("can't browse")
2032        || lower.contains("cannot browse")
2033        || lower.contains("web browsing")
2034        || lower.contains("browser not loading")
2035        || lower.contains("pages not loading")
2036        || lower.contains("websites not loading")
2037        || lower.contains("no network")
2038        || (lower.contains("network") && lower.contains("down"))
2039        || (lower.contains("can't") && lower.contains("connect to internet"))
2040        || (lower.contains("cannot") && lower.contains("connect to internet"))
2041        || (lower.contains("network") && lower.contains("not working"))
2042        || (lower.contains("internet") && lower.contains("not working"));
2043    let asks_wifi = lower.contains("wi-fi")
2044        || lower.contains("wifi")
2045        || lower.contains("wireless")
2046        || lower.contains("wlan")
2047        || lower.contains("signal strength")
2048        || lower.contains("ssid")
2049        || lower.contains("access point")
2050        || (lower.contains("wireless") && lower.contains("connect"));
2051    let asks_connections = lower.contains("tcp connection")
2052        || lower.contains("active connection")
2053        || lower.contains("established connection")
2054        || lower.contains("socket")
2055        || lower.contains("netstat")
2056        || lower.contains("outbound connection")
2057        || lower.contains("inbound connection")
2058        || lower.contains("remote connection")
2059        || lower.contains("connection list")
2060        || (lower.contains("connection") && lower.contains("active"))
2061        || (lower.contains("connection") && lower.contains("open"))
2062        || (lower.contains("what") && lower.contains("connecting"))
2063        || (lower.contains("which") && lower.contains("connecting"))
2064        || (lower.contains("process") && lower.contains("network") && lower.contains("connect"));
2065    let asks_vpn = lower.contains("vpn")
2066        || lower.contains("virtual private network")
2067        || lower.contains("wireguard")
2068        || lower.contains("anyconnect")
2069        || lower.contains("globalprotect")
2070        || lower.contains("pulse secure")
2071        || lower.contains("openvpn")
2072        || lower.contains("split tunnel")
2073        || lower.contains("vpn adapter")
2074        || (lower.contains("tunnel") && (lower.contains("network") || lower.contains("vpn")));
2075    let asks_proxy = lower.contains("proxy")
2076        || lower.contains("proxy setting")
2077        || lower.contains("winhttp proxy")
2078        || lower.contains("system proxy")
2079        || (lower.contains("routed") && lower.contains("proxy"));
2080    let asks_firewall_rules = (lower.contains("firewall")
2081        && (lower.contains("rule")
2082            || lower.contains("block")
2083            || lower.contains("allow")
2084            || lower.contains("inbound")
2085            || lower.contains("outbound")))
2086        || lower.contains("blocked port")
2087        || lower.contains("firewall rule");
2088    let asks_traceroute = lower.contains("traceroute")
2089        || lower.contains("tracert")
2090        || lower.contains("tracepath")
2091        || lower.contains("trace route")
2092        || lower.contains("trace the route")
2093        || lower.contains("trace the path")
2094        || lower.contains("network path")
2095        || lower.contains("how many hops")
2096        || lower.contains("where does traffic go")
2097        || (lower.contains("trace") && lower.contains("hop"))
2098        || (lower.contains("route") && lower.contains("traffic"))
2099        || (lower.contains("trace") && lower.contains("8.8.8.8"))
2100        || (lower.contains("path") && lower.contains("8.8.8.8"));
2101    let asks_dns_cache = lower.contains("dns cache")
2102        || lower.contains("cached dns")
2103        || lower.contains("dns lookup cache")
2104        || lower.contains("displaydns")
2105        || lower.contains("/displaydns")
2106        || lower.contains("get-dnsclientcache")
2107        || lower.contains("dns entries")
2108        || (lower.contains("dns") && lower.contains("cached"));
2109    let asks_arp = lower.contains("arp -")
2110        || lower.contains("arp table")
2111        || lower.contains("arp cache")
2112        || lower.contains("mac address")
2113        || lower.contains("neighbor table")
2114        || lower.contains("ip to mac")
2115        || lower.contains("ip neigh")
2116        || (lower.contains("arp")
2117            && (lower.contains("who") || lower.contains("entry") || lower.contains("entries")));
2118    let asks_route_table = lower.contains("route print")
2119        || lower.contains("route table")
2120        || lower.contains("routing table")
2121        || lower.contains("get-netroute")
2122        || lower.contains("default gateway")
2123        || lower.contains("network routes")
2124        || lower.contains("ip route")
2125        || lower.contains("next hop")
2126        || (lower.contains("route")
2127            && (lower.contains("table") || lower.contains("entry") || lower.contains("entries")));
2128    let asks_env = (lower.contains("environment variable")
2129        || lower.contains("env var")
2130        || lower.contains("env vars")
2131        || lower.contains("show env")
2132        || lower.contains("list env"))
2133        && !lower.contains("env doctor");
2134    let asks_hosts_file = lower.contains("hosts file")
2135        || lower.contains("/etc/hosts")
2136        || lower.contains("etc/hosts")
2137        || lower.contains("hosts entry")
2138        || lower.contains("hosts entries")
2139        || (lower.contains("hosts")
2140            && (lower.contains("redirect")
2141                || lower.contains("block")
2142                || lower.contains("loopback")));
2143    let asks_docker = lower.contains("docker")
2144        || lower.contains("container")
2145        || lower.contains("docker compose")
2146        || lower.contains("docker ps")
2147        || lower.contains("running container");
2148    let asks_docker_filesystems = (lower.contains("docker")
2149        || lower.contains("container")
2150        || lower.contains("compose")
2151        || lower.contains("volume")
2152        || lower.contains("bind mount"))
2153        && (lower.contains("mount")
2154            || lower.contains("volume")
2155            || lower.contains("bind")
2156            || lower.contains("filesystem")
2157            || lower.contains("storage")
2158            || lower.contains("path")
2159            || lower.contains("missing"));
2160    let asks_wsl = lower.contains("wsl")
2161        || lower.contains("windows subsystem")
2162        || lower.contains("linux distro")
2163        || lower.contains("ubuntu on windows")
2164        || (lower.contains("subsystem") && lower.contains("linux"));
2165    let asks_wsl_filesystems = (lower.contains("wsl")
2166        || lower.contains("windows subsystem")
2167        || lower.contains("linux distro")
2168        || lower.contains("ubuntu on windows")
2169        || (lower.contains("subsystem") && lower.contains("linux")))
2170        && (lower.contains("mount")
2171            || lower.contains("filesystem")
2172            || lower.contains("storage")
2173            || lower.contains("disk")
2174            || lower.contains("vhdx")
2175            || lower.contains("path bridge")
2176            || lower.contains("/mnt/c")
2177            || lower.contains("wsl df")
2178            || lower.contains("wsl du")
2179            || lower.contains("du -sh /mnt/c"));
2180    let asks_ssh = (lower.contains("ssh") && !lower.contains("ssh key") && !lower.contains("git"))
2181        || lower.contains("sshd")
2182        || lower.contains("ssh config")
2183        || lower.contains("ssh server")
2184        || lower.contains("ssh client")
2185        || lower.contains("known_hosts")
2186        || lower.contains("authorized_keys")
2187        || lower.contains("ssh key")
2188        || (lower.contains("ssh")
2189            && (lower.contains("running")
2190                || lower.contains("service")
2191                || lower.contains("port 22")));
2192    let asks_installed_software = lower.contains("installed software")
2193        || lower.contains("installed program")
2194        || lower.contains("installed app")
2195        || lower.contains("installed package")
2196        || lower.contains("what is installed")
2197        || lower.contains("what's installed")
2198        || lower.contains("winget list")
2199        || lower.contains("list programs")
2200        || lower.contains("list applications")
2201        || lower.contains("list all apps")
2202        || lower.contains("list apps")
2203        || lower.contains("show applications")
2204        || lower.contains("show programs")
2205        || (lower.contains("list") && lower.contains("application"))
2206        || (lower.contains("show") && lower.contains("application"))
2207        || (lower.contains("installed")
2208            && (lower.contains("on this machine")
2209                || lower.contains("on my machine")
2210                || lower.contains("on my pc")));
2211    let asks_databases = lower.contains("postgres")
2212        || lower.contains("postgresql")
2213        || lower.contains("mysql")
2214        || lower.contains("mariadb")
2215        || lower.contains("mongodb")
2216        || lower.contains("mongo")
2217        || lower.contains("redis")
2218        || lower.contains("sql server")
2219        || lower.contains("mssql")
2220        || lower.contains("sqlite")
2221        || lower.contains("elasticsearch")
2222        || lower.contains("cassandra")
2223        || lower.contains("couchdb")
2224        || (lower.contains("database")
2225            && (lower.contains("running")
2226                || lower.contains("service")
2227                || lower.contains("installed")
2228                || lower.contains("up")
2229                || lower.contains("local")))
2230        || lower.contains("db service")
2231        || lower.contains("database server")
2232        || (lower.contains("is")
2233            && lower.contains("running")
2234            && (lower.contains("db") || lower.contains("database")));
2235    let asks_git_config = (lower.contains("git config")
2236        || lower.contains("git configuration")
2237        || lower.contains("git global")
2238        || (lower.contains("git") && lower.contains("user.name"))
2239        || (lower.contains("git") && lower.contains("user.email"))
2240        || (lower.contains("git") && lower.contains("signing"))
2241        || (lower.contains("git") && lower.contains("credential"))
2242        || (lower.contains("git") && lower.contains("auth"))
2243        || lower.contains("git push denied")
2244        || lower.contains("git clone failed")
2245        || lower.contains("git identity")
2246        || lower.contains("git aliases"))
2247        && !lower.contains("github");
2248    let asks_audit_policy = lower.contains("audit policy")
2249        || lower.contains("auditpol")
2250        || lower.contains("audit log")
2251        || lower.contains("what is being logged")
2252        || lower.contains("security audit")
2253        || lower.contains("logon event")
2254        || lower.contains("login event")
2255        || lower.contains("audit category")
2256        || lower.contains("event auditing")
2257        || (lower.contains("audit") && lower.contains("event"))
2258        || (lower.contains("what") && lower.contains("being audited"))
2259        || (lower.contains("audit") && lower.contains("enable"));
2260    let asks_shares = lower.contains("smb share")
2261        || lower.contains("network share")
2262        || lower.contains("shared folder")
2263        || lower.contains("mapped drive")
2264        || lower.contains("mapped network drive")
2265        || lower.contains("get-smbshare")
2266        || lower.contains("what is shared")
2267        || lower.contains("what am i sharing")
2268        || lower.contains("file sharing")
2269        || lower.contains("smb session")
2270        || lower.contains("lanmanager")
2271        || lower.contains("netlanmanager")
2272        || lower.contains("smb1")
2273        || lower.contains("smb signing")
2274        || lower.contains("nfs export")
2275        || (lower.contains("folder") && lower.contains("shared"))
2276        || (lower.contains("sharing") && lower.contains("network"))
2277        || lower.contains("what am i sharing");
2278    let asks_dns_servers = (lower.contains("dns server")
2279        || lower.contains("dns resolver")
2280        || lower.contains("nameserver")
2281        || lower.contains("which dns")
2282        || lower.contains("what dns")
2283        || lower.contains("dns over https")
2284        || lower.contains("doh")
2285        || lower.contains("dns search suffix")
2286        || lower.contains("configured dns")
2287        || lower.contains("get-dnsclientserveraddress"))
2288        && !lower.contains("dns cache")
2289        && (!lower.contains("adapter")
2290            || contains_any(
2291                &lower,
2292                &[
2293                    "dns server",
2294                    "dns resolver",
2295                    "nameserver",
2296                    "configured dns",
2297                    "per adapter",
2298                    "which dns",
2299                    "what dns",
2300                    "get-dnsclientserveraddress",
2301                ],
2302            ))
2303        && !lower.contains("ip address")
2304        && !lower.contains("gateway");
2305    let asks_bitlocker = lower.contains("bitlocker")
2306        || (lower.contains("drive") && lower.contains("encrypt"))
2307        || (lower.contains("disk") && lower.contains("encrypt"))
2308        || (lower.contains("ssd") && lower.contains("encrypt"))
2309        || (lower.contains("volume") && lower.contains("encrypt"))
2310        || (lower.contains("machine") && lower.contains("encrypt"))
2311        || lower.contains("encryption status")
2312        || lower.contains("full disk encryption")
2313        || lower.contains("drive encryption");
2314    let asks_rdp = lower.contains("rdp")
2315        || lower.contains("remote desktop")
2316        || (lower.contains("remote") && lower.contains("access") && !lower.contains("git"));
2317    let asks_shadow_copies = lower.contains("shadow copy")
2318        || lower.contains("shadow copies")
2319        || lower.contains("vss")
2320        || lower.contains("snapshot")
2321        || lower.contains("restore point");
2322    let asks_pagefile = lower.contains("pagefile")
2323        || lower.contains("page file")
2324        || lower.contains("virtual memory")
2325        || lower.contains("swap file")
2326        || lower.contains("swap space")
2327        || lower.contains("memory swapping")
2328        || lower.contains("paging file")
2329        || (lower.contains("paging") && lower.contains("file"))
2330        || (lower.contains("paging") && lower.contains("active"));
2331    let asks_windows_features = (lower.contains("window") && lower.contains("feature"))
2332        || lower.contains("optional feature")
2333        || lower.contains("iis")
2334        || lower.contains("hyper-v")
2335        || (lower.contains("feature")
2336            && (lower.contains("install")
2337                || lower.contains("enabled")
2338                || lower.contains("turn on")));
2339    let asks_printers = lower.contains("printer")
2340        || lower.contains("print queue")
2341        || lower.contains("get-printer")
2342        || lower.contains("printing")
2343        || lower.contains("can't print")
2344        || lower.contains("cannot print")
2345        || lower.contains("print job")
2346        || lower.contains("print driver")
2347        || lower.contains("default printer")
2348        || lower.contains("add printer")
2349        || lower.contains("print to pdf")
2350        || (lower.contains("print") && lower.contains("not working"))
2351        || (lower.contains("print") && lower.contains("stuck"))
2352        || (lower.contains("print") && lower.contains("pending"))
2353        || (lower.contains("print") && lower.contains("offline"));
2354    let asks_winrm = lower.contains("winrm")
2355        || lower.contains("psremoting")
2356        || (lower.contains("ps") && lower.contains("remoting"))
2357        || (lower.contains("remote") && lower.contains("management") && !lower.contains("rdp"));
2358    let asks_network_stats = (lower.contains("network") && lower.contains("stat"))
2359        || (lower.contains("adapter")
2360            && lower.contains("stat")
2361            && !lower.contains("wlan")
2362            && !lower.contains("wireless"))
2363        || (lower.contains("nic") && lower.contains("stat"))
2364        || lower.contains("throughput")
2365        || lower.contains("dropped packet")
2366        || (lower.contains("network") && lower.contains("usage"))
2367        || (lower.contains("data") && lower.contains("transferred"))
2368        || (lower.contains("bytes") && lower.contains("transferred"))
2369        || lower.contains("network traffic")
2370        || (lower.contains("packet") && lower.contains("error"));
2371    let asks_udp_ports = lower.contains("udp port")
2372        || lower.contains("udp listener")
2373        || (lower.contains("udp")
2374            && (lower.contains("port")
2375                || lower.contains("listen")
2376                || lower.contains("open")
2377                || lower.contains("service")
2378                || lower.contains("connection")));
2379
2380    let asks_domain_health = lower.contains("domain health")
2381        || lower.contains("dc connectivity")
2382        || lower.contains("dc reachab")
2383        || lower.contains("can reach dc")
2384        || lower.contains("ldap port")
2385        || lower.contains("kerberos health")
2386        || lower.contains("kerberos connectivity")
2387        || lower.contains("ad connectivity")
2388        || lower.contains("active directory health")
2389        || lower.contains("domain controller connectivity")
2390        || lower.contains("domain controller reachab")
2391        || lower.contains("nltest")
2392        || lower.contains("dsgetdc")
2393        || lower.contains("gpo refresh")
2394        || (lower.contains("domain controller")
2395            && (lower.contains("reach")
2396                || lower.contains("connect")
2397                || lower.contains("test")
2398                || lower.contains("check")
2399                || lower.contains("online")
2400                || lower.contains("up")))
2401        || (lower.contains("active directory")
2402            && (lower.contains("connect")
2403                || lower.contains("reach")
2404                || lower.contains("health")
2405                || lower.contains("working")
2406                || lower.contains("accessible")))
2407        || (lower.contains("can reach") && lower.contains("domain"))
2408        || (lower.contains("kerberos") && lower.contains("issue"))
2409        || (lower.contains("kerberos") && lower.contains("fail"));
2410    let asks_service_dependencies = lower.contains("service depend")
2411        || lower.contains("services depend")
2412        || lower.contains("depends on")
2413        || lower.contains("service graph")
2414        || lower.contains("which services depend")
2415        || lower.contains("what depends on")
2416        || lower.contains("restart cascade")
2417        || lower.contains("svc dep")
2418        || lower.contains("service prerequisite")
2419        || (lower.contains("service")
2420            && (lower.contains("dependency")
2421                || lower.contains("dependencies")
2422                || lower.contains("required by")
2423                || lower.contains("needed by")
2424                || lower.contains("requirement")
2425                || lower.contains("relationship")));
2426    let asks_wmi_health = lower.contains("wmi health")
2427        || lower.contains("wmi corrupt")
2428        || lower.contains("wmi repository")
2429        || lower.contains("wmi broken")
2430        || lower.contains("winmgmt")
2431        || lower.contains("wmi query fail")
2432        || lower.contains("wmi not working")
2433        || (lower.contains("wmi")
2434            && (lower.contains("health")
2435                || lower.contains("status")
2436                || lower.contains("repair")
2437                || lower.contains("reset")
2438                || lower.contains("broken")));
2439    let asks_local_security_policy = lower.contains("password policy")
2440        || lower.contains("account lockout")
2441        || lower.contains("lockout policy")
2442        || lower.contains("lockout threshold")
2443        || lower.contains("lm compatibility")
2444        || lower.contains("ntlm level")
2445        || lower.contains("ntlm policy")
2446        || lower.contains("local security policy")
2447        || lower.contains("account policy")
2448        || lower.contains("uac level")
2449        || lower.contains("uac policy")
2450        || lower.contains("uac disabled")
2451        || lower.contains("uac prompt")
2452        || lower.contains("uac not")
2453        || lower.contains("user account control")
2454        || lower.contains("needs elevation")
2455        || lower.contains("needs admin")
2456        || lower.contains("run as administrator")
2457        || lower.contains("administrator permission")
2458        || lower.contains("lmcompatibilitylevel")
2459        || lower.contains("net accounts")
2460        || lower.contains("lockout")
2461        || (lower.contains("uac")
2462            && (lower.contains("off")
2463                || lower.contains("status")
2464                || lower.contains("check")
2465                || lower.contains("on")))
2466        || (lower.contains("password")
2467            && (lower.contains("minimum")
2468                || lower.contains("maximum age")
2469                || lower.contains("complexity")
2470                || lower.contains("history")
2471                || lower.contains("policy")));
2472    let asks_usb_history = lower.contains("usb history")
2473        || lower.contains("usb devices connected")
2474        || lower.contains("usb forensic")
2475        || lower.contains("usbstor")
2476        || lower.contains("usb registry")
2477        || lower.contains("what usb")
2478        || lower.contains("ever connected usb")
2479        || lower.contains("usb devices ever")
2480        || lower.contains("usb drives ever")
2481        || (lower.contains("usb")
2482            && (lower.contains("history")
2483                || lower.contains("forensic")
2484                || lower.contains("audit")
2485                || lower.contains("ever connected")
2486                || lower.contains("registry")
2487                || lower.contains("plugged")
2488                || lower.contains("were connected")
2489                || lower.contains("have been connected")
2490                || lower.contains("has been connected")));
2491    let asks_print_spooler = lower.contains("print spooler")
2492        || lower.contains("spooler service")
2493        || lower.contains("printnightmare")
2494        || lower.contains("print nightmar")
2495        || lower.contains("cve-2021-34527")
2496        || lower.contains("cve-2021-1675")
2497        || lower.contains("print security")
2498        || lower.contains("printer security")
2499        || lower.contains("printer service")
2500        || lower.contains("point and print")
2501        || lower.contains("rpcauthnlevel")
2502        || (lower.contains("print") && lower.contains("vulnerab"))
2503        || lower.contains("print service")
2504        || (lower.contains("printer") && lower.contains("spooler"))
2505        || (lower.contains("spooler")
2506            && (lower.contains("status")
2507                || lower.contains("running")
2508                || lower.contains("security")
2509                || lower.contains("hardening")));
2510
2511    // Host-remediation queries (e.g., "fix cargo not found on this machine") contain
2512    // code keywords like "cargo" that also trip the mutation guard. Check fix_plan
2513    // first so these read-only host inspection requests are never silently dropped.
2514    if asks_fix_plan && asks_mutation_intent {
2515        return Some("fix_plan");
2516    }
2517
2518    // If the user has a clear mutation intent (create folder, edit file),
2519    // we should NOT route to a read-only host inspection topic, as that would
2520    // trigger a pre-run crash. The main LLM turn will handle the mutation.
2521    if asks_mutation_intent {
2522        return None;
2523    }
2524
2525    // Priority 1: High-Precision Enterprise Triage (IT Pro Plus)
2526    if asks_overclocker {
2527        Some("overclocker")
2528    } else if asks_ad_user {
2529        Some("ad_user")
2530    } else if asks_user_accounts {
2531        Some("user_accounts")
2532    } else if asks_dns_lookup {
2533        Some("dns_lookup")
2534    } else if asks_event_query {
2535        Some("event_query")
2536    } else if asks_mdm {
2537        Some("mdm_enrollment")
2538    } else if asks_hyperv {
2539        Some("hyperv")
2540    } else if asks_ip_config {
2541        Some("ip_config")
2542    } else if asks_disk_benchmark {
2543        Some("disk_benchmark")
2544    } else if asks_fix_plan {
2545        Some("fix_plan")
2546    } else if asks_env_doctor {
2547        Some("env_doctor")
2548    } else if asks_traceroute {
2549        Some("traceroute")
2550    } else if asks_dhcp {
2551        Some("dhcp")
2552    } else if asks_mtu {
2553        Some("mtu")
2554    } else if asks_ipv6 {
2555        Some("ipv6")
2556    } else if asks_domain_health {
2557        Some("domain_health")
2558    } else if asks_service_dependencies {
2559        Some("service_dependencies")
2560    } else if asks_wmi_health {
2561        Some("wmi_health")
2562    } else if asks_local_security_policy {
2563        Some("local_security_policy")
2564    } else if asks_usb_history {
2565        Some("usb_history")
2566    } else if asks_print_spooler {
2567        Some("print_spooler")
2568    } else if asks_latency {
2569        Some("latency")
2570    } else if asks_nic_teaming {
2571        Some("nic_teaming")
2572    } else if asks_network_stats {
2573        Some("network_stats")
2574    } else if asks_share_access {
2575        Some("share_access")
2576    } else if asks_thermal {
2577        Some("thermal")
2578    } else if asks_activation {
2579        Some("activation")
2580    } else if asks_patch_history {
2581        Some("patch_history")
2582    } else if asks_bluetooth {
2583        Some("bluetooth")
2584    } else if asks_audio {
2585        Some("audio")
2586    } else if asks_camera {
2587        Some("camera")
2588    } else if asks_identity_auth {
2589        Some("identity_auth")
2590    } else if asks_sign_in {
2591        Some("sign_in")
2592    } else if asks_installer_health {
2593        Some("installer_health")
2594    } else if asks_teams {
2595        Some("teams")
2596    } else if asks_windows_backup {
2597        Some("windows_backup")
2598    } else if asks_onedrive {
2599        Some("onedrive")
2600    } else if asks_browser_health {
2601        Some("browser_health")
2602    } else if asks_outlook {
2603        Some("outlook")
2604    } else if asks_search_index {
2605        Some("search_index")
2606    } else if asks_display_config {
2607        Some("display_config")
2608    } else if asks_ntp {
2609        Some("ntp")
2610    } else if asks_cpu_power {
2611        Some("cpu_power")
2612    } else if asks_credentials {
2613        Some("credentials")
2614    } else if asks_tpm {
2615        Some("tpm")
2616    } else if asks_network_adapter {
2617        Some("network_adapter")
2618    } else if asks_tcp_params {
2619        Some("tcp_params")
2620    } else if asks_wlan_profiles {
2621        Some("wlan_profiles")
2622    } else if asks_ipsec {
2623        Some("ipsec")
2624    } else if asks_udp_ports {
2625        Some("udp_ports")
2626    } else if asks_port_test {
2627        Some("port_test")
2628    } else if asks_netbios {
2629        Some("netbios")
2630    } else if asks_snmp {
2631        Some("snmp")
2632    } else if asks_network_profile {
2633        Some("network_profile")
2634    } else if asks_permissions {
2635        Some("permissions")
2636    } else if asks_login_history {
2637        Some("login_history")
2638    } else if asks_registry_audit {
2639        Some("registry_audit")
2640    } else if asks_docker_filesystems {
2641        Some("docker_filesystems")
2642    } else if asks_wsl_filesystems {
2643        Some("wsl_filesystems")
2644    } else if asks_lan_discovery {
2645        Some("lan_discovery")
2646    } else if asks_storage_spaces {
2647        Some("storage_spaces")
2648    } else if asks_defender_quarantine {
2649        Some("defender_quarantine")
2650    } else if asks_storage_deep {
2651        Some("storage_deep")
2652    } else if asks_storage {
2653        Some("storage")
2654    } else if asks_gpo {
2655        Some("gpo")
2656    } else if asks_data_audit {
2657        Some("data_audit")
2658    } else if asks_certificates {
2659        Some("certificates")
2660    } else if asks_integrity {
2661        Some("integrity")
2662    } else if asks_domain {
2663        Some("domain")
2664    } else if asks_device_health {
2665        Some("device_health")
2666    } else if asks_drivers {
2667        Some("drivers")
2668    } else if asks_peripherals {
2669        Some("peripherals")
2670    } else if asks_sessions {
2671        Some("sessions")
2672    } else if asks_virtualization {
2673        Some("hardware")
2674    } else if asks_services {
2675        Some("services")
2676    } else if asks_startup {
2677        Some("startup_items")
2678    } else if asks_bitlocker {
2679        Some("bitlocker")
2680    } else if asks_rdp {
2681        Some("rdp")
2682    } else if asks_shadow_copies {
2683        Some("shadow_copies")
2684    } else if asks_pagefile {
2685        Some("pagefile")
2686    } else if asks_windows_features {
2687        Some("windows_features")
2688    } else if asks_printers {
2689        Some("printers")
2690    } else if asks_winrm {
2691        Some("winrm")
2692    } else if (asks_path && asks_toolchains)
2693        || (mentions_host_inspection_question(&lower) && asks_broad_readiness)
2694    {
2695        Some("summary")
2696    } else if asks_dns_servers {
2697        Some("dns_servers")
2698    } else if asks_connectivity {
2699        Some("connectivity")
2700    } else if asks_wifi {
2701        Some("wifi")
2702    } else if asks_connections {
2703        Some("connections")
2704    } else if asks_vpn {
2705        Some("vpn")
2706    } else if asks_proxy {
2707        Some("proxy")
2708    } else if asks_firewall_rules {
2709        Some("firewall_rules")
2710    } else if asks_dns_cache {
2711        Some("dns_cache")
2712    } else if asks_arp {
2713        Some("arp")
2714    } else if asks_route_table {
2715        Some("route_table")
2716    } else if asks_shares {
2717        Some("shares")
2718    } else if asks_network {
2719        Some("network")
2720    } else if asks_health_report {
2721        Some("health_report")
2722    } else if asks_os_config {
2723        Some("os_config")
2724    } else if asks_hardware || asks_virtualization {
2725        Some("hardware")
2726    } else if asks_updates {
2727        Some("updates")
2728    } else if asks_audit_policy {
2729        Some("audit_policy")
2730    } else if asks_security {
2731        Some("security")
2732    } else if asks_pending_reboot {
2733        Some("pending_reboot")
2734    } else if asks_disk_health {
2735        Some("disk_health")
2736    } else if asks_battery {
2737        Some("battery")
2738    } else if asks_app_crashes {
2739        Some("app_crashes")
2740    } else if asks_recent_crashes {
2741        Some("recent_crashes")
2742    } else if asks_log_check {
2743        Some("log_check")
2744    } else if asks_scheduled_tasks {
2745        Some("scheduled_tasks")
2746    } else if asks_dev_conflicts {
2747        Some("dev_conflicts")
2748    } else if asks_databases {
2749        Some("databases")
2750    } else if asks_docker {
2751        Some("docker")
2752    } else if asks_wsl {
2753        Some("wsl")
2754    } else if asks_ssh {
2755        Some("ssh")
2756    } else if asks_git_config {
2757        Some("git_config")
2758    } else if asks_installed_software {
2759        Some("installed_software")
2760    } else if asks_env {
2761        Some("env")
2762    } else if asks_hosts_file {
2763        Some("hosts_file")
2764    } else if asks_ports {
2765        Some("ports")
2766    } else if asks_processes {
2767        Some("processes")
2768    } else if asks_repo_doctor {
2769        Some("repo_doctor")
2770    } else if lower.contains("desktop")
2771        && (lower.contains("show")
2772            || lower.contains("list")
2773            || lower.contains("what is in")
2774            || lower.contains("what's in")
2775            || lower.contains("folder"))
2776    {
2777        Some("desktop")
2778    } else if lower.contains("downloads")
2779        && (lower.contains("show")
2780            || lower.contains("list")
2781            || lower.contains("what is in")
2782            || lower.contains("what's in")
2783            || lower.contains("folder"))
2784    {
2785        Some("downloads")
2786    } else if asks_path {
2787        Some("path")
2788    } else if asks_toolchains {
2789        Some("toolchains")
2790    } else if asks_resource_load {
2791        Some("resource_load")
2792    } else if asks_directory {
2793        Some("directory")
2794    } else if mentions_host_inspection_question(&lower) && !is_conversational_advisory(&lower) {
2795        Some("summary")
2796    } else {
2797        None
2798    }
2799}
2800
2801type TopicDetector = (&'static str, fn(&str) -> bool);
2802
2803pub fn all_host_inspection_topics(user_input: &str) -> Vec<&'static str> {
2804    // All topic detectors in priority order — ordered so more specific topics come
2805    // before generic fallbacks (e.g. traceroute before network).
2806    let lower = user_input.to_lowercase();
2807    let mut topics: Vec<&'static str> = Vec::with_capacity(4);
2808
2809    let detectors: &[TopicDetector] = &[
2810        ("overclocker", |l| {
2811            l.contains("overclocker")
2812                || l.contains("gpu clock")
2813                || l.contains("gpu throttle")
2814                || l.contains("throttle reason")
2815                || l.contains("root cause")
2816                || l.contains("nvidia stats")
2817                || l.contains("silicon health")
2818                || ((l.contains("voltage") || l.contains("volts"))
2819                    && (l.contains("gpu")
2820                        || l.contains("cpu")
2821                        || l.contains("nvidia")
2822                        || l.contains("silicon")))
2823                || (l.contains("gpu")
2824                    && (l.contains("throttle")
2825                        || l.contains("bottleneck")
2826                        || l.contains("performance")
2827                        || l.contains("overheating")
2828                        || l.contains("usage")
2829                        || l.contains("utilization")))
2830        }),
2831        ("data_audit", |l| {
2832            l.contains("data audit")
2833                || l.contains("audit data")
2834                || l.contains("csv schema")
2835                || l.contains("data schema")
2836                || l.contains("inspect file")
2837                || l.contains("profile data")
2838                || l.contains("data distribution")
2839                || (l.contains("schema") && (l.contains("csv") || l.contains("json")))
2840        }),
2841        ("directory", |l| {
2842            (l.contains("make")
2843                || l.contains("create")
2844                || l.contains("mkdir")
2845                || l.contains("organize"))
2846                && (l.contains("folder")
2847                    || l.contains("directory")
2848                    || l.contains("project area")
2849                    || l.contains("desktop"))
2850        }),
2851        ("ad_user", |l| {
2852            l.contains("ad user")
2853                || l.contains("domain user")
2854                || (l.contains("user") && (l.contains("sid") || l.contains("membership")))
2855        }),
2856        ("dns_lookup", |l| {
2857            l.contains("dns lookup")
2858                || l.contains("dns record")
2859                || l.contains("dns query")
2860                || l.contains("nslookup")
2861                || l.contains("resolve-dnsname")
2862                || l.contains("gethostaddresses")
2863                || l.contains("gethostentry")
2864                || l.contains("[system.net.dns]")
2865                || l.contains(" dig ")
2866                || l.starts_with("host ")
2867                || (l.contains("ip address") && l.contains(" of "))
2868                || l.contains("srv record")
2869                || l.contains("mx record")
2870        }),
2871        ("mdm_enrollment", |l| {
2872            l.contains("mdm")
2873                || l.contains("intune")
2874                || l.contains("autopilot")
2875                || l.contains("device enrollment")
2876                || l.contains("enrolled in")
2877                || l.contains("mdm enrollment")
2878                || l.contains("device management")
2879                || l.contains("managed device")
2880                || l.contains("azure ad join")
2881                || l.contains("aad join")
2882                || (l.contains("enrolled") && l.contains("device"))
2883                || (l.contains("enroll") && l.contains("device"))
2884                || (l.contains("microsoft") && l.contains("endpoint"))
2885        }),
2886        ("hyperv", |l| {
2887            l.contains("hyper-v")
2888                || l.contains("hyperv")
2889                || l.contains("hyper v")
2890                || l.contains("virtual machine")
2891                || l.contains("running vms")
2892                || l.contains("list vms")
2893                || l.contains("list vm")
2894                || l.contains("vmmem")
2895                || l.contains("vmms")
2896                || (l.contains("vm")
2897                    && !l.contains("nvme")
2898                    && (l.contains("checkpoint")
2899                        || l.contains("snapshot")
2900                        || l.contains("switch")
2901                        || l.contains("running")))
2902        }),
2903        ("ip_config", |l| {
2904            l.contains("ipconfig")
2905                || l.contains("ip config")
2906                || l.contains("adapter detail")
2907                || l.contains("dhcp lease")
2908        }),
2909        ("event_query", |l| {
2910            l.contains("event id")
2911                || l.contains("event_id")
2912                || l.contains("eventid")
2913                || l.contains("event log query")
2914                || l.contains("search event")
2915                || l.contains("query event")
2916                || l.contains("failed logon event")
2917                || l.contains("failed login event")
2918                || l.contains("application error event")
2919                || ((l.contains("event log")
2920                    || l.contains("system log")
2921                    || l.contains("application log")
2922                    || l.contains("security log"))
2923                    && (l.contains("last ")
2924                        || l.contains("past ")
2925                        || l.contains("today")
2926                        || l.contains("hour")
2927                        || l.contains("hours"))
2928                    && (l.contains("error")
2929                        || l.contains("errors")
2930                        || l.contains("warning")
2931                        || l.contains("warnings")
2932                        || l.contains("critical")))
2933                || (l.contains("event")
2934                    && (l.contains("4625")
2935                        || l.contains("7034")
2936                        || l.contains("7031")
2937                        || l.contains("4648")))
2938        }),
2939        ("fix_plan", |l| {
2940            l.contains("fix")
2941                && (l.contains("cargo")
2942                    || l.contains("port ")
2943                    || l.contains("lm studio")
2944                    || l.contains("toolchain"))
2945        }),
2946        ("updates", |l| {
2947            l.contains("up to date")
2948                || l.contains("windows update")
2949                || l.contains("pending update")
2950                || l.contains("update available")
2951                || l.contains("check updates")
2952                || l.contains("update windows")
2953                || (l.contains("windows") && l.contains("out of date"))
2954        }),
2955        ("security", |l| {
2956            l.contains("antivirus")
2957                || l.contains("defender")
2958                || l.contains("uac")
2959                || (l.contains("security") && !l.contains("git") && !l.contains("ssh"))
2960        }),
2961        ("defender_quarantine", |l| {
2962            l.contains("defender quarantine")
2963                || l.contains("quarantine threat")
2964                || l.contains("quarantine")
2965                || l.contains("threat history")
2966                || l.contains("malware history")
2967                || l.contains("defender history")
2968                || l.contains("detected threat")
2969                || l.contains("detected virus")
2970                || l.contains("malware detected")
2971                || l.contains("defender found")
2972                || l.contains("defender detected")
2973                || l.contains("defender find")
2974                || l.contains("virus found")
2975                || l.contains("threats found")
2976                || l.contains("threat detected")
2977                || l.contains("threat detection")
2978                || (l.contains("defender")
2979                    && (l.contains("malware") || l.contains("virus") || l.contains("threat")))
2980                || (l.contains("defender")
2981                    && l.contains("scan")
2982                    && (l.contains("result") || l.contains("history") || l.contains("found")))
2983        }),
2984        ("permissions", |l| {
2985            l.contains("permission") || l.contains("access control") || l.contains("get-acl")
2986        }),
2987        ("login_history", |l| {
2988            l.contains("login history")
2989                || l.contains("logon history")
2990                || l.contains("event id 4624")
2991        }),
2992        ("registry_audit", |l| {
2993            l.contains("registry audit")
2994                || l.contains("persistence")
2995                || l.contains("ifeo")
2996                || l.contains("reg query")
2997        }),
2998        ("share_access", |l| {
2999            l.contains("share access")
3000                || l.contains("unc path")
3001                || l.contains("smbshare")
3002                || l.contains("net share")
3003                || (l.contains("network drive") && !l.contains("network drives"))
3004                || l.contains("mapped drive")
3005                || l.contains("shared folder")
3006        }),
3007        ("thermal", |l| {
3008            l.contains("thermal")
3009                || l.contains("throttling")
3010                || l.contains("overheat")
3011                || l.contains("too hot")
3012                || l.contains("running hot")
3013                || l.contains("cpu temp")
3014                || l.contains("cpu temperature")
3015                || l.contains("temp sensor")
3016                || l.contains("fan loud")
3017                || l.contains("fan noise")
3018                || l.contains("fan spinning")
3019                || l.contains("fan running")
3020                || l.contains("fans running")
3021                || l.contains("fans spinning")
3022                || l.contains("loud fan")
3023                || l.contains("fan always on")
3024                || (l.contains("fan") && l.contains("always on"))
3025                || l.contains("fan at 100")
3026                || l.contains("fan constantly")
3027                || l.contains("laptop hot")
3028                || l.contains("pc getting hot")
3029                || l.contains("getting hot")
3030                || l.contains("check temps")
3031                || (l.contains("temperature")
3032                    && (l.contains("cpu")
3033                        || l.contains("gpu")
3034                        || l.contains("system")
3035                        || l.contains("sensor")
3036                        || l.contains("check")))
3037        }),
3038        ("overclocker", |l| {
3039            l.contains("overclocker")
3040                || l.contains("gpu clock")
3041                || l.contains("nvidia stats")
3042                || l.contains("silicon health")
3043                || l.contains("mhz")
3044                || (l.contains("gpu")
3045                    && (l.contains("usage")
3046                        || l.contains("utilization")
3047                        || l.contains("performance")))
3048        }),
3049        ("activation", |l| {
3050            l.contains("activation")
3051                || l.contains("slmgr")
3052                || l.contains("license status")
3053                || l.contains("licensed")
3054                || l.contains("unlicensed")
3055                || (l.contains("activate") && l.contains("window"))
3056                || l.contains("not activated")
3057                || l.contains("product key")
3058                || l.contains("license expired")
3059                || l.contains("is windows genuine")
3060                || (l.contains("windows") && l.contains("activated"))
3061        }),
3062        ("patch_history", |l| {
3063            l.contains("patch history")
3064                || l.contains("hotfix")
3065                || l.contains("kb history")
3066                || l.contains("security patch")
3067                || (l.contains("update") && l.contains("applied"))
3068        }),
3069        ("bluetooth", |l| {
3070            l.contains("bluetooth")
3071                || l.contains("pairing")
3072                || l.contains("paired device")
3073                || l.contains("paired devices")
3074                || l.contains("bthserv")
3075                || l.contains("bthavctpsvc")
3076                || l.contains("btagservice")
3077                || l.contains("bluetoothuserservice")
3078                || ((l.contains("headset") || l.contains("headphones"))
3079                    && (l.contains("disconnect")
3080                        || l.contains("pair")
3081                        || l.contains("reconnect")
3082                        || l.contains("bluetooth")))
3083        }),
3084        ("audio", |l| {
3085            l.contains("no sound")
3086                || l.contains("audio service")
3087                || l.contains("windows audio")
3088                || l.contains("speaker")
3089                || l.contains("speakers")
3090                || l.contains("microphone")
3091                || l.contains(" mic ")
3092                || l.starts_with("mic ")
3093                || l.contains("mic not")
3094                || l.contains("headset")
3095                || l.contains("headphones")
3096                || l.contains("playback device")
3097                || l.contains("recording device")
3098                || l.contains("audio endpoint")
3099                || l.contains("audioendpointbuilder")
3100                || l.contains("can't hear")
3101                || l.contains("cannot hear")
3102                || l.contains("cant hear")
3103                || l.contains("no audio")
3104                || (((l.contains("audio") || l.contains("sound"))
3105                    && (l.contains("device")
3106                        || l.contains("driver")
3107                        || l.contains("service")
3108                        || l.contains("working")
3109                        || l.contains("broken")
3110                        || l.contains("input")
3111                        || l.contains("output")
3112                        || l.contains("crackling")
3113                        || l.contains("mute")
3114                        || l.contains("muted")
3115                        || l.contains("volume")
3116                        || l.contains("speaker")
3117                        || l.contains("microphone")))
3118                    && !l.contains("audio file")
3119                    && !l.contains("voice engine"))
3120        }),
3121        ("camera", |l| {
3122            l.contains("camera")
3123                || l.contains("webcam")
3124                || l.contains("web cam")
3125                || (l.contains("camera") && l.contains("permission"))
3126                || (l.contains("camera") && l.contains("privacy"))
3127        }),
3128        ("sign_in", |l| {
3129            l.contains("windows hello")
3130                || l.contains("sign in")
3131                || l.contains("cant sign in")
3132                || l.contains("can't sign in")
3133                || l.contains("logon failure")
3134                || l.contains("login screen stuck")
3135                || l.contains("stuck on login")
3136                || l.contains("login loop")
3137                || l.contains("can't login")
3138                || l.contains("cannot login")
3139                || l.contains("cant login")
3140                || l.contains("login failed")
3141                || l.contains("login failure")
3142                || (l.contains("login") && l.contains("not working"))
3143                || (l.contains("login") && l.contains("problem"))
3144                || (l.contains("login") && l.contains("status"))
3145                || l.contains("sign-in status")
3146                || l.contains("sign in status")
3147                || (l.contains("can't log in") && !l.contains("vpn") && !l.contains("ssh"))
3148                || (l.contains("pin") && (l.contains("broken") || l.contains("not working")))
3149                || l.contains("credential provider")
3150                || l.contains("biometric service")
3151                || l.contains("wbiosrvc")
3152        }),
3153        ("identity_auth", |l| {
3154            l.contains("web account manager")
3155                || l.contains("token broker")
3156                || l.contains("tokenbroker")
3157                || l.contains("aad broker")
3158                || l.contains("broker plugin")
3159                || l.contains("identity broker")
3160                || l.contains("microsoft 365 sign-in")
3161                || l.contains("microsoft 365 signin")
3162                || l.contains("office sign-in")
3163                || l.contains("office signin")
3164                || l.contains("workplace join")
3165                || l.contains("device registration")
3166                || l.contains("device registered")
3167                || l.contains("entra")
3168                || l.contains("azure ad")
3169                || l.contains("azuread")
3170                || l.contains("azure ad prt")
3171                || l.contains("azureadprt")
3172                || l.contains("wamdefaultset")
3173                || l.contains("single sign-on")
3174                || l.contains("organizational account")
3175                || l.contains("corporate account")
3176                || (l.contains("azure") && l.contains("registered"))
3177                || ((l.contains("outlook")
3178                    || l.contains("teams")
3179                    || l.contains("onedrive")
3180                    || l.contains("office")
3181                    || l.contains("microsoft 365"))
3182                    && (l.contains("sign in")
3183                        || l.contains("signin")
3184                        || l.contains("signed in")
3185                        || l.contains("signed out")
3186                        || l.contains("keeps asking")
3187                        || l.contains("keep asking")
3188                        || l.contains("authentication")
3189                        || l.contains("auth")
3190                        || l.contains("token")
3191                        || l.contains("work account")
3192                        || l.contains("school account")
3193                        || l.contains("account mismatch")))
3194        }),
3195        ("installer_health", |l| {
3196            l.contains("installer health")
3197                || l.contains("installer broken")
3198                || l.contains("msiexec")
3199                || l.contains("msi installer")
3200                || l.contains("windows installer")
3201                || l.contains("app installer")
3202                || l.contains("desktopappinstaller")
3203                || l.contains("microsoft store")
3204                || l.contains("winget broken")
3205                || (l.contains("unable to install")
3206                    && (l.contains("app") || l.contains("program") || l.contains("software")))
3207                || ((l.contains("install") || l.contains("installer"))
3208                    && (l.contains("fail")
3209                        || l.contains("failing")
3210                        || l.contains("broken")
3211                        || l.contains("stuck")
3212                        || l.contains("hanging")
3213                        || l.contains("error"))
3214                    && !l.contains("windows update"))
3215        }),
3216        ("onedrive", |l| {
3217            l.contains("onedrive")
3218                || l.contains("one drive")
3219                || l.contains("files on-demand")
3220                || l.contains("known folder backup")
3221                || l.contains("known folder move")
3222                || l.contains("kfm")
3223                || l.contains("sharepoint sync")
3224                || l.contains("sync root")
3225                || ((l.contains("desktop") || l.contains("documents") || l.contains("pictures"))
3226                    && l.contains("backup")
3227                    && (l.contains("onedrive") || l.contains("cloud") || l.contains("sync")))
3228                || ((l.contains("desktop") || l.contains("documents") || l.contains("pictures"))
3229                    && l.contains("sync")
3230                    && (l.contains("onedrive") || l.contains("sharepoint") || l.contains("cloud")))
3231        }),
3232        ("browser_health", |l| {
3233            l.contains("browser health")
3234                || l.contains("webview2")
3235                || l.contains("default browser")
3236                || ((l.contains("browser")
3237                    || l.contains("chrome")
3238                    || l.contains("edge")
3239                    || l.contains("firefox"))
3240                    && (l.contains("slow")
3241                        || l.contains("sluggish")
3242                        || l.contains("lag")
3243                        || l.contains("crash")
3244                        || l.contains("crashing")
3245                        || l.contains("hang")
3246                        || l.contains("freeze")
3247                        || l.contains("frozen")
3248                        || l.contains("broken")
3249                        || l.contains("unresponsive")
3250                        || l.contains("not starting")
3251                        || l.contains("not loading")
3252                        || l.contains("extension")
3253                        || l.contains("extensions")
3254                        || l.contains("proxy")
3255                        || l.contains("policy")))
3256                || ((l.contains("links") || l.contains("link"))
3257                    && (l.contains("open wrong")
3258                        || l.contains("opens wrong")
3259                        || l.contains("wrong browser")
3260                        || l.contains("wrong app")))
3261        }),
3262        ("outlook", |l| {
3263            l.contains("outlook")
3264                || l.contains("ms outlook")
3265                || l.contains("microsoft outlook")
3266                || (l.contains("ost") && l.contains("mail"))
3267                || (l.contains("pst") && l.contains("mail"))
3268                || (l.contains("add-in") && l.contains("mail"))
3269        }),
3270        ("teams", |l| {
3271            (l.contains("teams")
3272                && !l.contains("nic team")
3273                && !l.contains("nic teaming")
3274                && !l.contains("link aggregation")
3275                && !l.contains("lbfo"))
3276                || l.contains("ms teams")
3277                || l.contains("microsoft teams")
3278        }),
3279        ("windows_backup", |l| {
3280            l.contains("file history")
3281                || l.contains("windows backup")
3282                || l.contains("wbadmin")
3283                || l.contains("system restore")
3284                || l.contains("restore point")
3285                || l.contains("known folder move")
3286                || l.contains("backed up")
3287                || (l.contains("backup")
3288                    && (l.contains("drive")
3289                        || l.contains("running")
3290                        || l.contains("health")
3291                        || l.contains("status")
3292                        || l.contains("failed")
3293                        || l.contains("enabled")
3294                        || l.contains("working")
3295                        || l.contains("set up")
3296                        || l.contains("configured")
3297                        || l.contains("schedule")))
3298        }),
3299        ("search_index", |l| {
3300            l.contains("search index")
3301                || l.contains("windows search")
3302                || l.contains("wsearch")
3303                || l.contains("indexer")
3304                || (l.contains("search") && l.contains("broken"))
3305                || (l.contains("search") && l.contains("not working"))
3306        }),
3307        ("display_config", |l| {
3308            l.contains("monitor")
3309                || l.contains("display")
3310                || l.contains("resolution")
3311                || l.contains("refresh rate")
3312                || l.contains("refresh hz")
3313                || l.contains("dpi")
3314                || l.contains("scaling")
3315                || l.contains("screen config")
3316                || l.contains("hdmi")
3317                || l.contains("displayport")
3318                || l.contains("how many screens")
3319                || l.contains("multi-monitor")
3320                || l.contains("second screen")
3321                || l.contains("external display")
3322        }),
3323        ("ntp", |l| {
3324            l.contains("ntp")
3325                || l.contains("time sync")
3326                || l.contains("clock sync")
3327                || l.contains("w32tm")
3328                || l.contains("clock drift")
3329                || l.contains("time server")
3330                || l.contains("time zone")
3331                || l.contains("timezone")
3332                || l.contains("wrong timezone")
3333                || l.contains("clock wrong")
3334                || l.contains("time wrong")
3335                || l.contains("system clock")
3336                || l.contains("system time")
3337                || (l.contains("time") && l.contains("drift"))
3338                || (l.contains("clock") && l.contains("off"))
3339                || (l.contains("time") && l.contains("accurate"))
3340                || (l.contains("time") && l.contains("correct"))
3341        }),
3342        ("cpu_power", |l| {
3343            l.contains("turbo boost")
3344                || l.contains("cpu frequency")
3345                || l.contains("cpu freq")
3346                || l.contains("cpu clock")
3347                || l.contains("cpu power")
3348                || l.contains("cpu speed")
3349                || l.contains("processor speed")
3350                || l.contains("processor frequency")
3351                || l.contains("power plan")
3352                || l.contains("cpu stuck")
3353                || l.contains("boost disabled")
3354                || (l.contains("boost") && l.contains("disabled"))
3355                || (l.contains("cpu") && l.contains("slow"))
3356                || (l.contains("processor") && l.contains("slow"))
3357                || (l.contains("cpu") && l.contains("underclocking"))
3358                || (l.contains("processor") && l.contains("running slow"))
3359                || l.contains("processor running at")
3360        }),
3361        ("credentials", |l| {
3362            l.contains("credential manager")
3363                || l.contains("credential store")
3364                || l.contains("saved password")
3365                || l.contains("stored credential")
3366                || l.contains("credential vault")
3367                || l.contains("cmdkey")
3368                || (l.contains("credential") && l.contains("list"))
3369                || (l.contains("windows") && l.contains("credential"))
3370                || (l.contains("credential")
3371                    && (l.contains("clear")
3372                        || l.contains("cached")
3373                        || l.contains("view")
3374                        || l.contains("delete")
3375                        || l.contains("remove")))
3376        }),
3377        ("tpm", |l| {
3378            l.contains("tpm")
3379                || l.contains("secure boot")
3380                || l.contains("trusted platform module")
3381                || l.contains("firmware security")
3382                || l.contains("uefi security")
3383                || l.contains("uefi mode")
3384                || l.contains("uefi enabled")
3385                || l.contains("legacy bios")
3386                || l.contains("uefi bios")
3387                || (l.contains("uefi")
3388                    && (l.contains("boot")
3389                        || l.contains("secure")
3390                        || l.contains("status")
3391                        || l.contains("check")))
3392        }),
3393        ("dhcp", |l| {
3394            l.contains("dhcp lease")
3395                || l.contains("lease expires")
3396                || l.contains("dhcp server")
3397                || l.contains("ip lease")
3398                || l.contains("lease renew")
3399                || (l.contains("dhcp")
3400                    && (l.contains("detail") || l.contains("info") || l.contains("check")))
3401        }),
3402        ("mtu", |l| {
3403            l.contains("mtu")
3404                || l.contains("path mtu")
3405                || l.contains("pmtu")
3406                || l.contains("frame size")
3407                || l.contains("fragmentation")
3408                || (l.contains("vpn") && l.contains("mtu"))
3409                || (l.contains("packet") && l.contains("size") && l.contains("max"))
3410        }),
3411        ("latency", |l| {
3412            l.contains("ping")
3413                || l.contains("latency")
3414                || l.contains("packet loss")
3415                || l.contains("rtt")
3416                || l.contains("round trip")
3417                || l.contains("network lag")
3418                || l.contains("jitter")
3419                || (l.contains("network") && l.contains("slow"))
3420                || (l.contains("internet") && l.contains("slow"))
3421        }),
3422        ("network_adapter", |l| {
3423            l.contains("nic settings")
3424                || l.contains("nic offload")
3425                || l.contains("adapter settings")
3426                || l.contains("jumbo frame")
3427                || l.contains("tcp offload")
3428                || l.contains("wake on lan")
3429                || l.contains("wake-on-lan")
3430                || l.contains("link speed")
3431                || l.contains("duplex mismatch")
3432                || l.contains("adapter error")
3433                || (l.contains("nic") && (l.contains("driver") || l.contains("error")))
3434        }),
3435        ("ipv6", |l| {
3436            l.contains("ipv6")
3437                || l.contains("slaac")
3438                || l.contains("dhcpv6")
3439                || l.contains("privacy extension")
3440                || l.contains("global unicast")
3441        }),
3442        ("tcp_params", |l| {
3443            l.contains("tcp autotuning")
3444                || l.contains("tcp congestion")
3445                || l.contains("congestion algorithm")
3446                || l.contains("tcp settings")
3447                || l.contains("tcp tuning")
3448                || l.contains("tcp chimney")
3449                || l.contains("ecn")
3450                || l.contains("receive window")
3451                || l.contains("tcp window")
3452                || (l.contains("tcp")
3453                    && (l.contains("slow")
3454                        || l.contains("throughput")
3455                        || l.contains("speed")
3456                        || l.contains("window size")))
3457        }),
3458        ("wlan_profiles", |l| {
3459            l.contains("saved wifi")
3460                || l.contains("wifi profile")
3461                || l.contains("wlan profile")
3462                || l.contains("wireless profile")
3463                || l.contains("saved network")
3464                || l.contains("netsh wlan")
3465                || (l.contains("wifi")
3466                    && (l.contains("security")
3467                        || l.contains("audit")
3468                        || l.contains("remember")
3469                        || l.contains("password")))
3470                || (l.contains("wireless") && l.contains("remember"))
3471        }),
3472        ("ipsec", |l| {
3473            l.contains("ipsec")
3474                || l.contains("security association")
3475                || l.contains("ike tunnel")
3476                || l.contains("ipsec tunnel")
3477                || l.contains("policy agent")
3478                || l.contains("xfrm")
3479        }),
3480        ("netbios", |l| {
3481            l.contains("netbios")
3482                || l.contains("nbtstat")
3483                || l.contains("wins server")
3484                || l.contains("nbns")
3485        }),
3486        ("nic_teaming", |l| {
3487            l.contains("nic team")
3488                || l.contains("lacp")
3489                || l.contains("link aggregation")
3490                || l.contains("lbfo")
3491                || l.contains("bonding")
3492                || (l.contains("bond")
3493                    && (l.contains("adapter") || l.contains("interface") || l.contains("nic")))
3494        }),
3495        ("snmp", |l| {
3496            l.contains("snmp")
3497                || l.contains("community string")
3498                || l.contains("community name")
3499                || l.contains("snmpd")
3500        }),
3501        ("port_test", |l| {
3502            l.contains("port test")
3503                || l.contains("test port")
3504                || l.contains("port check")
3505                || l.contains("can i reach")
3506                || l.contains("is port")
3507                || l.contains("port reachab")
3508                || (l.contains("port")
3509                    && (l.contains("open") || l.contains("blocked") || l.contains("reachable")))
3510        }),
3511        ("network_profile", |l| {
3512            l.contains("network profile")
3513                || l.contains("network location")
3514                || l.contains("network category")
3515                || l.contains("public network")
3516                || l.contains("private network")
3517        }),
3518        ("dns_lookup", |l| {
3519            l.contains("dns lookup")
3520                || l.contains("dns record")
3521                || l.contains("nslookup")
3522                || l.contains("resolve-dnsname")
3523                || l.contains("gethostaddresses")
3524                || l.contains("gethostentry")
3525                || l.contains("mx record")
3526                || l.contains("srv record")
3527                || l.contains("look up ")
3528                || l.contains(" dig ")
3529                || l.starts_with("host ")
3530                || (l.contains("ip address") && l.contains(" of "))
3531                || (l.contains("resolve") && (l.contains("hostname") || l.contains("domain")))
3532        }),
3533        ("pending_reboot", |l| {
3534            l.contains("pending reboot")
3535                || l.contains("pending restart")
3536                || l.contains("need to restart")
3537                || l.contains("reboot required")
3538                || (l.contains("reboot") && l.contains("pending"))
3539                || (l.contains("restart") && l.contains("pending"))
3540                || (l.contains("have to") && (l.contains("restart") || l.contains("reboot")))
3541                || (l.contains("do i need to") && (l.contains("restart") || l.contains("reboot")))
3542        }),
3543        ("disk_health", |l| {
3544            l.contains("disk health")
3545                || l.contains("drive health")
3546                || l.contains("smart status")
3547                || l.contains("smart data")
3548                || l.contains("bad sector")
3549                || l.contains("ssd health")
3550                || l.contains("nvme health")
3551                || l.contains("hard drive status")
3552                || (l.contains("drive") && l.contains("status") && !l.contains("backup"))
3553                || (l.contains("healthy")
3554                    && (l.contains("drive") || l.contains("disk") || l.contains("ssd")))
3555                || (l.contains("disk") && l.contains("fail"))
3556        }),
3557        ("battery", |l| {
3558            l.contains("battery")
3559                || l.contains("charge percentage")
3560                || l.contains("current charge")
3561                || l.contains("charge status")
3562        }),
3563        ("app_crashes", |l| {
3564            l.contains("application crash")
3565                || l.contains("application error")
3566                || l.contains("app hang")
3567                || l.contains("application hang")
3568                || l.contains("faulting application")
3569                || l.contains("faulting module")
3570                || l.contains("wer report")
3571                || l.contains("apps crashing")
3572                || l.contains("what crashed")
3573                || l.contains("which app crashed")
3574                || l.contains("what app crashed")
3575                || l.contains("app crash history")
3576                || l.contains("application crash log")
3577                || (l.contains("crash") && l.contains("program"))
3578                || (l.contains("applications") && l.contains("crashing"))
3579                || (l.contains("apps") && l.contains("crashing"))
3580                || (l.contains("crash")
3581                    && (l.contains("chrome")
3582                        || l.contains("edge")
3583                        || l.contains("firefox")
3584                        || l.contains("discord")
3585                        || l.contains("office")))
3586        }),
3587        ("recent_crashes", |l| {
3588            l.contains("crash")
3589                || l.contains("bsod")
3590                || l.contains("blue screen")
3591                || l.contains("unexpected restart")
3592                || l.contains("sudden restart")
3593                || l.contains("keep restarting")
3594                || l.contains("keeps restarting")
3595                || l.contains("restarts randomly")
3596                || l.contains("random restart")
3597                || l.contains("random reboot")
3598                || l.contains("kernel panic")
3599                || (l.contains("restart") && l.contains("itself"))
3600                || (l.contains("restart") && l.contains("by itself"))
3601                || (l.contains("why") && l.contains("restart"))
3602        }),
3603        ("scheduled_tasks", |l| {
3604            l.contains("scheduled task")
3605                || l.contains("task scheduler")
3606                || l.contains("scheduled job")
3607                || l.contains("runs automatically")
3608                || l.contains("running automatically")
3609                || l.contains("auto-run task")
3610                || l.contains("what runs on a timer")
3611                || l.contains("cron job")
3612                || l.contains("background task")
3613                || (l.contains("background") && l.contains("run"))
3614                || (l.contains("periodic") && l.contains("run"))
3615                || (l.contains("what") && l.contains("schedule"))
3616        }),
3617        ("ad_user", |l| {
3618            l.contains("ad user")
3619                || l.contains("domain user")
3620                || (l.contains("user") && l.contains("sid"))
3621        }),
3622        ("dns_lookup", |l| {
3623            (l.contains("dns") && (l.contains("lookup") || l.contains("srv") || l.contains("mx")))
3624                || l.contains("resolve-dnsname")
3625                || l.contains("gethostaddresses")
3626                || l.contains("gethostentry")
3627                || l.starts_with("host ")
3628                || (l.contains("ip address") && l.contains(" of "))
3629        }),
3630        ("hyperv", |l| {
3631            l.contains("hyper-v")
3632                || l.contains("hyperv")
3633                || l.contains("hyper v")
3634                || l.contains("virtual machine")
3635                || l.contains("running vms")
3636                || (l.contains("vm")
3637                    && !l.contains("nvme")
3638                    && (l.contains("running")
3639                        || l.contains("checkpoint")
3640                        || l.contains("snapshot")
3641                        || l.contains("switch")
3642                        || l.contains("ram")
3643                        || l.contains("memory")))
3644                || (l.contains("list") && l.contains("vm") && !l.contains("nvme"))
3645        }),
3646        ("ip_config", |l| {
3647            l.contains("ipconfig") && (l.contains("all") || l.contains("detail"))
3648        }),
3649        ("dev_conflicts", |l| {
3650            l.contains("dev conflict")
3651                || l.contains("toolchain conflict")
3652                || l.contains("duplicate path")
3653        }),
3654        ("storage_deep", |l| {
3655            (l.contains("where did my")
3656                && (l.contains("space") || l.contains("disk") || l.contains("storage")))
3657                || l.contains("what is taking up")
3658                || l.contains("what is using my disk")
3659                || l.contains("what is eating my disk")
3660                || l.contains("biggest folders")
3661                || l.contains("largest folders")
3662                || l.contains("largest directories")
3663                || l.contains("what is filling")
3664                || l.contains("storage breakdown")
3665                || l.contains("disk breakdown")
3666                || l.contains("find large files")
3667                || l.contains("find big files")
3668                || l.contains("storage analysis")
3669                || l.contains("disk analysis")
3670                || (l.contains("clean up")
3671                    && (l.contains("disk")
3672                        || l.contains("drive")
3673                        || l.contains("space")
3674                        || l.contains("storage")))
3675                || (l.contains("clean") && l.contains("c drive"))
3676                || (l.contains("analyze") && (l.contains("storage") || l.contains("disk")))
3677        }),
3678        ("storage", |l| {
3679            l.contains("disk space")
3680                || l.contains("storage")
3681                || l.contains("drive capacity")
3682                || l.contains("cache size")
3683                || l.contains("i/o pressure")
3684                || l.contains("disk usage")
3685                || ((l.contains("disk") || l.contains("drive")) && l.contains("full"))
3686                || l.contains("out of space")
3687                || l.contains("free space")
3688                || l.contains("how much space")
3689                || l.contains("space left")
3690                || l.contains("running out of space")
3691                || l.contains("how much disk")
3692                || l.contains("how full")
3693                || (l.contains("drive") && l.contains("usage"))
3694                || (l.contains("where") && l.contains("space"))
3695        }),
3696        ("storage_spaces", |l| {
3697            l.contains("storage spaces")
3698                || l.contains("storage space")
3699                || l.contains("storage pool")
3700                || l.contains("storage pools")
3701                || l.contains("virtual disk")
3702                || l.contains("virtual disks")
3703                || l.contains("windows raid")
3704                || l.contains("storage pool degraded")
3705                || l.contains("parity volume")
3706                || l.contains("disk pool")
3707                || l.contains("resiliency")
3708                || l.contains("mdadm")
3709                || l.contains("software raid")
3710                || l.contains("md array")
3711                || (l.contains("mirror") && l.contains("drive"))
3712                || (l.contains("storage") && l.contains("pool"))
3713        }),
3714        ("disk_benchmark", |l| {
3715            l.contains("benchmark")
3716                || l.contains("stress test")
3717                || l.contains("load test")
3718                || l.contains("intensity report")
3719                || l.contains("io intensity")
3720                || l.contains("disk intensity")
3721                || l.contains("thrash")
3722                || l.contains("latency report")
3723        }),
3724        ("log_check", |l| {
3725            l.contains("event log")
3726                || l.contains("recent errors")
3727                || l.contains("recent warnings")
3728                || l.contains("error log")
3729                || l.contains("event viewer")
3730                || l.contains("system log")
3731                || l.contains("windows log")
3732                || l.contains("application log")
3733                || l.contains("recent events")
3734                || l.contains("journal log")
3735                || l.contains("show me warnings")
3736                || (l.contains("log") && l.contains("recent") && l.contains("error"))
3737                || (l.contains("log") && l.contains("warning"))
3738                || (l.contains("what errors") && l.contains("log"))
3739        }),
3740        ("hardware", |l| {
3741            l.contains("cpu model")
3742                || l.contains("ram size")
3743                || l.contains("hardware spec")
3744                || (l.contains("what hardware") && l.contains("have"))
3745                || l.contains("hardware info")
3746                || l.contains("hardware inventory")
3747                || l.contains("system information")
3748                || l.contains("system info")
3749                || l.contains("system specs")
3750                || l.contains("computer spec")
3751                || l.contains("graphics card")
3752                || l.contains("video card")
3753                || l.contains("display adapter")
3754                || l.contains("bios version")
3755                || l.contains("motherboard")
3756                || l.contains("what gpu")
3757                || l.contains("what cpu")
3758                || (l.contains("hardware") && (l.contains("dna") || l.contains("inventory")))
3759        }),
3760        ("health_report", |l| {
3761            l.contains("health report") || l.contains("system health")
3762        }),
3763        ("resource_load", |l| {
3764            l.contains("resource load")
3765                || l.contains("cpu load")
3766                || l.contains("ram %")
3767                || l.contains("cpu %")
3768                || l.contains("performance")
3769                || l.contains("memory pressure")
3770                || l.contains("memory load")
3771                || l.contains("slow")
3772                || l.contains("lag")
3773                || l.contains("sluggish")
3774                || l.contains("hang")
3775                || l.contains("unresponsive")
3776                || l.contains("frozen")
3777                || l.contains("freezing up")
3778                || l.contains("computer freeze")
3779                || l.contains("utilization")
3780                || l.contains("high cpu")
3781                || l.contains("high ram")
3782                || l.contains("current load")
3783                || (l.contains("resource") && l.contains("usage"))
3784                || (l.contains("why") && l.contains("slow"))
3785                || (l.contains("why") && l.contains("laggy"))
3786        }),
3787        ("processes", |l| {
3788            l.contains("process")
3789                || l.contains("task manager")
3790                || l.contains("what is running")
3791                || l.contains("using my ram")
3792                || l.contains("using my cpu")
3793                || l.contains("using the cpu")
3794                || l.contains("hitting the disk")
3795                || l.contains("disk thrasher")
3796                || l.contains("cpu hog")
3797                || l.contains("memory hog")
3798                || l.contains("ram hog")
3799                || l.contains("hogging cpu")
3800                || l.contains("hogging ram")
3801                || l.contains("hogging memory")
3802                || l.contains("eating up cpu")
3803                || l.contains("eating up memory")
3804                || l.contains("eating my cpu")
3805                || l.contains("eating my memory")
3806                || (l.contains("hogging")
3807                    && (l.contains("cpu") || l.contains("ram") || l.contains("memory")))
3808                || (l.contains("eating up")
3809                    && (l.contains("cpu") || l.contains("ram") || l.contains("memory")))
3810        }),
3811        ("services", |l| {
3812            l.contains("service") || l.contains("daemon") || l.contains("windows service")
3813        }),
3814        ("ports", |l| {
3815            l.contains("listening port")
3816                || l.contains("open port")
3817                || l.contains("what is on port")
3818                || l.contains("port 3000")
3819                || l.contains("what is listening")
3820                || l.contains("what's listening")
3821                || l.contains("exposed port")
3822                || l.contains("listening on port")
3823                || (l.contains("listening") && l.contains("port"))
3824        }),
3825        ("traceroute", |l| {
3826            l.contains("traceroute")
3827                || l.contains("tracert")
3828                || l.contains("trace route")
3829                || l.contains("trace the path")
3830                || l.contains("network path")
3831                || l.contains("how many hops")
3832                || (l.contains("trace") && l.contains("hop"))
3833        }),
3834        ("dns_cache", |l| {
3835            l.contains("dns cache")
3836                || l.contains("cached dns")
3837                || l.contains("displaydns")
3838                || (l.contains("dns") && l.contains("cached"))
3839        }),
3840        ("arp", |l| {
3841            l.contains("arp table")
3842                || l.contains("arp cache")
3843                || l.contains("mac address")
3844                || l.contains("ip to mac")
3845                || l.contains("arp -")
3846        }),
3847        ("route_table", |l| {
3848            l.contains("route table")
3849                || l.contains("routing table")
3850                || l.contains("route print")
3851                || l.contains("network route")
3852                || l.contains("next hop")
3853        }),
3854        ("connectivity", |l| {
3855            l.contains("internet")
3856                || l.contains("am i connected")
3857                || l.contains("ping google")
3858                || l.contains("internet access")
3859                || l.contains("no internet")
3860                || l.contains("network connectivity")
3861                || l.contains("connectivity check")
3862                || l.contains("check connectivity")
3863                || l.contains("can't browse")
3864                || l.contains("cannot browse")
3865                || l.contains("web browsing")
3866                || l.contains("browser not loading")
3867                || l.contains("pages not loading")
3868                || l.contains("websites not loading")
3869                || l.contains("no network")
3870                || (l.contains("network") && l.contains("down"))
3871                || (l.contains("can't") && l.contains("connect to internet"))
3872                || (l.contains("cannot") && l.contains("connect to internet"))
3873                || (l.contains("network") && l.contains("not working"))
3874                || (l.contains("internet") && l.contains("not working"))
3875        }),
3876        ("wifi", |l| {
3877            l.contains("wi-fi")
3878                || l.contains("wifi")
3879                || l.contains("wireless")
3880                || l.contains("ssid")
3881                || l.contains("signal strength")
3882        }),
3883        ("connections", |l| {
3884            l.contains("tcp connection")
3885                || l.contains("active connection")
3886                || l.contains("established connection")
3887                || l.contains("socket")
3888                || l.contains("netstat")
3889                || l.contains("outbound connection")
3890                || l.contains("inbound connection")
3891                || l.contains("remote connection")
3892                || l.contains("connection list")
3893                || (l.contains("connection") && l.contains("active"))
3894                || (l.contains("connection") && l.contains("open"))
3895                || (l.contains("what") && l.contains("connecting"))
3896                || (l.contains("which") && l.contains("connecting"))
3897        }),
3898        ("vpn", |l| {
3899            l.contains("vpn")
3900                || l.contains("virtual private network")
3901                || l.contains("wireguard")
3902                || l.contains("anyconnect")
3903                || l.contains("globalprotect")
3904                || l.contains("pulse secure")
3905                || l.contains("split tunnel")
3906                || l.contains("vpn adapter")
3907        }),
3908        ("proxy", |l| {
3909            l.contains("proxy setting") || l.contains("system proxy") || l.contains("winhttp proxy")
3910        }),
3911        ("firewall_rules", |l| {
3912            (l.contains("firewall")
3913                && (l.contains("rule") || l.contains("inbound") || l.contains("outbound")))
3914                || l.contains("firewall rule")
3915        }),
3916        ("lan_discovery", |l| {
3917            l.contains("upnp")
3918                || l.contains("ssdp")
3919                || l.contains("mdns")
3920                || l.contains("bonjour")
3921                || l.contains("llmnr")
3922                || l.contains("network neighborhood")
3923                || l.contains("device discovery")
3924                || l.contains("local discovery")
3925                || l.contains("discover local devices")
3926                || l.contains("discover devices")
3927                || l.contains("browse computers")
3928                || (l.contains("local network")
3929                    && (l.contains("discover")
3930                        || l.contains("discovery")
3931                        || l.contains("neighborhood")
3932                        || l.contains("device")
3933                        || l.contains("devices")
3934                        || l.contains("aware of")))
3935                || ((l.contains("netbios") || l.contains("smb visibility"))
3936                    && !l.contains("active directory"))
3937                || ((l.contains("nas")
3938                    || l.contains("printer")
3939                    || l.contains("device")
3940                    || l.contains("computer")
3941                    || l.contains("pc"))
3942                    && ((l.contains("can't") && l.contains("see"))
3943                        || (l.contains("cannot") && l.contains("see"))
3944                        || (l.contains("cant") && l.contains("see"))
3945                        || l.contains("can't see")
3946                        || l.contains("cannot see")
3947                        || l.contains("cant see")
3948                        || l.contains("not visible")
3949                        || l.contains("not showing up")
3950                        || l.contains("not show up")
3951                        || l.contains("discover"))
3952                    && (l.contains("network")
3953                        || l.contains("lan")
3954                        || l.contains("local")
3955                        || l.contains("neighborhood")))
3956        }),
3957        ("network", |l| {
3958            l.contains("network adapter")
3959                || l.contains("ip address")
3960                || l.contains("ipconfig")
3961                || l.contains("gateway")
3962                || l.contains("subnet")
3963        }),
3964        ("env_doctor", |l| {
3965            l.contains("env doctor")
3966                || l.contains("environment doctor")
3967                || l.contains("package manager")
3968                || l.contains("path drift")
3969        }),
3970        ("os_config", |l| {
3971            l.contains("power plan")
3972                || l.contains("uptime")
3973                || l.contains("boot time")
3974                || l.contains("last boot")
3975        }),
3976        ("overclocker", |l| {
3977            l.contains("overclocker")
3978                || l.contains("gpu clock")
3979                || l.contains("gpu throttle")
3980                || l.contains("nvidia stats")
3981                || l.contains("silicon health")
3982                || l.contains("mhz")
3983                || ((l.contains("voltage") || l.contains("volts"))
3984                    && (l.contains("gpu")
3985                        || l.contains("cpu")
3986                        || l.contains("nvidia")
3987                        || l.contains("silicon")))
3988                || (l.contains("gpu")
3989                    && (l.contains("throttle")
3990                        || l.contains("bottleneck")
3991                        || l.contains("overheating")
3992                        || l.contains("usage")
3993                        || l.contains("utilization")
3994                        || l.contains("performance")))
3995        }),
3996        ("path", |l| {
3997            l.contains("path entries") || l.contains("raw path")
3998        }),
3999        ("toolchains", |l| {
4000            l.contains("developer tools")
4001                || l.contains("toolchains")
4002                || (l.contains("installed") && l.contains("version"))
4003        }),
4004        ("docker", |l| {
4005            l.contains("docker") || l.contains("container") || l.contains("running container")
4006        }),
4007        ("docker_filesystems", |l| {
4008            (l.contains("docker")
4009                || l.contains("container")
4010                || l.contains("compose")
4011                || l.contains("volume")
4012                || l.contains("bind mount"))
4013                && (l.contains("mount")
4014                    || l.contains("volume")
4015                    || l.contains("bind")
4016                    || l.contains("filesystem")
4017                    || l.contains("storage")
4018                    || l.contains("path")
4019                    || l.contains("missing"))
4020        }),
4021        ("wsl", |l| {
4022            l.contains("wsl")
4023                || l.contains("windows subsystem")
4024                || (l.contains("subsystem") && l.contains("linux"))
4025        }),
4026        ("wsl_filesystems", |l| {
4027            (l.contains("wsl")
4028                || l.contains("windows subsystem")
4029                || l.contains("linux distro")
4030                || (l.contains("subsystem") && l.contains("linux")))
4031                && (l.contains("mount")
4032                    || l.contains("filesystem")
4033                    || l.contains("storage")
4034                    || l.contains("disk")
4035                    || l.contains("vhdx")
4036                    || l.contains("path bridge")
4037                    || l.contains("/mnt/c")
4038                    || l.contains("wsl df")
4039                    || l.contains("wsl du")
4040                    || l.contains("du -sh /mnt/c"))
4041        }),
4042        ("ssh", |l| {
4043            l.contains("ssh")
4044                || l.contains("sshd")
4045                || l.contains("known_hosts")
4046                || l.contains("authorized_keys")
4047        }),
4048        ("git_config", |l| {
4049            (l.contains("git config") || l.contains("git global") || l.contains("git aliases"))
4050                && !l.contains("github")
4051        }),
4052        ("installed_software", |l| {
4053            l.contains("installed software")
4054                || l.contains("installed program")
4055                || l.contains("installed app")
4056                || l.contains("what is installed")
4057                || l.contains("what's installed")
4058                || l.contains("winget list")
4059                || l.contains("list programs")
4060                || l.contains("list applications")
4061                || l.contains("list apps")
4062                || l.contains("show applications")
4063                || l.contains("show programs")
4064                || (l.contains("list") && l.contains("application"))
4065                || (l.contains("show") && l.contains("application"))
4066        }),
4067        ("env", |l| {
4068            (l.contains("environment variable") || l.contains("env var") || l.contains("env vars"))
4069                && !l.contains("env doctor")
4070        }),
4071        ("hosts_file", |l| {
4072            l.contains("hosts file") || l.contains("/etc/hosts") || l.contains("hosts entry")
4073        }),
4074        ("databases", |l| {
4075            l.contains("postgres")
4076                || l.contains("mysql")
4077                || l.contains("mariadb")
4078                || l.contains("mongodb")
4079                || l.contains("redis")
4080                || l.contains("sqlite")
4081                || l.contains("sql server")
4082                || l.contains("elasticsearch")
4083                || (l.contains("database") && (l.contains("running") || l.contains("service")))
4084        }),
4085        ("user_accounts", |l| {
4086            l.contains("local user")
4087                || l.contains("user account")
4088                || l.contains("who is logged")
4089                || l.contains("who am i")
4090                || l.contains("logged in as")
4091                || l.contains("admin group")
4092                || l.contains("local admin")
4093                || l.contains("active sessions")
4094                || l.contains("running as admin")
4095                || l.contains("who has admin rights")
4096                || l.contains("list all users")
4097                || l.contains("list users")
4098                || l.contains("what accounts")
4099                || (l.contains("accounts") && l.contains("admin"))
4100        }),
4101        ("audit_policy", |l| {
4102            l.contains("audit policy")
4103                || l.contains("auditpol")
4104                || l.contains("what is being logged")
4105                || l.contains("security audit")
4106                || l.contains("event auditing")
4107                || l.contains("login event")
4108                || l.contains("logon event")
4109                || (l.contains("audit") && l.contains("event"))
4110                || (l.contains("what") && l.contains("being audited"))
4111                || (l.contains("audit") && l.contains("enable"))
4112        }),
4113        ("shares", |l| {
4114            l.contains("smb share")
4115                || l.contains("network share")
4116                || l.contains("shared folder")
4117                || l.contains("mapped drive")
4118                || l.contains("file sharing")
4119                || l.contains("what is shared")
4120                || l.contains("what am i sharing")
4121                || l.contains("smb1")
4122                || l.contains("nfs export")
4123                || (l.contains("folder") && l.contains("shared"))
4124                || (l.contains("sharing") && l.contains("network"))
4125        }),
4126        ("dns_servers", |l| {
4127            (l.contains("dns server")
4128                || l.contains("dns resolver")
4129                || l.contains("nameserver")
4130                || l.contains("which dns")
4131                || l.contains("dns over https")
4132                || l.contains("configured dns"))
4133                && !l.contains("dns cache")
4134        }),
4135        ("bitlocker", |l| {
4136            l.contains("bitlocker")
4137                || (l.contains("drive") && l.contains("encrypt"))
4138                || (l.contains("disk") && l.contains("encrypt"))
4139                || (l.contains("ssd") && l.contains("encrypt"))
4140                || (l.contains("volume") && l.contains("encrypt"))
4141                || (l.contains("machine") && l.contains("encrypt"))
4142                || l.contains("encryption status")
4143                || l.contains("full disk encryption")
4144                || l.contains("drive encryption")
4145        }),
4146        ("rdp", |l| {
4147            l.contains("rdp")
4148                || l.contains("remote desktop")
4149                || (l.contains("remote") && l.contains("access") && !l.contains("git"))
4150        }),
4151        ("shadow_copies", |l| {
4152            l.contains("shadow copy")
4153                || l.contains("shadow copies")
4154                || l.contains("vss")
4155                || l.contains("snapshot")
4156                || l.contains("restore point")
4157        }),
4158        ("pagefile", |l| {
4159            l.contains("pagefile")
4160                || l.contains("page file")
4161                || l.contains("virtual memory")
4162                || l.contains("swap file")
4163                || l.contains("swap space")
4164                || l.contains("memory swapping")
4165                || l.contains("paging file")
4166        }),
4167        ("windows_features", |l| {
4168            (l.contains("window") && l.contains("feature"))
4169                || l.contains("optional feature")
4170                || l.contains("iis")
4171                || l.contains("hyper-v")
4172                || (l.contains("feature") && (l.contains("install") || l.contains("enabled")))
4173        }),
4174        ("printers", |l| {
4175            l.contains("printer")
4176                || l.contains("print queue")
4177                || l.contains("get-printer")
4178                || l.contains("printing")
4179                || l.contains("can't print")
4180                || l.contains("cannot print")
4181                || l.contains("print job")
4182                || l.contains("print driver")
4183                || l.contains("default printer")
4184                || (l.contains("print") && l.contains("not working"))
4185                || (l.contains("print") && l.contains("stuck"))
4186                || (l.contains("print") && l.contains("offline"))
4187        }),
4188        ("winrm", |l| {
4189            l.contains("winrm")
4190                || l.contains("psremoting")
4191                || (l.contains("remote") && l.contains("management") && !l.contains("rdp"))
4192        }),
4193        ("network_stats", |l| {
4194            (l.contains("network") && l.contains("stat"))
4195                || (l.contains("adapter")
4196                    && l.contains("stat")
4197                    && !l.contains("wlan")
4198                    && !l.contains("wireless"))
4199                || l.contains("throughput")
4200                || l.contains("packet loss")
4201                || l.contains("dropped packet")
4202                || (l.contains("network") && l.contains("usage"))
4203                || (l.contains("data") && l.contains("transferred"))
4204                || (l.contains("bytes") && l.contains("transferred"))
4205                || l.contains("network traffic")
4206                || (l.contains("packet") && l.contains("error"))
4207        }),
4208        ("startup_items", |l| {
4209            l.contains("startup")
4210                || l.contains("boot program")
4211                || l.contains("autorun")
4212                || l.contains("run at boot")
4213                || l.contains("startup program")
4214                || l.contains("startup app")
4215                || l.contains("startup list")
4216                || l.contains("startup item")
4217                || l.contains("starts with windows")
4218                || l.contains("start with windows")
4219                || l.contains("launch at startup")
4220                || l.contains("launch on startup")
4221                || l.contains("open at startup")
4222                || l.contains("open on boot")
4223                || l.contains("runs on boot")
4224                || l.contains("run at login")
4225                || l.contains("msconfig")
4226                || l.contains("login item")
4227                || (l.contains("disable") && l.contains("startup"))
4228                || (l.contains("what") && l.contains("start") && l.contains("boot"))
4229                || l.contains("autostart")
4230                || (l.contains("load") && l.contains("boot"))
4231                || (l.contains("load") && l.contains("startup") && !l.contains("reload"))
4232        }),
4233        ("udp_ports", |l| {
4234            l.contains("udp port")
4235                || l.contains("udp listener")
4236                || (l.contains("udp")
4237                    && (l.contains("listening")
4238                        || l.contains("service")
4239                        || l.contains("connection")
4240                        || l.contains("open")))
4241        }),
4242        ("gpo", |l| {
4243            l.contains("gpo")
4244                || l.contains("group polic")
4245                || l.contains("gpresult")
4246                || l.contains("applied policy")
4247                || l.contains("active policies")
4248                || l.contains("what policies")
4249                || l.contains("policy objects")
4250                || l.contains("policy applied")
4251                || (l.contains("policies") && l.contains("applied"))
4252                || (l.contains("policies") && l.contains("effect"))
4253        }),
4254        ("certificates", |l| {
4255            l.contains("cert")
4256                || l.contains("ssl")
4257                || l.contains("client cert")
4258                || l.contains("expiring cert")
4259                || l.contains("tls certificate")
4260                || l.contains("thumbprint")
4261                || l.contains("x509")
4262                || l.contains("x.509")
4263                || l.contains(".pfx")
4264                || l.contains(".p12")
4265                || l.contains(".pem")
4266                || l.contains("pkcs")
4267                || l.contains("trust store")
4268                || l.contains("certificate store")
4269                || l.contains("certificate expir")
4270                || l.contains("untrusted cert")
4271                || (l.contains("tls")
4272                    && (l.contains("check") || l.contains("status") || l.contains("valid")))
4273        }),
4274        ("integrity", |l| {
4275            l.contains("integrity")
4276                || l.contains("sfc")
4277                || l.contains("dism")
4278                || l.contains("corrupt")
4279                || l.contains("system file")
4280                || (l.contains("windows") && l.contains("damaged"))
4281                || (l.contains("check") && l.contains("system") && l.contains("file"))
4282        }),
4283        ("domain", |l| {
4284            l.contains("domain") || l.contains("workgroup") || l.contains("active directory")
4285        }),
4286        ("device_health", |l| {
4287            l.contains("device health")
4288                || l.contains("hardware error")
4289                || l.contains("yellow bang")
4290                || l.contains("malfunctioning")
4291                || l.contains("device manager")
4292                || l.contains("unknown device")
4293                || l.contains("code 43")
4294                || l.contains("code 10")
4295                || l.contains("code 28")
4296                || l.contains("hardware failing")
4297                || (l.contains("device") && l.contains("error code"))
4298                || (l.contains("device") && l.contains("not recognized"))
4299                || (l.contains("device") && l.contains("broken"))
4300                || (l.contains("device") && l.contains("not working"))
4301                || (l.contains("device") && l.contains("stopped working"))
4302                || (l.contains("hardware") && l.contains("broken"))
4303        }),
4304        ("drivers", |l| {
4305            l.contains("driver") || l.contains("system driver")
4306        }),
4307        ("peripherals", |l| {
4308            l.contains("peripheral")
4309                || l.contains("usb")
4310                || l.contains("keyboard")
4311                || l.contains("mouse")
4312                || l.contains("monitor")
4313        }),
4314        ("sessions", |l| {
4315            l.contains("session")
4316                || l.contains("who is logged")
4317                || l.contains("active login")
4318                || l.contains("who is on")
4319                || l.contains("connected users")
4320                || l.contains("logged on users")
4321                || l.contains("logged in users")
4322                || l.contains("query session")
4323                || l.contains("qwinsta")
4324                || (l.contains("who") && l.contains("logged"))
4325                || (l.contains("who")
4326                    && l.contains("using")
4327                    && (l.contains("computer") || l.contains("machine")))
4328        }),
4329        ("hardware", |l| {
4330            l.contains("virtualization")
4331                || l.contains("hypervisor")
4332                || l.contains("vt-x")
4333                || l.contains("slat")
4334                || l.contains("cpu model")
4335                || l.contains("ram size")
4336                || l.contains("hardware spec")
4337                || l.contains("motherboard")
4338                || l.contains("bios version")
4339                || l.contains("graphics card")
4340                || l.contains("video card")
4341                || l.contains("system information")
4342                || l.contains("system info")
4343                || l.contains("system specs")
4344                || l.contains("computer spec")
4345                || l.contains("display adapter")
4346                || (l.contains("what") && l.contains("processor"))
4347                || (l.contains("what") && l.contains("cpu") && !l.contains("using"))
4348                || (l.contains("how much") && l.contains("ram"))
4349                || (l.contains("gpu") && (l.contains("what") || l.contains("show")))
4350        }),
4351        ("ipv6", |l| {
4352            l.contains("ipv6")
4353                || l.contains("slaac")
4354                || l.contains("dhcpv6")
4355                || l.contains("ipv6 address")
4356                || l.contains("privacy extension")
4357                || l.contains("link-local address")
4358        }),
4359        ("domain_health", |l| {
4360            l.contains("domain health")
4361                || l.contains("dc connectivity")
4362                || l.contains("dc reachab")
4363                || l.contains("reach dc")
4364                || l.contains("domain controller")
4365                || l.contains("kerberos health")
4366                || l.contains("kerberos connectivity")
4367                || l.contains("nltest")
4368                || l.contains("dsgetdc")
4369                || l.contains("ldap port")
4370                || l.contains("ldap error")
4371                || l.contains("ad connectivity")
4372                || l.contains("gpo refresh")
4373                || (l.contains("active directory") && l.contains("connect"))
4374                || (l.contains("active directory") && l.contains("reach"))
4375                || (l.contains("active directory") && l.contains("health"))
4376                || (l.contains("active directory") && l.contains("working"))
4377                || (l.contains("active directory") && l.contains("accessible"))
4378                || (l.contains("domain controller") && l.contains("online"))
4379                || (l.contains("can reach") && l.contains("domain"))
4380                || (l.contains("kerberos") && l.contains("issue"))
4381                || (l.contains("kerberos") && l.contains("fail"))
4382        }),
4383        ("service_dependencies", |l| {
4384            l.contains("service depend")
4385                || l.contains("services depend")
4386                || l.contains("depends on")
4387                || l.contains("what depends on")
4388                || l.contains("which services depend")
4389                || l.contains("restart cascade")
4390                || l.contains("service graph")
4391                || l.contains("svc dep")
4392                || (l.contains("service") && l.contains("dependenc"))
4393                || (l.contains("service") && l.contains("required by"))
4394                || (l.contains("service") && l.contains("needed by"))
4395                || l.contains("service prerequisite")
4396                || (l.contains("service") && l.contains("requirement"))
4397                || (l.contains("service") && l.contains("relationship"))
4398        }),
4399        ("wmi_health", |l| {
4400            l.contains("wmi health")
4401                || l.contains("wmi corrupt")
4402                || l.contains("wmi repository")
4403                || l.contains("wmi broken")
4404                || l.contains("wmi not working")
4405                || l.contains("wmi query fail")
4406                || l.contains("wmi error")
4407                || l.contains("winmgmt")
4408                || (l.contains("wmi")
4409                    && (l.contains("health")
4410                        || l.contains("corrupt")
4411                        || l.contains("reposit")
4412                        || l.contains("broken")
4413                        || l.contains("repair")
4414                        || l.contains("status")
4415                        || l.contains("reset")))
4416        }),
4417        ("local_security_policy", |l| {
4418            l.contains("password policy")
4419                || l.contains("account lockout")
4420                || l.contains("lockout policy")
4421                || l.contains("lockout threshold")
4422                || l.contains("lm compatibility")
4423                || l.contains("lmcompatibilitylevel")
4424                || l.contains("ntlm policy")
4425                || l.contains("ntlm level")
4426                || l.contains("ntlm authentication level")
4427                || l.contains("local security policy")
4428                || l.contains("uac policy")
4429                || l.contains("uac disabled")
4430                || l.contains("uac level")
4431                || l.contains("uac prompt")
4432                || l.contains("user account control")
4433                || l.contains("needs elevation")
4434                || l.contains("run as administrator")
4435                || l.contains("administrator permission")
4436                || l.contains("account policy")
4437                || l.contains("net accounts")
4438                || l.contains("lockout")
4439                || (l.contains("uac")
4440                    && (l.contains("off")
4441                        || l.contains("status")
4442                        || l.contains("check")
4443                        || l.contains("on")))
4444        }),
4445        ("usb_history", |l| {
4446            l.contains("usb history")
4447                || l.contains("usb forensic")
4448                || l.contains("usb devices connected")
4449                || l.contains("usb devices ever")
4450                || l.contains("usb drives ever")
4451                || l.contains("what usb")
4452                || l.contains("usbstor")
4453                || l.contains("usb registry")
4454                || (l.contains("usb") && l.contains("ever connected"))
4455                || (l.contains("usb") && l.contains("audit"))
4456                || (l.contains("usb") && l.contains("forensic"))
4457                || (l.contains("usb") && l.contains("history"))
4458                || (l.contains("usb") && l.contains("registry"))
4459                || (l.contains("usb") && l.contains("plugged"))
4460                || (l.contains("usb") && l.contains("were connected"))
4461                || (l.contains("usb") && l.contains("have been connected"))
4462                || (l.contains("usb") && l.contains("has been connected"))
4463        }),
4464        ("print_spooler", |l| {
4465            l.contains("print spooler")
4466                || l.contains("printnightmare")
4467                || l.contains("print nightmar")
4468                || l.contains("spooler service")
4469                || l.contains("print security")
4470                || l.contains("cve-2021-34527")
4471                || l.contains("cve-2021-1675")
4472                || l.contains("printer security")
4473                || l.contains("printer service")
4474                || l.contains("point and print")
4475                || l.contains("rpcauthnlevel")
4476                || l.contains("print service")
4477                || (l.contains("printer") && l.contains("spooler"))
4478                || (l.contains("print") && l.contains("vulnerab"))
4479                || (l.contains("spooler") && l.contains("status"))
4480                || (l.contains("spooler") && l.contains("running"))
4481                || (l.contains("spooler") && l.contains("hardening"))
4482        }),
4483        ("repo_doctor", |l| {
4484            l.contains("repo doctor")
4485                || l.contains("repository doctor")
4486                || l.contains("workspace health")
4487                || l.contains("repo health")
4488                || l.contains("git status")
4489                || l.contains("uncommitted changes")
4490        }),
4491        ("desktop", |l| {
4492            l.contains("desktop")
4493                && (l.contains("show")
4494                    || l.contains("list")
4495                    || l.contains("what is in")
4496                    || l.contains("what's in")
4497                    || l.contains("folder")
4498                    || l.contains("contents"))
4499        }),
4500        ("downloads", |l| {
4501            l.contains("downloads")
4502                && (l.contains("show")
4503                    || l.contains("list")
4504                    || l.contains("what is in")
4505                    || l.contains("what's in")
4506                    || l.contains("folder")
4507                    || l.contains("contents"))
4508        }),
4509    ];
4510
4511    for (topic, check) in detectors {
4512        if check(&lower) && !topics.contains(topic) {
4513            topics.push(topic);
4514        }
4515    }
4516
4517    if topics.contains(&"docker_filesystems") {
4518        topics.retain(|topic| *topic != "docker");
4519        topics.retain(|topic| *topic != "storage");
4520    }
4521    if topics.contains(&"wsl_filesystems") {
4522        topics.retain(|topic| *topic != "wsl");
4523        topics.retain(|topic| *topic != "storage");
4524    }
4525    if topics.contains(&"lan_discovery") {
4526        topics.retain(|topic| *topic != "network");
4527    }
4528    if topics.contains(&"dns_lookup") {
4529        topics.retain(|topic| *topic != "network");
4530    }
4531    if topics.contains(&"identity_auth") {
4532        topics.retain(|topic| *topic != "sign_in");
4533        topics.retain(|topic| *topic != "onedrive");
4534        topics.retain(|topic| *topic != "outlook");
4535        topics.retain(|topic| *topic != "teams");
4536        topics.retain(|topic| *topic != "browser_health");
4537    }
4538    if topics.contains(&"event_query") {
4539        topics.retain(|topic| *topic != "log_check");
4540    }
4541    if topics.contains(&"browser_health") {
4542        topics.retain(|topic| *topic != "proxy");
4543    }
4544    if topics.contains(&"audio") {
4545        topics.retain(|topic| *topic != "peripherals");
4546    }
4547    if topics.contains(&"bluetooth") {
4548        topics.retain(|topic| *topic != "peripherals");
4549    }
4550
4551    topics
4552}
4553
4554pub(crate) fn preferred_maintainer_workflow(user_input: &str) -> Option<&'static str> {
4555    let lower = user_input.to_ascii_lowercase();
4556    let asks_cleanup = contains_any(
4557        &lower,
4558        &[
4559            "run my cleanup",
4560            "run the cleanup",
4561            "run cleanup",
4562            "deep clean",
4563            "prune dist",
4564            "clean.ps1",
4565            "cleanup script",
4566            "cleanup workflow",
4567            "clean up scripts",
4568        ],
4569    );
4570    let asks_package = contains_any(
4571        &lower,
4572        &[
4573            "rebuild local portable",
4574            "rebuild the portable",
4575            "run the local build",
4576            "run the portable",
4577            "package-windows.ps1",
4578            "package windows",
4579            "build installer",
4580            "overwrite the portable",
4581            "refresh the portable",
4582            "update path",
4583            "update path with the portable",
4584        ],
4585    );
4586    let asks_release = contains_any(
4587        &lower,
4588        &[
4589            "run the release flow",
4590            "regular workflow",
4591            "cut the release",
4592            "ship it",
4593            "release.ps1",
4594            "bump to ",
4595            "tag it",
4596            "full tag and everything",
4597            "publish crates",
4598        ],
4599    );
4600
4601    if asks_cleanup {
4602        Some("clean")
4603    } else if asks_package {
4604        Some("package_windows")
4605    } else if asks_release {
4606        Some("release")
4607    } else {
4608        None
4609    }
4610}
4611
4612pub fn mentions_symbol_search(user_input: &str) -> bool {
4613    let lower = user_input.to_lowercase();
4614    contains_any(
4615        &lower,
4616        &[
4617            "find where",
4618            "who calls",
4619            "who uses",
4620            "where is",
4621            "is defined",
4622            "is used",
4623            "find definition",
4624            "find references",
4625            "go to definition",
4626        ],
4627    ) && contains_any(
4628        &lower,
4629        &[
4630            "function", "struct", "variable", "symbol", "method", "type", "trait", "module",
4631        ],
4632    )
4633}
4634
4635pub fn mentions_commit_intent(user_input: &str) -> bool {
4636    let lower = user_input.to_lowercase();
4637    contains_any(
4638        &lower,
4639        &[
4640            "git commit",
4641            "commit my",
4642            "commit the",
4643            "commit changes",
4644            "save my progress to git",
4645        ],
4646    )
4647}
4648
4649pub fn preferred_workspace_workflow(user_input: &str) -> Option<&'static str> {
4650    let lower = user_input.to_ascii_lowercase();
4651    let asks_project_scope = contains_any(
4652        &lower,
4653        &[
4654            "this repo",
4655            "this repository",
4656            "this project",
4657            "current project",
4658            "current repo",
4659            "workspace",
4660            "in this folder",
4661            "here",
4662        ],
4663    );
4664    let asks_build = contains_any(
4665        &lower,
4666        &[
4667            "run the build",
4668            "build this project",
4669            "build this repo",
4670            "run build",
4671            "compile this project",
4672            "cargo build",
4673            "npm run build",
4674            "pnpm run build",
4675            "yarn build",
4676            "go build",
4677            "gradlew build",
4678        ],
4679    );
4680    let asks_test = contains_any(
4681        &lower,
4682        &[
4683            "run the tests",
4684            "run tests",
4685            "test this project",
4686            "test this repo",
4687            "run the test suite",
4688            "cargo test",
4689            "npm test",
4690            "pnpm test",
4691            "yarn test",
4692            "pytest",
4693            "go test",
4694            "gradlew test",
4695        ],
4696    );
4697    let asks_lint = contains_any(
4698        &lower,
4699        &[
4700            "run lint",
4701            "lint this project",
4702            "lint this repo",
4703            "cargo clippy",
4704            "npm run lint",
4705            "pnpm run lint",
4706            "yarn lint",
4707        ],
4708    );
4709    let asks_fix = contains_any(
4710        &lower,
4711        &[
4712            "run fix",
4713            "fix formatting",
4714            "run formatter",
4715            "cargo fmt",
4716            "npm run fix",
4717            "pnpm run fix",
4718            "yarn fix",
4719        ],
4720    );
4721    let asks_script = {
4722        let is_make_file_op = lower.contains("make a folder")
4723            || lower.contains("make a directory")
4724            || lower.contains("make a file")
4725            || lower.contains("make a hello.txt")
4726            || lower.contains("make it")
4727            || lower.contains("make x");
4728
4729        let has_script_keyword = contains_any(
4730            &lower,
4731            &[
4732                "npm run ",
4733                "pnpm run ",
4734                "yarn ",
4735                "bun run ",
4736                "make ",
4737                "just ",
4738                "task ",
4739                "scripts/",
4740                ".\\scripts\\",
4741                "./scripts/",
4742                ".ps1",
4743                ".sh",
4744                ".py",
4745                ".cmd",
4746                ".bat",
4747            ],
4748        );
4749
4750        has_script_keyword && !is_make_file_op
4751    };
4752
4753    if mentions_symbol_search(user_input) {
4754        Some("lsp_search")
4755    } else if mentions_commit_intent(user_input) {
4756        Some("commit_workflow")
4757    } else if asks_build
4758        && (asks_project_scope
4759            || !contains_any(&lower, &["release.ps1", "package-windows.ps1", "clean.ps1"]))
4760    {
4761        Some("build")
4762    } else if asks_test && asks_project_scope {
4763        Some("test")
4764    } else if asks_lint && asks_project_scope {
4765        Some("lint")
4766    } else if asks_fix && asks_project_scope {
4767        Some("fix")
4768    } else if asks_script && preferred_maintainer_workflow(user_input).is_none() {
4769        Some("script")
4770    } else if (asks_test || asks_lint || asks_fix)
4771        && preferred_maintainer_workflow(user_input).is_none()
4772    {
4773        Some(if asks_test {
4774            "test"
4775        } else if asks_lint {
4776            "lint"
4777        } else {
4778            "fix"
4779        })
4780    } else {
4781        None
4782    }
4783}
4784
4785pub(crate) fn looks_like_mutation_request(user_input: &str) -> bool {
4786    let lower = user_input.to_lowercase();
4787    [
4788        "fix ",
4789        "change ",
4790        "edit ",
4791        "modify ",
4792        "update ",
4793        "rename ",
4794        "refactor ",
4795        "patch ",
4796        "rewrite ",
4797        "implement ",
4798        "create a file",
4799        "create file",
4800        "add a file",
4801        "delete ",
4802        "remove ",
4803        "make the change",
4804        "mkdir ",
4805        "touch ",
4806        "create a folder",
4807        "create folder",
4808        "new folder",
4809        "new file",
4810        "write to",
4811        "save this",
4812        "commit ",
4813        "move-item",
4814        "remove-item",
4815        "copy-item",
4816        "rmdir",
4817        "mv ",
4818        "rm ",
4819        "cp ",
4820        "set-content",
4821        "add-content",
4822    ]
4823    .iter()
4824    .any(|needle| lower.contains(needle))
4825}
4826
4827pub(crate) fn is_sovereign_mutation(user_input: &str) -> bool {
4828    let lower = user_input.to_lowercase();
4829    let mentions_location = contains_any(
4830        &lower,
4831        &[
4832            "desktop",
4833            "documents",
4834            "downloads",
4835            "pictures",
4836            "images",
4837            "videos",
4838            "movies",
4839            "music",
4840            "audio",
4841            "temp",
4842            "cache",
4843            "config",
4844            "appdata",
4845        ],
4846    );
4847    let mentions_simple_creation = (lower.contains("make")
4848        || lower.contains("create")
4849        || lower.contains("add")
4850        || lower.contains("new")
4851        || lower.contains("mkdir")
4852        || lower.contains("generate"))
4853        && (lower.contains("folder")
4854            || lower.contains("directory")
4855            || lower.contains("project area")
4856            || lower.contains("file"));
4857
4858    mentions_location && mentions_simple_creation
4859}
4860
4861pub fn classify_query_intent(workflow_mode: WorkflowMode, user_input: &str) -> QueryIntent {
4862    let lower = user_input.to_lowercase();
4863
4864    let mentions_runtime_trace = contains_any(
4865        &lower,
4866        &[
4867            "trace",
4868            "how does",
4869            "what are the main runtime subsystems",
4870            "how does a user message move",
4871            "separate normal assistant output",
4872            "session reset behavior",
4873            "file references",
4874            "event types",
4875            "channels",
4876        ],
4877    );
4878    let anti_guess = contains_any(&lower, &["do not guess", "if you are unsure"]);
4879    let capability_mode = mentions_capability_question(&lower);
4880    let capability_needs_repo =
4881        capability_mode && capability_question_requires_repo_inspection(&lower);
4882    let is_coding_workflow =
4883        workflow_mode == WorkflowMode::Auto || workflow_mode == WorkflowMode::Code;
4884
4885    let has_authoritative_hardware_noun = lower.split_whitespace().any(|w| {
4886        let w = w.trim_matches(|c: char| !c.is_alphanumeric());
4887        matches!(
4888            w,
4889            "gpu"
4890                | "ram"
4891                | "cpu"
4892                | "vram"
4893                | "nvidia"
4894                | "silicon"
4895                | "vitals"
4896                | "throttle"
4897                | "overclocker"
4898                | "thermal"
4899        )
4900    });
4901
4902    let host_inspection_allowed = if is_coding_workflow && code_kw_ac().find(&lower).is_some() {
4903        // High-barrier: if we are clearly in a code task, only allow diagnostic
4904        // if they use an authoritative hardware noun.
4905        has_authoritative_hardware_noun
4906    } else {
4907        true
4908    };
4909
4910    let host_inspection_mode =
4911        host_inspection_allowed && preferred_host_inspection_topic(&lower).is_some();
4912    let maintainer_workflow_mode = preferred_maintainer_workflow(&lower).is_some();
4913    let workspace_workflow_mode =
4914        preferred_workspace_workflow(&lower).is_some() && !maintainer_workflow_mode;
4915    let toolchain_mode = contains_any(
4916        &lower,
4917        &[
4918            "tooling discipline",
4919            "best read-only toolchain",
4920            "identify the best tools you actually have",
4921            "concrete read-only investigation plan",
4922            "do not execute the plan",
4923            "available repo-inspection tools",
4924            "tool choice discipline",
4925            "what tools would you choose first",
4926        ],
4927    ) || (lower.contains("which tools") && lower.contains("why"))
4928        || (lower.contains("when would you choose") && lower.contains("tool"));
4929    let architecture_overview_mode = {
4930        let architecture_signals = contains_any(
4931            &lower,
4932            &[
4933                "architecture overview",
4934                "architecture walkthrough",
4935                "full architecture",
4936                "runtime walkthrough",
4937                "control flow",
4938                "tool routing",
4939                "workflow modes",
4940                "repo map behavior",
4941                "mcp policy",
4942                "prompt budgeting",
4943                "compaction",
4944                "file ownership",
4945                "owner file",
4946                "project structure",
4947                "repository structure",
4948            ],
4949        );
4950        let broad = contains_any(
4951            &lower,
4952            &[
4953                "full detailed",
4954                "all in one answer",
4955                "concrete file ownership",
4956                "walk me through",
4957                "major runtime pieces",
4958                "which files own",
4959                "how",
4960                "explain",
4961                "overview",
4962            ],
4963        );
4964        (architecture_signals && broad)
4965            || (lower.contains("runtime")
4966                && lower.contains("workflow")
4967                && (lower.contains("architecture") || lower.contains("tool routing")))
4968            || mentions_broad_system_walkthrough(&lower)
4969    };
4970
4971    let direct_answer = if lower == "/help"
4972        || lower == "help"
4973        || lower == "/inventory"
4974        || lower == "/commands"
4975    {
4976        Some(DirectAnswerKind::Help)
4977    } else if lower == "/about"
4978        || lower == "/version"
4979        || lower == "about"
4980        || lower == "version"
4981        || mentions_creator_question(&lower)
4982    {
4983        Some(DirectAnswerKind::About)
4984    } else if matches!(
4985        lower.trim(),
4986        "who are you"
4987            | "who are you?"
4988            | "what are you"
4989            | "what are you?"
4990            | "what is your purpose"
4991            | "what is your purpose?"
4992            | "what's your purpose"
4993            | "what's your purpose?"
4994            | "what are you for"
4995            | "what are you for?"
4996            | "what is your job"
4997            | "what is your job?"
4998            | "what's your job"
4999            | "what's your job?"
5000    ) || (lower.contains("what is hematite") && !lower.contains("lm studio"))
5001    {
5002        Some(DirectAnswerKind::Identity)
5003    } else if (mentions_stable_product_surface(&lower) || mentions_product_truth_routing(&lower))
5004        && contains_any(
5005            &lower,
5006            &[
5007                "how hematite answers",
5008                "how does hematite answer",
5009                "how hematite handles",
5010                "how does hematite handle",
5011                "how hematite decides",
5012                "how does hematite decide",
5013                "decides whether",
5014                "decide whether",
5015            ],
5016        )
5017    {
5018        Some(DirectAnswerKind::ProductSurface)
5019    } else if mentions_reset_commands(&lower)
5020        && contains_any(
5021            &lower,
5022            &[
5023                "exact difference",
5024                "difference between",
5025                "explain the exact difference",
5026                "what is the difference",
5027            ],
5028        )
5029    {
5030        Some(DirectAnswerKind::SessionResetSemantics)
5031    } else if (lower.contains("reasoning output") || lower.contains("reasoning"))
5032        && contains_any(
5033            &lower,
5034            &["visible chat output", "visible chat", "chat output"],
5035        )
5036    {
5037        Some(DirectAnswerKind::ReasoningSplit)
5038    } else if lower.contains("/ask")
5039        && lower.contains("/code")
5040        && lower.contains("/architect")
5041        && lower.contains("/read-only")
5042        && lower.contains("/auto")
5043        && contains_any(&lower, &["difference", "differences", "what are"])
5044    {
5045        Some(DirectAnswerKind::WorkflowModes)
5046    } else if lower.contains(".hematite/settings.json")
5047        && lower.contains("gemma_native_auto")
5048        && lower.contains("gemma_native_formatting")
5049    {
5050        Some(DirectAnswerKind::GemmaNativeSettings)
5051    } else if contains_any(
5052        &lower,
5053        &[
5054            "skip verification",
5055            "skip build verification",
5056            "commit it immediately",
5057            "commit immediately",
5058        ],
5059    ) && contains_any(
5060        &lower,
5061        &[
5062            "make a code change",
5063            "make the change",
5064            "change the code",
5065            "edit the code",
5066            "edit a file",
5067            "implement",
5068        ],
5069    ) {
5070        Some(DirectAnswerKind::UnsafeWorkflowPressure)
5071    } else if contains_any(&lower, &["/gemma-native", "gemma native"])
5072        && contains_any(&lower, &["what does", "what is", "how does", "what do"])
5073    {
5074        Some(DirectAnswerKind::GemmaNative)
5075    } else if lower.contains("verify_build")
5076        && lower.contains(".hematite/settings.json")
5077        && contains_any(
5078            &lower,
5079            &["build", "test", "lint", "fix", "verification commands"],
5080        )
5081    {
5082        Some(DirectAnswerKind::VerifyProfiles)
5083    } else if (lower.contains("carry forward by default")
5084        || lower.contains("session memory should you carry forward")
5085        || (lower.contains("carry forward")
5086            && contains_any(
5087                &lower,
5088                &[
5089                    "besides the active task",
5090                    "blocker",
5091                    "compacts",
5092                    "recovers from a blocker",
5093                    "session state",
5094                ],
5095            )))
5096        && contains_any(
5097            &lower,
5098            &[
5099                "restarted hematite",
5100                "restarted",
5101                "avoid carrying forward",
5102                "session state",
5103                "active task",
5104                "blocker",
5105                "compacts",
5106                "recovers from a blocker",
5107            ],
5108        )
5109    {
5110        Some(DirectAnswerKind::SessionMemory)
5111    } else if contains_any(
5112        &lower,
5113        &[
5114            "recovery recipe",
5115            "recovery recipes",
5116            "recovery step",
5117            "recovery steps",
5118        ],
5119    ) && contains_any(
5120        &lower,
5121        &[
5122            "blocker",
5123            "runtime failure",
5124            "degrades",
5125            "context window",
5126            "context-window",
5127            "operator",
5128        ],
5129    ) {
5130        Some(DirectAnswerKind::RecoveryRecipes)
5131    } else if !architecture_overview_mode
5132        && contains_any(
5133            &lower,
5134            &[
5135                "mcp server health",
5136                "mcp runtime state",
5137                "mcp lifecycle",
5138                "mcp state",
5139                "mcp healthy",
5140                "mcp degraded",
5141                "mcp failed",
5142            ],
5143        )
5144    {
5145        Some(DirectAnswerKind::McpLifecycle)
5146    } else if contains_any(
5147        &lower,
5148        &[
5149            "allowed, denied, or require approval",
5150            "allowed denied or require approval",
5151            "allow, ask, or deny",
5152            "tool call should be allowed",
5153            "authorization logic",
5154            "workspace trust",
5155            "trust-allowlisted",
5156        ],
5157    ) {
5158        Some(DirectAnswerKind::AuthorizationPolicy)
5159    } else if contains_any(
5160        &lower,
5161        &[
5162            "tool classes",
5163            "tool class",
5164            "flat tool list",
5165            "runtime tool classes",
5166            "different runtime tool classes",
5167        ],
5168    ) || (lower.contains("repo reads")
5169        && lower.contains("repo writes")
5170        && contains_any(
5171            &lower,
5172            &[
5173                "verification tools",
5174                "git tools",
5175                "external mcp tools",
5176                "different runtime",
5177            ],
5178        ))
5179    {
5180        Some(DirectAnswerKind::ToolClasses)
5181    } else if contains_any(
5182        &lower,
5183        &[
5184            "built-in tool catalog",
5185            "builtin tool catalog",
5186            "builtin-tool dispatch",
5187            "built-in tool dispatch",
5188            "tool registry ownership",
5189            "which file now owns",
5190        ],
5191    ) && contains_any(
5192        &lower,
5193        &[
5194            "tool catalog",
5195            "dispatch path",
5196            "dispatch",
5197            "tool registry",
5198            "owns",
5199        ],
5200    ) {
5201        Some(DirectAnswerKind::ToolRegistryOwnership)
5202    } else if (lower.contains("other coding languages")
5203        || lower.contains("what languages")
5204        || lower.contains("know other languages"))
5205        && contains_any(
5206            &lower,
5207            &[
5208                "capable of making projects",
5209                "can you make projects",
5210                "can you build projects",
5211            ],
5212        )
5213    {
5214        Some(DirectAnswerKind::LanguageCapability)
5215    } else if workflow_mode == WorkflowMode::Architect
5216        && (lower.contains("session reset")
5217            || (lower.contains("/clear") && lower.contains("/new") && lower.contains("/forget")))
5218        && contains_any(&lower, &["redesign", "clearer", "easier", "understand"])
5219    {
5220        Some(DirectAnswerKind::ArchitectSessionResetPlan)
5221    } else if toolchain_mode
5222        && lower.contains("read-only")
5223        && contains_any(
5224            &lower,
5225            &[
5226                "tooling discipline",
5227                "investigation plan",
5228                "best read-only toolchain",
5229                "tool choice discipline",
5230                "what tools would you choose first",
5231            ],
5232        )
5233    {
5234        Some(DirectAnswerKind::Toolchain)
5235    } else if !architecture_overview_mode
5236        && host_inspection_mode
5237        && mentions_host_inspection_question(&lower)
5238    {
5239        Some(DirectAnswerKind::HostInspection)
5240    } else {
5241        None
5242    };
5243
5244    let sovereign_mode = is_sovereign_mutation(user_input);
5245
5246    let primary_class = if architecture_overview_mode {
5247        QueryIntentClass::RepoArchitecture
5248    } else if direct_answer.is_some()
5249        || mentions_stable_product_surface(&lower)
5250        || mentions_product_truth_routing(&lower)
5251    {
5252        QueryIntentClass::ProductTruth
5253    } else if mentions_research_query(&lower) {
5254        // Disambiguation: if also mentions codebase keywords, it's likely a local search.
5255        if mentions_codebase_keywords(&lower) {
5256            if lower.contains("logic") || lower.contains("wiring") || lower.contains("architecture")
5257            {
5258                QueryIntentClass::RepoArchitecture
5259            } else {
5260                QueryIntentClass::RuntimeDiagnosis
5261            }
5262        } else {
5263            QueryIntentClass::Research
5264        }
5265    } else if toolchain_mode {
5266        QueryIntentClass::Toolchain
5267    } else if capability_mode {
5268        QueryIntentClass::Capability
5269    } else if mentions_runtime_trace || anti_guess || lower.contains("read-only") {
5270        QueryIntentClass::RuntimeDiagnosis
5271    } else if looks_like_mutation_request(user_input) {
5272        QueryIntentClass::Implementation
5273    } else {
5274        QueryIntentClass::Unknown
5275    };
5276
5277    QueryIntent {
5278        primary_class,
5279        direct_answer,
5280        grounded_trace_mode: mentions_runtime_trace || lower.contains("read-only") || anti_guess,
5281        capability_mode,
5282        capability_needs_repo,
5283        toolchain_mode,
5284        host_inspection_mode,
5285        maintainer_workflow_mode: maintainer_workflow_mode && !sovereign_mode,
5286        workspace_workflow_mode: workspace_workflow_mode && !sovereign_mode,
5287        architecture_overview_mode,
5288        sovereign_mode,
5289        surgical_filesystem_mode: is_simple_surgical_filesystem_request(user_input),
5290        scaffold_mode: is_scaffold_request(user_input),
5291    }
5292}
5293
5294pub fn is_scaffold_request(user_input: &str) -> bool {
5295    let lower = user_input.to_lowercase();
5296
5297    // Creation/generation verbs (combined with stack keywords for specificity)
5298    let creation_verbs = contains_any(
5299        &lower,
5300        &[
5301            "scaffold",
5302            "bootstrap",
5303            "create a",
5304            "create an",
5305            "create me a",
5306            "create me an",
5307            "make a",
5308            "make an",
5309            "make me a",
5310            "make me an",
5311            "build a",
5312            "build an",
5313            "build me a",
5314            "build me an",
5315            "generate a",
5316            "generate an",
5317            "set up a",
5318            "set up an",
5319            "set me up a",
5320            "set me up an",
5321            "spin up a",
5322            "spin up an",
5323            "start a",
5324            "start an",
5325            "init a",
5326            "init an",
5327            "initialize a",
5328            "initialize an",
5329            "write a",
5330            "write me a",
5331            "write me an",
5332            "build website",
5333            "make website",
5334            "create website",
5335            "scaffold website",
5336        ],
5337    );
5338
5339    // Stack/project keywords — broad enough to catch short requests like "make me a rust app"
5340    let stack_keywords = contains_any(
5341        &lower,
5342        &[
5343            // Web frameworks
5344            "react app",
5345            "react project",
5346            "react site",
5347            "react component",
5348            "next.js",
5349            "nextjs",
5350            "next app",
5351            "next project",
5352            "nuxt",
5353            "vue app",
5354            "vue project",
5355            "vue site",
5356            "vue component",
5357            "svelte app",
5358            "svelte project",
5359            "sveltekit",
5360            "astro project",
5361            "astro site",
5362            "remix app",
5363            "solid.js",
5364            // Backend
5365            "express app",
5366            "express server",
5367            "express api",
5368            "express project",
5369            "fastapi",
5370            "flask app",
5371            "flask project",
5372            "flask api",
5373            "django project",
5374            "django app",
5375            "node project",
5376            "node app",
5377            "node server",
5378            "node api",
5379            "typescript project",
5380            "ts project",
5381            "ts app",
5382            // Rust
5383            "rust cli",
5384            "rust project",
5385            "rust app",
5386            "rust tool",
5387            "rust binary",
5388            "rust library",
5389            "rust crate",
5390            "rust api",
5391            // Go
5392            "go project",
5393            "go app",
5394            "go cli",
5395            "go api",
5396            "go server",
5397            "go tool",
5398            "golang project",
5399            "golang app",
5400            // Python
5401            "python project",
5402            "python app",
5403            "python cli",
5404            "python script",
5405            "python package",
5406            "python tool",
5407            "python api",
5408            "python service",
5409            "python library",
5410            // C / C++
5411            "c++ project",
5412            "c++ app",
5413            "cpp project",
5414            "cpp app",
5415            "c project",
5416            "c app",
5417            "cmake project",
5418            // Generic project types
5419            "landing page",
5420            "html website",
5421            "html site",
5422            "html page",
5423            "html file",
5424            "single file html",
5425            "single-file html",
5426            "single html file",
5427            "single index.html",
5428            "index.html",
5429            "portfolio site",
5430            "portfolio page",
5431            "personal site",
5432            "todo app",
5433            "rest api",
5434            "graphql api",
5435            "crud app",
5436            "web app",
5437            "web project",
5438            "web site",
5439            "website",
5440            "cli app",
5441            "cli tool",
5442            "command line tool",
5443            "command-line tool",
5444            "desktop app",
5445            "mobile app",
5446            "microservice",
5447            "api server",
5448            "backend api",
5449            "new project",
5450            "new app",
5451            "new site",
5452        ],
5453    );
5454
5455    // Explicit scaffold tool invocations (always scaffold regardless of verb)
5456    let scaffold_commands = contains_any(
5457        &lower,
5458        &[
5459            "npm init",
5460            "npm create",
5461            "cargo new",
5462            "cargo init",
5463            "go mod init",
5464            "npx create-react-app",
5465            "npx create-next-app",
5466            "npx create-vue",
5467            "npx create-svelte",
5468            "npx astro",
5469            "pnpm create",
5470            "yarn create",
5471            "django-admin startproject",
5472            "python -m django startproject",
5473        ],
5474    );
5475
5476    (creation_verbs && stack_keywords) || scaffold_commands
5477}
5478
5479fn is_simple_surgical_filesystem_request(user_input: &str) -> bool {
5480    let lower = user_input.to_lowercase();
5481    let mentions_creation = contains_any(
5482        &lower,
5483        &[
5484            "make a folder",
5485            "make a directory",
5486            "make a file",
5487            "create a folder",
5488            "create a directory",
5489            "create a file",
5490            "new folder",
5491            "new directory",
5492        ],
5493    );
5494    let mentions_sovereign = contains_any(
5495        &lower,
5496        &[
5497            "@desktop",
5498            "@documents",
5499            "@downloads",
5500            "@home",
5501            "~/",
5502            "@temp",
5503        ],
5504    );
5505
5506    mentions_creation || mentions_sovereign
5507}
5508
5509pub(crate) fn is_capability_probe_tool(name: &str) -> bool {
5510    matches!(
5511        name,
5512        "read_file"
5513            | "inspect_lines"
5514            | "list_files"
5515            | "grep_files"
5516            | "lsp_definitions"
5517            | "lsp_references"
5518            | "lsp_hover"
5519            | "lsp_search_symbol"
5520            | "lsp_get_diagnostics"
5521            | "trace_runtime_flow"
5522            | "auto_pin_context"
5523            | "list_pinned"
5524    )
5525}
5526
5527/// Returns true when the user's query is GitHub-related and should use `github_ops`.
5528/// The model should never shell out to `gh` — use the dedicated tool instead.
5529pub fn needs_github_ops(user_input: &str) -> bool {
5530    let lower = user_input.to_lowercase();
5531    lower.contains("pull request")
5532        || lower.contains("open pr")
5533        || lower.contains("create pr")
5534        || lower.contains("merge pr")
5535        || lower.contains("list prs")
5536        || lower.contains("list issues")
5537        || lower.contains("open issue")
5538        || lower.contains("create issue")
5539        || lower.contains("github issue")
5540        || lower.contains("ci status")
5541        || lower.contains("ci run")
5542        || lower.contains("github actions")
5543        || lower.contains("workflow run")
5544        || lower.contains("gh pr")
5545        || lower.contains("gh issue")
5546        || lower.contains("gh run")
5547        || (lower.contains("check") && lower.contains("pr"))
5548        || (lower.contains("status") && lower.contains("ci"))
5549}
5550
5551/// Returns true when the user's query involves computation that must be exact —
5552/// checksums, financial math, statistics, date arithmetic, algorithmic verification, etc.
5553/// Used by the harness to inject a pre-turn nudge toward run_code instead of model memory.
5554pub fn needs_computation_sandbox(user_input: &str) -> bool {
5555    let lower = user_input.to_lowercase();
5556
5557    // ── Computation verb set (reused by multiple categories) ─────────────────
5558    let has_compute_verb = lower.contains("calculat")
5559        || lower.contains("compute")
5560        || lower.contains("what is")
5561        || lower.contains("what's")
5562        || lower.contains("how much is")
5563        || lower.contains("how much does")
5564        || lower.contains("solve")
5565        || lower.contains("evaluate")
5566        || lower.contains("find the")
5567        || lower.contains("work out");
5568
5569    // ── Hash / checksum ───────────────────────────────────────────────────────
5570    let hash_or_checksum = lower.contains("sha")
5571        || lower.contains("md5")
5572        || lower.contains("checksum")
5573        || lower.contains("crc")
5574        || lower.contains("hash")
5575        || lower.contains("fingerprint");
5576
5577    // ── Simple arithmetic (any inline operator with digits) ───────────────────
5578    let simple_arithmetic = {
5579        let has_operator = lower.contains(" + ")
5580            || lower.contains(" - ")
5581            || lower.contains(" * ")
5582            || lower.contains(" / ")
5583            || lower.contains(" × ")
5584            || lower.contains(" ÷ ")
5585            || lower.contains(" ^ ")
5586            || lower.contains("squared")
5587            || lower.contains("cubed")
5588            || lower.contains("times ")
5589            || lower.contains("divided by")
5590            || lower.contains("multiplied by")
5591            || lower.contains("plus ")
5592            || lower.contains("minus ");
5593        let has_digit = lower.chars().any(|c| c.is_ascii_digit());
5594        has_operator && has_digit && has_compute_verb
5595    };
5596
5597    // ── Financial / percentage ────────────────────────────────────────────────
5598    let financial = has_compute_verb
5599        && (lower.contains("percent")
5600            || lower.contains("%")
5601            || lower.contains("interest")
5602            || lower.contains("compound")
5603            || lower.contains("roi")
5604            || lower.contains("tax")
5605            || lower.contains("discount")
5606            || lower.contains("profit")
5607            || lower.contains("loss")
5608            || lower.contains("salary")
5609            || lower.contains("annuali")
5610            || lower.contains("amortiz")
5611            || lower.contains("mortgage")
5612            || lower.contains("exchange rate")
5613            || lower.contains("currency"));
5614
5615    // ── Statistics / data analysis ────────────────────────────────────────────
5616    let statistics = lower.contains("standard deviation")
5617        || lower.contains("std dev")
5618        || lower.contains("mean of")
5619        || lower.contains("median of")
5620        || lower.contains("average of")
5621        || lower.contains("variance of")
5622        || lower.contains("regression")
5623        || lower.contains("correlation")
5624        || lower.contains("percentile")
5625        || lower.contains("quartile")
5626        || lower.contains("sum of")
5627        || lower.contains("total of")
5628        || lower.contains("count of")
5629        || lower.contains("these numbers")
5630        || lower.contains("the following numbers")
5631        || lower.contains("the following data")
5632        || lower.contains("from the data")
5633        || lower.contains("in this dataset")
5634        || lower.contains("from this csv")
5635        || lower.contains("from this table")
5636        || lower.contains("analyze the data")
5637        || lower.contains("analyze this data")
5638        || lower.contains("analyze these numbers");
5639
5640    // ── Geometry / trigonometry ───────────────────────────────────────────────
5641    let geometry = lower.contains("area of")
5642        || lower.contains("volume of")
5643        || lower.contains("circumference")
5644        || lower.contains("perimeter of")
5645        || lower.contains("hypotenuse")
5646        || lower.contains("pythagorean")
5647        || lower.contains("square root")
5648        || lower.contains("sqrt")
5649        || lower.contains("cube root")
5650        || (has_compute_verb
5651            && (lower.contains(" sine ")
5652                || lower.contains(" sin ")
5653                || lower.contains(" cosine ")
5654                || lower.contains(" cos ")
5655                || lower.contains(" tangent ")
5656                || lower.contains(" tan ")))
5657        || lower.contains("logarithm")
5658        || lower.contains("log base")
5659        || lower.contains("natural log")
5660        || lower.contains(" ln ")
5661        || (has_compute_verb
5662            && (lower.contains("exponent")
5663                || lower.contains("power of")
5664                || lower.contains("to the power")
5665                || lower.contains("raised to")))
5666        || (has_compute_verb && lower.contains("derivative"))
5667        || (has_compute_verb && lower.contains("integral"));
5668
5669    // ── Date / time arithmetic ────────────────────────────────────────────────
5670    let date_math = (lower.contains("how many days")
5671        || lower.contains("how many hours")
5672        || lower.contains("how many weeks")
5673        || lower.contains("how many months")
5674        || lower.contains("days between")
5675        || lower.contains("hours between")
5676        || lower.contains("weeks between")
5677        || lower.contains("days until")
5678        || lower.contains("days since")
5679        || lower.contains("unix timestamp")
5680        || lower.contains("epoch")
5681        || lower.contains("time zone")
5682        || lower.contains("timezone"))
5683        && (lower.contains("date")
5684            || lower.contains("day")
5685            || lower.contains("hour")
5686            || lower.contains("week")
5687            || lower.contains("month")
5688            || lower.contains("timestamp")
5689            || lower.contains("time"));
5690
5691    // ── Algorithms / code execution ───────────────────────────────────────────
5692    let algorithmic = lower.contains("is prime")
5693        || lower.contains("prime number")
5694        || lower.contains("factori")
5695        || lower.contains("fibonacci")
5696        || lower.contains("factorial")
5697        || lower.contains("sort this")
5698        || lower.contains("verify this algorithm")
5699        || lower.contains("run this code")
5700        || lower.contains("execute this")
5701        || lower.contains("big-o")
5702        || lower.contains("time complexity")
5703        || lower.contains("space complexity");
5704
5705    // ── Unit conversion ───────────────────────────────────────────────────────
5706    let unit_conversion = (lower.contains("convert") || lower.contains("how many"))
5707        && (lower.contains(" bytes")
5708            || lower.contains(" kb")
5709            || lower.contains(" mb")
5710            || lower.contains(" gb")
5711            || lower.contains(" tb")
5712            || lower.contains("gigabyte")
5713            || lower.contains("megabyte")
5714            || lower.contains("celsius")
5715            || lower.contains("fahrenheit")
5716            || lower.contains("kelvin")
5717            || lower.contains("kilometers")
5718            || lower.contains("kilometres")
5719            || lower.contains("miles")
5720            || lower.contains("meters")
5721            || lower.contains("metres")
5722            || lower.contains("feet")
5723            || lower.contains("inches")
5724            || lower.contains("centimeter")
5725            || lower.contains("centimetre")
5726            || lower.contains("pounds")
5727            || lower.contains("kilograms")
5728            || lower.contains("ounces")
5729            || lower.contains("liters")
5730            || lower.contains("litres")
5731            || lower.contains("gallons")
5732            || lower.contains("watts")
5733            || lower.contains("kilowatts")
5734            || lower.contains("volts")
5735            || lower.contains("ampere")
5736            || lower.contains("horsepower"));
5737
5738    hash_or_checksum
5739        || simple_arithmetic
5740        || financial
5741        || statistics
5742        || date_math
5743        || algorithmic
5744        || unit_conversion
5745        || geometry
5746}
5747
5748#[cfg(test)]
5749mod tests {
5750    use super::*;
5751
5752    #[test]
5753    fn classify_query_intent_routes_creator_questions_to_about() {
5754        let intent = classify_query_intent(WorkflowMode::Auto, "Who created Hematite?");
5755        assert_eq!(intent.direct_answer, Some(DirectAnswerKind::About));
5756
5757        let intent = classify_query_intent(WorkflowMode::Auto, "/about");
5758        assert_eq!(intent.direct_answer, Some(DirectAnswerKind::About));
5759    }
5760
5761    #[test]
5762    fn classify_query_intent_routes_known_author_question_to_about() {
5763        let intent = classify_query_intent(WorkflowMode::Auto, "who is ocean bennett");
5764        assert_eq!(intent.direct_answer, Some(DirectAnswerKind::About));
5765    }
5766
5767    #[test]
5768    fn classify_query_intent_marks_maintainer_workflow_requests() {
5769        let intent = classify_query_intent(
5770            WorkflowMode::Auto,
5771            "Run my cleanup scripts and prune old artifacts.",
5772        );
5773        assert!(intent.maintainer_workflow_mode);
5774        assert_eq!(
5775            preferred_maintainer_workflow("Rebuild the local portable and update PATH."),
5776            Some("package_windows")
5777        );
5778        assert_eq!(
5779            preferred_maintainer_workflow("Run the release flow and publish crates."),
5780            Some("release")
5781        );
5782    }
5783
5784    #[test]
5785    fn classify_query_intent_marks_workspace_workflow_requests() {
5786        let intent = classify_query_intent(WorkflowMode::Auto, "Run the tests in this project.");
5787        assert!(intent.workspace_workflow_mode);
5788        assert_eq!(
5789            preferred_workspace_workflow("Run the tests in this project."),
5790            Some("test")
5791        );
5792        assert_eq!(
5793            preferred_workspace_workflow("Run npm run dev in this repo."),
5794            Some("script")
5795        );
5796    }
5797
5798    #[test]
5799    fn test_overclocker_routing() {
5800        assert_eq!(
5801            preferred_host_inspection_topic("How's my silicon health looking?"),
5802            Some("overclocker")
5803        );
5804        assert_eq!(
5805            preferred_host_inspection_topic("Show me GPU clocks"),
5806            Some("overclocker")
5807        );
5808        assert_eq!(
5809            preferred_host_inspection_topic("nvidia stats"),
5810            Some("overclocker")
5811        );
5812        assert_eq!(
5813            preferred_host_inspection_topic("Show me GPU voltage telemetry"),
5814            Some("overclocker")
5815        );
5816        assert_eq!(
5817            preferred_host_inspection_topic("What are my CPU and GPU volts right now?"),
5818            Some("overclocker")
5819        );
5820    }
5821
5822    #[test]
5823    fn test_gpu_throttle_routing() {
5824        assert_eq!(
5825            preferred_host_inspection_topic("Is my GPU currently throttled and why?"),
5826            Some("overclocker")
5827        );
5828        assert_eq!(
5829            preferred_host_inspection_topic("Tell me if my GPU is throttled"),
5830            Some("overclocker")
5831        );
5832        assert_eq!(
5833            preferred_host_inspection_topic("Is the GPU overheating?"),
5834            Some("overclocker")
5835        );
5836    }
5837
5838    #[test]
5839    fn test_host_inspection_gateway() {
5840        assert!(mentions_host_inspection_question("is my gpu throttled?"));
5841        assert!(mentions_host_inspection_question(
5842            "check vram and silicon health"
5843        ));
5844        assert!(mentions_host_inspection_question("nvidia stats"));
5845
5846        // Negative tests: General coding/repo questions should NOT trigger the gate
5847        assert!(!mentions_host_inspection_question("What is a Rust macro?"));
5848        assert!(!mentions_host_inspection_question(
5849            "Explain the repository structure."
5850        ));
5851        assert!(!mentions_host_inspection_question(
5852            "is this code efficient?"
5853        ));
5854    }
5855
5856    #[test]
5857    fn test_web_mutation_routing() {
5858        // This is the prompt that previously failed by routing to HostInspection
5859        let input = "I want to change the primary brand color from whatever it is now to a vibrant 'Neon Hematite' (HSL 180, 100%, 50%). Update all CSS variables, update the JS theme toggle logic to support this as the new default highlight, and ensure the HTML icons match. Run verify_build when you are done.";
5860
5861        // Test in Auto mode (where it should stay in code)
5862        let intent = classify_query_intent(WorkflowMode::Auto, input);
5863        assert_eq!(intent.primary_class, QueryIntentClass::Implementation);
5864        assert_eq!(intent.direct_answer, None);
5865
5866        // Test in Code mode (where it should stay in code)
5867        let intent = classify_query_intent(WorkflowMode::Code, input);
5868        assert_eq!(intent.primary_class, QueryIntentClass::Implementation);
5869        assert_eq!(intent.direct_answer, None);
5870    }
5871
5872    #[test]
5873    fn test_explicit_diagnostic_during_code() {
5874        // Even if we are in Code mode, an authoritative hardware noun should trigger the diagnostic
5875        let input = "Check my GPU stats and tell me if it's throttled.";
5876        let intent = classify_query_intent(WorkflowMode::Code, input);
5877
5878        assert_eq!(intent.direct_answer, Some(DirectAnswerKind::HostInspection));
5879    }
5880
5881    #[test]
5882    fn test_coding_shield_logic_collision() {
5883        // "logic" should not collide with "log" when in code mode
5884        let input = "Fix the login logic in my typescript code.";
5885        let intent = classify_query_intent(WorkflowMode::Auto, input);
5886
5887        assert_eq!(intent.primary_class, QueryIntentClass::Implementation);
5888        assert_ne!(intent.direct_answer, Some(DirectAnswerKind::HostInspection));
5889    }
5890
5891    #[test]
5892    fn single_file_html_sovereign_prompt_counts_as_scaffold() {
5893        let input = "google uefn toolbelt then make a folder on my desktop called yourtask and inside it create a single index.html that explains what you found";
5894        let intent = classify_query_intent(WorkflowMode::Auto, input);
5895
5896        assert!(intent.sovereign_mode);
5897        assert!(intent.scaffold_mode);
5898    }
5899}