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

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.literal(true).optional(),
claudeAiKey: z.literal(true).optional(),
claudeAiEnabled: z.literal(true).optional(),
claudeAiModel: z.literal(true).optional(),
openAiModel: z.literal(true).optional(),
googleAiModel: z.literal(true).optional(),
dentalMgmtKey: z.literal(true).optional(),
dentalMgmtEnabled: z.literal(true).optional(),
afterHoursEnabled: z.literal(true).optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: SortOrderSchema.optional(),
claudeAiKey: SortOrderSchema.optional(),
claudeAiEnabled: SortOrderSchema.optional(),
claudeAiModel: SortOrderSchema.optional(),
openAiModel: SortOrderSchema.optional(),
googleAiModel: SortOrderSchema.optional(),
dentalMgmtKey: SortOrderSchema.optional(),
dentalMgmtEnabled: SortOrderSchema.optional(),
afterHoursEnabled: SortOrderSchema.optional(),

View File

@@ -9,6 +9,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.boolean().optional(),
claudeAiKey: z.string().optional(),
claudeAiEnabled: z.boolean().optional(),
claudeAiModel: z.string().optional(),
openAiModel: z.string().optional(),
googleAiModel: z.string().optional(),
dentalMgmtKey: z.string().optional(),
dentalMgmtEnabled: z.boolean().optional(),
afterHoursEnabled: z.boolean().optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.boolean().optional(),
claudeAiKey: z.string().optional(),
claudeAiEnabled: z.boolean().optional(),
claudeAiModel: z.string().optional(),
openAiModel: z.string().optional(),
googleAiModel: z.string().optional(),
dentalMgmtKey: z.string().optional(),
dentalMgmtEnabled: z.boolean().optional(),
afterHoursEnabled: z.boolean().optional(),

View File

@@ -9,6 +9,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.boolean().optional(),
claudeAiKey: z.string().optional(),
claudeAiEnabled: z.boolean().optional(),
claudeAiModel: z.string().optional(),
openAiModel: z.string().optional(),
googleAiModel: z.string().optional(),
dentalMgmtKey: z.string().optional(),
dentalMgmtEnabled: z.boolean().optional(),
afterHoursEnabled: z.boolean().optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.literal(true).optional(),
claudeAiKey: z.literal(true).optional(),
claudeAiEnabled: z.literal(true).optional(),
claudeAiModel: z.literal(true).optional(),
openAiModel: z.literal(true).optional(),
googleAiModel: z.literal(true).optional(),
dentalMgmtKey: z.literal(true).optional(),
dentalMgmtEnabled: z.literal(true).optional(),
afterHoursEnabled: z.literal(true).optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: SortOrderSchema.optional(),
claudeAiKey: SortOrderSchema.optional(),
claudeAiEnabled: SortOrderSchema.optional(),
claudeAiModel: SortOrderSchema.optional(),
openAiModel: SortOrderSchema.optional(),
googleAiModel: SortOrderSchema.optional(),
dentalMgmtKey: SortOrderSchema.optional(),
dentalMgmtEnabled: SortOrderSchema.optional(),
afterHoursEnabled: SortOrderSchema.optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.literal(true).optional(),
claudeAiKey: z.literal(true).optional(),
claudeAiEnabled: z.literal(true).optional(),
claudeAiModel: z.literal(true).optional(),
openAiModel: z.literal(true).optional(),
googleAiModel: z.literal(true).optional(),
dentalMgmtKey: z.literal(true).optional(),
dentalMgmtEnabled: z.literal(true).optional(),
afterHoursEnabled: z.literal(true).optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: SortOrderSchema.optional(),
claudeAiKey: SortOrderSchema.optional(),
claudeAiEnabled: SortOrderSchema.optional(),
claudeAiModel: SortOrderSchema.optional(),
openAiModel: SortOrderSchema.optional(),
googleAiModel: SortOrderSchema.optional(),
dentalMgmtKey: SortOrderSchema.optional(),
dentalMgmtEnabled: SortOrderSchema.optional(),
afterHoursEnabled: SortOrderSchema.optional(),

View File

