diff --git a/backend/ai/src/index.ts b/backend/ai/src/index.ts index f20687b..e773814 100644 --- a/backend/ai/src/index.ts +++ b/backend/ai/src/index.ts @@ -29,7 +29,7 @@ export default { return new Response("Missing instructions parameter", { status: 400 }); } - const prompt = `You are an intelligent programming assistant. Please respond to the following request: + const prompt = `You are an intelligent programming assistant. Please respond to the following request concisely: ${instructions} @@ -41,24 +41,38 @@ If your response includes code, please format it using triple backticks (\`\`\`) print("Hello, World!") \`\`\` -Provide a clear and concise explanation along with any code snippets.`; +Provide a clear and concise explanation along with any code snippets. Keep your response brief and to the point`; try { const anthropic = new Anthropic({ apiKey: env.ANTHROPIC_API_KEY }); - const response = await anthropic.messages.create({ + const stream = await anthropic.messages.create({ model: "claude-3-opus-20240229", max_tokens: 1024, messages: [{ role: "user", content: prompt }], + stream: true, }); - const assistantResponse = response.content[0].type === 'text' ? response.content[0].text : ''; + const encoder = new TextEncoder(); - // When sending the response, include CORS headers - return new Response(JSON.stringify({ "response": assistantResponse }), { + const streamResponse = new ReadableStream({ + async start(controller) { + for await (const chunk of stream) { + if (chunk.type === 'content_block_delta' && chunk.delta.type === 'text_delta') { + const bytes = encoder.encode(chunk.delta.text); + controller.enqueue(bytes); + } + } + controller.close(); + }, + }); + + return new Response(streamResponse, { headers: { - "Content-Type": "application/json", + "Content-Type": "text/plain; charset=utf-8", "Access-Control-Allow-Origin": "*", + "Cache-Control": "no-cache", + "Connection": "keep-alive", }, }); } catch (error) { diff --git a/frontend/components/editor/AIChat.tsx b/frontend/components/editor/AIChat.tsx index d89ef7e..a3073ec 100644 --- a/frontend/components/editor/AIChat.tsx +++ b/frontend/components/editor/AIChat.tsx @@ -1,4 +1,4 @@ -import React, { useState, useEffect, useRef } from 'react'; +import React, { useState, useEffect, useRef, useCallback } from 'react'; import { Button } from '../ui/button'; import { Send, StopCircle, Copy, Check, ChevronDown, ChevronUp, X, CornerUpLeft, Loader2 } from 'lucide-react'; import ReactMarkdown from 'react-markdown'; @@ -39,7 +39,7 @@ export default function AIChat() { } }; - const handleSend = async () => { + const handleSend = useCallback(async () => { if (input.trim() === '' && !context) return; const newMessage: Message = { @@ -51,20 +51,17 @@ export default function AIChat() { setInput(''); setIsContextExpanded(false); setIsGenerating(true); - setIsLoading(true); // Set loading state to true + setIsLoading(true); abortControllerRef.current = new AbortController(); try { const queryParams = new URLSearchParams({ instructions: input, - ...(context && { context }) // Include context only if it exists + ...(context && { context }) }); const response = await fetch(`http://127.0.0.1:8787/api?${queryParams}`, { method: 'GET', - headers: { - 'Content-Type': 'application/json', - }, signal: abortControllerRef.current.signal, }); @@ -72,22 +69,41 @@ export default function AIChat() { throw new Error('Failed to get AI response'); } - const data = await response.json(); + const reader = response.body?.getReader(); + const decoder = new TextDecoder(); const assistantMessage: Message = { role: 'assistant', content: '' }; setMessages(prev => [...prev, assistantMessage]); - setIsLoading(false); // Set loading state to false once we start receiving the response + setIsLoading(false); - // Simulate text generation - for (let i = 0; i <= data.response.length; i++) { - if (abortControllerRef.current.signal.aborted) { - break; + let buffer = ''; + const updateInterval = 100; // Update every 100ms + let lastUpdateTime = Date.now(); + + if (reader) { + while (true) { + const { done, value } = await reader.read(); + if (done) break; + buffer += decoder.decode(value, { stream: true }); + + const currentTime = Date.now(); + if (currentTime - lastUpdateTime > updateInterval) { + setMessages(prev => { + const updatedMessages = [...prev]; + const lastMessage = updatedMessages[updatedMessages.length - 1]; + lastMessage.content = buffer; + return updatedMessages; + }); + lastUpdateTime = currentTime; + } } + + // Final update to ensure all content is displayed setMessages(prev => { const updatedMessages = [...prev]; - updatedMessages[updatedMessages.length - 1].content = data.response.slice(0, i); + const lastMessage = updatedMessages[updatedMessages.length - 1]; + lastMessage.content = buffer; return updatedMessages; }); - await new Promise(resolve => setTimeout(resolve, 20)); } } catch (error: any) { if (error.name === 'AbortError') { @@ -99,10 +115,10 @@ export default function AIChat() { } } finally { setIsGenerating(false); - setIsLoading(false); // Ensure loading state is set to false + setIsLoading(false); abortControllerRef.current = null; } - }; + }, [input, context]); const handleStopGeneration = () => { if (abortControllerRef.current) { @@ -333,4 +349,4 @@ export default function AIChat() { ); -} +} \ No newline at end of file