fix: allow claims without appointment + support post/core multi-tooth input

- Make appointmentId nullable/optional in Prisma Zod schema via @zod rich
  comment so claims can be submitted without an existing appointment
- Convert undefined appointmentId to null in all claim form handlers and
  the backend claim creation endpoint
- Add AI classifier rule for expanding one procedure across multiple
  comma-separated tooth numbers (e.g. "post/core on #23, 24, 25, 26")
- Add "post/core" (slash) alias to CDT lookup maps

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-06-28 22:57:24 -04:00
parent c9d08028a9
commit e1fe4862d5
1613 changed files with 59200 additions and 422 deletions

View File

@@ -41,10 +41,14 @@ exports.PaymentGroupByResultSchema = z.array(z.object({
patientId: z.number().int(),
userId: z.number().int(),
updatedById: z.number().int(),
npiProviderId: z.number().int(),
totalBilled: z.number(),
totalPaid: z.number(),
totalAdjusted: z.number(),
totalDue: z.number(),
mhPaidAmount: z.number(),
copayment: z.number(),
adjustment: z.number(),
notes: z.string(),
icn: z.string(),
createdAt: z.date(),
@@ -55,10 +59,14 @@ exports.PaymentGroupByResultSchema = z.array(z.object({
patientId: z.number(),
userId: z.number(),
updatedById: z.number(),
npiProviderId: z.number(),
totalBilled: z.number(),
totalPaid: z.number(),
totalAdjusted: z.number(),
totalDue: z.number(),
mhPaidAmount: z.number(),
copayment: z.number(),
adjustment: z.number(),
status: z.number(),
notes: z.number(),
icn: z.number(),
@@ -67,8 +75,10 @@ exports.PaymentGroupByResultSchema = z.array(z.object({
claim: z.number(),
patient: z.number(),
updatedBy: z.number(),
npiProvider: z.number(),
serviceLineTransactions: z.number(),
serviceLines: z.number()
serviceLines: z.number(),
commissionBatchItems: z.number()
}).optional(),
_sum: z.object({
id: z.number().nullable(),
@@ -76,10 +86,14 @@ exports.PaymentGroupByResultSchema = z.array(z.object({
patientId: z.number().nullable(),
userId: z.number().nullable(),
updatedById: z.number().nullable(),
npiProviderId: z.number().nullable(),
totalBilled: z.number().nullable(),
totalPaid: z.number().nullable(),
totalAdjusted: z.number().nullable(),
totalDue: z.number().nullable()
totalDue: z.number().nullable(),
mhPaidAmount: z.number().nullable(),
copayment: z.number().nullable(),
adjustment: z.number().nullable()
}).nullable().optional(),
_avg: z.object({
id: z.number().nullable(),
@@ -87,10 +101,14 @@ exports.PaymentGroupByResultSchema = z.array(z.object({
patientId: z.number().nullable(),
userId: z.number().nullable(),
updatedById: z.number().nullable(),
npiProviderId: z.number().nullable(),
totalBilled: z.number().nullable(),
totalPaid: z.number().nullable(),
totalAdjusted: z.number().nullable(),
totalDue: z.number().nullable()
totalDue: z.number().nullable(),
mhPaidAmount: z.number().nullable(),
copayment: z.number().nullable(),
adjustment: z.number().nullable()
}).nullable().optional(),
_min: z.object({
id: z.number().int().nullable(),
@@ -98,10 +116,14 @@ exports.PaymentGroupByResultSchema = z.array(z.object({
patientId: z.number().int().nullable(),
userId: z.number().int().nullable(),
updatedById: z.number().int().nullable(),
npiProviderId: z.number().int().nullable(),
totalBilled: z.number().nullable(),
totalPaid: z.number().nullable(),
totalAdjusted: z.number().nullable(),
totalDue: z.number().nullable(),
mhPaidAmount: z.number().nullable(),
copayment: z.number().nullable(),
adjustment: z.number().nullable(),
notes: z.string().nullable(),
icn: z.string().nullable(),
createdAt: z.date().nullable(),
@@ -113,10 +135,14 @@ exports.PaymentGroupByResultSchema = z.array(z.object({
patientId: z.number().int().nullable(),
userId: z.number().int().nullable(),
updatedById: z.number().int().nullable(),
npiProviderId: z.number().int().nullable(),
totalBilled: z.number().nullable(),
totalPaid: z.number().nullable(),
totalAdjusted: z.number().nullable(),
totalDue: z.number().nullable(),
mhPaidAmount: z.number().nullable(),
copayment: z.number().nullable(),
adjustment: z.number().nullable(),
notes: z.string().nullable(),
icn: z.string().nullable(),
createdAt: z.date().nullable(),