mirror of
https://github.com/meta-llama/llama-stack.git
synced 2025-10-24 00:47:00 +00:00
170 lines
4.6 KiB
TypeScript
170 lines
4.6 KiB
TypeScript
"use client";
|
|
|
|
import {
|
|
OpenAIResponse,
|
|
ResponseInputMessageContent,
|
|
UsePaginationOptions,
|
|
} from "@/lib/types";
|
|
import { LogsTable, LogTableRow } from "@/components/logs/logs-table";
|
|
import { usePagination } from "@/hooks/use-pagination";
|
|
import type { ResponseListResponse } from "llama-stack-client/resources/responses/responses";
|
|
import {
|
|
isMessageInput,
|
|
isMessageItem,
|
|
isFunctionCallItem,
|
|
isWebSearchCallItem,
|
|
MessageItem,
|
|
FunctionCallItem,
|
|
WebSearchCallItem,
|
|
} from "./utils/item-types";
|
|
|
|
interface ResponsesTableProps {
|
|
/** Optional pagination configuration */
|
|
paginationOptions?: UsePaginationOptions;
|
|
}
|
|
|
|
/**
|
|
* Helper function to convert ResponseListResponse.Data to OpenAIResponse
|
|
*/
|
|
const convertResponseListData = (
|
|
responseData: ResponseListResponse.Data
|
|
): OpenAIResponse => {
|
|
return {
|
|
id: responseData.id,
|
|
created_at: responseData.created_at,
|
|
model: responseData.model,
|
|
object: responseData.object,
|
|
status: responseData.status,
|
|
output: responseData.output as OpenAIResponse["output"],
|
|
input: responseData.input as OpenAIResponse["input"],
|
|
error: responseData.error,
|
|
parallel_tool_calls: responseData.parallel_tool_calls,
|
|
previous_response_id: responseData.previous_response_id,
|
|
temperature: responseData.temperature,
|
|
top_p: responseData.top_p,
|
|
truncation: responseData.truncation,
|
|
user: responseData.user,
|
|
};
|
|
};
|
|
|
|
function getInputText(response: OpenAIResponse): string {
|
|
const firstInput = response.input.find(isMessageInput);
|
|
if (firstInput) {
|
|
return extractContentFromItem(firstInput);
|
|
}
|
|
return "";
|
|
}
|
|
|
|
function getOutputText(response: OpenAIResponse): string {
|
|
const firstMessage = response.output.find(item =>
|
|
isMessageItem(item as Record<string, unknown>)
|
|
);
|
|
if (firstMessage) {
|
|
const content = extractContentFromItem(firstMessage as MessageItem);
|
|
if (content) {
|
|
return content;
|
|
}
|
|
}
|
|
|
|
const functionCall = response.output.find(item =>
|
|
isFunctionCallItem(item as Record<string, unknown>)
|
|
);
|
|
if (functionCall) {
|
|
return formatFunctionCall(functionCall as FunctionCallItem);
|
|
}
|
|
|
|
const webSearchCall = response.output.find(item =>
|
|
isWebSearchCallItem(item as Record<string, unknown>)
|
|
);
|
|
if (webSearchCall) {
|
|
return formatWebSearchCall(webSearchCall as WebSearchCallItem);
|
|
}
|
|
|
|
return JSON.stringify(response.output);
|
|
}
|
|
|
|
function extractContentFromItem(item: {
|
|
content?: string | ResponseInputMessageContent[];
|
|
}): string {
|
|
if (!item.content) {
|
|
return "";
|
|
}
|
|
|
|
if (typeof item.content === "string") {
|
|
return item.content;
|
|
} else if (Array.isArray(item.content)) {
|
|
const textContent = item.content.find(
|
|
(c: ResponseInputMessageContent) =>
|
|
c.type === "input_text" || c.type === "output_text"
|
|
);
|
|
return textContent?.text || "";
|
|
}
|
|
return "";
|
|
}
|
|
|
|
function formatFunctionCall(functionCall: FunctionCallItem): string {
|
|
const args = functionCall.arguments || "{}";
|
|
const name = functionCall.name || "unknown";
|
|
return `${name}(${args})`;
|
|
}
|
|
|
|
function formatWebSearchCall(webSearchCall: WebSearchCallItem): string {
|
|
return `web_search_call(status: ${webSearchCall.status})`;
|
|
}
|
|
|
|
function formatResponseToRow(response: OpenAIResponse): LogTableRow {
|
|
return {
|
|
id: response.id,
|
|
input: getInputText(response),
|
|
output: getOutputText(response),
|
|
model: response.model,
|
|
createdTime: new Date(response.created_at * 1000).toLocaleString(),
|
|
detailPath: `/logs/responses/${response.id}`,
|
|
};
|
|
}
|
|
|
|
export function ResponsesTable({ paginationOptions }: ResponsesTableProps) {
|
|
const fetchFunction = async (
|
|
client: ReturnType<typeof import("@/hooks/use-auth-client").useAuthClient>,
|
|
params: {
|
|
after?: string;
|
|
limit: number;
|
|
model?: string;
|
|
order?: string;
|
|
}
|
|
) => {
|
|
const response = await client.responses.list({
|
|
after: params.after,
|
|
limit: params.limit,
|
|
...(params.model && { model: params.model }),
|
|
...(params.order && { order: params.order }),
|
|
} as Parameters<typeof client.responses.list>[0]);
|
|
|
|
const listResponse = response as ResponseListResponse;
|
|
|
|
return {
|
|
...listResponse,
|
|
data: listResponse.data.map(convertResponseListData),
|
|
};
|
|
};
|
|
|
|
const { data, status, hasMore, error, loadMore } = usePagination({
|
|
...paginationOptions,
|
|
fetchFunction,
|
|
errorMessagePrefix: "responses",
|
|
});
|
|
|
|
const formattedData = data.map(formatResponseToRow);
|
|
|
|
return (
|
|
<LogsTable
|
|
data={formattedData}
|
|
status={status}
|
|
hasMore={hasMore}
|
|
error={error}
|
|
onLoadMore={loadMore}
|
|
caption="A list of your recent responses."
|
|
emptyMessage="No responses found."
|
|
/>
|
|
);
|
|
}
|