@stamhoofd/backend 2.58.0 → 2.60.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 (56) hide show
  1. package/index.ts +10 -1
  2. package/package.json +12 -12
  3. package/src/audit-logs/DocumentTemplateLogger.ts +22 -0
  4. package/src/audit-logs/EventLogger.ts +30 -0
  5. package/src/audit-logs/GroupLogger.ts +95 -0
  6. package/src/audit-logs/MemberLogger.ts +24 -0
  7. package/src/audit-logs/MemberPlatformMembershipLogger.ts +60 -0
  8. package/src/audit-logs/MemberResponsibilityRecordLogger.ts +69 -0
  9. package/src/audit-logs/ModelLogger.ts +219 -0
  10. package/src/audit-logs/OrderLogger.ts +57 -0
  11. package/src/audit-logs/OrganizationLogger.ts +16 -0
  12. package/src/audit-logs/OrganizationRegistrationPeriodLogger.ts +77 -0
  13. package/src/audit-logs/PaymentLogger.ts +43 -0
  14. package/src/audit-logs/PlatformLogger.ts +13 -0
  15. package/src/audit-logs/RegistrationLogger.ts +53 -0
  16. package/src/audit-logs/RegistrationPeriodLogger.ts +21 -0
  17. package/src/audit-logs/StripeAccountLogger.ts +47 -0
  18. package/src/audit-logs/WebshopLogger.ts +35 -0
  19. package/src/crons/updateSetupSteps.ts +1 -1
  20. package/src/crons.ts +2 -1
  21. package/src/endpoints/global/events/PatchEventsEndpoint.ts +12 -24
  22. package/src/endpoints/global/members/PatchOrganizationMembersEndpoint.ts +7 -21
  23. package/src/endpoints/global/payments/StripeWebhookEndpoint.ts +5 -13
  24. package/src/endpoints/global/platform/PatchPlatformEnpoint.ts +5 -12
  25. package/src/endpoints/global/registration/PatchUserMembersEndpoint.ts +0 -15
  26. package/src/endpoints/global/registration/RegisterMembersEndpoint.ts +43 -27
  27. package/src/endpoints/global/registration-periods/PatchRegistrationPeriodsEndpoint.ts +0 -19
  28. package/src/endpoints/organization/dashboard/organization/PatchOrganizationEndpoint.ts +3 -13
  29. package/src/endpoints/organization/dashboard/registration-periods/PatchOrganizationRegistrationPeriodsEndpoint.ts +20 -43
  30. package/src/endpoints/organization/dashboard/stripe/ConnectStripeEndpoint.ts +0 -6
  31. package/src/endpoints/organization/dashboard/stripe/DeleteStripeAccountEndpoint.ts +0 -6
  32. package/src/endpoints/organization/dashboard/stripe/UpdateStripeAccountEndpoint.ts +5 -14
  33. package/src/endpoints/organization/dashboard/webshops/PatchWebshopOrdersEndpoint.ts +7 -4
  34. package/src/endpoints/organization/webshops/PlaceOrderEndpoint.ts +8 -2
  35. package/src/helpers/AuthenticatedStructures.ts +16 -1
  36. package/src/helpers/Context.ts +8 -2
  37. package/src/helpers/MemberUserSyncer.ts +45 -40
  38. package/src/helpers/PeriodHelper.ts +5 -5
  39. package/src/helpers/SetupStepUpdater.ts +503 -0
  40. package/src/helpers/TagHelper.ts +23 -20
  41. package/src/seeds/1722344162-update-membership.ts +2 -2
  42. package/src/seeds/1726572303-schedule-stock-updates.ts +2 -1
  43. package/src/seeds/1726847064-setup-steps.ts +1 -1
  44. package/src/seeds/1733319079-fill-paying-organization-ids.ts +68 -0
  45. package/src/services/AuditLogService.ts +81 -296
  46. package/src/services/BalanceItemPaymentService.ts +1 -1
  47. package/src/services/BalanceItemService.ts +14 -5
  48. package/src/services/DocumentService.ts +43 -0
  49. package/src/services/MemberNumberService.ts +120 -0
  50. package/src/services/PaymentService.ts +199 -193
  51. package/src/services/PlatformMembershipService.ts +284 -0
  52. package/src/services/RegistrationService.ts +78 -27
  53. package/src/services/diff.ts +512 -0
  54. package/src/sql-filters/events.ts +13 -1
  55. package/src/helpers/MembershipHelper.ts +0 -54
  56. package/src/services/explainPatch.ts +0 -782
