feat: add AI settings routes, storage, UI card, and migration

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Gitead
2026-05-06 08:58:58 -04:00
parent 8c162d7040
commit 4989201c62
5 changed files with 295 additions and 0 deletions

View File

@@ -0,0 +1,131 @@
import { useState, useEffect } from "react";
import { useQuery, useMutation } from "@tanstack/react-query";
import { Card, CardContent } from "@/components/ui/card";
import { Eye, EyeOff, CheckCircle } from "lucide-react";
import { useToast } from "@/hooks/use-toast";
import { apiRequest, queryClient } from "@/lib/queryClient";
type AiSettings = {
id?: number;
apiKey: string;
};
export function AiSettingsCard() {
const { toast } = useToast();
const [apiKey, setApiKey] = useState("");
const [showKey, setShowKey] = useState(false);
const { data: settings, isLoading } = useQuery<AiSettings | null>({
queryKey: ["/api/ai/settings"],
queryFn: async () => {
const res = await apiRequest("GET", "/api/ai/settings");
if (!res.ok) return null;
return res.json();
},
});
useEffect(() => {
if (settings) {
setApiKey(settings.apiKey ?? "");
}
}, [settings]);
const saveMutation = useMutation({
mutationFn: async (data: { apiKey: string }) => {
const res = await apiRequest("PUT", "/api/ai/settings", data);
if (!res.ok) {
const err = await res.json().catch(() => null);
throw new Error(err?.message || "Failed to save AI settings");
}
return res.json();
},
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["/api/ai/settings"] });
toast({ title: "AI Settings Saved", description: "Your Google AI API key has been saved." });
},
onError: (err: any) => {
toast({ title: "Error", description: err?.message || "Failed to save AI settings", variant: "destructive" });
},
});
const handleSubmit = (e: React.FormEvent) => {
e.preventDefault();
if (!apiKey.trim()) return;
saveMutation.mutate({ apiKey: apiKey.trim() });
};
const isConfigured = !!settings?.apiKey;
return (
<Card>
<CardContent className="space-y-4 py-6">
<div className="flex items-center gap-2">
<h3 className="text-lg font-semibold">Google AI Settings</h3>
{isConfigured && (
<span className="flex items-center gap-1 text-xs text-green-600 font-medium">
<CheckCircle className="h-3.5 w-3.5" /> Configured
</span>
)}
</div>
<p className="text-sm text-gray-500">
Enter your Google AI Studio API key to enable AI-powered SMS replies. When a patient responds to an
appointment reminder, the AI will automatically reply based on their answer.
</p>
{isLoading ? (
<p className="text-sm text-gray-400">Loading...</p>
) : (
<form className="space-y-4" onSubmit={handleSubmit}>
<div>
<label className="block text-sm font-medium">Google AI Studio API Key</label>
<div className="relative mt-1">
<input
type={showKey ? "text" : "password"}
value={apiKey}
onChange={(e) => setApiKey(e.target.value)}
className="p-2 border rounded w-full pr-10 font-mono text-sm"
placeholder="AIza••••••••••••••••••••••••••••••••••••••"
required
/>
<button
type="button"
onClick={() => setShowKey((v) => !v)}
className="absolute inset-y-0 right-2 flex items-center text-gray-500 hover:text-gray-700"
tabIndex={-1}
>
{showKey ? <EyeOff size={16} /> : <Eye size={16} />}
</button>
</div>
<p className="text-xs text-gray-500 mt-1">
Get your free key from{" "}
<span className="font-medium">Google AI Studio</span> Get API Key.
</p>
</div>
<div className="flex items-center gap-3 pt-1">
<button
type="submit"
className="bg-teal-600 text-white px-4 py-2 rounded hover:bg-teal-700 text-sm"
disabled={saveMutation.isPending}
>
{saveMutation.isPending ? "Saving..." : "Save AI Settings"}
</button>
</div>
{isConfigured && (
<div className="mt-2 p-3 bg-gray-50 rounded border text-xs text-gray-600 space-y-1">
<p className="font-medium text-gray-700">AI Auto-Reply Rules</p>
<p>
<span className="text-green-600 font-medium">Patient replies "Yes"</span> AI sends a thank-you confirmation
</p>
<p>
<span className="text-red-500 font-medium">Patient replies "No" / "Not available"</span> AI notifies them an assistant will follow up
</p>
</div>
)}
</form>
)}
</CardContent>
</Card>
);
}