llama.cpp/tools/server/webui/src/lib/components/app/mcp/McpSettingsSection.svelte

165 lines
4.5 KiB
Svelte

<script lang="ts">
import { Plus, X } from '@lucide/svelte';
import { Button } from '$lib/components/ui/button';
import * as Card from '$lib/components/ui/card';
import { parseMcpServerSettings } from '$lib/utils/mcp';
import type { MCPServerSettingsEntry } from '$lib/types/mcp';
import type { SettingsConfigType } from '$lib/types/settings';
import { DEFAULT_MCP_CONFIG } from '$lib/constants/mcp';
import { extractServerNameFromUrl, getFaviconUrl } from '$lib/utils/mcp';
import { McpServerCard } from '$lib/components/app/mcp/McpServerCard';
import McpServerForm from './McpServerForm.svelte';
interface Props {
localConfig: SettingsConfigType;
onConfigChange: (key: string, value: string | boolean) => void;
}
let { localConfig, onConfigChange }: Props = $props();
// Get servers from localConfig
let servers = $derived<MCPServerSettingsEntry[]>(parseMcpServerSettings(localConfig.mcpServers));
// New server form state
let isAddingServer = $state(false);
let newServerUrl = $state('');
let newServerHeaders = $state('');
// Validation for new server URL
let newServerUrlError = $derived.by(() => {
if (!newServerUrl.trim()) return 'URL is required';
try {
new URL(newServerUrl);
return null;
} catch {
return 'Invalid URL format';
}
});
function serializeServers(updatedServers: MCPServerSettingsEntry[]) {
onConfigChange('mcpServers', JSON.stringify(updatedServers));
}
function showAddServerForm() {
isAddingServer = true;
newServerUrl = '';
newServerHeaders = '';
}
function cancelAddServer() {
isAddingServer = false;
newServerUrl = '';
newServerHeaders = '';
}
function saveNewServer() {
if (newServerUrlError) return;
const newServer: MCPServerSettingsEntry = {
id: crypto.randomUUID ? crypto.randomUUID() : `server-${Date.now()}`,
enabled: true,
url: newServerUrl.trim(),
headers: newServerHeaders.trim() || undefined,
requestTimeoutSeconds: DEFAULT_MCP_CONFIG.requestTimeoutSeconds
};
serializeServers([...servers, newServer]);
isAddingServer = false;
newServerUrl = '';
newServerHeaders = '';
}
function updateServer(id: string, updates: Partial<MCPServerSettingsEntry>) {
const nextServers = servers.map((server) =>
server.id === id ? { ...server, ...updates } : server
);
serializeServers(nextServers);
}
function removeServer(id: string) {
serializeServers(servers.filter((server) => server.id !== id));
}
// Get display name for server
function getServerDisplayName(server: MCPServerSettingsEntry): string {
if (server.name) return server.name;
return extractServerNameFromUrl(server.url);
}
</script>
<div class="space-y-4">
<div class="flex items-center justify-between gap-4">
<div>
<h4 class="text-base font-semibold">Manage Servers</h4>
</div>
{#if !isAddingServer}
<Button variant="outline" size="sm" class="shrink-0" onclick={showAddServerForm}>
<Plus class="h-4 w-4" />
Add New Server
</Button>
{/if}
</div>
<!-- Add New Server Form -->
{#if isAddingServer}
<Card.Root class="bg-muted/30 p-4">
<div class="space-y-4">
<div class="flex items-center justify-between">
<p class="font-medium">Add New Server</p>
<Button
variant="ghost"
size="icon"
class="h-7 w-7"
onclick={cancelAddServer}
aria-label="Cancel"
>
<X class="h-3.5 w-3.5" />
</Button>
</div>
<McpServerForm
url={newServerUrl}
headers={newServerHeaders}
onUrlChange={(v) => (newServerUrl = v)}
onHeadersChange={(v) => (newServerHeaders = v)}
urlError={newServerUrl ? newServerUrlError : null}
id="new-server"
/>
<div class="flex items-center justify-end">
<Button
variant="default"
size="sm"
onclick={saveNewServer}
disabled={!!newServerUrlError}
aria-label="Save"
>
Add
</Button>
</div>
</div>
</Card.Root>
{/if}
{#if servers.length === 0 && !isAddingServer}
<div class="rounded-md border border-dashed p-4 text-sm text-muted-foreground">
No MCP Servers configured yet. Add one to enable agentic features.
</div>
{/if}
<!-- Server Cards -->
{#if servers.length > 0}
<div class="space-y-3">
{#each servers as server (server.id)}
<McpServerCard
{server}
displayName={getServerDisplayName(server)}
faviconUrl={getFaviconUrl(server.url)}
onToggle={(enabled) => updateServer(server.id, { enabled })}
onUpdate={(updates) => updateServer(server.id, updates)}
onDelete={() => removeServer(server.id)}
/>
{/each}
</div>
{/if}
</div>