@blackcode_sa/metaestetics-api 1.12.62 → 1.12.63

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 (273) hide show
  1. package/dist/admin/index.d.mts +4 -2
  2. package/dist/admin/index.d.ts +4 -2
  3. package/dist/admin/index.js +4 -45
  4. package/dist/admin/index.mjs +4 -45
  5. package/dist/backoffice/index.d.mts +9 -0
  6. package/dist/backoffice/index.d.ts +9 -0
  7. package/dist/backoffice/index.js +11 -0
  8. package/dist/backoffice/index.mjs +11 -0
  9. package/dist/index.d.mts +99 -3
  10. package/dist/index.d.ts +99 -3
  11. package/dist/index.js +545 -281
  12. package/dist/index.mjs +867 -603
  13. package/package.json +119 -119
  14. package/src/__mocks__/firstore.ts +10 -10
  15. package/src/admin/aggregation/README.md +79 -79
  16. package/src/admin/aggregation/appointment/README.md +128 -128
  17. package/src/admin/aggregation/appointment/appointment.aggregation.service.ts +1844 -1844
  18. package/src/admin/aggregation/appointment/index.ts +1 -1
  19. package/src/admin/aggregation/clinic/README.md +52 -52
  20. package/src/admin/aggregation/clinic/clinic.aggregation.service.ts +703 -703
  21. package/src/admin/aggregation/clinic/index.ts +1 -1
  22. package/src/admin/aggregation/forms/README.md +13 -13
  23. package/src/admin/aggregation/forms/filled-forms.aggregation.service.ts +322 -322
  24. package/src/admin/aggregation/forms/index.ts +1 -1
  25. package/src/admin/aggregation/index.ts +8 -8
  26. package/src/admin/aggregation/patient/README.md +27 -27
  27. package/src/admin/aggregation/patient/index.ts +1 -1
  28. package/src/admin/aggregation/patient/patient.aggregation.service.ts +141 -141
  29. package/src/admin/aggregation/practitioner/README.md +42 -42
  30. package/src/admin/aggregation/practitioner/index.ts +1 -1
  31. package/src/admin/aggregation/practitioner/practitioner.aggregation.service.ts +433 -433
  32. package/src/admin/aggregation/practitioner-invite/index.ts +1 -1
  33. package/src/admin/aggregation/practitioner-invite/practitioner-invite.aggregation.service.ts +961 -961
  34. package/src/admin/aggregation/procedure/README.md +43 -43
  35. package/src/admin/aggregation/procedure/index.ts +1 -1
  36. package/src/admin/aggregation/procedure/procedure.aggregation.service.ts +702 -702
  37. package/src/admin/aggregation/reviews/index.ts +1 -1
  38. package/src/admin/aggregation/reviews/reviews.aggregation.service.ts +641 -689
  39. package/src/admin/booking/README.md +125 -125
  40. package/src/admin/booking/booking.admin.ts +1037 -1037
  41. package/src/admin/booking/booking.calculator.ts +712 -712
  42. package/src/admin/booking/booking.types.ts +59 -59
  43. package/src/admin/booking/index.ts +3 -3
  44. package/src/admin/booking/timezones-problem.md +185 -185
  45. package/src/admin/calendar/README.md +7 -7
  46. package/src/admin/calendar/calendar.admin.service.ts +345 -345
  47. package/src/admin/calendar/index.ts +1 -1
  48. package/src/admin/documentation-templates/document-manager.admin.ts +260 -260
  49. package/src/admin/documentation-templates/index.ts +1 -1
  50. package/src/admin/free-consultation/free-consultation-utils.admin.ts +148 -148
  51. package/src/admin/free-consultation/index.ts +1 -1
  52. package/src/admin/index.ts +75 -75
  53. package/src/admin/logger/index.ts +78 -78
  54. package/src/admin/mailing/README.md +95 -95
  55. package/src/admin/mailing/appointment/appointment.mailing.service.ts +732 -732
  56. package/src/admin/mailing/appointment/index.ts +1 -1
  57. package/src/admin/mailing/appointment/templates/patient/appointment-confirmed.html +40 -40
  58. package/src/admin/mailing/base.mailing.service.ts +208 -208
  59. package/src/admin/mailing/index.ts +3 -3
  60. package/src/admin/mailing/practitionerInvite/existing-practitioner-invite.mailing.ts +611 -611
  61. package/src/admin/mailing/practitionerInvite/index.ts +2 -2
  62. package/src/admin/mailing/practitionerInvite/practitionerInvite.mailing.ts +395 -395
  63. package/src/admin/mailing/practitionerInvite/templates/existing-practitioner-invitation.template.ts +155 -155
  64. package/src/admin/mailing/practitionerInvite/templates/invitation.template.ts +101 -101
  65. package/src/admin/mailing/practitionerInvite/templates/invite-accepted-notification.template.ts +228 -228
  66. package/src/admin/mailing/practitionerInvite/templates/invite-rejected-notification.template.ts +242 -242
  67. package/src/admin/notifications/index.ts +1 -1
  68. package/src/admin/notifications/notifications.admin.ts +710 -710
  69. package/src/admin/requirements/README.md +128 -128
  70. package/src/admin/requirements/index.ts +1 -1
  71. package/src/admin/requirements/patient-requirements.admin.service.ts +475 -475
  72. package/src/admin/users/index.ts +1 -1
  73. package/src/admin/users/user-profile.admin.ts +405 -405
  74. package/src/backoffice/constants/certification.constants.ts +13 -13
  75. package/src/backoffice/constants/index.ts +1 -1
  76. package/src/backoffice/errors/backoffice.errors.ts +181 -181
  77. package/src/backoffice/errors/index.ts +1 -1
  78. package/src/backoffice/expo-safe/README.md +26 -26
  79. package/src/backoffice/expo-safe/index.ts +41 -41
  80. package/src/backoffice/index.ts +5 -5
  81. package/src/backoffice/services/FIXES_README.md +102 -102
  82. package/src/backoffice/services/README.md +40 -40
  83. package/src/backoffice/services/brand.service.ts +256 -256
  84. package/src/backoffice/services/category.service.ts +318 -318
  85. package/src/backoffice/services/constants.service.ts +385 -385
  86. package/src/backoffice/services/documentation-template.service.ts +202 -202
  87. package/src/backoffice/services/index.ts +8 -8
  88. package/src/backoffice/services/migrate-products.ts +116 -116
  89. package/src/backoffice/services/product.service.ts +553 -553
  90. package/src/backoffice/services/requirement.service.ts +235 -235
  91. package/src/backoffice/services/subcategory.service.ts +395 -395
  92. package/src/backoffice/services/technology.service.ts +1083 -1070
  93. package/src/backoffice/types/README.md +12 -12
  94. package/src/backoffice/types/admin-constants.types.ts +69 -69
  95. package/src/backoffice/types/brand.types.ts +29 -29
  96. package/src/backoffice/types/category.types.ts +62 -62
  97. package/src/backoffice/types/documentation-templates.types.ts +28 -28
  98. package/src/backoffice/types/index.ts +10 -10
  99. package/src/backoffice/types/procedure-product.types.ts +38 -38
  100. package/src/backoffice/types/product.types.ts +240 -240
  101. package/src/backoffice/types/requirement.types.ts +63 -63
  102. package/src/backoffice/types/static/README.md +18 -18
  103. package/src/backoffice/types/static/blocking-condition.types.ts +21 -21
  104. package/src/backoffice/types/static/certification.types.ts +37 -37
  105. package/src/backoffice/types/static/contraindication.types.ts +19 -19
  106. package/src/backoffice/types/static/index.ts +6 -6
  107. package/src/backoffice/types/static/pricing.types.ts +16 -16
  108. package/src/backoffice/types/static/procedure-family.types.ts +14 -14
  109. package/src/backoffice/types/static/treatment-benefit.types.ts +22 -22
  110. package/src/backoffice/types/subcategory.types.ts +34 -34
  111. package/src/backoffice/types/technology.types.ts +163 -161
  112. package/src/backoffice/validations/index.ts +1 -1
  113. package/src/backoffice/validations/schemas.ts +164 -163
  114. package/src/config/__mocks__/firebase.ts +99 -99
  115. package/src/config/firebase.ts +78 -78
  116. package/src/config/index.ts +9 -9
  117. package/src/errors/auth.error.ts +6 -6
  118. package/src/errors/auth.errors.ts +200 -200
  119. package/src/errors/clinic.errors.ts +32 -32
  120. package/src/errors/firebase.errors.ts +47 -47
  121. package/src/errors/user.errors.ts +99 -99
  122. package/src/index.backup.ts +407 -407
  123. package/src/index.ts +6 -6
  124. package/src/locales/en.ts +31 -31
  125. package/src/recommender/admin/index.ts +1 -1
  126. package/src/recommender/admin/services/recommender.service.admin.ts +5 -5
  127. package/src/recommender/front/index.ts +1 -1
  128. package/src/recommender/front/services/onboarding.service.ts +5 -5
  129. package/src/recommender/front/services/recommender.service.ts +3 -3
  130. package/src/recommender/index.ts +1 -1
  131. package/src/services/PATIENTAUTH.MD +197 -197
  132. package/src/services/README.md +106 -106
  133. package/src/services/__tests__/auth/auth.mock.test.ts +17 -17
  134. package/src/services/__tests__/auth/auth.setup.ts +293 -293
  135. package/src/services/__tests__/auth.service.test.ts +346 -346
  136. package/src/services/__tests__/base.service.test.ts +77 -77
  137. package/src/services/__tests__/user.service.test.ts +528 -528
  138. package/src/services/appointment/README.md +17 -17
  139. package/src/services/appointment/appointment.service.ts +2505 -2082
  140. package/src/services/appointment/index.ts +1 -1
  141. package/src/services/appointment/utils/appointment.utils.ts +552 -552
  142. package/src/services/appointment/utils/extended-procedure.utils.ts +314 -314
  143. package/src/services/appointment/utils/form-initialization.utils.ts +225 -225
  144. package/src/services/appointment/utils/recommended-procedure.utils.ts +195 -195
  145. package/src/services/appointment/utils/zone-management.utils.ts +353 -353
  146. package/src/services/appointment/utils/zone-photo.utils.ts +152 -152
  147. package/src/services/auth/auth.service.ts +989 -989
  148. package/src/services/auth/auth.v2.service.ts +961 -961
  149. package/src/services/auth/index.ts +7 -7
  150. package/src/services/auth/utils/error.utils.ts +90 -90
  151. package/src/services/auth/utils/firebase.utils.ts +49 -49
  152. package/src/services/auth/utils/index.ts +21 -21
  153. package/src/services/auth/utils/practitioner.utils.ts +125 -125
  154. package/src/services/base.service.ts +41 -41
  155. package/src/services/calendar/calendar.service.ts +1077 -1077
  156. package/src/services/calendar/calendar.v2.service.ts +1683 -1683
  157. package/src/services/calendar/calendar.v3.service.ts +313 -313
  158. package/src/services/calendar/externalCalendar.service.ts +178 -178
  159. package/src/services/calendar/index.ts +5 -5
  160. package/src/services/calendar/synced-calendars.service.ts +743 -743
  161. package/src/services/calendar/utils/appointment.utils.ts +265 -265
  162. package/src/services/calendar/utils/calendar-event.utils.ts +646 -646
  163. package/src/services/calendar/utils/clinic.utils.ts +237 -237
  164. package/src/services/calendar/utils/docs.utils.ts +157 -157
  165. package/src/services/calendar/utils/google-calendar.utils.ts +697 -697
  166. package/src/services/calendar/utils/index.ts +8 -8
  167. package/src/services/calendar/utils/patient.utils.ts +198 -198
  168. package/src/services/calendar/utils/practitioner.utils.ts +221 -221
  169. package/src/services/calendar/utils/synced-calendar.utils.ts +472 -472
  170. package/src/services/clinic/README.md +204 -204
  171. package/src/services/clinic/__tests__/clinic-admin.service.test.ts +287 -287
  172. package/src/services/clinic/__tests__/clinic-group.service.test.ts +352 -352
  173. package/src/services/clinic/__tests__/clinic.service.test.ts +354 -354
  174. package/src/services/clinic/billing-transactions.service.ts +217 -217
  175. package/src/services/clinic/clinic-admin.service.ts +202 -202
  176. package/src/services/clinic/clinic-group.service.ts +310 -310
  177. package/src/services/clinic/clinic.service.ts +708 -708
  178. package/src/services/clinic/index.ts +5 -5
  179. package/src/services/clinic/practitioner-invite.service.ts +519 -519
  180. package/src/services/clinic/utils/admin.utils.ts +551 -551
  181. package/src/services/clinic/utils/clinic-group.utils.ts +646 -646
  182. package/src/services/clinic/utils/clinic.utils.ts +949 -949
  183. package/src/services/clinic/utils/filter.utils.d.ts +23 -23
  184. package/src/services/clinic/utils/filter.utils.ts +446 -446
  185. package/src/services/clinic/utils/index.ts +11 -11
  186. package/src/services/clinic/utils/photos.utils.ts +188 -188
  187. package/src/services/clinic/utils/search.utils.ts +84 -84
  188. package/src/services/clinic/utils/tag.utils.ts +124 -124
  189. package/src/services/documentation-templates/documentation-template.service.ts +537 -537
  190. package/src/services/documentation-templates/filled-document.service.ts +587 -587
  191. package/src/services/documentation-templates/index.ts +2 -2
  192. package/src/services/index.ts +13 -13
  193. package/src/services/media/index.ts +1 -1
  194. package/src/services/media/media.service.ts +418 -418
  195. package/src/services/notifications/__tests__/notification.service.test.ts +242 -242
  196. package/src/services/notifications/index.ts +1 -1
  197. package/src/services/notifications/notification.service.ts +215 -215
  198. package/src/services/patient/README.md +48 -48
  199. package/src/services/patient/To-Do.md +43 -43
  200. package/src/services/patient/__tests__/patient.service.test.ts +294 -294
  201. package/src/services/patient/index.ts +2 -2
  202. package/src/services/patient/patient.service.ts +883 -883
  203. package/src/services/patient/patientRequirements.service.ts +285 -285
  204. package/src/services/patient/utils/aesthetic-analysis.utils.ts +176 -176
  205. package/src/services/patient/utils/clinic.utils.ts +80 -80
  206. package/src/services/patient/utils/docs.utils.ts +142 -142
  207. package/src/services/patient/utils/index.ts +9 -9
  208. package/src/services/patient/utils/location.utils.ts +126 -126
  209. package/src/services/patient/utils/medical-stuff.utils.ts +143 -143
  210. package/src/services/patient/utils/medical.utils.ts +458 -458
  211. package/src/services/patient/utils/practitioner.utils.ts +260 -260
  212. package/src/services/patient/utils/profile.utils.ts +510 -510
  213. package/src/services/patient/utils/sensitive.utils.ts +260 -260
  214. package/src/services/patient/utils/token.utils.ts +211 -211
  215. package/src/services/practitioner/README.md +145 -145
  216. package/src/services/practitioner/index.ts +1 -1
  217. package/src/services/practitioner/practitioner.service.ts +1742 -1742
  218. package/src/services/procedure/README.md +163 -163
  219. package/src/services/procedure/index.ts +1 -1
  220. package/src/services/procedure/procedure.service.ts +1682 -1682
  221. package/src/services/reviews/index.ts +1 -1
  222. package/src/services/reviews/reviews.service.ts +636 -683
  223. package/src/services/user/index.ts +1 -1
  224. package/src/services/user/user.service.ts +489 -489
  225. package/src/services/user/user.v2.service.ts +466 -466
  226. package/src/types/appointment/index.ts +481 -453
  227. package/src/types/calendar/index.ts +258 -258
  228. package/src/types/calendar/synced-calendar.types.ts +66 -66
  229. package/src/types/clinic/index.ts +489 -489
  230. package/src/types/clinic/practitioner-invite.types.ts +91 -91
  231. package/src/types/clinic/preferences.types.ts +159 -159
  232. package/src/types/clinic/to-do +3 -3
  233. package/src/types/documentation-templates/index.ts +308 -308
  234. package/src/types/index.ts +44 -44
  235. package/src/types/notifications/README.md +77 -77
  236. package/src/types/notifications/index.ts +265 -265
  237. package/src/types/patient/aesthetic-analysis.types.ts +66 -66
  238. package/src/types/patient/allergies.ts +58 -58
  239. package/src/types/patient/index.ts +275 -273
  240. package/src/types/patient/medical-info.types.ts +152 -152
  241. package/src/types/patient/patient-requirements.ts +92 -92
  242. package/src/types/patient/token.types.ts +61 -61
  243. package/src/types/practitioner/index.ts +206 -206
  244. package/src/types/procedure/index.ts +181 -181
  245. package/src/types/profile/index.ts +39 -39
  246. package/src/types/reviews/index.ts +130 -132
  247. package/src/types/tz-lookup.d.ts +4 -4
  248. package/src/types/user/index.ts +38 -38
  249. package/src/utils/TIMESTAMPS.md +176 -176
  250. package/src/utils/TimestampUtils.ts +241 -241
  251. package/src/utils/index.ts +1 -1
  252. package/src/validations/appointment.schema.ts +574 -574
  253. package/src/validations/calendar.schema.ts +225 -225
  254. package/src/validations/clinic.schema.ts +493 -493
  255. package/src/validations/common.schema.ts +25 -25
  256. package/src/validations/documentation-templates/index.ts +1 -1
  257. package/src/validations/documentation-templates/template.schema.ts +220 -220
  258. package/src/validations/documentation-templates.schema.ts +10 -10
  259. package/src/validations/index.ts +20 -20
  260. package/src/validations/media.schema.ts +10 -10
  261. package/src/validations/notification.schema.ts +90 -90
  262. package/src/validations/patient/aesthetic-analysis.schema.ts +55 -55
  263. package/src/validations/patient/medical-info.schema.ts +125 -125
  264. package/src/validations/patient/patient-requirements.schema.ts +84 -84
  265. package/src/validations/patient/token.schema.ts +29 -29
  266. package/src/validations/patient.schema.ts +217 -216
  267. package/src/validations/practitioner.schema.ts +222 -222
  268. package/src/validations/procedure-product.schema.ts +41 -41
  269. package/src/validations/procedure.schema.ts +124 -124
  270. package/src/validations/profile-info.schema.ts +41 -41
  271. package/src/validations/reviews.schema.ts +189 -195
  272. package/src/validations/schemas.ts +104 -104
  273. package/src/validations/shared.schema.ts +78 -78
