backend-manager 5.0.103 → 5.0.105

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 (60) hide show
  1. package/CHANGELOG.md +31 -0
  2. package/CLAUDE.md +113 -24
  3. package/README.md +8 -0
  4. package/TODO-PAYMENT-v2.md +5 -2
  5. package/package.json +1 -1
  6. package/src/cli/commands/deploy.js +2 -4
  7. package/src/cli/commands/emulator.js +30 -1
  8. package/src/cli/commands/test.js +33 -2
  9. package/src/manager/events/firestore/payments-webhooks/on-write.js +17 -3
  10. package/src/manager/events/firestore/payments-webhooks/transitions/index.js +6 -0
  11. package/src/manager/libraries/payment/processors/paypal.js +587 -0
  12. package/src/manager/libraries/{payment-processors → payment/processors}/stripe.js +86 -18
  13. package/src/manager/libraries/{payment-processors → payment/processors}/test.js +15 -8
  14. package/src/manager/routes/payments/cancel/processors/paypal.js +30 -0
  15. package/src/manager/routes/payments/cancel/processors/stripe.js +1 -1
  16. package/src/manager/routes/payments/cancel/processors/test.js +4 -6
  17. package/src/manager/routes/payments/intent/post.js +3 -3
  18. package/src/manager/routes/payments/intent/processors/paypal.js +150 -0
  19. package/src/manager/routes/payments/intent/processors/stripe.js +3 -5
  20. package/src/manager/routes/payments/intent/processors/test.js +7 -8
  21. package/src/manager/routes/payments/portal/processors/paypal.js +24 -0
  22. package/src/manager/routes/payments/portal/processors/stripe.js +1 -1
  23. package/src/manager/routes/payments/refund/post.js +85 -0
  24. package/src/manager/routes/payments/refund/processors/paypal.js +117 -0
  25. package/src/manager/routes/payments/refund/processors/stripe.js +103 -0
  26. package/src/manager/routes/payments/refund/processors/test.js +98 -0
  27. package/src/manager/routes/payments/webhook/processors/paypal.js +137 -0
  28. package/src/manager/schemas/payments/refund/post.js +18 -0
  29. package/src/test/test-accounts.js +46 -0
  30. package/templates/backend-manager-config.json +20 -24
  31. package/test/events/payments/journey-payments-cancel.js +3 -3
  32. package/test/events/payments/journey-payments-failure.js +1 -1
  33. package/test/events/payments/journey-payments-one-time.js +1 -1
  34. package/test/events/payments/journey-payments-plan-change.js +4 -4
  35. package/test/events/payments/journey-payments-suspend.js +3 -3
  36. package/test/events/payments/journey-payments-trial.js +2 -2
  37. package/test/fixtures/paypal/order-approved.json +62 -0
  38. package/test/fixtures/paypal/order-completed.json +110 -0
  39. package/test/fixtures/paypal/subscription-active.json +76 -0
  40. package/test/fixtures/paypal/subscription-cancelled.json +50 -0
  41. package/test/fixtures/paypal/subscription-suspended.json +65 -0
  42. package/test/helpers/payment/paypal/parse-webhook.js +539 -0
  43. package/test/helpers/payment/paypal/to-unified-one-time.js +382 -0
  44. package/test/helpers/payment/paypal/to-unified-subscription.js +820 -0
  45. package/test/helpers/{stripe-parse-webhook.js → payment/stripe/parse-webhook.js} +4 -4
  46. package/test/helpers/{stripe-to-unified-one-time.js → payment/stripe/to-unified-one-time.js} +8 -6
  47. package/test/helpers/{stripe-to-unified.js → payment/stripe/to-unified-subscription.js} +40 -33
  48. package/test/routes/payments/refund.js +174 -0
  49. package/src/manager/libraries/payment-processors/resolve-price-id.js +0 -19
  50. package/src/manager/routes/forms/delete.js +0 -37
  51. package/src/manager/routes/forms/get.js +0 -46
  52. package/src/manager/routes/forms/post.js +0 -45
  53. package/src/manager/routes/forms/public/get.js +0 -37
  54. package/src/manager/routes/forms/put.js +0 -52
  55. package/src/manager/schemas/forms/delete.js +0 -6
  56. package/src/manager/schemas/forms/get.js +0 -6
  57. package/src/manager/schemas/forms/post.js +0 -9
  58. package/src/manager/schemas/forms/public/get.js +0 -6
  59. package/src/manager/schemas/forms/put.js +0 -10
  60. /package/src/manager/libraries/{payment-processors → payment}/order-id.js +0 -0
@@ -0,0 +1,539 @@
1
+ /**
2
+ * Test: PayPal parseWebhook()
3
+ * Unit tests for the PayPal webhook processor's event categorization and routing
4
+ *
5
+ * Verifies that parseWebhook() correctly determines category, resourceType, resourceId,
6
+ * and uid for each supported event type. Mirrors stripe-parse-webhook.js for consistent coverage.
7
+ */
8
+ const paypalProcessor = require('../../../../src/manager/routes/payments/webhook/processors/paypal.js');
9
+
10
+ // Real PayPal sandbox fixtures
11
+ const FIXTURE_ORDER_APPROVED = require('../../../fixtures/paypal/order-approved.json');
12
+ const FIXTURE_SUBSCRIPTION_ACTIVE = require('../../../fixtures/paypal/subscription-active.json');
13
+
14
+ function parseWebhook(event) {
15
+ return paypalProcessor.parseWebhook({ body: event });
16
+ }
17
+
18
+ module.exports = {
19
+ description: 'PayPal parseWebhook() event categorization',
20
+ type: 'group',
21
+
22
+ tests: [
23
+ // ─── isSupported() ───
24
+
25
+ {
26
+ name: 'supports-subscription-activated',
27
+ async run({ assert }) {
28
+ assert.ok(paypalProcessor.isSupported('BILLING.SUBSCRIPTION.ACTIVATED'), 'Should support BILLING.SUBSCRIPTION.ACTIVATED');
29
+ },
30
+ },
31
+
32
+ {
33
+ name: 'supports-subscription-updated',
34
+ async run({ assert }) {
35
+ assert.ok(paypalProcessor.isSupported('BILLING.SUBSCRIPTION.UPDATED'), 'Should support BILLING.SUBSCRIPTION.UPDATED');
36
+ },
37
+ },
38
+
39
+ {
40
+ name: 'supports-subscription-cancelled',
41
+ async run({ assert }) {
42
+ assert.ok(paypalProcessor.isSupported('BILLING.SUBSCRIPTION.CANCELLED'), 'Should support BILLING.SUBSCRIPTION.CANCELLED');
43
+ },
44
+ },
45
+
46
+ {
47
+ name: 'supports-subscription-suspended',
48
+ async run({ assert }) {
49
+ assert.ok(paypalProcessor.isSupported('BILLING.SUBSCRIPTION.SUSPENDED'), 'Should support BILLING.SUBSCRIPTION.SUSPENDED');
50
+ },
51
+ },
52
+
53
+ {
54
+ name: 'supports-subscription-expired',
55
+ async run({ assert }) {
56
+ assert.ok(paypalProcessor.isSupported('BILLING.SUBSCRIPTION.EXPIRED'), 'Should support BILLING.SUBSCRIPTION.EXPIRED');
57
+ },
58
+ },
59
+
60
+ {
61
+ name: 'supports-subscription-reactivated',
62
+ async run({ assert }) {
63
+ assert.ok(paypalProcessor.isSupported('BILLING.SUBSCRIPTION.RE-ACTIVATED'), 'Should support BILLING.SUBSCRIPTION.RE-ACTIVATED');
64
+ },
65
+ },
66
+
67
+ {
68
+ name: 'supports-payment-sale-completed',
69
+ async run({ assert }) {
70
+ assert.ok(paypalProcessor.isSupported('PAYMENT.SALE.COMPLETED'), 'Should support PAYMENT.SALE.COMPLETED');
71
+ },
72
+ },
73
+
74
+ {
75
+ name: 'supports-payment-sale-denied',
76
+ async run({ assert }) {
77
+ assert.ok(paypalProcessor.isSupported('PAYMENT.SALE.DENIED'), 'Should support PAYMENT.SALE.DENIED');
78
+ },
79
+ },
80
+
81
+ {
82
+ name: 'supports-payment-sale-refunded',
83
+ async run({ assert }) {
84
+ assert.ok(paypalProcessor.isSupported('PAYMENT.SALE.REFUNDED'), 'Should support PAYMENT.SALE.REFUNDED');
85
+ },
86
+ },
87
+
88
+ {
89
+ name: 'supports-checkout-order-approved',
90
+ async run({ assert }) {
91
+ assert.ok(paypalProcessor.isSupported('CHECKOUT.ORDER.APPROVED'), 'Should support CHECKOUT.ORDER.APPROVED');
92
+ },
93
+ },
94
+
95
+ {
96
+ name: 'rejects-unsupported-event',
97
+ async run({ assert }) {
98
+ assert.equal(paypalProcessor.isSupported('PAYMENT.ORDER.CREATED'), false, 'Should not support PAYMENT.ORDER.CREATED');
99
+ assert.equal(paypalProcessor.isSupported('CUSTOMER.CREATED'), false, 'Should not support CUSTOMER.CREATED');
100
+ },
101
+ },
102
+
103
+ // ─── Validation ───
104
+
105
+ {
106
+ name: 'throws-on-empty-body',
107
+ async run({ assert }) {
108
+ try {
109
+ parseWebhook(null);
110
+ assert.fail('Should have thrown');
111
+ } catch (e) {
112
+ assert.match(e.message, /Invalid/, 'Should mention invalid payload');
113
+ }
114
+ },
115
+ },
116
+
117
+ {
118
+ name: 'throws-on-missing-id',
119
+ async run({ assert }) {
120
+ try {
121
+ parseWebhook({ event_type: 'BILLING.SUBSCRIPTION.ACTIVATED', resource: {} });
122
+ assert.fail('Should have thrown');
123
+ } catch (e) {
124
+ assert.match(e.message, /Invalid/, 'Should mention invalid payload');
125
+ }
126
+ },
127
+ },
128
+
129
+ {
130
+ name: 'throws-on-missing-event-type',
131
+ async run({ assert }) {
132
+ try {
133
+ parseWebhook({ id: 'WH-123', resource: {} });
134
+ assert.fail('Should have thrown');
135
+ } catch (e) {
136
+ assert.match(e.message, /Invalid/, 'Should mention invalid payload');
137
+ }
138
+ },
139
+ },
140
+
141
+ // ─── BILLING.SUBSCRIPTION.* events ───
142
+
143
+ {
144
+ name: 'subscription-activated-category',
145
+ async run({ assert }) {
146
+ const result = parseWebhook({
147
+ id: 'WH-activated',
148
+ event_type: 'BILLING.SUBSCRIPTION.ACTIVATED',
149
+ resource: { id: 'I-SUB123', custom_id: 'uid:user-abc,orderId:ord-1' },
150
+ });
151
+
152
+ assert.equal(result.category, 'subscription', 'Category should be subscription');
153
+ assert.equal(result.resourceType, 'subscription', 'Resource type should be subscription');
154
+ assert.equal(result.resourceId, 'I-SUB123', 'Resource ID should be subscription ID');
155
+ assert.equal(result.uid, 'user-abc', 'UID should come from custom_id');
156
+ assert.equal(result.eventId, 'WH-activated', 'Event ID should match');
157
+ assert.equal(result.eventType, 'BILLING.SUBSCRIPTION.ACTIVATED', 'Event type should match');
158
+ },
159
+ },
160
+
161
+ {
162
+ name: 'subscription-updated-category',
163
+ async run({ assert }) {
164
+ const result = parseWebhook({
165
+ id: 'WH-updated',
166
+ event_type: 'BILLING.SUBSCRIPTION.UPDATED',
167
+ resource: { id: 'I-SUB456', custom_id: 'uid:user-def,orderId:ord-2' },
168
+ });
169
+
170
+ assert.equal(result.category, 'subscription', 'Category should be subscription');
171
+ assert.equal(result.resourceType, 'subscription', 'Resource type should be subscription');
172
+ assert.equal(result.resourceId, 'I-SUB456', 'Resource ID should match');
173
+ assert.equal(result.uid, 'user-def', 'UID should match');
174
+ },
175
+ },
176
+
177
+ {
178
+ name: 'subscription-cancelled-category',
179
+ async run({ assert }) {
180
+ const result = parseWebhook({
181
+ id: 'WH-cancelled',
182
+ event_type: 'BILLING.SUBSCRIPTION.CANCELLED',
183
+ resource: { id: 'I-SUB789', custom_id: 'uid:user-ghi,orderId:ord-3' },
184
+ });
185
+
186
+ assert.equal(result.category, 'subscription', 'Category should be subscription');
187
+ assert.equal(result.resourceType, 'subscription', 'Resource type should be subscription');
188
+ assert.equal(result.resourceId, 'I-SUB789', 'Resource ID should match');
189
+ assert.equal(result.uid, 'user-ghi', 'UID should match');
190
+ },
191
+ },
192
+
193
+ {
194
+ name: 'subscription-suspended-category',
195
+ async run({ assert }) {
196
+ const result = parseWebhook({
197
+ id: 'WH-suspended',
198
+ event_type: 'BILLING.SUBSCRIPTION.SUSPENDED',
199
+ resource: { id: 'I-SUSPENDED', custom_id: 'uid:user-sus,orderId:ord-4' },
200
+ });
201
+
202
+ assert.equal(result.category, 'subscription', 'Category should be subscription');
203
+ assert.equal(result.resourceType, 'subscription', 'Resource type should be subscription');
204
+ assert.equal(result.resourceId, 'I-SUSPENDED', 'Resource ID should match');
205
+ assert.equal(result.uid, 'user-sus', 'UID should match');
206
+ },
207
+ },
208
+
209
+ {
210
+ name: 'subscription-expired-category',
211
+ async run({ assert }) {
212
+ const result = parseWebhook({
213
+ id: 'WH-expired',
214
+ event_type: 'BILLING.SUBSCRIPTION.EXPIRED',
215
+ resource: { id: 'I-EXPIRED', custom_id: 'uid:user-exp,orderId:ord-5' },
216
+ });
217
+
218
+ assert.equal(result.category, 'subscription', 'Category should be subscription');
219
+ assert.equal(result.resourceId, 'I-EXPIRED', 'Resource ID should match');
220
+ assert.equal(result.uid, 'user-exp', 'UID should match');
221
+ },
222
+ },
223
+
224
+ {
225
+ name: 'subscription-reactivated-category',
226
+ async run({ assert }) {
227
+ const result = parseWebhook({
228
+ id: 'WH-reactivated',
229
+ event_type: 'BILLING.SUBSCRIPTION.RE-ACTIVATED',
230
+ resource: { id: 'I-REACTIVATED', custom_id: 'uid:user-re,orderId:ord-6' },
231
+ });
232
+
233
+ assert.equal(result.category, 'subscription', 'Category should be subscription');
234
+ assert.equal(result.resourceId, 'I-REACTIVATED', 'Resource ID should match');
235
+ assert.equal(result.uid, 'user-re', 'UID should match');
236
+ },
237
+ },
238
+
239
+ {
240
+ name: 'subscription-event-null-uid-when-no-custom-id',
241
+ async run({ assert }) {
242
+ const result = parseWebhook({
243
+ id: 'WH-no-uid',
244
+ event_type: 'BILLING.SUBSCRIPTION.ACTIVATED',
245
+ resource: { id: 'I-NO-UID' },
246
+ });
247
+
248
+ assert.equal(result.uid, null, 'UID should be null when no custom_id');
249
+ assert.equal(result.category, 'subscription', 'Category should still be subscription');
250
+ },
251
+ },
252
+
253
+ {
254
+ name: 'subscription-event-null-uid-when-empty-custom-id',
255
+ async run({ assert }) {
256
+ const result = parseWebhook({
257
+ id: 'WH-empty-uid',
258
+ event_type: 'BILLING.SUBSCRIPTION.ACTIVATED',
259
+ resource: { id: 'I-EMPTY-UID', custom_id: '' },
260
+ });
261
+
262
+ assert.equal(result.uid, null, 'UID should be null when custom_id is empty');
263
+ },
264
+ },
265
+
266
+ // ─── PAYMENT.SALE.COMPLETED — subscription payment ───
267
+
268
+ {
269
+ name: 'sale-completed-subscription-payment',
270
+ async run({ assert }) {
271
+ const result = parseWebhook({
272
+ id: 'WH-sale-sub',
273
+ event_type: 'PAYMENT.SALE.COMPLETED',
274
+ resource: {
275
+ id: 'SALE-123',
276
+ billing_agreement_id: 'I-SUB-BILLING',
277
+ custom_id: 'uid:user-sale,orderId:ord-sale',
278
+ },
279
+ });
280
+
281
+ assert.equal(result.category, 'subscription', 'Subscription sale → subscription category');
282
+ assert.equal(result.resourceType, 'subscription', 'Should fetch subscription, not sale');
283
+ assert.equal(result.resourceId, 'I-SUB-BILLING', 'Resource ID should be the subscription ID');
284
+ assert.equal(result.uid, 'user-sale', 'UID from custom_id');
285
+ },
286
+ },
287
+
288
+ {
289
+ name: 'sale-completed-no-billing-agreement-skipped',
290
+ async run({ assert }) {
291
+ const result = parseWebhook({
292
+ id: 'WH-sale-onetime',
293
+ event_type: 'PAYMENT.SALE.COMPLETED',
294
+ resource: {
295
+ id: 'SALE-456',
296
+ // No billing_agreement_id → not a subscription payment
297
+ },
298
+ });
299
+
300
+ assert.equal(result.category, null, 'No billing agreement → null category (skipped)');
301
+ assert.equal(result.resourceType, null, 'No resource type');
302
+ assert.equal(result.resourceId, null, 'No resource ID');
303
+ },
304
+ },
305
+
306
+ // ─── PAYMENT.SALE.DENIED — subscription payment failure ───
307
+
308
+ {
309
+ name: 'sale-denied-subscription-payment',
310
+ async run({ assert }) {
311
+ const result = parseWebhook({
312
+ id: 'WH-sale-denied',
313
+ event_type: 'PAYMENT.SALE.DENIED',
314
+ resource: {
315
+ id: 'SALE-DENIED',
316
+ billing_agreement_id: 'I-SUB-DENIED',
317
+ custom_id: 'uid:user-denied,orderId:ord-denied',
318
+ },
319
+ });
320
+
321
+ assert.equal(result.category, 'subscription', 'Denied subscription sale → subscription');
322
+ assert.equal(result.resourceType, 'subscription', 'Should fetch subscription');
323
+ assert.equal(result.resourceId, 'I-SUB-DENIED', 'Resource ID should be subscription ID');
324
+ assert.equal(result.uid, 'user-denied', 'UID from custom_id');
325
+ },
326
+ },
327
+
328
+ {
329
+ name: 'sale-denied-no-billing-agreement-skipped',
330
+ async run({ assert }) {
331
+ const result = parseWebhook({
332
+ id: 'WH-sale-denied-onetime',
333
+ event_type: 'PAYMENT.SALE.DENIED',
334
+ resource: { id: 'SALE-DENIED-OT' },
335
+ });
336
+
337
+ assert.equal(result.category, null, 'No billing agreement → null (skipped)');
338
+ },
339
+ },
340
+
341
+ // ─── PAYMENT.SALE.REFUNDED ───
342
+
343
+ {
344
+ name: 'sale-refunded-subscription-linked',
345
+ async run({ assert }) {
346
+ const result = parseWebhook({
347
+ id: 'WH-refund-sub',
348
+ event_type: 'PAYMENT.SALE.REFUNDED',
349
+ resource: {
350
+ id: 'REFUND-123',
351
+ billing_agreement_id: 'I-SUB-REFUND',
352
+ custom_id: 'uid:user-refund,orderId:ord-refund',
353
+ },
354
+ });
355
+
356
+ assert.equal(result.category, 'subscription', 'Subscription refund → subscription');
357
+ assert.equal(result.resourceType, 'subscription', 'Should fetch subscription');
358
+ assert.equal(result.resourceId, 'I-SUB-REFUND', 'Resource ID should be subscription ID');
359
+ assert.equal(result.uid, 'user-refund', 'UID from custom_id');
360
+ },
361
+ },
362
+
363
+ {
364
+ name: 'sale-refunded-no-billing-agreement-skipped',
365
+ async run({ assert }) {
366
+ const result = parseWebhook({
367
+ id: 'WH-refund-onetime',
368
+ event_type: 'PAYMENT.SALE.REFUNDED',
369
+ resource: { id: 'REFUND-OT' },
370
+ });
371
+
372
+ assert.equal(result.category, null, 'No billing agreement → null (skipped)');
373
+ },
374
+ },
375
+
376
+ // ─── CHECKOUT.ORDER.APPROVED — one-time order ───
377
+
378
+ {
379
+ name: 'order-approved-one-time',
380
+ async run({ assert }) {
381
+ const result = parseWebhook({
382
+ id: 'WH-order-approved',
383
+ event_type: 'CHECKOUT.ORDER.APPROVED',
384
+ resource: {
385
+ id: 'ORDER-123',
386
+ purchase_units: [{
387
+ custom_id: 'uid:user-order,orderId:ord-order,productId:credits-100',
388
+ }],
389
+ },
390
+ });
391
+
392
+ assert.equal(result.category, 'one-time', 'Category should be one-time');
393
+ assert.equal(result.resourceType, 'order', 'Resource type should be order');
394
+ assert.equal(result.resourceId, 'ORDER-123', 'Resource ID should be the order ID');
395
+ assert.equal(result.uid, 'user-order', 'UID should come from purchase_units custom_id');
396
+ assert.equal(result.eventType, 'CHECKOUT.ORDER.APPROVED', 'Event type should match');
397
+ },
398
+ },
399
+
400
+ {
401
+ name: 'order-approved-null-uid-when-no-purchase-units',
402
+ async run({ assert }) {
403
+ const result = parseWebhook({
404
+ id: 'WH-order-no-units',
405
+ event_type: 'CHECKOUT.ORDER.APPROVED',
406
+ resource: { id: 'ORDER-NO-UNITS' },
407
+ });
408
+
409
+ assert.equal(result.category, 'one-time', 'Category should still be one-time');
410
+ assert.equal(result.resourceType, 'order', 'Resource type should be order');
411
+ assert.equal(result.uid, null, 'UID should be null when no purchase_units');
412
+ },
413
+ },
414
+
415
+ {
416
+ name: 'order-approved-null-uid-when-empty-custom-id',
417
+ async run({ assert }) {
418
+ const result = parseWebhook({
419
+ id: 'WH-order-empty',
420
+ event_type: 'CHECKOUT.ORDER.APPROVED',
421
+ resource: {
422
+ id: 'ORDER-EMPTY',
423
+ purchase_units: [{ custom_id: '' }],
424
+ },
425
+ });
426
+
427
+ assert.equal(result.uid, null, 'UID should be null when custom_id is empty');
428
+ assert.equal(result.category, 'one-time', 'Category should still be one-time');
429
+ },
430
+ },
431
+
432
+ // ─── Custom ID parsing edge cases ───
433
+
434
+ {
435
+ name: 'custom-id-with-colon-in-uid',
436
+ async run({ assert }) {
437
+ const result = parseWebhook({
438
+ id: 'WH-colon',
439
+ event_type: 'BILLING.SUBSCRIPTION.ACTIVATED',
440
+ resource: {
441
+ id: 'I-COLON',
442
+ custom_id: 'uid:firebase:auth:user123,orderId:ord-c',
443
+ },
444
+ });
445
+
446
+ assert.equal(result.uid, 'firebase:auth:user123', 'Should handle colons in uid value');
447
+ },
448
+ },
449
+
450
+ {
451
+ name: 'custom-id-uid-only',
452
+ async run({ assert }) {
453
+ const result = parseWebhook({
454
+ id: 'WH-uid-only',
455
+ event_type: 'BILLING.SUBSCRIPTION.ACTIVATED',
456
+ resource: {
457
+ id: 'I-UID-ONLY',
458
+ custom_id: 'uid:user-only',
459
+ },
460
+ });
461
+
462
+ assert.equal(result.uid, 'user-only', 'Should parse uid without orderId');
463
+ },
464
+ },
465
+
466
+ // ─── Raw passthrough ───
467
+
468
+ {
469
+ name: 'raw-contains-full-event',
470
+ async run({ assert }) {
471
+ const event = {
472
+ id: 'WH-raw',
473
+ event_type: 'BILLING.SUBSCRIPTION.UPDATED',
474
+ resource: { id: 'I-RAW', custom_id: 'uid:user-raw,orderId:ord-raw' },
475
+ extra_field: 'preserved',
476
+ };
477
+
478
+ const result = parseWebhook(event);
479
+ assert.equal(result.raw, event, 'Raw should be the full event object');
480
+ assert.equal(result.raw.extra_field, 'preserved', 'Extra fields preserved in raw');
481
+ },
482
+ },
483
+
484
+ // ─── Unsupported event type passthrough ───
485
+
486
+ {
487
+ name: 'unsupported-event-returns-null-category',
488
+ async run({ assert }) {
489
+ const result = parseWebhook({
490
+ id: 'WH-unsupported',
491
+ event_type: 'PAYMENT.ORDER.CREATED',
492
+ resource: { id: 'ORDER-123' },
493
+ });
494
+
495
+ assert.equal(result.category, null, 'Unsupported event → null category');
496
+ assert.equal(result.resourceType, null, 'No resource type');
497
+ assert.equal(result.resourceId, null, 'No resource ID');
498
+ assert.equal(result.eventId, 'WH-unsupported', 'Should still return event ID');
499
+ assert.equal(result.eventType, 'PAYMENT.ORDER.CREATED', 'Should still return event type');
500
+ },
501
+ },
502
+
503
+ // ─── Real PayPal sandbox fixtures ───
504
+
505
+ {
506
+ name: 'fixture-subscription-activated-parses-correctly',
507
+ async run({ assert }) {
508
+ const result = parseWebhook({
509
+ id: 'WH-fixture-sub',
510
+ event_type: 'BILLING.SUBSCRIPTION.ACTIVATED',
511
+ resource: FIXTURE_SUBSCRIPTION_ACTIVE,
512
+ });
513
+
514
+ assert.equal(result.category, 'subscription', 'Category should be subscription');
515
+ assert.equal(result.resourceType, 'subscription', 'Resource type should be subscription');
516
+ assert.equal(result.resourceId, 'I-MTPRX0B9LV4R', 'Resource ID from fixture');
517
+ assert.equal(result.uid, 'test-user-789', 'UID from fixture custom_id');
518
+ assert.equal(result.eventType, 'BILLING.SUBSCRIPTION.ACTIVATED', 'Event type should match');
519
+ },
520
+ },
521
+
522
+ {
523
+ name: 'fixture-order-approved-parses-correctly',
524
+ async run({ assert }) {
525
+ const result = parseWebhook({
526
+ id: 'WH-fixture-order',
527
+ event_type: 'CHECKOUT.ORDER.APPROVED',
528
+ resource: FIXTURE_ORDER_APPROVED,
529
+ });
530
+
531
+ assert.equal(result.category, 'one-time', 'Category should be one-time');
532
+ assert.equal(result.resourceType, 'order', 'Resource type should be order');
533
+ assert.equal(result.resourceId, '5UX02069M9686893E', 'Resource ID from fixture');
534
+ assert.equal(result.uid, 'test-user-123', 'UID from fixture purchase_units custom_id');
535
+ assert.equal(result.eventType, 'CHECKOUT.ORDER.APPROVED', 'Event type should match');
536
+ },
537
+ },
538
+ ],
539
+ };