mirror of
				https://github.com/meta-llama/llama-stack.git
				synced 2025-10-25 09:05:37 +00:00 
			
		
		
		
	
		
			Some checks failed
		
		
	
	Integration Auth Tests / test-matrix (oauth2_token) (push) Failing after 1s
				
			Test External Providers Installed via Module / test-external-providers-from-module (venv) (push) Has been skipped
				
			Test Llama Stack Build / generate-matrix (push) Successful in 6s
				
			Python Package Build Test / build (3.12) (push) Failing after 9s
				
			Test Llama Stack Build / build-ubi9-container-distribution (push) Failing after 12s
				
			Integration Tests (Replay) / Integration Tests (, , , client=, vision=) (push) Failing after 14s
				
			Unit Tests / unit-tests (3.12) (push) Failing after 12s
				
			Vector IO Integration Tests / test-matrix (push) Failing after 16s
				
			Test Llama Stack Build / build-single-provider (push) Failing after 15s
				
			SqlStore Integration Tests / test-postgres (3.13) (push) Failing after 16s
				
			Test Llama Stack Build / build-custom-container-distribution (push) Failing after 14s
				
			Test External API and Providers / test-external (venv) (push) Failing after 14s
				
			Test Llama Stack Build / build (push) Failing after 9s
				
			Unit Tests / unit-tests (3.13) (push) Failing after 14s
				
			SqlStore Integration Tests / test-postgres (3.12) (push) Failing after 21s
				
			Update ReadTheDocs / update-readthedocs (push) Failing after 1m2s
				
			Python Package Build Test / build (3.13) (push) Failing after 1m4s
				
			UI Tests / ui-tests (22) (push) Successful in 1m33s
				
			Pre-commit / pre-commit (push) Successful in 2m38s
				
			
		
			
				
	
	
		
			240 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			240 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import React, { Suspense, useEffect, useState } from "react";
 | |
| import Markdown from "react-markdown";
 | |
| import remarkGfm from "remark-gfm";
 | |
| 
 | |
| import { cn } from "@/lib/utils";
 | |
| import { CopyButton } from "@/components/ui/copy-button";
 | |
| 
 | |
| interface MarkdownRendererProps {
 | |
|   children: string;
 | |
| }
 | |
| 
 | |
| export function MarkdownRenderer({ children }: MarkdownRendererProps) {
 | |
|   return (
 | |
|     <div className="space-y-3">
 | |
|       <Markdown remarkPlugins={[remarkGfm]} components={COMPONENTS}>
 | |
|         {children}
 | |
|       </Markdown>
 | |
|     </div>
 | |
|   );
 | |
| }
 | |
| 
 | |
| interface HighlightedPre extends React.HTMLAttributes<HTMLPreElement> {
 | |
|   children: string;
 | |
|   language: string;
 | |
| }
 | |
| 
 | |
| const HighlightedPre = React.memo(
 | |
|   ({ children, language, ...props }: HighlightedPre) => {
 | |
|     const [tokens, setTokens] = useState<unknown[] | null>(null);
 | |
|     const [isSupported, setIsSupported] = useState(false);
 | |
| 
 | |
|     useEffect(() => {
 | |
|       let mounted = true;
 | |
| 
 | |
|       const loadAndHighlight = async () => {
 | |
|         try {
 | |
|           const { codeToTokens, bundledLanguages } = await import("shiki");
 | |
| 
 | |
|           if (!mounted) return;
 | |
| 
 | |
|           if (!(language in bundledLanguages)) {
 | |
|             setIsSupported(false);
 | |
|             return;
 | |
|           }
 | |
| 
 | |
|           setIsSupported(true);
 | |
| 
 | |
|           const { tokens: highlightedTokens } = await codeToTokens(children, {
 | |
|             lang: language as keyof typeof bundledLanguages,
 | |
|             defaultColor: false,
 | |
|             themes: {
 | |
|               light: "github-light",
 | |
|               dark: "github-dark",
 | |
|             },
 | |
|           });
 | |
| 
 | |
|           if (mounted) {
 | |
|             setTokens(highlightedTokens);
 | |
|           }
 | |
|         } catch {
 | |
|           if (mounted) {
 | |
|             setIsSupported(false);
 | |
|           }
 | |
|         }
 | |
|       };
 | |
| 
 | |
|       loadAndHighlight();
 | |
| 
 | |
|       return () => {
 | |
|         mounted = false;
 | |
|       };
 | |
|     }, [children, language]);
 | |
| 
 | |
|     if (!isSupported) {
 | |
|       return <pre {...props}>{children}</pre>;
 | |
|     }
 | |
| 
 | |
|     if (!tokens) {
 | |
|       return <pre {...props}>{children}</pre>;
 | |
|     }
 | |
| 
 | |
|     return (
 | |
|       <pre {...props}>
 | |
|         <code>
 | |
|           {tokens.map((line, lineIndex) => (
 | |
|             <React.Fragment key={lineIndex}>
 | |
|               <span>
 | |
|                 {line.map((token, tokenIndex) => {
 | |
|                   const style =
 | |
|                     typeof token.htmlStyle === "string"
 | |
|                       ? undefined
 | |
|                       : token.htmlStyle;
 | |
| 
 | |
|                   return (
 | |
|                     <span
 | |
|                       key={tokenIndex}
 | |
|                       className="text-shiki-light bg-shiki-light-bg dark:text-shiki-dark dark:bg-shiki-dark-bg"
 | |
|                       style={style}
 | |
|                     >
 | |
|                       {token.content}
 | |
|                     </span>
 | |
|                   );
 | |
|                 })}
 | |
|               </span>
 | |
|               {lineIndex !== tokens.length - 1 && "\n"}
 | |
|             </React.Fragment>
 | |
|           ))}
 | |
|         </code>
 | |
|       </pre>
 | |
|     );
 | |
|   }
 | |
| );
 | |
