mirror of
				https://github.com/meta-llama/llama-stack.git
				synced 2025-10-25 01:01:13 +00:00 
			
		
		
		
	# What does this PR do? - Introduces the Agent Session creation for the Playground and allows users to set tools - note tools are actually not usable yet and this is marked explicitly - this also caches sessions locally for faster loading on the UI and deletes them appropriately - allows users to easily create new sessions as well - Moved Model Configuration settings and "System Message" / Prompt to the left component - Added new logo and favicon - Added new typing animation when LLM is generating ### Create New Session <img width="1916" height="1393" alt="Screenshot 2025-08-21 at 4 18 08 PM" src="https://github.com/user-attachments/assets/52c70ae3-a33e-4338-8522-8184c692c320" /> ### List of Sessions <img width="1920" height="1391" alt="Screenshot 2025-08-21 at 4 18 56 PM" src="https://github.com/user-attachments/assets/ed78c3c6-08ec-486c-8bad-9b7382c11360" /> <!-- If resolving an issue, uncomment and update the line below --> <!-- Closes #[issue-number] --> ## Test Plan Unit tests added --------- Signed-off-by: Francisco Javier Arceo <farceo@redhat.com>
		
			
				
	
	
		
			407 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			407 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| "use client";
 | |
| 
 | |
| import React, { useMemo, useState } from "react";
 | |
| import { cva, type VariantProps } from "class-variance-authority";
 | |
| import { motion } from "framer-motion";
 | |
| import { Ban, ChevronRight, Code2, Loader2, Terminal } from "lucide-react";
 | |
| 
 | |
| import { cn } from "@/lib/utils";
 | |
| import {
 | |
|   Collapsible,
 | |
|   CollapsibleContent,
 | |
|   CollapsibleTrigger,
 | |
| } from "@/components/ui/collapsible";
 | |
| import { FilePreview } from "@/components/ui/file-preview";
 | |
| import { MarkdownRenderer } from "@/components/chat-playground/markdown-renderer";
 | |
| 
 | |
| const chatBubbleVariants = cva(
 | |
|   "group/message relative break-words rounded-lg p-3 text-sm sm:max-w-[70%]",
 | |
|   {
 | |
|     variants: {
 | |
|       isUser: {
 | |
|         true: "bg-primary text-primary-foreground",
 | |
|         false: "bg-muted text-foreground",
 | |
|       },
 | |
|       animation: {
 | |
|         none: "",
 | |
|         slide: "duration-300 animate-in fade-in-0",
 | |
|         scale: "duration-300 animate-in fade-in-0 zoom-in-75",
 | |
|         fade: "duration-500 animate-in fade-in-0",
 | |
|       },
 | |
|     },
 | |
|     compoundVariants: [
 | |
|       {
 | |
|         isUser: true,
 | |
|         animation: "slide",
 | |
|         class: "slide-in-from-right",
 | |
|       },
 | |
|       {
 | |
|         isUser: false,
 | |
|         animation: "slide",
 | |
|         class: "slide-in-from-left",
 | |
|       },
 | |
|       {
 | |
|         isUser: true,
 | |
|         animation: "scale",
 | |
|         class: "origin-bottom-right",
 | |
|       },
 | |
|       {
 | |
|         isUser: false,
 | |
|         animation: "scale",
 | |
|         class: "origin-bottom-left",
 | |
|       },
 | |
|     ],
 | |
|   }
 | |
| );
 | |
| 
 | |
| type Animation = VariantProps<typeof chatBubbleVariants>["animation"];
 | |
| 
 | |
| interface Attachment {
 | |
|   name?: string;
 | |
|   contentType?: string;
 | |
|   url: string;
 | |
| }
 | |
| 
 | |
| interface PartialToolCall {
 | |
|   state: "partial-call";
 | |
|   toolName: string;
 | |
| }
 | |
| 
 | |
| interface ToolCall {
 | |
|   state: "call";
 | |
|   toolName: string;
 | |
| }
 | |
| 
 | |
| interface ToolResult {
 | |
|   state: "result";
 | |
|   toolName: string;
 | |
|   result: {
 | |
|     __cancelled?: boolean;
 | |
|     [key: string]: unknown;
 | |
|   };
 | |
| }
 | |
| 
 | |
