@blackcode_sa/metaestetics-api 1.12.61 → 1.12.62
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.
- package/dist/admin/index.d.mts +2 -0
- package/dist/admin/index.d.ts +2 -0
- package/dist/admin/index.js +45 -4
- package/dist/admin/index.mjs +45 -4
- package/dist/index.d.mts +2 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +53 -11
- package/dist/index.mjs +53 -11
- package/package.json +119 -119
- package/src/__mocks__/firstore.ts +10 -10
- package/src/admin/aggregation/README.md +79 -79
- package/src/admin/aggregation/appointment/README.md +128 -128
- package/src/admin/aggregation/appointment/appointment.aggregation.service.ts +1844 -1844
- package/src/admin/aggregation/appointment/index.ts +1 -1
- package/src/admin/aggregation/clinic/README.md +52 -52
- package/src/admin/aggregation/clinic/clinic.aggregation.service.ts +703 -703
- package/src/admin/aggregation/clinic/index.ts +1 -1
- package/src/admin/aggregation/forms/README.md +13 -13
- package/src/admin/aggregation/forms/filled-forms.aggregation.service.ts +322 -322
- package/src/admin/aggregation/forms/index.ts +1 -1
- package/src/admin/aggregation/index.ts +8 -8
- package/src/admin/aggregation/patient/README.md +27 -27
- package/src/admin/aggregation/patient/index.ts +1 -1
- package/src/admin/aggregation/patient/patient.aggregation.service.ts +141 -141
- package/src/admin/aggregation/practitioner/README.md +42 -42
- package/src/admin/aggregation/practitioner/index.ts +1 -1
- package/src/admin/aggregation/practitioner/practitioner.aggregation.service.ts +433 -433
- package/src/admin/aggregation/practitioner-invite/index.ts +1 -1
- package/src/admin/aggregation/practitioner-invite/practitioner-invite.aggregation.service.ts +961 -961
- package/src/admin/aggregation/procedure/README.md +43 -43
- package/src/admin/aggregation/procedure/index.ts +1 -1
- package/src/admin/aggregation/procedure/procedure.aggregation.service.ts +702 -702
- package/src/admin/aggregation/reviews/index.ts +1 -1
- package/src/admin/aggregation/reviews/reviews.aggregation.service.ts +689 -641
- package/src/admin/booking/README.md +125 -125
- package/src/admin/booking/booking.admin.ts +1037 -1037
- package/src/admin/booking/booking.calculator.ts +712 -712
- package/src/admin/booking/booking.types.ts +59 -59
- package/src/admin/booking/index.ts +3 -3
- package/src/admin/booking/timezones-problem.md +185 -185
- package/src/admin/calendar/README.md +7 -7
- package/src/admin/calendar/calendar.admin.service.ts +345 -345
- package/src/admin/calendar/index.ts +1 -1
- package/src/admin/documentation-templates/document-manager.admin.ts +260 -260
- package/src/admin/documentation-templates/index.ts +1 -1
- package/src/admin/free-consultation/free-consultation-utils.admin.ts +148 -148
- package/src/admin/free-consultation/index.ts +1 -1
- package/src/admin/index.ts +75 -75
- package/src/admin/logger/index.ts +78 -78
- package/src/admin/mailing/README.md +95 -95
- package/src/admin/mailing/appointment/appointment.mailing.service.ts +732 -732
- package/src/admin/mailing/appointment/index.ts +1 -1
- package/src/admin/mailing/appointment/templates/patient/appointment-confirmed.html +40 -40
- package/src/admin/mailing/base.mailing.service.ts +208 -208
- package/src/admin/mailing/index.ts +3 -3
- package/src/admin/mailing/practitionerInvite/existing-practitioner-invite.mailing.ts +611 -611
- package/src/admin/mailing/practitionerInvite/index.ts +2 -2
- package/src/admin/mailing/practitionerInvite/practitionerInvite.mailing.ts +395 -395
- package/src/admin/mailing/practitionerInvite/templates/existing-practitioner-invitation.template.ts +155 -155
- package/src/admin/mailing/practitionerInvite/templates/invitation.template.ts +101 -101
- package/src/admin/mailing/practitionerInvite/templates/invite-accepted-notification.template.ts +228 -228
- package/src/admin/mailing/practitionerInvite/templates/invite-rejected-notification.template.ts +242 -242
- package/src/admin/notifications/index.ts +1 -1
- package/src/admin/notifications/notifications.admin.ts +710 -710
- package/src/admin/requirements/README.md +128 -128
- package/src/admin/requirements/index.ts +1 -1
- package/src/admin/requirements/patient-requirements.admin.service.ts +475 -475
- package/src/admin/users/index.ts +1 -1
- package/src/admin/users/user-profile.admin.ts +405 -405
- package/src/backoffice/constants/certification.constants.ts +13 -13
- package/src/backoffice/constants/index.ts +1 -1
- package/src/backoffice/errors/backoffice.errors.ts +181 -181
- package/src/backoffice/errors/index.ts +1 -1
- package/src/backoffice/expo-safe/README.md +26 -26
- package/src/backoffice/expo-safe/index.ts +41 -41
- package/src/backoffice/index.ts +5 -5
- package/src/backoffice/services/FIXES_README.md +102 -102
- package/src/backoffice/services/README.md +40 -40
- package/src/backoffice/services/brand.service.ts +256 -256
- package/src/backoffice/services/category.service.ts +318 -318
- package/src/backoffice/services/constants.service.ts +385 -385
- package/src/backoffice/services/documentation-template.service.ts +202 -202
- package/src/backoffice/services/index.ts +8 -8
- package/src/backoffice/services/migrate-products.ts +116 -116
- package/src/backoffice/services/product.service.ts +553 -553
- package/src/backoffice/services/requirement.service.ts +235 -235
- package/src/backoffice/services/subcategory.service.ts +395 -395
- package/src/backoffice/services/technology.service.ts +1070 -1070
- package/src/backoffice/types/README.md +12 -12
- package/src/backoffice/types/admin-constants.types.ts +69 -69
- package/src/backoffice/types/brand.types.ts +29 -29
- package/src/backoffice/types/category.types.ts +62 -62
- package/src/backoffice/types/documentation-templates.types.ts +28 -28
- package/src/backoffice/types/index.ts +10 -10
- package/src/backoffice/types/procedure-product.types.ts +38 -38
- package/src/backoffice/types/product.types.ts +240 -240
- package/src/backoffice/types/requirement.types.ts +63 -63
- package/src/backoffice/types/static/README.md +18 -18
- package/src/backoffice/types/static/blocking-condition.types.ts +21 -21
- package/src/backoffice/types/static/certification.types.ts +37 -37
- package/src/backoffice/types/static/contraindication.types.ts +19 -19
- package/src/backoffice/types/static/index.ts +6 -6
- package/src/backoffice/types/static/pricing.types.ts +16 -16
- package/src/backoffice/types/static/procedure-family.types.ts +14 -14
- package/src/backoffice/types/static/treatment-benefit.types.ts +22 -22
- package/src/backoffice/types/subcategory.types.ts +34 -34
- package/src/backoffice/types/technology.types.ts +161 -161
- package/src/backoffice/validations/index.ts +1 -1
- package/src/backoffice/validations/schemas.ts +163 -163
- package/src/config/__mocks__/firebase.ts +99 -99
- package/src/config/firebase.ts +78 -78
- package/src/config/index.ts +9 -9
- package/src/errors/auth.error.ts +6 -6
- package/src/errors/auth.errors.ts +200 -200
- package/src/errors/clinic.errors.ts +32 -32
- package/src/errors/firebase.errors.ts +47 -47
- package/src/errors/user.errors.ts +99 -99
- package/src/index.backup.ts +407 -407
- package/src/index.ts +6 -6
- package/src/locales/en.ts +31 -31
- package/src/recommender/admin/index.ts +1 -1
- package/src/recommender/admin/services/recommender.service.admin.ts +5 -5
- package/src/recommender/front/index.ts +1 -1
- package/src/recommender/front/services/onboarding.service.ts +5 -5
- package/src/recommender/front/services/recommender.service.ts +3 -3
- package/src/recommender/index.ts +1 -1
- package/src/services/PATIENTAUTH.MD +197 -197
- package/src/services/README.md +106 -106
- package/src/services/__tests__/auth/auth.mock.test.ts +17 -17
- package/src/services/__tests__/auth/auth.setup.ts +293 -293
- package/src/services/__tests__/auth.service.test.ts +346 -346
- package/src/services/__tests__/base.service.test.ts +77 -77
- package/src/services/__tests__/user.service.test.ts +528 -528
- package/src/services/appointment/README.md +17 -17
- package/src/services/appointment/appointment.service.ts +2082 -2082
- package/src/services/appointment/index.ts +1 -1
- package/src/services/appointment/utils/appointment.utils.ts +552 -552
- package/src/services/appointment/utils/extended-procedure.utils.ts +314 -314
- package/src/services/appointment/utils/form-initialization.utils.ts +225 -225
- package/src/services/appointment/utils/recommended-procedure.utils.ts +195 -195
- package/src/services/appointment/utils/zone-management.utils.ts +353 -353
- package/src/services/appointment/utils/zone-photo.utils.ts +152 -152
- package/src/services/auth/auth.service.ts +989 -989
- package/src/services/auth/auth.v2.service.ts +961 -961
- package/src/services/auth/index.ts +7 -7
- package/src/services/auth/utils/error.utils.ts +90 -90
- package/src/services/auth/utils/firebase.utils.ts +49 -49
- package/src/services/auth/utils/index.ts +21 -21
- package/src/services/auth/utils/practitioner.utils.ts +125 -125
- package/src/services/base.service.ts +41 -41
- package/src/services/calendar/calendar.service.ts +1077 -1077
- package/src/services/calendar/calendar.v2.service.ts +1683 -1683
- package/src/services/calendar/calendar.v3.service.ts +313 -313
- package/src/services/calendar/externalCalendar.service.ts +178 -178
- package/src/services/calendar/index.ts +5 -5
- package/src/services/calendar/synced-calendars.service.ts +743 -743
- package/src/services/calendar/utils/appointment.utils.ts +265 -265
- package/src/services/calendar/utils/calendar-event.utils.ts +646 -646
- package/src/services/calendar/utils/clinic.utils.ts +237 -237
- package/src/services/calendar/utils/docs.utils.ts +157 -157
- package/src/services/calendar/utils/google-calendar.utils.ts +697 -697
- package/src/services/calendar/utils/index.ts +8 -8
- package/src/services/calendar/utils/patient.utils.ts +198 -198
- package/src/services/calendar/utils/practitioner.utils.ts +221 -221
- package/src/services/calendar/utils/synced-calendar.utils.ts +472 -472
- package/src/services/clinic/README.md +204 -204
- package/src/services/clinic/__tests__/clinic-admin.service.test.ts +287 -287
- package/src/services/clinic/__tests__/clinic-group.service.test.ts +352 -352
- package/src/services/clinic/__tests__/clinic.service.test.ts +354 -354
- package/src/services/clinic/billing-transactions.service.ts +217 -217
- package/src/services/clinic/clinic-admin.service.ts +202 -202
- package/src/services/clinic/clinic-group.service.ts +310 -310
- package/src/services/clinic/clinic.service.ts +708 -708
- package/src/services/clinic/index.ts +5 -5
- package/src/services/clinic/practitioner-invite.service.ts +519 -519
- package/src/services/clinic/utils/admin.utils.ts +551 -551
- package/src/services/clinic/utils/clinic-group.utils.ts +646 -646
- package/src/services/clinic/utils/clinic.utils.ts +949 -949
- package/src/services/clinic/utils/filter.utils.d.ts +23 -23
- package/src/services/clinic/utils/filter.utils.ts +446 -446
- package/src/services/clinic/utils/index.ts +11 -11
- package/src/services/clinic/utils/photos.utils.ts +188 -188
- package/src/services/clinic/utils/search.utils.ts +84 -84
- package/src/services/clinic/utils/tag.utils.ts +124 -124
- package/src/services/documentation-templates/documentation-template.service.ts +537 -537
- package/src/services/documentation-templates/filled-document.service.ts +587 -587
- package/src/services/documentation-templates/index.ts +2 -2
- package/src/services/index.ts +13 -13
- package/src/services/media/index.ts +1 -1
- package/src/services/media/media.service.ts +418 -418
- package/src/services/notifications/__tests__/notification.service.test.ts +242 -242
- package/src/services/notifications/index.ts +1 -1
- package/src/services/notifications/notification.service.ts +215 -215
- package/src/services/patient/README.md +48 -48
- package/src/services/patient/To-Do.md +43 -43
- package/src/services/patient/__tests__/patient.service.test.ts +294 -294
- package/src/services/patient/index.ts +2 -2
- package/src/services/patient/patient.service.ts +883 -883
- package/src/services/patient/patientRequirements.service.ts +285 -285
- package/src/services/patient/utils/aesthetic-analysis.utils.ts +176 -176
- package/src/services/patient/utils/clinic.utils.ts +80 -80
- package/src/services/patient/utils/docs.utils.ts +142 -142
- package/src/services/patient/utils/index.ts +9 -9
- package/src/services/patient/utils/location.utils.ts +126 -126
- package/src/services/patient/utils/medical-stuff.utils.ts +143 -143
- package/src/services/patient/utils/medical.utils.ts +458 -458
- package/src/services/patient/utils/practitioner.utils.ts +260 -260
- package/src/services/patient/utils/profile.utils.ts +510 -510
- package/src/services/patient/utils/sensitive.utils.ts +260 -260
- package/src/services/patient/utils/token.utils.ts +211 -211
- package/src/services/practitioner/README.md +145 -145
- package/src/services/practitioner/index.ts +1 -1
- package/src/services/practitioner/practitioner.service.ts +1742 -1742
- package/src/services/procedure/README.md +163 -163
- package/src/services/procedure/index.ts +1 -1
- package/src/services/procedure/procedure.service.ts +1682 -1682
- package/src/services/reviews/index.ts +1 -1
- package/src/services/reviews/reviews.service.ts +683 -636
- package/src/services/user/index.ts +1 -1
- package/src/services/user/user.service.ts +489 -489
- package/src/services/user/user.v2.service.ts +466 -466
- package/src/types/appointment/index.ts +453 -453
- package/src/types/calendar/index.ts +258 -258
- package/src/types/calendar/synced-calendar.types.ts +66 -66
- package/src/types/clinic/index.ts +489 -489
- package/src/types/clinic/practitioner-invite.types.ts +91 -91
- package/src/types/clinic/preferences.types.ts +159 -159
- package/src/types/clinic/to-do +3 -3
- package/src/types/documentation-templates/index.ts +308 -308
- package/src/types/index.ts +44 -44
- package/src/types/notifications/README.md +77 -77
- package/src/types/notifications/index.ts +265 -265
- package/src/types/patient/aesthetic-analysis.types.ts +66 -66
- package/src/types/patient/allergies.ts +58 -58
- package/src/types/patient/index.ts +273 -273
- package/src/types/patient/medical-info.types.ts +152 -152
- package/src/types/patient/patient-requirements.ts +92 -92
- package/src/types/patient/token.types.ts +61 -61
- package/src/types/practitioner/index.ts +206 -206
- package/src/types/procedure/index.ts +181 -181
- package/src/types/profile/index.ts +39 -39
- package/src/types/reviews/index.ts +132 -130
- package/src/types/tz-lookup.d.ts +4 -4
- package/src/types/user/index.ts +38 -38
- package/src/utils/TIMESTAMPS.md +176 -176
- package/src/utils/TimestampUtils.ts +241 -241
- package/src/utils/index.ts +1 -1
- package/src/validations/appointment.schema.ts +574 -574
- package/src/validations/calendar.schema.ts +225 -225
- package/src/validations/clinic.schema.ts +493 -493
- package/src/validations/common.schema.ts +25 -25
- package/src/validations/documentation-templates/index.ts +1 -1
- package/src/validations/documentation-templates/template.schema.ts +220 -220
- package/src/validations/documentation-templates.schema.ts +10 -10
- package/src/validations/index.ts +20 -20
- package/src/validations/media.schema.ts +10 -10
- package/src/validations/notification.schema.ts +90 -90
- package/src/validations/patient/aesthetic-analysis.schema.ts +55 -55
- package/src/validations/patient/medical-info.schema.ts +125 -125
- package/src/validations/patient/patient-requirements.schema.ts +84 -84
- package/src/validations/patient/token.schema.ts +29 -29
- package/src/validations/patient.schema.ts +216 -216
- package/src/validations/practitioner.schema.ts +222 -222
- package/src/validations/procedure-product.schema.ts +41 -41
- package/src/validations/procedure.schema.ts +124 -124
- package/src/validations/profile-info.schema.ts +41 -41
- package/src/validations/reviews.schema.ts +195 -189
- package/src/validations/schemas.ts +104 -104
- 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";
|