@stamhoofd/backend 2.118.1 → 2.120.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (362) hide show
  1. package/package.json +32 -22
  2. package/src/audit-logs/ModelLogger.ts +4 -2
  3. package/src/audit-logs/OrderLogger.ts +1 -1
  4. package/src/boot.ts +32 -14
  5. package/src/crons/balance-emails.ts +4 -2
  6. package/src/crons/clearExcelCache.test.ts +8 -8
  7. package/src/crons/update-cached-balances.ts +40 -14
  8. package/src/debug.ts +3 -2
  9. package/src/decoders/StringArrayDecoder.ts +1 -1
  10. package/src/decoders/StringNullableDecoder.ts +1 -1
  11. package/src/email-recipient-loaders/documents.ts +2 -1
  12. package/src/email-recipient-loaders/members.ts +2 -1
  13. package/src/email-recipient-loaders/orders.ts +2 -1
  14. package/src/email-recipient-loaders/payments.ts +6 -3
  15. package/src/email-recipient-loaders/receivable-balances.ts +2 -1
  16. package/src/email-recipient-loaders/registrations.ts +2 -1
  17. package/src/email-replacements/getEmailReplacementsForPayment.ts +8 -7
  18. package/src/endpoints/admin/members/ChargeMembersEndpoint.ts +7 -6
  19. package/src/endpoints/admin/memberships/ChargeMembershipsEndpoint.ts +3 -2
  20. package/src/endpoints/admin/memberships/GetChargeMembershipsSummaryEndpoint.ts +2 -1
  21. package/src/endpoints/admin/organizations/ChargeOrganizationsEndpoint.ts +4 -3
  22. package/src/endpoints/admin/organizations/GetOrganizationsCountEndpoint.ts +3 -2
  23. package/src/endpoints/admin/organizations/GetOrganizationsEndpoint.ts +7 -5
  24. package/src/endpoints/admin/organizations/PatchOrganizationsEndpoint.ts +12 -10
  25. package/src/endpoints/admin/registrations/ChargeRegistrationsEndpoint.ts +4 -3
  26. package/src/endpoints/auth/CreateAdminEndpoint.ts +8 -6
  27. package/src/endpoints/auth/CreateTokenEndpoint.ts +14 -12
  28. package/src/endpoints/auth/DeleteTokenEndpoint.ts +2 -1
  29. package/src/endpoints/auth/DeleteUserEndpoint.ts +2 -1
  30. package/src/endpoints/auth/ForgotPasswordEndpoint.ts +5 -4
  31. package/src/endpoints/auth/GetOtherUserEndpoint.ts +3 -2
  32. package/src/endpoints/auth/GetUserEndpoint.ts +3 -2
  33. package/src/endpoints/auth/OpenIDConnectAuthTokenEndpoint.ts +2 -1
  34. package/src/endpoints/auth/OpenIDConnectCallbackEndpoint.ts +4 -2
  35. package/src/endpoints/auth/OpenIDConnectStartEndpoint.ts +3 -2
  36. package/src/endpoints/auth/PatchUserEndpoint.ts +15 -12
  37. package/src/endpoints/auth/PollEmailVerificationEndpoint.ts +3 -2
  38. package/src/endpoints/auth/RetryEmailVerificationEndpoint.ts +3 -2
  39. package/src/endpoints/auth/SignupEndpoint.ts +5 -4
  40. package/src/endpoints/auth/VerifyEmailEndpoint.ts +6 -5
  41. package/src/endpoints/frontend/FrontendEnvironmentEndpoint.ts +3 -2
  42. package/src/endpoints/global/addresses/SearchRegionsEndpoint.ts +8 -5
  43. package/src/endpoints/global/addresses/ValidateAddressEndpoint.ts +5 -3
  44. package/src/endpoints/global/audit-logs/GetAuditLogsEndpoint.ts +8 -5
  45. package/src/endpoints/global/billing/ActivatePackagesEndpoint.ts +9 -7
  46. package/src/endpoints/global/billing/DeactivatePackageEndpoint.ts +4 -3
  47. package/src/endpoints/global/caddy/CheckDomainCertEndpoint.ts +4 -2
  48. package/src/endpoints/global/email/CreateEmailEndpoint.ts +9 -7
  49. package/src/endpoints/global/email/GetAdminEmailsEndpoint.test.ts +2 -1
  50. package/src/endpoints/global/email/GetAdminEmailsEndpoint.ts +22 -19
  51. package/src/endpoints/global/email/GetEmailAddressEndpoint.ts +6 -4
  52. package/src/endpoints/global/email/GetEmailEndpoint.ts +4 -3
  53. package/src/endpoints/global/email/GetUserEmailsEndpoint.test.ts +3 -2
  54. package/src/endpoints/global/email/GetUserEmailsEndpoint.ts +7 -4
  55. package/src/endpoints/global/email/ManageEmailAddressEndpoint.ts +7 -5
  56. package/src/endpoints/global/email/PatchEmailEndpoint.test.ts +11 -11
  57. package/src/endpoints/global/email/PatchEmailEndpoint.ts +14 -11
  58. package/src/endpoints/global/email-recipients/GetEmailRecipientsCountEndpoint.ts +3 -2
  59. package/src/endpoints/global/email-recipients/GetEmailRecipientsEndpoint.test.ts +2 -1
  60. package/src/endpoints/global/email-recipients/GetEmailRecipientsEndpoint.ts +8 -5
  61. package/src/endpoints/global/email-recipients/RetryEmailRecipientEndpoint.ts +7 -5
  62. package/src/endpoints/global/email-recipients/helpers/validateEmailRecipientFilter.ts +4 -3
  63. package/src/endpoints/global/events/GetEventNotificationsCountEndpoint.ts +3 -2
  64. package/src/endpoints/global/events/GetEventNotificationsEndpoint.ts +9 -6
  65. package/src/endpoints/global/events/GetEventsEndpoint.ts +7 -4
  66. package/src/endpoints/global/events/PatchEventNotificationsEndpoint.test.ts +4 -2
  67. package/src/endpoints/global/events/PatchEventNotificationsEndpoint.ts +17 -15
  68. package/src/endpoints/global/events/PatchEventsEndpoint.test.ts +4 -2
  69. package/src/endpoints/global/events/PatchEventsEndpoint.ts +28 -26
  70. package/src/endpoints/global/files/ExportToExcelEndpoint.ts +8 -5
  71. package/src/endpoints/global/files/GetFileCache.ts +5 -3
  72. package/src/endpoints/global/files/UploadFile.ts +10 -4
  73. package/src/endpoints/global/files/UploadImage.ts +4 -2
  74. package/src/endpoints/global/groups/GetGroupsEndpoint.test.ts +4 -2
  75. package/src/endpoints/global/groups/GetGroupsEndpoint.ts +8 -5
  76. package/src/endpoints/global/members/GetMemberFamilyEndpoint.ts +7 -5
  77. package/src/endpoints/global/members/GetMembersCountEndpoint.ts +3 -2
  78. package/src/endpoints/global/members/GetMembersEndpoint.test.ts +4 -2
  79. package/src/endpoints/global/members/GetMembersEndpoint.ts +10 -8
  80. package/src/endpoints/global/members/PatchOrganizationMembersEndpoint.test.ts +257 -6
  81. package/src/endpoints/global/members/PatchOrganizationMembersEndpoint.ts +162 -107
  82. package/src/endpoints/global/members/helpers/validateGroupFilter.ts +4 -3
  83. package/src/endpoints/global/members/shouldCheckIfMemberIsDuplicate.ts +3 -2
  84. package/src/endpoints/global/organizations/CheckRegisterCodeEndpoint.ts +3 -2
  85. package/src/endpoints/global/organizations/CreateOrganizationEndpoint.test.ts +3 -2
  86. package/src/endpoints/global/organizations/CreateOrganizationEndpoint.ts +8 -7
  87. package/src/endpoints/global/organizations/GetOrganizationFromDomainEndpoint.ts +5 -3
  88. package/src/endpoints/global/organizations/GetOrganizationFromUriEndpoint.ts +5 -3
  89. package/src/endpoints/global/organizations/SearchOrganizationEndpoint.ts +8 -5
  90. package/src/endpoints/global/payments/StripeWebhookEndpoint.ts +4 -2
  91. package/src/endpoints/global/platform/GetPlatformAdminsEndpoint.ts +2 -1
  92. package/src/endpoints/global/platform/GetPlatformEndpoint.ts +3 -2
  93. package/src/endpoints/global/platform/PatchPlatformEnpoint.test.ts +3 -2
  94. package/src/endpoints/global/platform/PatchPlatformEnpoint.ts +7 -4
  95. package/src/endpoints/global/platform-memberships/GetPlatformMembershipsCountEndpoint.ts +47 -0
  96. package/src/endpoints/global/platform-memberships/GetPlatformMembershipsEndpoint.ts +206 -0
  97. package/src/endpoints/global/registration/GetRegistrationsCountEndpoint.ts +3 -2
  98. package/src/endpoints/global/registration/GetRegistrationsEndpoint.test.ts +2 -1
  99. package/src/endpoints/global/registration/GetRegistrationsEndpoint.ts +10 -7
  100. package/src/endpoints/global/registration/GetUserDetailedPayableBalanceEndpoint.ts +4 -2
  101. package/src/endpoints/global/registration/GetUserDocumentsEndpoint.ts +4 -2
  102. package/src/endpoints/global/registration/GetUserMembersEndpoint.ts +3 -2
  103. package/src/endpoints/global/registration/GetUserPayableBalanceEndpoint.ts +2 -1
  104. package/src/endpoints/global/registration/PatchUserMembersEndpoint.test.ts +5 -3
  105. package/src/endpoints/global/registration/PatchUserMembersEndpoint.ts +13 -9
  106. package/src/endpoints/global/registration/RegisterMembersEndpoint.test.ts +50 -49
  107. package/src/endpoints/global/registration/RegisterMembersEndpoint.ts +25 -22
  108. package/src/endpoints/global/registration-periods/GetRegistrationPeriodsEndpoint.ts +8 -5
  109. package/src/endpoints/global/registration-periods/PatchRegistrationPeriodsEndpoint.ts +9 -7
  110. package/src/endpoints/global/sso/GetSSOEndpoint.ts +4 -2
  111. package/src/endpoints/global/sso/SetSSOEndpoint.ts +3 -2
  112. package/src/endpoints/global/webshops/GetWebshopFromDomainEndpoint.ts +4 -2
  113. package/src/endpoints/global/webshops/GetWebshopsCountEndpoint.ts +43 -0
  114. package/src/endpoints/global/webshops/GetWebshopsEndpoint.test.ts +808 -0
  115. package/src/endpoints/global/webshops/GetWebshopsEndpoint.ts +221 -0
  116. package/src/endpoints/organization/dashboard/balance-items/GetBalanceItemEndpoint.ts +4 -3
  117. package/src/endpoints/organization/dashboard/balance-items/GetBalanceItemsCountEndpoint.ts +3 -2
  118. package/src/endpoints/organization/dashboard/balance-items/GetBalanceItemsEndpoint.ts +6 -4
  119. package/src/endpoints/organization/dashboard/balance-items/PatchBalanceItemsEndpoint.ts +15 -13
  120. package/src/endpoints/organization/dashboard/billing/GetOrganizationDetailedPayableBalanceEndpoint.ts +4 -2
  121. package/src/endpoints/organization/dashboard/billing/GetOrganizationPayableBalanceEndpoint.ts +4 -2
  122. package/src/endpoints/organization/dashboard/billing/GetPackagesEndpoint.ts +2 -1
  123. package/src/endpoints/organization/dashboard/documents/GetDocumentTemplateXML.ts +3 -2
  124. package/src/endpoints/organization/dashboard/documents/GetDocumentTemplatesCountEndpoint.ts +3 -2
  125. package/src/endpoints/organization/dashboard/documents/GetDocumentTemplatesEndpoint.ts +7 -4
  126. package/src/endpoints/organization/dashboard/documents/GetDocumentsCountEndpoint.ts +3 -2
  127. package/src/endpoints/organization/dashboard/documents/GetDocumentsEndpoint.ts +7 -4
  128. package/src/endpoints/organization/dashboard/documents/PatchDocumentEndpoint.ts +9 -7
  129. package/src/endpoints/organization/dashboard/documents/PatchDocumentTemplatesEndpoint.test.ts +4 -2
  130. package/src/endpoints/organization/dashboard/documents/PatchDocumentTemplatesEndpoint.ts +7 -5
  131. package/src/endpoints/organization/dashboard/email/CheckEmailBouncesEndpoint.ts +3 -2
  132. package/src/endpoints/organization/dashboard/email-templates/GetEmailTemplatesEndpoint.test.ts +36 -1
  133. package/src/endpoints/organization/dashboard/email-templates/GetEmailTemplatesEndpoint.ts +13 -4
  134. package/src/endpoints/organization/dashboard/email-templates/PatchEmailTemplatesEndpoint.test.ts +4 -2
  135. package/src/endpoints/organization/dashboard/email-templates/PatchEmailTemplatesEndpoint.ts +7 -5
  136. package/src/endpoints/organization/dashboard/invoices/GetInvoicesCountEndpoint.ts +3 -2
  137. package/src/endpoints/organization/dashboard/invoices/GetInvoicesEndpoint.ts +6 -4
  138. package/src/endpoints/organization/dashboard/invoices/PatchInvoicesEndpoint.ts +5 -3
  139. package/src/endpoints/organization/dashboard/mollie/CheckMollieEndpoint.ts +4 -2
  140. package/src/endpoints/organization/dashboard/mollie/ConnectMollieEndpoint.ts +6 -3
  141. package/src/endpoints/organization/dashboard/mollie/DisconnectMollieEndpoint.ts +4 -2
  142. package/src/endpoints/organization/dashboard/mollie/GetMollieDashboardEndpoint.ts +2 -1
  143. package/src/endpoints/organization/dashboard/nolt/CreateNoltTokenEndpoint.ts +2 -1
  144. package/src/endpoints/organization/dashboard/organization/GetOrganizationArchivedGroups.ts +4 -2
  145. package/src/endpoints/organization/dashboard/organization/GetOrganizationDeletedGroups.ts +3 -2
  146. package/src/endpoints/organization/dashboard/organization/GetUitpasClientIdEndpoint.ts +2 -1
  147. package/src/endpoints/organization/dashboard/organization/PatchOrganizationEndpoint.test.ts +5 -3
  148. package/src/endpoints/organization/dashboard/organization/PatchOrganizationEndpoint.ts +22 -19
  149. package/src/endpoints/organization/dashboard/organization/SearchUitpasOrganizersEndpoint.ts +5 -3
  150. package/src/endpoints/organization/dashboard/organization/SetOrganizationDomainEndpoint.ts +11 -9
  151. package/src/endpoints/organization/dashboard/organization/SetUitpasClientCredentialsEndpoint.ts +8 -7
  152. package/src/endpoints/organization/dashboard/payments/GetMemberBalanceEndpoint.ts +2 -1
  153. package/src/endpoints/organization/dashboard/payments/GetPaymentsCountEndpoint.ts +3 -2
  154. package/src/endpoints/organization/dashboard/payments/GetPaymentsEndpoint.ts +7 -5
  155. package/src/endpoints/organization/dashboard/payments/PatchPaymentsEndpoint.ts +18 -16
  156. package/src/endpoints/organization/dashboard/receivable-balances/GetReceivableBalanceEndpoint.ts +2 -1
  157. package/src/endpoints/organization/dashboard/receivable-balances/GetReceivableBalancesCountEndpoint.ts +3 -2
  158. package/src/endpoints/organization/dashboard/receivable-balances/GetReceivableBalancesEndpoint.ts +5 -3
  159. package/src/endpoints/organization/dashboard/registration-periods/GetOrganizationRegistrationPeriodsEndpoint.test.ts +2 -1
  160. package/src/endpoints/organization/dashboard/registration-periods/GetOrganizationRegistrationPeriodsEndpoint.ts +8 -5
  161. package/src/endpoints/organization/dashboard/registration-periods/MoveRegistrationPeriods.test.ts +4 -2
  162. package/src/endpoints/organization/dashboard/registration-periods/PatchOrganizationRegistrationPeriodsEndpoint.test.ts +4 -2
  163. package/src/endpoints/organization/dashboard/registration-periods/PatchOrganizationRegistrationPeriodsEndpoint.ts +32 -29
  164. package/src/endpoints/organization/dashboard/registration-periods/SetupStepReviewEndpoint.ts +6 -3
  165. package/src/endpoints/organization/dashboard/stripe/ConnectStripeEndpoint.ts +3 -2
  166. package/src/endpoints/organization/dashboard/stripe/DeleteStripeAccountEndpoint.ts +3 -2
  167. package/src/endpoints/organization/dashboard/stripe/GetStripeAccountLinkEndpoint.ts +5 -3
  168. package/src/endpoints/organization/dashboard/stripe/GetStripeAccountsEndpoint.ts +2 -1
  169. package/src/endpoints/organization/dashboard/stripe/GetStripeLoginLinkEndpoint.ts +5 -3
  170. package/src/endpoints/organization/dashboard/stripe/UpdateStripeAccountEndpoint.ts +3 -2
  171. package/src/endpoints/organization/dashboard/users/CreateApiUserEndpoint.ts +4 -3
  172. package/src/endpoints/organization/dashboard/users/DeleteUserEndpoint.ts +4 -3
  173. package/src/endpoints/organization/dashboard/users/GetApiUsersEndpoint.ts +3 -2
  174. package/src/endpoints/organization/dashboard/users/GetOrganizationAdminsEndpoint.ts +2 -1
  175. package/src/endpoints/organization/dashboard/users/PatchApiUserEndpoint.ts +6 -4
  176. package/src/endpoints/organization/dashboard/webshops/CreateWebshopEndpoint.ts +7 -6
  177. package/src/endpoints/organization/dashboard/webshops/DeleteWebshopEndpoint.ts +3 -2
  178. package/src/endpoints/organization/dashboard/webshops/GetDiscountCodesEndpoint.ts +4 -2
  179. package/src/endpoints/organization/dashboard/webshops/GetWebshopOrdersCountEndpoint.ts +3 -2
  180. package/src/endpoints/organization/dashboard/webshops/GetWebshopOrdersEndpoint.ts +7 -4
  181. package/src/endpoints/organization/dashboard/webshops/{GetWebshopTicketsCountEndpoint → GetWebshopTicketsCountEndpoint.ts} +7 -7
  182. package/src/endpoints/organization/dashboard/webshops/GetWebshopTicketsEndpoint.ts +9 -6
  183. package/src/endpoints/organization/dashboard/webshops/GetWebshopUriAvailabilityEndpoint.ts +4 -2
  184. package/src/endpoints/organization/dashboard/webshops/PatchDiscountCodesEndpoint.ts +6 -4
  185. package/src/endpoints/organization/dashboard/webshops/PatchWebshopEndpoint.ts +13 -12
  186. package/src/endpoints/organization/dashboard/webshops/PatchWebshopOrdersEndpoint.ts +4 -2
  187. package/src/endpoints/organization/dashboard/webshops/PatchWebshopTicketsEndpoint.ts +5 -3
  188. package/src/endpoints/organization/dashboard/webshops/SearchUitpasEventsEndpoint.ts +6 -4
  189. package/src/endpoints/organization/dashboard/webshops/VerifyWebshopDomainEndpoint.ts +2 -1
  190. package/src/endpoints/organization/shared/ExchangePaymentEndpoint.ts +6 -4
  191. package/src/endpoints/organization/shared/GetDocumentHtml.ts +4 -3
  192. package/src/endpoints/organization/shared/GetPaymentEndpoint.ts +4 -3
  193. package/src/endpoints/organization/shared/GetUitpasNumberDetailsEndpoint.ts +3 -2
  194. package/src/endpoints/organization/shared/auth/GetOrganizationEndpoint.ts +3 -2
  195. package/src/endpoints/organization/webshops/CheckWebshopDiscountCodesEndpoint.ts +4 -3
  196. package/src/endpoints/organization/webshops/GetOrderByPaymentEndpoint.ts +5 -4
  197. package/src/endpoints/organization/webshops/GetOrderEndpoint.ts +4 -3
  198. package/src/endpoints/organization/webshops/GetTicketsEndpoint.ts +7 -5
  199. package/src/endpoints/organization/webshops/GetWebshopEndpoint.ts +4 -3
  200. package/src/endpoints/organization/webshops/PlaceOrderEndpoint.test.ts +7 -3
  201. package/src/endpoints/organization/webshops/PlaceOrderEndpoint.ts +11 -10
  202. package/src/endpoints/organization/webshops/RetrieveUitpasSocialTariffPriceEndpoint.ts +5 -4
  203. package/src/endpoints/system/HealthEndpoint.test.ts +44 -0
  204. package/src/endpoints/system/HealthEndpoint.ts +14 -6
  205. package/src/excel-loaders/balance-items.ts +19 -17
  206. package/src/excel-loaders/event-notifications.ts +15 -13
  207. package/src/excel-loaders/index.ts +1 -0
  208. package/src/excel-loaders/members.ts +45 -43
  209. package/src/excel-loaders/organizations.ts +26 -25
  210. package/src/excel-loaders/payments.ts +44 -42
  211. package/src/excel-loaders/platform-memberships.ts +202 -0
  212. package/src/excel-loaders/receivable-balances.ts +25 -23
  213. package/src/excel-loaders/registrations.ts +30 -28
  214. package/src/helpers/AddressValidator.test.ts +2 -1
  215. package/src/helpers/AddressValidator.ts +13 -10
  216. package/src/helpers/AdminPermissionChecker.ts +193 -95
  217. package/src/helpers/AuthenticatedStructures.ts +13 -11
  218. package/src/helpers/BuckarooHelper.ts +3 -2
  219. package/src/helpers/Context.ts +8 -6
  220. package/src/helpers/CookieHelper.ts +2 -2
  221. package/src/helpers/FileCache.ts +9 -9
  222. package/src/helpers/ForwardHandler.ts +3 -2
  223. package/src/helpers/GlobalHelper.ts +2 -0
  224. package/src/helpers/GroupBuilder.ts +2 -1
  225. package/src/helpers/GroupedThrottledQueue.test.ts +19 -19
  226. package/src/helpers/LimitedFilteredRequestHelper.ts +1 -1
  227. package/src/helpers/MemberCharger.ts +2 -1
  228. package/src/helpers/MemberUserSyncer.ts +6 -3
  229. package/src/helpers/MembershipCharger.ts +2 -2
  230. package/src/helpers/OrganizationCharger.ts +2 -1
  231. package/src/helpers/PeriodHelper.ts +2 -1
  232. package/src/helpers/SQLTranslatedString.ts +3 -2
  233. package/src/helpers/ServiceFeeHelper.ts +1 -1
  234. package/src/helpers/SetupStepUpdater.ts +6 -5
  235. package/src/helpers/StripeHelper.ts +9 -8
  236. package/src/helpers/TagHelper.test.ts +5 -5
  237. package/src/helpers/TagHelper.ts +2 -1
  238. package/src/helpers/TemporaryMemberAccess.ts +2 -1
  239. package/src/helpers/ThrottledQueue.test.ts +20 -20
  240. package/src/helpers/UitpasTokenRepository.ts +7 -7
  241. package/src/helpers/ViesHelper.ts +5 -4
  242. package/src/helpers/XlsxTransformerColumnHelper.ts +21 -19
  243. package/src/helpers/email-html-helpers.ts +13 -12
  244. package/src/helpers/fetchToAsyncIterator.ts +1 -1
  245. package/src/helpers/outstandingBalanceJoin.ts +2 -1
  246. package/src/helpers/updateMemberDetailsUitpasNumber.ts +5 -4
  247. package/src/middleware/ContextMiddleware.ts +1 -1
  248. package/src/migrate.ts +21 -4
  249. package/src/seeds/0000000003-default-email-templates.ts +1 -1
  250. package/src/seeds/0000000004-single-organization.ts +2 -1
  251. package/src/seeds/1752848561-groups-registration-periods.ts +3 -2
  252. package/src/seeds/1754560914-groups-prices.test.ts +2 -1
  253. package/src/seeds/1754560914-groups-prices.ts +2 -1
  254. package/src/seeds/1755790070-fill-email-recipient-errors.ts +6 -6
  255. package/src/seeds/1755876819-remove-duplicate-members.ts +2 -1
  256. package/src/seeds/1765896674-document-update-year.test.ts +2 -1
  257. package/src/seeds/1773754928-force-save-members.ts +15 -0
  258. package/src/services/AuditLogService.ts +3 -2
  259. package/src/services/BalanceItemPaymentService.ts +2 -2
  260. package/src/services/BalanceItemService.ts +2 -1
  261. package/src/services/BootChecksService.test.ts +33 -0
  262. package/src/services/BootChecksService.ts +21 -0
  263. package/src/services/DatabaseCollationService.test.ts +18 -0
  264. package/src/services/DatabaseCollationService.ts +81 -0
  265. package/src/services/DocumentService.ts +1 -1
  266. package/src/services/EventNotificationService.ts +5 -4
  267. package/src/services/FileSignService.ts +2 -2
  268. package/src/services/InvoiceService.ts +3 -3
  269. package/src/services/MemberNumberService.ts +6 -4
  270. package/src/services/MemberRecordStore.ts +28 -19
  271. package/src/services/PaymentReallocationService.test.ts +2 -1
  272. package/src/services/PaymentReallocationService.ts +2 -1
  273. package/src/services/PaymentService.ts +28 -26
  274. package/src/services/RegistrationService.ts +65 -3
  275. package/src/services/SSOService.ts +13 -9
  276. package/src/services/STPackageService.ts +7 -5
  277. package/src/services/StartupHealthService.ts +15 -0
  278. package/src/services/uitpas/PassholderEndpoints.ts +2 -2
  279. package/src/services/uitpas/UitpasService.ts +11 -8
  280. package/src/services/uitpas/cancelTicketSales.ts +1 -1
  281. package/src/services/uitpas/checkPermissionsFor.ts +9 -9
  282. package/src/services/uitpas/getSocialTariffForEvent.ts +4 -4
  283. package/src/services/uitpas/getSocialTariffForUitpasNumbers.ts +5 -5
  284. package/src/services/uitpas/handleUitpasResponse.ts +1 -1
  285. package/src/services/uitpas/registerTicketSales.ts +4 -4
  286. package/src/services/uitpas/searchUitpasEvents.ts +3 -3
  287. package/src/services/uitpas/searchUitpasOrganizers.ts +3 -3
  288. package/src/sql-filters/audit-logs.ts +2 -1
  289. package/src/sql-filters/balance-item-payments.ts +2 -1
  290. package/src/sql-filters/balance-items.ts +2 -1
  291. package/src/sql-filters/base-registration-filter-compilers.ts +6 -4
  292. package/src/sql-filters/document-templates.ts +2 -1
  293. package/src/sql-filters/documents.ts +2 -1
  294. package/src/sql-filters/email-recipients.ts +2 -1
  295. package/src/sql-filters/emails.ts +2 -1
  296. package/src/sql-filters/event-notifications.ts +2 -1
  297. package/src/sql-filters/events.ts +2 -1
  298. package/src/sql-filters/groups.ts +2 -1
  299. package/src/sql-filters/invoiced-balance-items.ts +2 -1
  300. package/src/sql-filters/invoices.ts +2 -1
  301. package/src/sql-filters/member-responsibility-records.ts +2 -1
  302. package/src/sql-filters/members.ts +8 -7
  303. package/src/sql-filters/orders.ts +3 -2
  304. package/src/sql-filters/organization-registration-periods.ts +2 -1
  305. package/src/sql-filters/organizations.ts +2 -1
  306. package/src/sql-filters/payments.ts +2 -1
  307. package/src/sql-filters/platform-memberships.ts +67 -0
  308. package/src/sql-filters/receivable-balances.ts +2 -1
  309. package/src/sql-filters/registration-periods.ts +2 -1
  310. package/src/sql-filters/registrations.ts +2 -1
  311. package/src/sql-filters/tickets.ts +2 -1
  312. package/src/sql-filters/users.ts +2 -1
  313. package/src/sql-filters/webshops.ts +38 -0
  314. package/src/sql-sorters/audit-logs.ts +3 -2
  315. package/src/sql-sorters/balance-items.ts +3 -2
  316. package/src/sql-sorters/document-templates.ts +3 -2
  317. package/src/sql-sorters/documents.ts +3 -2
  318. package/src/sql-sorters/email-recipients.ts +3 -2
  319. package/src/sql-sorters/emails.ts +3 -2
  320. package/src/sql-sorters/event-notifications.ts +3 -2
  321. package/src/sql-sorters/events.ts +3 -2
  322. package/src/sql-sorters/groups.ts +3 -2
  323. package/src/sql-sorters/invoices.ts +3 -2
  324. package/src/sql-sorters/members.ts +3 -2
  325. package/src/sql-sorters/orders.ts +3 -2
  326. package/src/sql-sorters/organization-registration-periods.ts +3 -2
  327. package/src/sql-sorters/organizations.ts +3 -2
  328. package/src/sql-sorters/payments.ts +3 -2
  329. package/src/sql-sorters/platform-memberships.ts +40 -0
  330. package/src/sql-sorters/receivable-balances.ts +3 -2
  331. package/src/sql-sorters/registration-periods.ts +3 -2
  332. package/src/sql-sorters/registrations.ts +3 -2
  333. package/src/sql-sorters/tickets.ts +3 -2
  334. package/src/sql-sorters/webshops.ts +40 -0
  335. package/tests/actions/patchOrganizationMember.ts +5 -4
  336. package/tests/actions/patchPaymentStatus.ts +2 -2
  337. package/tests/actions/patchUserMember.ts +6 -4
  338. package/tests/e2e/api-rate-limits.test.ts +4 -5
  339. package/tests/e2e/bundle-discounts.test.ts +3 -2
  340. package/tests/e2e/charge-members.test.ts +7 -5
  341. package/tests/e2e/documents.test.ts +3 -2
  342. package/tests/e2e/private-files.test.ts +11 -13
  343. package/tests/e2e/register.test.ts +6 -5
  344. package/tests/e2e/stock.test.ts +6 -8
  345. package/tests/e2e/tickets.test.ts +4 -2
  346. package/tests/helpers/StripeMocker.ts +3 -3
  347. package/tests/init/initAdmin.ts +4 -2
  348. package/tests/init/initBundleDiscount.ts +3 -2
  349. package/tests/init/initPayconiq.ts +1 -1
  350. package/tests/init/initPermissionRole.ts +4 -2
  351. package/tests/init/initPlatformRecordCategory.ts +1 -1
  352. package/tests/init/initStripe.ts +1 -1
  353. package/tests/vitest.global.setup.ts +26 -0
  354. package/tests/{jest.setup.ts → vitest.setup.ts} +4 -3
  355. package/tsconfig.build.json +17 -0
  356. package/tsconfig.json +10 -41
  357. package/tsconfig.test.json +17 -0
  358. package/vitest.config.js +13 -0
  359. package/eslint.config.mjs +0 -5
  360. package/jest.config.cjs +0 -27
  361. package/tests/jest.global.setup.ts +0 -33
  362. package/tests/toMatchMap.ts +0 -68