@@ -0,0 +1,512 @@
1
+ import { ArrayDecoder, AutoEncoder, EnumDecoder, Field, getOptionalId, isPatchMap, SymbolDecoder } from '@simonbackx/simple-encoding';
2
+ import { AuditLogPatchItem, AuditLogPatchItemType, AuditLogReplacement, AuditLogReplacementType, Version } from '@stamhoofd/structures';
3
+ import { DataValidator, Formatter } from '@stamhoofd/utility';
4
+
5
+ export type PatchExplainer = {
6
+ key: string;
7
+ handler: (oldValue: unknown, value: unknown) => AuditLogPatchItem[];
8
+ };
9
+
10
+ function diffEnum(oldValue: unknown, value: unknown, key?: AuditLogReplacement) {
11
+ if (oldValue === value) {
12
+ return [];
13
+ }
14
+
15
+ if (value === undefined) {
16
+ // Not altered
17
+ return [];
18
+ }
19
+
20
+ return [
21
+ AuditLogPatchItem.create({
22
+ key,
23
+ oldValue: typeof oldValue === 'string' ? AuditLogReplacement.key(oldValue) : undefined,
24
+ value: typeof value === 'string' ? AuditLogReplacement.key(value) : undefined,
25
+ }).autoType(),
26
+ ];
27
+ }
28
+
29
+ function diffDate(oldValue: unknown, value: unknown, key?: AuditLogReplacement) {
30
+ if (!(oldValue instanceof Date) && oldValue !== null) {
31
+ return [];
32
+ }
33
+
34
+ if ((!(value instanceof Date)) && value !== null) {
35
+ return [];
36
+ }
37
+
38
+ if (oldValue?.getTime() === value?.getTime()) {
39
+ return [];
40
+ }
41
+ let dno = oldValue ? Formatter.dateNumber(oldValue, true) : undefined;
42
+ let dn = value ? Formatter.dateNumber(value, true) : undefined;
43
+
44
+ if (dno && dn && (dno === dn || (Formatter.time(oldValue!) !== Formatter.time(value!))) && key?.toKey() !== 'birthDay') {
45
+ // Add time
46
+ dno += ' ' + Formatter.time(oldValue!);
47
+ dn += ' ' + Formatter.time(value!);
48
+ }
49
+
50
+ return [
51
+ AuditLogPatchItem.create({
52
+ key,
53
+ oldValue: dno ? AuditLogReplacement.string(dno) : undefined,
54
+ value: dn ? AuditLogReplacement.string(dn) : undefined,
55
+ }).autoType(),
56
+ ];
57
+ }
58
+
59
+ function getDiffKey(autoEncoder: string): AuditLogReplacement;
60
+ function getDiffKey(autoEncoder: unknown): AuditLogReplacement | undefined;
61
+ function getDiffKey(autoEncoder: unknown): AuditLogReplacement | undefined {
62
+ if (typeof autoEncoder === 'string') {
63
+ if (DataValidator.isUuid(autoEncoder)) {
64
+ return AuditLogReplacement.uuid(autoEncoder);
65
+ }
66
+ return AuditLogReplacement.key(autoEncoder);
67
+ }
68
+ return undefined;
69
+ }
70
+
71
+ /**
72
+ * For arrays or maps, use this name instead of the index/key to identify an object
73
+ */
74
+ function getDiffName(autoEncoder: unknown): AuditLogReplacement | null {
75
+ if (typeof autoEncoder === 'string') {
76
+ if (DataValidator.isUuid(autoEncoder)) {
77
+ return AuditLogReplacement.uuid(autoEncoder);
78
+ }
79
+ return AuditLogReplacement.string(autoEncoder);
80
+ }
81
+
82
+ if (typeof autoEncoder === 'object' && autoEncoder !== null && 'getDiffName' in autoEncoder && typeof autoEncoder.getDiffName === 'function') {
83
+ const name = autoEncoder.getDiffName();
84
+ if (typeof name === 'string') {
85
+ return name ? AuditLogReplacement.string(name) : AuditLogReplacement.key('untitled');
86
+ }
87
+ }
88
+
89
+ if (typeof autoEncoder === 'object' && autoEncoder !== null && 'name' in autoEncoder && typeof autoEncoder.name === 'string') {
90
+ return autoEncoder.name ? AuditLogReplacement.string(autoEncoder.name) : AuditLogReplacement.key('untitled');
91
+ }
92
+ return null;
93
+ }
94
+ function getDiffPutValue(autoEncoder: unknown, key?: AuditLogReplacement): AuditLogReplacement | null {
95
+ if (typeof autoEncoder === 'object' && autoEncoder !== null && 'getDiffPutName' in autoEncoder && typeof autoEncoder.getDiffPutName === 'function') {
96
+ const name = autoEncoder.getDiffPutName();
97
+ if (typeof name === 'string') {
98
+ return AuditLogReplacement.string(name);
99
+ }
100
+ if (name instanceof AuditLogReplacement) {
101
+ return name;
102
+ }
103
+ }
104
+ return getDiffValue(autoEncoder, key);
105
+ }
106
+
107
+ function getDiffValue(autoEncoder: unknown, key?: AuditLogReplacement): AuditLogReplacement | null {
108
+ if (typeof autoEncoder === 'string') {
109
+ if (DataValidator.isUuid(autoEncoder)) {
110
+ return AuditLogReplacement.uuid(autoEncoder);
111
+ }
112
+ if (key && key?.lastValue() === 'status') {
113
+ // Will be an enum
114
+ return AuditLogReplacement.key(autoEncoder);
115
+ }
116
+ return AuditLogReplacement.string(autoEncoder);
117
+ }
118
+
119
+ if (typeof autoEncoder === 'symbol') {
120
+ const name = Symbol.keyFor(autoEncoder);
121
+ if (name) {
122
+ return AuditLogReplacement.key(name);
123
+ }
124
+ return AuditLogReplacement.key('unknown');
125
+ }
126
+
127
+ if (typeof autoEncoder === 'number') {
128
+ const k = key?.lastValue();
129
+ if (k && (k.toLowerCase().includes('price') || k.toLowerCase().includes('fee'))) {
130
+ return AuditLogReplacement.string(Formatter.price(autoEncoder));
131
+ }
132
+ return AuditLogReplacement.string(Formatter.integer(autoEncoder));
133
+ }
134
+
135
+ if (autoEncoder instanceof Date) {
136
+ return AuditLogReplacement.string(Formatter.dateTime(autoEncoder, true, true));
137
+ }
138
+
139
+ if (autoEncoder === true) {
140
+ return AuditLogReplacement.key('on');
141
+ }
142
+
143
+ if (autoEncoder === false) {
144
+ return AuditLogReplacement.key('off');
145
+ }
146
+
147
+ if (typeof autoEncoder === 'object' && autoEncoder !== null && 'getDiffValue' in autoEncoder && typeof autoEncoder.getDiffValue === 'function') {
148
+ const name = autoEncoder.getDiffValue();
149
+ if (typeof name === 'string') {
150
+ return AuditLogReplacement.string(name);
151
+ }
152
+ if (name instanceof AuditLogReplacement) {
153
+ return name;
154
+ }
155
+ }
156
+
157
+ return null;
158
+ }
159
+
160
+ function getKeySingular(key?: string) {
161
+ if (key === undefined) {
162
+ return undefined;
163
+ }
164
+ return key.replace(/ies$/, 'y').replace(/s$/, '');
165
+ }
166
+
167
+ function findOriginalById(id: unknown, oldArray: unknown[]): unknown | null {
168
+ return id ? oldArray.find(v => getId(v) === id) : null;
169
+ }
170
+
171
+ function findOriginalIndexById(id: unknown, oldArray: unknown[]): number {
172
+ return id ? oldArray.findIndex(v => getId(v) === id) : -1;
173
+ }
174
+
175
+ function getId(object: unknown): string | number | null {
176
+ const id = getOptionalId(object);
177
+ if (typeof id !== 'string' && typeof id !== 'number') {
178
+ if (object instanceof AutoEncoder) {
179
+ const encoded = object.encode({ version: Version });
180
+ return JSON.stringify(encoded);
181
+ }
182
+ return JSON.stringify(object);
183
+ }
184
+ return id;
185
+ }
186
+
187
+ function findOriginal(put: unknown, oldArray: unknown[]): unknown | null {
188
+ return findOriginalById(getId(put), oldArray);
189
+ }
190
+
191
+ function findIndex(put: unknown, oldArray: unknown[]): number {
192
+ return findOriginalIndexById(getId(put), oldArray);
193
+ }
194
+
195
+ function diffArray(oldValue: unknown, value: unknown, key?: AuditLogReplacement) {
196
+ if (!Array.isArray(oldValue) || !Array.isArray(value)) {
197
+ // Not supported
198
+ return [];
199
+ }
200
+ const items: AuditLogPatchItem[] = [];
201
+
202
+ // Search for puts
203
+ let orderChanged = false;
204
+ let added = 0;
205
+ for (const [index, newItem] of value.entries()) {
206
+ const originalIndex = findIndex(newItem, oldValue);
207
+
208
+ if (originalIndex === -1) {
209
+ // Has been added
210
+ items.push(...diffUnknown(
211
+ null,
212
+ newItem,
213
+ (getDiffName(newItem) || AuditLogReplacement.key('item')).prepend(key),
214
+ ));
215
+ added++;
216
+ }
217
+ else {
218
+ // Has been overwritten
219
+ const original = oldValue[originalIndex];
220
+ items.push(...diffUnknown(
221
+ original,
222
+ newItem,
223
+ (getDiffName(original) || getDiffName(newItem) || AuditLogReplacement.key('item')).prepend(key),
224
+ ));
225
+
226
+ if ((index - added) !== originalIndex) {
227
+ // Order has changed
228
+ orderChanged = true;
229
+ }
230
+ }
231
+ }
232
+
233
+ // Search for deletes
234
+ for (const original of oldValue) {
235
+ const newItem = findOriginal(original, value);
236
+ if (!newItem) {
237
+ // Has been deleted
238
+ items.push(...diffUnknown(
239
+ original,
240
+ null,
241
+ (getDiffName(original) || AuditLogReplacement.key('item')).prepend(key),
242
+ ));
243
+
244
+ orderChanged = false; // ignore order changed as delete will have set it to true
245
+ }
246
+ }
247
+
248
+ if (orderChanged) {
249
+ // Check if order has changed
250
+ items.push(
251
+ AuditLogPatchItem.create({
252
+ key,
253
+ type: AuditLogPatchItemType.Reordered,
254
+ }),
255
+ );
256
+ }
257
+ // Not supported
258
+ return items;
259
+ };
260
+
261
+ function diffMap(oldValue: unknown, value: unknown, key?: AuditLogReplacement) {
262
+ if (!(value instanceof Map)) {
263
+ // Not supported
264
+ return [];
265
+ }
266
+ if (!(oldValue instanceof Map) && oldValue !== undefined && oldValue !== null) {
267
+ // Not supported
268
+ return [];
269
+ }
270
+
271
+ const items: AuditLogPatchItem[] = [];
272
+ const isPatch = isPatchMap(value);
273
+
274
+ for (const [k, v] of value.entries()) {
275
+ if (typeof k !== 'string') {
276
+ // Not supported
277
+ continue;
278
+ }
279
+ const original = oldValue?.get(k);
280
+
281
+ if (v === null && isPatch) {
282
+ // Delete
283
+ if (original) {
284
+ items.push(
285
+ ...diffUnknown(
286
+ null,
287
+ original,
288
+ (getDiffName(original) || getDiffKey(k)).prepend(key),
289
+ ),
290
+ );
291
+ }
292
+ continue;
293
+ }
294
+
295
+ // Changed
296
+ items.push(
297
+ ...diffUnknown(
298
+ original,
299
+ v,
300
+ (getDiffName(original) || getDiffName(v) || getDiffKey(k)).prepend(key),
301
+ ),
302
+ );
303
+ }
304
+
305
+ if (!isPatch && oldValue) {
306
+ // Loop old values
307
+ for (const [k, v] of oldValue.entries()) {
308
+ if (typeof k !== 'string') {
309
+ // Not supported
310
+ continue;
311
+ }
312
+
313
+ if (value.has(k)) {
314
+ continue;
315
+ }
316
+
317
+ items.push(
318
+ ...diffUnknown(
319
+ null,
320
+ v,
321
+ (getDiffName(v) || getDiffKey(k)).prepend(key),
322
+ ),
323
+ );
324
+ }
325
+ }
326
+
327
+ return items;
328
+ }
329
+
330
+ export function transformValueForDiff(value: unknown) {
331
+ if (typeof value === 'object' && value !== null && 'transformForDiff' in value && typeof value.transformForDiff === 'function') {
332
+ return value.transformForDiff();
333
+ }
334
+ return value;
335
+ }
336
+
337
+ export function diffUnknown(oldValue: unknown, value: unknown, key?: AuditLogReplacement) {
338
+ oldValue = transformValueForDiff(oldValue);
339
+ value = transformValueForDiff(value);
340
+
341
+ if (oldValue === value) {
342
+ return [];
343
+ }
344
+
345
+ if ((oldValue === null || oldValue === undefined) && (value === null || value === undefined)) {
346
+ // Both removed
347
+ return [];
348
+ }
349
+
350
+ if (Array.isArray(oldValue) && Array.isArray(value)) {
351
+ return diffArray(oldValue, value, key);
352
+ }
353
+
354
+ if (value instanceof Map && (oldValue instanceof Map || oldValue === null || oldValue === undefined)) {
355
+ return diffMap(oldValue, value, key);
356
+ }
357
+
358
+ if ((value instanceof Date || value === null) && (oldValue instanceof Date || oldValue === null) && (oldValue !== null || value !== null)) {
359
+ return diffDate(oldValue, value, key);
360
+ }
361
+
362
+ if ((oldValue === null || oldValue === undefined) && (value !== null && value !== undefined)) {
363
+ // Simplify addition
364
+ return [
365
+ AuditLogPatchItem.create({
366
+ key,
367
+ value: getDiffPutValue(value, key) || undefined,
368
+ type: AuditLogPatchItemType.Added,
369
+ }),
370
+ ];
371
+ }
372
+
373
+ if ((oldValue !== null && oldValue !== undefined) && (value === null || value === undefined)) {
374
+ return [
375
+ AuditLogPatchItem.create({
376
+ key,
377
+ oldValue: getDiffPutValue(oldValue, key) || undefined,
378
+ type: AuditLogPatchItemType.Removed,
379
+ }),
380
+ ];
381
+ }
382
+
383
+ const items = diffObject(oldValue, value, key);
384
+ let v = getDiffValue(value, key);
385
+ let ov = getDiffValue(oldValue, key);
386
+
387
+ if (oldValue !== undefined && oldValue !== null && value !== undefined && value !== null && getDiffValue(value, key) && items.length === 0 && value instanceof AutoEncoder && value.isPatch()) {
388
+ return [
389
+ AuditLogPatchItem.create({
390
+ key,
391
+ value: v || undefined,
392
+ oldValue: ov || undefined,
393
+ type: AuditLogPatchItemType.Changed,
394
+ }),
395
+ ];
396
+ }
397
+
398
+ if (v && ov) {
399
+ // Simplify change
400
+ if (v.toString() === ov.toString()) {
401
+ v = null;
402
+ ov = null;
403
+
404
+ // if (items.length === 0) {
405
+ // Probably no change
406
+ return [];
407
+ // }
408
+ }
409
+
410
+ return [
411
+ AuditLogPatchItem.create({
412
+ key,
413
+ value: v || undefined,
414
+ oldValue: ov || undefined,
415
+ type: AuditLogPatchItemType.Changed,
416
+ }),
417
+ ];
418
+ }
419
+ return items;
420
+ };
421
+
422
+ /**
423
+ * Uses the autoencoder type information to provide a better change handler
424
+ */
425
+ function diffField(field: Field<any>, oldValue: unknown, value: unknown, key?: AuditLogReplacement): AuditLogPatchItem[] {
426
+ if (field.decoder instanceof EnumDecoder) {
427
+ return diffEnum(oldValue, value, key);
428
+ }
429
+
430
+ if (field.decoder instanceof SymbolDecoder) {
431
+ if (field.decoder.decoder instanceof EnumDecoder) {
432
+ return diffEnum(oldValue, value, key); ;
433
+ }
434
+ }
435
+
436
+ if (field.decoder instanceof ArrayDecoder) {
437
+ if (field.decoder.decoder instanceof EnumDecoder) {
438
+ // Map values to keys
439
+ const items = diffArray(oldValue, value, key);
440
+
441
+ for (const item of items) {
442
+ if (item.oldValue && !item.oldValue.type) {
443
+ item.oldValue.type = AuditLogReplacementType.Key;
444
+ }
445
+ if (item.value && !item.value.type) {
446
+ item.value.type = AuditLogReplacementType.Key;
447
+ }
448
+
449
+ // If item.key is an array that ends with a 'value', also change it
450
+ if (item.key.type === AuditLogReplacementType.Array) {
451
+ const lastKeyItem = item.key.values[item.key.values.length - 1];
452
+ if (!lastKeyItem.type) {
453
+ lastKeyItem.type = AuditLogReplacementType.Key;
454
+ }
455
+ }
456
+ else {
457
+ if (!item.key.type) {
458
+ item.key.type = AuditLogReplacementType.Key;
459
+ }
460
+ }
461
+ }
462
+ return items;
463
+ }
464
+
465
+ return diffArray(oldValue, value, key);
466
+ }
467
+
468
+ return diffUnknown(oldValue, value, key);
469
+ }
470
+
471
+ export function diffObject(original: unknown | null, patch: unknown, rootKey?: AuditLogReplacement): AuditLogPatchItem[] {
472
+ if (typeof patch !== 'object' || patch === null) {
473
+ return [];
474
+ }
475
+
476
+ if (original && typeof original !== 'object') {
477
+ return [];
478
+ }
479
+
480
+ const items: AuditLogPatchItem[] = [];
481
+
482
+ for (const key in patch) {
483
+ if (key === 'id') {
484
+ continue;
485
+ }
486
+
487
+ const oldValue = original?.[key] ?? null;
488
+ const value = patch[key];
489
+
490
+ if (!(patch instanceof AutoEncoder) && !(oldValue instanceof AutoEncoder)) {
491
+ // try manual without type information
492
+ items.push(...diffUnknown(oldValue, value, getDiffKey(key).prepend(rootKey)));
493
+ continue;
494
+ }
495
+
496
+ const field = original instanceof AutoEncoder
497
+ ? original.static.latestFields.find(f => f.property === key)
498
+ : (
499
+ patch instanceof AutoEncoder
500
+ ? patch.static.latestFields.find(f => f.property === key)
501
+ : null
502
+ );
503
+
504
+ if (!field) {
505
+ // Ignore: probably an internal field
506
+ continue;
507
+ }
508
+
509
+ items.push(...diffField(field, oldValue, value, getDiffKey(key).prepend(rootKey)));
510
+ }
511
+ return items;
512
+ }
@@ -1,4 +1,4 @@
1
- import { SQLFilterDefinitions, baseSQLFilterCompilers, createSQLColumnFilterCompiler, createSQLExpressionFilterCompiler, SQL, SQLValueType } from '@stamhoofd/sql';
1
+ import { SQLFilterDefinitions, baseSQLFilterCompilers, createSQLColumnFilterCompiler, createSQLExpressionFilterCompiler, SQL, SQLValueType, createSQLRelationFilterCompiler } from '@stamhoofd/sql';
2
2
 
