From be815937a59073517e05cd09ad2378ae215485dd Mon Sep 17 00:00:00 2001 From: "google-labs-jules[bot]" <161369871+google-labs-jules[bot]@users.noreply.github.com> Date: Sat, 28 Jun 2025 08:41:29 +0000 Subject: [PATCH 1/3] feat: Integrate OpenRouter as an LLM provider - Adds OpenRouter as a selectable LLM provider alongside OpenAI. - Users can choose the provider and select specific models from OpenRouter. - Includes UI for managing OpenRouter API keys, similar to Firecrawl keys. - Updates backend API to handle requests to OpenRouter. - Modifies API key checking to include OpenRouter keys. - Updates README with new features and environment variables. --- README.md | 12 +- app/api/fireplexity/check-env/route.ts | 4 +- app/api/fireplexity/search/route.ts | 55 +++-- app/api/openrouter/models/route.ts | 25 ++ app/page.tsx | 310 +++++++++++++++++++++---- app/search.tsx | 7 +- app/types.ts | 14 ++ 7 files changed, 357 insertions(+), 70 deletions(-) create mode 100644 app/api/openrouter/models/route.ts diff --git a/README.md b/README.md index 9c8f724..e816b9e 100644 --- a/README.md +++ b/README.md @@ -11,7 +11,9 @@ A blazing-fast AI search engine powered by Firecrawl's web scraping API. Get int ## Features - **Real-time Web Search** - Powered by Firecrawl's search API -- **AI Responses** - Streaming answers with GPT-4o-mini +- **Flexible AI Providers** - Choose between OpenAI and OpenRouter +- **Configurable Models** - Select from available models when using OpenRouter +- **AI Responses** - Streaming answers from your chosen LLM - **Source Citations** - Every claim backed by references - **Live Stock Data** - Automatic TradingView charts - **Smart Follow-ups** - AI-generated questions @@ -33,9 +35,12 @@ cp .env.example .env.local Add to `.env.local`: ``` FIRECRAWL_API_KEY=fc-your-api-key -OPENAI_API_KEY=sk-your-api-key +OPENAI_API_KEY=sk-your-api-key # Required if using the OpenAI provider +OPENROUTER_API_KEY=your-openrouter-key # Required if using the OpenRouter provider ``` +**Note:** If environment variables for API keys are not set, the application will prompt you to enter them in the UI. These will be stored in your browser's `localStorage`. + ### Run ```bash npm run dev @@ -47,9 +52,10 @@ Visit http://localhost:3000 - **Firecrawl** - Web scraping API - **Next.js 15** - React framework -- **OpenAI** - GPT-4o-mini +- **OpenAI / OpenRouter** - LLM providers - **Vercel AI SDK** - Streaming - **TradingView** - Stock charts +- **Shadcn UI & Tailwind CSS** - Frontend components ## Deploy diff --git a/app/api/fireplexity/check-env/route.ts b/app/api/fireplexity/check-env/route.ts index 42a998f..2948908 100644 --- a/app/api/fireplexity/check-env/route.ts +++ b/app/api/fireplexity/check-env/route.ts @@ -2,6 +2,8 @@ import { NextResponse } from 'next/server' export async function GET() { return NextResponse.json({ - hasFirecrawlKey: !!process.env.FIRECRAWL_API_KEY + hasFirecrawlKey: !!process.env.FIRECRAWL_API_KEY, + hasOpenAIKey: !!process.env.OPENAI_API_KEY, + hasOpenRouterKey: !!process.env.OPENROUTER_API_KEY }) } \ No newline at end of file diff --git a/app/api/fireplexity/search/route.ts b/app/api/fireplexity/search/route.ts index 43e511c..374d464 100644 --- a/app/api/fireplexity/search/route.ts +++ b/app/api/fireplexity/search/route.ts @@ -1,5 +1,5 @@ import { NextResponse } from 'next/server' -import { createOpenAI } from '@ai-sdk/openai' +import { createOpenAI, OpenAIProvider } from '@ai-sdk/openai' import { streamText, generateText, createDataStreamResponse } from 'ai' import { detectCompanyTicker } from '@/lib/company-ticker-map' import { selectRelevantContent } from '@/lib/content-selection' @@ -12,28 +12,51 @@ export async function POST(request: Request) { const body = await request.json() const messages = body.messages || [] const query = messages[messages.length - 1]?.content || body.query + const provider = body.provider || 'openai' // Default to openai + const openRouterModel = body.openRouterModel // Model for OpenRouter + const openRouterApiKey = body.openRouterApiKey || process.env.OPENROUTER_API_KEY + console.log(`[${requestId}] Query received:`, query) + console.log(`[${requestId}] Provider:`, provider) + if (provider === 'openrouter') { + console.log(`[${requestId}] OpenRouter Model:`, openRouterModel) + } if (!query) { return NextResponse.json({ error: 'Query is required' }, { status: 400 }) } - // Use API key from request body if provided, otherwise fall back to environment variable + // API key validation const firecrawlApiKey = body.firecrawlApiKey || process.env.FIRECRAWL_API_KEY - const openaiApiKey = process.env.OPENAI_API_KEY - if (!firecrawlApiKey) { return NextResponse.json({ error: 'Firecrawl API key not configured' }, { status: 500 }) } - - if (!openaiApiKey) { - return NextResponse.json({ error: 'OpenAI API key not configured' }, { status: 500 }) - } - // Configure OpenAI with API key - const openai = createOpenAI({ - apiKey: openaiApiKey - }) + let llmProvider: OpenAIProvider + let modelName: string = 'gpt-4o-mini' // Default OpenAI model + + if (provider === 'openai') { + const openaiApiKey = process.env.OPENAI_API_KEY + if (!openaiApiKey) { + return NextResponse.json({ error: 'OpenAI API key not configured' }, { status: 500 }) + } + llmProvider = createOpenAI({ apiKey: openaiApiKey }) + } else if (provider === 'openrouter') { + if (!openRouterApiKey) { + return NextResponse.json({ error: 'OpenRouter API key not configured' }, { status: 500 }) + } + if (!openRouterModel) { + return NextResponse.json({ error: 'OpenRouter model not selected' }, { status: 400 }) + } + llmProvider = createOpenAI({ + apiKey: openRouterApiKey, + baseURL: 'https://openrouter.ai/api/v1', + }) + modelName = openRouterModel + console.log(`[${requestId}] Using OpenRouter model: ${modelName}`) + } else { + return NextResponse.json({ error: 'Invalid provider specified' }, { status: 400 }) + } // Initialize Firecrawl const firecrawl = new FirecrawlApp({ apiKey: firecrawlApiKey }) @@ -176,7 +199,7 @@ export async function POST(request: Request) { : `user: ${query}` const followUpPromise = generateText({ - model: openai('gpt-4o-mini'), + model: llmProvider(modelName), // Use dynamic model and provider messages: [ { role: 'system', @@ -195,7 +218,7 @@ export async function POST(request: Request) { // Stream the text generation const result = streamText({ - model: openai('gpt-4o-mini'), + model: llmProvider(modelName), // Use dynamic model and provider messages: aiMessages, temperature: 0.7, maxTokens: 2000 @@ -239,11 +262,11 @@ export async function POST(request: Request) { const errorResponses: Record = { 401: { error: 'Invalid API key', - suggestion: 'Please check your Firecrawl API key is correct.' + suggestion: 'Please check your API key for the selected provider.' }, 402: { error: 'Insufficient credits', - suggestion: 'You\'ve run out of Firecrawl credits. Please upgrade your plan.' + suggestion: 'You\'ve run out of credits for the selected provider. Please upgrade your plan.' }, 429: { error: 'Rate limit exceeded', diff --git a/app/api/openrouter/models/route.ts b/app/api/openrouter/models/route.ts new file mode 100644 index 0000000..3bff158 --- /dev/null +++ b/app/api/openrouter/models/route.ts @@ -0,0 +1,25 @@ +import { NextResponse } from 'next/server' + +export async function GET() { + try { + const response = await fetch('https://openrouter.ai/api/v1/models', { + method: 'GET', + headers: { + // Authorization: `Bearer ${process.env.OPENROUTER_API_KEY}`, // Add API key if required by OpenRouter for this endpoint + }, + }) + + if (!response.ok) { + const errorData = await response.json() + console.error('OpenRouter API error:', errorData) + return NextResponse.json({ error: 'Failed to fetch models from OpenRouter', details: errorData }, { status: response.status }) + } + + const data = await response.json() + return NextResponse.json(data) + } catch (error) { + console.error('Error fetching OpenRouter models:', error) + const errorMessage = error instanceof Error ? error.message : 'Unknown error' + return NextResponse.json({ error: 'Internal server error', details: errorMessage }, { status: 500 }) + } +} diff --git a/app/page.tsx b/app/page.tsx index 65ea628..e9a617c 100644 --- a/app/page.tsx +++ b/app/page.tsx @@ -3,7 +3,7 @@ import { useChat } from 'ai/react' import { SearchComponent } from './search' import { ChatInterface } from './chat-interface' -import { SearchResult } from './types' +import { SearchResult, OpenRouterModel } from './types' import { Button } from '@/components/ui/button' import Link from 'next/link' import Image from 'next/image' @@ -18,6 +18,15 @@ import { import { Input } from "@/components/ui/input" import { toast } from "sonner" import { ErrorDisplay } from '@/components/error-display' +import { Label } from '@/components/ui/label' +import { + Select, + SelectContent, + SelectItem, + SelectTrigger, + SelectValue, +} from "@/components/ui/select" + interface MessageData { sources: SearchResult[] @@ -34,16 +43,33 @@ export default function FireplexityPage() { const [messageData, setMessageData] = useState>(new Map()) const currentMessageIndex = useRef(0) const [currentTicker, setCurrentTicker] = useState(null) + + // API Key States const [firecrawlApiKey, setFirecrawlApiKey] = useState('') - const [hasApiKey, setHasApiKey] = useState(false) - const [showApiKeyModal, setShowApiKeyModal] = useState(false) - const [, setIsCheckingEnv] = useState(true) + const [hasFirecrawlApiKey, setHasFirecrawlApiKey] = useState(false) + const [showFirecrawlApiKeyModal, setShowFirecrawlApiKeyModal] = useState(false) + + const [openRouterApiKey, setOpenRouterApiKey] = useState('') + const [hasOpenRouterApiKey, setHasOpenRouterApiKey] = useState(false) + const [showOpenRouterApiKeyModal, setShowOpenRouterApiKeyModal] = useState(false) + + const [isCheckingEnv, setIsCheckingEnv] = useState(true) const [pendingQuery, setPendingQuery] = useState('') + // Provider and Model States + const [provider, setProvider] = useState('openai') // 'openai' or 'openrouter' + const [openRouterModel, setOpenRouterModel] = useState('') + const [availableOpenRouterModels, setAvailableOpenRouterModels] = useState([]) + const [isFetchingModels, setIsFetchingModels] = useState(false) + + const { messages, input, handleInputChange, handleSubmit, isLoading, data } = useChat({ api: '/api/fireplexity/search', body: { - ...(firecrawlApiKey && { firecrawlApiKey }) + provider, + ...(provider === 'openrouter' && { openRouterModel }), + ...(firecrawlApiKey && { firecrawlApiKey }), + ...(provider === 'openrouter' && openRouterApiKey && { openRouterApiKey }), }, onResponse: () => { // Clear status when response starts @@ -59,6 +85,7 @@ export default function FireplexityPage() { onError: (error) => { console.error('Chat error:', error) setSearchStatus('') + toast.error(error.message || "An error occurred during the chat.") }, onFinish: () => { setSearchStatus('') @@ -76,7 +103,7 @@ export default function FireplexityPage() { newItems.forEach((item) => { if (!item || typeof item !== 'object' || !('type' in item)) return - const typedItem = item as unknown as { type: string; message?: string; sources?: SearchResult[]; questions?: string[]; symbol?: string } + const typedItem = item as unknown as { type: string; message?: string; sources?: SearchResult[]; questions?: string[]; symbol?: string, error?: string, suggestion?: string } if (typedItem.type === 'status') { setSearchStatus(typedItem.message || '') } @@ -104,73 +131,150 @@ export default function FireplexityPage() { newMap.set(currentMessageIndex.current, { ...existingData, followUpQuestions: typedItem.questions }) setMessageData(newMap) } + if (typedItem.type === 'error') { + toast.error(typedItem.error, { description: typedItem.suggestion }) + setSearchStatus('') + } }) // Update the last processed length lastDataLength.current = data.length } - }, [data, messageData]) + }, [data, messageData, messages]) - // Check for environment variables on mount + // Check for environment variables and stored API keys on mount useEffect(() => { - const checkApiKey = async () => { + const checkApiKeys = async () => { + setIsCheckingEnv(true) try { const response = await fetch('/api/fireplexity/check-env') const data = await response.json() if (data.hasFirecrawlKey) { - setHasApiKey(true) + setHasFirecrawlApiKey(true) } else { - // Check localStorage for user's API key - const storedKey = localStorage.getItem('firecrawl-api-key') - if (storedKey) { - setFirecrawlApiKey(storedKey) - setHasApiKey(true) + const storedFcKey = localStorage.getItem('firecrawl-api-key') + if (storedFcKey) { + setFirecrawlApiKey(storedFcKey) + setHasFirecrawlApiKey(true) } } + + if (data.hasOpenRouterKey) { + setHasOpenRouterApiKey(true) + } else { + const storedOrKey = localStorage.getItem('openrouter-api-key') + if (storedOrKey) { + setOpenRouterApiKey(storedOrKey) + setHasOpenRouterApiKey(true) + } + } + } catch (error) { console.error('Error checking environment:', error) + toast.error("Failed to check API key status.") } finally { setIsCheckingEnv(false) } } - checkApiKey() + checkApiKeys() }, []) - const handleApiKeySubmit = () => { + // Fetch OpenRouter models when provider is set to openrouter + useEffect(() => { + const fetchModels = async () => { + if (provider === 'openrouter' && availableOpenRouterModels.length === 0) { + setIsFetchingModels(true) + try { + const response = await fetch('/api/openrouter/models') + if (!response.ok) { + const errorData = await response.json() + toast.error('Failed to fetch OpenRouter models', { description: errorData.details?.error?.message || response.statusText }) + setAvailableOpenRouterModels([]) + return + } + const modelData = await response.json() + // Filter for chat models and sort by name + const chatModels = modelData.data + .filter((model: OpenRouterModel) => model.id.includes("chat") || model.name?.toLowerCase().includes("chat")) // Basic filtering + .sort((a: OpenRouterModel, b: OpenRouterModel) => (a.name || a.id).localeCompare(b.name || b.id)); + setAvailableOpenRouterModels(chatModels) + if (chatModels.length > 0 && !openRouterModel) { + setOpenRouterModel(chatModels[0].id) // Set default model + } + } catch (error) { + console.error('Error fetching OpenRouter models:', error) + toast.error('Error fetching OpenRouter models.') + setAvailableOpenRouterModels([]) + } finally { + setIsFetchingModels(false) + } + } + } + fetchModels() + }, [provider, openRouterModel, availableOpenRouterModels.length]) + + + const handleFirecrawlApiKeySubmit = () => { if (firecrawlApiKey.trim()) { localStorage.setItem('firecrawl-api-key', firecrawlApiKey) - setHasApiKey(true) - setShowApiKeyModal(false) - toast.success('API key saved successfully!') + setHasFirecrawlApiKey(true) + setShowFirecrawlApiKeyModal(false) + toast.success('Firecrawl API key saved!') - // If there's a pending query, submit it if (pendingQuery) { - const fakeEvent = { - preventDefault: () => {}, - currentTarget: { - querySelector: () => ({ value: pendingQuery }) - } - } as any - handleInputChange({ target: { value: pendingQuery } } as any) - setTimeout(() => { - handleSubmit(fakeEvent) - setPendingQuery('') - }, 100) + triggerPendingQuery() + } + } + } + + const handleOpenRouterApiKeySubmit = () => { + if (openRouterApiKey.trim()) { + localStorage.setItem('openrouter-api-key', openRouterApiKey) + setHasOpenRouterApiKey(true) + setShowOpenRouterApiKeyModal(false) + toast.success('OpenRouter API key saved!') + + if (pendingQuery) { + triggerPendingQuery() } } } + const triggerPendingQuery = () => { + if (pendingQuery) { + const fakeEvent = { + preventDefault: () => {}, + currentTarget: { + querySelector: () => ({ value: pendingQuery }) + } + } as any + handleInputChange({ target: { value: pendingQuery } } as any) + setTimeout(() => { + handleSubmit(fakeEvent) + setPendingQuery('') + }, 100) + } + } + + const handleSearch = (e: React.FormEvent) => { e.preventDefault() - if (!input.trim()) return + if (!input.trim() || isLoading) return - // Check if we have an API key - if (!hasApiKey) { + // Check for Firecrawl API key + if (!hasFirecrawlApiKey) { + setPendingQuery(input) + setShowFirecrawlApiKeyModal(true) + return + } + + // Check for OpenRouter API key if provider is openrouter + if (provider === 'openrouter' && !hasOpenRouterApiKey) { setPendingQuery(input) - setShowApiKeyModal(true) + setShowOpenRouterApiKeyModal(true) return } @@ -184,16 +288,28 @@ export default function FireplexityPage() { // Wrapped submit handler for chat interface const handleChatSubmit = (e: React.FormEvent) => { - // Check if we have an API key - if (!hasApiKey) { + if (isLoading) { + e.preventDefault() + return + } + // Check for Firecrawl API key + if (!hasFirecrawlApiKey) { + setPendingQuery(input) + setShowFirecrawlApiKeyModal(true) + e.preventDefault() + return + } + + // Check for OpenRouter API key if provider is openrouter + if (provider === 'openrouter' && !hasOpenRouterApiKey) { setPendingQuery(input) - setShowApiKeyModal(true) + setShowOpenRouterApiKeyModal(true) e.preventDefault() return } // Store current data in messageData before clearing - if (messages.length > 0 && sources.length > 0) { + if (messages.length > 0 && (sources.length > 0 || followUpQuestions.length > 0 || currentTicker)) { const assistantMessages = messages.filter(m => m.role === 'assistant') const lastAssistantIndex = assistantMessages.length - 1 if (lastAssistantIndex >= 0) { @@ -265,11 +381,71 @@ export default function FireplexityPage() {

