refac: tool name collision handling

This commit is contained in:
Timothy Jaeryang Baek 2025-08-18 21:28:28 +04:00
parent f592748011
commit 70d0477418
2 changed files with 104 additions and 82 deletions

View file

@ -5,6 +5,7 @@ import inspect
import aiohttp
import asyncio
import yaml
import json
from pydantic import BaseModel
from pydantic.fields import FieldInfo
@ -85,7 +86,9 @@ async def get_tools(
tool_server_data = server
break
assert tool_server_data is not None
if tool_server_data is None:
log.warning(f"Tool server data not found for {server_id}")
continue
tool_server_idx = tool_server_data.get("idx", 0)
tool_server_connection = (
@ -131,13 +134,14 @@ async def get_tools(
"spec": spec,
}
# TODO: if collision, prepend toolkit name
if function_name in tools_dict:
# Handle function name collisions
while function_name in tools_dict:
log.warning(
f"Tool {function_name} already exists in another tools!"
)
log.warning(f"Discarding {tool_id}.{function_name}")
else:
# Prepend server ID to function name
function_name = f"{server_id}_{function_name}"
tools_dict[function_name] = tool_dict
else:
continue
@ -198,13 +202,14 @@ async def get_tools(
},
}
# TODO: if collision, prepend toolkit name
if function_name in tools_dict:
# Handle function name collisions
while function_name in tools_dict:
log.warning(
f"Tool {function_name} already exists in another tools!"
)
log.warning(f"Discarding {tool_id}.{function_name}")
else:
# Prepend tool ID to function name
function_name = f"{tool_id}_{function_name}"
tools_dict[function_name] = tool_dict
return tools_dict
@ -453,8 +458,8 @@ async def set_tool_servers(request: Request):
)
if request.app.state.redis is not None:
await request.app.state.redis.hmset(
"tool_servers", request.app.state.TOOL_SERVERS
await request.app.state.redis.set(
"tool_servers", json.dumps(request.app.state.TOOL_SERVERS)
)
return request.app.state.TOOL_SERVERS
@ -463,7 +468,10 @@ async def set_tool_servers(request: Request):
async def get_tool_servers(request: Request):
tool_servers = []
if request.app.state.redis is not None:
tool_servers = await request.app.state.redis.hgetall("tool_servers")
try:
tool_servers = json.loads(await request.app.state.redis.get("tool_servers"))
except Exception as e:
log.error(f"Error fetching tool_servers from Redis: {e}")
if not tool_servers:
await set_tool_servers(request)
@ -536,7 +544,10 @@ async def get_tool_servers_data(
elif auth_type == "session":
token = session_token
id = info.get("id", idx)
id = info.get("id")
if not id:
id = str(idx)
server_entries.append((id, idx, server, full_url, info, token))
# Create async tasks to fetch data

View file

@ -17,6 +17,7 @@
import CameraSolid from '$lib/components/icons/CameraSolid.svelte';
import PhotoSolid from '$lib/components/icons/PhotoSolid.svelte';
import CommandLineSolid from '$lib/components/icons/CommandLineSolid.svelte';
import Spinner from '$lib/components/common/Spinner.svelte';
const i18n = getContext('i18n');
@ -34,7 +35,7 @@
export let onClose: Function;
let tools = {};
let tools = null;
let show = false;
let showAllTools = false;
@ -49,7 +50,7 @@
const init = async () => {
await _tools.set(await getTools(localStorage.token));
if ($_tools) {
tools = $_tools.reduce((a, tool, i, arr) => {
a[tool.id] = {
name: tool.name,
@ -58,6 +59,8 @@
};
return a;
}, {});
selectedToolIds = selectedToolIds.filter((id) => $_tools?.some((tool) => tool.id === id));
}
};
const detectMobile = () => {
@ -105,6 +108,7 @@
align="start"
transition={flyAndScale}
>
{#if tools}
{#if Object.keys(tools).length > 0}
<div class="{showAllTools ? '' : 'max-h-28'} overflow-y-auto scrollbar-thin">
{#each Object.keys(tools) as toolId}
@ -170,6 +174,13 @@
{/if}
<hr class="border-black/5 dark:border-white/5 my-1" />
{/if}
{:else}
<div class="py-4">
<Spinner />
</div>
<hr class="border-black/5 dark:border-white/5 my-1" />
{/if}
<Tooltip
content={fileUploadCapableModels.length !== selectedModels.length