mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-09-22 08:12:00 +02:00

* refactor: MCP UI Separation for Agents (Dustin WIP) feat: separate MCPs into their own lists away from tools + actions and add the status indicator functionality from chat to their dropdown ui fix: spotify mcp was not persisting on agent creation feat: show disconnected saved servers and their tools in agent mcp list in created agents fix: select-all regression fixed (caused by deleting tools we were drawing from for rendering list) fix: dont show all mcps, only those installed in agent in list feat: separate ToolSelectDialog for MCPServerTools fix: uninitialized mcp servers not showing as added in toolselectdialog refactor: reduce looping in AgentPanelContext for categorizing groups and mcps refactor: split ToolSelectDialog and MCPToolSelectDialog functionality (still needs customization for custom user vars) chore: address ESLint comments chore: address ESLint comments feat: one-click initialization on MCP servers in agent builder fix: stop propagation triggering reinit on caret click refactor: split uninitialized MCPs component from initialized MCPs feat: new mcp tool select dialog ui with custom user vars feat: show initialization state for CUV configurable MCPs too chore: remove unused localization string fix: deselecting all tools caused a re-render fix: remove subtools so removal from MCPToolSelectDialog works more consistently feat: added servers have all tools enabled by default feat: mcp server list now alphabetical to prevent annoying ui behavior of servers jumping around depending on tool selection fix: filter out placeholder group mcp tools from any actual tool calls / definitions feat: indicator now takes you to config dialog for uninitialized servers feat: show previously configured mcp servers that are now missing from the yaml feat: select all enabled by default on first add to mcp server list chore: address ESLint comments * refactor: MCP UI Separation for Agents (Danny WIP) chore: remove use of `{serverName}_mcp_{serverName}` chore: import order WIP: separate component concerns refactor: streamline agent mcp tools refactor: unify MCP server handling and improve tool visibility logic, remove unnecessary normalization or sorting, remove nesting button, make variable names clear refactor: rename mcpServerIds to mcpServerNames for clarity and consistency across components refactor: remove groupedMCPTools and toolToServerMap, streamline MCP server handling in context and components to effectively utilize mcpServersMap refactor: optimize tool selection logic by replacing array includes with Set for improved performance chore: add error logging for failed auth URL parsing in ToolCall component refactor: enhance MCP tool handling by improving server name management and updating UI elements for better clarity * refactor: decouple connection status from useMCPServerManager with useMCPConnectionStatus * fix: improve MCP tool validation logic to handle unconfigured servers * chore: enhance log message clarity for MCP server disconnection in updateUserPluginsController * refactor: simplify connection status extraction in useMCPConnectionStatus hook * refactor: improve initializing UX * chore: replace string literal with ResourceType constant in useResourcePermissions * refactor: cleanup code, remove redundancies, rename variables for clarity * chore: add back filtering and sorting for mcp tools dialog * refactor: initializeServer to return response and early return * refactor: enhance server initialization logic and improve UI for OAuth interaction * chore: clarify warning message for unconfigured MCP server in handleTools * refactor: prevent CustomUserVarsSection from submitting tools dialog form * fix: nested button of button issue in UninitializedMCPTool * feat: add functionality to revoke custom user variables in MCPToolSelectDialog --------- Co-authored-by: Danny Avila <danny@librechat.ai>
116 lines
3.5 KiB
TypeScript
116 lines
3.5 KiB
TypeScript
import { XCircle, PlusCircleIcon, Wrench } from 'lucide-react';
|
|
import type { AgentToolType } from 'librechat-data-provider';
|
|
import { useLocalize } from '~/hooks';
|
|
|
|
type MCPToolItemProps = {
|
|
tool: AgentToolType;
|
|
onAddTool: () => void;
|
|
onRemoveTool: () => void;
|
|
isInstalled?: boolean;
|
|
isConfiguring?: boolean;
|
|
isInitializing?: boolean;
|
|
};
|
|
|
|
function MCPToolItem({
|
|
tool,
|
|
onAddTool,
|
|
onRemoveTool,
|
|
isInstalled = false,
|
|
isConfiguring = false,
|
|
isInitializing = false,
|
|
}: MCPToolItemProps) {
|
|
const localize = useLocalize();
|
|
const handleClick = () => {
|
|
if (isInstalled) {
|
|
onRemoveTool();
|
|
} else {
|
|
onAddTool();
|
|
}
|
|
};
|
|
|
|
const name = tool.metadata?.name || tool.tool_id;
|
|
const description = tool.metadata?.description || '';
|
|
const icon = tool.metadata?.icon;
|
|
|
|
// Determine button state and text
|
|
const getButtonState = () => {
|
|
if (isInstalled) {
|
|
return {
|
|
text: localize('com_nav_tool_remove'),
|
|
icon: <XCircle className="flex h-4 w-4 items-center stroke-2" />,
|
|
className:
|
|
'btn relative bg-gray-300 hover:bg-gray-400 dark:bg-gray-50 dark:hover:bg-gray-200',
|
|
disabled: false,
|
|
};
|
|
}
|
|
|
|
if (isConfiguring) {
|
|
return {
|
|
text: localize('com_ui_confirm'),
|
|
icon: <PlusCircleIcon className="flex h-4 w-4 items-center stroke-2" />,
|
|
className: 'btn btn-primary relative',
|
|
disabled: false,
|
|
};
|
|
}
|
|
|
|
if (isInitializing) {
|
|
return {
|
|
text: localize('com_ui_initializing'),
|
|
icon: <Wrench className="flex h-4 w-4 items-center stroke-2" />,
|
|
className: 'btn btn-primary relative opacity-75 cursor-not-allowed',
|
|
disabled: true,
|
|
};
|
|
}
|
|
|
|
return {
|
|
text: localize('com_ui_add'),
|
|
icon: <PlusCircleIcon className="flex h-4 w-4 items-center stroke-2" />,
|
|
className: 'btn btn-primary relative',
|
|
disabled: false,
|
|
};
|
|
};
|
|
|
|
const buttonState = getButtonState();
|
|
|
|
return (
|
|
<div className="flex flex-col gap-4 rounded border border-border-medium bg-transparent p-6">
|
|
<div className="flex gap-4">
|
|
<div className="h-[70px] w-[70px] shrink-0">
|
|
<div className="relative h-full w-full">
|
|
{icon ? (
|
|
<img
|
|
src={icon}
|
|
alt={localize('com_ui_logo', { 0: name })}
|
|
className="h-full w-full rounded-[5px] bg-white"
|
|
/>
|
|
) : (
|
|
<div className="flex h-full w-full items-center justify-center rounded-[5px] border border-border-medium bg-transparent">
|
|
<Wrench className="h-8 w-8 text-text-secondary" />
|
|
</div>
|
|
)}
|
|
<div className="absolute inset-0 rounded-[5px] ring-1 ring-inset ring-black/10"></div>
|
|
</div>
|
|
</div>
|
|
<div className="flex min-w-0 flex-col items-start justify-between">
|
|
<div className="mb-2 line-clamp-1 max-w-full text-lg leading-5 text-text-primary">
|
|
{name}
|
|
</div>
|
|
<button
|
|
className={buttonState.className}
|
|
aria-label={`${buttonState.text} ${name}`}
|
|
onClick={handleClick}
|
|
disabled={buttonState.disabled}
|
|
>
|
|
<div className="flex w-full items-center justify-center gap-2">
|
|
{buttonState.text}
|
|
{buttonState.icon}
|
|
</div>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<div className="line-clamp-3 h-[60px] text-sm text-text-secondary">{description}</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default MCPToolItem;
|