@@ -16,6 +16,9 @@ const makeSchema = () => z.object({
openAiEnabled: SortOrderSchema.optional(),
claudeAiKey: SortOrderSchema.optional(),
claudeAiEnabled: SortOrderSchema.optional(),
claudeAiModel: SortOrderSchema.optional(),
openAiModel: SortOrderSchema.optional(),
googleAiModel: SortOrderSchema.optional(),
dentalMgmtKey: SortOrderSchema.optional(),
dentalMgmtEnabled: SortOrderSchema.optional(),
afterHoursEnabled: SortOrderSchema.optional(),

View File

@@ -12,6 +12,9 @@ const makeSchema = () => z.object({
openAiEnabled: SortOrderSchema.optional(),
claudeAiKey: SortOrderSchema.optional(),
claudeAiEnabled: SortOrderSchema.optional(),
claudeAiModel: SortOrderSchema.optional(),
openAiModel: SortOrderSchema.optional(),
googleAiModel: SortOrderSchema.optional(),
dentalMgmtKey: SortOrderSchema.optional(),
dentalMgmtEnabled: SortOrderSchema.optional(),
afterHoursEnabled: SortOrderSchema.optional(),

View File

@@ -16,6 +16,9 @@ const aisettingsscalarwherewithaggregatesinputSchema = z.object({
openAiEnabled: z.union([z.lazy(() => BoolWithAggregatesFilterObjectSchema), z.boolean()]).optional(),
claudeAiKey: z.union([z.lazy(() => StringWithAggregatesFilterObjectSchema), z.string()]).optional(),
claudeAiEnabled: z.union([z.lazy(() => BoolWithAggregatesFilterObjectSchema), z.boolean()]).optional(),
claudeAiModel: z.union([z.lazy(() => StringWithAggregatesFilterObjectSchema), z.string()]).optional(),
openAiModel: z.union([z.lazy(() => StringWithAggregatesFilterObjectSchema), z.string()]).optional(),
googleAiModel: z.union([z.lazy(() => StringWithAggregatesFilterObjectSchema), z.string()]).optional(),
dentalMgmtKey: z.union([z.lazy(() => StringWithAggregatesFilterObjectSchema), z.string()]).optional(),
dentalMgmtEnabled: z.union([z.lazy(() => BoolWithAggregatesFilterObjectSchema), z.boolean()]).optional(),
afterHoursEnabled: z.union([z.lazy(() => BoolWithAggregatesFilterObjectSchema), z.boolean()]).optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.boolean().optional(),
claudeAiKey: z.boolean().optional(),
claudeAiEnabled: z.boolean().optional(),
claudeAiModel: z.boolean().optional(),
openAiModel: z.boolean().optional(),
googleAiModel: z.boolean().optional(),
dentalMgmtKey: z.boolean().optional(),
dentalMgmtEnabled: z.boolean().optional(),
afterHoursEnabled: z.boolean().optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.boolean().optional(),
claudeAiKey: z.string().optional(),
claudeAiEnabled: z.boolean().optional(),
claudeAiModel: z.string().optional(),
openAiModel: z.string().optional(),
googleAiModel: z.string().optional(),
dentalMgmtKey: z.string().optional(),
dentalMgmtEnabled: z.boolean().optional(),
afterHoursEnabled: z.boolean().optional(),

View File

@@ -10,6 +10,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.boolean().optional(),
claudeAiKey: z.string().optional(),
claudeAiEnabled: z.boolean().optional(),
claudeAiModel: z.string().optional(),
openAiModel: z.string().optional(),
googleAiModel: z.string().optional(),
dentalMgmtKey: z.string().optional(),
dentalMgmtEnabled: z.boolean().optional(),
afterHoursEnabled: z.boolean().optional(),

View File

@@ -13,6 +13,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
openAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
googleAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
afterHoursEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),

View File

@@ -13,6 +13,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
openAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
googleAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
afterHoursEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),

View File

@@ -12,6 +12,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
openAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
googleAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
afterHoursEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),

View File

@@ -11,6 +11,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
openAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
googleAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
afterHoursEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),

View File

@@ -10,6 +10,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
openAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
googleAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
afterHoursEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),

View File

@@ -10,6 +10,9 @@ const makeSchema = () => z.object({
openAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
claudeAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
openAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
googleAiModel: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtKey: z.union([z.string(), z.lazy(() => StringFieldUpdateOperationsInputObjectSchema)]).optional(),
dentalMgmtEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),
afterHoursEnabled: z.union([z.boolean(), z.lazy(() => BoolFieldUpdateOperationsInputObjectSchema)]).optional(),

View File

@@ -18,6 +18,9 @@ const aisettingswhereinputSchema = z.object({
openAiEnabled: z.union([z.lazy(() => BoolFilterObjectSchema), z.boolean()]).optional(),
claudeAiKey: z.union([z.lazy(() => StringFilterObjectSchema), z.string()]).optional(),
claudeAiEnabled: z.union([z.lazy(() => BoolFilterObjectSchema), z.boolean()]).optional(),
claudeAiModel: z.union([z.lazy(() => StringFilterObjectSchema), z.string()]).optional(),
openAiModel: z.union([z.lazy(() => StringFilterObjectSchema), z.string()]).optional(),
googleAiModel: z.union([z.lazy(() => StringFilterObjectSchema), z.string()]).optional(),
dentalMgmtKey: z.union([z.lazy(() => StringFilterObjectSchema), z.string()]).optional(),
dentalMgmtEnabled: z.union([z.lazy(() => BoolFilterObjectSchema), z.boolean()]).optional(),
afterHoursEnabled: z.union([z.lazy(() => BoolFilterObjectSchema), z.boolean()]).optional(),