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