@@ -1 +1 @@
1
- export * from "./appointment.mailing.service";
1
+ export * from "./appointment.mailing.service";
@@ -1,40 +1,40 @@
1
- <!DOCTYPE html>
2
- <html lang="en">
3
- <head>
4
- <meta charset="UTF-8" />
5
- <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
- <title>Appointment Confirmed</title>
7
- <style>
8
- body {
9
- font-family: Arial, sans-serif;
10
- margin: 0;
11
- padding: 20px;
12
- color: #333;
13
- }
14
- .container {
15
- background-color: #f9f9f9;
16
- padding: 20px;
17
- border-radius: 5px;
18
- }
19
- h1 {
20
- color: #4caf50;
21
- }
22
- </style>
23
- </head>
24
- <body>
25
- <div class="container">
26
- <h1>Appointment Confirmed!</h1>
27
- <p>Dear {{patientName}},</p>
28
- <p>
29
- This is a placeholder email to confirm your appointment for
30
- <strong>{{procedureName}}</strong>.
31
- </p>
32
- <p>Date: {{appointmentDate}}</p>
33
- <p>Time: {{appointmentTime}}</p>
34
- <p>With: {{practitionerName}}</p>
35
- <p>At: {{clinicName}}</p>
36
- <p>We look forward to seeing you!</p>
37
- <p>Sincerely,<br />The {{clinicName}} Team</p>
38
- </div>
39
- </body>
40
- </html>
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
+ <title>Appointment Confirmed</title>
7
+ <style>
8
+ body {
9
+ font-family: Arial, sans-serif;
10
+ margin: 0;
11
+ padding: 20px;
12
+ color: #333;
13
+ }
14
+ .container {
15
+ background-color: #f9f9f9;
16
+ padding: 20px;
17
+ border-radius: 5px;
18
+ }
19
+ h1 {
20
+ color: #4caf50;
21
+ }
22
+ </style>
23
+ </head>
24
+ <body>
25
+ <div class="container">
26
+ <h1>Appointment Confirmed!</h1>
27
+ <p>Dear {{patientName}},</p>
28
+ <p>
29
+ This is a placeholder email to confirm your appointment for
30
+ <strong>{{procedureName}}</strong>.
31
+ </p>
32
+ <p>Date: {{appointmentDate}}</p>
33
+ <p>Time: {{appointmentTime}}</p>
34
+ <p>With: {{practitionerName}}</p>
35
+ <p>At: {{clinicName}}</p>
36
+ <p>We look forward to seeing you!</p>
37
+ <p>Sincerely,<br />The {{clinicName}} Team</p>
38
+ </div>
39
+ </body>
40
+ </html>
@@ -1,208 +1,208 @@
1
- import * as admin from "firebase-admin";
2
- // Import the new Mailgun types if available, or use a generic 'any' if not directly importable
3
- // For a shared library, it's often better to define a minimal interface for what you need
4
- // or expect the caller (Cloud Function) to pass a pre-configured client of 'any' type.
5
- // import Mailgun from "mailgun.js"; // This might not be a direct dep of the API pkg
6
- import { Logger } from "../logger"; // Assuming logger is in Api/src/logger
7
-
8
- /**
9
- * Minimal interface for the new mailgun.js client's messages API
10
- * This helps avoid a direct dependency on mailgun.js in the API package if not desired,
11
- * or provides clearer typing if it is.
12
- */
13
- interface NewMailgunMessagesAPI {
14
- create(domain: string, data: any): Promise<any>;
15
- }
16
-
17
- export interface NewMailgunClient {
18
- messages: NewMailgunMessagesAPI;
19
- // Add other methods/properties if your BaseMailingService uses them
20
- }
21
-
22
- /**
23
- * Base mailing service class that provides common functionality for all mailing services
24
- * This version is updated to expect a mailgun.js v10+ client.
25
- */
26
- export class BaseMailingService {
27
- protected db: FirebaseFirestore.Firestore;
28
- protected mailgunClient: NewMailgunClient; // Expecting the new mailgun.js client
29
-
30
- /**
31
- * Constructor for BaseMailingService
32
- * @param firestore Firestore instance provided by the caller
33
- * @param mailgunClient Mailgun client instance (mailgun.js v10+) provided by the caller
34
- */
35
- constructor(
36
- firestore: FirebaseFirestore.Firestore,
37
- mailgunClient: NewMailgunClient // Expect the new client
38
- ) {
39
- this.db = firestore;
40
- this.mailgunClient = mailgunClient;
41
-
42
- if (!this.db) {
43
- Logger.error("[BaseMailingService] No Firestore instance provided");
44
- throw new Error("Firestore instance is required");
45
- }
46
-
47
- if (
48
- !this.mailgunClient ||
49
- typeof this.mailgunClient.messages?.create !== "function"
50
- ) {
51
- Logger.error(
52
- "[BaseMailingService] Invalid Mailgun client provided (must be mailgun.js v10+ client)"
53
- );
54
- throw new Error("A valid mailgun.js v10+ client is required");
55
- }
56
-
57
- // The new client is configured with the URL (e.g., EU endpoint) by the caller.
58
- // Validation of this configuration can be logged if details are accessible,
59
- // but the primary responsibility for correct endpoint is now with client setup.
60
- Logger.info(
61
- "[BaseMailingService] Service initialized with mailgun.js client."
62
- );
63
- }
64
-
65
- /**
66
- * Sends an email using the new Mailgun client API.
67
- * @param domain The Mailgun domain to send from (e.g., mg.metaesthetics.net)
68
- * @param data Email data to send, compatible with mailgun.js messages.create API
69
- * @returns Promise with the sending result
70
- */
71
- protected async sendEmail(
72
- domain: string, // The new API requires the domain as the first argument to messages.create
73
- data: any // Data format according to mailgun.js messages.create
74
- ): Promise<any> {
75
- if (!domain) {
76
- throw new Error("Mailgun domain is required for sending email.");
77
- }
78
- if (!data) {
79
- throw new Error("Email data object is required");
80
- }
81
- if (!data.to) {
82
- throw new Error("Email 'to' address is required");
83
- }
84
- if (!data.from) {
85
- throw new Error("Email 'from' address is required");
86
- }
87
- if (!data.subject) {
88
- throw new Error("Email 'subject' is required");
89
- }
90
- if (!data.html && !data.text) {
91
- throw new Error("Email must have either 'html' or 'text' content");
92
- }
93
-
94
- Logger.info(
95
- "[BaseMailingService] Sending email via Mailgun (mailgun.js API)",
96
- {
97
- domain,
98
- to: data.to,
99
- from: data.from,
100
- subject: data.subject,
101
- hasHtml: !!data.html,
102
- hasText: !!data.text,
103
- }
104
- );
105
-
106
- try {
107
- // Use the new mailgun.js client API: client.messages.create(domain, data)
108
- const result = await this.mailgunClient.messages.create(domain, data);
109
- Logger.info(
110
- "[BaseMailingService] Email sent successfully with mailgun.js client",
111
- result
112
- );
113
- return result;
114
- } catch (error: any) {
115
- Logger.error(
116
- "[BaseMailingService] Error sending email with mailgun.js client",
117
- {
118
- errorMessage: error?.message,
119
- errorDetails: error?.details, // mailgun.js errors often have a details field
120
- errorStatusCode: error?.status, // and a status field for HTTP status code
121
- stack: error?.stack,
122
- domain,
123
- to: data.to,
124
- }
125
- );
126
- // Re-throw a more specific error or the original
127
- const mailgunError = new Error(
128
- `Mailgun API Error (${error?.status || "unknown"}): ${
129
- error?.details || error?.message || "Failed to send email"
130
- }`
131
- );
132
- (mailgunError as any).originalError = error;
133
- throw mailgunError;
134
- }
135
- }
136
-
137
- /**
138
- * Logs email sending attempt to Firestore for tracking
139
- * @param emailData Email data that was sent
140
- * @param success Whether the email was sent successfully
141
- * @param error Error object if the email failed to send
142
- */
143
- protected async logEmailAttempt(
144
- emailData: { to: string; subject: string; templateName?: string },
145
- success: boolean,
146
- error?: any
147
- ): Promise<void> {
148
- try {
149
- const emailLogRef = this.db.collection("email_logs").doc();
150
- await emailLogRef.set({
151
- to: emailData.to,
152
- subject: emailData.subject,
153
- templateName: emailData.templateName,
154
- success,
155
- error: error
156
- ? {
157
- message: error.message,
158
- details: (error as any).details,
159
- status: (error as any).status,
160
- stack: error.stack,
161
- }
162
- : null,
163
- sentAt: admin.firestore.FieldValue.serverTimestamp(),
164
- });
165
-
166
- Logger.info(
167
- `[BaseMailingService] Email log recorded. Success: ${success}`
168
- );
169
- } catch (logError: any) {
170
- Logger.error("[BaseMailingService] Error logging email attempt:", {
171
- errorMessage: logError.message,
172
- stack: logError.stack,
173
- });
174
- // Don't throw here to prevent disrupting the main flow
175
- }
176
- }
177
-
178
- /**
179
- * Renders a simple HTML email template with variables
180
- * @param template HTML template string
181
- * @param variables Key-value pairs to replace in the template
182
- * @returns Rendered HTML string
183
- */
184
- protected renderTemplate(
185
- template: string,
186
- variables: Record<string, string>
187
- ): string {
188
- if (!template) {
189
- throw new Error("Email template is required");
190
- }
191
-
192
- try {
193
- let rendered = template;
194
- Object.entries(variables).forEach(([key, value]) => {
195
- const regex = new RegExp(`{{\s*${key}\s*}}`, "g");
196
- rendered = rendered.replace(regex, value || "");
197
- });
198
- return rendered;
199
- } catch (renderError: any) {
200
- Logger.error("[BaseMailingService] Error rendering template:", {
201
- errorMessage: renderError.message,
202
- });
203
- throw new Error(
204
- `Template rendering failed: ${renderError.message || "Unknown error"}`
205
- );
206
- }
207
- }
208
- }
1
+ import * as admin from "firebase-admin";
2
+ // Import the new Mailgun types if available, or use a generic 'any' if not directly importable
3
+ // For a shared library, it's often better to define a minimal interface for what you need
4
+ // or expect the caller (Cloud Function) to pass a pre-configured client of 'any' type.
5
+ // import Mailgun from "mailgun.js"; // This might not be a direct dep of the API pkg
6
+ import { Logger } from "../logger"; // Assuming logger is in Api/src/logger
7
+
8
+ /**
9
+ * Minimal interface for the new mailgun.js client's messages API
10
+ * This helps avoid a direct dependency on mailgun.js in the API package if not desired,
11
+ * or provides clearer typing if it is.
12
+ */
13
+ interface NewMailgunMessagesAPI {
14
+ create(domain: string, data: any): Promise<any>;
15
+ }
16
+
17
+ export interface NewMailgunClient {
18
+ messages: NewMailgunMessagesAPI;
19
+ // Add other methods/properties if your BaseMailingService uses them
20
+ }
21
+
22
+ /**
23
+ * Base mailing service class that provides common functionality for all mailing services
24
+ * This version is updated to expect a mailgun.js v10+ client.
25
+ */
26
+ export class BaseMailingService {
27
+ protected db: FirebaseFirestore.Firestore;
28
+ protected mailgunClient: NewMailgunClient; // Expecting the new mailgun.js client
29
+
30
+ /**
31
+ * Constructor for BaseMailingService
32
+ * @param firestore Firestore instance provided by the caller
33
+ * @param mailgunClient Mailgun client instance (mailgun.js v10+) provided by the caller
34
+ */
35
+ constructor(
36
+ firestore: FirebaseFirestore.Firestore,
37
+ mailgunClient: NewMailgunClient // Expect the new client
38
+ ) {
39
+ this.db = firestore;
40
+ this.mailgunClient = mailgunClient;
41
+
42
+ if (!this.db) {
43
+ Logger.error("[BaseMailingService] No Firestore instance provided");
44
+ throw new Error("Firestore instance is required");
45
+ }
46
+
47
+ if (
48
+ !this.mailgunClient ||
49
+ typeof this.mailgunClient.messages?.create !== "function"
50
+ ) {
51
+ Logger.error(
52
+ "[BaseMailingService] Invalid Mailgun client provided (must be mailgun.js v10+ client)"
53
+ );
54
+ throw new Error("A valid mailgun.js v10+ client is required");
55
+ }
56
+
57
+ // The new client is configured with the URL (e.g., EU endpoint) by the caller.
58
+ // Validation of this configuration can be logged if details are accessible,
59
+ // but the primary responsibility for correct endpoint is now with client setup.
60
+ Logger.info(
61
+ "[BaseMailingService] Service initialized with mailgun.js client."
62
+ );
63
+ }
64
+
65
+ /**
66
+ * Sends an email using the new Mailgun client API.
67
+ * @param domain The Mailgun domain to send from (e.g., mg.metaesthetics.net)
68
+ * @param data Email data to send, compatible with mailgun.js messages.create API
69
+ * @returns Promise with the sending result
70
+ */
71
+ protected async sendEmail(
72
+ domain: string, // The new API requires the domain as the first argument to messages.create
73
+ data: any // Data format according to mailgun.js messages.create
74
+ ): Promise<any> {
75
+ if (!domain) {
76
+ throw new Error("Mailgun domain is required for sending email.");
77
+ }
78
+ if (!data) {
79
+ throw new Error("Email data object is required");
80
+ }
81
+ if (!data.to) {
82
+ throw new Error("Email 'to' address is required");
83
+ }
84
+ if (!data.from) {
85
+ throw new Error("Email 'from' address is required");
86
+ }
87
+ if (!data.subject) {
88
+ throw new Error("Email 'subject' is required");
89
+ }
90
+ if (!data.html && !data.text) {
91
+ throw new Error("Email must have either 'html' or 'text' content");
92
+ }
93
+
94
+ Logger.info(
95
+ "[BaseMailingService] Sending email via Mailgun (mailgun.js API)",
96
+ {
97
+ domain,
98
+ to: data.to,
99
+ from: data.from,
100
+ subject: data.subject,
101
+ hasHtml: !!data.html,
102
+ hasText: !!data.text,
103
+ }
104
+ );
105
+
106
+ try {
107
+ // Use the new mailgun.js client API: client.messages.create(domain, data)
108
+ const result = await this.mailgunClient.messages.create(domain, data);
109
+ Logger.info(
110
+ "[BaseMailingService] Email sent successfully with mailgun.js client",
111
+ result
112
+ );
113
+ return result;
114
+ } catch (error: any) {
115
+ Logger.error(
116
+ "[BaseMailingService] Error sending email with mailgun.js client",
117
+ {
118
+ errorMessage: error?.message,
119
+ errorDetails: error?.details, // mailgun.js errors often have a details field
120
+ errorStatusCode: error?.status, // and a status field for HTTP status code
121
+ stack: error?.stack,
122
+ domain,
123
+ to: data.to,
124
+ }
125
+ );
126
+ // Re-throw a more specific error or the original
127
+ const mailgunError = new Error(
128
+ `Mailgun API Error (${error?.status || "unknown"}): ${
129
+ error?.details || error?.message || "Failed to send email"
130
+ }`
131
+ );
132
+ (mailgunError as any).originalError = error;
133
+ throw mailgunError;
134
+ }
135
+ }
136
+
137
+ /**
138
+ * Logs email sending attempt to Firestore for tracking
139
+ * @param emailData Email data that was sent
140
+ * @param success Whether the email was sent successfully
141
+ * @param error Error object if the email failed to send
142
+ */
143
+ protected async logEmailAttempt(
144
+ emailData: { to: string; subject: string; templateName?: string },
145
+ success: boolean,
146
+ error?: any
147
+ ): Promise<void> {
148
+ try {
149
+ const emailLogRef = this.db.collection("email_logs").doc();
150
+ await emailLogRef.set({
151
+ to: emailData.to,
152
+ subject: emailData.subject,
153
+ templateName: emailData.templateName,
154
+ success,
155
+ error: error
156
+ ? {
157
+ message: error.message,
158
+ details: (error as any).details,
159
+ status: (error as any).status,
160
+ stack: error.stack,
161
+ }
162
+ : null,
163
+ sentAt: admin.firestore.FieldValue.serverTimestamp(),
164
+ });
165
+
166
+ Logger.info(
167
+ `[BaseMailingService] Email log recorded. Success: ${success}`
168
+ );
169
+ } catch (logError: any) {
170
+ Logger.error("[BaseMailingService] Error logging email attempt:", {
171
+ errorMessage: logError.message,
172
+ stack: logError.stack,
173
+ });
174
+ // Don't throw here to prevent disrupting the main flow
175
+ }
176
+ }
177
+
178
+ /**
179
+ * Renders a simple HTML email template with variables
180
+ * @param template HTML template string
181
+ * @param variables Key-value pairs to replace in the template
182
+ * @returns Rendered HTML string
183
+ */
184
+ protected renderTemplate(
185
+ template: string,
186
+ variables: Record<string, string>
187
+ ): string {
188
+ if (!template) {
189
+ throw new Error("Email template is required");
190
+ }
191
+
192
+ try {
193
+ let rendered = template;
194
+ Object.entries(variables).forEach(([key, value]) => {
195
+ const regex = new RegExp(`{{\s*${key}\s*}}`, "g");
196
+ rendered = rendered.replace(regex, value || "");
197
+ });
198
+ return rendered;
199
+ } catch (renderError: any) {
200
+ Logger.error("[BaseMailingService] Error rendering template:", {
201
+ errorMessage: renderError.message,
202
+ });
203
+ throw new Error(
204
+ `Template rendering failed: ${renderError.message || "Unknown error"}`
205
+ );
206
+ }
207
+ }
208
+ }
@@ -1,3 +1,3 @@
1
- export * from "./base.mailing.service";
2
- export * from "./practitionerInvite";
3
- export * from "./appointment";
1
+ export * from "./base.mailing.service";
2
+ export * from "./practitionerInvite";
3
+ export * from "./appointment";