@blackcode_sa/metaestetics-api 1.12.65 → 1.12.67

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 +2 -0
  2. package/dist/admin/index.d.ts +2 -0
  3. package/dist/admin/index.js +45 -4
  4. package/dist/admin/index.mjs +45 -4
  5. package/dist/backoffice/index.d.mts +33 -0
  6. package/dist/backoffice/index.d.ts +33 -0
  7. package/dist/backoffice/index.js +63 -0
  8. package/dist/backoffice/index.mjs +63 -0
  9. package/dist/index.d.mts +35 -0
  10. package/dist/index.d.ts +35 -0
  11. package/dist/index.js +116 -11
  12. package/dist/index.mjs +116 -11
  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 +689 -641
  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 +341 -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 +10 -10
  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 +417 -395
  92. package/src/backoffice/services/technology.service.ts +1104 -1083
  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 +67 -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 +168 -163
  112. package/src/backoffice/validations/index.ts +1 -1
  113. package/src/backoffice/validations/schemas.ts +164 -164
  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 -2505
  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 +1715 -1715
  221. package/src/services/reviews/index.ts +1 -1
  222. package/src/services/reviews/reviews.service.ts +683 -636
  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 +480 -480
  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 -275
  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 +132 -130
  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 -217
  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 +195 -189
  272. package/src/validations/schemas.ts +104 -104
  273. package/src/validations/shared.schema.ts +78 -78
