feat: optimized agent response time

This commit is contained in:
Akhileshrangani4 2024-10-13 23:04:16 -04:00
parent dd59608d73
commit 1365fecb08
2 changed files with 55 additions and 25 deletions

View File

@ -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) {

View File

@ -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) {