- AI-powered web search with instant results and follow-up questions + AI-powered web search with instant results and follow-up questions. Now with OpenRouter support!

+ {/* Configuration Section - Only show before first search */} + {!isChatActive && ( +
+
+

Configuration

+
+
+ + +
+ + {provider === 'openrouter' && ( +
+ + + {!hasOpenRouterApiKey && ( + + )} +
+ )} +
+ {!hasFirecrawlApiKey && ( +
+ +
+ )} +
+
+ )} + + {/* Main content wrapper */}
@@ -278,7 +454,8 @@ export default function FireplexityPage() { handleSubmit={handleSearch} input={input} handleInputChange={handleInputChange} - isLoading={isLoading} + isLoading={isLoading || isCheckingEnv || (provider === 'openrouter' && isFetchingModels)} + disabled={isCheckingEnv || (provider === 'openrouter' && isFetchingModels)} /> ) : ( - {/* API Key Modal */} - + {/* Firecrawl API Key Modal */} + Firecrawl API Key Required @@ -333,19 +510,58 @@ export default function FireplexityPage() {
setFirecrawlApiKey(e.target.value)} onKeyDown={(e) => { if (e.key === 'Enter') { e.preventDefault() - handleApiKeySubmit() + handleFirecrawlApiKeySubmit() + } + }} + className="h-12" + /> + +
+
+
+ + {/* OpenRouter API Key Modal */} + + + + OpenRouter API Key Required + + To use the OpenRouter provider, you need an OpenRouter API key. Get one from{' '} + + openrouter.ai + + + +
+ setOpenRouterApiKey(e.target.value)} + onKeyDown={(e) => { + if (e.key === 'Enter') { + e.preventDefault() + handleOpenRouterApiKeySubmit() } }} className="h-12" /> -
diff --git a/app/search.tsx b/app/search.tsx index 8c0ddea..1d476a1 100644 --- a/app/search.tsx +++ b/app/search.tsx @@ -9,9 +9,10 @@ interface SearchComponentProps { input: string handleInputChange: (e: React.ChangeEvent | React.ChangeEvent) => void isLoading: boolean + disabled?: boolean } -export function SearchComponent({ handleSubmit, input, handleInputChange, isLoading }: SearchComponentProps) { +export function SearchComponent({ handleSubmit, input, handleInputChange, isLoading, disabled }: SearchComponentProps) { return (
@@ -21,11 +22,11 @@ export function SearchComponent({ handleSubmit, input, handleInputChange, isLoad onChange={handleInputChange} placeholder="Ask anything..." className="pr-24 h-14 text-lg rounded-xl border border-gray-200 dark:border-gray-700 bg-white dark:bg-zinc-800 transition-colors" - disabled={isLoading} + disabled={isLoading || disabled} /> + + + + + + {emptyStateMessage} + + {options.map((option) => ( + { + onChange(currentValue === value ? "" : currentValue) + setOpen(false) + }} + > + + {option.label} + + ))} + + + + + + ) +} diff --git a/components/ui/command.tsx b/components/ui/command.tsx new file mode 100644 index 0000000..6fbed44 --- /dev/null +++ b/components/ui/command.tsx @@ -0,0 +1,155 @@ +"use client" + +import * as React from "react" +import { type DialogProps } from "@radix-ui/react-dialog" +import { Command as CommandPrimitive } from "cmdk" +import { Search } from "lucide-react" + +import { cn } from "@/lib/utils" +import { Dialog, DialogContent } from "@/components/ui/dialog" + +const Command = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>(({ className, ...props }, ref) => ( + +)) +Command.displayName = CommandPrimitive.displayName + +interface CommandDialogProps extends DialogProps {} + +const CommandDialog = ({ children, ...props }: CommandDialogProps) => { + return ( + + + + {children} + + + + ) +} + +const CommandInput = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>(({ className, ...props }, ref) => ( +
+ + +
+)) + +CommandInput.displayName = CommandPrimitive.Input.displayName + +const CommandList = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>(({ className, ...props }, ref) => ( + +)) + +CommandList.displayName = CommandPrimitive.List.displayName + +const CommandEmpty = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>((props, ref) => ( + +)) + +CommandEmpty.displayName = CommandPrimitive.Empty.displayName + +const CommandGroup = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>(({ className, ...props }, ref) => ( + +)) + +CommandGroup.displayName = CommandPrimitive.Group.displayName + +const CommandSeparator = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>(({ className, ...props }, ref) => ( + +)) +CommandSeparator.displayName = CommandPrimitive.Separator.displayName + +const CommandItem = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>(({ className, ...props }, ref) => ( + +)) + +CommandItem.displayName = CommandPrimitive.Item.displayName + +const CommandShortcut = ({ + className, + ...props +}: React.HTMLAttributes) => { + return ( + + ) +} +CommandShortcut.displayName = "CommandShortcut" + +export { + Command, + CommandDialog, + CommandInput, + CommandList, + CommandEmpty, + CommandGroup, + CommandItem, + CommandSeparator, + CommandShortcut, +} diff --git a/components/ui/popover.tsx b/components/ui/popover.tsx new file mode 100644 index 0000000..a0ec48b --- /dev/null +++ b/components/ui/popover.tsx @@ -0,0 +1,31 @@ +"use client" + +import * as React from "react" +import * as PopoverPrimitive from "@radix-ui/react-popover" + +import { cn } from "@/lib/utils" + +const Popover = PopoverPrimitive.Root + +const PopoverTrigger = PopoverPrimitive.Trigger + +const PopoverContent = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>(({ className, align = "center", sideOffset = 4, ...props }, ref) => ( + + + +)) +PopoverContent.displayName = PopoverPrimitive.Content.displayName + +export { Popover, PopoverTrigger, PopoverContent } diff --git a/package-lock.json b/package-lock.json index 6fc6c43..1f2026f 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9,16 +9,20 @@ "version": "0.1.0", "dependencies": { "@ai-sdk/openai": "^1.3.22", + "@mendable/firecrawl-js": "^1.10.0", + "@radix-ui/react-dialog": "^1.1.4", "@radix-ui/react-slot": "^1.2.3", "ai": "^4.3.16", "class-variance-authority": "^0.7.1", "clsx": "^2.1.1", + "cmdk": "^1.1.1", "lucide-react": "^0.511.0", "next": "15.3.2", "react": "^19.0.0", "react-dom": "^19.0.0", "react-markdown": "^10.1.0", "remark-gfm": "^4.0.1", + "sonner": "^1.7.2", "tailwind-merge": "^3.3.0" }, "devDependencies": { @@ -861,6 +865,21 @@ "@jridgewell/sourcemap-codec": "^1.4.14" } }, + "node_modules/@mendable/firecrawl-js": { + "version": "1.29.0", + "resolved": "https://registry.npmjs.org/@mendable/firecrawl-js/-/firecrawl-js-1.29.0.tgz", + "integrity": "sha512-ZS97rwri5ZZmqDWy7VQJlzCmNFATSvUj+LNBtMj//Rs6fm/uIsyOU5Noq6zWVWKLqFsuQnDM5wnMz8q0JFRi/w==", + "license": "MIT", + "dependencies": { + "axios": "^1.6.8", + "typescript-event-target": "^1.1.1", + "zod": "^3.23.8", + "zod-to-json-schema": "^3.23.0" + }, + "engines": { + "node": ">=22.0.0" + } + }, "node_modules/@napi-rs/wasm-runtime": { "version": "0.2.11", "resolved": "https://registry.npmjs.org/@napi-rs/wasm-runtime/-/wasm-runtime-0.2.11.tgz", @@ -1075,6 +1094,12 @@ "node": ">=8.0.0" } }, + "node_modules/@radix-ui/primitive": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@radix-ui/primitive/-/primitive-1.1.2.tgz", + "integrity": "sha512-XnbHrrprsNqZKQhStrSwgRUQzoCI1glLzdw79xiZPoofhGICeZRSQ3dIxAKH1gb3OHfNf4d6f+vAv3kil2eggA==", + "license": "MIT" + }, "node_modules/@radix-ui/react-compose-refs": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/@radix-ui/react-compose-refs/-/react-compose-refs-1.1.2.tgz", @@ -1090,6 +1115,213 @@ } } }, + "node_modules/@radix-ui/react-context": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-context/-/react-context-1.1.2.tgz", + "integrity": "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-dialog": { + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/@radix-ui/react-dialog/-/react-dialog-1.1.14.tgz", + "integrity": "sha512-+CpweKjqpzTmwRwcYECQcNYbI8V9VSQt0SNFKeEBLgfucbsLssU6Ppq7wUdNXEGb573bMjFhVjKVll8rmV6zMw==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.2", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-dismissable-layer": "1.1.10", + "@radix-ui/react-focus-guards": "1.1.2", + "@radix-ui/react-focus-scope": "1.1.7", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-presence": "1.1.4", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-slot": "1.2.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "aria-hidden": "^1.2.4", + "react-remove-scroll": "^2.6.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-dismissable-layer": { + "version": "1.1.10", + "resolved": "https://registry.npmjs.org/@radix-ui/react-dismissable-layer/-/react-dismissable-layer-1.1.10.tgz", + "integrity": "sha512-IM1zzRV4W3HtVgftdQiiOmA0AdJlCtMLe00FXaHwgt3rAnNsIyDqshvkIW3hj/iu5hu8ERP7KIYki6NkqDxAwQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.2", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-escape-keydown": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-focus-guards": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-focus-guards/-/react-focus-guards-1.1.2.tgz", + "integrity": "sha512-fyjAACV62oPV925xFCrH8DR5xWhg9KYtJT4s3u54jxp+L/hbpTY2kIeEFFbFe+a/HCE94zGQMZLIpVTPVZDhaA==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-focus-scope": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-focus-scope/-/react-focus-scope-1.1.7.tgz", + "integrity": "sha512-t2ODlkXBQyn7jkl6TNaw/MtVEVvIGelJDCG41Okq/KwUsJBwQ4XVZsHAVUkK4mBv3ewiAS3PGuUWuY2BoK4ZUw==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-id": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-id/-/react-id-1.1.1.tgz", + "integrity": "sha512-kGkGegYIdQsOb4XjsfM97rXsiHaBwco+hFI66oO4s9LU+PLAC5oJ7khdOVFxkhsmlbpUqDAvXw11CluXP+jkHg==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-portal": { + "version": "1.1.9", + "resolved": "https://registry.npmjs.org/@radix-ui/react-portal/-/react-portal-1.1.9.tgz", + "integrity": "sha512-bpIxvq03if6UNwXZ+HTK71JLh4APvnXntDc6XOX8UVq4XQOVl7lwok0AvIl+b8zgCw3fSaVTZMpAPPagXbKmHQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-presence": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@radix-ui/react-presence/-/react-presence-1.1.4.tgz", + "integrity": "sha512-ueDqRbdc4/bkaQT3GIpLQssRlFgWaL/U2z/S31qRwwLWoxHLgry3SIfCwhxeQNbirEUXFa+lq3RL3oBYXtcmIA==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-primitive": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/@radix-ui/react-primitive/-/react-primitive-2.1.3.tgz", + "integrity": "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-slot": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, "node_modules/@radix-ui/react-slot": { "version": "1.2.3", "resolved": "https://registry.npmjs.org/@radix-ui/react-slot/-/react-slot-1.2.3.tgz", @@ -1108,6 +1340,91 @@ } } }, + "node_modules/@radix-ui/react-use-callback-ref": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-callback-ref/-/react-use-callback-ref-1.1.1.tgz", + "integrity": "sha512-FkBMwD+qbGQeMu1cOHnuGB6x4yzPjho8ap5WtbEJ26umhgqVXbhekKUQO+hZEL1vU92a3wHwdp0HAcqAUF5iDg==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-controllable-state": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-controllable-state/-/react-use-controllable-state-1.2.2.tgz", + "integrity": "sha512-BjasUjixPFdS+NKkypcyyN5Pmg83Olst0+c6vGov0diwTEo6mgdqVR6hxcEgFuh4QrAs7Rc+9KuGJ9TVCj0Zzg==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-effect-event": "0.0.2", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-effect-event": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-effect-event/-/react-use-effect-event-0.0.2.tgz", + "integrity": "sha512-Qp8WbZOBe+blgpuUT+lw2xheLP8q0oatc9UpmiemEICxGvFLYmHm9QowVZGHtJlGbS6A6yJ3iViad/2cVjnOiA==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-escape-keydown": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-escape-keydown/-/react-use-escape-keydown-1.1.1.tgz", + "integrity": "sha512-Il0+boE7w/XebUHyBjroE+DbByORGR9KKmITzbR7MyQ4akpORYP/ZmbhAr0DG7RmmBqoOnZdy2QlvajJ2QA59g==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-callback-ref": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-layout-effect": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-layout-effect/-/react-use-layout-effect-1.1.1.tgz", + "integrity": "sha512-RbJRS4UWQFkzHTTwVymMTUv8EqYhOp8dOOviLj2ugtTiXRaRQS7GLGxZTLL1jWhMeoSCf5zmcZkqTl9IiYfXcQ==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, "node_modules/@rtsao/scc": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/@rtsao/scc/-/scc-1.1.0.tgz", @@ -1515,7 +1832,7 @@ "version": "19.1.6", "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.1.6.tgz", "integrity": "sha512-4hOiT/dwO8Ko0gV1m/TJZYk3y0KBnY9vzDh7W+DH17b2HFSOGgdj33dhihPeuy3l0q23+4e+hoXHV6hCC4dCXw==", - "dev": true, + "devOptional": true, "license": "MIT", "peerDependencies": { "@types/react": "^19.0.0" @@ -2178,6 +2495,18 @@ "dev": true, "license": "Python-2.0" }, + "node_modules/aria-hidden": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/aria-hidden/-/aria-hidden-1.2.6.tgz", + "integrity": "sha512-ik3ZgC9dY/lYVVM++OISsaYDeg1tb0VtP5uL3ouh1koGOaUMDPpbFIei4JkFimWUFPn90sbMNMXQAIVOlnYKJA==", + "license": "MIT", + "dependencies": { + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/aria-query": { "version": "5.3.2", "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.2.tgz", @@ -2365,6 +2694,12 @@ "node": ">= 0.4" } }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", + "license": "MIT" + }, "node_modules/available-typed-arrays": { "version": "1.0.7", "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", @@ -2391,6 +2726,17 @@ "node": ">=4" } }, + "node_modules/axios": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.10.0.tgz", + "integrity": "sha512-/1xYAC4MP/HEG+3duIhFr4ZQXR4sQXOIe+o6sdqzeykGLx6Upp/1p8MHqhINOvGeP7xyNHe7tsiJByc4SSVUxw==", + "license": "MIT", + "dependencies": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, "node_modules/axobject-query": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-4.1.0.tgz", @@ -2476,7 +2822,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", - "dev": true, "license": "MIT", "dependencies": { "es-errors": "^1.3.0", @@ -2637,6 +2982,22 @@ "node": ">=6" } }, + "node_modules/cmdk": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/cmdk/-/cmdk-1.1.1.tgz", + "integrity": "sha512-Vsv7kFaXm+ptHDMZ7izaRsP70GgrW9NBNGswt9OZaVBLlE0SNpDq8eu/VGXyF9r7M0azK3Wy7OlYXsuyYLFzHg==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-compose-refs": "^1.1.1", + "@radix-ui/react-dialog": "^1.1.6", + "@radix-ui/react-id": "^1.1.0", + "@radix-ui/react-primitive": "^2.0.2" + }, + "peerDependencies": { + "react": "^18 || ^19 || ^19.0.0-rc", + "react-dom": "^18 || ^19 || ^19.0.0-rc" + } + }, "node_modules/color": { "version": "4.2.3", "resolved": "https://registry.npmjs.org/color/-/color-4.2.3.tgz", @@ -2682,6 +3043,18 @@ "simple-swizzle": "^0.2.2" } }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "license": "MIT", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, "node_modules/comma-separated-tokens": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz", @@ -2854,6 +3227,15 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "license": "MIT", + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/dequal": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", @@ -2873,6 +3255,12 @@ "node": ">=8" } }, + "node_modules/detect-node-es": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/detect-node-es/-/detect-node-es-1.1.0.tgz", + "integrity": "sha512-ypdmJU/TbBby2Dxibuv7ZLW3Bs1QEmM7nHjEANfohJLvE0XVujisn1qPJcZxg+qDucsr+bP6fLD1rPS3AhJ7EQ==", + "license": "MIT" + }, "node_modules/devlop": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/devlop/-/devlop-1.1.0.tgz", @@ -2909,7 +3297,6 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", - "dev": true, "license": "MIT", "dependencies": { "call-bind-apply-helpers": "^1.0.1", @@ -3014,7 +3401,6 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", - "dev": true, "license": "MIT", "engines": { "node": ">= 0.4" @@ -3024,7 +3410,6 @@ "version": "1.3.0", "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", - "dev": true, "license": "MIT", "engines": { "node": ">= 0.4" @@ -3062,7 +3447,6 @@ "version": "1.1.1", "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", - "dev": true, "license": "MIT", "dependencies": { "es-errors": "^1.3.0" @@ -3075,7 +3459,6 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", - "dev": true, "license": "MIT", "dependencies": { "es-errors": "^1.3.0", @@ -3698,6 +4081,26 @@ "dev": true, "license": "ISC" }, + "node_modules/follow-redirects": { + "version": "1.15.9", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", + "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "license": "MIT", + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, "node_modules/for-each": { "version": "0.3.5", "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.5.tgz", @@ -3714,11 +4117,26 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/form-data": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.3.tgz", + "integrity": "sha512-qsITQPfmvMOSAdeyZ+12I1c+CKSstAFAwu+97zrnWAbIr5u8wfsExUzCesVLC8NgHuRUqNN4Zy6UPWUTRGslcA==", + "license": "MIT", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "es-set-tostringtag": "^2.1.0", + "hasown": "^2.0.2", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/function-bind": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "dev": true, "license": "MIT", "funding": { "url": "https://github.com/sponsors/ljharb" @@ -3759,7 +4177,6 @@ "version": "1.3.0", "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", - "dev": true, "license": "MIT", "dependencies": { "call-bind-apply-helpers": "^1.0.2", @@ -3780,11 +4197,19 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/get-nonce": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-nonce/-/get-nonce-1.0.1.tgz", + "integrity": "sha512-FJhYRoDaiatfEkUK8HKlicmu/3SGFD51q3itKDGoSTysQJBnfOcxU5GxnhE1E6soB76MbT0MBtnKJuXyAx+96Q==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, "node_modules/get-proto": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", - "dev": true, "license": "MIT", "dependencies": { "dunder-proto": "^1.0.1", @@ -3872,7 +4297,6 @@ "version": "1.2.0", "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", - "dev": true, "license": "MIT", "engines": { "node": ">= 0.4" @@ -3951,7 +4375,6 @@ "version": "1.1.0", "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", - "dev": true, "license": "MIT", "engines": { "node": ">= 0.4" @@ -3964,7 +4387,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", - "dev": true, "license": "MIT", "dependencies": { "has-symbols": "^1.0.3" @@ -3980,7 +4402,6 @@ "version": "2.0.2", "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", - "dev": true, "license": "MIT", "dependencies": { "function-bind": "^1.1.2" @@ -5070,7 +5491,6 @@ "version": "1.1.0", "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", - "dev": true, "license": "MIT", "engines": { "node": ">= 0.4" @@ -5945,6 +6365,27 @@ "node": ">=8.6" } }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, "node_modules/minimatch": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", @@ -6482,6 +6923,12 @@ "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", + "license": "MIT" + }, "node_modules/punycode": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", @@ -6568,6 +7015,75 @@ "react": ">=18" } }, + "node_modules/react-remove-scroll": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/react-remove-scroll/-/react-remove-scroll-2.7.1.tgz", + "integrity": "sha512-HpMh8+oahmIdOuS5aFKKY6Pyog+FNaZV/XyJOq7b4YFwsFHe5yYfdbIalI4k3vU2nSDql7YskmUseHsRrJqIPA==", + "license": "MIT", + "dependencies": { + "react-remove-scroll-bar": "^2.3.7", + "react-style-singleton": "^2.2.3", + "tslib": "^2.1.0", + "use-callback-ref": "^1.3.3", + "use-sidecar": "^1.1.3" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/react-remove-scroll-bar": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/react-remove-scroll-bar/-/react-remove-scroll-bar-2.3.8.tgz", + "integrity": "sha512-9r+yi9+mgU33AKcj6IbT9oRCO78WriSj6t/cF8DWBZJ9aOGPOTEDvdUDz1FwKim7QXWwmHqtdHnRJfhAxEG46Q==", + "license": "MIT", + "dependencies": { + "react-style-singleton": "^2.2.2", + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/react-style-singleton": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/react-style-singleton/-/react-style-singleton-2.2.3.tgz", + "integrity": "sha512-b6jSvxvVnyptAiLjbkWLE/lOnR4lfTtDAl+eUC7RZy+QQWc6wRzIV2CE6xBuMmDxc2qIihtDCZD5NPOFl7fRBQ==", + "license": "MIT", + "dependencies": { + "get-nonce": "^1.0.0", + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, "node_modules/reflect.getprototypeof": { "version": "1.0.10", "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.10.tgz", @@ -7034,6 +7550,16 @@ "is-arrayish": "^0.3.1" } }, + "node_modules/sonner": { + "version": "1.7.4", + "resolved": "https://registry.npmjs.org/sonner/-/sonner-1.7.4.tgz", + "integrity": "sha512-DIS8z4PfJRbIyfVFDVnK9rO3eYDtse4Omcm6bt0oEr5/jtLgysmjuBl1frJ9E/EQZrFmKx2A8m/s5s9CRXIzhw==", + "license": "MIT", + "peerDependencies": { + "react": "^18.0.0 || ^19.0.0 || ^19.0.0-rc", + "react-dom": "^18.0.0 || ^19.0.0 || ^19.0.0-rc" + } + }, "node_modules/source-map-js": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", @@ -7584,6 +8110,12 @@ "node": ">=14.17" } }, + "node_modules/typescript-event-target": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/typescript-event-target/-/typescript-event-target-1.1.1.tgz", + "integrity": "sha512-dFSOFBKV6uwaloBCCUhxlD3Pr/P1a/tJdcmPrTXCHlEFD3faj0mztjcGn6VBAhQ0/Bdy8K3VWrrqwbt/ffsYsg==", + "license": "MIT" + }, "node_modules/unbox-primitive": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.1.0.tgz", @@ -7742,6 +8274,49 @@ "punycode": "^2.1.0" } }, + "node_modules/use-callback-ref": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/use-callback-ref/-/use-callback-ref-1.3.3.tgz", + "integrity": "sha512-jQL3lRnocaFtu3V00JToYz/4QkNWswxijDaCVNZRiRTO3HQDLsdu1ZtmIUvV4yPp+rvWm5j0y0TG/S61cuijTg==", + "license": "MIT", + "dependencies": { + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/use-sidecar": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/use-sidecar/-/use-sidecar-1.1.3.tgz", + "integrity": "sha512-Fedw0aZvkhynoPYlA5WXrMCAMm+nSWdZt6lzJQ7Ok8S6Q+VsHmHpRWndVRJ8Be0ZbkfPc5LRYH+5XrzXcEeLRQ==", + "license": "MIT", + "dependencies": { + "detect-node-es": "^1.1.0", + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, "node_modules/use-sync-external-store": { "version": "1.5.0", "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.5.0.tgz", @@ -7922,7 +8497,6 @@ "resolved": "https://registry.npmjs.org/zod/-/zod-3.25.67.tgz", "integrity": "sha512-idA2YXwpCdqUSKRCACDE6ItZD9TZzy3OZMtpfLoh6oPR47lipysRrJfjzMqFxQ3uJuUPyUeWe1r9vLH33xO/Qw==", "license": "MIT", - "peer": true, "funding": { "url": "https://github.com/sponsors/colinhacks" } diff --git a/package.json b/package.json index 37bdfa0..0154889 100644 --- a/package.json +++ b/package.json @@ -16,6 +16,7 @@ "ai": "^4.3.16", "class-variance-authority": "^0.7.1", "clsx": "^2.1.1", + "cmdk": "^1.1.1", "lucide-react": "^0.511.0", "next": "15.3.2", "react": "^19.0.0", @@ -36,4 +37,4 @@ "tailwindcss": "^4", "typescript": "^5" } -} \ No newline at end of file +}