elevenlabs-webhook-nodejs 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (969) hide show
  1. package/.claude/settings.local.json +27 -0
  2. package/.dockerignore +6 -0
  3. package/.env.example +43 -0
  4. package/BEFORE-PRODUCTION.md +32 -0
  5. package/Dockerfile +23 -0
  6. package/SYSTEM_OVERVIEW.md +942 -0
  7. package/SYSTEM_STATUS.md +332 -0
  8. package/backup_script/main.py +146 -0
  9. package/baileys/.dockerignore +7 -0
  10. package/baileys/Dockerfile +13 -0
  11. package/baileys/README.md +412 -0
  12. package/baileys/index.js +499 -0
  13. package/baileys/package-lock.json +2532 -0
  14. package/baileys/package.json +25 -0
  15. package/baileys/server.js +96 -0
  16. package/baileys/src/config.js +55 -0
  17. package/baileys/src/middleware/api-key.js +16 -0
  18. package/baileys/src/routes/accounts.js +51 -0
  19. package/baileys/src/routes/chats.js +103 -0
  20. package/baileys/src/routes/webhooks.js +34 -0
  21. package/baileys/src/services/account-store.js +259 -0
  22. package/baileys/src/services/webhook-dispatcher.js +161 -0
  23. package/baileys/src/services/worker-manager.js +597 -0
  24. package/baileys/src/utils/jid.js +79 -0
  25. package/baileys/src/utils/logger.js +16 -0
  26. package/baileys/src/utils/use-mongodb-auth-state.js +122 -0
  27. package/baileys/worker.js +721 -0
  28. package/dist/app.d.ts +1 -0
  29. package/dist/app.js +84 -0
  30. package/dist/app.js.map +1 -0
  31. package/dist/config/agentPrompts.json +19 -0
  32. package/dist/config/database.d.ts +1 -0
  33. package/dist/config/database.js +26 -0
  34. package/dist/config/database.js.map +1 -0
  35. package/dist/config/env.d.ts +41 -0
  36. package/dist/config/env.js +81 -0
  37. package/dist/config/env.js.map +1 -0
  38. package/dist/config/index.d.ts +3 -0
  39. package/dist/config/index.js +73 -0
  40. package/dist/config/index.js.map +1 -0
  41. package/dist/controllers/webhook.controller.d.ts +10 -0
  42. package/dist/controllers/webhook.controller.js +128 -0
  43. package/dist/controllers/webhook.controller.js.map +1 -0
  44. package/dist/errors/index.d.ts +4 -0
  45. package/dist/errors/index.js +13 -0
  46. package/dist/errors/index.js.map +1 -0
  47. package/dist/hooks/webhookValidator.d.ts +2 -0
  48. package/dist/hooks/webhookValidator.js +47 -0
  49. package/dist/hooks/webhookValidator.js.map +1 -0
  50. package/dist/index.d.ts +1 -0
  51. package/dist/index.js +55 -0
  52. package/dist/index.js.map +1 -0
  53. package/dist/migrations/001_session-architecture.d.ts +3 -0
  54. package/dist/migrations/001_session-architecture.js +119 -0
  55. package/dist/migrations/001_session-architecture.js.map +1 -0
  56. package/dist/migrations/002_agent-ref.d.ts +3 -0
  57. package/dist/migrations/002_agent-ref.js +55 -0
  58. package/dist/migrations/002_agent-ref.js.map +1 -0
  59. package/dist/migrations/003_shift-schedule.d.ts +3 -0
  60. package/dist/migrations/003_shift-schedule.js +48 -0
  61. package/dist/migrations/003_shift-schedule.js.map +1 -0
  62. package/dist/migrations/004_invert-shift-to-clinic-hours.d.ts +3 -0
  63. package/dist/migrations/004_invert-shift-to-clinic-hours.js +27 -0
  64. package/dist/migrations/004_invert-shift-to-clinic-hours.js.map +1 -0
  65. package/dist/migrations/005_composite-baileys-chat-ids.d.ts +3 -0
  66. package/dist/migrations/005_composite-baileys-chat-ids.js +47 -0
  67. package/dist/migrations/005_composite-baileys-chat-ids.js.map +1 -0
  68. package/dist/migrations/migration.model.d.ts +18 -0
  69. package/dist/migrations/migration.model.js +45 -0
  70. package/dist/migrations/migration.model.js.map +1 -0
  71. package/dist/migrations/migration.routes.d.ts +2 -0
  72. package/dist/migrations/migration.routes.js +37 -0
  73. package/dist/migrations/migration.routes.js.map +1 -0
  74. package/dist/migrations/migration.runner.d.ts +19 -0
  75. package/dist/migrations/migration.runner.js +74 -0
  76. package/dist/migrations/migration.runner.js.map +1 -0
  77. package/dist/models/ConversationClaim.d.ts +13 -0
  78. package/dist/models/ConversationClaim.js +44 -0
  79. package/dist/models/ConversationClaim.js.map +1 -0
  80. package/dist/models/ConversationEvaluation.d.ts +23 -0
  81. package/dist/models/ConversationEvaluation.js +92 -0
  82. package/dist/models/ConversationEvaluation.js.map +1 -0
  83. package/dist/models/Summary.d.ts +37 -0
  84. package/dist/models/Summary.js +78 -0
  85. package/dist/models/Summary.js.map +1 -0
  86. package/dist/models/Transcription.d.ts +34 -0
  87. package/dist/models/Transcription.js +71 -0
  88. package/dist/models/Transcription.js.map +1 -0
  89. package/dist/models/WhatsAppRecipient.d.ts +23 -0
  90. package/dist/models/WhatsAppRecipient.js +53 -0
  91. package/dist/models/WhatsAppRecipient.js.map +1 -0
  92. package/dist/modules/admin/admin.routes.d.ts +2 -0
  93. package/dist/modules/admin/admin.routes.js +1966 -0
  94. package/dist/modules/admin/admin.routes.js.map +1 -0
  95. package/dist/modules/admin/elevenlabs-test-chat.routes.d.ts +2 -0
  96. package/dist/modules/admin/elevenlabs-test-chat.routes.js +158 -0
  97. package/dist/modules/admin/elevenlabs-test-chat.routes.js.map +1 -0
  98. package/dist/modules/admin/whatsapp-test-chat.routes.d.ts +2 -0
  99. package/dist/modules/admin/whatsapp-test-chat.routes.js +204 -0
  100. package/dist/modules/admin/whatsapp-test-chat.routes.js.map +1 -0
  101. package/dist/modules/agents/agent.model.d.ts +38 -0
  102. package/dist/modules/agents/agent.model.js +92 -0
  103. package/dist/modules/agents/agent.model.js.map +1 -0
  104. package/dist/modules/agents/agent.routes.d.ts +2 -0
  105. package/dist/modules/agents/agent.routes.js +61 -0
  106. package/dist/modules/agents/agent.routes.js.map +1 -0
  107. package/dist/modules/appointment-validation/appointment-validation.model.d.ts +76 -0
  108. package/dist/modules/appointment-validation/appointment-validation.model.js +118 -0
  109. package/dist/modules/appointment-validation/appointment-validation.model.js.map +1 -0
  110. package/dist/modules/appointment-validation/appointment-validation.routes.d.ts +2 -0
  111. package/dist/modules/appointment-validation/appointment-validation.routes.js +202 -0
  112. package/dist/modules/appointment-validation/appointment-validation.routes.js.map +1 -0
  113. package/dist/modules/appointment-validation/appointment-validation.service.d.ts +53 -0
  114. package/dist/modules/appointment-validation/appointment-validation.service.js +827 -0
  115. package/dist/modules/appointment-validation/appointment-validation.service.js.map +1 -0
  116. package/dist/modules/auth/auth.model.d.ts +17 -0
  117. package/dist/modules/auth/auth.model.js +64 -0
  118. package/dist/modules/auth/auth.model.js.map +1 -0
  119. package/dist/modules/auth/auth.routes.d.ts +2 -0
  120. package/dist/modules/auth/auth.routes.js +202 -0
  121. package/dist/modules/auth/auth.routes.js.map +1 -0
  122. package/dist/modules/auth/auth.service.d.ts +28 -0
  123. package/dist/modules/auth/auth.service.js +183 -0
  124. package/dist/modules/auth/auth.service.js.map +1 -0
  125. package/dist/modules/auth/refresh-token.model.d.ts +13 -0
  126. package/dist/modules/auth/refresh-token.model.js +52 -0
  127. package/dist/modules/auth/refresh-token.model.js.map +1 -0
  128. package/dist/modules/billing/billing-alert.model.d.ts +16 -0
  129. package/dist/modules/billing/billing-alert.model.js +47 -0
  130. package/dist/modules/billing/billing-alert.model.js.map +1 -0
  131. package/dist/modules/billing/billing-period-snapshot.model.d.ts +35 -0
  132. package/dist/modules/billing/billing-period-snapshot.model.js +68 -0
  133. package/dist/modules/billing/billing-period-snapshot.model.js.map +1 -0
  134. package/dist/modules/billing/billing.model.d.ts +18 -0
  135. package/dist/modules/billing/billing.model.js +62 -0
  136. package/dist/modules/billing/billing.model.js.map +1 -0
  137. package/dist/modules/billing/billing.routes.d.ts +2 -0
  138. package/dist/modules/billing/billing.routes.js +63 -0
  139. package/dist/modules/billing/billing.routes.js.map +1 -0
  140. package/dist/modules/billing/billing.service.d.ts +69 -0
  141. package/dist/modules/billing/billing.service.js +498 -0
  142. package/dist/modules/billing/billing.service.js.map +1 -0
  143. package/dist/modules/billing/payment.model.d.ts +24 -0
  144. package/dist/modules/billing/payment.model.js +57 -0
  145. package/dist/modules/billing/payment.model.js.map +1 -0
  146. package/dist/modules/calls/call.model.d.ts +41 -0
  147. package/dist/modules/calls/call.model.js +97 -0
  148. package/dist/modules/calls/call.model.js.map +1 -0
  149. package/dist/modules/calls/call.routes.d.ts +2 -0
  150. package/dist/modules/calls/call.routes.js +103 -0
  151. package/dist/modules/calls/call.routes.js.map +1 -0
  152. package/dist/modules/campaigns/campaign.model.d.ts +45 -0
  153. package/dist/modules/campaigns/campaign.model.js +98 -0
  154. package/dist/modules/campaigns/campaign.model.js.map +1 -0
  155. package/dist/modules/campaigns/campaign.routes.d.ts +2 -0
  156. package/dist/modules/campaigns/campaign.routes.js +323 -0
  157. package/dist/modules/campaigns/campaign.routes.js.map +1 -0
  158. package/dist/modules/campaigns/campaign.service.d.ts +11 -0
  159. package/dist/modules/campaigns/campaign.service.js +86 -0
  160. package/dist/modules/campaigns/campaign.service.js.map +1 -0
  161. package/dist/modules/google-calendar/google-calendar.routes.d.ts +2 -0
  162. package/dist/modules/google-calendar/google-calendar.routes.js +32 -0
  163. package/dist/modules/google-calendar/google-calendar.routes.js.map +1 -0
  164. package/dist/modules/inbound-call/inbound-call-config.model.d.ts +20 -0
  165. package/dist/modules/inbound-call/inbound-call-config.model.js +68 -0
  166. package/dist/modules/inbound-call/inbound-call-config.model.js.map +1 -0
  167. package/dist/modules/inbound-call/inbound-call.routes.d.ts +2 -0
  168. package/dist/modules/inbound-call/inbound-call.routes.js +243 -0
  169. package/dist/modules/inbound-call/inbound-call.routes.js.map +1 -0
  170. package/dist/modules/leads/lead.model.d.ts +24 -0
  171. package/dist/modules/leads/lead.model.js +54 -0
  172. package/dist/modules/leads/lead.model.js.map +1 -0
  173. package/dist/modules/leads/lead.routes.d.ts +2 -0
  174. package/dist/modules/leads/lead.routes.js +201 -0
  175. package/dist/modules/leads/lead.routes.js.map +1 -0
  176. package/dist/modules/plans/plan.model.d.ts +25 -0
  177. package/dist/modules/plans/plan.model.js +59 -0
  178. package/dist/modules/plans/plan.model.js.map +1 -0
  179. package/dist/modules/surveys/survey.model.d.ts +30 -0
  180. package/dist/modules/surveys/survey.model.js +75 -0
  181. package/dist/modules/surveys/survey.model.js.map +1 -0
  182. package/dist/modules/surveys/survey.routes.d.ts +2 -0
  183. package/dist/modules/surveys/survey.routes.js +153 -0
  184. package/dist/modules/surveys/survey.routes.js.map +1 -0
  185. package/dist/modules/tenants/tenant.model.d.ts +86 -0
  186. package/dist/modules/tenants/tenant.model.js +127 -0
  187. package/dist/modules/tenants/tenant.model.js.map +1 -0
  188. package/dist/modules/tenants/tenant.routes.d.ts +2 -0
  189. package/dist/modules/tenants/tenant.routes.js +65 -0
  190. package/dist/modules/tenants/tenant.routes.js.map +1 -0
  191. package/dist/modules/users/user.routes.d.ts +2 -0
  192. package/dist/modules/users/user.routes.js +106 -0
  193. package/dist/modules/users/user.routes.js.map +1 -0
  194. package/dist/modules/webhooks/elevenlabs-tool.routes.d.ts +20 -0
  195. package/dist/modules/webhooks/elevenlabs-tool.routes.js +85 -0
  196. package/dist/modules/webhooks/elevenlabs-tool.routes.js.map +1 -0
  197. package/dist/modules/webhooks/elevenlabs-tool.service.d.ts +11 -0
  198. package/dist/modules/webhooks/elevenlabs-tool.service.js +360 -0
  199. package/dist/modules/webhooks/elevenlabs-tool.service.js.map +1 -0
  200. package/dist/modules/webhooks/elevenlabs.routes.d.ts +2 -0
  201. package/dist/modules/webhooks/elevenlabs.routes.js +34 -0
  202. package/dist/modules/webhooks/elevenlabs.routes.js.map +1 -0
  203. package/dist/modules/webhooks/elevenlabs.service.d.ts +6 -0
  204. package/dist/modules/webhooks/elevenlabs.service.js +512 -0
  205. package/dist/modules/webhooks/elevenlabs.service.js.map +1 -0
  206. package/dist/modules/webhooks/unipile.routes.d.ts +2 -0
  207. package/dist/modules/webhooks/unipile.routes.js +780 -0
  208. package/dist/modules/webhooks/unipile.routes.js.map +1 -0
  209. package/dist/modules/whatsapp/appointment.model.d.ts +27 -0
  210. package/dist/modules/whatsapp/appointment.model.js +58 -0
  211. package/dist/modules/whatsapp/appointment.model.js.map +1 -0
  212. package/dist/modules/whatsapp/operator-request.model.d.ts +29 -0
  213. package/dist/modules/whatsapp/operator-request.model.js +65 -0
  214. package/dist/modules/whatsapp/operator-request.model.js.map +1 -0
  215. package/dist/modules/whatsapp/stt-usage.model.d.ts +16 -0
  216. package/dist/modules/whatsapp/stt-usage.model.js +53 -0
  217. package/dist/modules/whatsapp/stt-usage.model.js.map +1 -0
  218. package/dist/modules/whatsapp/whatsapp-chat.model.d.ts +23 -0
  219. package/dist/modules/whatsapp/whatsapp-chat.model.js +55 -0
  220. package/dist/modules/whatsapp/whatsapp-chat.model.js.map +1 -0
  221. package/dist/modules/whatsapp/whatsapp-contact-profile.model.d.ts +23 -0
  222. package/dist/modules/whatsapp/whatsapp-contact-profile.model.js +54 -0
  223. package/dist/modules/whatsapp/whatsapp-contact-profile.model.js.map +1 -0
  224. package/dist/modules/whatsapp/whatsapp-message.model.d.ts +30 -0
  225. package/dist/modules/whatsapp/whatsapp-message.model.js +52 -0
  226. package/dist/modules/whatsapp/whatsapp-message.model.js.map +1 -0
  227. package/dist/modules/whatsapp/whatsapp-session.model.d.ts +33 -0
  228. package/dist/modules/whatsapp/whatsapp-session.model.js +65 -0
  229. package/dist/modules/whatsapp/whatsapp-session.model.js.map +1 -0
  230. package/dist/modules/whatsapp/whatsapp.routes.d.ts +2 -0
  231. package/dist/modules/whatsapp/whatsapp.routes.js +1237 -0
  232. package/dist/modules/whatsapp/whatsapp.routes.js.map +1 -0
  233. package/dist/plugins/cors.d.ts +3 -0
  234. package/dist/plugins/cors.js +11 -0
  235. package/dist/plugins/cors.js.map +1 -0
  236. package/dist/plugins/jwt.d.ts +3 -0
  237. package/dist/plugins/jwt.js +22 -0
  238. package/dist/plugins/jwt.js.map +1 -0
  239. package/dist/plugins/rawBody.d.ts +3 -0
  240. package/dist/plugins/rawBody.js +16 -0
  241. package/dist/plugins/rawBody.js.map +1 -0
  242. package/dist/plugins/rbac.d.ts +5 -0
  243. package/dist/plugins/rbac.js +29 -0
  244. package/dist/plugins/rbac.js.map +1 -0
  245. package/dist/routes/admin.routes.d.ts +2 -0
  246. package/dist/routes/admin.routes.js +169 -0
  247. package/dist/routes/admin.routes.js.map +1 -0
  248. package/dist/routes/health.routes.d.ts +2 -0
  249. package/dist/routes/health.routes.js +16 -0
  250. package/dist/routes/health.routes.js.map +1 -0
  251. package/dist/routes/webhook.routes.d.ts +2 -0
  252. package/dist/routes/webhook.routes.js +17 -0
  253. package/dist/routes/webhook.routes.js.map +1 -0
  254. package/dist/services/ai/base.ai.d.ts +19 -0
  255. package/dist/services/ai/base.ai.js +120 -0
  256. package/dist/services/ai/base.ai.js.map +1 -0
  257. package/dist/services/ai/gemini.service.d.ts +11 -0
  258. package/dist/services/ai/gemini.service.js +43 -0
  259. package/dist/services/ai/gemini.service.js.map +1 -0
  260. package/dist/services/ai/index.d.ts +2 -0
  261. package/dist/services/ai/index.js +26 -0
  262. package/dist/services/ai/index.js.map +1 -0
  263. package/dist/services/ai/openai.service.d.ts +11 -0
  264. package/dist/services/ai/openai.service.js +50 -0
  265. package/dist/services/ai/openai.service.js.map +1 -0
  266. package/dist/services/elevenlabs.service.d.ts +52 -0
  267. package/dist/services/elevenlabs.service.js +447 -0
  268. package/dist/services/elevenlabs.service.js.map +1 -0
  269. package/dist/services/google-calendar.service.d.ts +60 -0
  270. package/dist/services/google-calendar.service.js +494 -0
  271. package/dist/services/google-calendar.service.js.map +1 -0
  272. package/dist/services/inbound-call-schedule.service.d.ts +11 -0
  273. package/dist/services/inbound-call-schedule.service.js +162 -0
  274. package/dist/services/inbound-call-schedule.service.js.map +1 -0
  275. package/dist/services/netgsm.service.d.ts +41 -0
  276. package/dist/services/netgsm.service.js +89 -0
  277. package/dist/services/netgsm.service.js.map +1 -0
  278. package/dist/services/unipile.service.d.ts +41 -0
  279. package/dist/services/unipile.service.js +149 -0
  280. package/dist/services/unipile.service.js.map +1 -0
  281. package/dist/services/whatsapp-agent.service.d.ts +139 -0
  282. package/dist/services/whatsapp-agent.service.js +2055 -0
  283. package/dist/services/whatsapp-agent.service.js.map +1 -0
  284. package/dist/services/whatsapp.service.d.ts +26 -0
  285. package/dist/services/whatsapp.service.js +206 -0
  286. package/dist/services/whatsapp.service.js.map +1 -0
  287. package/dist/templates/index.d.ts +39 -0
  288. package/dist/templates/index.js +35 -0
  289. package/dist/templates/index.js.map +1 -0
  290. package/dist/templates/receptionist.d.ts +2 -0
  291. package/dist/templates/receptionist.js +39 -0
  292. package/dist/templates/receptionist.js.map +1 -0
  293. package/dist/templates/survey.d.ts +2 -0
  294. package/dist/templates/survey.js +41 -0
  295. package/dist/templates/survey.js.map +1 -0
  296. package/dist/types/index.d.ts +173 -0
  297. package/dist/types/index.js +3 -0
  298. package/dist/types/index.js.map +1 -0
  299. package/dist/utils/logger.d.ts +3 -0
  300. package/dist/utils/logger.js +105 -0
  301. package/dist/utils/logger.js.map +1 -0
  302. package/docker-compose.nestjs.yml +89 -0
  303. package/docker-compose.yml +78 -0
  304. package/docs/AI_AGENT_ENHANCEMENT_PLAN.md +164 -0
  305. package/docs/API.md +1193 -0
  306. package/docs/API_ENDPOINTS.md +344 -0
  307. package/docs/ARCHITECTURE.md +305 -0
  308. package/docs/AUTH_API.md +252 -0
  309. package/docs/BILLING_SMS_ALERTS.md +94 -0
  310. package/docs/CHAT_ASSIGNMENT_SYSTEM.md +118 -0
  311. package/docs/CLIENT_TOOLS_AND_FEATURES.md +337 -0
  312. package/docs/ELEVENLABS_WEBHOOK_TOOLS.md +644 -0
  313. package/docs/FRONTEND_CHECKLIST.md +227 -0
  314. package/docs/IMPLEMENTATION_STATUS.md +470 -0
  315. package/docs/MIGRATION_GUIDE.md +96 -0
  316. package/docs/MISSINGS_REPORT.md +507 -0
  317. package/docs/NESTJS_MIGRATION_REFERENCE.md +5136 -0
  318. package/docs/PROJECT_DESCRIPTION.md +1038 -0
  319. package/docs/SCALING.md +148 -0
  320. package/docs/SESSION_SUMMARY_2026_03_17.md +135 -0
  321. package/docs/WHATSAPP_AGENT.md +1086 -0
  322. package/docs/architecture/00-SYSTEM-OVERVIEW.md +318 -0
  323. package/docs/architecture/01-DATABASE-SCHEMA.md +2564 -0
  324. package/docs/architecture/02-AUTHENTICATION.md +1040 -0
  325. package/docs/architecture/03-MULTI-CLINIC.md +742 -0
  326. package/docs/architecture/04-WHATSAPP-AGENT.md +608 -0
  327. package/docs/architecture/05-OPERATOR-WORKFLOW.md +444 -0
  328. package/docs/architecture/06-BAILEYS-MICROSERVICE.md +616 -0
  329. package/docs/architecture/07-APPOINTMENTS.md +849 -0
  330. package/docs/architecture/08-VOICE-CALLS.md +470 -0
  331. package/docs/architecture/09-OUTBOUND-CAMPAIGNS.md +542 -0
  332. package/docs/architecture/10-CLIENT-TOOLS.md +665 -0
  333. package/docs/architecture/11-BILLING.md +458 -0
  334. package/docs/architecture/12-SECURITY.md +216 -0
  335. package/docs/architecture/13-LOGGING-AUDIT.md +549 -0
  336. package/docs/architecture/14-META-BUSINESS-API.md +454 -0
  337. package/docs/architecture/15-AI-MODULE.md +479 -0
  338. package/docs/architecture/16-BACKGROUND-JOBS.md +469 -0
  339. package/docs/architecture/17-REALTIME-LIVECHAT.md +447 -0
  340. package/docs/architecture/18-FILE-STORAGE.md +410 -0
  341. package/docs/architecture/19-PATIENTS.md +1034 -0
  342. package/docs/architecture/20-TREATMENTS-AND-PLANS.md +774 -0
  343. package/docs/architecture/21-BEFORE-AFTER-PHOTOS.md +519 -0
  344. package/docs/database.md +456 -0
  345. package/docs/ornek-randevu-onay.csv +3 -0
  346. package/ecosystem.config.js +16 -0
  347. package/elevenlabs-convai-api-reference.md +1171 -0
  348. package/frontend/.dockerignore +2 -0
  349. package/frontend/Dockerfile +24 -0
  350. package/frontend/README.md +75 -0
  351. package/frontend/components.json +25 -0
  352. package/frontend/eslint.config.js +23 -0
  353. package/frontend/index.html +13 -0
  354. package/frontend/nginx.conf +37 -0
  355. package/frontend/package-lock.json +8709 -0
  356. package/frontend/package.json +71 -0
  357. package/frontend/public/favicon.svg +1 -0
  358. package/frontend/public/icons.svg +24 -0
  359. package/frontend/src/App.tsx +125 -0
  360. package/frontend/src/components/error-boundary.tsx +50 -0
  361. package/frontend/src/components/shared/activity-timeline.tsx +66 -0
  362. package/frontend/src/components/shared/appointment-calendar.css +80 -0
  363. package/frontend/src/components/shared/appointment-calendar.tsx +245 -0
  364. package/frontend/src/components/shared/chat-bubble.tsx +72 -0
  365. package/frontend/src/components/shared/combobox.tsx +119 -0
  366. package/frontend/src/components/shared/confirm-dialog.tsx +57 -0
  367. package/frontend/src/components/shared/data-table-pagination.tsx +97 -0
  368. package/frontend/src/components/shared/data-table-toolbar.tsx +39 -0
  369. package/frontend/src/components/shared/empty-state.tsx +27 -0
  370. package/frontend/src/components/shared/file-upload.tsx +140 -0
  371. package/frontend/src/components/shared/index.ts +20 -0
  372. package/frontend/src/components/shared/language-switcher.tsx +29 -0
  373. package/frontend/src/components/shared/notification-dropdown.tsx +115 -0
  374. package/frontend/src/components/shared/page-header.tsx +23 -0
  375. package/frontend/src/components/shared/stat-card.tsx +37 -0
  376. package/frontend/src/components/shared/status-badge.tsx +70 -0
  377. package/frontend/src/components/shared/status-dot.tsx +43 -0
  378. package/frontend/src/components/ui/alert-dialog.tsx +187 -0
  379. package/frontend/src/components/ui/alert.tsx +76 -0
  380. package/frontend/src/components/ui/avatar.tsx +109 -0
  381. package/frontend/src/components/ui/badge.tsx +52 -0
  382. package/frontend/src/components/ui/breadcrumb.tsx +125 -0
  383. package/frontend/src/components/ui/button.tsx +60 -0
  384. package/frontend/src/components/ui/calendar.tsx +219 -0
  385. package/frontend/src/components/ui/card.tsx +103 -0
  386. package/frontend/src/components/ui/chart.tsx +371 -0
  387. package/frontend/src/components/ui/checkbox.tsx +29 -0
  388. package/frontend/src/components/ui/collapsible.tsx +19 -0
  389. package/frontend/src/components/ui/command.tsx +194 -0
  390. package/frontend/src/components/ui/dialog.tsx +158 -0
  391. package/frontend/src/components/ui/dropdown-menu.tsx +268 -0
  392. package/frontend/src/components/ui/input-group.tsx +156 -0
  393. package/frontend/src/components/ui/input.tsx +20 -0
  394. package/frontend/src/components/ui/label.tsx +20 -0
  395. package/frontend/src/components/ui/pagination.tsx +130 -0
  396. package/frontend/src/components/ui/popover.tsx +90 -0
  397. package/frontend/src/components/ui/progress.tsx +83 -0
  398. package/frontend/src/components/ui/radio-group.tsx +36 -0
  399. package/frontend/src/components/ui/scroll-area.tsx +54 -0
  400. package/frontend/src/components/ui/select.tsx +199 -0
  401. package/frontend/src/components/ui/separator.tsx +23 -0
  402. package/frontend/src/components/ui/sheet.tsx +138 -0
  403. package/frontend/src/components/ui/sidebar.tsx +723 -0
  404. package/frontend/src/components/ui/skeleton.tsx +13 -0
  405. package/frontend/src/components/ui/sonner.tsx +47 -0
  406. package/frontend/src/components/ui/switch.tsx +30 -0
  407. package/frontend/src/components/ui/table.tsx +114 -0
  408. package/frontend/src/components/ui/tabs.tsx +82 -0
  409. package/frontend/src/components/ui/textarea.tsx +18 -0
  410. package/frontend/src/components/ui/toggle-group.tsx +89 -0
  411. package/frontend/src/components/ui/toggle.tsx +43 -0
  412. package/frontend/src/components/ui/tooltip.tsx +64 -0
  413. package/frontend/src/hooks/use-mobile.ts +19 -0
  414. package/frontend/src/i18n/index.ts +20 -0
  415. package/frontend/src/i18n/locales/en.json +786 -0
  416. package/frontend/src/i18n/locales/tr.json +786 -0
  417. package/frontend/src/index.css +134 -0
  418. package/frontend/src/layouts/admin-layout.tsx +111 -0
  419. package/frontend/src/layouts/app-header.tsx +130 -0
  420. package/frontend/src/layouts/app-layout.tsx +19 -0
  421. package/frontend/src/layouts/app-sidebar.tsx +212 -0
  422. package/frontend/src/layouts/auth-guard.tsx +31 -0
  423. package/frontend/src/layouts/mobile-sidebar.tsx +120 -0
  424. package/frontend/src/lib/api.ts +68 -0
  425. package/frontend/src/lib/hooks/use-appointments.ts +224 -0
  426. package/frontend/src/lib/hooks/use-availability-blocks.ts +83 -0
  427. package/frontend/src/lib/hooks/use-chats.ts +236 -0
  428. package/frontend/src/lib/hooks/use-clinic-detail.ts +171 -0
  429. package/frontend/src/lib/hooks/use-clinics.ts +37 -0
  430. package/frontend/src/lib/hooks/use-doctor-calendar.ts +80 -0
  431. package/frontend/src/lib/hooks/use-examinations.ts +89 -0
  432. package/frontend/src/lib/hooks/use-medical-history.ts +52 -0
  433. package/frontend/src/lib/hooks/use-patients.ts +296 -0
  434. package/frontend/src/lib/hooks/use-photo-sets.ts +118 -0
  435. package/frontend/src/lib/hooks/use-treatment-plans.ts +152 -0
  436. package/frontend/src/lib/hooks/use-treatments.ts +125 -0
  437. package/frontend/src/lib/hooks/use-users.ts +172 -0
  438. package/frontend/src/lib/utils.ts +6 -0
  439. package/frontend/src/main.tsx +17 -0
  440. package/frontend/src/pages/admin/agents/detail.tsx +774 -0
  441. package/frontend/src/pages/admin/agents/import.tsx +280 -0
  442. package/frontend/src/pages/admin/agents/index.tsx +245 -0
  443. package/frontend/src/pages/admin/ai-playground.tsx +543 -0
  444. package/frontend/src/pages/appointments/create-appointment-dialog.tsx +390 -0
  445. package/frontend/src/pages/appointments/index.tsx +860 -0
  446. package/frontend/src/pages/audit/index.tsx +24 -0
  447. package/frontend/src/pages/auth/login.tsx +194 -0
  448. package/frontend/src/pages/billing/index.tsx +24 -0
  449. package/frontend/src/pages/calendar/index.tsx +704 -0
  450. package/frontend/src/pages/calendar-connections/index.tsx +295 -0
  451. package/frontend/src/pages/calls/index.tsx +24 -0
  452. package/frontend/src/pages/campaigns/index.tsx +24 -0
  453. package/frontend/src/pages/chats/index.tsx +981 -0
  454. package/frontend/src/pages/clinics/index.tsx +224 -0
  455. package/frontend/src/pages/clinics/settings.tsx +412 -0
  456. package/frontend/src/pages/components-showcase.tsx +773 -0
  457. package/frontend/src/pages/connections/index.tsx +328 -0
  458. package/frontend/src/pages/dashboard.tsx +50 -0
  459. package/frontend/src/pages/leads/index.tsx +24 -0
  460. package/frontend/src/pages/my-schedule/index.tsx +496 -0
  461. package/frontend/src/pages/patients/create-patient-dialog.tsx +358 -0
  462. package/frontend/src/pages/patients/detail.tsx +1195 -0
  463. package/frontend/src/pages/patients/edit-patient-dialog.tsx +387 -0
  464. package/frontend/src/pages/patients/examinations-tab.tsx +460 -0
  465. package/frontend/src/pages/patients/index.tsx +381 -0
  466. package/frontend/src/pages/patients/medical-history-dialog.tsx +207 -0
  467. package/frontend/src/pages/patients/photo-sets-tab.tsx +616 -0
  468. package/frontend/src/pages/patients/timeline-tab.tsx +164 -0
  469. package/frontend/src/pages/patients/treatment-plans-tab.tsx +598 -0
  470. package/frontend/src/pages/phone-numbers/detail.tsx +427 -0
  471. package/frontend/src/pages/phone-numbers/index.tsx +455 -0
  472. package/frontend/src/pages/platform/index.tsx +454 -0
  473. package/frontend/src/pages/settings/index.tsx +126 -0
  474. package/frontend/src/pages/treatments/index.tsx +487 -0
  475. package/frontend/src/pages/users/doctor-profile.tsx +672 -0
  476. package/frontend/src/pages/users/edit.tsx +329 -0
  477. package/frontend/src/pages/users/index.tsx +407 -0
  478. package/frontend/src/pages/validation/index.tsx +24 -0
  479. package/frontend/src/stores/auth.store.ts +108 -0
  480. package/frontend/src/stores/ui.store.ts +41 -0
  481. package/frontend/tsconfig.app.json +32 -0
  482. package/frontend/tsconfig.json +13 -0
  483. package/frontend/tsconfig.node.json +26 -0
  484. package/frontend/vite.config.ts +29 -0
  485. package/nestjs/.dockerignore +5 -0
  486. package/nestjs/.env.docker +64 -0
  487. package/nestjs/.prettierrc +4 -0
  488. package/nestjs/Dockerfile +36 -0
  489. package/nestjs/README.md +98 -0
  490. package/nestjs/eslint.config.mjs +35 -0
  491. package/nestjs/nest-cli.json +8 -0
  492. package/nestjs/package-lock.json +13390 -0
  493. package/nestjs/package.json +114 -0
  494. package/nestjs/prisma/migrations/20260409161536_add_message_metadata_fields/migration.sql +1746 -0
  495. package/nestjs/prisma/migrations/20260410140436_add_agent_ai_fields/migration.sql +36 -0
  496. package/nestjs/prisma/migrations/20260410175519_add_agent_clinic_assignments/migration.sql +21 -0
  497. package/nestjs/prisma/migrations/20260412094344_make_agent_tenant_optional/migration.sql +2 -0
  498. package/nestjs/prisma/migrations/20260412110008_add_admin_chat_sessions/migration.sql +47 -0
  499. package/nestjs/prisma/migrations/migration_lock.toml +3 -0
  500. package/nestjs/prisma/schema.prisma +1843 -0
  501. package/nestjs/prisma/seed.ts +375 -0
  502. package/nestjs/prisma.config.ts +14 -0
  503. package/nestjs/src/admin/admin.controller.ts +27 -0
  504. package/nestjs/src/admin/admin.module.ts +16 -0
  505. package/nestjs/src/admin/admin.service.ts +91 -0
  506. package/nestjs/src/admin/ai-chat-session.service.ts +454 -0
  507. package/nestjs/src/admin/ai-test.controller.ts +191 -0
  508. package/nestjs/src/admin/superadmin.controller.ts +106 -0
  509. package/nestjs/src/agents/agents.controller.ts +262 -0
  510. package/nestjs/src/agents/agents.module.ts +13 -0
  511. package/nestjs/src/agents/agents.service.ts +733 -0
  512. package/nestjs/src/agents/dto/create-agent.dto.ts +99 -0
  513. package/nestjs/src/agents/dto/index.ts +2 -0
  514. package/nestjs/src/agents/dto/update-agent.dto.ts +148 -0
  515. package/nestjs/src/app.module.ts +115 -0
  516. package/nestjs/src/appointment-validation/appointment-validation.controller.ts +194 -0
  517. package/nestjs/src/appointment-validation/appointment-validation.module.ts +16 -0
  518. package/nestjs/src/appointment-validation/appointment-validation.service.ts +450 -0
  519. package/nestjs/src/appointment-validation/dto/create-batch.dto.ts +105 -0
  520. package/nestjs/src/appointment-validation/dto/index.ts +1 -0
  521. package/nestjs/src/appointment-validation/processors/validation-dispatch.processor.ts +26 -0
  522. package/nestjs/src/appointment-validation/processors/validation-sync.processor.ts +23 -0
  523. package/nestjs/src/appointments/appointments.controller.ts +268 -0
  524. package/nestjs/src/appointments/appointments.module.ts +13 -0
  525. package/nestjs/src/appointments/appointments.service.ts +773 -0
  526. package/nestjs/src/appointments/dto/create-appointment.dto.ts +72 -0
  527. package/nestjs/src/appointments/dto/index.ts +4 -0
  528. package/nestjs/src/appointments/dto/query-appointments.dto.ts +60 -0
  529. package/nestjs/src/appointments/dto/update-appointment.dto.ts +43 -0
  530. package/nestjs/src/appointments/dto/update-status.dto.ts +18 -0
  531. package/nestjs/src/appointments/processors/reminder.processor.ts +243 -0
  532. package/nestjs/src/audit/audit.controller.ts +84 -0
  533. package/nestjs/src/audit/audit.decorator.ts +20 -0
  534. package/nestjs/src/audit/audit.interceptor.ts +67 -0
  535. package/nestjs/src/audit/audit.interfaces.ts +15 -0
  536. package/nestjs/src/audit/audit.module.ts +12 -0
  537. package/nestjs/src/audit/audit.service.ts +177 -0
  538. package/nestjs/src/auth/auth.controller.ts +116 -0
  539. package/nestjs/src/auth/auth.module.ts +25 -0
  540. package/nestjs/src/auth/auth.service.ts +612 -0
  541. package/nestjs/src/auth/dto/change-password.dto.ts +13 -0
  542. package/nestjs/src/auth/dto/forgot-password.dto.ts +8 -0
  543. package/nestjs/src/auth/dto/index.ts +6 -0
  544. package/nestjs/src/auth/dto/login.dto.ts +13 -0
  545. package/nestjs/src/auth/dto/refresh.dto.ts +8 -0
  546. package/nestjs/src/auth/dto/register.dto.ts +28 -0
  547. package/nestjs/src/auth/dto/reset-password.dto.ts +13 -0
  548. package/nestjs/src/auth/permissions.config.ts +85 -0
  549. package/nestjs/src/availability-blocks/availability-blocks.controller.ts +83 -0
  550. package/nestjs/src/availability-blocks/availability-blocks.module.ts +10 -0
  551. package/nestjs/src/availability-blocks/availability-blocks.service.ts +202 -0
  552. package/nestjs/src/billing/billing.controller.ts +104 -0
  553. package/nestjs/src/billing/billing.module.ts +12 -0
  554. package/nestjs/src/billing/billing.service.ts +398 -0
  555. package/nestjs/src/billing/dto/index.ts +2 -0
  556. package/nestjs/src/billing/dto/query-billing.dto.ts +29 -0
  557. package/nestjs/src/billing/dto/record-payment.dto.ts +60 -0
  558. package/nestjs/src/billing/processors/alerts.processor.ts +216 -0
  559. package/nestjs/src/billing/processors/snapshot.processor.ts +181 -0
  560. package/nestjs/src/calls/calls.controller.ts +92 -0
  561. package/nestjs/src/calls/calls.module.ts +10 -0
  562. package/nestjs/src/calls/calls.service.ts +359 -0
  563. package/nestjs/src/calls/dto/index.ts +1 -0
  564. package/nestjs/src/calls/dto/query-calls.dto.ts +46 -0
  565. package/nestjs/src/clinics/clinics.controller.ts +226 -0
  566. package/nestjs/src/clinics/clinics.module.ts +11 -0
  567. package/nestjs/src/clinics/clinics.service.ts +203 -0
  568. package/nestjs/src/clinics/dto/create-clinic.dto.ts +40 -0
  569. package/nestjs/src/clinics/dto/create-meta-connection.dto.ts +44 -0
  570. package/nestjs/src/clinics/dto/index.ts +4 -0
  571. package/nestjs/src/clinics/dto/update-clinic.dto.ts +133 -0
  572. package/nestjs/src/clinics/dto/update-meta-connection.dto.ts +22 -0
  573. package/nestjs/src/clinics/meta-connections.service.ts +210 -0
  574. package/nestjs/src/common/decorators/accessible-clinics.decorator.ts +9 -0
  575. package/nestjs/src/common/decorators/current-user.decorator.ts +10 -0
  576. package/nestjs/src/common/decorators/features.decorator.ts +7 -0
  577. package/nestjs/src/common/decorators/index.ts +4 -0
  578. package/nestjs/src/common/decorators/permissions.decorator.ts +13 -0
  579. package/nestjs/src/common/gateways/events.gateway.ts +157 -0
  580. package/nestjs/src/common/guards/clinic-access.guard.ts +40 -0
  581. package/nestjs/src/common/guards/features.guard.ts +38 -0
  582. package/nestjs/src/common/guards/index.ts +5 -0
  583. package/nestjs/src/common/guards/jwt-auth.guard.ts +54 -0
  584. package/nestjs/src/common/guards/permissions.guard.ts +50 -0
  585. package/nestjs/src/common/guards/superadmin.guard.ts +35 -0
  586. package/nestjs/src/common/interceptors/request-context.interceptor.ts +32 -0
  587. package/nestjs/src/common/interceptors/superadmin-tenant.interceptor.ts +42 -0
  588. package/nestjs/src/common/interfaces/jwt-payload.interface.ts +11 -0
  589. package/nestjs/src/config/config.module.ts +13 -0
  590. package/nestjs/src/database/database.module.ts +9 -0
  591. package/nestjs/src/database/prisma.service.ts +20 -0
  592. package/nestjs/src/database/tenant-context.ts +27 -0
  593. package/nestjs/src/google-calendar/google-calendar.controller.ts +259 -0
  594. package/nestjs/src/google-calendar/google-calendar.module.ts +10 -0
  595. package/nestjs/src/google-calendar/google-calendar.service.ts +811 -0
  596. package/nestjs/src/integrations/ai/ai.interface.ts +74 -0
  597. package/nestjs/src/integrations/ai/ai.module.ts +11 -0
  598. package/nestjs/src/integrations/ai/ai.service.ts +148 -0
  599. package/nestjs/src/integrations/ai/providers/anthropic.provider.ts +146 -0
  600. package/nestjs/src/integrations/ai/providers/openai.provider.ts +158 -0
  601. package/nestjs/src/integrations/elevenlabs/elevenlabs.module.ts +8 -0
  602. package/nestjs/src/integrations/elevenlabs/elevenlabs.service.ts +226 -0
  603. package/nestjs/src/integrations/encryption/encryption.module.ts +9 -0
  604. package/nestjs/src/integrations/encryption/encryption.service.ts +31 -0
  605. package/nestjs/src/integrations/image/image.module.ts +9 -0
  606. package/nestjs/src/integrations/image/image.service.ts +61 -0
  607. package/nestjs/src/integrations/meta-business/meta-business.module.ts +10 -0
  608. package/nestjs/src/integrations/meta-business/meta-instagram.service.ts +94 -0
  609. package/nestjs/src/integrations/meta-business/meta-webhook.service.ts +52 -0
  610. package/nestjs/src/integrations/meta-business/meta-whatsapp.service.ts +254 -0
  611. package/nestjs/src/integrations/minio/minio.module.ts +9 -0
  612. package/nestjs/src/integrations/minio/minio.service.ts +88 -0
  613. package/nestjs/src/integrations/netgsm/netgsm.module.ts +8 -0
  614. package/nestjs/src/integrations/netgsm/netgsm.service.ts +17 -0
  615. package/nestjs/src/integrations/tektippay/tektippay.module.ts +8 -0
  616. package/nestjs/src/integrations/tektippay/tektippay.service.ts +23 -0
  617. package/nestjs/src/leads/dto/index.ts +2 -0
  618. package/nestjs/src/leads/dto/query-leads.dto.ts +50 -0
  619. package/nestjs/src/leads/dto/update-lead.dto.ts +26 -0
  620. package/nestjs/src/leads/leads.controller.ts +184 -0
  621. package/nestjs/src/leads/leads.module.ts +10 -0
  622. package/nestjs/src/leads/leads.service.ts +375 -0
  623. package/nestjs/src/logging/logging.controller.ts +82 -0
  624. package/nestjs/src/logging/logging.module.ts +11 -0
  625. package/nestjs/src/logging/logging.service.ts +180 -0
  626. package/nestjs/src/main.ts +86 -0
  627. package/nestjs/src/outbound-campaigns/dto/create-campaign.dto.ts +47 -0
  628. package/nestjs/src/outbound-campaigns/dto/index.ts +3 -0
  629. package/nestjs/src/outbound-campaigns/dto/update-campaign.dto.ts +31 -0
  630. package/nestjs/src/outbound-campaigns/dto/upload-entries.dto.ts +35 -0
  631. package/nestjs/src/outbound-campaigns/outbound-campaigns.controller.ts +307 -0
  632. package/nestjs/src/outbound-campaigns/outbound-campaigns.module.ts +13 -0
  633. package/nestjs/src/outbound-campaigns/outbound-campaigns.service.ts +471 -0
  634. package/nestjs/src/outbound-campaigns/processors/campaign-dispatch.processor.ts +366 -0
  635. package/nestjs/src/patients/documents.service.ts +231 -0
  636. package/nestjs/src/patients/dto/create-examination.dto.ts +34 -0
  637. package/nestjs/src/patients/dto/create-note.dto.ts +14 -0
  638. package/nestjs/src/patients/dto/create-patient.dto.ts +86 -0
  639. package/nestjs/src/patients/dto/create-photo-set.dto.ts +32 -0
  640. package/nestjs/src/patients/dto/index.ts +10 -0
  641. package/nestjs/src/patients/dto/update-examination.dto.ts +29 -0
  642. package/nestjs/src/patients/dto/update-medical-history.dto.ts +47 -0
  643. package/nestjs/src/patients/dto/update-patient.dto.ts +87 -0
  644. package/nestjs/src/patients/dto/update-photo-set.dto.ts +28 -0
  645. package/nestjs/src/patients/dto/upload-document.dto.ts +31 -0
  646. package/nestjs/src/patients/dto/upload-photo.dto.ts +27 -0
  647. package/nestjs/src/patients/examinations.service.ts +271 -0
  648. package/nestjs/src/patients/medical-history.service.ts +149 -0
  649. package/nestjs/src/patients/notes.service.ts +172 -0
  650. package/nestjs/src/patients/patients.controller.ts +485 -0
  651. package/nestjs/src/patients/patients.module.ts +22 -0
  652. package/nestjs/src/patients/patients.service.ts +412 -0
  653. package/nestjs/src/patients/photo-sets.service.ts +389 -0
  654. package/nestjs/src/phone-numbers/dto/create-phone-number.dto.ts +57 -0
  655. package/nestjs/src/phone-numbers/dto/index.ts +3 -0
  656. package/nestjs/src/phone-numbers/dto/update-phone-number.dto.ts +38 -0
  657. package/nestjs/src/phone-numbers/dto/update-schedule.dto.ts +39 -0
  658. package/nestjs/src/phone-numbers/phone-numbers.controller.ts +125 -0
  659. package/nestjs/src/phone-numbers/phone-numbers.module.ts +10 -0
  660. package/nestjs/src/phone-numbers/phone-numbers.service.ts +209 -0
  661. package/nestjs/src/plans/dto/create-plan.dto.ts +70 -0
  662. package/nestjs/src/plans/dto/index.ts +2 -0
  663. package/nestjs/src/plans/dto/update-plan.dto.ts +80 -0
  664. package/nestjs/src/plans/plans.controller.ts +50 -0
  665. package/nestjs/src/plans/plans.module.ts +10 -0
  666. package/nestjs/src/plans/plans.service.ts +115 -0
  667. package/nestjs/src/platform/dto/create-tenant.dto.ts +36 -0
  668. package/nestjs/src/platform/dto/index.ts +2 -0
  669. package/nestjs/src/platform/dto/update-tenant-platform.dto.ts +44 -0
  670. package/nestjs/src/platform/platform.controller.ts +79 -0
  671. package/nestjs/src/platform/platform.module.ts +10 -0
  672. package/nestjs/src/platform/platform.service.ts +301 -0
  673. package/nestjs/src/queue/queue.module.ts +56 -0
  674. package/nestjs/src/redis/redis.module.ts +20 -0
  675. package/nestjs/src/tenants/dto/index.ts +1 -0
  676. package/nestjs/src/tenants/dto/update-tenant.dto.ts +15 -0
  677. package/nestjs/src/tenants/tenants.controller.ts +45 -0
  678. package/nestjs/src/tenants/tenants.module.ts +10 -0
  679. package/nestjs/src/tenants/tenants.service.ts +41 -0
  680. package/nestjs/src/tools/adapters/elevenlabs-http.adapter.ts +51 -0
  681. package/nestjs/src/tools/adapters/elevenlabs-ws.adapter.ts +59 -0
  682. package/nestjs/src/tools/handlers/calendar.tools.ts +441 -0
  683. package/nestjs/src/tools/handlers/notification.tools.ts +34 -0
  684. package/nestjs/src/tools/handlers/operator.tools.ts +303 -0
  685. package/nestjs/src/tools/handlers/patient.tools.ts +242 -0
  686. package/nestjs/src/tools/handlers/payment.tools.ts +43 -0
  687. package/nestjs/src/tools/handlers/validation.tools.ts +152 -0
  688. package/nestjs/src/tools/tool-registry.service.ts +221 -0
  689. package/nestjs/src/tools/tool.decorator.ts +16 -0
  690. package/nestjs/src/tools/tool.interfaces.ts +26 -0
  691. package/nestjs/src/tools/tools.module.ts +50 -0
  692. package/nestjs/src/treatments/dto/create-plan-item.dto.ts +27 -0
  693. package/nestjs/src/treatments/dto/create-treatment-plan.dto.ts +69 -0
  694. package/nestjs/src/treatments/dto/create-treatment.dto.ts +59 -0
  695. package/nestjs/src/treatments/dto/index.ts +6 -0
  696. package/nestjs/src/treatments/dto/update-plan-item.dto.ts +23 -0
  697. package/nestjs/src/treatments/dto/update-treatment-plan.dto.ts +22 -0
  698. package/nestjs/src/treatments/dto/update-treatment.dto.ts +70 -0
  699. package/nestjs/src/treatments/treatment-plans.service.ts +362 -0
  700. package/nestjs/src/treatments/treatments.controller.ts +265 -0
  701. package/nestjs/src/treatments/treatments.module.ts +14 -0
  702. package/nestjs/src/treatments/treatments.service.ts +165 -0
  703. package/nestjs/src/users/doctor-profiles.service.ts +202 -0
  704. package/nestjs/src/users/dto/index.ts +4 -0
  705. package/nestjs/src/users/dto/invite-user.dto.ts +52 -0
  706. package/nestjs/src/users/dto/update-clinic-assignments.dto.ts +9 -0
  707. package/nestjs/src/users/dto/update-doctor-profile.dto.ts +49 -0
  708. package/nestjs/src/users/dto/update-user.dto.ts +41 -0
  709. package/nestjs/src/users/users.controller.ts +142 -0
  710. package/nestjs/src/users/users.module.ts +11 -0
  711. package/nestjs/src/users/users.service.ts +250 -0
  712. package/nestjs/src/webhooks/elevenlabs-tool.controller.ts +66 -0
  713. package/nestjs/src/webhooks/elevenlabs-webhook.controller.ts +60 -0
  714. package/nestjs/src/webhooks/meta-webhook.controller.ts +178 -0
  715. package/nestjs/src/webhooks/processors/elevenlabs-webhook.processor.ts +28 -0
  716. package/nestjs/src/webhooks/webhooks.module.ts +17 -0
  717. package/nestjs/src/whatsapp/chat-context.service.ts +281 -0
  718. package/nestjs/src/whatsapp/dto/add-blacklist.dto.ts +13 -0
  719. package/nestjs/src/whatsapp/dto/index.ts +2 -0
  720. package/nestjs/src/whatsapp/dto/send-message.dto.ts +14 -0
  721. package/nestjs/src/whatsapp/listeners/meta-message.listener.ts +579 -0
  722. package/nestjs/src/whatsapp/meta-auth.controller.ts +268 -0
  723. package/nestjs/src/whatsapp/meta-instagram-auth.controller.ts +244 -0
  724. package/nestjs/src/whatsapp/operator.service.ts +692 -0
  725. package/nestjs/src/whatsapp/processors/cost-sweep.processor.ts +130 -0
  726. package/nestjs/src/whatsapp/processors/grace.processor.ts +191 -0
  727. package/nestjs/src/whatsapp/processors/message-buffer.processor.ts +138 -0
  728. package/nestjs/src/whatsapp/processors/message-cleanup.processor.ts +148 -0
  729. package/nestjs/src/whatsapp/processors/meta-token-refresh.processor.ts +114 -0
  730. package/nestjs/src/whatsapp/processors/operator-expiry.processor.ts +105 -0
  731. package/nestjs/src/whatsapp/processors/profile-update.processor.ts +234 -0
  732. package/nestjs/src/whatsapp/processors/session-cleanup.processor.ts +178 -0
  733. package/nestjs/src/whatsapp/processors/session-labels.processor.ts +248 -0
  734. package/nestjs/src/whatsapp/whatsapp-agent.service.ts +2506 -0
  735. package/nestjs/src/whatsapp/whatsapp-recovery.service.ts +117 -0
  736. package/nestjs/src/whatsapp/whatsapp.controller.ts +398 -0
  737. package/nestjs/src/whatsapp/whatsapp.module.ts +51 -0
  738. package/nestjs/src/whatsapp/whatsapp.service.ts +592 -0
  739. package/nestjs/test/app.e2e-spec.ts +25 -0
  740. package/nestjs/test/jest-e2e.json +9 -0
  741. package/nestjs/tsconfig.build.json +4 -0
  742. package/nestjs/tsconfig.json +25 -0
  743. package/nginx.example.conf +18 -0
  744. package/package.json +47 -0
  745. package/scripts/addRecipient.ts +48 -0
  746. package/scripts/listRecipients.ts +31 -0
  747. package/scripts/migrate-agent-ref.ts +86 -0
  748. package/scripts/migrate-sessions.ts +183 -0
  749. package/scripts/promote.ts +27 -0
  750. package/scripts/retrigger.ts +84 -0
  751. package/scripts/seed.ts +435 -0
  752. package/scripts/testSend.ts +63 -0
  753. package/src/app.ts +85 -0
  754. package/src/config/agentPrompts.json +19 -0
  755. package/src/config/database.ts +21 -0
  756. package/src/config/env.ts +94 -0
  757. package/src/config/index.ts +86 -0
  758. package/src/controllers/webhook.controller.ts +150 -0
  759. package/src/errors/index.ts +9 -0
  760. package/src/hooks/webhookValidator.ts +55 -0
  761. package/src/index.ts +68 -0
  762. package/src/migrations/001_session-architecture.ts +138 -0
  763. package/src/migrations/002_agent-ref.ts +65 -0
  764. package/src/migrations/003_shift-schedule.ts +55 -0
  765. package/src/migrations/004_invert-shift-to-clinic-hours.ts +30 -0
  766. package/src/migrations/005_composite-baileys-chat-ids.ts +60 -0
  767. package/src/migrations/migration.model.ts +27 -0
  768. package/src/migrations/migration.routes.ts +40 -0
  769. package/src/migrations/migration.runner.ts +112 -0
  770. package/src/models/ConversationClaim.ts +17 -0
  771. package/src/models/ConversationEvaluation.ts +91 -0
  772. package/src/models/Summary.ts +77 -0
  773. package/src/models/Transcription.ts +68 -0
  774. package/src/models/WhatsAppRecipient.ts +37 -0
  775. package/src/modules/admin/admin.routes.ts +2385 -0
  776. package/src/modules/admin/elevenlabs-test-chat.routes.ts +193 -0
  777. package/src/modules/admin/whatsapp-test-chat.routes.ts +244 -0
  778. package/src/modules/agents/agent.model.ts +93 -0
  779. package/src/modules/agents/agent.routes.ts +65 -0
  780. package/src/modules/appointment-validation/appointment-validation.model.ts +163 -0
  781. package/src/modules/appointment-validation/appointment-validation.routes.ts +275 -0
  782. package/src/modules/appointment-validation/appointment-validation.service.ts +1028 -0
  783. package/src/modules/auth/auth.model.ts +42 -0
  784. package/src/modules/auth/auth.routes.ts +199 -0
  785. package/src/modules/auth/auth.service.ts +210 -0
  786. package/src/modules/auth/refresh-token.model.ts +26 -0
  787. package/src/modules/billing/billing-alert.model.ts +28 -0
  788. package/src/modules/billing/billing-period-snapshot.model.ts +68 -0
  789. package/src/modules/billing/billing.model.ts +42 -0
  790. package/src/modules/billing/billing.routes.ts +67 -0
  791. package/src/modules/billing/billing.service.ts +562 -0
  792. package/src/modules/billing/payment.model.ts +42 -0
  793. package/src/modules/calls/call.model.ts +102 -0
  794. package/src/modules/calls/call.routes.ts +118 -0
  795. package/src/modules/campaigns/campaign.model.ts +111 -0
  796. package/src/modules/campaigns/campaign.routes.ts +402 -0
  797. package/src/modules/campaigns/campaign.service.ts +99 -0
  798. package/src/modules/google-calendar/google-calendar.routes.ts +31 -0
  799. package/src/modules/inbound-call/inbound-call-config.model.ts +49 -0
  800. package/src/modules/inbound-call/inbound-call.routes.ts +289 -0
  801. package/src/modules/leads/lead.model.ts +40 -0
  802. package/src/modules/leads/lead.routes.ts +246 -0
  803. package/src/modules/logs/log.model.ts +27 -0
  804. package/src/modules/logs/log.routes.ts +102 -0
  805. package/src/modules/plans/plan.model.ts +45 -0
  806. package/src/modules/surveys/survey.model.ts +70 -0
  807. package/src/modules/surveys/survey.routes.ts +187 -0
  808. package/src/modules/tenants/tenant.model.ts +181 -0
  809. package/src/modules/tenants/tenant.routes.ts +78 -0
  810. package/src/modules/users/user.routes.ts +126 -0
  811. package/src/modules/webhooks/elevenlabs-tool.routes.ts +94 -0
  812. package/src/modules/webhooks/elevenlabs-tool.service.ts +491 -0
  813. package/src/modules/webhooks/elevenlabs.routes.ts +34 -0
  814. package/src/modules/webhooks/elevenlabs.service.ts +565 -0
  815. package/src/modules/webhooks/unipile.routes.ts +917 -0
  816. package/src/modules/whatsapp/appointment.model.ts +47 -0
  817. package/src/modules/whatsapp/operator-request.model.ts +58 -0
  818. package/src/modules/whatsapp/stt-usage.model.ts +30 -0
  819. package/src/modules/whatsapp/whatsapp-chat.model.ts +39 -0
  820. package/src/modules/whatsapp/whatsapp-contact-profile.model.ts +41 -0
  821. package/src/modules/whatsapp/whatsapp-message.model.ts +41 -0
  822. package/src/modules/whatsapp/whatsapp-session.model.ts +60 -0
  823. package/src/modules/whatsapp/whatsapp.routes.ts +1435 -0
  824. package/src/plugins/cors.ts +7 -0
  825. package/src/plugins/jwt.ts +18 -0
  826. package/src/plugins/rawBody.ts +12 -0
  827. package/src/plugins/rbac.ts +24 -0
  828. package/src/routes/admin.routes.ts +208 -0
  829. package/src/routes/health.routes.ts +12 -0
  830. package/src/routes/webhook.routes.ts +12 -0
  831. package/src/services/ai/base.ai.ts +132 -0
  832. package/src/services/ai/gemini.service.ts +41 -0
  833. package/src/services/ai/index.ts +24 -0
  834. package/src/services/ai/openai.service.ts +48 -0
  835. package/src/services/elevenlabs.service.ts +532 -0
  836. package/src/services/google-calendar.service.ts +656 -0
  837. package/src/services/inbound-call-schedule.service.ts +174 -0
  838. package/src/services/netgsm.service.ts +128 -0
  839. package/src/services/unipile.service.ts +200 -0
  840. package/src/services/whatsapp-agent.service.ts +2479 -0
  841. package/src/services/whatsapp.service.ts +245 -0
  842. package/src/templates/index.ts +71 -0
  843. package/src/templates/receptionist.ts +44 -0
  844. package/src/templates/survey.ts +44 -0
  845. package/src/types/index.ts +218 -0
  846. package/src/utils/logger.ts +83 -0
  847. package/tsconfig.json +19 -0
  848. package/web/.dockerignore +4 -0
  849. package/web/Dockerfile +18 -0
  850. package/web/README.md +73 -0
  851. package/web/components.json +23 -0
  852. package/web/eslint.config.js +23 -0
  853. package/web/index.html +14 -0
  854. package/web/nginx.conf +18 -0
  855. package/web/package-lock.json +10292 -0
  856. package/web/package.json +48 -0
  857. package/web/public/favicon.ico +0 -0
  858. package/web/public/vite.svg +1 -0
  859. package/web/src/App.tsx +191 -0
  860. package/web/src/assets/react.svg +1 -0
  861. package/web/src/components/Layout.tsx +261 -0
  862. package/web/src/components/LeadConversation.tsx +251 -0
  863. package/web/src/components/ProtectedRoute.tsx +20 -0
  864. package/web/src/components/conversation-review/CardAudioPlayer.tsx +200 -0
  865. package/web/src/components/conversation-review/CardEvaluation.tsx +351 -0
  866. package/web/src/components/conversation-review/CardMetadata.tsx +44 -0
  867. package/web/src/components/conversation-review/ConversationCard.tsx +95 -0
  868. package/web/src/components/conversation-review/ReviewContainer.tsx +120 -0
  869. package/web/src/components/conversation-review/ReviewFilters.tsx +88 -0
  870. package/web/src/components/empty-state.tsx +15 -0
  871. package/web/src/components/page-header.tsx +19 -0
  872. package/web/src/components/page-loader.tsx +32 -0
  873. package/web/src/components/pagination.tsx +38 -0
  874. package/web/src/components/stat-card.tsx +27 -0
  875. package/web/src/components/status-badge.tsx +125 -0
  876. package/web/src/components/ui/alert.tsx +66 -0
  877. package/web/src/components/ui/badge.tsx +48 -0
  878. package/web/src/components/ui/button.tsx +64 -0
  879. package/web/src/components/ui/card.tsx +92 -0
  880. package/web/src/components/ui/checkbox.tsx +32 -0
  881. package/web/src/components/ui/dialog.tsx +158 -0
  882. package/web/src/components/ui/dropdown-menu.tsx +255 -0
  883. package/web/src/components/ui/input.tsx +21 -0
  884. package/web/src/components/ui/label.tsx +22 -0
  885. package/web/src/components/ui/progress.tsx +29 -0
  886. package/web/src/components/ui/select.tsx +188 -0
  887. package/web/src/components/ui/separator.tsx +28 -0
  888. package/web/src/components/ui/sheet.tsx +123 -0
  889. package/web/src/components/ui/skeleton.tsx +13 -0
  890. package/web/src/components/ui/sonner.tsx +35 -0
  891. package/web/src/components/ui/table.tsx +116 -0
  892. package/web/src/components/ui/tabs.tsx +89 -0
  893. package/web/src/components/ui/textarea.tsx +18 -0
  894. package/web/src/components/ui/tooltip.tsx +57 -0
  895. package/web/src/components/whatsapp/ChatDetail.tsx +417 -0
  896. package/web/src/components/whatsapp/ChatHeader.tsx +78 -0
  897. package/web/src/components/whatsapp/ChatList.tsx +107 -0
  898. package/web/src/components/whatsapp/ChatListItem.tsx +60 -0
  899. package/web/src/components/whatsapp/MessageBubble.tsx +46 -0
  900. package/web/src/components/whatsapp/MessageInput.tsx +63 -0
  901. package/web/src/components/whatsapp/MessageStream.tsx +135 -0
  902. package/web/src/components/whatsapp/SessionDivider.tsx +65 -0
  903. package/web/src/components/whatsapp/SessionHistory.tsx +119 -0
  904. package/web/src/components/whatsapp/settings/AiSettingsTab.tsx +268 -0
  905. package/web/src/components/whatsapp/settings/CalendarTab.tsx +339 -0
  906. package/web/src/components/whatsapp/settings/ConnectionTab.tsx +236 -0
  907. package/web/src/components/whatsapp/settings/NotificationsTab.tsx +109 -0
  908. package/web/src/components/whatsapp/settings/OperatorTab.tsx +303 -0
  909. package/web/src/contexts/AuthContext.tsx +103 -0
  910. package/web/src/index.css +130 -0
  911. package/web/src/lib/api.ts +92 -0
  912. package/web/src/lib/utils.ts +6 -0
  913. package/web/src/main.tsx +10 -0
  914. package/web/src/pages/AppointmentDetail.tsx +206 -0
  915. package/web/src/pages/AppointmentValidation.tsx +157 -0
  916. package/web/src/pages/AppointmentValidationDetail.tsx +617 -0
  917. package/web/src/pages/AppointmentValidationNew.tsx +1005 -0
  918. package/web/src/pages/Appointments.tsx +283 -0
  919. package/web/src/pages/Billing.tsx +126 -0
  920. package/web/src/pages/CallDetail.tsx +293 -0
  921. package/web/src/pages/CallSettings.tsx +313 -0
  922. package/web/src/pages/Calls.tsx +188 -0
  923. package/web/src/pages/CampaignDetail.tsx +216 -0
  924. package/web/src/pages/CampaignNew.tsx +277 -0
  925. package/web/src/pages/CampaignResults.tsx +185 -0
  926. package/web/src/pages/Campaigns.tsx +171 -0
  927. package/web/src/pages/Dashboard.tsx +336 -0
  928. package/web/src/pages/InboundSchedule.tsx +246 -0
  929. package/web/src/pages/LeadDetail.tsx +183 -0
  930. package/web/src/pages/Leads.tsx +258 -0
  931. package/web/src/pages/Login.tsx +99 -0
  932. package/web/src/pages/Register.tsx +129 -0
  933. package/web/src/pages/Settings.tsx +133 -0
  934. package/web/src/pages/SurveyDetail.tsx +232 -0
  935. package/web/src/pages/SurveyPreview.tsx +179 -0
  936. package/web/src/pages/Surveys.tsx +207 -0
  937. package/web/src/pages/Users.tsx +199 -0
  938. package/web/src/pages/WhatsApp.tsx +147 -0
  939. package/web/src/pages/WhatsAppSettings.tsx +215 -0
  940. package/web/src/pages/admin/AdminBaileysMessageLog.tsx +331 -0
  941. package/web/src/pages/admin/AdminBaileysRawMessages.tsx +318 -0
  942. package/web/src/pages/admin/AdminConversationReview.tsx +116 -0
  943. package/web/src/pages/admin/AdminCredits.tsx +467 -0
  944. package/web/src/pages/admin/AdminElevenLabsAgentDetail.tsx +332 -0
  945. package/web/src/pages/admin/AdminElevenLabsAgents.tsx +164 -0
  946. package/web/src/pages/admin/AdminElevenLabsBatchCallDetail.tsx +214 -0
  947. package/web/src/pages/admin/AdminElevenLabsBatchCalls.tsx +293 -0
  948. package/web/src/pages/admin/AdminElevenLabsConversationDetail.tsx +230 -0
  949. package/web/src/pages/admin/AdminElevenLabsConversations.tsx +228 -0
  950. package/web/src/pages/admin/AdminElevenLabsInboundCalls.tsx +293 -0
  951. package/web/src/pages/admin/AdminElevenLabsPhoneNumbers.tsx +506 -0
  952. package/web/src/pages/admin/AdminElevenLabsTestChat.tsx +258 -0
  953. package/web/src/pages/admin/AdminElevenLabsWebhooks.tsx +289 -0
  954. package/web/src/pages/admin/AdminEvaluationDashboard.tsx +520 -0
  955. package/web/src/pages/admin/AdminLeads.tsx +339 -0
  956. package/web/src/pages/admin/AdminLogs.tsx +283 -0
  957. package/web/src/pages/admin/AdminMigrations.tsx +247 -0
  958. package/web/src/pages/admin/AdminPlans.tsx +313 -0
  959. package/web/src/pages/admin/AdminSystem.tsx +391 -0
  960. package/web/src/pages/admin/AdminTenantBillableItems.tsx +464 -0
  961. package/web/src/pages/admin/AdminTenantDetail.tsx +1317 -0
  962. package/web/src/pages/admin/AdminTenants.tsx +274 -0
  963. package/web/src/pages/admin/AdminWhatsApp.tsx +618 -0
  964. package/web/src/pages/admin/AdminWhatsAppTestChat.tsx +328 -0
  965. package/web/src/types/index.ts +242 -0
  966. package/web/tsconfig.app.json +32 -0
  967. package/web/tsconfig.json +13 -0
  968. package/web/tsconfig.node.json +26 -0
  969. package/web/vite.config.ts +23 -0
