2025-06-09 15:50:15 -07:00
|
|
|
|
"use client";
|
|
|
|
|
|
|
2025-12-11 13:42:33 +02:00
|
|
|
|
import { useAtomValue } from "jotai";
|
2025-07-27 10:05:37 -07:00
|
|
|
|
import {
|
2025-07-27 10:41:15 -07:00
|
|
|
|
AlertCircle,
|
|
|
|
|
|
Bot,
|
2025-11-13 12:46:48 -08:00
|
|
|
|
Check,
|
2025-07-27 10:41:15 -07:00
|
|
|
|
CheckCircle,
|
2025-11-13 12:46:48 -08:00
|
|
|
|
ChevronsUpDown,
|
2025-07-27 10:41:15 -07:00
|
|
|
|
Clock,
|
|
|
|
|
|
Edit3,
|
|
|
|
|
|
Loader2,
|
|
|
|
|
|
Plus,
|
|
|
|
|
|
RefreshCw,
|
|
|
|
|
|
Settings2,
|
|
|
|
|
|
Trash2,
|
|
|
|
|
|
} from "lucide-react";
|
2025-09-30 21:53:10 -07:00
|
|
|
|
import { AnimatePresence, motion } from "motion/react";
|
2025-07-27 10:41:15 -07:00
|
|
|
|
import { useEffect, useState } from "react";
|
|
|
|
|
|
import { toast } from "sonner";
|
2025-12-11 13:42:33 +02:00
|
|
|
|
import {
|
|
|
|
|
|
createLLMConfigMutationAtom,
|
|
|
|
|
|
deleteLLMConfigMutationAtom,
|
|
|
|
|
|
updateLLMConfigMutationAtom,
|
|
|
|
|
|
} from "@/atoms/llm-config/llm-config-mutation.atoms";
|
|
|
|
|
|
import { globalLLMConfigsAtom, llmConfigsAtom } from "@/atoms/llm-config/llm-config-query.atoms";
|
2025-07-27 10:41:15 -07:00
|
|
|
|
import { Alert, AlertDescription } from "@/components/ui/alert";
|
2025-12-13 22:43:38 -08:00
|
|
|
|
import {
|
|
|
|
|
|
AlertDialog,
|
|
|
|
|
|
AlertDialogAction,
|
|
|
|
|
|
AlertDialogCancel,
|
|
|
|
|
|
AlertDialogContent,
|
|
|
|
|
|
AlertDialogDescription,
|
|
|
|
|
|
AlertDialogFooter,
|
|
|
|
|
|
AlertDialogHeader,
|
|
|
|
|
|
AlertDialogTitle,
|
|
|
|
|
|
} from "@/components/ui/alert-dialog";
|
2025-07-27 10:05:37 -07:00
|
|
|
|
import { Badge } from "@/components/ui/badge";
|
2025-07-27 10:41:15 -07:00
|
|
|
|
import { Button } from "@/components/ui/button";
|
|
|
|
|
|
import { Card, CardContent } from "@/components/ui/card";
|
2025-11-13 12:46:48 -08:00
|
|
|
|
import {
|
|
|
|
|
|
Command,
|
|
|
|
|
|
CommandEmpty,
|
|
|
|
|
|
CommandGroup,
|
|
|
|
|
|
CommandInput,
|
|
|
|
|
|
CommandItem,
|
|
|
|
|
|
CommandList,
|
|
|
|
|
|
} from "@/components/ui/command";
|
2025-07-27 10:05:37 -07:00
|
|
|
|
import {
|
|
|
|
|
|
Dialog,
|
|
|
|
|
|
DialogContent,
|
|
|
|
|
|
DialogDescription,
|
|
|
|
|
|
DialogHeader,
|
|
|
|
|
|
DialogTitle,
|
|
|
|
|
|
} from "@/components/ui/dialog";
|
2025-07-27 10:41:15 -07:00
|
|
|
|
import { Input } from "@/components/ui/input";
|
|
|
|
|
|
import { Label } from "@/components/ui/label";
|
2025-11-13 12:46:48 -08:00
|
|
|
|
import { Popover, PopoverContent, PopoverTrigger } from "@/components/ui/popover";
|
2025-07-27 10:05:37 -07:00
|
|
|
|
import {
|
2025-07-27 10:41:15 -07:00
|
|
|
|
Select,
|
|
|
|
|
|
SelectContent,
|
|
|
|
|
|
SelectItem,
|
|
|
|
|
|
SelectTrigger,
|
|
|
|
|
|
SelectValue,
|
|
|
|
|
|
} from "@/components/ui/select";
|
2025-10-12 13:13:42 +05:30
|
|
|
|
import { LANGUAGES } from "@/contracts/enums/languages";
|
2025-11-13 02:41:30 -08:00
|
|
|
|
import { getModelsByProvider } from "@/contracts/enums/llm-models";
|
2025-10-13 20:11:11 -07:00
|
|
|
|
import { LLM_PROVIDERS } from "@/contracts/enums/llm-providers";
|
2025-12-11 13:42:33 +02:00
|
|
|
|
import type {
|
|
|
|
|
|
CreateLLMConfigRequest,
|
|
|
|
|
|
CreateLLMConfigResponse,
|
|
|
|
|
|
LLMConfig,
|
|
|
|
|
|
UpdateLLMConfigResponse,
|
|
|
|
|
|
} from "@/contracts/types/llm-config.types";
|
2025-11-13 12:46:48 -08:00
|
|
|
|
import { cn } from "@/lib/utils";
|
2025-10-07 04:59:01 +05:30
|
|
|
|
import InferenceParamsEditor from "../inference-params-editor";
|
2025-06-09 15:50:15 -07:00
|
|
|
|
|
2025-10-10 00:50:29 -07:00
|
|
|
|
interface ModelConfigManagerProps {
|
|
|
|
|
|
searchSpaceId: number;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function ModelConfigManager({ searchSpaceId }: ModelConfigManagerProps) {
|
2025-12-11 13:42:33 +02:00
|
|
|
|
const {
|
|
|
|
|
|
mutateAsync: createLLMConfig,
|
|
|
|
|
|
isPending: isCreatingLLMConfig,
|
|
|
|
|
|
error: createLLMConfigError,
|
|
|
|
|
|
} = useAtomValue(createLLMConfigMutationAtom);
|
|
|
|
|
|
const {
|
|
|
|
|
|
mutateAsync: updateLLMConfig,
|
|
|
|
|
|
isPending: isUpdatingLLMConfig,
|
|
|
|
|
|
error: updateLLMConfigError,
|
|
|
|
|
|
} = useAtomValue(updateLLMConfigMutationAtom);
|
|
|
|
|
|
const {
|
|
|
|
|
|
mutateAsync: deleteLLMConfig,
|
|
|
|
|
|
isPending: isDeletingLLMConfig,
|
|
|
|
|
|
error: deleteLLMConfigError,
|
|
|
|
|
|
} = useAtomValue(deleteLLMConfigMutationAtom);
|
|
|
|
|
|
const {
|
|
|
|
|
|
data: llmConfigs,
|
|
|
|
|
|
isFetching: isFetchingLLMConfigs,
|
|
|
|
|
|
error: LLMConfigsFetchError,
|
|
|
|
|
|
refetch: refreshConfigs,
|
|
|
|
|
|
} = useAtomValue(llmConfigsAtom);
|
|
|
|
|
|
const { data: globalConfigs = [] } = useAtomValue(globalLLMConfigsAtom);
|
2025-07-27 10:05:37 -07:00
|
|
|
|
const [isAddingNew, setIsAddingNew] = useState(false);
|
|
|
|
|
|
const [editingConfig, setEditingConfig] = useState<LLMConfig | null>(null);
|
2025-12-09 19:39:25 +00:00
|
|
|
|
const [formData, setFormData] = useState<CreateLLMConfigRequest>({
|
2025-07-27 10:05:37 -07:00
|
|
|
|
name: "",
|
2025-12-09 19:39:25 +00:00
|
|
|
|
provider: "" as CreateLLMConfigRequest["provider"], // Allow it as Default,
|
2025-07-27 10:05:37 -07:00
|
|
|
|
custom_provider: "",
|
|
|
|
|
|
model_name: "",
|
|
|
|
|
|
api_key: "",
|
|
|
|
|
|
api_base: "",
|
2025-10-12 13:13:42 +05:30
|
|
|
|
language: "English",
|
2025-07-27 10:05:37 -07:00
|
|
|
|
litellm_params: {},
|
2025-10-10 00:50:29 -07:00
|
|
|
|
search_space_id: searchSpaceId,
|
2025-07-27 10:05:37 -07:00
|
|
|
|
});
|
2025-12-11 13:42:33 +02:00
|
|
|
|
const isSubmitting = isCreatingLLMConfig || isUpdatingLLMConfig;
|
|
|
|
|
|
const errors = [
|
|
|
|
|
|
createLLMConfigError,
|
|
|
|
|
|
updateLLMConfigError,
|
|
|
|
|
|
deleteLLMConfigError,
|
|
|
|
|
|
LLMConfigsFetchError,
|
|
|
|
|
|
] as Error[];
|
|
|
|
|
|
const isError = Boolean(errors.filter(Boolean).length);
|
2025-11-13 12:46:48 -08:00
|
|
|
|
const [modelComboboxOpen, setModelComboboxOpen] = useState(false);
|
2025-12-13 22:43:38 -08:00
|
|
|
|
const [configToDelete, setConfigToDelete] = useState<LLMConfig | null>(null);
|
|
|
|
|
|
const [isDeleting, setIsDeleting] = useState(false);
|
2025-07-27 10:05:37 -07:00
|
|
|
|
|
|
|
|
|
|
// Populate form when editing
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (editingConfig) {
|
|
|
|
|
|
setFormData({
|
|
|
|
|
|
name: editingConfig.name,
|
|
|
|
|
|
provider: editingConfig.provider,
|
|
|
|
|
|
custom_provider: editingConfig.custom_provider || "",
|
|
|
|
|
|
model_name: editingConfig.model_name,
|
|
|
|
|
|
api_key: editingConfig.api_key,
|
|
|
|
|
|
api_base: editingConfig.api_base || "",
|
2025-10-12 13:44:45 +05:30
|
|
|
|
language: editingConfig.language || "English",
|
2025-07-27 10:05:37 -07:00
|
|
|
|
litellm_params: editingConfig.litellm_params || {},
|
2025-10-10 00:50:29 -07:00
|
|
|
|
search_space_id: searchSpaceId,
|
2025-07-27 10:05:37 -07:00
|
|
|
|
});
|
|
|
|
|
|
}
|
2025-10-10 00:50:29 -07:00
|
|
|
|
}, [editingConfig, searchSpaceId]);
|
2025-07-27 10:05:37 -07:00
|
|
|
|
|
2025-12-09 19:39:25 +00:00
|
|
|
|
const handleInputChange = (field: keyof CreateLLMConfigRequest, value: string) => {
|
2025-07-27 10:05:37 -07:00
|
|
|
|
setFormData((prev) => ({ ...prev, [field]: value }));
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-11-13 02:41:30 -08:00
|
|
|
|
// Handle provider change with auto-fill API Base URL and reset model / 处理 Provider 变更并自动填充 API Base URL 并重置模型
|
2025-12-11 13:42:33 +02:00
|
|
|
|
const handleProviderChange = (providerValue: CreateLLMConfigRequest["provider"]) => {
|
2025-10-12 19:10:46 +08:00
|
|
|
|
const provider = LLM_PROVIDERS.find((p) => p.value === providerValue);
|
|
|
|
|
|
setFormData((prev) => ({
|
|
|
|
|
|
...prev,
|
|
|
|
|
|
provider: providerValue,
|
2025-11-13 02:41:30 -08:00
|
|
|
|
model_name: "", // Reset model when provider changes
|
2025-10-12 19:10:46 +08:00
|
|
|
|
// Auto-fill API Base URL if provider has a default / 如果提供商有默认值则自动填充
|
|
|
|
|
|
api_base: provider?.apiBase || prev.api_base,
|
|
|
|
|
|
}));
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-07-27 10:05:37 -07:00
|
|
|
|
const handleSubmit = async (e: React.FormEvent) => {
|
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
|
if (!formData.name || !formData.provider || !formData.model_name || !formData.api_key) {
|
|
|
|
|
|
toast.error("Please fill in all required fields");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-09 19:39:25 +00:00
|
|
|
|
let result: CreateLLMConfigResponse | UpdateLLMConfigResponse | null = null;
|
2025-07-27 10:05:37 -07:00
|
|
|
|
if (editingConfig) {
|
|
|
|
|
|
// Update existing config
|
2025-12-11 13:42:33 +02:00
|
|
|
|
result = await updateLLMConfig({ id: editingConfig.id, data: formData });
|
2025-07-27 10:05:37 -07:00
|
|
|
|
} else {
|
|
|
|
|
|
// Create new config
|
|
|
|
|
|
result = await createLLMConfig(formData);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (result) {
|
|
|
|
|
|
setFormData({
|
|
|
|
|
|
name: "",
|
2025-12-09 19:39:25 +00:00
|
|
|
|
provider: "" as CreateLLMConfigRequest["provider"],
|
2025-07-27 10:05:37 -07:00
|
|
|
|
custom_provider: "",
|
|
|
|
|
|
model_name: "",
|
|
|
|
|
|
api_key: "",
|
|
|
|
|
|
api_base: "",
|
2025-10-12 13:13:42 +05:30
|
|
|
|
language: "English",
|
2025-07-27 10:05:37 -07:00
|
|
|
|
litellm_params: {},
|
2025-10-10 00:50:29 -07:00
|
|
|
|
search_space_id: searchSpaceId,
|
2025-07-27 10:05:37 -07:00
|
|
|
|
});
|
|
|
|
|
|
setIsAddingNew(false);
|
|
|
|
|
|
setEditingConfig(null);
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-12-13 22:43:38 -08:00
|
|
|
|
const handleDeleteClick = (config: LLMConfig) => {
|
|
|
|
|
|
setConfigToDelete(config);
|
2025-07-27 10:05:37 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
2025-12-13 22:43:38 -08:00
|
|
|
|
const handleConfirmDelete = async () => {
|
|
|
|
|
|
if (!configToDelete) return;
|
|
|
|
|
|
try {
|
2025-12-14 20:38:30 +02:00
|
|
|
|
await deleteLLMConfig({ id: configToDelete.id });
|
2025-12-13 22:43:38 -08:00
|
|
|
|
} catch (error) {
|
|
|
|
|
|
toast.error("Failed to delete configuration");
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
setConfigToDelete(null);
|
|
|
|
|
|
}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const selectedProvider = LLM_PROVIDERS.find((p) => p.value === formData.provider);
|
2025-11-13 02:41:30 -08:00
|
|
|
|
const availableModels = formData.provider ? getModelsByProvider(formData.provider) : [];
|
2025-07-27 10:05:37 -07:00
|
|
|
|
|
|
|
|
|
|
const getProviderInfo = (providerValue: string) => {
|
|
|
|
|
|
return LLM_PROVIDERS.find((p) => p.value === providerValue);
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div className="space-y-6">
|
|
|
|
|
|
{/* Header */}
|
|
|
|
|
|
<div className="flex flex-col space-y-4 sm:flex-row sm:items-center sm:justify-between sm:space-y-0">
|
|
|
|
|
|
<div className="space-y-1">
|
|
|
|
|
|
<div className="flex items-center space-x-3">
|
|
|
|
|
|
<div className="flex h-10 w-10 items-center justify-center rounded-lg bg-blue-500/10">
|
|
|
|
|
|
<Settings2 className="h-5 w-5 text-blue-600" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<h2 className="text-2xl font-bold tracking-tight">Model Configurations</h2>
|
|
|
|
|
|
<p className="text-muted-foreground">
|
|
|
|
|
|
Manage your LLM provider configurations and API settings.
|
|
|
|
|
|
</p>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="flex items-center space-x-2">
|
|
|
|
|
|
<Button
|
|
|
|
|
|
variant="outline"
|
|
|
|
|
|
size="sm"
|
2025-12-09 19:39:25 +00:00
|
|
|
|
onClick={() => refreshConfigs()}
|
|
|
|
|
|
disabled={isFetchingLLMConfigs}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
className="flex items-center gap-2"
|
|
|
|
|
|
>
|
2025-12-09 19:39:25 +00:00
|
|
|
|
<RefreshCw className={`h-4 w-4 ${isFetchingLLMConfigs ? "animate-spin" : ""}`} />
|
2025-07-27 10:05:37 -07:00
|
|
|
|
Refresh
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
{/* Error Alert */}
|
2025-12-11 13:42:33 +02:00
|
|
|
|
{isError &&
|
|
|
|
|
|
errors.filter(Boolean).map((err, i) => {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Alert key={`err.message-${i}`} variant="destructive">
|
|
|
|
|
|
<AlertCircle className="h-4 w-4" />
|
|
|
|
|
|
<AlertDescription>{err?.message ?? "Something went wrong"}</AlertDescription>
|
|
|
|
|
|
</Alert>
|
|
|
|
|
|
);
|
|
|
|
|
|
})}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
|
2025-11-14 21:53:46 -08:00
|
|
|
|
{/* Global Configs Info Alert */}
|
2025-12-09 19:39:25 +00:00
|
|
|
|
{!isFetchingLLMConfigs && !isError && globalConfigs.length > 0 && (
|
2025-11-14 21:53:46 -08:00
|
|
|
|
<Alert>
|
|
|
|
|
|
<CheckCircle className="h-4 w-4" />
|
|
|
|
|
|
<AlertDescription>
|
|
|
|
|
|
<strong>
|
|
|
|
|
|
{globalConfigs.length} global configuration{globalConfigs.length > 1 ? "s" : ""}
|
|
|
|
|
|
</strong>{" "}
|
|
|
|
|
|
available for use. You can assign them in the LLM Roles tab without adding your own API
|
|
|
|
|
|
keys.
|
|
|
|
|
|
</AlertDescription>
|
|
|
|
|
|
</Alert>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
2025-07-27 10:05:37 -07:00
|
|
|
|
{/* Loading State */}
|
2025-12-09 19:39:25 +00:00
|
|
|
|
{isFetchingLLMConfigs && (
|
2025-07-27 10:05:37 -07:00
|
|
|
|
<Card>
|
|
|
|
|
|
<CardContent className="flex items-center justify-center py-12">
|
|
|
|
|
|
<div className="flex items-center gap-2 text-muted-foreground">
|
|
|
|
|
|
<Loader2 className="w-5 h-5 animate-spin" />
|
|
|
|
|
|
<span>Loading configurations...</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</CardContent>
|
|
|
|
|
|
</Card>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
|
|
{/* Stats Overview */}
|
2025-12-11 13:42:33 +02:00
|
|
|
|
{!isFetchingLLMConfigs && !isError && (
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<div className="grid gap-3 grid-cols-3">
|
|
|
|
|
|
<Card className="overflow-hidden">
|
|
|
|
|
|
<div className="h-1 bg-blue-500" />
|
|
|
|
|
|
<CardContent className="p-4">
|
|
|
|
|
|
<div className="flex items-start justify-between gap-2">
|
|
|
|
|
|
<div className="space-y-1 min-w-0">
|
2025-12-09 19:39:25 +00:00
|
|
|
|
<p className="text-2xl font-bold tracking-tight">{llmConfigs?.length}</p>
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<p className="text-xs font-medium text-muted-foreground">Total Configs</p>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<div className="flex h-9 w-9 shrink-0 items-center justify-center rounded-lg bg-blue-500/10">
|
|
|
|
|
|
<Bot className="h-4 w-4 text-blue-600" />
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</CardContent>
|
|
|
|
|
|
</Card>
|
|
|
|
|
|
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<Card className="overflow-hidden">
|
|
|
|
|
|
<div className="h-1 bg-green-500" />
|
|
|
|
|
|
<CardContent className="p-4">
|
|
|
|
|
|
<div className="flex items-start justify-between gap-2">
|
|
|
|
|
|
<div className="space-y-1 min-w-0">
|
|
|
|
|
|
<p className="text-2xl font-bold tracking-tight">
|
2025-12-09 19:39:25 +00:00
|
|
|
|
{new Set(llmConfigs?.map((c) => c.provider)).size}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</p>
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<p className="text-xs font-medium text-muted-foreground">Providers</p>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<div className="flex h-9 w-9 shrink-0 items-center justify-center rounded-lg bg-green-500/10">
|
|
|
|
|
|
<CheckCircle className="h-4 w-4 text-green-600" />
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</CardContent>
|
|
|
|
|
|
</Card>
|
|
|
|
|
|
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<Card className="overflow-hidden">
|
|
|
|
|
|
<div className="h-1 bg-emerald-500" />
|
|
|
|
|
|
<CardContent className="p-4">
|
|
|
|
|
|
<div className="flex items-start justify-between gap-2">
|
|
|
|
|
|
<div className="space-y-1 min-w-0">
|
|
|
|
|
|
<p className="text-2xl font-bold tracking-tight text-emerald-600">Active</p>
|
|
|
|
|
|
<p className="text-xs font-medium text-muted-foreground">Status</p>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<div className="flex h-9 w-9 shrink-0 items-center justify-center rounded-lg bg-emerald-500/10">
|
|
|
|
|
|
<CheckCircle className="h-4 w-4 text-emerald-600" />
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</CardContent>
|
|
|
|
|
|
</Card>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
|
|
{/* Configuration Management */}
|
2025-12-09 19:39:25 +00:00
|
|
|
|
{!isFetchingLLMConfigs && !isError && (
|
2025-07-27 10:05:37 -07:00
|
|
|
|
<div className="space-y-6">
|
|
|
|
|
|
<div className="flex flex-col space-y-4 sm:flex-row sm:items-center sm:justify-between sm:space-y-0">
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<h3 className="text-xl font-semibold tracking-tight">Your Configurations</h3>
|
|
|
|
|
|
<p className="text-sm text-muted-foreground">
|
|
|
|
|
|
Manage and configure your LLM providers
|
|
|
|
|
|
</p>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<Button onClick={() => setIsAddingNew(true)} className="flex items-center gap-2">
|
|
|
|
|
|
<Plus className="h-4 w-4" />
|
|
|
|
|
|
Add Configuration
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2025-12-09 19:39:25 +00:00
|
|
|
|
{llmConfigs?.length === 0 ? (
|
2025-07-27 10:05:37 -07:00
|
|
|
|
<Card className="border-dashed border-2 border-muted-foreground/25">
|
|
|
|
|
|
<CardContent className="flex flex-col items-center justify-center py-16 text-center">
|
|
|
|
|
|
<div className="rounded-full bg-muted p-4 mb-6">
|
|
|
|
|
|
<Bot className="h-10 w-10 text-muted-foreground" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2 mb-6">
|
|
|
|
|
|
<h3 className="text-xl font-semibold">No Configurations Yet</h3>
|
|
|
|
|
|
<p className="text-muted-foreground max-w-sm">
|
2025-11-14 21:53:46 -08:00
|
|
|
|
Add your own LLM provider configurations.
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</p>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<Button onClick={() => setIsAddingNew(true)} size="lg">
|
|
|
|
|
|
<Plus className="h-4 w-4 mr-2" />
|
|
|
|
|
|
Add First Configuration
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</CardContent>
|
|
|
|
|
|
</Card>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<div className="grid gap-4">
|
|
|
|
|
|
<AnimatePresence>
|
2025-12-09 19:39:25 +00:00
|
|
|
|
{llmConfigs?.map((config) => {
|
2025-07-27 10:05:37 -07:00
|
|
|
|
const providerInfo = getProviderInfo(config.provider);
|
|
|
|
|
|
return (
|
|
|
|
|
|
<motion.div
|
|
|
|
|
|
key={config.id}
|
|
|
|
|
|
initial={{ opacity: 0, y: 10 }}
|
|
|
|
|
|
animate={{ opacity: 1, y: 0 }}
|
|
|
|
|
|
exit={{ opacity: 0, y: -10 }}
|
|
|
|
|
|
transition={{ duration: 0.2 }}
|
|
|
|
|
|
>
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<Card className="group overflow-hidden hover:shadow-md transition-all duration-200">
|
|
|
|
|
|
<CardContent className="p-0">
|
|
|
|
|
|
<div className="flex">
|
|
|
|
|
|
{/* Left accent bar */}
|
|
|
|
|
|
<div className="w-1 bg-primary/50 group-hover:bg-primary transition-colors" />
|
|
|
|
|
|
|
|
|
|
|
|
<div className="flex-1 p-5">
|
|
|
|
|
|
<div className="flex items-start justify-between gap-4">
|
|
|
|
|
|
{/* Main content */}
|
|
|
|
|
|
<div className="flex items-start gap-4 flex-1 min-w-0">
|
|
|
|
|
|
<div className="flex h-11 w-11 shrink-0 items-center justify-center rounded-xl bg-primary/10 group-hover:bg-primary/15 transition-colors">
|
|
|
|
|
|
<Bot className="h-5 w-5 text-primary" />
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<div className="flex-1 min-w-0 space-y-3">
|
|
|
|
|
|
{/* Title row */}
|
|
|
|
|
|
<div className="flex items-center gap-2 flex-wrap">
|
|
|
|
|
|
<h4 className="text-base font-semibold tracking-tight truncate">
|
|
|
|
|
|
{config.name}
|
|
|
|
|
|
</h4>
|
|
|
|
|
|
<div className="flex items-center gap-1.5">
|
|
|
|
|
|
<Badge
|
|
|
|
|
|
variant="secondary"
|
|
|
|
|
|
className="text-[10px] font-medium px-1.5 py-0"
|
|
|
|
|
|
>
|
|
|
|
|
|
{config.provider}
|
|
|
|
|
|
</Badge>
|
|
|
|
|
|
{config.language && (
|
|
|
|
|
|
<Badge
|
|
|
|
|
|
variant="outline"
|
|
|
|
|
|
className="text-[10px] px-1.5 py-0 text-muted-foreground"
|
|
|
|
|
|
>
|
|
|
|
|
|
{config.language}
|
|
|
|
|
|
</Badge>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
{/* Model name */}
|
2025-10-12 13:13:42 +05:30
|
|
|
|
<div className="flex items-center gap-2">
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<code className="text-xs font-mono text-muted-foreground bg-muted/50 px-2 py-0.5 rounded">
|
|
|
|
|
|
{config.model_name}
|
|
|
|
|
|
</code>
|
2025-10-12 13:13:42 +05:30
|
|
|
|
</div>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
|
2025-12-13 22:43:38 -08:00
|
|
|
|
{/* Footer row */}
|
|
|
|
|
|
<div className="flex items-center gap-3 pt-2">
|
|
|
|
|
|
{config.created_at && (
|
|
|
|
|
|
<div className="flex items-center gap-1.5 text-xs text-muted-foreground">
|
|
|
|
|
|
<Clock className="h-3 w-3" />
|
|
|
|
|
|
<span>
|
|
|
|
|
|
{new Date(config.created_at).toLocaleDateString()}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</div>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
)}
|
2025-12-13 22:43:38 -08:00
|
|
|
|
<div className="flex items-center gap-1.5 text-xs">
|
|
|
|
|
|
<div className="h-1.5 w-1.5 rounded-full bg-emerald-500" />
|
|
|
|
|
|
<span className="text-emerald-600 dark:text-emerald-400 font-medium">
|
|
|
|
|
|
Active
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2025-12-13 22:43:38 -08:00
|
|
|
|
{/* Actions */}
|
|
|
|
|
|
<div className="flex items-center gap-1 shrink-0">
|
|
|
|
|
|
<Button
|
|
|
|
|
|
variant="ghost"
|
|
|
|
|
|
size="sm"
|
|
|
|
|
|
onClick={() => setEditingConfig(config)}
|
|
|
|
|
|
className="h-8 w-8 p-0 text-muted-foreground hover:text-foreground"
|
|
|
|
|
|
>
|
|
|
|
|
|
<Edit3 className="h-4 w-4" />
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
<Button
|
|
|
|
|
|
variant="ghost"
|
|
|
|
|
|
size="sm"
|
|
|
|
|
|
onClick={() => handleDeleteClick(config)}
|
|
|
|
|
|
className="h-8 w-8 p-0 text-muted-foreground hover:text-destructive hover:bg-destructive/10"
|
|
|
|
|
|
>
|
|
|
|
|
|
<Trash2 className="h-4 w-4" />
|
|
|
|
|
|
</Button>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</CardContent>
|
|
|
|
|
|
</Card>
|
|
|
|
|
|
</motion.div>
|
|
|
|
|
|
);
|
|
|
|
|
|
})}
|
|
|
|
|
|
</AnimatePresence>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
|
|
{/* Add/Edit Configuration Dialog */}
|
|
|
|
|
|
<Dialog
|
|
|
|
|
|
open={isAddingNew || !!editingConfig}
|
|
|
|
|
|
onOpenChange={(open) => {
|
|
|
|
|
|
if (!open) {
|
|
|
|
|
|
setIsAddingNew(false);
|
|
|
|
|
|
setEditingConfig(null);
|
|
|
|
|
|
setFormData({
|
|
|
|
|
|
name: "",
|
2025-12-09 19:39:25 +00:00
|
|
|
|
provider: "" as LLMConfig["provider"],
|
2025-07-27 10:05:37 -07:00
|
|
|
|
custom_provider: "",
|
|
|
|
|
|
model_name: "",
|
|
|
|
|
|
api_key: "",
|
|
|
|
|
|
api_base: "",
|
2025-10-12 13:13:42 +05:30
|
|
|
|
language: "",
|
2025-07-27 10:05:37 -07:00
|
|
|
|
litellm_params: {},
|
2025-10-10 00:50:29 -07:00
|
|
|
|
search_space_id: searchSpaceId,
|
2025-07-27 10:05:37 -07:00
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}}
|
|
|
|
|
|
>
|
|
|
|
|
|
<DialogContent className="max-w-2xl max-h-[90vh] overflow-y-auto">
|
|
|
|
|
|
<DialogHeader>
|
|
|
|
|
|
<DialogTitle className="flex items-center gap-2">
|
|
|
|
|
|
{editingConfig ? <Edit3 className="w-5 h-5" /> : <Plus className="w-5 h-5" />}
|
|
|
|
|
|
{editingConfig ? "Edit LLM Configuration" : "Add New LLM Configuration"}
|
|
|
|
|
|
</DialogTitle>
|
|
|
|
|
|
<DialogDescription>
|
|
|
|
|
|
{editingConfig
|
|
|
|
|
|
? "Update your language model provider configuration"
|
|
|
|
|
|
: "Configure a new language model provider for your AI assistant"}
|
|
|
|
|
|
</DialogDescription>
|
|
|
|
|
|
</DialogHeader>
|
|
|
|
|
|
|
|
|
|
|
|
<form onSubmit={handleSubmit} className="space-y-4">
|
|
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label htmlFor="name">Configuration Name *</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="name"
|
|
|
|
|
|
placeholder="e.g., My OpenAI GPT-4"
|
|
|
|
|
|
value={formData.name}
|
|
|
|
|
|
onChange={(e) => handleInputChange("name", e.target.value)}
|
|
|
|
|
|
required
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label htmlFor="provider">Provider *</Label>
|
2025-10-13 20:11:11 -07:00
|
|
|
|
<Select value={formData.provider} onValueChange={handleProviderChange}>
|
2025-10-10 00:50:29 -07:00
|
|
|
|
<SelectTrigger>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
<SelectValue placeholder="Select a provider">
|
|
|
|
|
|
{formData.provider && (
|
2025-10-10 00:50:29 -07:00
|
|
|
|
<span className="font-medium">
|
|
|
|
|
|
{LLM_PROVIDERS.find((p) => p.value === formData.provider)?.label}
|
|
|
|
|
|
</span>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
)}
|
|
|
|
|
|
</SelectValue>
|
|
|
|
|
|
</SelectTrigger>
|
|
|
|
|
|
<SelectContent>
|
|
|
|
|
|
{LLM_PROVIDERS.map((provider) => (
|
|
|
|
|
|
<SelectItem key={provider.value} value={provider.value}>
|
|
|
|
|
|
<div className="space-y-1 py-1">
|
|
|
|
|
|
<div className="font-medium">{provider.label}</div>
|
|
|
|
|
|
<div className="text-xs text-muted-foreground">
|
|
|
|
|
|
{provider.description}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</SelectItem>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</SelectContent>
|
|
|
|
|
|
</Select>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
{formData.provider === "CUSTOM" && (
|
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label htmlFor="custom_provider">Custom Provider Name *</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="custom_provider"
|
|
|
|
|
|
placeholder="e.g., my-custom-provider"
|
2025-12-09 19:39:25 +00:00
|
|
|
|
value={formData.custom_provider ?? ""}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
onChange={(e) => handleInputChange("custom_provider", e.target.value)}
|
|
|
|
|
|
required
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label htmlFor="model_name">Model Name *</Label>
|
2025-11-13 12:46:48 -08:00
|
|
|
|
<Popover open={modelComboboxOpen} onOpenChange={setModelComboboxOpen}>
|
|
|
|
|
|
<PopoverTrigger asChild>
|
|
|
|
|
|
<Button
|
|
|
|
|
|
variant="outline"
|
|
|
|
|
|
aria-expanded={modelComboboxOpen}
|
|
|
|
|
|
className="w-full justify-between font-normal"
|
2025-11-13 02:41:30 -08:00
|
|
|
|
>
|
2025-11-13 12:46:48 -08:00
|
|
|
|
<span className={cn(!formData.model_name && "text-muted-foreground")}>
|
|
|
|
|
|
{formData.model_name || "Select or type model name..."}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</PopoverTrigger>
|
|
|
|
|
|
<PopoverContent className="w-full p-0" align="start" side="bottom">
|
|
|
|
|
|
<Command shouldFilter={false}>
|
|
|
|
|
|
<CommandInput
|
|
|
|
|
|
placeholder={selectedProvider?.example || "Type model name..."}
|
|
|
|
|
|
value={formData.model_name}
|
|
|
|
|
|
onValueChange={(value) => handleInputChange("model_name", value)}
|
|
|
|
|
|
/>
|
|
|
|
|
|
<CommandList>
|
|
|
|
|
|
<CommandEmpty>
|
|
|
|
|
|
<div className="py-2 text-center text-sm text-muted-foreground">
|
|
|
|
|
|
{formData.model_name
|
|
|
|
|
|
? `Using custom model: "${formData.model_name}"`
|
|
|
|
|
|
: "Type your model name above"}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</CommandEmpty>
|
|
|
|
|
|
{availableModels.length > 0 && (
|
|
|
|
|
|
<CommandGroup heading="Suggested Models">
|
|
|
|
|
|
{availableModels
|
|
|
|
|
|
.filter(
|
|
|
|
|
|
(model) =>
|
|
|
|
|
|
!formData.model_name ||
|
|
|
|
|
|
model.value
|
|
|
|
|
|
.toLowerCase()
|
|
|
|
|
|
.includes(formData.model_name.toLowerCase()) ||
|
|
|
|
|
|
model.label
|
|
|
|
|
|
.toLowerCase()
|
|
|
|
|
|
.includes(formData.model_name.toLowerCase())
|
|
|
|
|
|
)
|
|
|
|
|
|
.map((model) => (
|
|
|
|
|
|
<CommandItem
|
|
|
|
|
|
key={model.value}
|
|
|
|
|
|
value={model.value}
|
|
|
|
|
|
onSelect={(currentValue) => {
|
|
|
|
|
|
handleInputChange("model_name", currentValue);
|
|
|
|
|
|
setModelComboboxOpen(false);
|
|
|
|
|
|
}}
|
|
|
|
|
|
className="flex flex-col items-start py-3"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div className="flex w-full items-center">
|
|
|
|
|
|
<Check
|
|
|
|
|
|
className={cn(
|
|
|
|
|
|
"mr-2 h-4 w-4 shrink-0",
|
|
|
|
|
|
formData.model_name === model.value
|
|
|
|
|
|
? "opacity-100"
|
|
|
|
|
|
: "opacity-0"
|
|
|
|
|
|
)}
|
|
|
|
|
|
/>
|
|
|
|
|
|
<div className="flex-1">
|
|
|
|
|
|
<div className="font-medium">{model.label}</div>
|
|
|
|
|
|
{model.contextWindow && (
|
|
|
|
|
|
<div className="text-xs text-muted-foreground">
|
|
|
|
|
|
Context: {model.contextWindow}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</CommandItem>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</CommandGroup>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</CommandList>
|
|
|
|
|
|
</Command>
|
|
|
|
|
|
</PopoverContent>
|
|
|
|
|
|
</Popover>
|
|
|
|
|
|
<p className="text-xs text-muted-foreground">
|
|
|
|
|
|
{availableModels.length > 0
|
|
|
|
|
|
? `Type freely or select from ${availableModels.length} model suggestions`
|
|
|
|
|
|
: selectedProvider?.example
|
|
|
|
|
|
? `Examples: ${selectedProvider.example}`
|
|
|
|
|
|
: "Type your model name freely"}
|
|
|
|
|
|
</p>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2025-10-12 13:13:42 +05:30
|
|
|
|
<div className="space-y-2">
|
2025-10-13 20:11:11 -07:00
|
|
|
|
<Label htmlFor="language">Language (Optional)</Label>
|
|
|
|
|
|
<Select
|
|
|
|
|
|
value={formData.language || "English"}
|
|
|
|
|
|
onValueChange={(value) => handleInputChange("language", value)}
|
|
|
|
|
|
>
|
|
|
|
|
|
<SelectTrigger>
|
|
|
|
|
|
<SelectValue placeholder="Select language" />
|
|
|
|
|
|
</SelectTrigger>
|
|
|
|
|
|
<SelectContent>
|
|
|
|
|
|
{LANGUAGES.map((language) => (
|
|
|
|
|
|
<SelectItem key={language.value} value={language.value}>
|
|
|
|
|
|
{language.label}
|
|
|
|
|
|
</SelectItem>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</SelectContent>
|
|
|
|
|
|
</Select>
|
|
|
|
|
|
</div>
|
2025-10-12 13:13:42 +05:30
|
|
|
|
|
2025-07-27 10:05:37 -07:00
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label htmlFor="api_key">API Key *</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="api_key"
|
|
|
|
|
|
type="password"
|
2025-12-09 01:50:08 -08:00
|
|
|
|
placeholder={
|
|
|
|
|
|
formData.provider === "OLLAMA" ? "Any value (e.g., ollama)" : "Your API key"
|
|
|
|
|
|
}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
value={formData.api_key}
|
|
|
|
|
|
onChange={(e) => handleInputChange("api_key", e.target.value)}
|
|
|
|
|
|
required
|
|
|
|
|
|
/>
|
2025-12-09 01:49:27 -08:00
|
|
|
|
{formData.provider === "OLLAMA" && (
|
|
|
|
|
|
<p className="text-xs text-muted-foreground">
|
|
|
|
|
|
💡 Ollama doesn't require authentication — enter any value (e.g., "ollama")
|
|
|
|
|
|
</p>
|
|
|
|
|
|
)}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div className="space-y-2">
|
2025-10-12 19:10:46 +08:00
|
|
|
|
<Label htmlFor="api_base">
|
2025-10-13 20:11:11 -07:00
|
|
|
|
API Base URL
|
2025-10-12 19:10:46 +08:00
|
|
|
|
{selectedProvider?.apiBase && (
|
|
|
|
|
|
<span className="text-xs font-normal text-muted-foreground ml-2">
|
|
|
|
|
|
(Auto-filled for {selectedProvider.label})
|
|
|
|
|
|
</span>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</Label>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
<Input
|
|
|
|
|
|
id="api_base"
|
2025-10-12 19:10:46 +08:00
|
|
|
|
placeholder={selectedProvider?.apiBase || "e.g., https://api.openai.com/v1"}
|
2025-12-09 19:39:25 +00:00
|
|
|
|
value={formData.api_base ?? ""}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
onChange={(e) => handleInputChange("api_base", e.target.value)}
|
|
|
|
|
|
/>
|
2025-10-12 19:10:46 +08:00
|
|
|
|
{selectedProvider?.apiBase && formData.api_base === selectedProvider.apiBase && (
|
|
|
|
|
|
<p className="text-xs text-green-600 flex items-center gap-1">
|
|
|
|
|
|
<CheckCircle className="h-3 w-3" />
|
|
|
|
|
|
Using recommended API endpoint for {selectedProvider.label}
|
|
|
|
|
|
</p>
|
|
|
|
|
|
)}
|
|
|
|
|
|
{selectedProvider?.apiBase && !formData.api_base && (
|
|
|
|
|
|
<p className="text-xs text-amber-600 flex items-center gap-1">
|
|
|
|
|
|
<AlertCircle className="h-3 w-3" />
|
2025-10-13 20:11:11 -07:00
|
|
|
|
⚠️ API Base URL is required for {selectedProvider.label}. Click to auto-fill:
|
|
|
|
|
|
<button
|
2025-10-12 19:10:46 +08:00
|
|
|
|
type="button"
|
|
|
|
|
|
className="underline font-medium"
|
|
|
|
|
|
onClick={() => handleInputChange("api_base", selectedProvider.apiBase || "")}
|
|
|
|
|
|
>
|
|
|
|
|
|
{selectedProvider.apiBase}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</p>
|
|
|
|
|
|
)}
|
2025-12-09 01:49:27 -08:00
|
|
|
|
{/* Ollama-specific help */}
|
|
|
|
|
|
{formData.provider === "OLLAMA" && (
|
|
|
|
|
|
<div className="mt-2 p-3 bg-muted/50 rounded-lg border border-muted">
|
|
|
|
|
|
<p className="text-xs font-medium mb-2">💡 Ollama API Base URL Examples:</p>
|
|
|
|
|
|
<div className="space-y-1.5">
|
|
|
|
|
|
<button
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
className="flex items-center gap-2 text-xs text-muted-foreground hover:text-foreground transition-colors"
|
|
|
|
|
|
onClick={() => handleInputChange("api_base", "http://localhost:11434")}
|
|
|
|
|
|
>
|
2025-12-09 01:50:08 -08:00
|
|
|
|
<code className="px-1.5 py-0.5 bg-background rounded border">
|
|
|
|
|
|
http://localhost:11434
|
|
|
|
|
|
</code>
|
2025-12-09 01:49:27 -08:00
|
|
|
|
<span>— Standard local installation</span>
|
|
|
|
|
|
</button>
|
|
|
|
|
|
<button
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
className="flex items-center gap-2 text-xs text-muted-foreground hover:text-foreground transition-colors"
|
2025-12-09 01:50:08 -08:00
|
|
|
|
onClick={() =>
|
|
|
|
|
|
handleInputChange("api_base", "http://host.docker.internal:11434")
|
|
|
|
|
|
}
|
2025-12-09 01:49:27 -08:00
|
|
|
|
>
|
2025-12-09 01:50:08 -08:00
|
|
|
|
<code className="px-1.5 py-0.5 bg-background rounded border">
|
|
|
|
|
|
http://host.docker.internal:11434
|
|
|
|
|
|
</code>
|
2025-12-09 01:49:27 -08:00
|
|
|
|
<span>— If using SurfSense Docker image</span>
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2025-10-07 04:59:01 +05:30
|
|
|
|
{/* Optional Inference Parameters */}
|
|
|
|
|
|
<div className="pt-4">
|
|
|
|
|
|
<InferenceParamsEditor
|
|
|
|
|
|
params={formData.litellm_params || {}}
|
|
|
|
|
|
setParams={(newParams) =>
|
2025-10-10 00:50:29 -07:00
|
|
|
|
setFormData((prev) => ({ ...prev, litellm_params: newParams }))
|
2025-10-07 04:59:01 +05:30
|
|
|
|
}
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2025-07-27 10:05:37 -07:00
|
|
|
|
<div className="flex gap-2 pt-4">
|
|
|
|
|
|
<Button type="submit" disabled={isSubmitting}>
|
|
|
|
|
|
{isSubmitting
|
|
|
|
|
|
? editingConfig
|
|
|
|
|
|
? "Updating..."
|
|
|
|
|
|
: "Adding..."
|
|
|
|
|
|
: editingConfig
|
|
|
|
|
|
? "Update Configuration"
|
|
|
|
|
|
: "Add Configuration"}
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
<Button
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
variant="outline"
|
|
|
|
|
|
onClick={() => {
|
|
|
|
|
|
setIsAddingNew(false);
|
|
|
|
|
|
setEditingConfig(null);
|
|
|
|
|
|
setFormData({
|
|
|
|
|
|
name: "",
|
2025-12-09 19:39:25 +00:00
|
|
|
|
provider: "" as LLMConfig["provider"],
|
2025-07-27 10:05:37 -07:00
|
|
|
|
custom_provider: "",
|
|
|
|
|
|
model_name: "",
|
|
|
|
|
|
api_key: "",
|
|
|
|
|
|
api_base: "",
|
2025-10-12 13:13:42 +05:30
|
|
|
|
language: "",
|
2025-07-27 10:05:37 -07:00
|
|
|
|
litellm_params: {},
|
2025-10-10 00:50:29 -07:00
|
|
|
|
search_space_id: searchSpaceId,
|
2025-07-27 10:05:37 -07:00
|
|
|
|
});
|
|
|
|
|
|
}}
|
|
|
|
|
|
disabled={isSubmitting}
|
|
|
|
|
|
>
|
|
|
|
|
|
Cancel
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</form>
|
|
|
|
|
|
</DialogContent>
|
|
|
|
|
|
</Dialog>
|
2025-12-13 22:43:38 -08:00
|
|
|
|
|
|
|
|
|
|
{/* Delete Confirmation Dialog */}
|
|
|
|
|
|
<AlertDialog
|
|
|
|
|
|
open={!!configToDelete}
|
|
|
|
|
|
onOpenChange={(open) => !open && setConfigToDelete(null)}
|
|
|
|
|
|
>
|
|
|
|
|
|
<AlertDialogContent>
|
|
|
|
|
|
<AlertDialogHeader>
|
|
|
|
|
|
<AlertDialogTitle className="flex items-center gap-2">
|
|
|
|
|
|
<Trash2 className="h-5 w-5 text-destructive" />
|
|
|
|
|
|
Delete Configuration
|
|
|
|
|
|
</AlertDialogTitle>
|
|
|
|
|
|
<AlertDialogDescription>
|
|
|
|
|
|
Are you sure you want to delete{" "}
|
|
|
|
|
|
<span className="font-semibold text-foreground">{configToDelete?.name}</span>? This
|
|
|
|
|
|
action cannot be undone and will permanently remove this model configuration.
|
|
|
|
|
|
</AlertDialogDescription>
|
|
|
|
|
|
</AlertDialogHeader>
|
|
|
|
|
|
<AlertDialogFooter>
|
|
|
|
|
|
<AlertDialogCancel disabled={isDeleting}>Cancel</AlertDialogCancel>
|
|
|
|
|
|
<AlertDialogAction
|
|
|
|
|
|
onClick={handleConfirmDelete}
|
|
|
|
|
|
disabled={isDeleting}
|
|
|
|
|
|
className="bg-destructive text-destructive-foreground hover:bg-destructive/90"
|
|
|
|
|
|
>
|
|
|
|
|
|
{isDeleting ? (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<Loader2 className="mr-2 h-4 w-4 animate-spin" />
|
|
|
|
|
|
Deleting...
|
|
|
|
|
|
</>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<Trash2 className="mr-2 h-4 w-4" />
|
|
|
|
|
|
Delete
|
|
|
|
|
|
</>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</AlertDialogAction>
|
|
|
|
|
|
</AlertDialogFooter>
|
|
|
|
|
|
</AlertDialogContent>
|
|
|
|
|
|
</AlertDialog>
|
2025-07-27 10:05:37 -07:00
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|