Files
DentalManagementMH06/apps/Backend/src/storage/ai-settings-storage.ts
ff 4899ab8368 feat: multi-provider AI support with per-provider model selection
- Add llm-factory.ts: unified LLM provider abstraction (Google/Claude/OpenAI)
- Install @langchain/anthropic and @langchain/openai packages
- resolveAiProvider picks active provider from DB settings (Claude > OpenAI > Google)
- All AI graphs (reminder, new-patient, reschedule, internal-chat) now accept provider+model params
- Add claudeAiModel, openAiModel, googleAiModel columns to ai_settings table
- New PUT /api/ai/provider-model route to save selected model per provider
- UI model dropdowns for Claude (Haiku/Sonnet/Opus), OpenAI (GPT-5.x series), Google (Gemini 2.5/3.x)
- Google AI section also gets model selector alongside existing API key field

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-06-06 09:34:11 -04:00

64 lines
2.4 KiB
TypeScript

import { prisma as db } from "@repo/db/client";
export const aiSettingsStorage = {
async getAiSettings(userId: number) {
return db.aiSettings.findUnique({ where: { userId } });
},
async upsertAiSettings(userId: number, apiKey: string, aiEnabled?: boolean) {
return db.aiSettings.upsert({
where: { userId },
update: { apiKey, ...(aiEnabled !== undefined && { aiEnabled }) },
create: { userId, apiKey, aiEnabled: aiEnabled ?? true },
});
},
async setAiEnabled(userId: number, enabled: boolean): Promise<void> {
await db.aiSettings.upsert({
where: { userId },
update: { aiEnabled: enabled },
create: { userId, apiKey: "", aiEnabled: enabled },
});
},
async upsertProviderKey(userId: number, provider: "openAi" | "claudeAi" | "dentalMgmt", key: string): Promise<void> {
const field = provider === "openAi" ? "openAiKey" : provider === "claudeAi" ? "claudeAiKey" : "dentalMgmtKey";
await db.aiSettings.upsert({
where: { userId },
update: { [field]: key },
create: { userId, apiKey: "", [field]: key },
});
},
async setProviderModel(userId: number, provider: "claudeAi" | "openAi" | "googleAi", model: string): Promise<void> {
const field = provider === "claudeAi" ? "claudeAiModel" : provider === "openAi" ? "openAiModel" : "googleAiModel";
await db.aiSettings.upsert({
where: { userId },
update: { [field]: model },
create: { userId, apiKey: "", [field]: model },
});
},
async setProviderEnabled(userId: number, provider: "openAi" | "claudeAi" | "dentalMgmt", enabled: boolean): Promise<void> {
const field = provider === "openAi" ? "openAiEnabled" : provider === "claudeAi" ? "claudeAiEnabled" : "dentalMgmtEnabled";
await db.aiSettings.upsert({
where: { userId },
update: { [field]: enabled },
create: { userId, apiKey: "", [field]: enabled, openAiEnabled: false, claudeAiEnabled: false, dentalMgmtEnabled: false },
});
},
async getOpenPhoneReply(userId: number): Promise<boolean> {
const row = await db.aiSettings.findUnique({ where: { userId } });
return row?.openPhoneReply ?? false;
},
async setOpenPhoneReply(userId: number, enabled: boolean): Promise<void> {
await db.aiSettings.upsert({
where: { userId },
update: { openPhoneReply: enabled },
create: { userId, apiKey: "", openPhoneReply: enabled },
});
},
};