- Updated docker-compose.yml to allow disabling embedded Ollama via environment variable. - Refactored Dockerfile and Dockerfile.dev for improved dependency management and installation process. - Enhanced FastAPI migration scripts to handle orphaned Alembic revisions and added new database migration logic. - Improved error handling in background tasks and Codex authentication endpoints. - Added support for font file uploads with better validation and extraction of font names. - Introduced new image search functionality with support for Pexels and Pixabay APIs.
243 lines
9.5 KiB
TypeScript
243 lines
9.5 KiB
TypeScript
"use client";
|
|
import { useEffect, useState } from "react";
|
|
import { Check, ChevronsUpDown, Loader2 } from "lucide-react";
|
|
import { Button } from "./ui/button";
|
|
import {
|
|
Command,
|
|
CommandEmpty,
|
|
CommandGroup,
|
|
CommandInput,
|
|
CommandItem,
|
|
CommandList,
|
|
} from "./ui/command";
|
|
import { Popover, PopoverContent, PopoverTrigger } from "./ui/popover";
|
|
import { cn } from "@/lib/utils";
|
|
import { toast } from "sonner";
|
|
import { Switch } from "./ui/switch";
|
|
import { getApiUrl } from "@/utils/api";
|
|
|
|
interface GoogleConfigProps {
|
|
googleApiKey: string;
|
|
googleModel: string;
|
|
webGrounding?: boolean;
|
|
onInputChange: (value: string | boolean, field: string) => void;
|
|
}
|
|
|
|
export default function GoogleConfig({
|
|
googleApiKey,
|
|
googleModel,
|
|
webGrounding,
|
|
onInputChange
|
|
}: GoogleConfigProps) {
|
|
const [openModelSelect, setOpenModelSelect] = useState(false);
|
|
const [availableModels, setAvailableModels] = useState<string[]>([]);
|
|
const [modelsLoading, setModelsLoading] = useState(false);
|
|
const [modelsChecked, setModelsChecked] = useState(false);
|
|
const [apiKey, setApiKey] = useState(googleApiKey);
|
|
|
|
useEffect(() => {
|
|
setAvailableModels([]);
|
|
setModelsChecked(false);
|
|
onInputChange("", "google_model");
|
|
}, [apiKey]);
|
|
|
|
const onApiKeyChange = (value: string) => {
|
|
setApiKey(value);
|
|
onInputChange(value, "google_api_key");
|
|
};
|
|
|
|
const fetchAvailableModels = async () => {
|
|
if (!googleApiKey) return;
|
|
|
|
setModelsLoading(true);
|
|
try {
|
|
const response = await fetch(getApiUrl("/api/v1/ppt/google/models/available"), {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
body: JSON.stringify({
|
|
api_key: googleApiKey
|
|
}),
|
|
});
|
|
|
|
if (response.ok) {
|
|
const data = await response.json();
|
|
setAvailableModels(data);
|
|
setModelsChecked(true);
|
|
onInputChange("models/gemini-2.5-flash", "google_model");
|
|
} else {
|
|
console.error('Failed to fetch models');
|
|
setAvailableModels([]);
|
|
setModelsChecked(true);
|
|
}
|
|
} catch (error) {
|
|
console.error('Error fetching models:', error);
|
|
toast.error('Error fetching models');
|
|
setAvailableModels([]);
|
|
setModelsChecked(true);
|
|
} finally {
|
|
setModelsLoading(false);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className="space-y-6 ">
|
|
{/* API Key Input */}
|
|
<div className="mb-4 flex items-center justify-between bg-white p-10">
|
|
<div className="">
|
|
<h3 className="text-xl font-normal text-[#191919]">Google API key</h3>
|
|
<p className="mt-2 text-sm max-w-[205px] text-gray-500">
|
|
Your API key will be stored locally and never shared
|
|
</p>
|
|
</div>
|
|
<div className="flex items-center gap-4">
|
|
<div className="relative w-[275px] ">
|
|
<div className="flex flex-col justify-start gap-2">
|
|
<label className="block text-sm font-medium text-gray-700 mb-2">
|
|
Google API Key
|
|
</label>
|
|
<input
|
|
type="text"
|
|
value={googleApiKey}
|
|
onChange={(e) => onApiKeyChange(e.target.value)}
|
|
className="w-full px-2 py-3 outline-none border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500/20 focus:border-blue-500 transition-colors"
|
|
placeholder="Enter your API key"
|
|
/>
|
|
</div>
|
|
|
|
{/* Check for available models button - show when no models checked or no models found */}
|
|
{(!modelsChecked || (modelsChecked && availableModels.length === 0)) && (
|
|
<button
|
|
onClick={fetchAvailableModels}
|
|
disabled={modelsLoading || !googleApiKey}
|
|
className={` mt-7 py-2.5 bg-[#F7F6F9] px-3.5 rounded-[48px] text-xs font-semibold text-[#101323] transition-all duration-200 border ${modelsLoading || !googleApiKey
|
|
? " border-gray-300 cursor-not-allowed text-gray-500"
|
|
: " border-[#EDEEEF] text-blue-600 hover:bg-[#E8F0FF]/90 focus:ring-2 focus:ring-blue-500/20"
|
|
}`}
|
|
>
|
|
{modelsLoading ? (
|
|
<span className="flex items-center justify-center gap-2">
|
|
<Loader2 className="w-4 h-4 animate-spin" />
|
|
Checking for models...
|
|
</span>
|
|
) : (
|
|
"Check for available models"
|
|
)}
|
|
</button>
|
|
)}
|
|
</div>
|
|
<div className="w-[295px]">
|
|
{/* Show message if no models found */}
|
|
{modelsChecked && availableModels.length === 0 && (
|
|
<div className="mb-4 p-3 bg-yellow-50 border border-yellow-200 rounded-lg">
|
|
<p className="text-sm text-yellow-800">
|
|
No models found. Please make sure your API key is valid and has access to Google models.
|
|
</p>
|
|
</div>
|
|
)}
|
|
|
|
{/* Model Selection - only show if models are available */}
|
|
{modelsChecked && availableModels.length > 0 ? (
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-3">
|
|
Select Google Model
|
|
</label>
|
|
<div className="w-full">
|
|
<Popover
|
|
open={openModelSelect}
|
|
onOpenChange={setOpenModelSelect}
|
|
>
|
|
<PopoverTrigger asChild>
|
|
<Button
|
|
variant="outline"
|
|
role="combobox"
|
|
aria-expanded={openModelSelect}
|
|
className="w-full h-12 px-4 py-4 outline-none border border-gray-300 rounded-lg focus:ring-2 focus:ring-blue-500/20 focus:border-blue-500 transition-colors hover:border-gray-400 justify-between"
|
|
>
|
|
<div className="flex gap-3 items-center">
|
|
<span className="text-sm font-medium text-gray-900">
|
|
{googleModel
|
|
? availableModels.find(model => model === googleModel) || googleModel
|
|
: "Select a model"}
|
|
</span>
|
|
</div>
|
|
<ChevronsUpDown className="w-4 h-4 text-gray-500" />
|
|
</Button>
|
|
</PopoverTrigger>
|
|
<PopoverContent
|
|
className="p-0"
|
|
align="start"
|
|
style={{ width: "var(--radix-popover-trigger-width)" }}
|
|
>
|
|
<Command>
|
|
<CommandInput placeholder="Search models..." />
|
|
<CommandList>
|
|
<CommandEmpty>No model found.</CommandEmpty>
|
|
<CommandGroup>
|
|
{availableModels.map((model, index) => (
|
|
<CommandItem
|
|
key={index}
|
|
value={model}
|
|
onSelect={(value) => {
|
|
onInputChange(value, "google_model");
|
|
setOpenModelSelect(false);
|
|
}}
|
|
>
|
|
<Check
|
|
className={cn(
|
|
"mr-2 h-4 w-4",
|
|
googleModel === model
|
|
? "opacity-100"
|
|
: "opacity-0"
|
|
)}
|
|
/>
|
|
<div className="flex gap-3 items-center">
|
|
<div className="flex flex-col space-y-1 flex-1">
|
|
<div className="flex items-center justify-between gap-2">
|
|
<span className="text-sm font-medium text-gray-900">
|
|
{model}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</CommandItem>
|
|
))}
|
|
</CommandGroup>
|
|
</CommandList>
|
|
</Command>
|
|
</PopoverContent>
|
|
</Popover>
|
|
</div>
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Web Grounding Toggle - show at the end, below models dropdown */}
|
|
<div className="bg-white flex justify-between items-center p-10 rounded-[12px]">
|
|
<div>
|
|
<h4 className="text-xl font-normal text-[#191919]">Model Controls</h4>
|
|
<p className="mt-2 text-sm max-w-[205px] text-gray-500">
|
|
Configure web access and advanced AI features.
|
|
</p>
|
|
</div>
|
|
<div className="flex items-center gap-4">
|
|
<div className="w-[275px]">
|
|
<div className="flex items-center mb-4 gap-2.5 ">
|
|
<Switch
|
|
checked={!!webGrounding}
|
|
onCheckedChange={(checked) => onInputChange(checked, "web_grounding")}
|
|
/>
|
|
<label className="text-sm font-medium text-gray-700">
|
|
Enable Web Grounding
|
|
</label>
|
|
</div>
|
|
</div>
|
|
<div className="w-[295px]"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|