package/package.json CHANGED
@@ -1,35 +1,45 @@
1
1
  {
2
2
  "name": "@stamhoofd/backend",
3
- "version": "2.118.1",
3
+ "version": "2.120.0",
4
+ "type": "module",
4
5
  "main": "./dist/index.js",
5
6
  "exports": {
6
7
  ".": {
8
+ "types": "./dist/index.d.ts",
9
+ "import": "./dist/index.js",
7
10
  "require": "./dist/index.js"
8
11
  },
9
- "./src/migrate": {
12
+ "./migrate": {
13
+ "types": "./dist/src/migrate.d.ts",
14
+ "import": "./dist/src/migrate.js",
10
15
  "require": "./dist/src/migrate.js"
11
16
  },
12
- "./src/boot": {
17
+ "./boot": {
18
+ "types": "./dist/src/boot.d.ts",
19
+ "import": "./dist/src/boot.js",
13
20
  "require": "./dist/src/boot.js"
14
21
  },
15
22
  "./tests/helpers": {
23
+ "@stamhoofd/source": "./tests/helpers/index.ts",
24
+ "types": "./dist/tests/helpers/index.d.ts",
25
+ "import": "./dist/tests/helpers/index.js",
16
26
  "require": "./dist/tests/helpers/index.js"
17
27
  }
18
28
  },
19
29
  "license": "UNLICENCED",
20
30
  "scripts": {
21
31
  "dev": "wait-on ../../shared/middleware/dist/index.js && concurrently -r 'yarn -s build --watch --preserveWatchOutput' \"yarn -s dev:watch\"",
32
+ "dev:build": "yarn -s build",
22
33
  "dev:full": "yarn -s dev",
23
34
  "dev:watch": "wait-on ./dist/index.js && nodemon --quiet --inspect=5858 --watch dist --watch '../../../shared/*/dist/' --watch '../../shared/*/dist/' --ext .ts,.json,.sql,.js --delay 2000ms --exec 'node --enable-source-maps ./dist/index.js' --signal SIGTERM",
24
35
  "dev:backend": "yarn -s dev",
25
- "build": "rm -rf ./dist/src/migrations && rm -rf ./dist/src/seeds && yarn -s copy-assets && tsc -b",
26
- "copy-assets": "rsync --delete --mkpath --exclude='*.ts' --exclude='*.js' -r --checksum ./src/ ./dist/src/",
36
+ "build": "tsc --build tsconfig.build.json --clean && yarn -s copy-assets && tsc --build tsconfig.build.json",
37
+ "copy-assets": "rsync --delete --mkpath --exclude='*.ts' --exclude='*.js' --exclude='*.map' -r --checksum ./src/ ./dist/src/",
27
38
  "build:full": "yarn -s clear && yarn -s build",
28
- "clear": "rm -rf ./dist",
39
+ "clear": "rm -rf ./dist && rm -f *.tsbuildinfo",
29
40
  "start": "yarn -s build && node --enable-source-maps ./dist/index.js",
30
- "test": "jest --runInBand",
31
- "test:build": "yarn -s build:full && yarn -s test",
32
- "test:reset": "yarn -s build:full && jest --runInBand",
41
+ "test": "vitest",
42
+ "test:reset": "yarn -s build:full && yarn -s test",
33
43
  "migrations": "yarn -s build:full && node ./dist/migrations.js",
34
44
  "lint": "eslint"
35
45
  },
@@ -51,18 +61,18 @@
51
61
  "@aws-sdk/s3-request-presigner": "^3.839.0",
52
62
  "@bwip-js/node": "^4.5.1",
53
63
  "@mollie/api-client": "4.3.3",
54
- "@simonbackx/simple-database": "1.34.0",
55
- "@simonbackx/simple-encoding": "2.23.1",
56
- "@simonbackx/simple-endpoints": "1.20.1",
64
+ "@simonbackx/simple-database": "1.36.12",
65
+ "@simonbackx/simple-encoding": "2.26.5",
66
+ "@simonbackx/simple-endpoints": "1.21.0",
57
67
  "@simonbackx/simple-logging": "^1.0.1",
58
- "@stamhoofd/backend-i18n": "2.118.1",
59
- "@stamhoofd/backend-middleware": "2.118.1",
60
- "@stamhoofd/email": "2.118.1",
61
- "@stamhoofd/models": "2.118.1",
62
- "@stamhoofd/queues": "2.118.1",
63
- "@stamhoofd/sql": "2.118.1",
64
- "@stamhoofd/structures": "2.118.1",
65
- "@stamhoofd/utility": "2.118.1",
68
+ "@stamhoofd/backend-i18n": "2.120.0",
69
+ "@stamhoofd/backend-middleware": "2.120.0",
70
+ "@stamhoofd/email": "2.120.0",
71
+ "@stamhoofd/models": "2.120.0",
72
+ "@stamhoofd/queues": "2.120.0",
73
+ "@stamhoofd/sql": "2.120.0",
74
+ "@stamhoofd/structures": "2.120.0",
75
+ "@stamhoofd/utility": "2.120.0",
66
76
  "archiver": "^7.0.1",
67
77
  "axios": "^1.13.2",
68
78
  "cookie": "^0.7.0",
@@ -72,7 +82,7 @@
72
82
  "luxon": "3.4.4",
73
83
  "mailparser": "^3.9.1",
74
84
  "mockdate": "^3.0.2",
75
- "mysql2": "^3.14.1",
85
+ "mysql2": "^3.20.0",
76
86
  "node-rsa": "1.1.1",
77
87
  "openid-client": "^5.4.0",
78
88
  "stripe": "^16.6.0"
@@ -80,5 +90,5 @@
80
90
  "publishConfig": {
81
91
  "access": "public"
82
92
  },
83
- "gitHead": "7461e7ca17f68233be8a8acf1943f2d7882244fd"
93
+ "gitHead": "f38f79c15ce16b0c8c14743ff3eb61feda5a18d4"
84
94
  }
@@ -1,7 +1,9 @@
1
- import { Model, ModelEvent } from '@simonbackx/simple-database';
1
+ import type { ModelEvent } from '@simonbackx/simple-database';
2
+ import { Model } from '@simonbackx/simple-database';
2
3
  import { AuditLog } from '@stamhoofd/models';
3
4
  import { ObjectDiffer } from '@stamhoofd/object-differ';
4
- import { AuditLogPatchItem, AuditLogPatchItemType, AuditLogReplacement, AuditLogSource, AuditLogType } from '@stamhoofd/structures';
5
+ import type { AuditLogType } from '@stamhoofd/structures';
6
+ import { AuditLogPatchItem, AuditLogPatchItemType, AuditLogReplacement, AuditLogSource } from '@stamhoofd/structures';
5
7
  import { ContextInstance } from '../helpers/Context.js';
6
8
  import { AuditLogService } from '../services/AuditLogService.js';
7
9
 
@@ -50,7 +50,7 @@ export const OrderLogger = new ModelLogger(Order, {
50
50
  id: model.id,
51
51
  value: model.number ? `bestelling #${model.number}` : `bestelling van ${model.data.customer.name}`,
52
52
  type: AuditLogReplacementType.Order,
53
- description: model.number ? model.data.customer.name : $t(`4597a3f5-6a6a-44a3-85da-5240ab08447a`),
53
+ description: model.number ? model.data.customer.name : $t(`%Cq`),
54
54
  })],
55
55
  ]);