| HighlightedPre.displayName = "HighlightedCode";
 | |
| 
 | |
| interface CodeBlockProps extends React.HTMLAttributes<HTMLPreElement> {
 | |
|   children: React.ReactNode;
 | |
|   className?: string;
 | |
|   language: string;
 | |
| }
 | |
| 
 | |
| const CodeBlock = ({
 | |
|   children,
 | |
|   className,
 | |
|   language,
 | |
|   ...restProps
 | |
| }: CodeBlockProps) => {
 | |
|   const code =
 | |
|     typeof children === "string"
 | |
|       ? children
 | |
|       : childrenTakeAllStringContents(children);
 | |
| 
 | |
|   const preClass = cn(
 | |
|     "overflow-x-scroll rounded-md border bg-background/50 p-4 font-mono text-sm [scrollbar-width:none]",
 | |
|     className
 | |
|   );
 | |
| 
 | |
|   return (
 | |
|     <div className="group/code relative mb-4">
 | |
|       <Suspense
 | |
|         fallback={
 | |
|           <pre className={preClass} {...restProps}>
 | |
|             {children}
 | |
|           </pre>
 | |
|         }
 | |
|       >
 | |
|         <HighlightedPre language={language} className={preClass}>
 | |
|           {code}
 | |
|         </HighlightedPre>
 | |
|       </Suspense>
 | |
| 
 | |
|       <div className="invisible absolute right-2 top-2 flex space-x-1 rounded-lg p-1 opacity-0 transition-all duration-200 group-hover/code:visible group-hover/code:opacity-100">
 | |
|         <CopyButton content={code} copyMessage="Copied code to clipboard" />
 | |
|       </div>
 | |
|     </div>
 | |
|   );
 | |
| };
 | |
| 
 | |
| function childrenTakeAllStringContents(element: unknown): string {
 | |
|   if (typeof element === "string") {
 | |
|     return element;
 | |
|   }
 | |
| 
 | |
|   if (element?.props?.children) {
 | |
|     const children = element.props.children;
 | |
| 
 | |
|     if (Array.isArray(children)) {
 | |
|       return children
 | |
|         .map(child => childrenTakeAllStringContents(child))
 | |
|         .join("");
 | |
|     } else {
 | |
|       return childrenTakeAllStringContents(children);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return "";
 | |
| }
 | |
| 
 | |
| const COMPONENTS = {
 | |
|   h1: withClass("h1", "text-2xl font-semibold"),
 | |
|   h2: withClass("h2", "font-semibold text-xl"),
 | |
|   h3: withClass("h3", "font-semibold text-lg"),
 | |
|   h4: withClass("h4", "font-semibold text-base"),
 | |
|   h5: withClass("h5", "font-medium"),
 | |
|   strong: withClass("strong", "font-semibold"),
 | |
|   a: withClass("a", "text-primary underline underline-offset-2"),
 | |
|   blockquote: withClass("blockquote", "border-l-2 border-primary pl-4"),
 | |
|   code: ({
 | |
|     children,
 | |
|     className,
 | |
|     ...rest
 | |
|   }: {
 | |
|     children: React.ReactNode;
 | |
|     className?: string;
 | |
|   }) => {
 | |
|     const match = /language-(\w+)/.exec(className || "");
 | |
|     return match ? (
 | |
|       <CodeBlock className={className} language={match[1]} {...rest}>
 | |
|         {children}
 | |
|       </CodeBlock>
 | |
|     ) : (
 | |
|       <code
 | |
|         className={cn(
 | |
|           "font-mono [:not(pre)>&]:rounded-md [:not(pre)>&]:bg-background/50 [:not(pre)>&]:px-1 [:not(pre)>&]:py-0.5"
 | |
|         )}
 | |
|         {...rest}
 | |
|       >
 | |
|         {children}
 | |
|       </code>
 | |
|     );
 | |
|   },
 | |
|   pre: ({ children }: { children: React.ReactNode }) => children,
 | |
|   ol: withClass("ol", "list-decimal space-y-2 pl-6"),
 | |
|   ul: withClass("ul", "list-disc space-y-2 pl-6"),
 | |
|   li: withClass("li", "my-1.5"),
 | |
|   table: withClass(
 | |
|     "table",
 | |
|     "w-full border-collapse overflow-y-auto rounded-md border border-foreground/20"
 | |
|   ),
 | |
|   th: withClass(
 | |
|     "th",
 | |
|     "border border-foreground/20 px-4 py-2 text-left font-bold [&[align=center]]:text-center [&[align=right]]:text-right"
 | |
|   ),
 | |
|   td: withClass(
 | |
|     "td",
 | |
|     "border border-foreground/20 px-4 py-2 text-left [&[align=center]]:text-center [&[align=right]]:text-right"
 | |
|   ),
 | |
|   tr: withClass("tr", "m-0 border-t p-0 even:bg-muted"),
 | |
|   p: withClass("p", "whitespace-pre-wrap"),
 | |
|   hr: withClass("hr", "border-foreground/20"),
 | |
| };
 | |
| 
 | |
| function withClass(Tag: keyof JSX.IntrinsicElements, classes: string) {
 | |
|   const Component = ({ ...props }: Record<string, unknown>) => (
 | |
|     <Tag className={classes} {...props} />
 | |
|   );
 | |
|   Component.displayName = Tag;
 | |
|   return Component;
 | |
| }
 | |
| 
 | |
| export default MarkdownRenderer;
 |