feat: Add intelligent auto-router and enhanced integrations

- Add intelligent-router.sh hook for automatic agent routing
- Add AUTO-TRIGGER-SUMMARY.md documentation
- Add FINAL-INTEGRATION-SUMMARY.md documentation
- Complete Prometheus integration (6 commands + 4 tools)
- Complete Dexto integration (12 commands + 5 tools)
- Enhanced Ralph with access to all agents
- Fix /clawd command (removed disable-model-invocation)
- Update hooks.json to v5 with intelligent routing
- 291 total skills now available
- All 21 commands with automatic routing

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
admin
2026-01-28 00:27:56 +04:00
Unverified
parent 3b128ba3bd
commit b52318eeae
1724 changed files with 351216 additions and 0 deletions

View File

@@ -0,0 +1,291 @@
import React, { useState, useEffect, useCallback } from 'react';
import { useDebounce } from 'use-debounce';
import { useHotkeys } from 'react-hotkeys-hook';
import { useSearchMessages, type SearchResult } from './hooks/useSearch';
import { Input } from './ui/input';
import { ScrollArea } from './ui/scroll-area';
import { Search, MessageSquare, User, Bot, Settings, ChevronRight, RefreshCw } from 'lucide-react';
import { cn } from '@/lib/utils';
import { Badge } from './ui/badge';
interface GlobalSearchModalProps {
isOpen: boolean;
onClose: () => void;
onNavigateToSession: (sessionId: string, messageIndex?: number) => void;
}
export default function GlobalSearchModal({
isOpen,
onClose,
onNavigateToSession,
}: GlobalSearchModalProps) {
const [searchQuery, setSearchQuery] = useState('');
const [debouncedQuery] = useDebounce(searchQuery, 300);
const [selectedIndex, setSelectedIndex] = useState(0);
// Use TanStack Query for search with debouncing
const { data, isLoading, error } = useSearchMessages(debouncedQuery, undefined, 10, isOpen);
const results = data?.results || [];
const searchError = error?.message ?? null;
// Clamp selectedIndex when results change to prevent out-of-bounds selection
useEffect(() => {
if (selectedIndex >= results.length && results.length > 0) {
setSelectedIndex(results.length - 1);
} else if (results.length === 0) {
setSelectedIndex(0);
}
}, [results.length, selectedIndex]);
const handleResultClick = useCallback(
(result: SearchResult) => {
onNavigateToSession(result.sessionId, result.messageIndex);
onClose();
},
[onNavigateToSession, onClose]
);
// Reset when modal opens/closes
useEffect(() => {
if (isOpen) {
setSearchQuery('');
setSelectedIndex(0);
}
}, [isOpen]);
// Keyboard navigation (using react-hotkeys-hook)
// ArrowDown to navigate down in results
useHotkeys(
'down',
() => {
setSelectedIndex((prev) => Math.min(prev + 1, results.length - 1));
},
{ enabled: isOpen, preventDefault: true },
[isOpen, results.length]
);
// ArrowUp to navigate up in results
useHotkeys(
'up',
() => {
setSelectedIndex((prev) => Math.max(prev - 1, 0));
},
{ enabled: isOpen, preventDefault: true },
[isOpen]
);
// Enter to select current result
useHotkeys(
'enter',
() => {
if (results[selectedIndex]) {
handleResultClick(results[selectedIndex]);
setSelectedIndex(0);
}
},
{ enabled: isOpen, preventDefault: true },
[isOpen, results, selectedIndex, handleResultClick]
);
// Escape to close modal
useHotkeys(
'escape',
() => {
onClose();
},
{ enabled: isOpen, preventDefault: true },
[isOpen, onClose]
);
const getRoleIcon = (role: string) => {
switch (role) {
case 'user':
return <User className="w-4 h-4" />;
case 'assistant':
return <Bot className="w-4 h-4" />;
case 'system':
return <Settings className="w-4 h-4" />;
default:
return <MessageSquare className="w-4 h-4" />;
}
};
const getRoleColor = (role: string) => {
switch (role) {
case 'user':
return 'bg-blue-100 text-blue-800 dark:bg-blue-900 dark:text-blue-200';
case 'assistant':
return 'bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200';
case 'system':
return 'bg-yellow-100 text-yellow-800 dark:bg-yellow-900 dark:text-yellow-200';
default:
return 'bg-gray-100 text-gray-800 dark:bg-gray-800 dark:text-gray-200';
}
};
const highlightText = (text: string, query: string) => {
if (!query) return text;
const escapedQuery = query.replace(/[.*+?^${}()|\[\]\\]/g, '\\$&');
const regex = new RegExp(`(${escapedQuery})`, 'gi');
const parts = text.split(regex);
return parts.map((part, index) =>
part.toLowerCase() === query.toLowerCase() ? (
<mark
key={index}
className="bg-yellow-200 dark:bg-yellow-800 font-medium rounded px-1"
>
{part}
</mark>
) : (
part
)
);
};
if (!isOpen) return null;
return (
<>
{/* Backdrop */}
<div className="fixed inset-0 z-40 bg-black/10 backdrop-blur-[2px]" onClick={onClose} />
{/* Search popover */}
<div className="fixed left-1/2 top-[15%] -translate-x-1/2 z-50 w-full max-w-2xl bg-popover/70 backdrop-blur-xl border border-border/30 rounded-xl shadow-2xl overflow-hidden">
<div className="flex flex-col max-h-[70vh]">
{/* Search Header */}
<div className="p-4 border-b border-border/30">
<div className="relative">
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 w-5 h-5 text-muted-foreground" />
<Input
placeholder="Search conversations..."
value={searchQuery}
onChange={(e) => setSearchQuery(e.target.value)}
className="pl-12 text-lg border-0 shadow-none focus-visible:ring-0 bg-transparent"
autoFocus
/>
</div>
</div>
{/* Results */}
<div className="flex-1 overflow-hidden">
{isLoading ? (
<div className="flex items-center justify-center py-12">
<RefreshCw className="h-6 w-6 animate-spin mr-3" />
<span className="text-muted-foreground">Searching...</span>
</div>
) : error ? (
<div className="flex items-center justify-center py-12">
<div className="text-center">
<Search className="w-12 h-12 mx-auto mb-4 text-destructive opacity-50" />
<p className="text-destructive font-medium">Search Error</p>
<p className="text-sm text-muted-foreground mt-2">
{searchError}
</p>
<p className="text-xs text-muted-foreground mt-2">
Try again or check your connection.
</p>
</div>
</div>
) : (
<ScrollArea className="h-full max-h-[calc(70vh-80px)]">
<div className="p-2">
{results.length > 0 ? (
<div className="space-y-1">
{results.map((result, index) => (
<div
key={index}
className={cn(
'flex items-start gap-3 p-3 rounded-lg cursor-pointer transition-colors',
index === selectedIndex
? 'bg-accent text-accent-foreground'
: 'hover:bg-accent/50'
)}
onClick={() => handleResultClick(result)}
>
<div className="flex-shrink-0 mt-1">
<Badge
className={cn(
'text-xs',
getRoleColor(result.message.role)
)}
>
{getRoleIcon(result.message.role)}
</Badge>
</div>
<div className="flex-1 min-w-0">
<div className="flex items-center gap-2 mb-1">
<span className="text-sm font-medium">
{result.sessionId.length > 20
? `${result.sessionId.slice(0, 20)}...`
: result.sessionId}
</span>
<span className="text-xs text-muted-foreground">
{result.message.role}
</span>
</div>
<div className="text-sm text-muted-foreground line-clamp-2">
{highlightText(
result.context,
debouncedQuery
)}
</div>
</div>
<ChevronRight className="w-4 h-4 text-muted-foreground flex-shrink-0 mt-1" />
</div>
))}
</div>
) : debouncedQuery ? (
<div className="text-center py-12">
<Search className="w-12 h-12 mx-auto mb-4 text-muted-foreground opacity-50" />
<p className="text-muted-foreground">
No messages found matching your search.
</p>
<p className="text-sm text-muted-foreground mt-2">
Try different keywords.
</p>
</div>
) : (
<div className="text-center py-12">
<Search className="w-12 h-12 mx-auto mb-4 text-muted-foreground opacity-50" />
<p className="text-muted-foreground">
Start typing to search your conversations.
</p>
<div className="flex items-center justify-center gap-4 mt-4 text-xs text-muted-foreground">
<div className="flex items-center gap-1">
<kbd className="px-2 py-1 bg-muted/50 rounded text-xs">
</kbd>
<kbd className="px-2 py-1 bg-muted/50 rounded text-xs">
</kbd>
<span>to navigate</span>
</div>
<div className="flex items-center gap-1">
<kbd className="px-2 py-1 bg-muted/50 rounded text-xs">
Enter
</kbd>
<span>to select</span>
</div>
<div className="flex items-center gap-1">
<kbd className="px-2 py-1 bg-muted/50 rounded text-xs">
Esc
</kbd>
<span>to close</span>
</div>
</div>
</div>
)}
</div>
</ScrollArea>
)}
</div>
</div>
</div>
</>
);
}