@catalyst-team/poly-sdk 0.2.0 → 0.2.1

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 (197) hide show
  1. package/LICENSE +1 -1
  2. package/README.en.md +8 -44
  3. package/README.md +5 -3
  4. package/README.zh-CN.md +502 -0
  5. package/dist/__tests__/clob-api.test.d.ts +5 -0
  6. package/dist/__tests__/clob-api.test.d.ts.map +1 -0
  7. package/dist/__tests__/clob-api.test.js +240 -0
  8. package/dist/__tests__/clob-api.test.js.map +1 -0
  9. package/dist/__tests__/integration/bridge-client.integration.test.d.ts +11 -0
  10. package/dist/__tests__/integration/bridge-client.integration.test.d.ts.map +1 -0
  11. package/dist/__tests__/integration/bridge-client.integration.test.js +260 -0
  12. package/dist/__tests__/integration/bridge-client.integration.test.js.map +1 -0
  13. package/dist/__tests__/integration/clob-api.integration.test.d.ts +13 -0
  14. package/dist/__tests__/integration/clob-api.integration.test.d.ts.map +1 -0
  15. package/dist/__tests__/integration/clob-api.integration.test.js +170 -0
  16. package/dist/__tests__/integration/clob-api.integration.test.js.map +1 -0
  17. package/dist/__tests__/integration/ctf-client.integration.test.d.ts +17 -0
  18. package/dist/__tests__/integration/ctf-client.integration.test.d.ts.map +1 -0
  19. package/dist/__tests__/integration/ctf-client.integration.test.js +234 -0
  20. package/dist/__tests__/integration/ctf-client.integration.test.js.map +1 -0
  21. package/dist/__tests__/integration/data-api.integration.test.d.ts +9 -0
  22. package/dist/__tests__/integration/data-api.integration.test.d.ts.map +1 -0
  23. package/dist/__tests__/integration/data-api.integration.test.js +161 -0
  24. package/dist/__tests__/integration/data-api.integration.test.js.map +1 -0
  25. package/dist/__tests__/integration/gamma-api.integration.test.d.ts +9 -0
  26. package/dist/__tests__/integration/gamma-api.integration.test.d.ts.map +1 -0
  27. package/dist/__tests__/integration/gamma-api.integration.test.js +170 -0
  28. package/dist/__tests__/integration/gamma-api.integration.test.js.map +1 -0
  29. package/dist/__tests__/test-utils.d.ts +92 -0
  30. package/dist/__tests__/test-utils.d.ts.map +1 -0
  31. package/dist/__tests__/test-utils.js +143 -0
  32. package/dist/__tests__/test-utils.js.map +1 -0
  33. package/dist/clients/bridge-client.d.ts +388 -0
  34. package/dist/clients/bridge-client.d.ts.map +1 -0
  35. package/dist/clients/bridge-client.js +587 -0
  36. package/dist/clients/bridge-client.js.map +1 -0
  37. package/dist/clients/clob-api.d.ts +318 -0
  38. package/dist/clients/clob-api.d.ts.map +1 -0
  39. package/dist/clients/clob-api.js +388 -0
  40. package/dist/clients/clob-api.js.map +1 -0
  41. package/dist/clients/ctf-client.d.ts +473 -0
  42. package/dist/clients/ctf-client.d.ts.map +1 -0
  43. package/dist/clients/ctf-client.js +915 -0
  44. package/dist/clients/ctf-client.js.map +1 -0
  45. package/dist/clients/data-api.d.ts +134 -0
  46. package/dist/clients/data-api.d.ts.map +1 -0
  47. package/dist/clients/data-api.js +265 -0
  48. package/dist/clients/data-api.js.map +1 -0
  49. package/dist/clients/gamma-api.d.ts +401 -0
  50. package/dist/clients/gamma-api.d.ts.map +1 -0
  51. package/dist/clients/gamma-api.js +352 -0
  52. package/dist/clients/gamma-api.js.map +1 -0
  53. package/dist/clients/trading-client.d.ts +252 -0
  54. package/dist/clients/trading-client.d.ts.map +1 -0
  55. package/dist/clients/trading-client.js +543 -0
  56. package/dist/clients/trading-client.js.map +1 -0
  57. package/dist/clients/websocket-manager.d.ts +100 -0
  58. package/dist/clients/websocket-manager.d.ts.map +1 -0
  59. package/dist/clients/websocket-manager.js +193 -0
  60. package/dist/clients/websocket-manager.js.map +1 -0
  61. package/dist/core/cache-adapter-bridge.d.ts +36 -0
  62. package/dist/core/cache-adapter-bridge.d.ts.map +1 -0
  63. package/dist/core/cache-adapter-bridge.js +81 -0
  64. package/dist/core/cache-adapter-bridge.js.map +1 -0
  65. package/dist/core/cache.d.ts +40 -0
  66. package/dist/core/cache.d.ts.map +1 -0
  67. package/dist/core/cache.js +71 -0
  68. package/dist/core/cache.js.map +1 -0
  69. package/dist/core/errors.d.ts +38 -0
  70. package/dist/core/errors.d.ts.map +1 -0
  71. package/dist/core/errors.js +84 -0
  72. package/dist/core/errors.js.map +1 -0
  73. package/dist/core/rate-limiter.d.ts +31 -0
  74. package/dist/core/rate-limiter.d.ts.map +1 -0
  75. package/dist/core/rate-limiter.js +70 -0
  76. package/dist/core/rate-limiter.js.map +1 -0
  77. package/{src/core/types.ts → dist/core/types.d.ts} +169 -215
  78. package/dist/core/types.d.ts.map +1 -0
  79. package/dist/core/types.js +19 -0
  80. package/dist/core/types.js.map +1 -0
  81. package/dist/core/unified-cache.d.ts +63 -0
  82. package/dist/core/unified-cache.d.ts.map +1 -0
  83. package/dist/core/unified-cache.js +114 -0
  84. package/dist/core/unified-cache.js.map +1 -0
  85. package/dist/index.d.ts +93 -0
  86. package/dist/index.d.ts.map +1 -0
  87. package/dist/index.js +255 -0
  88. package/dist/index.js.map +1 -0
  89. package/dist/services/arbitrage-service.d.ts +408 -0
  90. package/dist/services/arbitrage-service.d.ts.map +1 -0
  91. package/dist/services/arbitrage-service.js +1422 -0
  92. package/dist/services/arbitrage-service.js.map +1 -0
  93. package/dist/services/authorization-service.d.ts +97 -0
  94. package/dist/services/authorization-service.d.ts.map +1 -0
  95. package/dist/services/authorization-service.js +279 -0
  96. package/dist/services/authorization-service.js.map +1 -0
  97. package/dist/services/market-service.d.ts +108 -0
  98. package/dist/services/market-service.d.ts.map +1 -0
  99. package/dist/services/market-service.js +458 -0
  100. package/dist/services/market-service.js.map +1 -0
  101. package/dist/services/realtime-service.d.ts +82 -0
  102. package/dist/services/realtime-service.d.ts.map +1 -0
  103. package/dist/services/realtime-service.js +150 -0
  104. package/dist/services/realtime-service.js.map +1 -0
  105. package/dist/services/swap-service.d.ts +217 -0
  106. package/dist/services/swap-service.d.ts.map +1 -0
  107. package/dist/services/swap-service.js +695 -0
  108. package/dist/services/swap-service.js.map +1 -0
  109. package/dist/services/wallet-service.d.ts +94 -0
  110. package/dist/services/wallet-service.d.ts.map +1 -0
  111. package/dist/services/wallet-service.js +173 -0
  112. package/dist/services/wallet-service.js.map +1 -0
  113. package/dist/utils/price-utils.d.ts +153 -0
  114. package/dist/utils/price-utils.d.ts.map +1 -0
  115. package/dist/utils/price-utils.js +236 -0
  116. package/dist/utils/price-utils.js.map +1 -0
  117. package/package.json +7 -2
  118. package/docs/00-design.md +0 -760
  119. package/docs/02-API.md +0 -1148
  120. package/docs/arb/test-plan.md +0 -387
  121. package/docs/arb/test-results.md +0 -336
  122. package/docs/arbitrage.md +0 -754
  123. package/docs/reports/smart-money-analysis-2025-12-23-cn.md +0 -840
  124. package/examples/01-basic-usage.ts +0 -68
  125. package/examples/02-smart-money.ts +0 -95
  126. package/examples/03-market-analysis.ts +0 -108
  127. package/examples/04-kline-aggregation.ts +0 -158
  128. package/examples/05-follow-wallet-strategy.ts +0 -156
  129. package/examples/06-services-demo.ts +0 -124
  130. package/examples/07-realtime-websocket.ts +0 -117
  131. package/examples/08-trading-orders.ts +0 -278
  132. package/examples/09-rewards-tracking.ts +0 -187
  133. package/examples/10-ctf-operations.ts +0 -336
  134. package/examples/11-live-arbitrage-scan.ts +0 -221
  135. package/examples/12-trending-arb-monitor.ts +0 -406
  136. package/examples/13-arbitrage-service.ts +0 -211
  137. package/examples/README.md +0 -179
  138. package/scripts/README.md +0 -163
  139. package/scripts/approvals/approve-erc1155.ts +0 -129
  140. package/scripts/approvals/approve-neg-risk-erc1155.ts +0 -149
  141. package/scripts/approvals/approve-neg-risk.ts +0 -102
  142. package/scripts/approvals/check-all-allowances.ts +0 -150
  143. package/scripts/approvals/check-allowance.ts +0 -129
  144. package/scripts/approvals/check-ctf-approval.ts +0 -158
  145. package/scripts/arb/faze-bo3-arb.ts +0 -385
  146. package/scripts/arb/settle-position.ts +0 -190
  147. package/scripts/arb/token-rebalancer.ts +0 -420
  148. package/scripts/arb-tests/01-unit-tests.ts +0 -495
  149. package/scripts/arb-tests/02-integration-tests.ts +0 -412
  150. package/scripts/arb-tests/03-e2e-tests.ts +0 -503
  151. package/scripts/arb-tests/README.md +0 -109
  152. package/scripts/datas/001-report.md +0 -486
  153. package/scripts/datas/clone-modal-screenshot.png +0 -0
  154. package/scripts/deposit/deposit-native-usdc.ts +0 -179
  155. package/scripts/deposit/deposit-usdc.ts +0 -155
  156. package/scripts/deposit/swap-usdc-to-usdce.ts +0 -375
  157. package/scripts/research/research-markets.ts +0 -166
  158. package/scripts/trading/check-orders.ts +0 -50
  159. package/scripts/trading/sell-nvidia-positions.ts +0 -206
  160. package/scripts/trading/test-order.ts +0 -172
  161. package/scripts/verify/test-approve-trading.ts +0 -98
  162. package/scripts/verify/test-provider-fix.ts +0 -43
  163. package/scripts/verify/test-search-mcp.ts +0 -113
  164. package/scripts/verify/verify-all-apis.ts +0 -160
  165. package/scripts/wallet/check-wallet-balances.ts +0 -75
  166. package/scripts/wallet/test-wallet-operations.ts +0 -191
  167. package/scripts/wallet/verify-wallet-tools.ts +0 -124
  168. package/src/__tests__/clob-api.test.ts +0 -301
  169. package/src/__tests__/integration/bridge-client.integration.test.ts +0 -314
  170. package/src/__tests__/integration/clob-api.integration.test.ts +0 -218
  171. package/src/__tests__/integration/ctf-client.integration.test.ts +0 -331
  172. package/src/__tests__/integration/data-api.integration.test.ts +0 -194
  173. package/src/__tests__/integration/gamma-api.integration.test.ts +0 -206
  174. package/src/__tests__/test-utils.ts +0 -170
  175. package/src/clients/bridge-client.ts +0 -841
  176. package/src/clients/clob-api.ts +0 -629
  177. package/src/clients/ctf-client.ts +0 -1216
  178. package/src/clients/data-api.ts +0 -469
  179. package/src/clients/gamma-api.ts +0 -597
  180. package/src/clients/trading-client.ts +0 -749
  181. package/src/clients/websocket-manager.ts +0 -267
  182. package/src/core/cache-adapter-bridge.ts +0 -94
  183. package/src/core/cache.ts +0 -85
  184. package/src/core/errors.ts +0 -117
  185. package/src/core/rate-limiter.ts +0 -74
  186. package/src/core/unified-cache.ts +0 -153
  187. package/src/index.ts +0 -461
  188. package/src/services/arbitrage-service.ts +0 -1807
  189. package/src/services/authorization-service.ts +0 -357
  190. package/src/services/market-service.ts +0 -544
  191. package/src/services/realtime-service.ts +0 -196
  192. package/src/services/swap-service.ts +0 -896
  193. package/src/services/wallet-service.ts +0 -259
  194. package/src/utils/price-utils.ts +0 -307
  195. package/tsconfig.json +0 -8
  196. package/vitest.config.ts +0 -19
  197. package/vitest.integration.config.ts +0 -18