| type ToolInvocation = PartialToolCall | ToolCall | ToolResult;
 | |
| 
 | |
| interface ReasoningPart {
 | |
|   type: "reasoning";
 | |
|   reasoning: string;
 | |
| }
 | |
| 
 | |
| interface ToolInvocationPart {
 | |
|   type: "tool-invocation";
 | |
|   toolInvocation: ToolInvocation;
 | |
| }
 | |
| 
 | |
| interface TextPart {
 | |
|   type: "text";
 | |
|   text: string;
 | |
| }
 | |
| 
 | |
| // For compatibility with AI SDK types, not used
 | |
| interface SourcePart {
 | |
|   type: "source";
 | |
|   source?: unknown;
 | |
| }
 | |
| 
 | |
| interface FilePart {
 | |
|   type: "file";
 | |
|   mimeType: string;
 | |
|   data: string;
 | |
| }
 | |
| 
 | |
| interface StepStartPart {
 | |
|   type: "step-start";
 | |
| }
 | |
| 
 | |
| type MessagePart =
 | |
|   | TextPart
 | |
|   | ReasoningPart
 | |
|   | ToolInvocationPart
 | |
|   | SourcePart
 | |
|   | FilePart
 | |
|   | StepStartPart;
 | |
| 
 | |
| export interface Message {
 | |
|   id: string;
 | |
|   role: "user" | "assistant" | (string & {});
 | |
|   content: string;
 | |
|   createdAt?: Date;
 | |
|   experimental_attachments?: Attachment[];
 | |
|   toolInvocations?: ToolInvocation[];
 | |
|   parts?: MessagePart[];
 | |
| }
 | |
| 
 | |
| export interface ChatMessageProps extends Message {
 | |
|   showTimeStamp?: boolean;
 | |
|   animation?: Animation;
 | |
|   actions?: React.ReactNode;
 | |
| }
 | |
| 
 | |
