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,721 @@
1
+ /**
2
+ * Baileys Worker Process
3
+ *
4
+ * Runs a single Baileys WhatsApp socket in an isolated child process.
5
+ * Communicates with the gateway (server.js) via IPC (process.send / process.on).
6
+ *
7
+ * Auth state is stored in MongoDB (shared across all workers via MONGODB_URI).
8
+ * Falls back to local file auth if MONGODB_URI is not set.
9
+ *
10
+ * IPC Protocol — Worker → Parent:
11
+ * { type: 'qr', qr }
12
+ * { type: 'connected', phone }
13
+ * { type: 'disconnected', reason }
14
+ * { type: 'logged_out' }
15
+ * { type: 'status', status }
16
+ * { type: 'message', payload } — Unipile-format webhook payload
17
+ * { type: 'reply', reqId, data } — Response to a request
18
+ * { type: 'reply', reqId, error } — Error response
19
+ * { type: 'ready' } — Worker initialized
20
+ *
21
+ * IPC Protocol — Parent → Worker:
22
+ * { type: 'init', accountId, authDir, mongodbUri }
23
+ * { type: 'send', chatId, text, reqId }
24
+ * { type: 'start_chat', phone, text, reqId }
25
+ * { type: 'forward_media', messageId, toPhone, caption, reqId }
26
+ * { type: 'disconnect' }
27
+ */
28
+
29
+ import makeWASocket, {
30
+ useMultiFileAuthState,
31
+ DisconnectReason,
32
+ Browsers,
33
+ fetchLatestBaileysVersion,
34
+ makeCacheableSignalKeyStore,
35
+ isJidBroadcast,
36
+ isJidGroup,
37
+ isLidUser,
38
+ getContentType,
39
+ jidNormalizedUser,
40
+ downloadMediaMessage,
41
+ } from '@whiskeysockets/baileys';
42
+ import { MongoClient } from 'mongodb';
43
+ import pino from 'pino';
44
+ import { jidToPhone, phoneFromSockUser, extractText, MEDIA_PLACEHOLDER, MEDIA_TYPE_LABELS } from './src/utils/jid.js';
45
+ import { useMongoDBAuthState, deleteMongoDBAuthState } from './src/utils/use-mongodb-auth-state.js';
46
+
47
+ const workerLogger = pino({ level: 'warn' });
48
+
49
+ let sock = null;
50
+ let accountId = null;
51
+ let authDir = null;
52
+ let mongodbUri = null;
53
+ let mongoClient = null;
54
+ let mongoDb = null;
55
+ let connectedPhone = null;
56
+ let mediaForwardTo = null;
57
+
58
+ // ─── Media Message Cache ────────────────────────────────
59
+ // Caches recent media messages so we can re-download them for forwarding.
60
+ // downloadMediaMessage() requires the original Baileys message object (with encryption keys).
61
+ const messageCache = new Map(); // key: messageId, value: { msg, contentType, timestamp }
62
+ const MESSAGE_CACHE_TTL_MS = 60 * 60 * 1000; // 60 minutes
63
+
64
+ function cleanMessageCache() {
65
+ const now = Date.now();
66
+ for (const [key, entry] of messageCache) {
67
+ if (now - entry.timestamp > MESSAGE_CACHE_TTL_MS) {
68
+ messageCache.delete(key);
69
+ }
70
+ }
71
+ }
72
+
73
+ // Clean cache every 5 minutes
74
+ setInterval(cleanMessageCache, 5 * 60 * 1000);
75
+
76
+ function send(msg) {
77
+ if (process.send) process.send(msg);
78
+ }
79
+
80
+ /**
81
+ * Extract ad attribution from contextInfo for the webhook payload.
82
+ * Returns null if the message didn't come from an ad click.
83
+ */
84
+ function sanitizeAdContext(ctx) {
85
+ if (!ctx.externalAdReply && !ctx.entryPointConversionSource) return null;
86
+
87
+ const clean = {};
88
+
89
+ if (ctx.externalAdReply) {
90
+ const ad = ctx.externalAdReply;
91
+ clean.ad = {
92
+ title: ad.title || null,
93
+ body: ad.body || null,
94
+ source_url: ad.sourceUrl || null,
95
+ };
96
+ }
97
+
98
+ if (ctx.entryPointConversionSource) clean.source = ctx.entryPointConversionSource;
99
+ if (ctx.entryPointConversionApp) clean.app = ctx.entryPointConversionApp;
100
+
101
+ return clean;
102
+ }
103
+
104
+ /**
105
+ * Strip binary/heavy fields from contextInfo for logging.
106
+ * Keeps: externalAdReply, quotedMessage (text-only summary), mentionedJid,
107
+ * isForwarded, forwardingScore, entryPointConversion* fields.
108
+ */
109
+ function sanitizeContextInfo(ctx) {
110
+ const clean = {};
111
+
112
+ if (ctx.externalAdReply) {
113
+ const ad = ctx.externalAdReply;
114
+ clean.externalAdReply = {
115
+ title: ad.title || null,
116
+ body: ad.body || null,
117
+ mediaUrl: ad.mediaUrl || null,
118
+ sourceUrl: ad.sourceUrl || null,
119
+ mediaType: ad.mediaType ?? null,
120
+ showAdAttribution: ad.showAdAttribution ?? null,
121
+ };
122
+ }
123
+
124
+ if (ctx.quotedMessage) {
125
+ // Only store text summary, not the full nested message proto
126
+ const qm = ctx.quotedMessage;
127
+ clean.quotedMessage = {
128
+ text: qm.conversation || qm.extendedTextMessage?.text || qm.imageMessage?.caption || null,
129
+ type: qm.conversation ? 'conversation' : Object.keys(qm)[0] || null,
130
+ };
131
+ if (ctx.stanzaId) clean.quotedStanzaId = ctx.stanzaId;
132
+ if (ctx.participant) clean.quotedParticipant = ctx.participant;
133
+ }
134
+
135
+ if (ctx.mentionedJid?.length) clean.mentionedJid = ctx.mentionedJid;
136
+ if (ctx.isForwarded) clean.isForwarded = true;
137
+ if (ctx.forwardingScore) clean.forwardingScore = ctx.forwardingScore;
138
+ if (ctx.entryPointConversionSource) clean.entryPointConversionSource = ctx.entryPointConversionSource;
139
+ if (ctx.entryPointConversionApp) clean.entryPointConversionApp = ctx.entryPointConversionApp;
140
+ if (ctx.entryPointConversionDelaySeconds) clean.entryPointConversionDelaySeconds = ctx.entryPointConversionDelaySeconds;
141
+
142
+ return Object.keys(clean).length > 0 ? clean : null;
143
+ }
144
+
145
+ // ─── IPC Message Handler ─────────────────────────────────
146
+
147
+ process.on('message', async (msg) => {
148
+ switch (msg.type) {
149
+ case 'init':
150
+ accountId = msg.accountId;
151
+ authDir = msg.authDir;
152
+ mongodbUri = msg.mongodbUri || '';
153
+ mediaForwardTo = msg.mediaForwardTo || '';
154
+ await connectSocket();
155
+ break;
156
+
157
+ case 'send':
158
+ await handleSend(msg);
159
+ break;
160
+
161
+ case 'start_chat':
162
+ await handleStartChat(msg);
163
+ break;
164
+
165
+ case 'forward_media':
166
+ await handleForwardMedia(msg);
167
+ break;
168
+
169
+ case 'send_image':
170
+ await handleSendImage(msg);
171
+ break;
172
+
173
+ case 'download_audio':
174
+ await handleDownloadAudio(msg);
175
+ break;
176
+
177
+ case 'disconnect':
178
+ await handleDisconnect();
179
+ break;
180
+
181
+ case 'shutdown':
182
+ await handleShutdown();
183
+ break;
184
+ }
185
+ });
186
+
187
+ // ─── Auth State ──────────────────────────────────────────
188
+
189
+ async function getAuthState() {
190
+ if (mongodbUri) {
191
+ try {
192
+ mongoClient = new MongoClient(mongodbUri);
193
+ await mongoClient.connect();
194
+ // Verify connection works
195
+ await mongoClient.db().command({ ping: 1 });
196
+ mongoDb = mongoClient.db();
197
+ return useMongoDBAuthState(mongoDb, accountId);
198
+ } catch (err) {
199
+ // If URI has credentials and auth fails, try without credentials
200
+ try {
201
+ const url = new URL(mongodbUri);
202
+ if (url.username) {
203
+ url.username = '';
204
+ url.password = '';
205
+ const noAuthUri = url.toString();
206
+ if (mongoClient) try { await mongoClient.close(); } catch {}
207
+ mongoClient = new MongoClient(noAuthUri);
208
+ await mongoClient.connect();
209
+ await mongoClient.db().command({ ping: 1 });
210
+ mongoDb = mongoClient.db();
211
+ return useMongoDBAuthState(mongoDb, accountId);
212
+ }
213
+ } catch {}
214
+ // All MongoDB attempts failed, fall back to file auth
215
+ if (mongoClient) try { await mongoClient.close(); } catch {}
216
+ mongoClient = null;
217
+ mongoDb = null;
218
+ }
219
+ }
220
+
221
+ // Fallback to file-based auth
222
+ return useMultiFileAuthState(authDir);
223
+ }
224
+
225
+ // ─── Baileys Socket ──────────────────────────────────────
226
+
227
+ async function connectSocket() {
228
+ try {
229
+ const { state, saveCreds } = await getAuthState();
230
+ const { version } = await fetchLatestBaileysVersion();
231
+
232
+ sock = makeWASocket({
233
+ version,
234
+ auth: {
235
+ creds: state.creds,
236
+ keys: makeCacheableSignalKeyStore(state.keys, workerLogger),
237
+ },
238
+ logger: workerLogger,
239
+ browser: Browsers.ubuntu('Portal'),
240
+ markOnlineOnConnect: false,
241
+ generateHighQualityLinkPreview: false,
242
+ syncFullHistory: false,
243
+ shouldSyncHistoryMessage: () => false,
244
+ connectTimeoutMs: 60_000,
245
+ qrTimeout: 40_000,
246
+ });
247
+
248
+ // ─── Connection Events ─────────────────────────────
249
+
250
+ sock.ev.on('connection.update', (update) => {
251
+ const { connection, lastDisconnect, qr } = update;
252
+
253
+ if (qr) {
254
+ send({ type: 'qr', qr });
255
+ send({ type: 'status', status: 'qr_pending' });
256
+ }
257
+
258
+ if (connection === 'connecting') {
259
+ send({ type: 'status', status: 'connecting' });
260
+ }
261
+
262
+ if (connection === 'open') {
263
+ connectedPhone = phoneFromSockUser(sock.user?.id);
264
+ send({ type: 'connected', phone: connectedPhone });
265
+ send({ type: 'status', status: 'connected' });
266
+ }
267
+
268
+ if (connection === 'close') {
269
+ const statusCode = lastDisconnect?.error?.output?.statusCode;
270
+ const reason = lastDisconnect?.error?.message || 'Unknown';
271
+
272
+ if (statusCode === DisconnectReason.loggedOut) {
273
+ // Clean auth state from MongoDB on logout
274
+ if (mongoDb) {
275
+ deleteMongoDBAuthState(mongoDb, accountId).catch(() => {});
276
+ }
277
+ send({ type: 'logged_out' });
278
+ send({ type: 'status', status: 'logged_out' });
279
+ } else {
280
+ send({ type: 'disconnected', reason, statusCode });
281
+ send({ type: 'status', status: 'disconnected' });
282
+ // Auto-reconnect after short delay
283
+ setTimeout(() => connectSocket(), 3_000);
284
+ }
285
+ }
286
+ });
287
+
288
+ // ─── Credentials ───────────────────────────────────
289
+
290
+ sock.ev.on('creds.update', saveCreds);
291
+
292
+ // ─── Incoming Messages ─────────────────────────────
293
+
294
+ sock.ev.on('messages.upsert', async ({ messages, type }) => {
295
+ // Helper to build base log entry from a message
296
+ function msgLogBase(msg) {
297
+ return {
298
+ messageId: msg.key?.id || null,
299
+ remoteJid: msg.key?.remoteJid || null,
300
+ fromMe: msg.key?.fromMe || false,
301
+ pushName: msg.pushName || null,
302
+ timestamp: msg.messageTimestamp
303
+ ? new Date(Number(msg.messageTimestamp) * 1000).toISOString()
304
+ : new Date().toISOString(),
305
+ };
306
+ }
307
+
308
+ function logFiltered(msg, reason, extra = {}) {
309
+ send({
310
+ type: 'message_log',
311
+ action: 'filtered',
312
+ reason,
313
+ ...msgLogBase(msg),
314
+ ...extra,
315
+ });
316
+ }
317
+
318
+ function logDispatched(msg, extra = {}) {
319
+ send({
320
+ type: 'message_log',
321
+ action: 'dispatched',
322
+ reason: null,
323
+ ...msgLogBase(msg),
324
+ ...extra,
325
+ });
326
+ }
327
+
328
+ if (type !== 'notify') {
329
+ // Log once for the whole batch
330
+ if (messages.length > 0) {
331
+ send({
332
+ type: 'message_log',
333
+ action: 'filtered',
334
+ reason: `not_notify:${type}`,
335
+ messageId: null,
336
+ remoteJid: null,
337
+ fromMe: false,
338
+ pushName: null,
339
+ timestamp: new Date().toISOString(),
340
+ batchSize: messages.length,
341
+ });
342
+ }
343
+ return;
344
+ }
345
+
346
+ for (const msg of messages) {
347
+ // Log raw WhatsApp message before any filtering
348
+ const rawContentType = msg.message ? getContentType(msg.message) : null;
349
+ const rawContextInfo = (rawContentType && msg.message?.[rawContentType]?.contextInfo) || null;
350
+ send({
351
+ type: 'raw_message',
352
+ key: msg.key,
353
+ messageStubType: msg.messageStubType || null,
354
+ pushName: msg.pushName || null,
355
+ messageTimestamp: msg.messageTimestamp || null,
356
+ contentType: rawContentType,
357
+ // Store message structure keys (not binary data) for debugging
358
+ messageKeys: msg.message ? Object.keys(msg.message) : [],
359
+ text: msg.message ? extractText(msg) : null,
360
+ // Store contextInfo for ad attribution, quoted messages, mentions etc.
361
+ // Strip binary fields (jpegThumbnail, thumbnailDirectPath) to keep logs lean
362
+ contextInfo: rawContextInfo ? sanitizeContextInfo(rawContextInfo) : null,
363
+ });
364
+
365
+ if (!msg.message) {
366
+ logFiltered(msg, 'no_message');
367
+ continue;
368
+ }
369
+ if (isJidBroadcast(msg.key.remoteJid)) {
370
+ logFiltered(msg, 'broadcast');
371
+ continue;
372
+ }
373
+ if (isJidGroup(msg.key.remoteJid)) {
374
+ logFiltered(msg, 'group_message');
375
+ continue;
376
+ }
377
+ if (msg.key.remoteJid === 'status@broadcast') {
378
+ logFiltered(msg, 'status_broadcast');
379
+ continue;
380
+ }
381
+
382
+ // Skip stub/system messages (encryption notifications, group events, etc.)
383
+ if (msg.messageStubType) {
384
+ logFiltered(msg, 'stub_message', { stubType: msg.messageStubType });
385
+ continue;
386
+ }
387
+
388
+ const contentType = getContentType(msg.message);
389
+
390
+ // Skip protocol and system content types — these are not user messages
391
+ const SKIP_CONTENT_TYPES = [
392
+ 'protocolMessage', // message edits, revokes, key changes
393
+ 'senderKeyDistributionMessage', // encryption key distribution
394
+ 'messageContextInfo', // context metadata
395
+ 'reactionMessage', // emoji reactions
396
+ 'pollUpdateMessage', // poll vote updates
397
+ 'pinInChatMessage', // pin/unpin
398
+ 'keepInChatMessage', // keep messages
399
+ 'encReactionMessage', // encrypted reactions
400
+ 'editedMessage', // edited message container
401
+ 'viewOnceMessageV2', // view-once protocol wrapper
402
+ 'viewOnceMessageV2Extension',
403
+ 'eventMessage', // calendar events
404
+ 'newsletterAdminInviteMessage',
405
+ 'messageHistoryBundle', // history sync
406
+ ];
407
+ if (SKIP_CONTENT_TYPES.includes(contentType)) {
408
+ logFiltered(msg, 'skip_content_type', { contentType });
409
+ continue;
410
+ }
411
+
412
+ // Build Unipile-compatible webhook payload (1:1 chats only — groups filtered above)
413
+ const jid = msg.key.remoteJid;
414
+ const jidAlt = msg.key.remoteJidAlt || null;
415
+ const fromMe = msg.key.fromMe || false;
416
+
417
+ // Resolve LID to phone-based JID
418
+ // 1. Use remoteJidAlt if available (already in the message key)
419
+ // 2. Fall back to lidMapping lookup
420
+ let contactJid = jid;
421
+ if (isLidUser(contactJid)) {
422
+ if (jidAlt && !isLidUser(jidAlt)) {
423
+ contactJid = jidNormalizedUser(jidAlt);
424
+ } else if (sock.signalRepository?.lidMapping) {
425
+ try {
426
+ const pnJid = await sock.signalRepository.lidMapping.getPNForLID(contactJid);
427
+ if (pnJid) {
428
+ contactJid = jidNormalizedUser(pnJid);
429
+ }
430
+ } catch {}
431
+ }
432
+ }
433
+
434
+ // Resolve chat JID (same logic)
435
+ let chatJid = jidNormalizedUser(jid);
436
+ if (isLidUser(jid)) {
437
+ if (jidAlt && !isLidUser(jidAlt)) {
438
+ chatJid = jidNormalizedUser(jidAlt);
439
+ } else if (sock.signalRepository?.lidMapping) {
440
+ try {
441
+ const pnJid = await sock.signalRepository.lidMapping.getPNForLID(jid);
442
+ if (pnJid) {
443
+ chatJid = jidNormalizedUser(pnJid);
444
+ }
445
+ } catch {}
446
+ }
447
+ }
448
+
449
+ // Skip if all resolution attempts failed
450
+ if (isLidUser(contactJid)) {
451
+ logFiltered(msg, 'unresolved_lid', { contentType, contactJid });
452
+ continue;
453
+ }
454
+
455
+ const contactPhone = jidToPhone(contactJid);
456
+ const contactName = (!fromMe && msg.pushName) ? msg.pushName : contactPhone;
457
+
458
+ // Extract text or use media placeholder for non-text content
459
+ const text = extractText(msg);
460
+
461
+ // User-facing content types that contain actual messages
462
+ const textContentTypes = ['conversation', 'extendedTextMessage'];
463
+ const mediaContentTypes = [
464
+ 'imageMessage', 'videoMessage', 'audioMessage', 'documentMessage',
465
+ 'stickerMessage', 'contactMessage', 'contactsArrayMessage',
466
+ 'locationMessage', 'liveLocationMessage',
467
+ 'listMessage', 'listResponseMessage',
468
+ 'buttonsMessage', 'buttonsResponseMessage',
469
+ 'templateMessage', 'templateButtonReplyMessage',
470
+ 'interactiveMessage', 'interactiveResponseMessage',
471
+ 'pollCreationMessage', 'pollCreationMessageV3',
472
+ 'orderMessage', 'productMessage',
473
+ 'requestPhoneNumberMessage',
474
+ 'albumMessage', 'stickerPackMessage',
475
+ ];
476
+
477
+ const isMediaMsg = mediaContentTypes.includes(contentType);
478
+ const mediaLabel = MEDIA_TYPE_LABELS[contentType] || MEDIA_PLACEHOLDER;
479
+ // For media with caption: prepend label so AI knows the type
480
+ // For media without caption: just the label
481
+ // For text-only: just the text
482
+ const messageText = isMediaMsg
483
+ ? (text ? `${mediaLabel} ${text}` : mediaLabel)
484
+ : (text || '');
485
+
486
+ if (!messageText) {
487
+ logFiltered(msg, 'empty_text', { contentType });
488
+ continue;
489
+ }
490
+
491
+ // Cache media messages for on-demand forwarding (operator workflow)
492
+ // Cache both incoming and outgoing media (operator may reply with documents from same device)
493
+ if (isMediaMsg) {
494
+ messageCache.set(msg.key.id, { msg, contentType, timestamp: Date.now() });
495
+ }
496
+
497
+ // Forward media messages to dev number for inspection
498
+ if (isMediaMsg && !fromMe && mediaForwardTo && sock) {
499
+ const forwardJid = mediaForwardTo.replace(/[^0-9]/g, '') + '@s.whatsapp.net';
500
+ forwardMedia(msg, contentType, forwardJid, contactName, contactPhone);
501
+ }
502
+
503
+ const timestamp = msg.messageTimestamp
504
+ ? new Date(Number(msg.messageTimestamp) * 1000).toISOString()
505
+ : new Date().toISOString();
506
+
507
+ const payload = {
508
+ account_id: accountId,
509
+ account_type: 'WHATSAPP',
510
+ event: 'message_received',
511
+ chat_id: `${accountId}:${chatJid}`,
512
+ message_id: msg.key.id,
513
+ message: messageText,
514
+ content_type: contentType,
515
+ timestamp,
516
+ is_sender: fromMe,
517
+ is_event: false,
518
+ hidden: false,
519
+ sender: {
520
+ attendee_id: contactPhone,
521
+ attendee_name: contactName,
522
+ attendee_provider_id: contactPhone,
523
+ attendee_specifics: { phone_number: contactPhone },
524
+ },
525
+ attendees: [{
526
+ attendee_id: contactPhone,
527
+ attendee_name: contactName,
528
+ attendee_provider_id: contactPhone,
529
+ attendee_specifics: { phone_number: contactPhone },
530
+ }],
531
+ account_info: {
532
+ type: 'WHATSAPP',
533
+ phone_number: connectedPhone || '',
534
+ },
535
+ ad_context: rawContextInfo ? sanitizeAdContext(rawContextInfo) : null,
536
+ quoted_text: rawContextInfo?.quotedMessage
537
+ ? (rawContextInfo.quotedMessage.conversation || rawContextInfo.quotedMessage.extendedTextMessage?.text || rawContextInfo.quotedMessage.imageMessage?.caption || null)
538
+ : null,
539
+ };
540
+
541
+ console.log(`[MSG] ${fromMe ? '→' : '←'} ${contactPhone} (${contactName}) | ${contentType} | ${messageText.substring(0, 100)}`);
542
+ send({ type: 'message', payload });
543
+ logDispatched(msg, { contentType, contactPhone, messageText: messageText.slice(0, 200) });
544
+ }
545
+ });
546
+
547
+ send({ type: 'ready' });
548
+ } catch (err) {
549
+ send({ type: 'reply', reqId: '__init__', error: err.message });
550
+ process.exit(1);
551
+ }
552
+ }
553
+
554
+ // ─── Media Forwarding ───────────────────────────────────
555
+
556
+ async function forwardMedia(msg, contentType, toJid, senderName, senderPhone) {
557
+ const label = `[${senderName} ${senderPhone}]`;
558
+ try {
559
+ const mediaMsg = msg.message;
560
+ const DOWNLOADABLE = ['imageMessage', 'videoMessage', 'audioMessage', 'documentMessage', 'stickerMessage'];
561
+
562
+ if (DOWNLOADABLE.includes(contentType)) {
563
+ const buffer = await downloadMediaMessage(msg, 'buffer', {});
564
+ const caption = `${label} — ${contentType.replace('Message', '')}`;
565
+
566
+ if (contentType === 'imageMessage') {
567
+ await sock.sendMessage(toJid, { image: buffer, caption });
568
+ } else if (contentType === 'videoMessage') {
569
+ await sock.sendMessage(toJid, { video: buffer, caption });
570
+ } else if (contentType === 'audioMessage') {
571
+ const mimetype = mediaMsg.audioMessage?.mimetype || 'audio/ogg; codecs=opus';
572
+ await sock.sendMessage(toJid, { audio: buffer, mimetype });
573
+ await sock.sendMessage(toJid, { text: caption });
574
+ } else if (contentType === 'documentMessage') {
575
+ const mimetype = mediaMsg.documentMessage?.mimetype || 'application/octet-stream';
576
+ const fileName = mediaMsg.documentMessage?.fileName || 'document';
577
+ await sock.sendMessage(toJid, { document: buffer, mimetype, fileName, caption });
578
+ } else if (contentType === 'stickerMessage') {
579
+ await sock.sendMessage(toJid, { sticker: buffer });
580
+ await sock.sendMessage(toJid, { text: caption });
581
+ }
582
+ } else {
583
+ // Non-downloadable media (location, contacts, polls, etc.)
584
+ await sock.sendMessage(toJid, { text: `${label} — ${contentType}` });
585
+ }
586
+ } catch (err) {
587
+ // Fallback: send text notification if download/forward fails
588
+ try {
589
+ await sock.sendMessage(toJid, { text: `${label} — ${contentType} (download failed)` });
590
+ } catch {}
591
+ }
592
+ }
593
+
594
+ // ─── Command Handlers ────────────────────────────────────
595
+
596
+ async function handleSend({ chatId, text, reqId }) {
597
+ try {
598
+ if (!sock) throw new Error('Socket not connected');
599
+ const result = await sock.sendMessage(chatId, { text });
600
+ send({ type: 'reply', reqId, data: { message_id: result?.key?.id || null } });
601
+ } catch (err) {
602
+ send({ type: 'reply', reqId, error: err.message });
603
+ }
604
+ }
605
+
606
+ async function handleForwardMedia({ messageId, toPhone, caption, reqId }) {
607
+ try {
608
+ if (!sock) throw new Error('Socket not connected');
609
+
610
+ const cached = messageCache.get(messageId);
611
+ if (!cached) throw new Error('Message not in cache (expired or not found)');
612
+
613
+ const { msg, contentType } = cached;
614
+ const toJid = toPhone.replace(/[^0-9]/g, '') + '@s.whatsapp.net';
615
+
616
+ const DOWNLOADABLE = ['imageMessage', 'videoMessage', 'audioMessage', 'documentMessage', 'stickerMessage'];
617
+
618
+ if (DOWNLOADABLE.includes(contentType)) {
619
+ const buffer = await downloadMediaMessage(msg, 'buffer', {});
620
+
621
+ if (contentType === 'imageMessage') {
622
+ await sock.sendMessage(toJid, { image: buffer, caption: caption || '' });
623
+ } else if (contentType === 'videoMessage') {
624
+ await sock.sendMessage(toJid, { video: buffer, caption: caption || '' });
625
+ } else if (contentType === 'audioMessage') {
626
+ const mimetype = msg.message?.audioMessage?.mimetype || 'audio/ogg; codecs=opus';
627
+ await sock.sendMessage(toJid, { audio: buffer, mimetype });
628
+ if (caption) await sock.sendMessage(toJid, { text: caption });
629
+ } else if (contentType === 'documentMessage') {
630
+ const mimetype = msg.message?.documentMessage?.mimetype || 'application/octet-stream';
631
+ const fileName = msg.message?.documentMessage?.fileName || 'document';
632
+ await sock.sendMessage(toJid, { document: buffer, mimetype, fileName, caption: caption || '' });
633
+ } else if (contentType === 'stickerMessage') {
634
+ await sock.sendMessage(toJid, { sticker: buffer });
635
+ if (caption) await sock.sendMessage(toJid, { text: caption });
636
+ }
637
+
638
+ send({ type: 'reply', reqId, data: { success: true } });
639
+ } else {
640
+ // Non-downloadable media — send caption as text
641
+ await sock.sendMessage(toJid, { text: caption || `[Media: ${contentType}]` });
642
+ send({ type: 'reply', reqId, data: { success: true } });
643
+ }
644
+ } catch (err) {
645
+ send({ type: 'reply', reqId, error: err.message });
646
+ }
647
+ }
648
+
649
+ async function handleSendImage({ chatId, imageBase64, caption, reqId }) {
650
+ try {
651
+ if (!sock) throw new Error('Socket not connected');
652
+ const buffer = Buffer.from(imageBase64, 'base64');
653
+ await sock.sendMessage(chatId, { image: buffer, caption: caption || '' });
654
+ send({ type: 'reply', reqId, data: { success: true } });
655
+ } catch (err) {
656
+ send({ type: 'reply', reqId, error: err.message });
657
+ }
658
+ }
659
+
660
+ async function handleDownloadAudio({ messageId, reqId }) {
661
+ try {
662
+ const cached = messageCache.get(messageId);
663
+ if (!cached) throw new Error('Message not in cache');
664
+ if (cached.contentType !== 'audioMessage') throw new Error('Not an audio message');
665
+ const buffer = await downloadMediaMessage(cached.msg, 'buffer', {});
666
+ const mimetype = cached.msg.message?.audioMessage?.mimetype || 'audio/ogg; codecs=opus';
667
+ send({ type: 'reply', reqId, data: { audio_base64: buffer.toString('base64'), mimetype } });
668
+ } catch (err) {
669
+ send({ type: 'reply', reqId, error: err.message });
670
+ }
671
+ }
672
+
673
+ async function handleStartChat({ phone, text, reqId }) {
674
+ try {
675
+ if (!sock) throw new Error('Socket not connected');
676
+ const jid = phone.replace(/[^0-9]/g, '') + '@s.whatsapp.net';
677
+ const result = await sock.sendMessage(jid, { text });
678
+ send({ type: 'reply', reqId, data: { chat_id: `${accountId}:${jid}`, message_id: result?.key?.id || null } });
679
+ } catch (err) {
680
+ send({ type: 'reply', reqId, error: err.message });
681
+ }
682
+ }
683
+
684
+ async function handleDisconnect() {
685
+ // Explicit logout — clears WhatsApp session and auth state
686
+ try {
687
+ if (sock) {
688
+ await sock.logout();
689
+ }
690
+ } catch {
691
+ // Ignore logout errors
692
+ }
693
+ if (mongoClient) {
694
+ try { await mongoClient.close(); } catch {}
695
+ }
696
+ process.exit(0);
697
+ }
698
+
699
+ async function handleShutdown() {
700
+ // Graceful close — preserves auth state for reconnection after restart
701
+ try {
702
+ if (sock) {
703
+ sock.end(undefined);
704
+ }
705
+ } catch {}
706
+ if (mongoClient) {
707
+ try { await mongoClient.close(); } catch {}
708
+ }
709
+ process.exit(0);
710
+ }
711
+
712
+ // ─── Handle unexpected errors ────────────────────────────
713
+
714
+ process.on('uncaughtException', (err) => {
715
+ send({ type: 'reply', reqId: '__error__', error: `Uncaught: ${err.message}` });
716
+ process.exit(1);
717
+ });
718
+
719
+ process.on('unhandledRejection', (err) => {
720
+ send({ type: 'reply', reqId: '__error__', error: `Unhandled: ${err?.message || err}` });
721
+ });