@skillsmith/core 2.1.0 → 2.1.2

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 (204) hide show
  1. package/dist/.tsbuildinfo +1 -1
  2. package/dist/src/analysis/types.d.ts +2 -0
  3. package/dist/src/analysis/types.d.ts.map +1 -1
  4. package/dist/src/analysis/types.js +13 -1
  5. package/dist/src/analysis/types.js.map +1 -1
  6. package/dist/src/analytics/AnalyticsRepository.d.ts +4 -0
  7. package/dist/src/analytics/AnalyticsRepository.d.ts.map +1 -1
  8. package/dist/src/analytics/AnalyticsRepository.js +26 -44
  9. package/dist/src/analytics/AnalyticsRepository.js.map +1 -1
  10. package/dist/src/analytics/schema.d.ts +1 -1
  11. package/dist/src/analytics/schema.d.ts.map +1 -1
  12. package/dist/src/analytics/schema.js +68 -0
  13. package/dist/src/analytics/schema.js.map +1 -1
  14. package/dist/src/api/client.d.ts +33 -29
  15. package/dist/src/api/client.d.ts.map +1 -1
  16. package/dist/src/api/client.js +15 -10
  17. package/dist/src/api/client.js.map +1 -1
  18. package/dist/src/billing/BillingService.d.ts +139 -0
  19. package/dist/src/billing/BillingService.d.ts.map +1 -0
  20. package/dist/src/billing/BillingService.js +393 -0
  21. package/dist/src/billing/BillingService.js.map +1 -0
  22. package/dist/src/billing/GDPRComplianceService.d.ts +176 -0
  23. package/dist/src/billing/GDPRComplianceService.d.ts.map +1 -0
  24. package/dist/src/billing/GDPRComplianceService.js +361 -0
  25. package/dist/src/billing/GDPRComplianceService.js.map +1 -0
  26. package/dist/src/billing/StripeClient.d.ts +177 -0
  27. package/dist/src/billing/StripeClient.d.ts.map +1 -0
  28. package/dist/src/billing/StripeClient.js +462 -0
  29. package/dist/src/billing/StripeClient.js.map +1 -0
  30. package/dist/src/billing/StripeReconciliationJob.d.ts +95 -0
  31. package/dist/src/billing/StripeReconciliationJob.d.ts.map +1 -0
  32. package/dist/src/billing/StripeReconciliationJob.js +405 -0
  33. package/dist/src/billing/StripeReconciliationJob.js.map +1 -0
  34. package/dist/src/billing/StripeWebhookHandler.d.ts +92 -0
  35. package/dist/src/billing/StripeWebhookHandler.d.ts.map +1 -0
  36. package/dist/src/billing/StripeWebhookHandler.js +409 -0
  37. package/dist/src/billing/StripeWebhookHandler.js.map +1 -0
  38. package/dist/src/billing/index.d.ts +18 -0
  39. package/dist/src/billing/index.d.ts.map +1 -0
  40. package/dist/src/billing/index.js +19 -0
  41. package/dist/src/billing/index.js.map +1 -0
  42. package/dist/src/billing/types.d.ts +266 -0
  43. package/dist/src/billing/types.d.ts.map +1 -0
  44. package/dist/src/billing/types.js +23 -0
  45. package/dist/src/billing/types.js.map +1 -0
  46. package/dist/src/embeddings/hnsw-store.d.ts +568 -0
  47. package/dist/src/embeddings/hnsw-store.d.ts.map +1 -0
  48. package/dist/src/embeddings/hnsw-store.js +805 -0
  49. package/dist/src/embeddings/hnsw-store.js.map +1 -0
  50. package/dist/src/embeddings/index.d.ts +2 -0
  51. package/dist/src/embeddings/index.d.ts.map +1 -1
  52. package/dist/src/embeddings/index.js +2 -0
  53. package/dist/src/embeddings/index.js.map +1 -1
  54. package/dist/src/index.d.ts +1 -0
  55. package/dist/src/index.d.ts.map +1 -1
  56. package/dist/src/index.js +2 -0
  57. package/dist/src/index.js.map +1 -1
  58. package/dist/src/learning/PatternStore.d.ts +457 -0
  59. package/dist/src/learning/PatternStore.d.ts.map +1 -0
  60. package/dist/src/learning/PatternStore.js +893 -0
  61. package/dist/src/learning/PatternStore.js.map +1 -0
  62. package/dist/src/learning/ReasoningBankIntegration.d.ts +403 -0
  63. package/dist/src/learning/ReasoningBankIntegration.d.ts.map +1 -0
  64. package/dist/src/learning/ReasoningBankIntegration.js +627 -0
  65. package/dist/src/learning/ReasoningBankIntegration.js.map +1 -0
  66. package/dist/src/learning/index.d.ts +15 -0
  67. package/dist/src/learning/index.d.ts.map +1 -0
  68. package/dist/src/learning/index.js +15 -0
  69. package/dist/src/learning/index.js.map +1 -0
  70. package/dist/src/routing/SONARouter.d.ts +154 -0
  71. package/dist/src/routing/SONARouter.d.ts.map +1 -0
  72. package/dist/src/routing/SONARouter.js +679 -0
  73. package/dist/src/routing/SONARouter.js.map +1 -0
  74. package/dist/src/routing/index.d.ts +9 -0
  75. package/dist/src/routing/index.d.ts.map +1 -0
  76. package/dist/src/routing/index.js +10 -0
  77. package/dist/src/routing/index.js.map +1 -0
  78. package/dist/src/routing/types.d.ts +331 -0
  79. package/dist/src/routing/types.d.ts.map +1 -0
  80. package/dist/src/routing/types.js +203 -0
  81. package/dist/src/routing/types.js.map +1 -0
  82. package/dist/src/scripts/__tests__/scan-imported-skills.test.js +5 -0
  83. package/dist/src/scripts/__tests__/scan-imported-skills.test.js.map +1 -1
  84. package/dist/src/security/SkillSandbox.d.ts +156 -0
  85. package/dist/src/security/SkillSandbox.d.ts.map +1 -0
  86. package/dist/src/security/SkillSandbox.js +303 -0
  87. package/dist/src/security/SkillSandbox.js.map +1 -0
  88. package/dist/src/security/index.d.ts +3 -1
  89. package/dist/src/security/index.d.ts.map +1 -1
  90. package/dist/src/security/index.js +5 -1
  91. package/dist/src/security/index.js.map +1 -1
  92. package/dist/src/security/rate-limiter/presets.d.ts +12 -0
  93. package/dist/src/security/rate-limiter/presets.d.ts.map +1 -1
  94. package/dist/src/security/rate-limiter/presets.js +12 -0
  95. package/dist/src/security/rate-limiter/presets.js.map +1 -1
  96. package/dist/src/security/sanitization.d.ts +85 -0
  97. package/dist/src/security/sanitization.d.ts.map +1 -1
  98. package/dist/src/security/sanitization.js +133 -0
  99. package/dist/src/security/sanitization.js.map +1 -1
  100. package/dist/src/security/scanner/SecurityScanner.d.ts +23 -0
  101. package/dist/src/security/scanner/SecurityScanner.d.ts.map +1 -1
  102. package/dist/src/security/scanner/SecurityScanner.js +232 -28
  103. package/dist/src/security/scanner/SecurityScanner.js.map +1 -1
  104. package/dist/src/security/scanner/patterns.d.ts +13 -0
  105. package/dist/src/security/scanner/patterns.d.ts.map +1 -1
  106. package/dist/src/security/scanner/patterns.js +51 -0
  107. package/dist/src/security/scanner/patterns.js.map +1 -1
  108. package/dist/src/security/scanner/types.d.ts +13 -1
  109. package/dist/src/security/scanner/types.d.ts.map +1 -1
  110. package/dist/src/security/scanner/weights.d.ts.map +1 -1
  111. package/dist/src/security/scanner/weights.js +1 -0
  112. package/dist/src/security/scanner/weights.js.map +1 -1
  113. package/dist/src/session/SessionManager.d.ts +7 -0
  114. package/dist/src/session/SessionManager.d.ts.map +1 -1
  115. package/dist/src/session/SessionManager.js +117 -10
  116. package/dist/src/session/SessionManager.js.map +1 -1
  117. package/dist/src/sync/SyncEngine.d.ts.map +1 -1
  118. package/dist/src/sync/SyncEngine.js +52 -32
  119. package/dist/src/sync/SyncEngine.js.map +1 -1
  120. package/dist/src/testing/MultiLLMProvider.d.ts +374 -0
  121. package/dist/src/testing/MultiLLMProvider.d.ts.map +1 -0
  122. package/dist/src/testing/MultiLLMProvider.js +720 -0
  123. package/dist/src/testing/MultiLLMProvider.js.map +1 -0
  124. package/dist/src/testing/index.d.ts +8 -0
  125. package/dist/src/testing/index.d.ts.map +1 -0
  126. package/dist/src/testing/index.js +9 -0
  127. package/dist/src/testing/index.js.map +1 -0
  128. package/dist/src/types.d.ts +3 -0
  129. package/dist/src/types.d.ts.map +1 -1
  130. package/dist/tests/SecurityScanner.test.js +337 -1
  131. package/dist/tests/SecurityScanner.test.js.map +1 -1
  132. package/dist/tests/billing/BillingService.test.d.ts +7 -0
  133. package/dist/tests/billing/BillingService.test.d.ts.map +1 -0
  134. package/dist/tests/billing/BillingService.test.js +168 -0
  135. package/dist/tests/billing/BillingService.test.js.map +1 -0
  136. package/dist/tests/billing/GDPRCompliance.test.d.ts +7 -0
  137. package/dist/tests/billing/GDPRCompliance.test.d.ts.map +1 -0
  138. package/dist/tests/billing/GDPRCompliance.test.js +195 -0
  139. package/dist/tests/billing/GDPRCompliance.test.js.map +1 -0
  140. package/dist/tests/billing/StripeReconciliation.test.d.ts +7 -0
  141. package/dist/tests/billing/StripeReconciliation.test.d.ts.map +1 -0
  142. package/dist/tests/billing/StripeReconciliation.test.js +266 -0
  143. package/dist/tests/billing/StripeReconciliation.test.js.map +1 -0
  144. package/dist/tests/billing/stripe-validators.test.d.ts +7 -0
  145. package/dist/tests/billing/stripe-validators.test.d.ts.map +1 -0
  146. package/dist/tests/billing/stripe-validators.test.js +107 -0
  147. package/dist/tests/billing/stripe-validators.test.js.map +1 -0
  148. package/dist/tests/embeddings/hnsw-store.test.d.ts +7 -0
  149. package/dist/tests/embeddings/hnsw-store.test.d.ts.map +1 -0
  150. package/dist/tests/embeddings/hnsw-store.test.js +295 -0
  151. package/dist/tests/embeddings/hnsw-store.test.js.map +1 -0
  152. package/dist/tests/integration/neural/e2e-learning.test.d.ts +17 -0
  153. package/dist/tests/integration/neural/e2e-learning.test.d.ts.map +1 -0
  154. package/dist/tests/integration/neural/e2e-learning.test.js +238 -0
  155. package/dist/tests/integration/neural/e2e-learning.test.js.map +1 -0
  156. package/dist/tests/integration/neural/helpers.d.ts +132 -0
  157. package/dist/tests/integration/neural/helpers.d.ts.map +1 -0
  158. package/dist/tests/integration/neural/helpers.js +287 -0
  159. package/dist/tests/integration/neural/helpers.js.map +1 -0
  160. package/dist/tests/integration/neural/personalization.test.d.ts +21 -0
  161. package/dist/tests/integration/neural/personalization.test.d.ts.map +1 -0
  162. package/dist/tests/integration/neural/personalization.test.js +304 -0
  163. package/dist/tests/integration/neural/personalization.test.js.map +1 -0
  164. package/dist/tests/integration/neural/preference-learner.test.d.ts +23 -0
  165. package/dist/tests/integration/neural/preference-learner.test.d.ts.map +1 -0
  166. package/dist/tests/integration/neural/preference-learner.test.js +289 -0
  167. package/dist/tests/integration/neural/preference-learner.test.js.map +1 -0
  168. package/dist/tests/integration/neural/privacy.test.d.ts +19 -0
  169. package/dist/tests/integration/neural/privacy.test.d.ts.map +1 -0
  170. package/dist/tests/integration/neural/privacy.test.js +249 -0
  171. package/dist/tests/integration/neural/privacy.test.js.map +1 -0
  172. package/dist/tests/integration/neural/setup.d.ts +175 -0
  173. package/dist/tests/integration/neural/setup.d.ts.map +1 -0
  174. package/dist/tests/integration/neural/setup.js +487 -0
  175. package/dist/tests/integration/neural/setup.js.map +1 -0
  176. package/dist/tests/integration/neural/signal-collection.test.d.ts +21 -0
  177. package/dist/tests/integration/neural/signal-collection.test.d.ts.map +1 -0
  178. package/dist/tests/integration/neural/signal-collection.test.js +232 -0
  179. package/dist/tests/integration/neural/signal-collection.test.js.map +1 -0
  180. package/dist/tests/learning/PatternStore.test.d.ts +8 -0
  181. package/dist/tests/learning/PatternStore.test.d.ts.map +1 -0
  182. package/dist/tests/learning/PatternStore.test.js +589 -0
  183. package/dist/tests/learning/PatternStore.test.js.map +1 -0
  184. package/dist/tests/learning/ReasoningBankIntegration.test.d.ts +8 -0
  185. package/dist/tests/learning/ReasoningBankIntegration.test.d.ts.map +1 -0
  186. package/dist/tests/learning/ReasoningBankIntegration.test.js +269 -0
  187. package/dist/tests/learning/ReasoningBankIntegration.test.js.map +1 -0
  188. package/dist/tests/routing/SONARouter.test.d.ts +8 -0
  189. package/dist/tests/routing/SONARouter.test.d.ts.map +1 -0
  190. package/dist/tests/routing/SONARouter.test.js +400 -0
  191. package/dist/tests/routing/SONARouter.test.js.map +1 -0
  192. package/dist/tests/security/ContinuousSecurity.test.js +10 -12
  193. package/dist/tests/security/ContinuousSecurity.test.js.map +1 -1
  194. package/dist/tests/security/SkillSandbox.test.d.ts +8 -0
  195. package/dist/tests/security/SkillSandbox.test.d.ts.map +1 -0
  196. package/dist/tests/security/SkillSandbox.test.js +321 -0
  197. package/dist/tests/security/SkillSandbox.test.js.map +1 -0
  198. package/dist/tests/sync/SyncEngine.test.js +4 -2
  199. package/dist/tests/sync/SyncEngine.test.js.map +1 -1
  200. package/dist/tests/testing/MultiLLMProvider.test.d.ts +14 -0
  201. package/dist/tests/testing/MultiLLMProvider.test.d.ts.map +1 -0
  202. package/dist/tests/testing/MultiLLMProvider.test.js +438 -0
  203. package/dist/tests/testing/MultiLLMProvider.test.js.map +1 -0
  204. package/package.json +16 -3
@@ -0,0 +1,266 @@
1
+ /**
2
+ * SMI-1069: Stripe Reconciliation Job Tests
3
+ *
4
+ * Tests for data reconciliation between local database and Stripe.
5
+ */
6
+ import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
7
+ import Database from 'better-sqlite3';
8
+ import { StripeReconciliationJob } from '../../src/billing/StripeReconciliationJob.js';
9
+ import { initializeAnalyticsSchema } from '../../src/analytics/schema.js';
10
+ import { randomUUID } from 'crypto';
11
+ // Mock StripeClient
12
+ function createMockStripeClient() {
13
+ return {
14
+ getSubscription: vi.fn().mockResolvedValue({
15
+ id: 'sub_test123',
16
+ status: 'active',
17
+ metadata: { tier: 'team' },
18
+ items: {
19
+ data: [{ quantity: 5 }],
20
+ },
21
+ current_period_start: 1704067200,
22
+ current_period_end: 1706745600,
23
+ }),
24
+ getInvoice: vi.fn().mockResolvedValue({
25
+ id: 'in_test123',
26
+ status: 'paid',
27
+ amount_paid: 2500,
28
+ amount_due: 2500,
29
+ }),
30
+ getStripeInstance: vi.fn(),
31
+ };
32
+ }
33
+ describe('StripeReconciliationJob', () => {
34
+ let db;
35
+ let mockStripeClient;
36
+ beforeEach(() => {
37
+ // Create in-memory database
38
+ db = new Database(':memory:');
39
+ initializeAnalyticsSchema(db);
40
+ mockStripeClient = createMockStripeClient();
41
+ });
42
+ afterEach(() => {
43
+ db.close();
44
+ vi.restoreAllMocks();
45
+ });
46
+ // Helper to create test subscription
47
+ function createTestSubscription(customerId, options = {}) {
48
+ const id = randomUUID();
49
+ const now = new Date().toISOString();
50
+ db.prepare(`INSERT INTO user_subscriptions (
51
+ id, customer_id, email, stripe_customer_id, stripe_subscription_id,
52
+ stripe_price_id, tier, status, seat_count,
53
+ current_period_start, current_period_end, created_at, updated_at
54
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`).run(id, customerId, 'test@example.com', `cus_${customerId}`, options.stripeSubscriptionId ?? `sub_${customerId}`, 'price_test', options.tier ?? 'team', options.status ?? 'active', options.seatCount ?? 5, now, now, now, now);
55
+ return id;
56
+ }
57
+ // Helper to create test invoice
58
+ function createTestInvoice(customerId, options = {}) {
59
+ const id = randomUUID();
60
+ const now = new Date().toISOString();
61
+ db.prepare(`INSERT INTO invoices (
62
+ id, customer_id, stripe_invoice_id, amount_cents, currency, status, created_at
63
+ ) VALUES (?, ?, ?, ?, ?, ?, ?)`).run(id, customerId, options.stripeInvoiceId ?? `in_${customerId}`, options.amountCents ?? 2500, 'usd', options.status ?? 'paid', now);
64
+ return id;
65
+ }
66
+ describe('Basic Operation', () => {
67
+ it('should run reconciliation job successfully', async () => {
68
+ const job = new StripeReconciliationJob({
69
+ stripeClient: mockStripeClient,
70
+ db,
71
+ });
72
+ const result = await job.run();
73
+ expect(result.success).toBe(true);
74
+ expect(result.startedAt).toBeDefined();
75
+ expect(result.completedAt).toBeDefined();
76
+ expect(result.durationMs).toBeGreaterThanOrEqual(0);
77
+ });
78
+ it('should report subscription counts', async () => {
79
+ createTestSubscription('customer1');
80
+ createTestSubscription('customer2');
81
+ const job = new StripeReconciliationJob({
82
+ stripeClient: mockStripeClient,
83
+ db,
84
+ });
85
+ const result = await job.run();
86
+ expect(result.stats.subscriptionsChecked).toBe(2);
87
+ });
88
+ it('should report invoice counts', async () => {
89
+ createTestInvoice('customer1');
90
+ createTestInvoice('customer2');
91
+ createTestInvoice('customer3');
92
+ const job = new StripeReconciliationJob({
93
+ stripeClient: mockStripeClient,
94
+ db,
95
+ });
96
+ const result = await job.run();
97
+ expect(result.stats.invoicesChecked).toBe(3);
98
+ });
99
+ });
100
+ describe('Subscription Discrepancy Detection', () => {
101
+ it('should detect status mismatch', async () => {
102
+ createTestSubscription('customer1', { status: 'past_due' });
103
+ const job = new StripeReconciliationJob({
104
+ stripeClient: mockStripeClient,
105
+ db,
106
+ });
107
+ const result = await job.run();
108
+ expect(result.discrepancies).toHaveLength(1);
109
+ expect(result.discrepancies[0].type).toBe('status_mismatch');
110
+ expect(result.discrepancies[0].localValue).toBe('past_due');
111
+ expect(result.discrepancies[0].stripeValue).toBe('active');
112
+ });
113
+ it('should detect tier mismatch', async () => {
114
+ createTestSubscription('customer1', { tier: 'individual' });
115
+ const job = new StripeReconciliationJob({
116
+ stripeClient: mockStripeClient,
117
+ db,
118
+ });
119
+ const result = await job.run();
120
+ const tierMismatch = result.discrepancies.find((d) => d.type === 'tier_mismatch');
121
+ expect(tierMismatch).toBeDefined();
122
+ expect(tierMismatch?.localValue).toBe('individual');
123
+ expect(tierMismatch?.stripeValue).toBe('team');
124
+ });
125
+ it('should detect seat count mismatch', async () => {
126
+ createTestSubscription('customer1', { seatCount: 10 });
127
+ const job = new StripeReconciliationJob({
128
+ stripeClient: mockStripeClient,
129
+ db,
130
+ });
131
+ const result = await job.run();
132
+ const seatMismatch = result.discrepancies.find((d) => d.type === 'seat_count_mismatch');
133
+ expect(seatMismatch).toBeDefined();
134
+ expect(seatMismatch?.localValue).toBe(10);
135
+ expect(seatMismatch?.stripeValue).toBe(5);
136
+ });
137
+ it('should detect missing Stripe subscription', async () => {
138
+ createTestSubscription('customer1');
139
+ // Mock Stripe returning null (subscription not found)
140
+ vi.mocked(mockStripeClient.getSubscription).mockResolvedValue(null);
141
+ const job = new StripeReconciliationJob({
142
+ stripeClient: mockStripeClient,
143
+ db,
144
+ });
145
+ const result = await job.run();
146
+ expect(result.discrepancies).toHaveLength(1);
147
+ expect(result.discrepancies[0].type).toBe('missing_stripe');
148
+ });
149
+ });
150
+ describe('Invoice Discrepancy Detection', () => {
151
+ it('should detect invoice status mismatch', async () => {
152
+ createTestInvoice('customer1', { status: 'open' });
153
+ const job = new StripeReconciliationJob({
154
+ stripeClient: mockStripeClient,
155
+ db,
156
+ });
157
+ const result = await job.run();
158
+ const statusMismatch = result.discrepancies.find((d) => d.type === 'invoice_status_mismatch');
159
+ expect(statusMismatch).toBeDefined();
160
+ expect(statusMismatch?.localValue).toBe('open');
161
+ expect(statusMismatch?.stripeValue).toBe('paid');
162
+ });
163
+ it('should detect invoice amount mismatch', async () => {
164
+ createTestInvoice('customer1', { amountCents: 5000 });
165
+ const job = new StripeReconciliationJob({
166
+ stripeClient: mockStripeClient,
167
+ db,
168
+ });
169
+ const result = await job.run();
170
+ const amountMismatch = result.discrepancies.find((d) => d.type === 'invoice_amount_mismatch');
171
+ expect(amountMismatch).toBeDefined();
172
+ expect(amountMismatch?.localValue).toBe(5000);
173
+ expect(amountMismatch?.stripeValue).toBe(2500);
174
+ });
175
+ });
176
+ describe('Auto-Fix', () => {
177
+ it('should fix status mismatch when autoFix is enabled', async () => {
178
+ const subscriptionId = createTestSubscription('customer1', { status: 'past_due' });
179
+ const job = new StripeReconciliationJob({
180
+ stripeClient: mockStripeClient,
181
+ db,
182
+ autoFix: true,
183
+ });
184
+ const result = await job.run();
185
+ expect(result.stats.discrepanciesFixed).toBe(1);
186
+ expect(result.discrepancies[0].fixed).toBe(true);
187
+ // Verify the fix was applied
188
+ const updated = db
189
+ .prepare(`SELECT status FROM user_subscriptions WHERE id = ?`)
190
+ .get(subscriptionId);
191
+ expect(updated.status).toBe('active');
192
+ });
193
+ it('should not fix discrepancies when autoFix is disabled', async () => {
194
+ createTestSubscription('customer1', { status: 'past_due' });
195
+ const job = new StripeReconciliationJob({
196
+ stripeClient: mockStripeClient,
197
+ db,
198
+ autoFix: false,
199
+ });
200
+ const result = await job.run();
201
+ expect(result.stats.discrepanciesFixed).toBe(0);
202
+ expect(result.discrepancies[0].fixed).toBe(false);
203
+ });
204
+ it('should fix invoice status mismatch', async () => {
205
+ const invoiceId = createTestInvoice('customer1', { status: 'open' });
206
+ const job = new StripeReconciliationJob({
207
+ stripeClient: mockStripeClient,
208
+ db,
209
+ autoFix: true,
210
+ });
211
+ await job.run();
212
+ // Verify the fix was applied
213
+ const updated = db.prepare(`SELECT status FROM invoices WHERE id = ?`).get(invoiceId);
214
+ expect(updated.status).toBe('paid');
215
+ });
216
+ });
217
+ describe('Error Handling', () => {
218
+ it('should handle Stripe API errors gracefully', async () => {
219
+ createTestSubscription('customer1');
220
+ vi.mocked(mockStripeClient.getSubscription).mockRejectedValue(new Error('Stripe API error'));
221
+ const job = new StripeReconciliationJob({
222
+ stripeClient: mockStripeClient,
223
+ db,
224
+ });
225
+ const result = await job.run();
226
+ expect(result.errors).toHaveLength(1);
227
+ expect(result.errors[0]).toContain('Stripe API error');
228
+ });
229
+ it('should continue processing after individual errors', async () => {
230
+ createTestSubscription('customer1');
231
+ createTestSubscription('customer2');
232
+ // First call fails, second succeeds
233
+ vi.mocked(mockStripeClient.getSubscription)
234
+ .mockRejectedValueOnce(new Error('API error'))
235
+ .mockResolvedValueOnce({
236
+ id: 'sub_test123',
237
+ status: 'active',
238
+ metadata: { tier: 'team' },
239
+ items: { data: [{ quantity: 5 }] },
240
+ });
241
+ const job = new StripeReconciliationJob({
242
+ stripeClient: mockStripeClient,
243
+ db,
244
+ });
245
+ const result = await job.run();
246
+ expect(result.stats.subscriptionsChecked).toBe(2);
247
+ expect(result.errors).toHaveLength(1);
248
+ });
249
+ });
250
+ describe('Batch Size', () => {
251
+ it('should respect batch size limit', async () => {
252
+ // Create more subscriptions than batch size
253
+ for (let i = 0; i < 10; i++) {
254
+ createTestSubscription(`customer${i}`);
255
+ }
256
+ const job = new StripeReconciliationJob({
257
+ stripeClient: mockStripeClient,
258
+ db,
259
+ batchSize: 3,
260
+ });
261
+ const result = await job.run();
262
+ expect(result.stats.subscriptionsChecked).toBe(3);
263
+ });
264
+ });
265
+ });
266
+ //# sourceMappingURL=StripeReconciliation.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"StripeReconciliation.test.js","sourceRoot":"","sources":["../../../tests/billing/StripeReconciliation.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * SMI-1062: Stripe ID Validator Tests
3
+ *
4
+ * Tests for Stripe ID validation and sanitization functions.
5
+ */
6
+ export {};
7
+ //# sourceMappingURL=stripe-validators.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stripe-validators.test.d.ts","sourceRoot":"","sources":["../../../tests/billing/stripe-validators.test.ts"],"names":[],"mappings":"AAAA;;;;GAIG"}
@@ -0,0 +1,107 @@
1
+ /**
2
+ * SMI-1062: Stripe ID Validator Tests
3
+ *
4
+ * Tests for Stripe ID validation and sanitization functions.
5
+ */
6
+ import { describe, it, expect } from 'vitest';
7
+ import { isValidStripeId, sanitizeStripeCustomerId, sanitizeStripeSubscriptionId, sanitizeStripePriceId, sanitizeStripeInvoiceId, sanitizeStripeEventId, } from '../../src/security/sanitization.js';
8
+ describe('Stripe ID Validators', () => {
9
+ describe('isValidStripeId', () => {
10
+ it('should validate customer IDs', () => {
11
+ expect(isValidStripeId('cus_OtLqEJvHu1Mvxm', 'customer')).toBe(true);
12
+ expect(isValidStripeId('cus_123456789abcdef', 'customer')).toBe(true);
13
+ expect(isValidStripeId('cus_a', 'customer')).toBe(true);
14
+ });
15
+ it('should validate subscription IDs', () => {
16
+ expect(isValidStripeId('sub_1234567890abcdef', 'subscription')).toBe(true);
17
+ expect(isValidStripeId('sub_ABC123', 'subscription')).toBe(true);
18
+ });
19
+ it('should validate price IDs', () => {
20
+ expect(isValidStripeId('price_1234567890abcdef', 'price')).toBe(true);
21
+ expect(isValidStripeId('price_team_monthly', 'price')).toBe(false); // underscores not allowed
22
+ });
23
+ it('should validate invoice IDs', () => {
24
+ expect(isValidStripeId('in_1234567890abcdef', 'invoice')).toBe(true);
25
+ });
26
+ it('should validate event IDs', () => {
27
+ expect(isValidStripeId('evt_1234567890abcdef', 'event')).toBe(true);
28
+ });
29
+ it('should reject invalid IDs', () => {
30
+ // Wrong prefix
31
+ expect(isValidStripeId('sub_123', 'customer')).toBe(false);
32
+ expect(isValidStripeId('cus_123', 'subscription')).toBe(false);
33
+ // No prefix
34
+ expect(isValidStripeId('123456789', 'customer')).toBe(false);
35
+ // Special characters in suffix
36
+ expect(isValidStripeId('cus_123-456', 'customer')).toBe(false);
37
+ expect(isValidStripeId('cus_123_456', 'customer')).toBe(false);
38
+ expect(isValidStripeId('cus_<script>', 'customer')).toBe(false);
39
+ // Too short
40
+ expect(isValidStripeId('cus_', 'customer')).toBe(false);
41
+ expect(isValidStripeId('cus', 'customer')).toBe(false);
42
+ // Empty or null - use unknown cast to test defensive behavior
43
+ expect(isValidStripeId('', 'customer')).toBe(false);
44
+ expect(isValidStripeId(null, 'customer')).toBe(false);
45
+ expect(isValidStripeId(undefined, 'customer')).toBe(false);
46
+ });
47
+ it('should reject IDs exceeding max length', () => {
48
+ // ID exactly at max length (4 char prefix + 96 chars = 100)
49
+ const maxLengthId = 'cus_' + 'a'.repeat(96);
50
+ expect(isValidStripeId(maxLengthId, 'customer')).toBe(true); // exactly at default max of 100
51
+ // ID exceeding max length
52
+ const longId = 'cus_' + 'a'.repeat(100); // 104 chars total
53
+ expect(isValidStripeId(longId, 'customer')).toBe(false); // exceeds default max of 100
54
+ expect(isValidStripeId(longId, 'customer', 50)).toBe(false);
55
+ });
56
+ });
57
+ describe('sanitizeStripeCustomerId', () => {
58
+ it('should return valid customer IDs unchanged', () => {
59
+ const id = 'cus_OtLqEJvHu1Mvxm';
60
+ expect(sanitizeStripeCustomerId(id)).toBe(id);
61
+ });
62
+ it('should return empty string for invalid customer IDs', () => {
63
+ expect(sanitizeStripeCustomerId('invalid')).toBe('');
64
+ expect(sanitizeStripeCustomerId('sub_123456')).toBe('');
65
+ expect(sanitizeStripeCustomerId('cus_<script>alert(1)</script>')).toBe('');
66
+ });
67
+ });
68
+ describe('sanitizeStripeSubscriptionId', () => {
69
+ it('should return valid subscription IDs unchanged', () => {
70
+ const id = 'sub_1234567890abcdef';
71
+ expect(sanitizeStripeSubscriptionId(id)).toBe(id);
72
+ });
73
+ it('should return empty string for invalid subscription IDs', () => {
74
+ expect(sanitizeStripeSubscriptionId('cus_123456')).toBe('');
75
+ expect(sanitizeStripeSubscriptionId('invalid')).toBe('');
76
+ });
77
+ });
78
+ describe('sanitizeStripePriceId', () => {
79
+ it('should return valid price IDs unchanged', () => {
80
+ const id = 'price_1234567890abcdef';
81
+ expect(sanitizeStripePriceId(id)).toBe(id);
82
+ });
83
+ it('should return empty string for invalid price IDs', () => {
84
+ expect(sanitizeStripePriceId('invalid')).toBe('');
85
+ expect(sanitizeStripePriceId('prod_123456')).toBe('');
86
+ });
87
+ });
88
+ describe('sanitizeStripeInvoiceId', () => {
89
+ it('should return valid invoice IDs unchanged', () => {
90
+ const id = 'in_1234567890abcdef';
91
+ expect(sanitizeStripeInvoiceId(id)).toBe(id);
92
+ });
93
+ it('should return empty string for invalid invoice IDs', () => {
94
+ expect(sanitizeStripeInvoiceId('invalid')).toBe('');
95
+ });
96
+ });
97
+ describe('sanitizeStripeEventId', () => {
98
+ it('should return valid event IDs unchanged', () => {
99
+ const id = 'evt_1234567890abcdef';
100
+ expect(sanitizeStripeEventId(id)).toBe(id);
101
+ });
102
+ it('should return empty string for invalid event IDs', () => {
103
+ expect(sanitizeStripeEventId('invalid')).toBe('');
104
+ });
105
+ });
106
+ });
107
+ //# sourceMappingURL=stripe-validators.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stripe-validators.test.js","sourceRoot":"","sources":["../../../tests/billing/stripe-validators.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * SMI-1519: HNSW Embedding Store Tests
3
+ *
4
+ * Tests for the hybrid HNSW + SQLite embedding storage.
5
+ */
6
+ export {};
7
+ //# sourceMappingURL=hnsw-store.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hnsw-store.test.d.ts","sourceRoot":"","sources":["../../../tests/embeddings/hnsw-store.test.ts"],"names":[],"mappings":"AAAA;;;;GAIG"}