@@ -1,206 +0,0 @@
1
- /**
2
- * Gamma API Client Integration Tests
3
- *
4
- * These tests make REAL API calls to Polymarket.
5
- *
6
- * Run with: pnpm test:integration
7
- */
8
-
9
- import { describe, it, expect, beforeAll } from 'vitest';
10
- import { GammaApiClient } from '../../clients/gamma-api.js';
11
- import { RateLimiter } from '../../core/rate-limiter.js';
12
- import { createUnifiedCache } from '../../core/unified-cache.js';
13
-
14
- describe('GammaApiClient Integration', () => {
15
- let client: GammaApiClient;
16
-
17
- beforeAll(() => {
18
- client = new GammaApiClient(new RateLimiter(), createUnifiedCache());
19
- });
20
-
21
- describe('getMarkets', () => {
22
- it('should fetch active markets sorted by volume', async () => {
23
- const markets = await client.getMarkets({
24
- active: true,
25
- closed: false,
26
- order: 'volume24hr',
27
- ascending: false,
28
- limit: 10,
29
- });
30
-
31
- expect(Array.isArray(markets)).toBe(true);
32
- expect(markets.length).toBeGreaterThan(0);
33
- expect(markets.length).toBeLessThanOrEqual(10);
34
-
35
- // Verify each market structure
36
- for (const market of markets) {
37
- expect(typeof market.id).toBe('string');
38
- expect(typeof market.conditionId).toBe('string');
39
- expect(typeof market.question).toBe('string');
40
- expect(market.question.length).toBeGreaterThan(0);
41
- expect(Array.isArray(market.outcomes)).toBe(true);
42
- expect(Array.isArray(market.outcomePrices)).toBe(true);
43
- expect(typeof market.volume).toBe('number');
44
- expect(typeof market.active).toBe('boolean');
45
- expect(market.active).toBe(true); // We filtered for active
46
- }
47
-
48
- // Verify volume24hr is sorted descending
49
- for (let i = 1; i < markets.length; i++) {
50
- const prev = markets[i - 1].volume24hr ?? 0;
51
- const curr = markets[i].volume24hr ?? 0;
52
- expect(prev).toBeGreaterThanOrEqual(curr);
53
- }
54
-
55
- console.log(`✓ Fetched ${markets.length} active markets sorted by 24h volume`);
56
- console.log(` Top market: "${markets[0].question.slice(0, 50)}..."`);
57
- console.log(` 24h volume: $${markets[0].volume24hr?.toLocaleString()}`);
58
- }, 30000);
59
-
60
- it('should return correct outcome prices', async () => {
61
- const markets = await client.getMarkets({
62
- active: true,
63
- limit: 5,
64
- });
65
-
66
- for (const market of markets) {
67
- // Outcome prices should be between 0 and 1
68
- for (const price of market.outcomePrices) {
69
- expect(price).toBeGreaterThanOrEqual(0);
70
- expect(price).toBeLessThanOrEqual(1);
71
- }
72
-
73
- // For binary markets, prices should roughly sum to 1
74
- // But some markets might have no liquidity (0 prices)
75
- if (market.outcomes.length === 2) {
76
- const sum = market.outcomePrices.reduce((a, b) => a + b, 0);
77
- // Allow for no liquidity (0) or normal spread
78
- if (sum > 0) {
79
- expect(sum).toBeGreaterThan(0.8);
80
- expect(sum).toBeLessThan(1.2);
81
- }
82
- }
83
- }
84
-
85
- console.log('✓ Outcome prices validated for all markets');
86
- }, 30000);
87
-
88
- it('should handle pagination correctly', async () => {
89
- const page1 = await client.getMarkets({
90
- active: true,
91
- limit: 5,
92
- offset: 0,
93
- });
94
-
95
- const page2 = await client.getMarkets({
96
- active: true,
97
- limit: 5,
98
- offset: 5,
99
- });
100
-
101
- // Pages should have different markets
102
- const page1Ids = new Set(page1.map(m => m.id));
103
- const page2Ids = new Set(page2.map(m => m.id));
104
-
105
- for (const id of page2Ids) {
106
- expect(page1Ids.has(id)).toBe(false);
107
- }
108
-
109
- console.log(`✓ Pagination works: page1 has ${page1.length} markets, page2 has ${page2.length} markets`);
110
- }, 30000);
111
- });
112
-
113
- describe('getTrendingMarkets', () => {
114
- it('should return trending markets by 24h volume', async () => {
115
- const trending = await client.getTrendingMarkets(10);
116
-
117
- expect(trending.length).toBeGreaterThan(0);
118
-
119
- // Should all be active and not closed
120
- for (const market of trending) {
121
- expect(market.active).toBe(true);
122
- expect(market.closed).toBe(false);
123
- }
124
-
125
- // Should be sorted by volume24hr descending
126
- for (let i = 1; i < trending.length; i++) {
127
- const prev = trending[i - 1].volume24hr ?? 0;
128
- const curr = trending[i].volume24hr ?? 0;
129
- expect(prev).toBeGreaterThanOrEqual(curr);
130
- }
131
-
132
- console.log(`✓ Top ${trending.length} trending markets retrieved`);
133
- }, 30000);
134
- });
135
-
136
- describe('getEvents', () => {
137
- it('should fetch events with associated markets', async () => {
138
- const events = await client.getEvents({
139
- active: true,
140
- limit: 5,
141
- });
142
-
143
- expect(Array.isArray(events)).toBe(true);
144
-
145
- if (events.length > 0) {
146
- const event = events[0];
147
-
148
- expect(typeof event.id).toBe('string');
149
- expect(typeof event.title).toBe('string');
150
- expect(event.title.length).toBeGreaterThan(0);
151
- expect(Array.isArray(event.markets)).toBe(true);
152
-
153
- console.log(`✓ Fetched ${events.length} events`);
154
- console.log(` First event: "${event.title}"`);
155
- console.log(` Markets in event: ${event.markets.length}`);
156
- } else {
157
- console.log('✓ No active events found (this is ok)');
158
- }
159
- }, 30000);
160
- });
161
-
162
- describe('getMarketBySlug', () => {
163
- it('should find market by slug', async () => {
164
- // First get a market to know its slug
165
- const markets = await client.getMarkets({ active: true, limit: 1 });
166
-
167
- if (markets.length === 0) {
168
- console.log('No markets found, skipping test');
169
- return;
170
- }
171
-
172
- const slug = markets[0].slug;
173
- const market = await client.getMarketBySlug(slug);
174
-
175
- expect(market).not.toBeNull();
176
- expect(market?.slug).toBe(slug);
177
- expect(market?.question).toBe(markets[0].question);
178
-
179
- console.log(`✓ Found market by slug: "${slug}"`);
180
- }, 30000);
181
-
182
- it('should return null for non-existent slug', async () => {
183
- const market = await client.getMarketBySlug('this-market-definitely-does-not-exist-12345');
184
- expect(market).toBeNull();
185
- }, 30000);
186
- });
187
-
188
- describe('getMarketByConditionId', () => {
189
- it('should find market by condition ID', async () => {
190
- const markets = await client.getMarkets({ active: true, limit: 1 });
191
-
192
- if (markets.length === 0) {
193
- console.log('No markets found, skipping test');
194
- return;
195
- }
196
-
197
- const conditionId = markets[0].conditionId;
198
- const market = await client.getMarketByConditionId(conditionId);
199
-
200
- expect(market).not.toBeNull();
201
- expect(market?.conditionId).toBe(conditionId);
202
-
203
- console.log(`✓ Found market by conditionId: "${conditionId.slice(0, 20)}..."`);
204
- }, 30000);
205
- });
206
- });
@@ -1,170 +0,0 @@
1
- /**
2
- * Test Utilities for poly-sdk
3
- *
4
- * Provides mock implementations and test helpers for unit testing.
5
- */
6
-
7
- import { vi } from 'vitest';
8
-
9
- // ===== Mock Data =====
10
-
11
- export const mockClobMarket = {
12
- conditionId: '0x82ace55cdcba920112a2b3548f21e6e117730144db4dd580456aaecf1a2ad751',
13
- question: 'Will BTC reach $100k by end of 2024?',
14
- description: 'This market resolves YES if Bitcoin reaches $100,000 USD.',
15
- marketSlug: 'will-btc-reach-100k',
16
- tokens: [
17
- {
18
- tokenId: '21742633143463906290569050155826241533067272736897614950488156847949938836455',
19
- outcome: 'Yes',
20
- price: 0.65,
21
- },
22
- {
23
- tokenId: '79707176503087022804581743747659199568535926050896809695934676056217229461419',
24
- outcome: 'No',
25
- price: 0.35,
26
- },
27
- ],
28
- acceptingOrders: true,
29
- endDateIso: '2024-12-31T23:59:59Z',
30
- active: true,
31
- closed: false,
32
- };
33
-
34
- export const mockOrderbook = {
35
- bids: [
36
- { price: 0.55, size: 1000 },
37
- { price: 0.54, size: 500 },
38
- { price: 0.53, size: 750 },
39
- ],
40
- asks: [
41
- { price: 0.57, size: 800 },
42
- { price: 0.58, size: 600 },
43
- { price: 0.59, size: 400 },
44
- ],
45
- timestamp: Date.now(),
46
- };
47
-
48
- export const mockNoOrderbook = {
49
- bids: [
50
- { price: 0.43, size: 900 },
51
- { price: 0.42, size: 450 },
52
- ],
53
- asks: [
54
- { price: 0.45, size: 700 },
55
- { price: 0.46, size: 500 },
56
- ],
57
- timestamp: Date.now(),
58
- };
59
-
60
- export const mockGammaMarket = {
61
- id: '12345',
62
- conditionId: '0x82ace55cdcba920112a2b3548f21e6e117730144db4dd580456aaecf1a2ad751',
63
- slug: 'will-btc-reach-100k',
64
- question: 'Will BTC reach $100k by end of 2024?',
65
- outcomes: ['Yes', 'No'],
66
- outcomePrices: [0.65, 0.35],
67
- volume: 5000000,
68
- volume24hr: 250000,
69
- liquidity: 100000,
70
- spread: 0.02,
71
- endDate: new Date('2024-12-31T23:59:59Z'),
72
- active: true,
73
- closed: false,
74
- };
75
-
76
- export const mockPosition = {
77
- asset: '21742633143463906290569050155826241533067272736897614950488156847949938836455',
78
- conditionId: '0x82ace55cdcba920112a2b3548f21e6e117730144db4dd580456aaecf1a2ad751',
79
- outcome: 'Yes',
80
- outcomeIndex: 0,
81
- size: 100,
82
- avgPrice: 0.50,
83
- curPrice: 0.65,
84
- cashPnl: 15,
85
- percentPnl: 0.30,
86
- title: 'Will BTC reach $100k?',
87
- };
88
-
89
- // ===== Mock RateLimiter =====
90
-
91
- export class MockRateLimiter {
92
- async execute<T>(_apiType: string, fn: () => Promise<T>): Promise<T> {
93
- return fn();
94
- }
95
- }
96
-
97
- // ===== Mock Cache =====
98
-
99
- export class MockCache {
100
- private cache = new Map<string, { value: unknown; expires: number }>();
101
-
102
- async getOrSet<T>(
103
- key: string,
104
- ttl: number,
105
- fn: () => Promise<T>
106
- ): Promise<T> {
107
- const cached = this.cache.get(key);
108
- if (cached && cached.expires > Date.now()) {
109
- return cached.value as T;
110
- }
111
- const value = await fn();
112
- this.cache.set(key, { value, expires: Date.now() + ttl });
113
- return value;
114
- }
115
-
116
- clear(): void {
117
- this.cache.clear();
118
- }
119
- }
120
-
121
- // ===== Mock Fetch =====
122
-
123
- export function createMockFetch(responses: Record<string, unknown>) {
124
- return vi.fn(async (url: string) => {
125
- const urlObj = new URL(url);
126
- const path = urlObj.pathname;
127
-
128
- for (const [pattern, response] of Object.entries(responses)) {
129
- if (path.includes(pattern)) {
130
- return {
131
- ok: true,
132
- status: 200,
133
- json: async () => response,
134
- };
135
- }
136
- }
137
-
138
- return {
139
- ok: false,
140
- status: 404,
141
- json: async () => ({ error: 'Not found' }),
142
- };
143
- });
144
- }
145
-
146
- // ===== Assertion Helpers =====
147
-
148
- export function expectOrderbookSorted(orderbook: {
149
- bids: Array<{ price: number }>;
150
- asks: Array<{ price: number }>;
151
- }): void {
152
- // Bids should be descending
153
- for (let i = 1; i < orderbook.bids.length; i++) {
154
- if (orderbook.bids[i].price > orderbook.bids[i - 1].price) {
155
- throw new Error('Bids are not sorted descending');
156
- }
157
- }
158
- // Asks should be ascending
159
- for (let i = 1; i < orderbook.asks.length; i++) {
160
- if (orderbook.asks[i].price < orderbook.asks[i - 1].price) {
161
- throw new Error('Asks are not sorted ascending');
162
- }
163
- }
164
- }
165
-
166
- // ===== Time Helpers =====
167
-
168
- export async function waitFor(ms: number): Promise<void> {
169
- return new Promise((resolve) => setTimeout(resolve, ms));
170
- }