@@ -0,0 +1,827 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.normalizePhone = normalizePhone;
7
+ exports.recalcStats = recalcStats;
8
+ exports.createBatch = createBatch;
9
+ exports.submitBatch = submitBatch;
10
+ exports.retryEntry = retryEntry;
11
+ exports.cancelBatch = cancelBatch;
12
+ exports.retryBatch = retryBatch;
13
+ exports.recordValidationResult = recordValidationResult;
14
+ exports.syncBatchConversations = syncBatchConversations;
15
+ exports.resumeInterruptedBatches = resumeInterruptedBatches;
16
+ exports.syncActiveBatches = syncActiveBatches;
17
+ exports.exportCSV = exportCSV;
18
+ const mongoose_1 = __importDefault(require("mongoose"));
19
+ const appointment_validation_model_1 = __importDefault(require("./appointment-validation.model"));
20
+ const agent_model_1 = __importDefault(require("../agents/agent.model"));
21
+ const tenant_model_1 = __importDefault(require("../tenants/tenant.model"));
22
+ const elevenlabs_service_1 = __importDefault(require("../../services/elevenlabs.service"));
23
+ const logger_1 = __importDefault(require("../../utils/logger"));
24
+ // ─── Phone Normalization ────────────────────────────────────
25
+ function normalizePhone(raw) {
26
+ const cleaned = raw.replace(/[^+\d]/g, '');
27
+ if (cleaned.length < 10)
28
+ return '';
29
+ return cleaned.startsWith('+') ? cleaned : `+${cleaned}`;
30
+ }
31
+ // ─── Stats ──────────────────────────────────────────────────
32
+ function recalcStats(batch) {
33
+ const entries = batch.entries;
34
+ batch.stats = {
35
+ total: entries.length,
36
+ confirmed: entries.filter(e => e.validation_result === 'confirmed').length,
37
+ cancelled: entries.filter(e => e.validation_result === 'cancelled').length,
38
+ reschedule: entries.filter(e => e.validation_result === 'reschedule').length,
39
+ unresolved: entries.filter(e => e.validation_result === 'unresolved' || (e.call_status === 'completed' && !e.validation_result)).length,
40
+ no_answer: entries.filter(e => e.call_status === 'no_answer').length,
41
+ voicemail: entries.filter(e => e.call_status === 'voicemail').length,
42
+ pending: entries.filter(e => e.call_status === 'pending' || e.call_status === 'calling').length,
43
+ failed: entries.filter(e => e.call_status === 'failed').length,
44
+ };
45
+ }
46
+ async function createBatch(tenantId, userId, data) {
47
+ const entries = data.entries.map(e => ({
48
+ _id: new mongoose_1.default.Types.ObjectId(),
49
+ row_index: e.row_index,
50
+ external_id: e.external_id || '',
51
+ patient_name: e.patient_name,
52
+ patient_phone: normalizePhone(e.patient_phone),
53
+ appointment_date: e.appointment_date,
54
+ appointment_time: e.appointment_time,
55
+ doctor_name: e.doctor_name || '',
56
+ department: e.department || '',
57
+ notes: e.notes || '',
58
+ raw_data: e.raw_data || {},
59
+ call_status: 'pending',
60
+ conversation_id: null,
61
+ validation_result: null,
62
+ reschedule_request: null,
63
+ patient_message: null,
64
+ call_duration_secs: 0,
65
+ called_at: null,
66
+ result_at: null,
67
+ }));
68
+ const batch = await appointment_validation_model_1.default.create({
69
+ tenant_id: new mongoose_1.default.Types.ObjectId(tenantId),
70
+ created_by: new mongoose_1.default.Types.ObjectId(userId),
71
+ name: data.name,
72
+ status: 'draft',
73
+ agent_id: null,
74
+ phone_number_id: null,
75
+ column_mapping: data.column_mapping,
76
+ call_schedule: {
77
+ call_date: data.call_schedule.call_date,
78
+ call_start_time: data.call_schedule.call_start_time || '09:00',
79
+ call_end_time: data.call_schedule.call_end_time || '18:00',
80
+ max_concurrent: data.call_schedule.max_concurrent || 3,
81
+ max_retries: data.call_schedule.max_retries || 2,
82
+ retry_delay_minutes: data.call_schedule.retry_delay_minutes || 1, // TODO: change to 30 for production
83
+ },
84
+ entries,
85
+ stats: {
86
+ total: entries.length,
87
+ confirmed: 0,
88
+ cancelled: 0,
89
+ reschedule: 0,
90
+ unresolved: 0,
91
+ no_answer: 0,
92
+ voicemail: 0,
93
+ pending: entries.length,
94
+ failed: 0,
95
+ },
96
+ });
97
+ return batch;
98
+ }
99
+ // ─── Submit ─────────────────────────────────────────────────
100
+ async function submitBatch(batchId, tenantId, options) {
101
+ const batch = await appointment_validation_model_1.default.findOne({ _id: batchId, tenant_id: tenantId });
102
+ if (!batch)
103
+ throw Object.assign(new Error('Batch not found'), { statusCode: 404 });
104
+ if (batch.status !== 'draft') {
105
+ throw Object.assign(new Error('Batch must be in draft status to submit'), { statusCode: 400 });
106
+ }
107
+ if (batch.entries.length === 0) {
108
+ throw Object.assign(new Error('Batch has no entries'), { statusCode: 400 });
109
+ }
110
+ // Resolve agent + phone from tenant settings
111
+ const tenant = await tenant_model_1.default.findById(tenantId).lean();
112
+ if (!tenant)
113
+ throw Object.assign(new Error('Tenant not found'), { statusCode: 404 });
114
+ const agentId = tenant.settings?.whatsapp_agent?.appointment_validation_agent_id;
115
+ const phoneNumberId = tenant.settings?.whatsapp_agent?.appointment_validation_phone_number_id;
116
+ if (!agentId) {
117
+ throw Object.assign(new Error('Randevu onay agenti tenant ayarlarinda yapilandirilmamis'), { statusCode: 400 });
118
+ }
119
+ if (!phoneNumberId) {
120
+ throw Object.assign(new Error('Randevu onay telefon numarasi tenant ayarlarinda yapilandirilmamis'), { statusCode: 400 });
121
+ }
122
+ const agent = await agent_model_1.default.findById(agentId).lean();
123
+ if (!agent || !agent.is_active) {
124
+ throw Object.assign(new Error('Yapilandirilan agent bulunamadi veya aktif degil'), { statusCode: 400 });
125
+ }
126
+ // Store resolved values on the batch
127
+ batch.agent_id = agent._id;
128
+ batch.phone_number_id = phoneNumberId;
129
+ batch.status = 'in_progress';
130
+ await batch.save();
131
+ // Fire individual SIP trunk calls (non-blocking)
132
+ dispatchCalls(batch, agent.elevenlabs_agent_id, phoneNumberId, tenant.name, options?.force).catch(err => logger_1.default.error('Batch dispatch error', { batchId: batch._id, error: err.message }));
133
+ logger_1.default.info('Appointment validation batch submitted', {
134
+ batchId: batch._id,
135
+ entryCount: batch.entries.length,
136
+ });
137
+ return batch;
138
+ }
139
+ // ─── Time Window Helpers ─────────────────────────────────────
140
+ function isWithinCallWindow(schedule) {
141
+ const now = new Date();
142
+ const today = now.toISOString().split('T')[0]; // YYYY-MM-DD
143
+ // Check date
144
+ if (schedule.call_date !== today)
145
+ return false;
146
+ // Check time window
147
+ const hhmm = `${String(now.getHours()).padStart(2, '0')}:${String(now.getMinutes()).padStart(2, '0')}`;
148
+ return hhmm >= schedule.call_start_time && hhmm < schedule.call_end_time;
149
+ }
150
+ function msUntilCallWindow(schedule) {
151
+ const now = new Date();
152
+ const today = now.toISOString().split('T')[0];
153
+ if (schedule.call_date > today) {
154
+ // Future date — calculate ms until start
155
+ const startDt = new Date(`${schedule.call_date}T${schedule.call_start_time}:00`);
156
+ return startDt.getTime() - now.getTime();
157
+ }
158
+ if (schedule.call_date === today) {
159
+ const hhmm = `${String(now.getHours()).padStart(2, '0')}:${String(now.getMinutes()).padStart(2, '0')}`;
160
+ if (hhmm < schedule.call_start_time) {
161
+ const startDt = new Date(`${today}T${schedule.call_start_time}:00`);
162
+ return startDt.getTime() - now.getTime();
163
+ }
164
+ }
165
+ return -1; // Past
166
+ }
167
+ function isAppointmentPassed(entry) {
168
+ const now = new Date();
169
+ const apptDate = entry.appointment_date; // YYYY-MM-DD
170
+ const apptTime = entry.appointment_time; // HH:MM
171
+ if (!apptDate)
172
+ return false;
173
+ const dt = new Date(`${apptDate}T${apptTime || '23:59'}:00`);
174
+ return !isNaN(dt.getTime()) && dt.getTime() <= now.getTime();
175
+ }
176
+ // ─── Concurrent Call Dispatcher ──────────────────────────────
177
+ async function dispatchCalls(batch, elevenlabsAgentId, phoneNumberId, clinicName, force) {
178
+ const schedule = batch.call_schedule;
179
+ const maxConcurrent = schedule.max_concurrent || 3;
180
+ const maxRetries = schedule.max_retries || 2;
181
+ const retryDelayMs = (schedule.retry_delay_minutes || 30) * 60 * 1000;
182
+ if (!force) {
183
+ // Wait until call window opens (if not yet)
184
+ const waitMs = msUntilCallWindow(schedule);
185
+ if (waitMs > 0) {
186
+ logger_1.default.info('Waiting for call window to open', {
187
+ batchId: batch._id, waitMs, callDate: schedule.call_date, startTime: schedule.call_start_time,
188
+ });
189
+ await new Promise(resolve => setTimeout(resolve, Math.min(waitMs, 12 * 60 * 60 * 1000)));
190
+ }
191
+ else if (waitMs < 0 && schedule.call_date < new Date().toISOString().split('T')[0]) {
192
+ logger_1.default.warn('Call date is in the past, proceeding anyway', { batchId: batch._id, callDate: schedule.call_date });
193
+ }
194
+ }
195
+ else {
196
+ logger_1.default.info('Force mode — skipping time window checks', { batchId: batch._id });
197
+ }
198
+ let activeCount = 0;
199
+ async function callEntry(entry) {
200
+ const entryId = entry._id;
201
+ try {
202
+ // Atomic update: set calling status + increment attempts
203
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries._id': entryId }, { $set: { 'entries.$.call_status': 'calling' }, $inc: { 'entries.$.attempts': 1 } });
204
+ const result = await elevenlabs_service_1.default.sipTrunkOutboundCall({
205
+ agentId: elevenlabsAgentId,
206
+ agentPhoneNumberId: phoneNumberId,
207
+ toNumber: entry.patient_phone,
208
+ conversationInitiationClientData: {
209
+ dynamicVariables: {
210
+ clinic_name: clinicName,
211
+ patient_name: entry.patient_name,
212
+ patient_phone: entry.patient_phone,
213
+ appointment_date: entry.appointment_date,
214
+ appointment_time: entry.appointment_time,
215
+ doctor_name: entry.doctor_name || '',
216
+ department: entry.department || '',
217
+ notes: entry.notes || '',
218
+ current_date: new Date().toISOString().split('T')[0],
219
+ row_id: entryId.toString(),
220
+ },
221
+ },
222
+ });
223
+ // Atomic update: set conversation_id + called_at
224
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries._id': entryId }, { $set: {
225
+ 'entries.$.conversation_id': result.conversationId || null,
226
+ 'entries.$.called_at': new Date(),
227
+ } });
228
+ logger_1.default.info('Appointment validation call initiated', {
229
+ batchId: batch._id,
230
+ phone: entry.patient_phone,
231
+ conversationId: result.conversationId,
232
+ });
233
+ }
234
+ catch (err) {
235
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries._id': entryId }, { $set: { 'entries.$.call_status': 'failed' } });
236
+ logger_1.default.error('Failed to initiate appointment validation call', {
237
+ batchId: batch._id,
238
+ phone: entry.patient_phone,
239
+ error: err.message,
240
+ });
241
+ }
242
+ }
243
+ // Process pending entries with concurrency limit
244
+ async function processQueue() {
245
+ // Re-read from DB to get fresh state
246
+ const fresh = await appointment_validation_model_1.default.findById(batch._id);
247
+ if (!fresh || fresh.status === 'cancelled')
248
+ return;
249
+ batch.entries = fresh.entries;
250
+ const pendingEntries = batch.entries.filter(e => e.call_status === 'pending');
251
+ // Skip entries whose appointment has already passed (atomic)
252
+ for (const entry of pendingEntries) {
253
+ if (isAppointmentPassed(entry)) {
254
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries._id': entry._id }, { $set: {
255
+ 'entries.$.call_status': 'failed',
256
+ 'entries.$.validation_result': 'unresolved',
257
+ 'entries.$.result_at': new Date(),
258
+ } });
259
+ logger_1.default.info('Skipped entry — appointment already passed', {
260
+ batchId: batch._id, phone: entry.patient_phone,
261
+ appointmentDate: entry.appointment_date, appointmentTime: entry.appointment_time,
262
+ });
263
+ }
264
+ }
265
+ // Re-read after marking passed entries
266
+ const updated = await appointment_validation_model_1.default.findById(batch._id);
267
+ if (!updated || updated.status === 'cancelled')
268
+ return;
269
+ batch.entries = updated.entries;
270
+ const callableEntries = batch.entries
271
+ .filter(e => e.call_status === 'pending')
272
+ .sort((a, b) => {
273
+ const dtA = `${a.appointment_date}T${a.appointment_time || '00:00'}`;
274
+ const dtB = `${b.appointment_date}T${b.appointment_time || '00:00'}`;
275
+ return dtA < dtB ? -1 : dtA > dtB ? 1 : 0;
276
+ });
277
+ const promises = [];
278
+ for (const entry of callableEntries) {
279
+ // Check time window before each call (skip in force mode)
280
+ if (!force && !isWithinCallWindow(schedule)) {
281
+ logger_1.default.info('Call window closed, pausing dispatch', {
282
+ batchId: batch._id,
283
+ endTime: schedule.call_end_time,
284
+ remaining: pendingEntries.length - promises.length,
285
+ });
286
+ break;
287
+ }
288
+ // Wait for a slot to open
289
+ while (activeCount >= maxConcurrent) {
290
+ await new Promise(resolve => setTimeout(resolve, 1000));
291
+ }
292
+ activeCount++;
293
+ const p = callEntry(entry).finally(() => { activeCount--; });
294
+ promises.push(p);
295
+ // Small stagger between launching calls
296
+ await new Promise(resolve => setTimeout(resolve, 1500));
297
+ }
298
+ await Promise.all(promises);
299
+ }
300
+ // First pass
301
+ await processQueue();
302
+ if (maxRetries <= 0) {
303
+ await waitAndComplete(batch._id);
304
+ return;
305
+ }
306
+ // Wait for all active calls to finish before checking retries
307
+ await waitForCallsToSettle(batch._id);
308
+ // Retry loop
309
+ for (let retry = 0; retry < maxRetries; retry++) {
310
+ const current = await appointment_validation_model_1.default.findById(batch._id);
311
+ if (!current || current.status === 'cancelled')
312
+ return;
313
+ const retriable = current.entries.filter(e => (e.call_status === 'failed' || e.call_status === 'no_answer' || e.call_status === 'voicemail' || e.validation_result === 'unresolved')
314
+ && (e.attempts || 0) <= retry + 1 // only retry entries from previous round
315
+ && !isAppointmentPassed(e));
316
+ if (retriable.length === 0)
317
+ break;
318
+ if (!force && !isWithinCallWindow(schedule)) {
319
+ logger_1.default.info('Call window closed, skipping retries', { batchId: batch._id });
320
+ break;
321
+ }
322
+ logger_1.default.info('Waiting for retry delay', {
323
+ batchId: current._id,
324
+ retryRound: retry + 1,
325
+ retriable: retriable.length,
326
+ retryDelayMinutes: schedule.retry_delay_minutes,
327
+ });
328
+ await new Promise(resolve => setTimeout(resolve, retryDelayMs));
329
+ if (!force && !isWithinCallWindow(schedule))
330
+ break;
331
+ // Reset retriable entries atomically
332
+ for (const entry of retriable) {
333
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries._id': entry._id }, { $set: {
334
+ 'entries.$.call_status': 'pending',
335
+ 'entries.$.conversation_id': null,
336
+ 'entries.$.validation_result': null,
337
+ 'entries.$.result_at': null,
338
+ } });
339
+ }
340
+ // Re-read batch for processQueue
341
+ const refreshed = await appointment_validation_model_1.default.findById(batch._id);
342
+ if (!refreshed || refreshed.status === 'cancelled')
343
+ return;
344
+ // Reassign batch ref for processQueue/callEntry closures
345
+ batch.entries = refreshed.entries;
346
+ await processQueue();
347
+ await waitForCallsToSettle(batch._id);
348
+ }
349
+ await waitAndComplete(batch._id);
350
+ }
351
+ async function waitForCallsToSettle(batchId) {
352
+ // Poll until no entries are in 'calling' status (max 10 min)
353
+ const maxWait = 10 * 60 * 1000;
354
+ const start = Date.now();
355
+ while (Date.now() - start < maxWait) {
356
+ const b = await appointment_validation_model_1.default.findById(batchId).lean();
357
+ if (!b)
358
+ return;
359
+ const hasCalling = b.entries.some(e => e.call_status === 'calling');
360
+ if (!hasCalling)
361
+ return;
362
+ await new Promise(resolve => setTimeout(resolve, 5000));
363
+ }
364
+ // Timeout — mark stuck calling entries as failed so they can be retried
365
+ logger_1.default.warn('waitForCallsToSettle timed out, marking stuck entries as failed', { batchId });
366
+ const b = await appointment_validation_model_1.default.findById(batchId);
367
+ if (!b)
368
+ return;
369
+ for (const entry of b.entries) {
370
+ if (entry.call_status === 'calling') {
371
+ entry.call_status = 'failed';
372
+ }
373
+ }
374
+ recalcStats(b);
375
+ await b.save();
376
+ }
377
+ async function waitAndComplete(batchId) {
378
+ await waitForCallsToSettle(batchId);
379
+ const final = await appointment_validation_model_1.default.findById(batchId);
380
+ if (final && final.status === 'in_progress') {
381
+ final.status = 'completed';
382
+ recalcStats(final);
383
+ await final.save();
384
+ logger_1.default.info('Appointment validation batch completed', { batchId: final._id });
385
+ }
386
+ }
387
+ // ─── Retry Single Entry ─────────────────────────────────────
388
+ async function retryEntry(batchId, entryId, tenantId) {
389
+ const batch = await appointment_validation_model_1.default.findOne({ _id: batchId, tenant_id: tenantId });
390
+ if (!batch)
391
+ throw Object.assign(new Error('Batch not found'), { statusCode: 404 });
392
+ const entry = batch.entries.find(e => e._id.toString() === entryId);
393
+ if (!entry)
394
+ throw Object.assign(new Error('Entry not found'), { statusCode: 404 });
395
+ const tenant = await tenant_model_1.default.findById(tenantId).lean();
396
+ if (!tenant)
397
+ throw Object.assign(new Error('Tenant not found'), { statusCode: 404 });
398
+ const agentObjId = tenant.settings?.whatsapp_agent?.appointment_validation_agent_id;
399
+ const phoneNumberId = tenant.settings?.whatsapp_agent?.appointment_validation_phone_number_id;
400
+ if (!agentObjId || !phoneNumberId) {
401
+ throw Object.assign(new Error('Agent veya telefon numarasi yapilandirilmamis'), { statusCode: 400 });
402
+ }
403
+ const agent = await agent_model_1.default.findById(agentObjId).lean();
404
+ if (!agent || !agent.is_active) {
405
+ throw Object.assign(new Error('Agent bulunamadi veya aktif degil'), { statusCode: 400 });
406
+ }
407
+ if (batch.status === 'completed' || batch.status === 'cancelled') {
408
+ batch.status = 'in_progress';
409
+ await batch.save();
410
+ }
411
+ // Fire a single call directly (no dispatch loop / retries)
412
+ (async () => {
413
+ try {
414
+ // Reset entry atomically
415
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries._id': entry._id }, { $set: {
416
+ 'entries.$.call_status': 'calling',
417
+ 'entries.$.conversation_id': null,
418
+ 'entries.$.validation_result': null,
419
+ 'entries.$.reschedule_request': null,
420
+ 'entries.$.patient_message': null,
421
+ 'entries.$.call_duration_secs': 0,
422
+ 'entries.$.called_at': null,
423
+ 'entries.$.result_at': null,
424
+ }, $inc: { 'entries.$.attempts': 1 } });
425
+ const result = await elevenlabs_service_1.default.sipTrunkOutboundCall({
426
+ agentId: agent.elevenlabs_agent_id,
427
+ agentPhoneNumberId: phoneNumberId,
428
+ toNumber: entry.patient_phone,
429
+ conversationInitiationClientData: {
430
+ dynamicVariables: {
431
+ clinic_name: tenant.name,
432
+ patient_name: entry.patient_name,
433
+ patient_phone: entry.patient_phone,
434
+ appointment_date: entry.appointment_date,
435
+ appointment_time: entry.appointment_time,
436
+ doctor_name: entry.doctor_name || '',
437
+ department: entry.department || '',
438
+ notes: entry.notes || '',
439
+ current_date: new Date().toISOString().split('T')[0],
440
+ row_id: entry._id.toString(),
441
+ },
442
+ },
443
+ });
444
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries._id': entry._id }, { $set: {
445
+ 'entries.$.conversation_id': result.conversationId || null,
446
+ 'entries.$.called_at': new Date(),
447
+ } });
448
+ logger_1.default.info('Single entry retry call initiated', {
449
+ batchId: batch._id, entryId, conversationId: result.conversationId,
450
+ });
451
+ }
452
+ catch (err) {
453
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries._id': entry._id }, { $set: { 'entries.$.call_status': 'failed' } });
454
+ logger_1.default.error('Single entry retry call failed', {
455
+ batchId: batch._id, entryId, error: err.message,
456
+ });
457
+ }
458
+ // Recalc stats
459
+ const refreshed = await appointment_validation_model_1.default.findById(batch._id);
460
+ if (refreshed) {
461
+ recalcStats(refreshed);
462
+ await refreshed.save();
463
+ }
464
+ })().catch(err => logger_1.default.error('Single entry retry error', { batchId: batch._id, entryId, error: err.message }));
465
+ // Return current batch state (call fires in background)
466
+ recalcStats(batch);
467
+ return batch;
468
+ }
469
+ // ─── Cancel ─────────────────────────────────────────────────
470
+ async function cancelBatch(batchId, tenantId) {
471
+ const batch = await appointment_validation_model_1.default.findOne({ _id: batchId, tenant_id: tenantId });
472
+ if (!batch)
473
+ throw Object.assign(new Error('Batch not found'), { statusCode: 404 });
474
+ if (batch.status === 'completed' || batch.status === 'cancelled') {
475
+ throw Object.assign(new Error('Batch is already finished'), { statusCode: 400 });
476
+ }
477
+ // Mark remaining pending/calling entries as failed
478
+ for (const entry of batch.entries) {
479
+ if (entry.call_status === 'pending' || entry.call_status === 'calling') {
480
+ entry.call_status = 'failed';
481
+ }
482
+ }
483
+ batch.status = 'cancelled';
484
+ recalcStats(batch);
485
+ await batch.save();
486
+ return batch;
487
+ }
488
+ // ─── Retry ──────────────────────────────────────────────────
489
+ async function retryBatch(batchId, tenantId, options) {
490
+ const batch = await appointment_validation_model_1.default.findOne({ _id: batchId, tenant_id: tenantId });
491
+ if (!batch)
492
+ throw Object.assign(new Error('Batch not found'), { statusCode: 404 });
493
+ const tenant = await tenant_model_1.default.findById(tenantId).lean();
494
+ if (!tenant)
495
+ throw Object.assign(new Error('Tenant not found'), { statusCode: 404 });
496
+ const agentId = tenant.settings?.whatsapp_agent?.appointment_validation_agent_id;
497
+ const phoneNumberId = tenant.settings?.whatsapp_agent?.appointment_validation_phone_number_id;
498
+ if (!agentId || !phoneNumberId) {
499
+ throw Object.assign(new Error('Agent veya telefon numarasi yapilandirilmamis'), { statusCode: 400 });
500
+ }
501
+ const agent = await agent_model_1.default.findById(agentId).lean();
502
+ if (!agent || !agent.is_active) {
503
+ throw Object.assign(new Error('Agent bulunamadi veya aktif degil'), { statusCode: 400 });
504
+ }
505
+ // Reset failed/no_answer/voicemail/unresolved entries to pending
506
+ for (const entry of batch.entries) {
507
+ if (entry.call_status === 'failed' || entry.call_status === 'no_answer' || entry.call_status === 'voicemail'
508
+ || entry.validation_result === 'unresolved') {
509
+ entry.call_status = 'pending';
510
+ entry.conversation_id = null;
511
+ entry.validation_result = null;
512
+ entry.reschedule_request = null;
513
+ entry.patient_message = null;
514
+ entry.result_at = null;
515
+ }
516
+ }
517
+ batch.status = 'in_progress';
518
+ recalcStats(batch);
519
+ await batch.save();
520
+ // Fire calls for reset entries
521
+ dispatchCalls(batch, agent.elevenlabs_agent_id, phoneNumberId, tenant.name, options?.force).catch(err => logger_1.default.error('Batch retry dispatch error', { batchId: batch._id, error: err.message }));
522
+ return batch;
523
+ }
524
+ // ─── Record Validation Result (called by webhook tool) ──────
525
+ async function recordValidationResult(conversationId, tenantId, result, rescheduleRequest, patientMessage) {
526
+ const batch = await appointment_validation_model_1.default.findOne({
527
+ tenant_id: tenantId,
528
+ 'entries.conversation_id': conversationId,
529
+ });
530
+ if (!batch) {
531
+ return { recorded: false };
532
+ }
533
+ const entry = batch.entries.find(e => e.conversation_id === conversationId);
534
+ if (!entry) {
535
+ return { recorded: false };
536
+ }
537
+ // Atomic update to avoid race conditions with concurrent dispatchCalls
538
+ await appointment_validation_model_1.default.updateOne({ _id: batch._id, 'entries.conversation_id': conversationId }, { $set: {
539
+ 'entries.$.validation_result': result,
540
+ 'entries.$.reschedule_request': rescheduleRequest || null,
541
+ 'entries.$.patient_message': patientMessage || null,
542
+ 'entries.$.result_at': new Date(),
543
+ 'entries.$.call_status': 'completed',
544
+ } });
545
+ // Recalc stats on fresh document
546
+ const refreshed = await appointment_validation_model_1.default.findById(batch._id);
547
+ if (refreshed) {
548
+ recalcStats(refreshed);
549
+ await refreshed.save();
550
+ }
551
+ logger_1.default.info('Appointment validation result recorded', {
552
+ batchId: batch._id,
553
+ conversationId,
554
+ result,
555
+ });
556
+ return { recorded: true, batchId: batch._id.toString() };
557
+ }
558
+ // ─── Sync Conversations from ElevenLabs API ─────────────────
559
+ async function syncBatchConversations(batch) {
560
+ const entriesWithConvId = batch.entries.filter(e => e.conversation_id && (e.call_status === 'calling' || e.call_status === 'completed' || !e.validation_result));
561
+ if (entriesWithConvId.length === 0) {
562
+ recalcStats(batch);
563
+ await batch.save();
564
+ return batch;
565
+ }
566
+ const EL_STATUS_MAP = {
567
+ done: 'completed',
568
+ failed: 'failed',
569
+ 'in-progress': 'calling',
570
+ initiated: 'calling',
571
+ processing: 'completed',
572
+ };
573
+ let synced = 0;
574
+ for (const entry of entriesWithConvId) {
575
+ try {
576
+ const conv = await elevenlabs_service_1.default.getConversation(entry.conversation_id);
577
+ const elStatus = (conv.status || '').toLowerCase();
578
+ const mappedStatus = EL_STATUS_MAP[elStatus];
579
+ if (mappedStatus && entry.call_status !== mappedStatus) {
580
+ entry.call_status = mappedStatus;
581
+ }
582
+ // Update duration from conversation metadata
583
+ const duration = conv.metadata?.duration;
584
+ if (typeof duration === 'number' && duration > 0) {
585
+ entry.call_duration_secs = Math.round(duration);
586
+ }
587
+ // If call is done but no validation_result, check if tool was called in transcript
588
+ if (elStatus === 'done' && !entry.validation_result) {
589
+ let foundResult = false;
590
+ const transcript = conv.transcript || [];
591
+ for (const turn of transcript) {
592
+ const toolCalls = turn.tool_calls || [];
593
+ for (const tc of toolCalls) {
594
+ if (tc.tool_name === 'confirm_appointment_validation' && tc.params) {
595
+ const result = tc.params.result;
596
+ if (['confirmed', 'cancelled', 'reschedule'].includes(result)) {
597
+ entry.validation_result = result;
598
+ entry.reschedule_request = tc.params.reschedule_request || null;
599
+ entry.patient_message = tc.params.patient_message || null;
600
+ entry.result_at = new Date();
601
+ foundResult = true;
602
+ }
603
+ }
604
+ }
605
+ }
606
+ // Call done but tool never called — mark as unresolved
607
+ if (!foundResult) {
608
+ entry.validation_result = 'unresolved';
609
+ entry.result_at = new Date();
610
+ }
611
+ }
612
+ synced++;
613
+ // Small delay to avoid rate limiting
614
+ if (synced % 5 === 0) {
615
+ await new Promise(resolve => setTimeout(resolve, 500));
616
+ }
617
+ }
618
+ catch (err) {
619
+ logger_1.default.warn('Failed to sync conversation', {
620
+ conversationId: entry.conversation_id,
621
+ error: err.message,
622
+ });
623
+ }
624
+ }
625
+ // Check if batch is complete
626
+ const allDone = batch.entries.every(e => e.call_status !== 'pending' && e.call_status !== 'calling');
627
+ if (allDone && batch.status === 'in_progress') {
628
+ batch.status = 'completed';
629
+ }
630
+ recalcStats(batch);
631
+ await batch.save();
632
+ logger_1.default.info('Synced batch conversations from ElevenLabs', {
633
+ batchId: batch._id,
634
+ synced,
635
+ total: entriesWithConvId.length,
636
+ });
637
+ return batch;
638
+ }
639
+ // ─── Sync Active Batches ────────────────────────────────────
640
+ // ─── Resume Interrupted Batches (on server startup) ──────────
641
+ async function resumeInterruptedBatches() {
642
+ const interrupted = await appointment_validation_model_1.default.find({
643
+ status: 'in_progress',
644
+ });
645
+ if (interrupted.length === 0)
646
+ return;
647
+ logger_1.default.info('Found interrupted appointment validation batches', { count: interrupted.length });
648
+ for (const batch of interrupted) {
649
+ try {
650
+ // Fix stale "calling" entries — server died mid-call, webhook never arrived
651
+ let fixed = 0;
652
+ for (const entry of batch.entries) {
653
+ if (entry.call_status === 'calling') {
654
+ // If we have a conversation_id, try to sync from ElevenLabs
655
+ if (entry.conversation_id) {
656
+ try {
657
+ const conv = await elevenlabs_service_1.default.getConversation(entry.conversation_id);
658
+ const elStatus = (conv.status || '').toLowerCase();
659
+ const termReason = (conv.metadata?.termination_reason || '').toLowerCase();
660
+ if (elStatus === 'done' || elStatus === 'failed') {
661
+ entry.call_status = termReason.includes('voicemail') ? 'voicemail'
662
+ : ['failed', 'error'].includes(elStatus) ? 'failed'
663
+ : 'completed';
664
+ if (entry.call_status === 'completed' && !entry.validation_result) {
665
+ entry.validation_result = 'unresolved';
666
+ entry.result_at = new Date();
667
+ }
668
+ }
669
+ else {
670
+ // Still in progress or unknown — mark failed so it can be retried
671
+ entry.call_status = 'failed';
672
+ }
673
+ }
674
+ catch {
675
+ // Can't reach ElevenLabs — mark failed
676
+ entry.call_status = 'failed';
677
+ }
678
+ }
679
+ else {
680
+ // No conversation_id — call was never actually placed, reset to pending
681
+ entry.call_status = 'pending';
682
+ }
683
+ fixed++;
684
+ }
685
+ }
686
+ if (fixed > 0) {
687
+ logger_1.default.info('Fixed stale calling entries after restart', {
688
+ batchId: batch._id, fixed,
689
+ });
690
+ }
691
+ // Check if there are still pending entries to dispatch
692
+ const hasPending = batch.entries.some(e => e.call_status === 'pending');
693
+ const hasRetriable = batch.entries.some(e => (e.call_status === 'failed' || e.call_status === 'no_answer' || e.call_status === 'voicemail')
694
+ && (e.attempts || 0) < (batch.call_schedule.max_retries || 2));
695
+ if (hasPending || hasRetriable) {
696
+ // Resolve tenant + agent to re-dispatch
697
+ const tenant = await tenant_model_1.default.findById(batch.tenant_id).lean();
698
+ const agentObjId = tenant?.settings?.whatsapp_agent?.appointment_validation_agent_id;
699
+ const phoneNumberId = tenant?.settings?.whatsapp_agent?.appointment_validation_phone_number_id;
700
+ if (tenant && agentObjId && phoneNumberId) {
701
+ const agent = await agent_model_1.default.findById(agentObjId).lean();
702
+ if (agent?.is_active) {
703
+ // Reset retriable entries to pending for re-dispatch
704
+ if (hasRetriable && !hasPending) {
705
+ for (const entry of batch.entries) {
706
+ if ((entry.call_status === 'failed' || entry.call_status === 'no_answer' || entry.call_status === 'voicemail')
707
+ && (entry.attempts || 0) < (batch.call_schedule.max_retries || 2)) {
708
+ entry.call_status = 'pending';
709
+ entry.conversation_id = null;
710
+ entry.validation_result = null;
711
+ }
712
+ }
713
+ }
714
+ recalcStats(batch);
715
+ await batch.save();
716
+ logger_1.default.info('Resuming interrupted batch', {
717
+ batchId: batch._id,
718
+ pending: batch.entries.filter(e => e.call_status === 'pending').length,
719
+ });
720
+ dispatchCalls(batch, agent.elevenlabs_agent_id, phoneNumberId, tenant.name).catch(err => logger_1.default.error('Failed to resume batch dispatch', { batchId: batch._id, error: err.message }));
721
+ continue; // skip the save below
722
+ }
723
+ }
724
+ }
725
+ // No pending/retriable — just complete it
726
+ const allDone = batch.entries.every(e => e.call_status !== 'pending' && e.call_status !== 'calling');
727
+ if (allDone) {
728
+ batch.status = 'completed';
729
+ }
730
+ recalcStats(batch);
731
+ await batch.save();
732
+ }
733
+ catch (err) {
734
+ logger_1.default.error('Failed to resume interrupted batch', {
735
+ batchId: batch._id,
736
+ error: err.message,
737
+ });
738
+ }
739
+ }
740
+ }
741
+ // ─── Sync Active Batches (periodic) ─────────────────────────
742
+ async function syncActiveBatches() {
743
+ const activeBatches = await appointment_validation_model_1.default.find({
744
+ status: 'in_progress',
745
+ });
746
+ const staleThreshold = Date.now() - 5 * 60 * 1000; // 5 minutes
747
+ for (const batch of activeBatches) {
748
+ try {
749
+ let changed = false;
750
+ // Mark stale "calling" entries as completed (webhook probably missed)
751
+ for (const entry of batch.entries) {
752
+ if (entry.call_status === 'calling' && entry.called_at && entry.called_at.getTime() < staleThreshold) {
753
+ entry.call_status = 'completed';
754
+ changed = true;
755
+ logger_1.default.warn('Marked stale calling entry as completed', {
756
+ batchId: batch._id,
757
+ entryId: entry._id,
758
+ phone: entry.patient_phone,
759
+ });
760
+ }
761
+ }
762
+ // Only auto-complete if no pending/calling entries AND batch has been
763
+ // in_progress for at least 5 minutes (avoids racing with dispatch retry loop)
764
+ const allDone = batch.entries.every(e => e.call_status !== 'pending' && e.call_status !== 'calling');
765
+ const batchAge = Date.now() - (batch.updatedAt?.getTime() || Date.now());
766
+ if (allDone && batchAge > 5 * 60 * 1000) {
767
+ batch.status = 'completed';
768
+ changed = true;
769
+ }
770
+ if (changed) {
771
+ recalcStats(batch);
772
+ }
773
+ await batch.save();
774
+ }
775
+ catch (err) {
776
+ logger_1.default.error('Failed to sync appointment validation batch', {
777
+ batchId: batch._id,
778
+ error: err.message,
779
+ });
780
+ }
781
+ }
782
+ }
783
+ // ─── CSV Export ─────────────────────────────────────────────
784
+ function escapeCSV(value) {
785
+ if (value.includes(',') || value.includes('"') || value.includes('\n')) {
786
+ return `"${value.replace(/"/g, '""')}"`;
787
+ }
788
+ return value;
789
+ }
790
+ function exportCSV(batch) {
791
+ const header = [
792
+ 'Harici ID',
793
+ 'Satir No',
794
+ 'Hasta Adi',
795
+ 'Telefon',
796
+ 'Randevu Tarihi',
797
+ 'Randevu Saati',
798
+ 'Doktor',
799
+ 'Bolum',
800
+ 'Arama Durumu',
801
+ 'Deneme',
802
+ 'Sonuc',
803
+ 'Erteleme Talebi',
804
+ 'Hasta Mesaji',
805
+ 'Arama Suresi (sn)',
806
+ 'Aranma Zamani',
807
+ ].join(',');
808
+ const rows = batch.entries.map(e => [
809
+ escapeCSV(e.external_id || ''),
810
+ String(e.row_index),
811
+ escapeCSV(e.patient_name),
812
+ e.patient_phone,
813
+ e.appointment_date,
814
+ e.appointment_time,
815
+ escapeCSV(e.doctor_name),
816
+ escapeCSV(e.department),
817
+ e.call_status,
818
+ String(e.attempts || 0),
819
+ e.validation_result || '',
820
+ escapeCSV(e.reschedule_request || ''),
821
+ escapeCSV(e.patient_message || ''),
822
+ String(e.call_duration_secs),
823
+ e.called_at ? e.called_at.toISOString() : '',
824
+ ].join(','));
825
+ return [header, ...rows].join('\n');
826
+ }
827
+ //# sourceMappingURL=appointment-validation.service.js.map