3
3
  export const eventFilterCompilers: SQLFilterDefinitions = {
4
4
  ...baseSQLFilterCompilers,
@@ -25,4 +25,16 @@ export const eventFilterCompilers: SQLFilterDefinitions = {
25
25
  SQL.jsonValue(SQL.column('meta'), '$.value.visible'),
26
26
  { isJSONValue: true, type: SQLValueType.JSONBoolean },
27
27
  ),
28
+ 'group': createSQLRelationFilterCompiler(
29
+ SQL.select()
30
+ .from(SQL.table('groups'))
31
+ .where(
32
+ SQL.column('id'),
33
+ SQL.column('events', 'groupId'),
34
+ ),
35
+ {
36
+ ...baseSQLFilterCompilers,
37
+ organizationId: createSQLColumnFilterCompiler('organizationId'),
38
+ },
39
+ ),
28
40
  };
@@ -1,54 +0,0 @@
1
- import { logger } from '@simonbackx/simple-logging';
2
- import { Member } from '@stamhoofd/models';
3
- import { QueueHandler } from '@stamhoofd/queues';
4
-
5
- export class MembershipHelper {
6
- static async updateAll() {
7
- console.log('Scheduling updateAllMemberships');
8
-
9
- let c = 0;
10
- let id: string = '';
11
- const tag = 'updateAllMemberships';
12
- const batch = 100;
13
-
14
- QueueHandler.cancel(tag);
15
-
16
- await QueueHandler.schedule(tag, async () => {
17
- console.log('Starting updateAllMemberships');
18
- await logger.setContext({ tags: ['silent-seed', 'seed'] }, async () => {
19
- while (true) {
20
- const rawMembers = await Member.where({
21
- id: {
22
- value: id,
23
- sign: '>',
24
- },
25
- }, { limit: batch, sort: ['id'] });
26
-
27
- if (rawMembers.length === 0) {
28
- break;
29
- }
30
-
31
- const promises: Promise<any>[] = [];
32
-
33
- for (const member of rawMembers) {
34
- promises.push((async () => {
35
- await Member.updateMembershipsForId(member.id, true);
36
- c++;
37
-
38
- if (c % 10000 === 0) {
39
- process.stdout.write(c + ' members updated\n');
40
- }
41
- })());
42
- }
43
-
44
- await Promise.all(promises);
45
- id = rawMembers[rawMembers.length - 1].id;
46
-
47
- if (rawMembers.length < batch) {
48
- break;
49
- }
50
- }
51
- });
52
- });
53
- }
54
- }