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>
This commit is contained in:
ff
2026-06-06 09:34:11 -04:00
parent d5bc96ff39
commit 4899ab8368
57 changed files with 681 additions and 138 deletions

File diff suppressed because one or more lines are too long

View File

@@ -441,6 +441,9 @@ exports.Prisma.AiSettingsScalarFieldEnum = {
openAiEnabled: 'openAiEnabled',
claudeAiKey: 'claudeAiKey',
claudeAiEnabled: 'claudeAiEnabled',
claudeAiModel: 'claudeAiModel',
openAiModel: 'openAiModel',
googleAiModel: 'googleAiModel',
dentalMgmtKey: 'dentalMgmtKey',
dentalMgmtEnabled: 'dentalMgmtEnabled',
afterHoursEnabled: 'afterHoursEnabled',

View File

@@ -36467,6 +36467,9 @@ export namespace Prisma {
openAiEnabled: boolean | null
claudeAiKey: string | null
claudeAiEnabled: boolean | null
claudeAiModel: string | null
openAiModel: string | null
googleAiModel: string | null
dentalMgmtKey: string | null
dentalMgmtEnabled: boolean | null
afterHoursEnabled: boolean | null
@@ -36482,6 +36485,9 @@ export namespace Prisma {
openAiEnabled: boolean | null
claudeAiKey: string | null
claudeAiEnabled: boolean | null
claudeAiModel: string | null
openAiModel: string | null
googleAiModel: string | null
dentalMgmtKey: string | null
dentalMgmtEnabled: boolean | null
afterHoursEnabled: boolean | null
@@ -36497,6 +36503,9 @@ export namespace Prisma {
openAiEnabled: number
claudeAiKey: number
claudeAiEnabled: number
claudeAiModel: number
openAiModel: number
googleAiModel: number
dentalMgmtKey: number
dentalMgmtEnabled: number
afterHoursEnabled: number
@@ -36524,6 +36533,9 @@ export namespace Prisma {
openAiEnabled?: true
claudeAiKey?: true
claudeAiEnabled?: true
claudeAiModel?: true
openAiModel?: true
googleAiModel?: true
dentalMgmtKey?: true
dentalMgmtEnabled?: true
afterHoursEnabled?: true
@@ -36539,6 +36551,9 @@ export namespace Prisma {
openAiEnabled?: true
claudeAiKey?: true
claudeAiEnabled?: true
claudeAiModel?: true
openAiModel?: true
googleAiModel?: true
dentalMgmtKey?: true
dentalMgmtEnabled?: true
afterHoursEnabled?: true
@@ -36554,6 +36569,9 @@ export namespace Prisma {
openAiEnabled?: true
claudeAiKey?: true
claudeAiEnabled?: true
claudeAiModel?: true
openAiModel?: true
googleAiModel?: true
dentalMgmtKey?: true
dentalMgmtEnabled?: true
afterHoursEnabled?: true
@@ -36656,6 +36674,9 @@ export namespace Prisma {
openAiEnabled: boolean
claudeAiKey: string
claudeAiEnabled: boolean
claudeAiModel: string
openAiModel: string
googleAiModel: string
dentalMgmtKey: string
dentalMgmtEnabled: boolean
afterHoursEnabled: boolean
@@ -36690,6 +36711,9 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: boolean
claudeAiEnabled?: boolean
claudeAiModel?: boolean
openAiModel?: boolean
googleAiModel?: boolean
dentalMgmtKey?: boolean
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
@@ -36706,6 +36730,9 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: boolean
claudeAiEnabled?: boolean
claudeAiModel?: boolean
openAiModel?: boolean
googleAiModel?: boolean
dentalMgmtKey?: boolean
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
@@ -36722,6 +36749,9 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: boolean
claudeAiEnabled?: boolean
claudeAiModel?: boolean
openAiModel?: boolean
googleAiModel?: boolean
dentalMgmtKey?: boolean
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
@@ -36738,13 +36768,16 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: boolean
claudeAiEnabled?: boolean
claudeAiModel?: boolean
openAiModel?: boolean
googleAiModel?: boolean
dentalMgmtKey?: boolean
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
openPhoneReply?: boolean
}
export type AiSettingsOmit<ExtArgs extends $Extensions.InternalArgs = $Extensions.DefaultArgs> = $Extensions.GetOmit<"id" | "userId" | "apiKey" | "aiEnabled" | "openAiKey" | "openAiEnabled" | "claudeAiKey" | "claudeAiEnabled" | "dentalMgmtKey" | "dentalMgmtEnabled" | "afterHoursEnabled" | "openPhoneReply", ExtArgs["result"]["aiSettings"]>
export type AiSettingsOmit<ExtArgs extends $Extensions.InternalArgs = $Extensions.DefaultArgs> = $Extensions.GetOmit<"id" | "userId" | "apiKey" | "aiEnabled" | "openAiKey" | "openAiEnabled" | "claudeAiKey" | "claudeAiEnabled" | "claudeAiModel" | "openAiModel" | "googleAiModel" | "dentalMgmtKey" | "dentalMgmtEnabled" | "afterHoursEnabled" | "openPhoneReply", ExtArgs["result"]["aiSettings"]>
export type AiSettingsInclude<ExtArgs extends $Extensions.InternalArgs = $Extensions.DefaultArgs> = {
user?: boolean | UserDefaultArgs<ExtArgs>
}
@@ -36769,6 +36802,9 @@ export namespace Prisma {
openAiEnabled: boolean
claudeAiKey: string
claudeAiEnabled: boolean
claudeAiModel: string
openAiModel: string
googleAiModel: string
dentalMgmtKey: string
dentalMgmtEnabled: boolean
afterHoursEnabled: boolean
@@ -37205,6 +37241,9 @@ export namespace Prisma {
readonly openAiEnabled: FieldRef<"AiSettings", 'Boolean'>
readonly claudeAiKey: FieldRef<"AiSettings", 'String'>
readonly claudeAiEnabled: FieldRef<"AiSettings", 'Boolean'>
readonly claudeAiModel: FieldRef<"AiSettings", 'String'>
readonly openAiModel: FieldRef<"AiSettings", 'String'>
readonly googleAiModel: FieldRef<"AiSettings", 'String'>
readonly dentalMgmtKey: FieldRef<"AiSettings", 'String'>
readonly dentalMgmtEnabled: FieldRef<"AiSettings", 'Boolean'>
readonly afterHoursEnabled: FieldRef<"AiSettings", 'Boolean'>
@@ -45818,6 +45857,9 @@ export namespace Prisma {
openAiEnabled: 'openAiEnabled',
claudeAiKey: 'claudeAiKey',
claudeAiEnabled: 'claudeAiEnabled',
claudeAiModel: 'claudeAiModel',
openAiModel: 'openAiModel',
googleAiModel: 'googleAiModel',
dentalMgmtKey: 'dentalMgmtKey',
dentalMgmtEnabled: 'dentalMgmtEnabled',
afterHoursEnabled: 'afterHoursEnabled',
@@ -48464,6 +48506,9 @@ export namespace Prisma {
openAiEnabled?: BoolFilter<"AiSettings"> | boolean
claudeAiKey?: StringFilter<"AiSettings"> | string
claudeAiEnabled?: BoolFilter<"AiSettings"> | boolean
claudeAiModel?: StringFilter<"AiSettings"> | string
openAiModel?: StringFilter<"AiSettings"> | string
googleAiModel?: StringFilter<"AiSettings"> | string
dentalMgmtKey?: StringFilter<"AiSettings"> | string
dentalMgmtEnabled?: BoolFilter<"AiSettings"> | boolean
afterHoursEnabled?: BoolFilter<"AiSettings"> | boolean
@@ -48480,6 +48525,9 @@ export namespace Prisma {
openAiEnabled?: SortOrder
claudeAiKey?: SortOrder
claudeAiEnabled?: SortOrder
claudeAiModel?: SortOrder
openAiModel?: SortOrder
googleAiModel?: SortOrder
dentalMgmtKey?: SortOrder
dentalMgmtEnabled?: SortOrder
afterHoursEnabled?: SortOrder
@@ -48499,6 +48547,9 @@ export namespace Prisma {
openAiEnabled?: BoolFilter<"AiSettings"> | boolean
claudeAiKey?: StringFilter<"AiSettings"> | string
claudeAiEnabled?: BoolFilter<"AiSettings"> | boolean
claudeAiModel?: StringFilter<"AiSettings"> | string
openAiModel?: StringFilter<"AiSettings"> | string
googleAiModel?: StringFilter<"AiSettings"> | string
dentalMgmtKey?: StringFilter<"AiSettings"> | string
dentalMgmtEnabled?: BoolFilter<"AiSettings"> | boolean
afterHoursEnabled?: BoolFilter<"AiSettings"> | boolean
@@ -48515,6 +48566,9 @@ export namespace Prisma {
openAiEnabled?: SortOrder
claudeAiKey?: SortOrder
claudeAiEnabled?: SortOrder
claudeAiModel?: SortOrder
openAiModel?: SortOrder
googleAiModel?: SortOrder
dentalMgmtKey?: SortOrder
dentalMgmtEnabled?: SortOrder
afterHoursEnabled?: SortOrder
@@ -48538,6 +48592,9 @@ export namespace Prisma {
openAiEnabled?: BoolWithAggregatesFilter<"AiSettings"> | boolean
claudeAiKey?: StringWithAggregatesFilter<"AiSettings"> | string
claudeAiEnabled?: BoolWithAggregatesFilter<"AiSettings"> | boolean
claudeAiModel?: StringWithAggregatesFilter<"AiSettings"> | string
openAiModel?: StringWithAggregatesFilter<"AiSettings"> | string
googleAiModel?: StringWithAggregatesFilter<"AiSettings"> | string
dentalMgmtKey?: StringWithAggregatesFilter<"AiSettings"> | string
dentalMgmtEnabled?: BoolWithAggregatesFilter<"AiSettings"> | boolean
afterHoursEnabled?: BoolWithAggregatesFilter<"AiSettings"> | boolean
@@ -51245,6 +51302,9 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: string
claudeAiEnabled?: boolean
claudeAiModel?: string
openAiModel?: string
googleAiModel?: string
dentalMgmtKey?: string
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
@@ -51261,6 +51321,9 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: string
claudeAiEnabled?: boolean
claudeAiModel?: string
openAiModel?: string
googleAiModel?: string
dentalMgmtKey?: string
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
@@ -51274,6 +51337,9 @@ export namespace Prisma {
openAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiKey?: StringFieldUpdateOperationsInput | string
claudeAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiModel?: StringFieldUpdateOperationsInput | string
openAiModel?: StringFieldUpdateOperationsInput | string
googleAiModel?: StringFieldUpdateOperationsInput | string
dentalMgmtKey?: StringFieldUpdateOperationsInput | string
dentalMgmtEnabled?: BoolFieldUpdateOperationsInput | boolean
afterHoursEnabled?: BoolFieldUpdateOperationsInput | boolean
@@ -51290,6 +51356,9 @@ export namespace Prisma {
openAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiKey?: StringFieldUpdateOperationsInput | string
claudeAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiModel?: StringFieldUpdateOperationsInput | string
openAiModel?: StringFieldUpdateOperationsInput | string
googleAiModel?: StringFieldUpdateOperationsInput | string
dentalMgmtKey?: StringFieldUpdateOperationsInput | string
dentalMgmtEnabled?: BoolFieldUpdateOperationsInput | boolean
afterHoursEnabled?: BoolFieldUpdateOperationsInput | boolean
@@ -51305,6 +51374,9 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: string
claudeAiEnabled?: boolean
claudeAiModel?: string
openAiModel?: string
googleAiModel?: string
dentalMgmtKey?: string
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
@@ -51318,6 +51390,9 @@ export namespace Prisma {
openAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiKey?: StringFieldUpdateOperationsInput | string
claudeAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiModel?: StringFieldUpdateOperationsInput | string
openAiModel?: StringFieldUpdateOperationsInput | string
googleAiModel?: StringFieldUpdateOperationsInput | string
dentalMgmtKey?: StringFieldUpdateOperationsInput | string
dentalMgmtEnabled?: BoolFieldUpdateOperationsInput | boolean
afterHoursEnabled?: BoolFieldUpdateOperationsInput | boolean
@@ -51333,6 +51408,9 @@ export namespace Prisma {
openAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiKey?: StringFieldUpdateOperationsInput | string
claudeAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiModel?: StringFieldUpdateOperationsInput | string
openAiModel?: StringFieldUpdateOperationsInput | string
googleAiModel?: StringFieldUpdateOperationsInput | string
dentalMgmtKey?: StringFieldUpdateOperationsInput | string
dentalMgmtEnabled?: BoolFieldUpdateOperationsInput | boolean
afterHoursEnabled?: BoolFieldUpdateOperationsInput | boolean
@@ -53948,6 +54026,9 @@ export namespace Prisma {
openAiEnabled?: SortOrder
claudeAiKey?: SortOrder
claudeAiEnabled?: SortOrder
claudeAiModel?: SortOrder
openAiModel?: SortOrder
googleAiModel?: SortOrder
dentalMgmtKey?: SortOrder
dentalMgmtEnabled?: SortOrder
afterHoursEnabled?: SortOrder
@@ -53968,6 +54049,9 @@ export namespace Prisma {
openAiEnabled?: SortOrder
claudeAiKey?: SortOrder
claudeAiEnabled?: SortOrder
claudeAiModel?: SortOrder
openAiModel?: SortOrder
googleAiModel?: SortOrder
dentalMgmtKey?: SortOrder
dentalMgmtEnabled?: SortOrder
afterHoursEnabled?: SortOrder
@@ -53983,6 +54067,9 @@ export namespace Prisma {
openAiEnabled?: SortOrder
claudeAiKey?: SortOrder
claudeAiEnabled?: SortOrder
claudeAiModel?: SortOrder
openAiModel?: SortOrder
googleAiModel?: SortOrder
dentalMgmtKey?: SortOrder
dentalMgmtEnabled?: SortOrder
afterHoursEnabled?: SortOrder
@@ -58336,6 +58423,9 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: string
claudeAiEnabled?: boolean
claudeAiModel?: string
openAiModel?: string
googleAiModel?: string
dentalMgmtKey?: string
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
@@ -58350,6 +58440,9 @@ export namespace Prisma {
openAiEnabled?: boolean
claudeAiKey?: string
claudeAiEnabled?: boolean
claudeAiModel?: string
openAiModel?: string
googleAiModel?: string
dentalMgmtKey?: string
dentalMgmtEnabled?: boolean
afterHoursEnabled?: boolean
@@ -58962,6 +59055,9 @@ export namespace Prisma {
openAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiKey?: StringFieldUpdateOperationsInput | string
claudeAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiModel?: StringFieldUpdateOperationsInput | string
openAiModel?: StringFieldUpdateOperationsInput | string
googleAiModel?: StringFieldUpdateOperationsInput | string
dentalMgmtKey?: StringFieldUpdateOperationsInput | string
dentalMgmtEnabled?: BoolFieldUpdateOperationsInput | boolean
afterHoursEnabled?: BoolFieldUpdateOperationsInput | boolean
@@ -58976,6 +59072,9 @@ export namespace Prisma {
openAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiKey?: StringFieldUpdateOperationsInput | string
claudeAiEnabled?: BoolFieldUpdateOperationsInput | boolean
claudeAiModel?: StringFieldUpdateOperationsInput | string
openAiModel?: StringFieldUpdateOperationsInput | string
googleAiModel?: StringFieldUpdateOperationsInput | string
dentalMgmtKey?: StringFieldUpdateOperationsInput | string
dentalMgmtEnabled?: BoolFieldUpdateOperationsInput | boolean
afterHoursEnabled?: BoolFieldUpdateOperationsInput | boolean

File diff suppressed because one or more lines are too long

View File

@@ -1,5 +1,5 @@
{
"name": "prisma-client-5274dcc359e2ac7cb6d4e95a13b684de504baf5945107a0270a97148d1ad0620",
"name": "prisma-client-81ca44b9d3e5c16b41f6767b04f58e5bfe8927e8c0e21f3f03dff81c6db38733",
"main": "index.js",
"types": "index.d.ts",
"browser": "default.js",

View File

@@ -609,6 +609,9 @@ model AiSettings {
openAiEnabled Boolean @default(false)
claudeAiKey String @default("")
claudeAiEnabled Boolean @default(false)
claudeAiModel String @default("claude-haiku-4-5-20251001")
openAiModel String @default("gpt-5.2")
googleAiModel String @default("gemini-2.5-flash")
dentalMgmtKey String @default("")
dentalMgmtEnabled Boolean @default(false)
afterHoursEnabled Boolean @default(true)