| export const ChatMessage: React.FC<ChatMessageProps> = ({
 | |
|   role,
 | |
|   content,
 | |
|   createdAt,
 | |
|   showTimeStamp = false,
 | |
|   animation = "scale",
 | |
|   actions,
 | |
|   experimental_attachments,
 | |
|   toolInvocations,
 | |
|   parts,
 | |
| }) => {
 | |
|   const files = useMemo(() => {
 | |
|     return experimental_attachments?.map(attachment => {
 | |
|       const dataArray = dataUrlToUint8Array(attachment.url);
 | |
|       const file = new File([dataArray], attachment.name ?? "Unknown", {
 | |
|         type: attachment.contentType,
 | |
|       });
 | |
|       return file;
 | |
|     });
 | |
|   }, [experimental_attachments]);
 | |
| 
 | |
|   const isUser = role === "user";
 | |
| 
 | |
|   const formattedTime = createdAt
 | |
|     ? new Date(createdAt).toLocaleTimeString("en-US", {
 | |
|         hour: "2-digit",
 | |
|         minute: "2-digit",
 | |
|       })
 | |
|     : undefined;
 | |
| 
 | |
|   if (isUser) {
 | |
|     return (
 | |
|       <div
 | |
|         className={cn("flex flex-col", isUser ? "items-end" : "items-start")}
 | |
|       >
 | |
|         {files ? (
 | |
|           <div className="mb-1 flex flex-wrap gap-2">
 | |
|             {files.map((file, index) => {
 | |
|               return <FilePreview file={file} key={index} />;
 | |
|             })}
 | |
|           </div>
 | |
|         ) : null}
 | |
| 
 | |
|         <div className={cn(chatBubbleVariants({ isUser, animation }))}>
 | |
|           <MarkdownRenderer>{content}</MarkdownRenderer>
 | |
|         </div>
 | |
| 
 | |
|         {showTimeStamp && createdAt ? (
 | |
|           <time
 | |
|             dateTime={new Date(createdAt).toISOString()}
 | |
|             className={cn(
 | |
|               "mt-1 block px-1 text-xs opacity-50",
 | |
|               animation !== "none" && "duration-500 animate-in fade-in-0"
 | |
|             )}
 | |
|           >
 | |
|             {formattedTime}
 | |
|           </time>
 | |
|         ) : null}
 | |
|       </div>
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   if (parts && parts.length > 0) {
 | |
|     return parts.map((part, index) => {
 | |
|       if (part.type === "text") {
 | |
|         return (
 | |
|           <div
 | |
|             className={cn(
 | |
|               "flex flex-col",
 | |
|               isUser ? "items-end" : "items-start"
 | |
|             )}
 | |
|             key={`text-${index}`}
 | |
|           >
 | |
|             <div className={cn(chatBubbleVariants({ isUser, animation }))}>
 | |
|               <MarkdownRenderer>{part.text}</MarkdownRenderer>
 | |
|               {actions ? (
 | |
|                 <div className="absolute -bottom-4 right-2 flex space-x-1 rounded-lg border bg-background p-1 text-foreground opacity-0 transition-opacity group-hover/message:opacity-100">
 | |
|                   {actions}
 | |
|                 </div>
 | |
|               ) : null}
 | |
|             </div>
 | |
| 
 | |
|             {showTimeStamp && createdAt ? (
 | |
|               <time
 | |
|                 dateTime={new Date(createdAt).toISOString()}
 | |
|                 className={cn(
 | |
|                   "mt-1 block px-1 text-xs opacity-50",
 | |
|                   animation !== "none" && "duration-500 animate-in fade-in-0"
 | |
|                 )}
 | |
|               >
 | |
|                 {formattedTime}
 | |
|               </time>
 | |
|             ) : null}
 | |
|           </div>
 | |
|         );
 | |
|       } else if (part.type === "reasoning") {
 | |
|         return <ReasoningBlock key={`reasoning-${index}`} part={part} />;
 | |
|       } else if (part.type === "tool-invocation") {
 | |
|         return (
 | |
|           <ToolCall
 | |
|             key={`tool-${index}`}
 | |
|             toolInvocations={[part.toolInvocation]}
 | |
|           />
 | |
|         );
 | |
|       }
 | |
|       return null;
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   if (toolInvocations && toolInvocations.length > 0) {
 | |
|     return <ToolCall toolInvocations={toolInvocations} />;
 | |
|   }
 | |
| 
 | |
|   return (
 | |
|     <div className={cn("flex flex-col", isUser ? "items-end" : "items-start")}>
 | |
|       <div className={cn(chatBubbleVariants({ isUser, animation }))}>
 | |
|         <MarkdownRenderer>{content}</MarkdownRenderer>
 | |
|         {actions ? (
 | |
|           <div className="absolute -bottom-4 right-2 flex space-x-1 rounded-lg border bg-background p-1 text-foreground opacity-0 transition-opacity group-hover/message:opacity-100">
 | |
|             {actions}
 | |
|           </div>
 | |
|         ) : null}
 | |
|       </div>
 | |
| 
 | |
|       {showTimeStamp && createdAt ? (
 | |
|         <time
 | |
|           dateTime={new Date(createdAt).toISOString()}
 | |
|           className={cn(
 | |
|             "mt-1 block px-1 text-xs opacity-50",
 | |
|             animation !== "none" && "duration-500 animate-in fade-in-0"
 | |
|           )}
 | |
|         >
 | |
|           {formattedTime}
 | |
|         </time>
 | |
|       ) : null}
 | |
|     </div>
 | |
|   );
 | |
| };
 | |
| 
 | |
| function dataUrlToUint8Array(data: string) {
 | |
|   const base64 = data.split(",")[1];
 | |
|   const buf = Buffer.from(base64, "base64");
 | |
|   return new Uint8Array(buf);
 | |
| }
 | |
| 
 | |
| const ReasoningBlock = ({ part }: { part: ReasoningPart }) => {
 | |
|   const [isOpen, setIsOpen] = useState(false);
 | |
| 
 | |
|   return (
 | |
|     <div className="mb-2 flex flex-col items-start sm:max-w-[70%]">
 | |
|       <Collapsible
 | |
|         open={isOpen}
 | |
|         onOpenChange={setIsOpen}
 | |
|         className="group w-full overflow-hidden rounded-lg border bg-muted/50"
 | |
|       >
 | |
|         <div className="flex items-center p-2">
 | |
|           <CollapsibleTrigger asChild>
 | |
|             <button className="flex items-center gap-2 text-sm text-muted-foreground hover:text-foreground">
 | |
|               <ChevronRight className="h-4 w-4 transition-transform group-data-[state=open]:rotate-90" />
 | |
|               <span>Thinking</span>
 | |
|             </button>
 | |
|           </CollapsibleTrigger>
 | |
|         </div>
 | |
|         <CollapsibleContent forceMount>
 | |
|           <motion.div
 | |
|             initial={false}
 | |
|             animate={isOpen ? "open" : "closed"}
 | |
|             variants={{
 | |
|               open: { height: "auto", opacity: 1 },
 | |
|               closed: { height: 0, opacity: 0 },
 | |
|             }}
 | |
|             transition={{ duration: 0.3, ease: [0.04, 0.62, 0.23, 0.98] }}
 | |
|             className="border-t"
 | |
|           >
 | |
|             <div className="p-2">
 | |
|               <div className="whitespace-pre-wrap text-xs">
 | |
|                 {part.reasoning}
 | |
|               </div>
 | |
|             </div>
 | |
|           </motion.div>
 | |
|         </CollapsibleContent>
 | |
|       </Collapsible>
 | |
|     </div>
 | |
|   );
 | |
| };
 | |
| 
 | |
| function ToolCall({
 | |
|   toolInvocations,
 | |
| }: Pick<ChatMessageProps, "toolInvocations">) {
 | |
|   if (!toolInvocations?.length) return null;
 | |
| 
 | |
|   return (
 | |
|     <div className="flex flex-col items-start gap-2">
 | |
|       {toolInvocations.map((invocation, index) => {
 | |
|         const isCancelled =
 | |
|           invocation.state === "result" &&
 | |
|           invocation.result.__cancelled === true;
 | |
| 
 | |
|         if (isCancelled) {
 | |
|           return (
 | |
|             <div
 | |
|               key={index}
 | |
|               className="flex items-center gap-2 rounded-lg border bg-muted/50 px-3 py-2 text-sm text-muted-foreground"
 | |
|             >
 | |
|               <Ban className="h-4 w-4" />
 | |
|               <span>
 | |
|                 Cancelled{" "}
 | |
|                 <span className="font-mono">
 | |
|                   {"`"}
 | |
|                   {invocation.toolName}
 | |
|                   {"`"}
 | |
|                 </span>
 | |
|               </span>
 | |
|             </div>
 | |
|           );
 | |
|         }
 | |
| 
 | |
|         switch (invocation.state) {
 | |
|           case "partial-call":
 | |
|           case "call":
 | |
|             return (
 | |
|               <div
 | |
|                 key={index}
 | |
|                 className="flex items-center gap-2 rounded-lg border bg-muted/50 px-3 py-2 text-sm text-muted-foreground"
 | |
|               >
 | |
|                 <Terminal className="h-4 w-4" />
 | |
|                 <span>
 | |
|                   Calling{" "}
 | |
|                   <span className="font-mono">
 | |
|                     {"`"}
 | |
|                     {invocation.toolName}
 | |
|                     {"`"}
 | |
|                   </span>
 | |
|                   ...
 | |
|                 </span>
 | |
|                 <Loader2 className="h-3 w-3 animate-spin" />
 | |
|               </div>
 | |
|             );
 | |
|           case "result":
 | |
|             return (
 | |
|               <div
 | |
|                 key={index}
 | |
|                 className="flex flex-col gap-1.5 rounded-lg border bg-muted/50 px-3 py-2 text-sm"
 | |
|               >
 | |
|                 <div className="flex items-center gap-2 text-muted-foreground">
 | |
|                   <Code2 className="h-4 w-4" />
 | |
|                   <span>
 | |
|                     Result from{" "}
 | |
|                     <span className="font-mono">
 | |
|                       {"`"}
 | |
|                       {invocation.toolName}
 | |
|                       {"`"}
 | |
|                     </span>
 | |
|                   </span>
 | |
|                 </div>
 | |
|                 <pre className="overflow-x-auto whitespace-pre-wrap text-foreground">
 | |
|                   {JSON.stringify(invocation.result, null, 2)}
 | |
|                 </pre>
 | |
|               </div>
 | |
|             );
 | |
|           default:
 | |
|             return null;
 | |
|         }
 | |
|       })}
 | |
|     </div>
 | |
|   );
 | |
| }
 |