@@ -1,574 +1,574 @@
1
- import { z } from 'zod';
2
- import { AppointmentStatus, PaymentStatus, MediaType } from '../types/appointment';
3
- import { filledDocumentStatusSchema } from './documentation-templates.schema';
4
- import { Currency, PricingMeasure } from '../backoffice/types/static/pricing.types';
5
- import { mediaResourceSchema } from './media.schema';
6
-
7
- // Define common constants locally if not available from common.schema.ts
8
- const MIN_STRING_LENGTH = 1;
9
- const MAX_STRING_LENGTH = 1024; // Example value
10
- const MAX_STRING_LENGTH_LONG = 4096; // Example value for longer notes/comments
11
- const MAX_ARRAY_LENGTH = 100; // Example value
12
-
13
- // --- Enum Schemas ---
14
- export const appointmentStatusSchema = z.nativeEnum(AppointmentStatus);
15
- export const paymentStatusSchema = z.nativeEnum(PaymentStatus);
16
- export const mediaTypeSchema = z.nativeEnum(MediaType);
17
-
18
- // --- Schemas for Nested Objects from types/appointment/index.ts ---
19
-
20
- export const appointmentMediaItemSchema = z.object({
21
- id: z.string().min(MIN_STRING_LENGTH, 'Media item ID is required'),
22
- type: mediaTypeSchema,
23
- url: z.string().url('Media URL must be a valid URL'),
24
- fileName: z.string().optional(),
25
- uploadedAt: z
26
- .any()
27
- .refine(
28
- val =>
29
- val instanceof Date ||
30
- val?._seconds !== undefined ||
31
- val?.seconds !== undefined ||
32
- typeof val === 'number' ||
33
- typeof val === 'string' ||
34
- (val && typeof val.toMillis === 'function'),
35
- 'uploadedAt must be a valid timestamp or Date object',
36
- ),
37
- uploadedBy: z.string().min(MIN_STRING_LENGTH, 'Uploaded by user ID is required'),
38
- description: z.string().max(MAX_STRING_LENGTH, 'Description too long').optional(),
39
- });
40
-
41
- export const procedureExtendedInfoSchema = z.object({
42
- id: z.string().min(MIN_STRING_LENGTH),
43
- name: z.string().min(MIN_STRING_LENGTH),
44
- description: z.string(),
45
- cost: z.number().min(0),
46
- duration: z.number().min(0),
47
- procedureFamily: z.any(),
48
- procedureCategoryId: z.string(),
49
- procedureCategoryName: z.string(),
50
- procedureSubCategoryId: z.string(),
51
- procedureSubCategoryName: z.string(),
52
- procedureTechnologyId: z.string(),
53
- procedureTechnologyName: z.string(),
54
- procedureProductBrandId: z.string(),
55
- procedureProductBrandName: z.string(),
56
- procedureProducts: z.array(
57
- z.object({
58
- productId: z.string().min(MIN_STRING_LENGTH, 'Product ID is required'),
59
- productName: z.string().min(MIN_STRING_LENGTH, 'Product name is required'),
60
- brandId: z.string().min(MIN_STRING_LENGTH, 'Brand ID is required'),
61
- brandName: z.string().min(MIN_STRING_LENGTH, 'Brand name is required'),
62
- }),
63
- ),
64
- });
65
-
66
- export const linkedFormInfoSchema = z.object({
67
- formId: z.string().min(MIN_STRING_LENGTH, 'Form ID is required'),
68
- templateId: z.string().min(MIN_STRING_LENGTH, 'Template ID is required'),
69
- templateVersion: z.number().int().positive('Template version must be a positive integer'),
70
- title: z.string().min(MIN_STRING_LENGTH, 'Form title is required'),
71
- isUserForm: z.boolean(),
72
- isRequired: z.boolean().optional(),
73
- status: filledDocumentStatusSchema,
74
- path: z.string().min(MIN_STRING_LENGTH, 'Form path is required'),
75
- submittedAt: z
76
- .any()
77
- .refine(
78
- val =>
79
- val === undefined ||
80
- val instanceof Date ||
81
- val?._seconds !== undefined ||
82
- val?.seconds !== undefined ||
83
- typeof val === 'number' ||
84
- typeof val === 'string' ||
85
- (val && typeof val.toMillis === 'function'),
86
- 'submittedAt must be a valid timestamp or Date object',
87
- )
88
- .optional(),
89
- completedAt: z
90
- .any()
91
- .refine(
92
- val =>
93
- val === undefined ||
94
- val instanceof Date ||
95
- val?._seconds !== undefined ||
96
- val?.seconds !== undefined ||
97
- typeof val === 'number' ||
98
- typeof val === 'string' ||
99
- (val && typeof val.toMillis === 'function'),
100
- 'completedAt must be a valid timestamp or Date object',
101
- )
102
- .optional(),
103
- });
104
-
105
- export const patientReviewInfoSchema = z.object({
106
- reviewId: z.string().min(MIN_STRING_LENGTH, 'Review ID is required'),
107
- rating: z.number().min(1).max(5, 'Rating must be between 1 and 5'),
108
- comment: z.string().max(MAX_STRING_LENGTH_LONG, 'Comment too long').optional(),
109
- reviewedAt: z
110
- .any()
111
- .refine(
112
- val =>
113
- val instanceof Date ||
114
- val?._seconds !== undefined ||
115
- val?.seconds !== undefined ||
116
- typeof val === 'number' ||
117
- typeof val === 'string' ||
118
- (val && typeof val.toMillis === 'function'),
119
- 'reviewedAt must be a valid timestamp or Date object',
120
- ),
121
- });
122
-
123
- export const finalizedDetailsSchema = z.object({
124
- by: z.string().min(MIN_STRING_LENGTH, 'Finalized by user ID is required'),
125
- at: z
126
- .any()
127
- .refine(
128
- val =>
129
- val instanceof Date ||
130
- val?._seconds !== undefined ||
131
- val?.seconds !== undefined ||
132
- typeof val === 'number' ||
133
- typeof val === 'string' ||
134
- (val && typeof val.toMillis === 'function'),
135
- 'Finalized at must be a valid timestamp or Date object',
136
- ),
137
- notes: z.string().max(MAX_STRING_LENGTH_LONG, 'Finalization notes too long').optional(),
138
- });
139
-
140
- /**
141
- * Schema for before/after photos and notes per zone
142
- */
143
- export const beforeAfterPerZoneSchema = z.object({
144
- before: mediaResourceSchema.nullable(),
145
- after: mediaResourceSchema.nullable(),
146
- afterNote: z.string().nullable().optional(),
147
- beforeNote: z.string().nullable().optional(),
148
- });
149
-
150
- /**
151
- * Schema for billing information per zone
152
- */
153
- export const billingPerZoneSchema = z.object({
154
- Product: z.string().min(MIN_STRING_LENGTH, 'Product name is required'),
155
- ProductId: z.string().nullable(),
156
- Quantity: z.number().min(0, 'Quantity must be non-negative'),
157
- UnitOfMeasurement: z.nativeEnum(PricingMeasure),
158
- UnitPrice: z.number().min(0, 'Unit price must be non-negative'),
159
- UnitCurency: z.nativeEnum(Currency),
160
- Subtotal: z.number().min(0, 'Subtotal must be non-negative'),
161
- Note: z.string().nullable(),
162
- IonNumber: z.string().nullable(),
163
- });
164
-
165
- /**
166
- * Schema for final billing calculations of the appointment
167
- */
168
- export const finalBillingSchema = z.object({
169
- subtotalAll: z.number().min(0, 'Subtotal all must be non-negative'),
170
- taxRate: z.number().min(0).max(1, 'Tax rate must be between 0 and 1'),
171
- taxPrice: z.number().min(0, 'Tax price must be non-negative'),
172
- finalPrice: z.number().min(0, 'Final price must be non-negative'),
173
- currency: z.nativeEnum(Currency),
174
- });
175
-
176
- /**
177
- * Schema for zone item data (product or note per zone)
178
- */
179
- export const zoneItemDataSchema = z
180
- .object({
181
- productId: z.string().optional(),
182
- productName: z.string().optional(),
183
- productBrandId: z.string().optional(),
184
- productBrandName: z.string().optional(),
185
- belongingProcedureId: z.string().min(MIN_STRING_LENGTH, 'Belonging procedure ID is required'),
186
- type: z.enum(['item', 'note'], {
187
- required_error: 'Type must be either "item" or "note"',
188
- }),
189
- stage: z
190
- .enum(['before', 'after'], {
191
- required_error: 'Stage must be either "before" or "after"',
192
- })
193
- .optional(),
194
- price: z.number().min(0, 'Price must be non-negative').optional(),
195
- currency: z.nativeEnum(Currency).optional(),
196
- unitOfMeasurement: z.nativeEnum(PricingMeasure).optional(),
197
- priceOverrideAmount: z.number().min(0, 'Price override amount must be non-negative').optional(),
198
- quantity: z.number().min(0, 'Quantity must be non-negative').optional(),
199
- parentZone: z
200
- .string()
201
- .min(MIN_STRING_LENGTH, 'Parent zone is required')
202
- .refine(
203
- val => {
204
- const parts = val.split('.');
205
- return parts.length === 2;
206
- },
207
- {
208
- message: 'Parent zone must be in "category.zone" format (e.g., "face.forehead")',
209
- },
210
- ),
211
- subzones: z.array(z.string()).refine(
212
- val => {
213
- return val.every(subzone => {
214
- const parts = subzone.split('.');
215
- return parts.length === 3;
216
- });
217
- },
218
- {
219
- message: 'Subzones must be in "category.zone.subzone" format (e.g., "face.forehead.left")',
220
- },
221
- ),
222
- notes: z.string().max(MAX_STRING_LENGTH_LONG, 'Notes too long').optional(),
223
- subtotal: z.number().min(0, 'Subtotal must be non-negative').optional(),
224
- ionNumber: z.string().optional(),
225
- createdAt: z.string().optional(),
226
- updatedAt: z.string().optional(),
227
- })
228
- .refine(
229
- data => {
230
- if (data.type === 'item') {
231
- return !!(data.productId && data.productName && (data.price || data.priceOverrideAmount));
232
- }
233
- return true;
234
- },
235
- {
236
- message: 'Item type requires productId, productName, and either price or priceOverrideAmount',
237
- },
238
- );
239
-
240
- /**
241
- * Schema for appointment product metadata
242
- */
243
- export const appointmentProductMetadataSchema = z.object({
244
- productId: z.string().min(MIN_STRING_LENGTH, 'Product ID is required'),
245
- productName: z.string().min(MIN_STRING_LENGTH, 'Product name is required'),
246
- brandId: z.string().min(MIN_STRING_LENGTH, 'Brand ID is required'),
247
- brandName: z.string().min(MIN_STRING_LENGTH, 'Brand name is required'),
248
- procedureId: z.string().min(MIN_STRING_LENGTH, 'Procedure ID is required'),
249
- price: z.number().min(0, 'Price must be non-negative'),
250
- currency: z.nativeEnum(Currency),
251
- unitOfMeasurement: z.nativeEnum(PricingMeasure),
252
- });
253
-
254
- /**
255
- * Schema for extended procedure info
256
- */
257
- export const extendedProcedureInfoSchema = z.object({
258
- procedureId: z.string().min(MIN_STRING_LENGTH, 'Procedure ID is required'),
259
- procedureName: z.string().min(MIN_STRING_LENGTH, 'Procedure name is required'),
260
- procedureFamily: z.any(),
261
- procedureCategoryId: z.string().min(MIN_STRING_LENGTH, 'Category ID is required'),
262
- procedureCategoryName: z.string().min(MIN_STRING_LENGTH, 'Category name is required'),
263
- procedureSubCategoryId: z.string().min(MIN_STRING_LENGTH, 'Subcategory ID is required'),
264
- procedureSubCategoryName: z.string().min(MIN_STRING_LENGTH, 'Subcategory name is required'),
265
- procedureTechnologyId: z.string().min(MIN_STRING_LENGTH, 'Technology ID is required'),
266
- procedureTechnologyName: z.string().min(MIN_STRING_LENGTH, 'Technology name is required'),
267
- procedureProducts: z.array(
268
- z.object({
269
- productId: z.string().min(MIN_STRING_LENGTH, 'Product ID is required'),
270
- productName: z.string().min(MIN_STRING_LENGTH, 'Product name is required'),
271
- brandId: z.string().min(MIN_STRING_LENGTH, 'Brand ID is required'),
272
- brandName: z.string().min(MIN_STRING_LENGTH, 'Brand name is required'),
273
- }),
274
- ),
275
- });
276
-
277
- /**
278
- * Schema for recommended procedure within metadata
279
- */
280
- export const recommendedProcedureTimeframeSchema = z.object({
281
- value: z.number().int().positive('Timeframe value must be a positive integer'),
282
- unit: z.enum(['day', 'week', 'month', 'year']),
283
- });
284
-
285
- export const recommendedProcedureSchema = z.object({
286
- procedure: extendedProcedureInfoSchema,
287
- note: z.string().max(MAX_STRING_LENGTH_LONG, 'Note too long'), // Note is now optional (no min length)
288
- timeframe: recommendedProcedureTimeframeSchema,
289
- });
290
-
291
- /**
292
- * Schema for appointment metadata containing zone-specific information
293
- */
294
- export const appointmentMetadataSchema = z.object({
295
- selectedZones: z.array(z.string()).nullable(),
296
- zonePhotos: z.record(z.string(), z.array(beforeAfterPerZoneSchema).max(10)).nullable(),
297
- zonesData: z.record(z.string(), z.array(zoneItemDataSchema)).nullable().optional(),
298
- appointmentProducts: z.array(appointmentProductMetadataSchema).optional().default([]),
299
- extendedProcedures: z.array(extendedProcedureInfoSchema).optional().default([]),
300
- recommendedProcedures: z.array(recommendedProcedureSchema).optional().default([]),
301
- zoneBilling: z.record(z.string(), billingPerZoneSchema).nullable().optional(),
302
- finalbilling: finalBillingSchema.nullable(),
303
- finalizationNotes: z.string().nullable(),
304
- });
305
-
306
- // --- Main Appointment Schemas ---
307
-
308
- /**
309
- * Schema for validating appointment creation data
310
- */
311
- export const createAppointmentSchema = z
312
- .object({
313
- clinicBranchId: z.string().min(MIN_STRING_LENGTH, 'Clinic branch ID is required'),
314
- practitionerId: z.string().min(MIN_STRING_LENGTH, 'Practitioner ID is required'),
315
- patientId: z.string().min(MIN_STRING_LENGTH, 'Patient ID is required'),
316
- procedureId: z.string().min(MIN_STRING_LENGTH, 'Procedure ID is required'),
317
- appointmentStartTime: z
318
- .any()
319
- .refine(
320
- val =>
321
- val instanceof Date ||
322
- val?._seconds !== undefined ||
323
- val?.seconds !== undefined ||
324
- typeof val === 'number' ||
325
- typeof val === 'string' ||
326
- (val && typeof val.toMillis === 'function'),
327
- 'Appointment start time must be a valid timestamp or Date object',
328
- ),
329
- appointmentEndTime: z
330
- .any()
331
- .refine(
332
- val =>
333
- val instanceof Date ||
334
- val?._seconds !== undefined ||
335
- val?.seconds !== undefined ||
336
- typeof val === 'number' ||
337
- typeof val === 'string' ||
338
- (val && typeof val.toMillis === 'function'),
339
- 'Appointment end time must be a valid timestamp or Date object',
340
- ),
341
- cost: z.number().min(0, 'Cost must be a non-negative number'),
342
- currency: z.string().min(1, 'Currency is required'),
343
- patientNotes: z.string().max(MAX_STRING_LENGTH, 'Patient notes too long').nullable().optional(),
344
- initialStatus: appointmentStatusSchema,
345
- initialPaymentStatus: paymentStatusSchema.optional().default(PaymentStatus.UNPAID),
346
- clinic_tz: z.string().min(1, 'Timezone is required'),
347
- })
348
- .refine(data => data.appointmentEndTime > data.appointmentStartTime, {
349
- message: 'Appointment end time must be after start time',
350
- path: ['appointmentEndTime'],
351
- });
352
-
353
- /**
354
- * Schema for validating appointment update data
355
- */
356
- export const updateAppointmentSchema = z
357
- .object({
358
- status: appointmentStatusSchema.optional(),
359
- confirmationTime: z.any().optional().nullable(),
360
- cancellationTime: z.any().optional().nullable(),
361
- rescheduleTime: z.any().optional().nullable(),
362
- procedureActualStartTime: z.any().optional().nullable(),
363
- actualDurationMinutes: z
364
- .number()
365
- .int()
366
- .positive('Duration must be a positive integer')
367
- .optional(),
368
- cancellationReason: z
369
- .string()
370
- .max(MAX_STRING_LENGTH, 'Cancellation reason too long')
371
- .nullable()
372
- .optional(),
373
- canceledBy: z.enum(['patient', 'clinic', 'practitioner', 'system']).optional(),
374
- internalNotes: z
375
- .string()
376
- .max(MAX_STRING_LENGTH_LONG, 'Internal notes too long')
377
- .nullable()
378
- .optional(),
379
- patientNotes: z.any().optional().nullable(),
380
- paymentStatus: paymentStatusSchema.optional(),
381
- paymentTransactionId: z.any().optional().nullable(),
382
- completedPreRequirements: z.union([z.array(z.string()), z.any()]).optional(),
383
- completedPostRequirements: z.union([z.array(z.string()), z.any()]).optional(),
384
- linkedFormIds: z.union([z.array(z.string()), z.any()]).optional(),
385
- pendingUserFormsIds: z.union([z.array(z.string()), z.any()]).optional(),
386
- appointmentStartTime: z
387
- .any()
388
- .refine(
389
- val =>
390
- val === undefined ||
391
- val instanceof Date ||
392
- val?._seconds !== undefined ||
393
- val?.seconds !== undefined ||
394
- typeof val === 'number' ||
395
- typeof val === 'string' ||
396
- (val && typeof val.toMillis === 'function'),
397
- 'Appointment start time must be a valid timestamp or Date object',
398
- )
399
- .optional(),
400
- appointmentEndTime: z
401
- .any()
402
- .refine(
403
- val =>
404
- val === undefined ||
405
- val instanceof Date ||
406
- val?._seconds !== undefined ||
407
- val?.seconds !== undefined ||
408
- typeof val === 'number' ||
409
- typeof val === 'string' ||
410
- (val && typeof val.toMillis === 'function'),
411
- 'Appointment end time must be a valid timestamp or Date object',
412
- )
413
- .optional(),
414
- calendarEventId: z.string().min(MIN_STRING_LENGTH).optional(),
415
- cost: z.number().min(0).optional(),
416
- clinicBranchId: z.string().min(MIN_STRING_LENGTH).optional(),
417
- practitionerId: z.string().min(MIN_STRING_LENGTH).optional(),
418
- clinic_tz: z.string().min(MIN_STRING_LENGTH).optional(),
419
- linkedForms: z.union([z.array(linkedFormInfoSchema).max(MAX_ARRAY_LENGTH), z.any()]).optional(),
420
- media: z.union([z.array(appointmentMediaItemSchema).max(MAX_ARRAY_LENGTH), z.any()]).optional(),
421
- reviewInfo: z.union([patientReviewInfoSchema.nullable(), z.any()]).optional(),
422
- finalizedDetails: z.union([finalizedDetailsSchema.nullable(), z.any()]).optional(),
423
- isArchived: z.boolean().optional(),
424
- updatedAt: z.any().optional(),
425
- metadata: appointmentMetadataSchema.optional(),
426
- })
427
- .refine(
428
- data => {
429
- if (
430
- data.status === AppointmentStatus.CANCELED_CLINIC ||
431
- data.status === AppointmentStatus.CANCELED_PATIENT ||
432
- data.status === AppointmentStatus.CANCELED_PATIENT_RESCHEDULED
433
- ) {
434
- return !!data.cancellationReason && !!data.canceledBy;
435
- }
436
- return true;
437
- },
438
- {
439
- message:
440
- 'Cancellation reason and canceled by must be provided when canceling an appointment with patient or clinic origin.',
441
- path: ['status'],
442
- },
443
- )
444
- .refine(
445
- data => {
446
- if (data.appointmentStartTime && data.appointmentEndTime) {
447
- return data.appointmentEndTime > data.appointmentStartTime;
448
- }
449
- return true;
450
- },
451
- {
452
- message: 'Appointment end time must be after start time if both are provided',
453
- path: ['appointmentEndTime'],
454
- },
455
- );
456
-
457
- /**
458
- * Schema for validating appointment search parameters
459
- */
460
- export const searchAppointmentsSchema = z
461
- .object({
462
- patientId: z.string().optional(),
463
- practitionerId: z.string().optional(),
464
- clinicBranchId: z.string().optional(),
465
- startDate: z
466
- .any()
467
- .refine(
468
- val =>
469
- val === undefined ||
470
- val instanceof Date ||
471
- val?._seconds !== undefined ||
472
- val?.seconds !== undefined ||
473
- typeof val === 'number' ||
474
- typeof val === 'string' ||
475
- (val && typeof val.toMillis === 'function'),
476
- 'Start date must be a valid timestamp or Date object',
477
- )
478
- .optional(),
479
- endDate: z
480
- .any()
481
- .refine(
482
- val =>
483
- val === undefined ||
484
- val instanceof Date ||
485
- val?._seconds !== undefined ||
486
- val?.seconds !== undefined ||
487
- typeof val === 'number' ||
488
- typeof val === 'string' ||
489
- (val && typeof val.toMillis === 'function'),
490
- 'End date must be a valid timestamp or Date object',
491
- )
492
- .optional(),
493
- status: z
494
- .union([appointmentStatusSchema, z.array(appointmentStatusSchema).nonempty()])
495
- .optional(),
496
- limit: z.number().positive().int().optional().default(20),
497
- startAfter: z.any().optional(),
498
- })
499
- .refine(
500
- data => {
501
- if (!data.startDate && !data.endDate && !data.status) {
502
- return !!(data.patientId || data.practitionerId || data.clinicBranchId);
503
- }
504
- return true;
505
- },
506
- {
507
- message:
508
- 'At least one of patientId, practitionerId, or clinicBranchId must be provided if no date or status filters are set.',
509
- path: ['patientId'],
510
- },
511
- )
512
- .refine(
513
- data => {
514
- if (data.startDate && data.endDate) {
515
- return data.endDate >= data.startDate;
516
- }
517
- return true;
518
- },
519
- {
520
- message: 'End date must be after or the same as start date',
521
- path: ['endDate'],
522
- },
523
- );
524
-
525
- /**
526
- * Schema for validating appointment reschedule data
527
- */
528
- export const rescheduleAppointmentSchema = z.object({
529
- appointmentId: z.string().min(MIN_STRING_LENGTH, 'Appointment ID is required'),
530
- newStartTime: z
531
- .any()
532
- .refine(
533
- val =>
534
- val instanceof Date ||
535
- val?._seconds !== undefined ||
536
- val?.seconds !== undefined ||
537
- typeof val === 'number' ||
538
- typeof val === 'string' ||
539
- (val && typeof val.toMillis === 'function'),
540
- 'New start time must be a valid timestamp, Date object, number, or string',
541
- ),
542
- newEndTime: z
543
- .any()
544
- .refine(
545
- val =>
546
- val instanceof Date ||
547
- val?._seconds !== undefined ||
548
- val?.seconds !== undefined ||
549
- typeof val === 'number' ||
550
- typeof val === 'string' ||
551
- (val && typeof val.toMillis === 'function'),
552
- 'New end time must be a valid timestamp, Date object, number, or string',
553
- ),
554
- });
555
-
556
- /**
557
- * Schema for validating zone photo upload data
558
- */
559
- export const zonePhotoUploadSchema = z.object({
560
- appointmentId: z.string().min(MIN_STRING_LENGTH, 'Appointment ID is required'),
561
- zoneId: z.string().min(MIN_STRING_LENGTH, 'Zone ID is required'),
562
- photoType: z.enum(['before', 'after'], {
563
- required_error: 'Photo type must be either "before" or "after"',
564
- }),
565
- file: z.any().refine(file => {
566
- // Check if it's a File or Blob object
567
- return (
568
- file instanceof File ||
569
- file instanceof Blob ||
570
- (file && typeof file.size === 'number' && typeof file.type === 'string')
571
- );
572
- }, 'File must be a valid File or Blob object'),
573
- notes: z.string().max(MAX_STRING_LENGTH_LONG, 'Notes too long').optional(),
574
- });
1
+ import { z } from 'zod';
2
+ import { AppointmentStatus, PaymentStatus, MediaType } from '../types/appointment';
3
+ import { filledDocumentStatusSchema } from './documentation-templates.schema';
4
+ import { Currency, PricingMeasure } from '../backoffice/types/static/pricing.types';
5
+ import { mediaResourceSchema } from './media.schema';
6
+
7
+ // Define common constants locally if not available from common.schema.ts
8
+ const MIN_STRING_LENGTH = 1;
9
+ const MAX_STRING_LENGTH = 1024; // Example value
10
+ const MAX_STRING_LENGTH_LONG = 4096; // Example value for longer notes/comments
11
+ const MAX_ARRAY_LENGTH = 100; // Example value
12
+
13
+ // --- Enum Schemas ---
14
+ export const appointmentStatusSchema = z.nativeEnum(AppointmentStatus);
15
+ export const paymentStatusSchema = z.nativeEnum(PaymentStatus);
16
+ export const mediaTypeSchema = z.nativeEnum(MediaType);
17
+
18
+ // --- Schemas for Nested Objects from types/appointment/index.ts ---
19
+
20
+ export const appointmentMediaItemSchema = z.object({
21
+ id: z.string().min(MIN_STRING_LENGTH, 'Media item ID is required'),
22
+ type: mediaTypeSchema,
23
+ url: z.string().url('Media URL must be a valid URL'),
24
+ fileName: z.string().optional(),
25
+ uploadedAt: z
26
+ .any()
27
+ .refine(
28
+ val =>
29
+ val instanceof Date ||
30
+ val?._seconds !== undefined ||
31
+ val?.seconds !== undefined ||
32
+ typeof val === 'number' ||
33
+ typeof val === 'string' ||
34
+ (val && typeof val.toMillis === 'function'),
35
+ 'uploadedAt must be a valid timestamp or Date object',
36
+ ),
37
+ uploadedBy: z.string().min(MIN_STRING_LENGTH, 'Uploaded by user ID is required'),
38
+ description: z.string().max(MAX_STRING_LENGTH, 'Description too long').optional(),
39
+ });
40
+
41
+ export const procedureExtendedInfoSchema = z.object({
42
+ id: z.string().min(MIN_STRING_LENGTH),
43
+ name: z.string().min(MIN_STRING_LENGTH),
44
+ description: z.string(),
45
+ cost: z.number().min(0),
46
+ duration: z.number().min(0),
47
+ procedureFamily: z.any(),
48
+ procedureCategoryId: z.string(),
49
+ procedureCategoryName: z.string(),
50
+ procedureSubCategoryId: z.string(),
51
+ procedureSubCategoryName: z.string(),
52
+ procedureTechnologyId: z.string(),
53
+ procedureTechnologyName: z.string(),
54
+ procedureProductBrandId: z.string(),
55
+ procedureProductBrandName: z.string(),
56
+ procedureProducts: z.array(
57
+ z.object({
58
+ productId: z.string().min(MIN_STRING_LENGTH, 'Product ID is required'),
59
+ productName: z.string().min(MIN_STRING_LENGTH, 'Product name is required'),
60
+ brandId: z.string().min(MIN_STRING_LENGTH, 'Brand ID is required'),
61
+ brandName: z.string().min(MIN_STRING_LENGTH, 'Brand name is required'),
62
+ }),
63
+ ),
64
+ });
65
+
66
+ export const linkedFormInfoSchema = z.object({
67
+ formId: z.string().min(MIN_STRING_LENGTH, 'Form ID is required'),
68
+ templateId: z.string().min(MIN_STRING_LENGTH, 'Template ID is required'),
69
+ templateVersion: z.number().int().positive('Template version must be a positive integer'),
70
+ title: z.string().min(MIN_STRING_LENGTH, 'Form title is required'),
71
+ isUserForm: z.boolean(),
72
+ isRequired: z.boolean().optional(),
73
+ status: filledDocumentStatusSchema,
74
+ path: z.string().min(MIN_STRING_LENGTH, 'Form path is required'),
75
+ submittedAt: z
76
+ .any()
77
+ .refine(
78
+ val =>
79
+ val === undefined ||
80
+ val instanceof Date ||
81
+ val?._seconds !== undefined ||
82
+ val?.seconds !== undefined ||
83
+ typeof val === 'number' ||
84
+ typeof val === 'string' ||
85
+ (val && typeof val.toMillis === 'function'),
86
+ 'submittedAt must be a valid timestamp or Date object',
87
+ )
88
+ .optional(),
89
+ completedAt: z
90
+ .any()
91
+ .refine(
92
+ val =>
93
+ val === undefined ||
94
+ val instanceof Date ||
95
+ val?._seconds !== undefined ||
96
+ val?.seconds !== undefined ||
97
+ typeof val === 'number' ||
98
+ typeof val === 'string' ||
99
+ (val && typeof val.toMillis === 'function'),
100
+ 'completedAt must be a valid timestamp or Date object',
101
+ )
102
+ .optional(),
103
+ });
104
+
105
+ export const patientReviewInfoSchema = z.object({
106
+ reviewId: z.string().min(MIN_STRING_LENGTH, 'Review ID is required'),
107
+ rating: z.number().min(1).max(5, 'Rating must be between 1 and 5'),
108
+ comment: z.string().max(MAX_STRING_LENGTH_LONG, 'Comment too long').optional(),
109
+ reviewedAt: z
110
+ .any()
111
+ .refine(
112
+ val =>
113
+ val instanceof Date ||
114
+ val?._seconds !== undefined ||
115
+ val?.seconds !== undefined ||
116
+ typeof val === 'number' ||
117
+ typeof val === 'string' ||
118
+ (val && typeof val.toMillis === 'function'),
119
+ 'reviewedAt must be a valid timestamp or Date object',
120
+ ),
121
+ });
122
+
123
+ export const finalizedDetailsSchema = z.object({
124
+ by: z.string().min(MIN_STRING_LENGTH, 'Finalized by user ID is required'),
125
+ at: z
126
+ .any()
127
+ .refine(
128
+ val =>
129
+ val instanceof Date ||
130
+ val?._seconds !== undefined ||
131
+ val?.seconds !== undefined ||
132
+ typeof val === 'number' ||
133
+ typeof val === 'string' ||
134
+ (val && typeof val.toMillis === 'function'),
135
+ 'Finalized at must be a valid timestamp or Date object',
136
+ ),
137
+ notes: z.string().max(MAX_STRING_LENGTH_LONG, 'Finalization notes too long').optional(),
138
+ });
139
+
140
+ /**
141
+ * Schema for before/after photos and notes per zone
142
+ */
143
+ export const beforeAfterPerZoneSchema = z.object({
144
+ before: mediaResourceSchema.nullable(),
145
+ after: mediaResourceSchema.nullable(),
146
+ afterNote: z.string().nullable().optional(),
147
+ beforeNote: z.string().nullable().optional(),
148
+ });
149
+
150
+ /**
151
+ * Schema for billing information per zone
152
+ */
153
+ export const billingPerZoneSchema = z.object({
154
+ Product: z.string().min(MIN_STRING_LENGTH, 'Product name is required'),
155
+ ProductId: z.string().nullable(),
156
+ Quantity: z.number().min(0, 'Quantity must be non-negative'),
157
+ UnitOfMeasurement: z.nativeEnum(PricingMeasure),
158
+ UnitPrice: z.number().min(0, 'Unit price must be non-negative'),
159
+ UnitCurency: z.nativeEnum(Currency),
160
+ Subtotal: z.number().min(0, 'Subtotal must be non-negative'),
161
+ Note: z.string().nullable(),
162
+ IonNumber: z.string().nullable(),
163
+ });
164
+
165
+ /**
166
+ * Schema for final billing calculations of the appointment
167
+ */
168
+ export const finalBillingSchema = z.object({
169
+ subtotalAll: z.number().min(0, 'Subtotal all must be non-negative'),
170
+ taxRate: z.number().min(0).max(1, 'Tax rate must be between 0 and 1'),
171
+ taxPrice: z.number().min(0, 'Tax price must be non-negative'),
172
+ finalPrice: z.number().min(0, 'Final price must be non-negative'),
173
+ currency: z.nativeEnum(Currency),
174
+ });
175
+
176
+ /**
177
+ * Schema for zone item data (product or note per zone)
178
+ */
179
+ export const zoneItemDataSchema = z
180
+ .object({
181
+ productId: z.string().optional(),
182
+ productName: z.string().optional(),
183
+ productBrandId: z.string().optional(),
184
+ productBrandName: z.string().optional(),
185
+ belongingProcedureId: z.string().min(MIN_STRING_LENGTH, 'Belonging procedure ID is required'),
186
+ type: z.enum(['item', 'note'], {
187
+ required_error: 'Type must be either "item" or "note"',
188
+ }),
189
+ stage: z
190
+ .enum(['before', 'after'], {
191
+ required_error: 'Stage must be either "before" or "after"',
192
+ })
193
+ .optional(),
194
+ price: z.number().min(0, 'Price must be non-negative').optional(),
195
+ currency: z.nativeEnum(Currency).optional(),
196
+ unitOfMeasurement: z.nativeEnum(PricingMeasure).optional(),
197
+ priceOverrideAmount: z.number().min(0, 'Price override amount must be non-negative').optional(),
198
+ quantity: z.number().min(0, 'Quantity must be non-negative').optional(),
199
+ parentZone: z
200
+ .string()
201
+ .min(MIN_STRING_LENGTH, 'Parent zone is required')
202
+ .refine(
203
+ val => {
204
+ const parts = val.split('.');
205
+ return parts.length === 2;
206
+ },
207
+ {
208
+ message: 'Parent zone must be in "category.zone" format (e.g., "face.forehead")',
209
+ },
210
+ ),
211
+ subzones: z.array(z.string()).refine(
212
+ val => {
213
+ return val.every(subzone => {
214
+ const parts = subzone.split('.');
215
+ return parts.length === 3;
216
+ });
217
+ },
218
+ {
219
+ message: 'Subzones must be in "category.zone.subzone" format (e.g., "face.forehead.left")',
220
+ },
221
+ ),
222
+ notes: z.string().max(MAX_STRING_LENGTH_LONG, 'Notes too long').optional(),
223
+ subtotal: z.number().min(0, 'Subtotal must be non-negative').optional(),
224
+ ionNumber: z.string().optional(),
225
+ createdAt: z.string().optional(),
226
+ updatedAt: z.string().optional(),
227
+ })
228
+ .refine(
229
+ data => {
230
+ if (data.type === 'item') {
231
+ return !!(data.productId && data.productName && (data.price || data.priceOverrideAmount));
232
+ }
233
+ return true;
234
+ },
235
+ {
236
+ message: 'Item type requires productId, productName, and either price or priceOverrideAmount',
237
+ },
238
+ );
239
+
240
+ /**
241
+ * Schema for appointment product metadata
242
+ */
243
+ export const appointmentProductMetadataSchema = z.object({
244
+ productId: z.string().min(MIN_STRING_LENGTH, 'Product ID is required'),
245
+ productName: z.string().min(MIN_STRING_LENGTH, 'Product name is required'),
246
+ brandId: z.string().min(MIN_STRING_LENGTH, 'Brand ID is required'),
247
+ brandName: z.string().min(MIN_STRING_LENGTH, 'Brand name is required'),
248
+ procedureId: z.string().min(MIN_STRING_LENGTH, 'Procedure ID is required'),
249
+ price: z.number().min(0, 'Price must be non-negative'),
250
+ currency: z.nativeEnum(Currency),
251
+ unitOfMeasurement: z.nativeEnum(PricingMeasure),
252
+ });
253
+
254
+ /**
255
+ * Schema for extended procedure info
256
+ */
257
+ export const extendedProcedureInfoSchema = z.object({
258
+ procedureId: z.string().min(MIN_STRING_LENGTH, 'Procedure ID is required'),
259
+ procedureName: z.string().min(MIN_STRING_LENGTH, 'Procedure name is required'),
260
+ procedureFamily: z.any(),
261
+ procedureCategoryId: z.string().min(MIN_STRING_LENGTH, 'Category ID is required'),
262
+ procedureCategoryName: z.string().min(MIN_STRING_LENGTH, 'Category name is required'),
263
+ procedureSubCategoryId: z.string().min(MIN_STRING_LENGTH, 'Subcategory ID is required'),
264
+ procedureSubCategoryName: z.string().min(MIN_STRING_LENGTH, 'Subcategory name is required'),
265
+ procedureTechnologyId: z.string().min(MIN_STRING_LENGTH, 'Technology ID is required'),
266
+ procedureTechnologyName: z.string().min(MIN_STRING_LENGTH, 'Technology name is required'),
267
+ procedureProducts: z.array(
268
+ z.object({
269
+ productId: z.string().min(MIN_STRING_LENGTH, 'Product ID is required'),
270
+ productName: z.string().min(MIN_STRING_LENGTH, 'Product name is required'),
271
+ brandId: z.string().min(MIN_STRING_LENGTH, 'Brand ID is required'),
272
+ brandName: z.string().min(MIN_STRING_LENGTH, 'Brand name is required'),
273
+ }),
274
+ ),
275
+ });
276
+
277
+ /**
278
+ * Schema for recommended procedure within metadata
279
+ */
280
+ export const recommendedProcedureTimeframeSchema = z.object({
281
+ value: z.number().int().positive('Timeframe value must be a positive integer'),
282
+ unit: z.enum(['day', 'week', 'month', 'year']),
283
+ });
284
+
285
+ export const recommendedProcedureSchema = z.object({
286
+ procedure: extendedProcedureInfoSchema,
287
+ note: z.string().max(MAX_STRING_LENGTH_LONG, 'Note too long'), // Note is now optional (no min length)
288
+ timeframe: recommendedProcedureTimeframeSchema,
289
+ });
290
+
291
+ /**
292
+ * Schema for appointment metadata containing zone-specific information
293
+ */
294
+ export const appointmentMetadataSchema = z.object({
295
+ selectedZones: z.array(z.string()).nullable(),
296
+ zonePhotos: z.record(z.string(), z.array(beforeAfterPerZoneSchema).max(10)).nullable(),
297
+ zonesData: z.record(z.string(), z.array(zoneItemDataSchema)).nullable().optional(),
298
+ appointmentProducts: z.array(appointmentProductMetadataSchema).optional().default([]),
299
+ extendedProcedures: z.array(extendedProcedureInfoSchema).optional().default([]),
300
+ recommendedProcedures: z.array(recommendedProcedureSchema).optional().default([]),
301
+ zoneBilling: z.record(z.string(), billingPerZoneSchema).nullable().optional(),
302
+ finalbilling: finalBillingSchema.nullable(),
303
+ finalizationNotes: z.string().nullable(),
304
+ });
305
+
306
+ // --- Main Appointment Schemas ---
307
+
308
+ /**
309
+ * Schema for validating appointment creation data
310
+ */
311
+ export const createAppointmentSchema = z
312
+ .object({
313
+ clinicBranchId: z.string().min(MIN_STRING_LENGTH, 'Clinic branch ID is required'),
314
+ practitionerId: z.string().min(MIN_STRING_LENGTH, 'Practitioner ID is required'),
315
+ patientId: z.string().min(MIN_STRING_LENGTH, 'Patient ID is required'),
316
+ procedureId: z.string().min(MIN_STRING_LENGTH, 'Procedure ID is required'),
317
+ appointmentStartTime: z
318
+ .any()
319
+ .refine(
320
+ val =>
321
+ val instanceof Date ||
322
+ val?._seconds !== undefined ||
323
+ val?.seconds !== undefined ||
324
+ typeof val === 'number' ||
325
+ typeof val === 'string' ||
326
+ (val && typeof val.toMillis === 'function'),
327
+ 'Appointment start time must be a valid timestamp or Date object',
328
+ ),
329
+ appointmentEndTime: z
330
+ .any()
331
+ .refine(
332
+ val =>
333
+ val instanceof Date ||
334
+ val?._seconds !== undefined ||
335
+ val?.seconds !== undefined ||
336
+ typeof val === 'number' ||
337
+ typeof val === 'string' ||
338
+ (val && typeof val.toMillis === 'function'),
339
+ 'Appointment end time must be a valid timestamp or Date object',
340
+ ),
341
+ cost: z.number().min(0, 'Cost must be a non-negative number'),
342
+ currency: z.string().min(1, 'Currency is required'),
343
+ patientNotes: z.string().max(MAX_STRING_LENGTH, 'Patient notes too long').nullable().optional(),
344
+ initialStatus: appointmentStatusSchema,
345
+ initialPaymentStatus: paymentStatusSchema.optional().default(PaymentStatus.UNPAID),
346
+ clinic_tz: z.string().min(1, 'Timezone is required'),
347
+ })
348
+ .refine(data => data.appointmentEndTime > data.appointmentStartTime, {
349
+ message: 'Appointment end time must be after start time',
350
+ path: ['appointmentEndTime'],
351
+ });
352
+
353
+ /**
354
+ * Schema for validating appointment update data
355
+ */
356
+ export const updateAppointmentSchema = z
357
+ .object({
358
+ status: appointmentStatusSchema.optional(),
359
+ confirmationTime: z.any().optional().nullable(),
360
+ cancellationTime: z.any().optional().nullable(),
361
+ rescheduleTime: z.any().optional().nullable(),
362
+ procedureActualStartTime: z.any().optional().nullable(),
363
+ actualDurationMinutes: z
364
+ .number()
365
+ .int()
366
+ .positive('Duration must be a positive integer')
367
+ .optional(),
368
+ cancellationReason: z
369
+ .string()
370
+ .max(MAX_STRING_LENGTH, 'Cancellation reason too long')
371
+ .nullable()
372
+ .optional(),
373
+ canceledBy: z.enum(['patient', 'clinic', 'practitioner', 'system']).optional(),
374
+ internalNotes: z
375
+ .string()
376
+ .max(MAX_STRING_LENGTH_LONG, 'Internal notes too long')
377
+ .nullable()
378
+ .optional(),
379
+ patientNotes: z.any().optional().nullable(),
380
+ paymentStatus: paymentStatusSchema.optional(),
381
+ paymentTransactionId: z.any().optional().nullable(),
382
+ completedPreRequirements: z.union([z.array(z.string()), z.any()]).optional(),
383
+ completedPostRequirements: z.union([z.array(z.string()), z.any()]).optional(),
384
+ linkedFormIds: z.union([z.array(z.string()), z.any()]).optional(),
385
+ pendingUserFormsIds: z.union([z.array(z.string()), z.any()]).optional(),
386
+ appointmentStartTime: z
387
+ .any()
388
+ .refine(
389
+ val =>
390
+ val === undefined ||
391
+ val instanceof Date ||
392
+ val?._seconds !== undefined ||
393
+ val?.seconds !== undefined ||
394
+ typeof val === 'number' ||
395
+ typeof val === 'string' ||
396
+ (val && typeof val.toMillis === 'function'),
397
+ 'Appointment start time must be a valid timestamp or Date object',
398
+ )
399
+ .optional(),
400
+ appointmentEndTime: z
401
+ .any()
402
+ .refine(
403
+ val =>
404
+ val === undefined ||
405
+ val instanceof Date ||
406
+ val?._seconds !== undefined ||
407
+ val?.seconds !== undefined ||
408
+ typeof val === 'number' ||
409
+ typeof val === 'string' ||
410
+ (val && typeof val.toMillis === 'function'),
411
+ 'Appointment end time must be a valid timestamp or Date object',
412
+ )
413
+ .optional(),
414
+ calendarEventId: z.string().min(MIN_STRING_LENGTH).optional(),
415
+ cost: z.number().min(0).optional(),
416
+ clinicBranchId: z.string().min(MIN_STRING_LENGTH).optional(),
417
+ practitionerId: z.string().min(MIN_STRING_LENGTH).optional(),
418
+ clinic_tz: z.string().min(MIN_STRING_LENGTH).optional(),
419
+ linkedForms: z.union([z.array(linkedFormInfoSchema).max(MAX_ARRAY_LENGTH), z.any()]).optional(),
420
+ media: z.union([z.array(appointmentMediaItemSchema).max(MAX_ARRAY_LENGTH), z.any()]).optional(),
421
+ reviewInfo: z.union([patientReviewInfoSchema.nullable(), z.any()]).optional(),
422
+ finalizedDetails: z.union([finalizedDetailsSchema.nullable(), z.any()]).optional(),
423
+ isArchived: z.boolean().optional(),
424
+ updatedAt: z.any().optional(),
425
+ metadata: appointmentMetadataSchema.optional(),
426
+ })
427
+ .refine(
428
+ data => {
429
+ if (
430
+ data.status === AppointmentStatus.CANCELED_CLINIC ||
431
+ data.status === AppointmentStatus.CANCELED_PATIENT ||
432
+ data.status === AppointmentStatus.CANCELED_PATIENT_RESCHEDULED
433
+ ) {
434
+ return !!data.cancellationReason && !!data.canceledBy;
435
+ }
436
+ return true;
437
+ },
438
+ {
439
+ message:
440
+ 'Cancellation reason and canceled by must be provided when canceling an appointment with patient or clinic origin.',
441
+ path: ['status'],
442
+ },
443
+ )
444
+ .refine(
445
+ data => {
446
+ if (data.appointmentStartTime && data.appointmentEndTime) {
447
+ return data.appointmentEndTime > data.appointmentStartTime;
448
+ }
449
+ return true;
450
+ },
451
+ {
452
+ message: 'Appointment end time must be after start time if both are provided',
453
+ path: ['appointmentEndTime'],
454
+ },
455
+ );
456
+
457
+ /**
458
+ * Schema for validating appointment search parameters
459
+ */
460
+ export const searchAppointmentsSchema = z
461
+ .object({
462
+ patientId: z.string().optional(),
463
+ practitionerId: z.string().optional(),
464
+ clinicBranchId: z.string().optional(),
465
+ startDate: z
466
+ .any()
467
+ .refine(
468
+ val =>
469
+ val === undefined ||
470
+ val instanceof Date ||
471
+ val?._seconds !== undefined ||
472
+ val?.seconds !== undefined ||
473
+ typeof val === 'number' ||
474
+ typeof val === 'string' ||
475
+ (val && typeof val.toMillis === 'function'),
476
+ 'Start date must be a valid timestamp or Date object',
477
+ )
478
+ .optional(),
479
+ endDate: z
480
+ .any()
481
+ .refine(
482
+ val =>
483
+ val === undefined ||
484
+ val instanceof Date ||
485
+ val?._seconds !== undefined ||
486
+ val?.seconds !== undefined ||
487
+ typeof val === 'number' ||
488
+ typeof val === 'string' ||
489
+ (val && typeof val.toMillis === 'function'),
490
+ 'End date must be a valid timestamp or Date object',
491
+ )
492
+ .optional(),
493
+ status: z
494
+ .union([appointmentStatusSchema, z.array(appointmentStatusSchema).nonempty()])
495
+ .optional(),
496
+ limit: z.number().positive().int().optional().default(20),
497
+ startAfter: z.any().optional(),
498
+ })
499
+ .refine(
500
+ data => {
501
+ if (!data.startDate && !data.endDate && !data.status) {
502
+ return !!(data.patientId || data.practitionerId || data.clinicBranchId);
503
+ }
504
+ return true;
505
+ },
506
+ {
507
+ message:
508
+ 'At least one of patientId, practitionerId, or clinicBranchId must be provided if no date or status filters are set.',
509
+ path: ['patientId'],
510
+ },
511
+ )
512
+ .refine(
513
+ data => {
514
+ if (data.startDate && data.endDate) {
515
+ return data.endDate >= data.startDate;
516
+ }
517
+ return true;
518
+ },
519
+ {
520
+ message: 'End date must be after or the same as start date',
521
+ path: ['endDate'],
522
+ },
523
+ );
524
+
525
+ /**
526
+ * Schema for validating appointment reschedule data
527
+ */
528
+ export const rescheduleAppointmentSchema = z.object({
529
+ appointmentId: z.string().min(MIN_STRING_LENGTH, 'Appointment ID is required'),
530
+ newStartTime: z
531
+ .any()
532
+ .refine(
533
+ val =>
534
+ val instanceof Date ||
535
+ val?._seconds !== undefined ||
536
+ val?.seconds !== undefined ||
537
+ typeof val === 'number' ||
538
+ typeof val === 'string' ||
539
+ (val && typeof val.toMillis === 'function'),
540
+ 'New start time must be a valid timestamp, Date object, number, or string',
541
+ ),
542
+ newEndTime: z
543
+ .any()
544
+ .refine(
545
+ val =>
546
+ val instanceof Date ||
547
+ val?._seconds !== undefined ||
548
+ val?.seconds !== undefined ||
549
+ typeof val === 'number' ||
550
+ typeof val === 'string' ||
551
+ (val && typeof val.toMillis === 'function'),
552
+ 'New end time must be a valid timestamp, Date object, number, or string',
553
+ ),
554
+ });
555
+
556
+ /**
557
+ * Schema for validating zone photo upload data
558
+ */
559
+ export const zonePhotoUploadSchema = z.object({
560
+ appointmentId: z.string().min(MIN_STRING_LENGTH, 'Appointment ID is required'),
561
+ zoneId: z.string().min(MIN_STRING_LENGTH, 'Zone ID is required'),
562
+ photoType: z.enum(['before', 'after'], {
563
+ required_error: 'Photo type must be either "before" or "after"',
564
+ }),
565
+ file: z.any().refine(file => {
566
+ // Check if it's a File or Blob object
567
+ return (
568
+ file instanceof File ||
569
+ file instanceof Blob ||
570
+ (file && typeof file.size === 'number' && typeof file.type === 'string')
571
+ );
572
+ }, 'File must be a valid File or Blob object'),
573
+ notes: z.string().max(MAX_STRING_LENGTH_LONG, 'Notes too long').optional(),
574
+ });