Move tool calls to a separate file.
This commit is contained in:
parent
e72cd513e5
commit
d32f675db8
|
|
@ -5,6 +5,7 @@ import { useState, useRef, useEffect, useMemo, useCallback } from "preact/hooks"
|
||||||
import LLM from "../utils/llm";
|
import LLM from "../utils/llm";
|
||||||
import { highlight } from "../utils/highlight";
|
import { highlight } from "../utils/highlight";
|
||||||
import Prompt from "../utils/prompt";
|
import Prompt from "../utils/prompt";
|
||||||
|
import { Tools } from "../utils/tools";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
|
|
||||||
export const ChatSidebar = () => {
|
export const ChatSidebar = () => {
|
||||||
|
|
@ -105,11 +106,11 @@ export const ChatSidebar = () => {
|
||||||
if (tool_calls) {
|
if (tool_calls) {
|
||||||
const toolMessages: ChatMessage[] = [];
|
const toolMessages: ChatMessage[] = [];
|
||||||
for (const tool of tool_calls) {
|
for (const tool of tool_calls) {
|
||||||
if (tool.function.name === 'test') {
|
const content = await Tools.executeTool(appState, tool);
|
||||||
const message: ChatMessage = {
|
const message: ChatMessage = {
|
||||||
id: crypto.randomUUID(),
|
id: crypto.randomUUID(),
|
||||||
role: 'tool',
|
role: 'tool',
|
||||||
content: `Test successful, received: ${JSON.stringify(tool.function.arguments)}`,
|
content,
|
||||||
tool_call_id: tool.id,
|
tool_call_id: tool.id,
|
||||||
};
|
};
|
||||||
dispatch({
|
dispatch({
|
||||||
|
|
@ -119,7 +120,6 @@ export const ChatSidebar = () => {
|
||||||
});
|
});
|
||||||
toolMessages.push(message);
|
toolMessages.push(message);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return sendMessage([...newMessages, assistantMessage, ...toolMessages]);
|
return sendMessage([...newMessages, assistantMessage, ...toolMessages]);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,27 +1,8 @@
|
||||||
import LLM from "./llm";
|
import LLM from "./llm";
|
||||||
import type { AppState } from "../contexts/state";
|
import type { AppState } from "../contexts/state";
|
||||||
|
import { Tools } from "./tools";
|
||||||
|
|
||||||
namespace Prompt {
|
namespace Prompt {
|
||||||
const tools: LLM.Tool[] = [
|
|
||||||
{
|
|
||||||
type: 'function',
|
|
||||||
function: {
|
|
||||||
name: 'test',
|
|
||||||
description: 'A simple test function',
|
|
||||||
parameters: {
|
|
||||||
type: 'object',
|
|
||||||
properties: {
|
|
||||||
message: {
|
|
||||||
type: 'string',
|
|
||||||
description: 'The test message',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
required: ['message'],
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
export function compilePrompt(state: AppState, newMessages: LLM.ChatMessage[] = []): LLM.ChatCompletionRequest | null {
|
export function compilePrompt(state: AppState, newMessages: LLM.ChatMessage[] = []): LLM.ChatCompletionRequest | null {
|
||||||
const { currentStory, model } = state;
|
const { currentStory, model } = state;
|
||||||
|
|
||||||
|
|
@ -40,7 +21,7 @@ namespace Prompt {
|
||||||
return {
|
return {
|
||||||
model,
|
model,
|
||||||
messages,
|
messages,
|
||||||
tools,
|
tools: Tools.getTools(),
|
||||||
// TODO banned_tokens
|
// TODO banned_tokens
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,65 @@
|
||||||
|
import { formatError } from "@common/errors";
|
||||||
|
import type { AppState } from "../contexts/state";
|
||||||
|
import LLM from "./llm";
|
||||||
|
|
||||||
|
export namespace Tools {
|
||||||
|
interface Tool {
|
||||||
|
description: string;
|
||||||
|
parameters: LLM.ToolObjectParameter;
|
||||||
|
handler(args: string | Record<string, any>, appState: AppState): unknown;
|
||||||
|
}
|
||||||
|
|
||||||
|
const TOOLS: Record<string, Tool> = {
|
||||||
|
'test': {
|
||||||
|
handler: async (args) => (
|
||||||
|
`Test successful, received: ${JSON.stringify(args)}`
|
||||||
|
),
|
||||||
|
description: 'A simple test function',
|
||||||
|
parameters: {
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
message: {
|
||||||
|
type: 'string',
|
||||||
|
description: 'The test message',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
required: ['message'],
|
||||||
|
},
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export function getTools(): LLM.Tool[] {
|
||||||
|
return Object.entries(TOOLS).map(([key, tool]) => {
|
||||||
|
return {
|
||||||
|
type: 'function',
|
||||||
|
function: {
|
||||||
|
name: key,
|
||||||
|
description: tool.description,
|
||||||
|
parameters: tool.parameters,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function executeTool(appState: AppState, toolCall: LLM.ToolCall): Promise<string> {
|
||||||
|
const { function: fn } = toolCall;
|
||||||
|
let args = fn.arguments;
|
||||||
|
try {
|
||||||
|
if (typeof fn.arguments === 'string') {
|
||||||
|
args = JSON.parse(fn.arguments);
|
||||||
|
}
|
||||||
|
} catch { }
|
||||||
|
|
||||||
|
const handler = TOOLS[fn.name]?.handler;
|
||||||
|
if (!handler) {
|
||||||
|
return `Unknown tool: ${fn.name}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await handler(args, appState);
|
||||||
|
return JSON.stringify(result);
|
||||||
|
} catch (err) {
|
||||||
|
return formatError(err, 'Error executing tool');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue