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