SurfSense/surfsense_web/components/onboard/add-provider-step.tsx

302 lines
9.3 KiB
TypeScript
Raw Normal View History

2025-06-09 15:50:15 -07:00
"use client";
2025-07-27 10:41:15 -07:00
import { AlertCircle, Bot, Plus, Trash2 } from "lucide-react";
2025-09-30 21:53:10 -07:00
import { motion } from "motion/react";
import { useTranslations } from "next-intl";
2025-07-27 10:41:15 -07:00
import { useState } from "react";
import { toast } from "sonner";
import { Alert, AlertDescription } from "@/components/ui/alert";
import { Badge } from "@/components/ui/badge";
2025-07-27 10:05:37 -07:00
import { Button } from "@/components/ui/button";
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
import { Input } from "@/components/ui/input";
import { Label } from "@/components/ui/label";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select";
import { LANGUAGES } from "@/contracts/enums/languages";
2025-10-15 14:38:26 -07:00
import { LLM_PROVIDERS } from "@/contracts/enums/llm-providers";
2025-07-27 10:41:15 -07:00
import { type CreateLLMConfig, useLLMConfigs } from "@/hooks/use-llm-configs";
2025-06-09 15:50:15 -07:00
2025-10-07 04:59:01 +05:30
import InferenceParamsEditor from "../inference-params-editor";
2025-06-09 17:04:06 -07:00
interface AddProviderStepProps {
searchSpaceId: number;
2025-07-27 10:05:37 -07:00
onConfigCreated?: () => void;
onConfigDeleted?: () => void;
2025-06-09 17:04:06 -07:00
}
export function AddProviderStep({
searchSpaceId,
onConfigCreated,
onConfigDeleted,
}: AddProviderStepProps) {
const t = useTranslations("onboard");
const { llmConfigs, createLLMConfig, deleteLLMConfig } = useLLMConfigs(searchSpaceId);
2025-07-27 10:05:37 -07:00
const [isAddingNew, setIsAddingNew] = useState(false);
const [formData, setFormData] = useState<CreateLLMConfig>({
name: "",
provider: "",
custom_provider: "",
model_name: "",
api_key: "",
api_base: "",
language: "English",
2025-07-27 10:05:37 -07:00
litellm_params: {},
search_space_id: searchSpaceId,
2025-07-27 10:05:37 -07:00
});
const [isSubmitting, setIsSubmitting] = useState(false);
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
const handleInputChange = (field: keyof CreateLLMConfig, value: string) => {
setFormData((prev) => ({ ...prev, [field]: value }));
};
2025-06-09 15:50:15 -07:00
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-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
setIsSubmitting(true);
const result = await createLLMConfig(formData);
setIsSubmitting(false);
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
if (result) {
setFormData({
name: "",
provider: "",
custom_provider: "",
model_name: "",
api_key: "",
api_base: "",
language: "English",
2025-07-27 10:05:37 -07:00
litellm_params: {},
search_space_id: searchSpaceId,
2025-07-27 10:05:37 -07:00
});
setIsAddingNew(false);
// Notify parent component that a config was created
onConfigCreated?.();
}
};
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
const selectedProvider = LLM_PROVIDERS.find((p) => p.value === formData.provider);
2025-06-09 15:50:15 -07:00
2025-10-07 04:59:01 +05:30
const handleParamsChange = (newParams: Record<string, number | string>) => {
setFormData((prev) => ({ ...prev, litellm_params: newParams }));
};
2025-07-27 10:05:37 -07:00
return (
<div className="space-y-6">
{/* Info Alert */}
<Alert>
<AlertCircle className="h-4 w-4" />
<AlertDescription>{t("add_provider_instruction")}</AlertDescription>
2025-07-27 10:05:37 -07:00
</Alert>
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
{/* Existing Configurations */}
{llmConfigs.length > 0 && (
<div className="space-y-4">
<h3 className="text-lg font-semibold">{t("your_llm_configs")}</h3>
2025-07-27 10:05:37 -07:00
<div className="grid gap-4">
{llmConfigs.map((config) => (
<motion.div
key={config.id}
initial={{ opacity: 0, y: 10 }}
animate={{ opacity: 1, y: 0 }}
exit={{ opacity: 0, y: -10 }}
>
<Card className="border-l-4 border-l-primary">
<CardContent className="pt-4">
<div className="flex items-center justify-between">
<div className="flex-1">
<div className="flex items-center gap-2 mb-2">
<Bot className="w-4 h-4" />
<h4 className="font-medium">{config.name}</h4>
<Badge variant="secondary">{config.provider}</Badge>
</div>
<p className="text-sm text-muted-foreground">
{t("model")}: {config.model_name}
{config.language && `${t("language")}: ${config.language}`}
{config.api_base && `${t("base")}: ${config.api_base}`}
2025-07-27 10:05:37 -07:00
</p>
</div>
<Button
variant="ghost"
size="sm"
onClick={async () => {
const success = await deleteLLMConfig(config.id);
if (success) {
onConfigDeleted?.();
}
}}
className="text-destructive hover:text-destructive"
>
<Trash2 className="w-4 h-4" />
</Button>
</div>
</CardContent>
</Card>
</motion.div>
))}
</div>
</div>
)}
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
{/* Add New Provider */}
{!isAddingNew ? (
<Card className="border-dashed border-2 hover:border-primary/50 transition-colors">
<CardContent className="flex flex-col items-center justify-center py-12">
<Plus className="w-12 h-12 text-muted-foreground mb-4" />
<h3 className="text-lg font-semibold mb-2">{t("add_provider_title")}</h3>
<p className="text-muted-foreground text-center mb-4">{t("add_provider_subtitle")}</p>
2025-07-27 10:05:37 -07:00
<Button onClick={() => setIsAddingNew(true)}>
<Plus className="w-4 h-4 mr-2" />
{t("add_provider_button")}
2025-07-27 10:05:37 -07:00
</Button>
</CardContent>
</Card>
) : (
<Card>
<CardHeader>
<CardTitle>{t("add_new_llm_provider")}</CardTitle>
<CardDescription>{t("configure_new_provider")}</CardDescription>
2025-07-27 10:05:37 -07:00
</CardHeader>
<CardContent>
<form onSubmit={handleSubmit} className="space-y-4">
<div className="grid grid-cols-1 md:grid-cols-3 gap-4">
2025-07-27 10:05:37 -07:00
<div className="space-y-2">
<Label htmlFor="name">{t("config_name_required")}</Label>
2025-07-27 10:05:37 -07:00
<Input
id="name"
placeholder={t("config_name_placeholder")}
2025-07-27 10:05:37 -07:00
value={formData.name}
onChange={(e) => handleInputChange("name", e.target.value)}
required
/>
</div>
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
<div className="space-y-2">
<Label htmlFor="provider">{t("provider_required")}</Label>
2025-07-27 10:05:37 -07:00
<Select
value={formData.provider}
onValueChange={(value) => handleInputChange("provider", value)}
>
<SelectTrigger>
<SelectValue placeholder={t("provider_placeholder")} />
2025-07-27 10:05:37 -07:00
</SelectTrigger>
<SelectContent>
{LLM_PROVIDERS.map((provider) => (
<SelectItem key={provider.value} value={provider.value}>
{provider.label}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
{/* language */}
<div className="space-y-2">
<Label htmlFor="language">{t("language_optional")}</Label>
<Select
value={formData.language || "English"}
onValueChange={(value) => handleInputChange("language", value)}
>
<SelectTrigger>
<SelectValue placeholder={t("language_placeholder")} />
</SelectTrigger>
<SelectContent>
{LANGUAGES.map((language) => (
<SelectItem key={language.value} value={language.value}>
{language.label}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
2025-07-27 10:05:37 -07:00
</div>
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
{formData.provider === "CUSTOM" && (
<div className="space-y-2">
<Label htmlFor="custom_provider">{t("custom_provider_name")}</Label>
2025-07-27 10:05:37 -07:00
<Input
id="custom_provider"
placeholder={t("custom_provider_placeholder")}
2025-07-27 10:05:37 -07:00
value={formData.custom_provider}
onChange={(e) => handleInputChange("custom_provider", e.target.value)}
required
/>
</div>
)}
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
<div className="space-y-2">
<Label htmlFor="model_name">{t("model_name_required")}</Label>
2025-07-27 10:05:37 -07:00
<Input
id="model_name"
placeholder={selectedProvider?.example || t("model_name_placeholder")}
2025-07-27 10:05:37 -07:00
value={formData.model_name}
onChange={(e) => handleInputChange("model_name", e.target.value)}
required
/>
{selectedProvider && (
<p className="text-xs text-muted-foreground">
{t("examples")}: {selectedProvider.example}
2025-07-27 10:05:37 -07:00
</p>
)}
</div>
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
<div className="space-y-2">
<Label htmlFor="api_key">{t("api_key_required")}</Label>
2025-07-27 10:05:37 -07:00
<Input
id="api_key"
type="password"
placeholder={t("api_key_placeholder")}
2025-07-27 10:05:37 -07:00
value={formData.api_key}
onChange={(e) => handleInputChange("api_key", e.target.value)}
required
/>
</div>
2025-06-09 15:50:15 -07:00
2025-07-27 10:05:37 -07:00
<div className="space-y-2">
<Label htmlFor="api_base">{t("api_base_optional")}</Label>
2025-07-27 10:05:37 -07:00
<Input
id="api_base"
placeholder={t("api_base_placeholder")}
2025-07-27 10:05:37 -07:00
value={formData.api_base}
onChange={(e) => handleInputChange("api_base", e.target.value)}
/>
</div>
2025-06-09 15:50:15 -07:00
2025-10-07 04:59:01 +05:30
{/* Optional Inference Parameters */}
<div className="pt-4">
<InferenceParamsEditor
params={formData.litellm_params || {}}
setParams={handleParamsChange}
/>
</div>
2025-07-27 10:05:37 -07:00
<div className="flex gap-2 pt-4">
<Button type="submit" disabled={isSubmitting}>
{isSubmitting ? t("adding") : t("add_provider")}
2025-07-27 10:05:37 -07:00
</Button>
<Button
type="button"
variant="outline"
onClick={() => setIsAddingNew(false)}
disabled={isSubmitting}
>
{t("cancel")}
2025-07-27 10:05:37 -07:00
</Button>
</div>
</form>
</CardContent>
</Card>
)}
</div>
);
}