56
56
  },
package/src/boot.ts CHANGED
@@ -16,12 +16,14 @@ import { SetupStepUpdater } from './helpers/SetupStepUpdater.js';
16
16
  import { ContextMiddleware } from './middleware/ContextMiddleware.js';
17
17
  import { AuditLogService } from './services/AuditLogService.js';
18
18
  import { BalanceItemService } from './services/BalanceItemService.js';
19
+ import { BootChecksService } from './services/BootChecksService.js';
19
20
  import { CpuService } from './services/CpuService.js';
20
21
  import { DocumentService } from './services/DocumentService.js';
21
22
  import { FileSignService } from './services/FileSignService.js';
22
23
  import { PlatformMembershipService } from './services/PlatformMembershipService.js';
23
24
  import { UitpasService } from './services/uitpas/UitpasService.js';
24
25
  import { UniqueUserService } from './services/UniqueUserService.js';
26
+ import { AutoEncoder } from '@simonbackx/simple-encoding';
25
27
 
26
28
  process.on('unhandledRejection', (error: Error) => {
27
29
  console.error('unhandledRejection');
@@ -35,16 +37,25 @@ Column.setJSONVersion(Version);
35
37
  // Set timezone
36
38
  process.env.TZ = 'UTC';
37
39
 
40
+ if (STAMHOOFD.environment === 'development') {
41
+ Error.stackTraceLimit = 100;
42
+ }
43
+
38
44
  // Quick check
39
45
  if (new Date().getTimezoneOffset() !== 0) {
40
46
  throw new Error('Process should always run in UTC timezone');
41
47
  }
42
48
 
43
- const seeds = async () => {
49
+ const seeds = async (options: {shutdown: () => Promise<void>}) => {
44
50
  try {
45
51
  // Internal
46
52
  await AuditLogService.disable(async () => {
47
- await Migration.runAll(__dirname + '/seeds');
53
+ if (!await Migration.runAll(import.meta.dirname + '/seeds')) {
54
+ console.error('Seeds failed!')
55
+ if (STAMHOOFD.environment === 'test' || STAMHOOFD.environment === 'development') {
56
+ await options.shutdown();
57
+ }
58
+ }
48
59
  });
49
60
  }
50
61
  catch (e) {
@@ -61,14 +72,21 @@ function productionLog(message: string) {
61
72
  }
62
73
 
63
74
  export const boot = async (options: { killProcess: boolean }) => {
75
+ // Make sure we use the current environment for connecting to the database
76
+ await Database.reload({})
77
+
64
78
  productionLog('Running server at v' + Version);
65
79
  productionLog('Running server at port ' + STAMHOOFD.PORT);
66
80
  productionLog('Running server on DB ' + process.env.DB_DATABASE); // note, should always use process env here
81
+ console.log('Node version is: ' + process.version);
82
+
83
+ AutoEncoder.skipDefaultValuesVersion = 10000; // todo
67
84
 
68
85
  loadLogger();
69
86
 
70
87
  await GlobalHelper.load();
71
88
  await UniqueUserService.check();
89
+ await BootChecksService.checkDatabaseCollation();
72
90
 
73
91
  // Init platform shared struct: otherwise permissions won't work with missing responsibilities
74
92
  productionLog('Loading platform...');
@@ -79,17 +97,17 @@ export const boot = async (options: { killProcess: boolean }) => {
79
97
  productionLog('Loading endpoints...');
80
98
 
81
99
  // Note: we should load endpoints one by once to have a reliable order of url matching
82
- await router.loadAllEndpoints(__dirname + '/endpoints/global/*');
83
- await router.loadAllEndpoints(__dirname + '/endpoints/admin/*');
84
- await router.loadAllEndpoints(__dirname + '/endpoints/frontend');
85
-
86
- await router.loadAllEndpoints(__dirname + '/endpoints/auth');
87
- await router.loadAllEndpoints(__dirname + '/endpoints/organization/dashboard/*');
88
- await router.loadAllEndpoints(__dirname + '/endpoints/organization/registration');
89
- await router.loadAllEndpoints(__dirname + '/endpoints/organization/webshops');
90
- await router.loadAllEndpoints(__dirname + '/endpoints/organization/shared');
91
- await router.loadAllEndpoints(__dirname + '/endpoints/organization/shared/*');
92
- await router.loadAllEndpoints(__dirname + '/endpoints/system');
100
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/global/*');
101
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/admin/*');
102
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/frontend');
103
+
104
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/auth');
105
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/organization/dashboard/*');
106
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/organization/registration');
107
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/organization/webshops');
108
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/organization/shared');
109
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/organization/shared/*');
110
+ await router.loadAllEndpoints(import.meta.dirname + '/endpoints/system');
93
111
 
94
112
  router.endpoints.push(new CORSPreflightEndpoint());
95
113
 
@@ -257,7 +275,7 @@ export const boot = async (options: { killProcess: boolean }) => {
257
275
  UitpasService.listen();
258
276
 
259
277
  startCrons();
260
- seeds().catch(console.error);
278
+ seeds({ shutdown }).catch(console.error);
261
279
 
262
280
  return { shutdown };
263
281
  };
@@ -1,7 +1,9 @@
1
1
  import { registerCron } from '@stamhoofd/crons';
2
2
  import { CachedBalance, Email, EmailRecipient, Organization, User } from '@stamhoofd/models';
3
- import { IterableSQLSelect, readDynamicSQLExpression, SQL, SQLCalculation, SQLPlusSign } from '@stamhoofd/sql';
4
- import { EmailRecipientFilter, EmailRecipientFilterType, EmailRecipientSubfilter, EmailTemplateType, OrganizationEmail, ReceivableBalanceType, StamhoofdFilter } from '@stamhoofd/structures';
3
+ import type { IterableSQLSelect} from '@stamhoofd/sql';
4
+ import { readDynamicSQLExpression, SQL, SQLCalculation, SQLPlusSign } from '@stamhoofd/sql';
5
+ import type { OrganizationEmail, StamhoofdFilter } from '@stamhoofd/structures';
6
+ import { EmailRecipientFilter, EmailRecipientFilterType, EmailRecipientSubfilter, EmailTemplateType, ReceivableBalanceType } from '@stamhoofd/structures';
5
7
  import { ContextInstance } from '../helpers/Context.js';
6
8
 
7
9
  registerCron('balanceEmails', balanceEmails);
@@ -3,8 +3,8 @@ import fs from 'fs/promises';
3
3
  import { clearExcelCacheHelper } from './clearExcelCache.js';
4
4
 
5
5
  const testPath = '/Users/user/project/backend/app/api/.cache';
6
- jest.mock('fs/promises');
7
- const fsMock = jest.mocked(fs, { shallow: true });
6
+ vitest.mock('fs/promises');
7
+ const fsMock = vitest.mocked(fs, { deep: false });
8
8
 
9
9
  describe('clearExcelCacheHelper', () => {
10
10
  it('should only run between 3 and 6 AM', async () => {
@@ -34,7 +34,7 @@ describe('clearExcelCacheHelper', () => {
34
34
  environment: 'production',
35
35
  });
36
36
 
37
- expect(didClear).toBeFalse();
37
+ expect(didClear).toBe(false);
38
38
  }
39
39
 
40
40
  for (const date of shouldPass) {
@@ -45,7 +45,7 @@ describe('clearExcelCacheHelper', () => {
45
45
  environment: 'production',
46
46
  });
47
47
 
48
- expect(didClear).toBeTrue();
48
+ expect(didClear).toBe(true);
49
49
  }
50
50
  // #endregion
51
51
  });
@@ -70,7 +70,7 @@ describe('clearExcelCacheHelper', () => {
70
70
  environment: 'production',
71
71
  });
72
72
 
73
- expect(didClearSecondTry).toBeFalse();
73
+ expect(didClearSecondTry).toBe(false);
74
74
 
75
75
  // third try, should pass because the last clear was more than a day ago
76
76
  const didClearThirdTry = await clearExcelCacheHelper({
@@ -80,7 +80,7 @@ describe('clearExcelCacheHelper', () => {
80
80
  environment: 'production',
81
81
  });
82
82
 
83
- expect(didClearThirdTry).toBeTrue();
83
+ expect(didClearThirdTry).toBe(true);
84
84
 
85
85
  // fourth try, should fail because 5 min earlier the cache was cleared
86
86
  const didClearFourthTry = await clearExcelCacheHelper({
@@ -90,7 +90,7 @@ describe('clearExcelCacheHelper', () => {
90
90
  environment: 'production',
91
91
  });
92
92
 
93
- expect(didClearFourthTry).toBeFalse();
93
+ expect(didClearFourthTry).toBe(false);
94
94
  // #endregion
95
95
  });
96
96
 
@@ -132,7 +132,7 @@ describe('clearExcelCacheHelper', () => {
132
132
  });
133
133
 
134
134
  fsMock.readdir.mockReturnValue(
135
- // eslint-disable-next-line @typescript-eslint/no-unsafe-argument
135
+
136
136
  Promise.resolve([...directories, file1]) as any,
137
137
  );
138
138
  // #endregion
@@ -1,5 +1,6 @@
1
- import { CachedBalance } from '@stamhoofd/models';
2
1
  import { registerCron } from '@stamhoofd/crons';
2
+ import { CachedBalance } from '@stamhoofd/models';
3
+ import { ReceivableBalanceType } from '@stamhoofd/structures';
3
4
 
4
5
  registerCron('updateCachedBalances', updateCachedBalances);
5
6
 
@@ -11,28 +12,53 @@ async function updateCachedBalances() {
11
12
 
12
13
  const balances = await CachedBalance.select().where(
13
14
  CachedBalance.whereNeedsUpdate(),
14
- ).limit(100).fetch();
15
+ )
16
+ /**
17
+ * Cached balances of member should be updated before the cached balances of the user
18
+ * because the cached balances of the user are dependent on the cached balances of the member.
19
+ */
20
+ .orderBy('objectType', 'ASC')
21
+ // minimize the number of queries when updating
22
+ .orderBy('organizationId', 'ASC')
23
+ .limit(100).fetch();
15
24
 
16
- // Group by object type and by organization id
17
- const grouped = new Map<string, CachedBalance[]>();
25
+ // Group by object type, next by organization id
26
+ const grouped = new Map<ReceivableBalanceType, Map<string, CachedBalance[]>>();
18
27
 
19
28
  for (const balance of balances) {
20
- const key = balance.objectType + '-' + balance.organizationId;
21
- const arr = grouped.get(key);
29
+ const objectType = balance.objectType;
30
+ let organizationMap = grouped.get(objectType);
31
+ if (!organizationMap) {
32
+ organizationMap = new Map<string, CachedBalance[]>();
33
+ grouped.set(objectType, organizationMap);
34
+ }
22
35
 
23
- if (!arr) {
24
- grouped.set(key, [balance]);
36
+ const organizationId = balance.organizationId;
37
+ const balances = organizationMap.get(organizationId);
38
+
39
+ if (!balances) {
40
+ organizationMap.set(organizationId, [balance]);
25
41
  continue;
26
42
  }
27
43
 
28
- arr.push(balance);
44
+ balances.push(balance);
29
45
  }
30
46
 
31
- for (const [_, balances] of grouped) {
32
- const balance = balances[0];
47
+ // update member balances first
48
+ const objectTypeMemberFirst = Array.from(grouped.entries());
49
+ objectTypeMemberFirst.sort((a, b) => {
50
+ const aIsMember = a[0] === ReceivableBalanceType.member;
51
+ const bIsMember = b[0] === ReceivableBalanceType.member;
52
+ if (aIsMember && !bIsMember) return -1;
53
+ if (!aIsMember && bIsMember) return 1;
54
+ return 0;
55
+ });
33
56
 
34
- const ids = balances.map(b => b.objectId);
35
- console.log('Updating', ids.length, balance.objectType, 'for', balance.organizationId);
36
- await CachedBalance.updateForObjects(balance.organizationId, ids, balance.objectType);
57
+ for (const [objectType, organizationMap] of objectTypeMemberFirst) {
58
+ for (const [organizationId, balances] of organizationMap) {
59
+ const ids = balances.map(b => b.objectId);
60
+ console.log('Updating', ids.length, objectType, 'for', organizationId);
61
+ await CachedBalance.updateForObjects(organizationId, ids, objectType);
62
+ }
37
63
  }
38
64
  }
package/src/debug.ts CHANGED
@@ -1,10 +1,11 @@
1
1
  import { Column } from '@simonbackx/simple-database';
2
- import { EncodedResponse, ResponseMiddleware, Router, RouterServer } from '@simonbackx/simple-endpoints';
2
+ import type { ResponseMiddleware, RouterServer } from '@simonbackx/simple-endpoints';
3
+ import { EncodedResponse, Router } from '@simonbackx/simple-endpoints';
3
4
  import { logger } from '@simonbackx/simple-logging';
4
5
  import { requestPrefix } from '@stamhoofd/backend-middleware';
5
6
 
6
7
  import { AutoEncoder } from '@simonbackx/simple-encoding';
7
- import { Request, Response } from '@simonbackx/simple-endpoints';
8
+ import type { Request, Response } from '@simonbackx/simple-endpoints';
8
9
  import { SQLLogger } from '@stamhoofd/sql';
9
10
  import { ContextInstance } from './helpers/Context.js';
10
11
 
@@ -1,4 +1,4 @@
1
- import { Data, Decoder } from '@simonbackx/simple-encoding';
1
+ import type { Data, Decoder } from '@simonbackx/simple-encoding';
2
2
 
3
3
  export class StringArrayDecoder<T> implements Decoder<T[]> {
4
4
  decoder: Decoder<T>;
@@ -1,4 +1,4 @@
1
- import { Decoder, Data } from '@simonbackx/simple-encoding';
1
+ import type { Decoder, Data } from '@simonbackx/simple-encoding';
2
2
 
3
3
  export class StringNullableDecoder<T> implements Decoder<T | null> {
4
4
  decoder: Decoder<T>;
@@ -1,6 +1,7 @@
1
1
  import { Email, Member } from '@stamhoofd/models';
2
2
  import { SQL } from '@stamhoofd/sql';
3
- import { EmailRecipient, EmailRecipientFilterType, LimitedFilteredRequest, PaginatedResponse, Replacement } from '@stamhoofd/structures';
3
+ import type { LimitedFilteredRequest} from '@stamhoofd/structures';
4
+ import { EmailRecipient, EmailRecipientFilterType, PaginatedResponse, Replacement } from '@stamhoofd/structures';
4
5
  import { GetDocumentsEndpoint } from '../endpoints/organization/dashboard/documents/GetDocumentsEndpoint.js';
5
6
 
6
7
  async function fetch(query: LimitedFilteredRequest) {
@@ -1,6 +1,7 @@
1
1
  import { Email } from '@stamhoofd/models';
2
2
  import { SQL } from '@stamhoofd/sql';
3
- import { EmailRecipient, EmailRecipientFilterType, LimitedFilteredRequest, MembersBlob, PaginatedResponse, mergeFilters } from '@stamhoofd/structures';
3
+ import type { EmailRecipient, LimitedFilteredRequest, MembersBlob} from '@stamhoofd/structures';
4
+ import { EmailRecipientFilterType, PaginatedResponse, mergeFilters } from '@stamhoofd/structures';
4
5
  import { GetMembersEndpoint } from '../endpoints/global/members/GetMembersEndpoint.js';
5
6
 
6
7
  async function getRecipients(result: PaginatedResponse<MembersBlob, LimitedFilteredRequest>, type: 'member' | 'parents' | 'unverified') {
@@ -1,5 +1,6 @@
1
1
  import { Email, Webshop } from '@stamhoofd/models';
2
- import { EmailRecipient, EmailRecipientFilterType, LimitedFilteredRequest, mergeFilters, PaginatedResponse, WebshopPreview } from '@stamhoofd/structures';
2
+ import type { EmailRecipient, LimitedFilteredRequest, WebshopPreview } from '@stamhoofd/structures';
3
+ import { EmailRecipientFilterType, mergeFilters, PaginatedResponse } from '@stamhoofd/structures';
3
4
  import { GetWebshopOrdersEndpoint } from '../endpoints/organization/dashboard/webshops/GetWebshopOrdersEndpoint.js';
4
5
  import { AuthenticatedStructures } from '../helpers/AuthenticatedStructures.js';
5
6
  import { Context } from '../helpers/Context.js';
@@ -1,10 +1,13 @@
1
- import { BalanceItem, BalanceItemPayment, Email, Member, MemberResponsibilityRecord, Order, Organization, Payment, RecipientLoader, User, Webshop } from '@stamhoofd/models';
1
+ import type { RecipientLoader} from '@stamhoofd/models';
2
+ import { BalanceItem, BalanceItemPayment, Email, Member, MemberResponsibilityRecord, Order, Organization, Payment, User, Webshop } from '@stamhoofd/models';
2
3
  import { compileToSQLFilter, SQL } from '@stamhoofd/sql';
3
- import { CountFilteredRequest, EmailRecipient, EmailRecipientFilterType, LimitedFilteredRequest, PaginatedResponse, PaymentGeneral, PaymentMethod, StamhoofdFilter } from '@stamhoofd/structures';
4
+ import type { LimitedFilteredRequest, PaymentGeneral, StamhoofdFilter } from '@stamhoofd/structures';
5
+ import { CountFilteredRequest, EmailRecipient, EmailRecipientFilterType, PaginatedResponse, PaymentMethod } from '@stamhoofd/structures';
4
6
  import { Formatter } from '@stamhoofd/utility';
5
7
  import { GetPaymentsEndpoint } from '../endpoints/organization/dashboard/payments/GetPaymentsEndpoint.js';
6
8
  import { memberResponsibilityRecordFilterCompilers } from '../sql-filters/member-responsibility-records.js';
7
- import { buildReplacementOptions, getEmailReplacementsForPayment, ReplacementsOptions } from '../email-replacements/getEmailReplacementsForPayment.js';
9
+ import type { ReplacementsOptions } from '../email-replacements/getEmailReplacementsForPayment.js';
10
+ import { buildReplacementOptions, getEmailReplacementsForPayment } from '../email-replacements/getEmailReplacementsForPayment.js';
8
11
 
9
12
  type BeforeFetchAllResult = {
10
13
  doesIncludePaymentWithoutOrders: boolean;
@@ -1,5 +1,6 @@
1
1
  import { CachedBalance, Email } from '@stamhoofd/models';
2
- import { BalanceItem as BalanceItemStruct, compileToInMemoryFilter, EmailRecipient, EmailRecipientFilterType, LimitedFilteredRequest, PaginatedResponse, receivableBalanceObjectContactInMemoryFilterCompilers, ReceivableBalanceType, Replacement, StamhoofdFilter } from '@stamhoofd/structures';
2
+ import type { LimitedFilteredRequest, StamhoofdFilter } from '@stamhoofd/structures';
3
+ import { BalanceItem as BalanceItemStruct, compileToInMemoryFilter, EmailRecipient, EmailRecipientFilterType, PaginatedResponse, receivableBalanceObjectContactInMemoryFilterCompilers, ReceivableBalanceType, Replacement } from '@stamhoofd/structures';
3
4
  import { Formatter } from '@stamhoofd/utility';
4
5
  import { GetReceivableBalancesEndpoint } from '../endpoints/organization/dashboard/receivable-balances/GetReceivableBalancesEndpoint.js';
5
6
 
@@ -1,6 +1,7 @@
1
1
  import { Email, Member } from '@stamhoofd/models';
2
2
  import { SQL } from '@stamhoofd/sql';
3
- import { EmailRecipient, EmailRecipientFilterType, LimitedFilteredRequest, PaginatedResponse, RegistrationsBlob, mergeFilters } from '@stamhoofd/structures';
3
+ import type { EmailRecipient, LimitedFilteredRequest, RegistrationsBlob} from '@stamhoofd/structures';
4
+ import { EmailRecipientFilterType, PaginatedResponse, mergeFilters } from '@stamhoofd/structures';
4
5
  import { GetRegistrationsEndpoint } from '../endpoints/global/registration/GetRegistrationsEndpoint.js';
5
6
  import { memberJoin } from '../sql-filters/registrations.js';
6
7
 
@@ -1,5 +1,6 @@
1
1
  import { Order, Organization, Webshop } from '@stamhoofd/models';
2
- import { BalanceItemPaymentsHtmlTableItem, BalanceItemRelationType, BalanceItemType, getBalanceItemPaymentsHtmlTable, PaymentGeneral, PaymentMethod, PaymentMethodHelper, Replacement, Webshop as WebshopStruct } from '@stamhoofd/structures';
2
+ import type { BalanceItemPaymentsHtmlTableItem, PaymentGeneral} from '@stamhoofd/structures';
3
+ import { BalanceItemRelationType, BalanceItemType, getBalanceItemPaymentsHtmlTable, PaymentMethod, PaymentMethodHelper, Replacement, Webshop as WebshopStruct } from '@stamhoofd/structures';
3
4
  import { Formatter } from '@stamhoofd/utility';
4
5
  import { createOrderDataHTMLTable } from '../helpers/email-html-helpers.js';
5
6
 
@@ -216,7 +217,7 @@ function getPaymentContext(payment: PaymentGeneral, { orderMap, webshopMap }: Re
216
217
  const webshop = webshopMap.get(order.webshopId);
217
218
  if (webshop) {
218
219
  if (order.number) {
219
- overviewContext.add($t('80c86ad4-5ca1-481a-a408-db018a81f25e', {
220
+ overviewContext.add($t('%1Kz', {
220
221
  webshop: webshop.meta.name,
221
222
  orderNumber: order.number ?? '',
222
223
  }));
@@ -226,7 +227,7 @@ function getPaymentContext(payment: PaymentGeneral, { orderMap, webshopMap }: Re
226
227
  }
227
228
  }
228
229
  else {
229
- overviewContext.add($t('8bb27033-cf3c-4af1-a3f4-3373734ab189', {
230
+ overviewContext.add($t('%1L0', {
230
231
  orderNumber: order.number ?? '',
231
232
  }));
232
233
  }
@@ -252,8 +253,8 @@ function getPaymentContext(payment: PaymentGeneral, { orderMap, webshopMap }: Re
252
253
  }
253
254
 
254
255
  if (registrationMemberNames.size > 0) {
255
- const memberNames = Formatter.joinLast([...registrationMemberNames], ', ', ' ' + $t(`6a156458-b396-4d0f-b562-adb3e38fc51b`) + ' ');
256
- overviewContext.add($t(`01d5fd7e-2960-4eb4-ab3a-2ac6dcb2e39c`) + ' ' + memberNames);
256
+ const memberNames = Formatter.joinLast([...registrationMemberNames], ', ', ' ' + $t(`%M1`) + ' ');
257
+ overviewContext.add($t(`%x6`) + ' ' + memberNames);
257
258
  }
258
259
 
259
260
  if (overviewContext.size === 0) {
@@ -273,11 +274,11 @@ function getPaymentContext(payment: PaymentGeneral, { orderMap, webshopMap }: Re
273
274
  }
274
275
 
275
276
  // else return default text for multiple items
276
- return $t('7462f89e-db05-452f-804f-b9f2d51bff0d', { count: payment.balanceItemPayments.length });
277
+ return $t('%1Kx', { count: payment.balanceItemPayments.length });
277
278
  }
278
279
 
279
280
  // else return default text for single item
280
- return $t('fe349d88-fecf-4cd8-8ea4-9e3b13cfa0dc');
281
+ return $t('%1Ky');
281
282
  }
282
283
 
283
284
  // join texts for balance items with type order or registration
@@ -1,5 +1,6 @@
1
- import { Decoder } from '@simonbackx/simple-encoding';
2
- import { DecodedRequest, Endpoint, Request, Response } from '@simonbackx/simple-endpoints';
1
+ import type { Decoder } from '@simonbackx/simple-encoding';
2
+ import type { DecodedRequest, Request} from '@simonbackx/simple-endpoints';
3
+ import { Endpoint, Response } from '@simonbackx/simple-endpoints';
3
4
  import { SimpleError } from '@simonbackx/simple-errors';
4
5
  import { ChargeRequest, LimitedFilteredRequest, PermissionLevel } from '@stamhoofd/structures';
5
6
 
@@ -35,7 +36,7 @@ export class ChargeMembersEndpoint extends Endpoint<Params, Query, Body, Respons
35
36
  throw new SimpleError({
36
37
  code: 'invalid_field',
37
38
  message: 'Invalid description',
38
- human: $t(`2449fba5-99dc-496f-a9d6-a67263d56616`),
39
+ human: $t(`%Cr`),
39
40
  field: 'description',
40
41
  });
41
42
  }
@@ -44,7 +45,7 @@ export class ChargeMembersEndpoint extends Endpoint<Params, Query, Body, Respons
44
45
  throw new SimpleError({
45
46
  code: 'invalid_field',
46
47
  message: 'Invalid price',
47
- human: $t(`1e165aac-8a58-45c5-bdd8-c58131a7b7f5`),
48
+ human: $t(`%Cs`),
48
49
  field: 'price',
49
50
  });
50
51
  }
@@ -53,7 +54,7 @@ export class ChargeMembersEndpoint extends Endpoint<Params, Query, Body, Respons
53
54
  throw new SimpleError({
54
55
  code: 'invalid_field',
55
56
  message: 'Invalid amount',
56
- human: $t(`0bdf4953-1eae-41fd-b142-5ad3287f17a7`),
57
+ human: $t(`%Ct`),
57
58
  field: 'amount',
58
59
  });
59
60
  }
@@ -77,7 +78,7 @@ export class ChargeMembersEndpoint extends Endpoint<Params, Query, Body, Respons
77
78
  throw new SimpleError({
78
79
  code: 'charge_pending',
79
80
  message: 'Charge members already pending',
80
- human: $t(`d2b84fdd-035b-4307-a897-000081aa814f`),
81
+ human: $t(`%Cu`),
81
82
  });
82
83
  }
83
84
 
@@ -1,4 +1,5 @@
1
- import { DecodedRequest, Endpoint, Request, Response } from '@simonbackx/simple-endpoints';
1
+ import type { DecodedRequest, Request} from '@simonbackx/simple-endpoints';
2
+ import { Endpoint, Response } from '@simonbackx/simple-endpoints';
2
3
  import { SimpleError } from '@simonbackx/simple-errors';
3
4
  import { QueueHandler } from '@stamhoofd/queues';
4
5
  import { Context } from '../../../helpers/Context.js';
@@ -34,7 +35,7 @@ export class ChargeMembershipsEndpoint extends Endpoint<Params, Query, Body, Res
34
35
  throw new SimpleError({
35
36
  code: 'charge_pending',
36
37
  message: 'Charge already pending',
37
- human: $t(`d2b84fdd-035b-4307-a897-000081aa814f`),
38
+ human: $t(`%Cu`),
38
39
  });
39
40
  }
40
41
 
@@ -1,4 +1,5 @@
1
- import { DecodedRequest, Endpoint, Request, Response } from '@simonbackx/simple-endpoints';
1
+ import type { DecodedRequest, Request} from '@simonbackx/simple-endpoints';
2
+ import { Endpoint, Response } from '@simonbackx/simple-endpoints';
2
3
  import { Platform } from '@stamhoofd/models';
3
4
  import { QueueHandler } from '@stamhoofd/queues';
4
5
  import { SQL, SQLAlias, SQLCount, SQLDistinct, SQLSelectAs, SQLSum, SQLWhereSign } from '@stamhoofd/sql';
@@ -1,5 +1,6 @@
1
- import { Decoder } from '@simonbackx/simple-encoding';
2
- import { DecodedRequest, Endpoint, Request, Response } from '@simonbackx/simple-endpoints';
1
+ import type { Decoder } from '@simonbackx/simple-encoding';
2
+ import type { DecodedRequest, Request} from '@simonbackx/simple-endpoints';
3
+ import { Endpoint, Response } from '@simonbackx/simple-endpoints';
3
4
  import { SimpleError } from '@simonbackx/simple-errors';
4
5
  import { ChargeRequest, LimitedFilteredRequest } from '@stamhoofd/structures';
5
6
 
@@ -48,7 +49,7 @@ export class ChargeOrganizationsEndpoint extends Endpoint<Params, Query, Body, R
48
49
  throw new SimpleError({
49
50
  code: 'charge_pending',
50
51
  message: 'Charge organizations already pending',
51
- human: $t(`d2b84fdd-035b-4307-a897-000081aa814f`),
52
+ human: $t(`%Cu`),
52
53
  });
53
54
  }
54
55
 
@@ -1,5 +1,6 @@
1
- import { Decoder } from '@simonbackx/simple-encoding';
2
- import { DecodedRequest, Endpoint, Request, Response } from '@simonbackx/simple-endpoints';
1
+ import type { Decoder } from '@simonbackx/simple-encoding';
2
+ import type { DecodedRequest, Request} from '@simonbackx/simple-endpoints';
3
+ import { Endpoint, Response } from '@simonbackx/simple-endpoints';
3
4
  import { CountFilteredRequest, CountResponse } from '@stamhoofd/structures';
4
5
 
5
6
  import { Context } from '../../../helpers/Context.js';