@catalyst-team/poly-sdk 0.1.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 (244) hide show
  1. package/.env +0 -0
  2. package/README.md +803 -0
  3. package/dist/__tests__/clob-api.test.d.ts +5 -0
  4. package/dist/__tests__/clob-api.test.d.ts.map +1 -0
  5. package/dist/__tests__/clob-api.test.js +240 -0
  6. package/dist/__tests__/clob-api.test.js.map +1 -0
  7. package/dist/__tests__/integration/bridge-client.integration.test.d.ts +11 -0
  8. package/dist/__tests__/integration/bridge-client.integration.test.d.ts.map +1 -0
  9. package/dist/__tests__/integration/bridge-client.integration.test.js +260 -0
  10. package/dist/__tests__/integration/bridge-client.integration.test.js.map +1 -0
  11. package/dist/__tests__/integration/clob-api.integration.test.d.ts +13 -0
  12. package/dist/__tests__/integration/clob-api.integration.test.d.ts.map +1 -0
  13. package/dist/__tests__/integration/clob-api.integration.test.js +170 -0
  14. package/dist/__tests__/integration/clob-api.integration.test.js.map +1 -0
  15. package/dist/__tests__/integration/ctf-client.integration.test.d.ts +17 -0
  16. package/dist/__tests__/integration/ctf-client.integration.test.d.ts.map +1 -0
  17. package/dist/__tests__/integration/ctf-client.integration.test.js +234 -0
  18. package/dist/__tests__/integration/ctf-client.integration.test.js.map +1 -0
  19. package/dist/__tests__/integration/data-api.integration.test.d.ts +9 -0
  20. package/dist/__tests__/integration/data-api.integration.test.d.ts.map +1 -0
  21. package/dist/__tests__/integration/data-api.integration.test.js +161 -0
  22. package/dist/__tests__/integration/data-api.integration.test.js.map +1 -0
  23. package/dist/__tests__/integration/gamma-api.integration.test.d.ts +9 -0
  24. package/dist/__tests__/integration/gamma-api.integration.test.d.ts.map +1 -0
  25. package/dist/__tests__/integration/gamma-api.integration.test.js +170 -0
  26. package/dist/__tests__/integration/gamma-api.integration.test.js.map +1 -0
  27. package/dist/__tests__/test-utils.d.ts +92 -0
  28. package/dist/__tests__/test-utils.d.ts.map +1 -0
  29. package/dist/__tests__/test-utils.js +143 -0
  30. package/dist/__tests__/test-utils.js.map +1 -0
  31. package/dist/clients/bridge-client.d.ts +388 -0
  32. package/dist/clients/bridge-client.d.ts.map +1 -0
  33. package/dist/clients/bridge-client.js +587 -0
  34. package/dist/clients/bridge-client.js.map +1 -0
  35. package/dist/clients/clob-api.d.ts +318 -0
  36. package/dist/clients/clob-api.d.ts.map +1 -0
  37. package/dist/clients/clob-api.js +388 -0
  38. package/dist/clients/clob-api.js.map +1 -0
  39. package/dist/clients/ctf-client.d.ts +473 -0
  40. package/dist/clients/ctf-client.d.ts.map +1 -0
  41. package/dist/clients/ctf-client.js +915 -0
  42. package/dist/clients/ctf-client.js.map +1 -0
  43. package/dist/clients/data-api.d.ts +134 -0
  44. package/dist/clients/data-api.d.ts.map +1 -0
  45. package/dist/clients/data-api.js +265 -0
  46. package/dist/clients/data-api.js.map +1 -0
  47. package/dist/clients/gamma-api.d.ts +401 -0
  48. package/dist/clients/gamma-api.d.ts.map +1 -0
  49. package/dist/clients/gamma-api.js +352 -0
  50. package/dist/clients/gamma-api.js.map +1 -0
  51. package/dist/clients/trading-client.d.ts +252 -0
  52. package/dist/clients/trading-client.d.ts.map +1 -0
  53. package/dist/clients/trading-client.js +543 -0
  54. package/dist/clients/trading-client.js.map +1 -0
  55. package/dist/clients/websocket-manager.d.ts +100 -0
  56. package/dist/clients/websocket-manager.d.ts.map +1 -0
  57. package/dist/clients/websocket-manager.js +193 -0
  58. package/dist/clients/websocket-manager.js.map +1 -0
  59. package/dist/core/cache-adapter-bridge.d.ts +36 -0
  60. package/dist/core/cache-adapter-bridge.d.ts.map +1 -0
  61. package/dist/core/cache-adapter-bridge.js +81 -0
  62. package/dist/core/cache-adapter-bridge.js.map +1 -0
  63. package/dist/core/cache.d.ts +40 -0
  64. package/dist/core/cache.d.ts.map +1 -0
  65. package/dist/core/cache.js +71 -0
  66. package/dist/core/cache.js.map +1 -0
  67. package/dist/core/errors.d.ts +38 -0
  68. package/dist/core/errors.d.ts.map +1 -0
  69. package/dist/core/errors.js +84 -0
  70. package/dist/core/errors.js.map +1 -0
  71. package/dist/core/rate-limiter.d.ts +31 -0
  72. package/dist/core/rate-limiter.d.ts.map +1 -0
  73. package/dist/core/rate-limiter.js +70 -0
  74. package/dist/core/rate-limiter.js.map +1 -0
  75. package/dist/core/types.d.ts +314 -0
  76. package/dist/core/types.d.ts.map +1 -0
  77. package/dist/core/types.js +19 -0
  78. package/dist/core/types.js.map +1 -0
  79. package/dist/core/unified-cache.d.ts +63 -0
  80. package/dist/core/unified-cache.d.ts.map +1 -0
  81. package/dist/core/unified-cache.js +114 -0
  82. package/dist/core/unified-cache.js.map +1 -0
  83. package/dist/index.d.ts +94 -0
  84. package/dist/index.d.ts.map +1 -0
  85. package/dist/index.js +258 -0
  86. package/dist/index.js.map +1 -0
  87. package/dist/mcp/errors.d.ts +33 -0
  88. package/dist/mcp/errors.d.ts.map +1 -0
  89. package/dist/mcp/errors.js +86 -0
  90. package/dist/mcp/errors.js.map +1 -0
  91. package/dist/mcp/index.d.ts +62 -0
  92. package/dist/mcp/index.d.ts.map +1 -0
  93. package/dist/mcp/index.js +173 -0
  94. package/dist/mcp/index.js.map +1 -0
  95. package/dist/mcp/server.d.ts +17 -0
  96. package/dist/mcp/server.d.ts.map +1 -0
  97. package/dist/mcp/server.js +155 -0
  98. package/dist/mcp/server.js.map +1 -0
  99. package/dist/mcp/tools/guide.d.ts +12 -0
  100. package/dist/mcp/tools/guide.d.ts.map +1 -0
  101. package/dist/mcp/tools/guide.js +801 -0
  102. package/dist/mcp/tools/guide.js.map +1 -0
  103. package/dist/mcp/tools/index.d.ts +11 -0
  104. package/dist/mcp/tools/index.d.ts.map +1 -0
  105. package/dist/mcp/tools/index.js +27 -0
  106. package/dist/mcp/tools/index.js.map +1 -0
  107. package/dist/mcp/tools/market.d.ts +11 -0
  108. package/dist/mcp/tools/market.d.ts.map +1 -0
  109. package/dist/mcp/tools/market.js +314 -0
  110. package/dist/mcp/tools/market.js.map +1 -0
  111. package/dist/mcp/tools/order.d.ts +10 -0
  112. package/dist/mcp/tools/order.d.ts.map +1 -0
  113. package/dist/mcp/tools/order.js +258 -0
  114. package/dist/mcp/tools/order.js.map +1 -0
  115. package/dist/mcp/tools/trade.d.ts +38 -0
  116. package/dist/mcp/tools/trade.d.ts.map +1 -0
  117. package/dist/mcp/tools/trade.js +314 -0
  118. package/dist/mcp/tools/trade.js.map +1 -0
  119. package/dist/mcp/tools/trader.d.ts +11 -0
  120. package/dist/mcp/tools/trader.d.ts.map +1 -0
  121. package/dist/mcp/tools/trader.js +277 -0
  122. package/dist/mcp/tools/trader.js.map +1 -0
  123. package/dist/mcp/tools/wallet.d.ts +274 -0
  124. package/dist/mcp/tools/wallet.d.ts.map +1 -0
  125. package/dist/mcp/tools/wallet.js +579 -0
  126. package/dist/mcp/tools/wallet.js.map +1 -0
  127. package/dist/mcp/types.d.ts +413 -0
  128. package/dist/mcp/types.d.ts.map +1 -0
  129. package/dist/mcp/types.js +5 -0
  130. package/dist/mcp/types.js.map +1 -0
  131. package/dist/services/authorization-service.d.ts +97 -0
  132. package/dist/services/authorization-service.d.ts.map +1 -0
  133. package/dist/services/authorization-service.js +279 -0
  134. package/dist/services/authorization-service.js.map +1 -0
  135. package/dist/services/market-service.d.ts +108 -0
  136. package/dist/services/market-service.d.ts.map +1 -0
  137. package/dist/services/market-service.js +458 -0
  138. package/dist/services/market-service.js.map +1 -0
  139. package/dist/services/realtime-service.d.ts +82 -0
  140. package/dist/services/realtime-service.d.ts.map +1 -0
  141. package/dist/services/realtime-service.js +150 -0
  142. package/dist/services/realtime-service.js.map +1 -0
  143. package/dist/services/swap-service.d.ts +217 -0
  144. package/dist/services/swap-service.d.ts.map +1 -0
  145. package/dist/services/swap-service.js +695 -0
  146. package/dist/services/swap-service.js.map +1 -0
  147. package/dist/services/wallet-service.d.ts +94 -0
  148. package/dist/services/wallet-service.d.ts.map +1 -0
  149. package/dist/services/wallet-service.js +173 -0
  150. package/dist/services/wallet-service.js.map +1 -0
  151. package/dist/utils/price-utils.d.ts +153 -0
  152. package/dist/utils/price-utils.d.ts.map +1 -0
  153. package/dist/utils/price-utils.js +236 -0
  154. package/dist/utils/price-utils.js.map +1 -0
  155. package/docs/00-design.md +760 -0
  156. package/docs/01-mcp.md +2041 -0
  157. package/docs/02-API.md +1148 -0
  158. package/docs/e2e/01-trader-tools.md +159 -0
  159. package/docs/e2e/02-market-tools.md +180 -0
  160. package/docs/e2e/03-order-tools.md +166 -0
  161. package/docs/e2e/04-wallet-tools.md +224 -0
  162. package/docs/e2e/05-trading-tools.md +327 -0
  163. package/docs/e2e/06-integration-scenarios.md +481 -0
  164. package/docs/e2e/coordinator.md +376 -0
  165. package/examples/01-basic-usage.ts +68 -0
  166. package/examples/02-smart-money.ts +95 -0
  167. package/examples/03-market-analysis.ts +108 -0
  168. package/examples/04-kline-aggregation.ts +158 -0
  169. package/examples/05-follow-wallet-strategy.ts +156 -0
  170. package/examples/06-services-demo.ts +124 -0
  171. package/examples/07-realtime-websocket.ts +117 -0
  172. package/examples/08-trading-orders.ts +278 -0
  173. package/examples/09-rewards-tracking.ts +187 -0
  174. package/examples/10-ctf-operations.ts +336 -0
  175. package/examples/11-live-arbitrage-scan.ts +221 -0
  176. package/examples/12-trending-arb-monitor.ts +406 -0
  177. package/examples/README.md +179 -0
  178. package/package.json +62 -0
  179. package/scripts/README.md +163 -0
  180. package/scripts/approvals/approve-erc1155.ts +129 -0
  181. package/scripts/approvals/approve-neg-risk-erc1155.ts +149 -0
  182. package/scripts/approvals/approve-neg-risk.ts +102 -0
  183. package/scripts/approvals/check-all-allowances.ts +150 -0
  184. package/scripts/approvals/check-allowance.ts +129 -0
  185. package/scripts/approvals/check-ctf-approval.ts +158 -0
  186. package/scripts/datas/001-report.md +486 -0
  187. package/scripts/datas/clone-modal-screenshot.png +0 -0
  188. package/scripts/deposit/deposit-native-usdc.ts +179 -0
  189. package/scripts/deposit/deposit-usdc.ts +155 -0
  190. package/scripts/deposit/swap-usdc-to-usdce.ts +375 -0
  191. package/scripts/research/research-markets.ts +166 -0
  192. package/scripts/trading/check-orders.ts +50 -0
  193. package/scripts/trading/sell-nvidia-positions.ts +206 -0
  194. package/scripts/trading/test-order.ts +172 -0
  195. package/scripts/truth.md +440 -0
  196. package/scripts/verify/test-approve-trading.ts +98 -0
  197. package/scripts/verify/test-provider-fix.ts +43 -0
  198. package/scripts/verify/test-search-mcp.ts +113 -0
  199. package/scripts/verify/verify-all-apis.ts +160 -0
  200. package/scripts/wallet/check-wallet-balances.ts +75 -0
  201. package/scripts/wallet/test-wallet-operations.ts +191 -0
  202. package/scripts/wallet/verify-wallet-tools.ts +124 -0
  203. package/src/__tests__/clob-api.test.ts +301 -0
  204. package/src/__tests__/integration/bridge-client.integration.test.ts +314 -0
  205. package/src/__tests__/integration/clob-api.integration.test.ts +218 -0
  206. package/src/__tests__/integration/ctf-client.integration.test.ts +331 -0
  207. package/src/__tests__/integration/data-api.integration.test.ts +194 -0
  208. package/src/__tests__/integration/gamma-api.integration.test.ts +206 -0
  209. package/src/__tests__/test-utils.ts +170 -0
  210. package/src/clients/bridge-client.ts +841 -0
  211. package/src/clients/clob-api.ts +629 -0
  212. package/src/clients/ctf-client.ts +1216 -0
  213. package/src/clients/data-api.ts +469 -0
  214. package/src/clients/gamma-api.ts +597 -0
  215. package/src/clients/trading-client.ts +749 -0
  216. package/src/clients/websocket-manager.ts +267 -0
  217. package/src/core/cache-adapter-bridge.ts +94 -0
  218. package/src/core/cache.ts +85 -0
  219. package/src/core/errors.ts +117 -0
  220. package/src/core/rate-limiter.ts +74 -0
  221. package/src/core/types.ts +360 -0
  222. package/src/core/unified-cache.ts +153 -0
  223. package/src/index.ts +455 -0
  224. package/src/mcp/README.md +380 -0
  225. package/src/mcp/errors.ts +124 -0
  226. package/src/mcp/index.ts +309 -0
  227. package/src/mcp/server.ts +183 -0
  228. package/src/mcp/tools/guide.ts +821 -0
  229. package/src/mcp/tools/index.ts +73 -0
  230. package/src/mcp/tools/market.ts +363 -0
  231. package/src/mcp/tools/order.ts +326 -0
  232. package/src/mcp/tools/trade.ts +417 -0
  233. package/src/mcp/tools/trader.ts +322 -0
  234. package/src/mcp/tools/wallet.ts +683 -0
  235. package/src/mcp/types.ts +472 -0
  236. package/src/services/authorization-service.ts +357 -0
  237. package/src/services/market-service.ts +544 -0
  238. package/src/services/realtime-service.ts +196 -0
  239. package/src/services/swap-service.ts +896 -0
  240. package/src/services/wallet-service.ts +259 -0
  241. package/src/utils/price-utils.ts +307 -0
  242. package/tsconfig.json +8 -0
  243. package/vitest.config.ts +19 -0
  244. package/vitest.integration.config.ts +18 -0
@@ -0,0 +1,166 @@
1
+ /**
2
+ * Market Research - ARB + MM + HYBRID Analysis
3
+ */
4
+
5
+ import { PolymarketSDK } from '../src/index.js';
6
+
7
+ const sdk = new PolymarketSDK();
8
+
9
+ async function main() {
10
+ console.log('╔══════════════════════════════════════════════════════════════════════════╗');
11
+ console.log('║ ARB + MM + HYBRID Market Research ║');
12
+ console.log('╚══════════════════════════════════════════════════════════════════════════╝');
13
+
14
+ console.log('\nFetching top markets by volume...\n');
15
+
16
+ const markets = await sdk.gammaApi.getMarkets({
17
+ active: true,
18
+ closed: false,
19
+ order: 'volume24hr',
20
+ ascending: false,
21
+ limit: 50,
22
+ });
23
+
24
+ console.log(`Found ${markets.length} markets. Analyzing...\n`);
25
+
26
+ interface MarketResult {
27
+ question: string;
28
+ conditionId: string;
29
+ vol24h: number;
30
+ avgSpread: number;
31
+ longArb: number;
32
+ shortArb: number;
33
+ mid: number;
34
+ minDepth: number;
35
+ yesBid: number;
36
+ yesAsk: number;
37
+ noBid: number;
38
+ noAsk: number;
39
+ }
40
+
41
+ const results: MarketResult[] = [];
42
+
43
+ for (const market of markets) {
44
+ if (!market.conditionId) continue;
45
+
46
+ try {
47
+ const ob = await sdk.clobApi.getProcessedOrderbook(market.conditionId);
48
+
49
+ if (ob.yes.bid < 0.01 || ob.no.bid < 0.01) continue;
50
+
51
+ const vol24h = market.volume24hr || 0;
52
+ const yesSpread = ob.summary.yesSpread || (ob.yes.ask - ob.yes.bid);
53
+ const noSpread = ob.summary.noSpread || (ob.no.ask - ob.no.bid);
54
+ const avgSpread = (yesSpread + noSpread) / 2;
55
+ const longCost = ob.summary.effectiveLongCost;
56
+ const shortRev = ob.summary.effectiveShortRevenue;
57
+ const longArb = 1 - longCost;
58
+ const shortArb = shortRev - 1;
59
+ const mid = (ob.yes.bid + ob.yes.ask) / 2;
60
+ const minDepth = Math.min(ob.yes.bidDepth, ob.yes.askDepth, ob.no.bidDepth, ob.no.askDepth);
61
+
62
+ results.push({
63
+ question: market.question || 'Unknown',
64
+ conditionId: market.conditionId,
65
+ vol24h,
66
+ avgSpread,
67
+ longArb,
68
+ shortArb,
69
+ mid,
70
+ minDepth,
71
+ yesBid: ob.yes.bid,
72
+ yesAsk: ob.yes.ask,
73
+ noBid: ob.no.bid,
74
+ noAsk: ob.no.ask,
75
+ });
76
+
77
+ process.stdout.write('.');
78
+ } catch {
79
+ process.stdout.write('x');
80
+ }
81
+
82
+ await new Promise(r => setTimeout(r, 100));
83
+ }
84
+
85
+ console.log('\n');
86
+
87
+ // Categorize
88
+ const arb = results.filter(r => r.longArb > 0 || r.shortArb > 0);
89
+ const mm = results.filter(r => r.avgSpread >= 0.01 && r.vol24h >= 10000 && r.minDepth >= 1000);
90
+ const hybrid = results.filter(r => r.mid >= 0.25 && r.mid <= 0.75 && r.vol24h >= 5000);
91
+
92
+ console.log('═══════════════════════════════════════════════════════════════════════════');
93
+ console.log('🎯 ARBITRAGE OPPORTUNITIES:', arb.length);
94
+ console.log('═══════════════════════════════════════════════════════════════════════════');
95
+
96
+ if (arb.length === 0) {
97
+ console.log(' No direct arbitrage found (market efficient)\n');
98
+
99
+ // Show closest to arb
100
+ const sorted = [...results].sort((a, b) => Math.max(a.longArb, a.shortArb) - Math.max(b.longArb, b.shortArb)).reverse();
101
+ console.log(' Closest to arbitrage:');
102
+ for (const r of sorted.slice(0, 3)) {
103
+ const profit = Math.max(r.longArb, r.shortArb);
104
+ console.log(` ${r.question.slice(0, 50)}... → ${(profit * 100).toFixed(3)}%`);
105
+ }
106
+ } else {
107
+ for (const r of arb.slice(0, 5)) {
108
+ const type = r.longArb > r.shortArb ? 'LONG' : 'SHORT';
109
+ const profit = Math.max(r.longArb, r.shortArb);
110
+ console.log(`\n ${r.question.slice(0, 60)}...`);
111
+ console.log(` ${type} ARB: ${(profit * 100).toFixed(3)}%`);
112
+ }
113
+ }
114
+
115
+ console.log('\n═══════════════════════════════════════════════════════════════════════════');
116
+ console.log('📊 MARKET MAKING CANDIDATES:', mm.length);
117
+ console.log('═══════════════════════════════════════════════════════════════════════════');
118
+
119
+ const mmSorted = [...mm].sort((a, b) => {
120
+ // Score: spread * volume * sqrt(depth)
121
+ const scoreA = a.avgSpread * Math.log(a.vol24h) * Math.sqrt(a.minDepth);
122
+ const scoreB = b.avgSpread * Math.log(b.vol24h) * Math.sqrt(b.minDepth);
123
+ return scoreB - scoreA;
124
+ });
125
+
126
+ for (const r of mmSorted.slice(0, 10)) {
127
+ console.log(`\n ${r.question.slice(0, 60)}...`);
128
+ console.log(` Spread: ${(r.avgSpread * 100).toFixed(2)}% | Vol: $${(r.vol24h/1000).toFixed(0)}K | Depth: $${(r.minDepth/1000).toFixed(0)}K`);
129
+ console.log(` YES: ${r.yesBid.toFixed(3)}/${r.yesAsk.toFixed(3)} | NO: ${r.noBid.toFixed(3)}/${r.noAsk.toFixed(3)}`);
130
+ }
131
+
132
+ console.log('\n═══════════════════════════════════════════════════════════════════════════');
133
+ console.log('🔄 HYBRID CANDIDATES (uncertain markets 25-75%):', hybrid.length);
134
+ console.log('═══════════════════════════════════════════════════════════════════════════');
135
+
136
+ const hybridSorted = [...hybrid].sort((a, b) => {
137
+ // Prefer prices closer to 50%
138
+ const distA = Math.abs(a.mid - 0.5);
139
+ const distB = Math.abs(b.mid - 0.5);
140
+ return distA - distB;
141
+ });
142
+
143
+ for (const r of hybridSorted.slice(0, 10)) {
144
+ console.log(`\n ${r.question.slice(0, 60)}...`);
145
+ console.log(` Price: ${(r.mid * 100).toFixed(1)}¢ | Vol: $${(r.vol24h/1000).toFixed(0)}K | Spread: ${(r.avgSpread * 100).toFixed(2)}%`);
146
+ }
147
+
148
+ console.log('\n═══════════════════════════════════════════════════════════════════════════');
149
+ console.log('SUMMARY');
150
+ console.log('═══════════════════════════════════════════════════════════════════════════');
151
+ console.log(` Total analyzed: ${results.length}`);
152
+ console.log(` Arb opportunities: ${arb.length}`);
153
+ console.log(` MM candidates: ${mm.length}`);
154
+ console.log(` Hybrid candidates: ${hybrid.length}`);
155
+
156
+ if (mm.length > 0) {
157
+ console.log(`\n 🏆 Top MM recommendation:`);
158
+ console.log(` ${mmSorted[0].question.slice(0, 60)}...`);
159
+ }
160
+ if (hybrid.length > 0) {
161
+ console.log(` 🏆 Top Hybrid recommendation:`);
162
+ console.log(` ${hybridSorted[0].question.slice(0, 60)}...`);
163
+ }
164
+ }
165
+
166
+ main().catch(console.error);
@@ -0,0 +1,50 @@
1
+ /**
2
+ * 检查 Polymarket 真实订单
3
+ */
4
+
5
+ import { TradingClient, RateLimiter } from '../src/index.js';
6
+
7
+ async function main() {
8
+ const privateKey = process.env.POLY_PRIVKEY || process.env.PRIVATE_KEY;
9
+ if (!privateKey) {
10
+ console.error('❌ 没有配置私钥');
11
+ process.exit(1);
12
+ }
13
+
14
+ const rateLimiter = new RateLimiter();
15
+ const client = new TradingClient(rateLimiter, { privateKey });
16
+
17
+ console.log('⏳ 初始化...');
18
+ await client.initialize();
19
+ console.log(`✅ 钱包: ${client.getAddress()}`);
20
+
21
+ // 检查 USDC 余额
22
+ console.log('\n📊 余额:');
23
+ const balance = await client.getBalanceAllowance('COLLATERAL');
24
+ console.log(` USDC: $${(Number(balance.balance) / 1e6).toFixed(2)}`);
25
+ console.log(` Allowance: $${(Number(balance.allowance) / 1e6).toFixed(2)}`);
26
+
27
+ // 检查未完成订单
28
+ console.log('\n📋 Open Orders:');
29
+ const orders = await client.getOpenOrders();
30
+ if (orders.length === 0) {
31
+ console.log(' (无)');
32
+ } else {
33
+ for (const order of orders) {
34
+ console.log(` ${order.id}: ${order.side} ${order.remainingSize}@${order.price} (${order.status})`);
35
+ }
36
+ }
37
+
38
+ // 检查最近交易
39
+ console.log('\n💰 Recent Trades:');
40
+ const trades = await client.getTrades();
41
+ if (trades.length === 0) {
42
+ console.log(' (无)');
43
+ } else {
44
+ for (const trade of trades.slice(0, 10)) {
45
+ console.log(` ${trade.id}: ${trade.side} ${trade.size}@${trade.price} (fee: ${trade.fee})`);
46
+ }
47
+ }
48
+ }
49
+
50
+ main().catch(console.error);
@@ -0,0 +1,206 @@
1
+ #!/usr/bin/env npx tsx
2
+ /**
3
+ * Sell NVIDIA Market Positions
4
+ *
5
+ * Sells both YES and NO tokens from the NVIDIA market to recover USDC.
6
+ *
7
+ * Usage:
8
+ * # Dry run
9
+ * npx tsx scripts/sell-nvidia-positions.ts
10
+ *
11
+ * # Execute
12
+ * npx tsx scripts/sell-nvidia-positions.ts --execute
13
+ */
14
+
15
+ import * as fs from 'fs';
16
+ import * as path from 'path';
17
+ import { fileURLToPath } from 'url';
18
+ import { ethers } from 'ethers';
19
+ import {
20
+ TradingClient,
21
+ RateLimiter,
22
+ ClobApiClient,
23
+ DataApiClient,
24
+ Cache,
25
+ CTFClient,
26
+ } from '../src/index.js';
27
+
28
+ const __filename = fileURLToPath(import.meta.url);
29
+ const __dirname = path.dirname(__filename);
30
+
31
+ // Read private key
32
+ const envPath = path.resolve(__dirname, '../../earning-engine/dashboard-api/.env');
33
+ const envContent = fs.readFileSync(envPath, 'utf8');
34
+ const match = envContent.match(/^PRIVATE_KEY=(.+)$/m);
35
+ const PRIVATE_KEY = match ? match[1].trim() : '';
36
+
37
+ if (!PRIVATE_KEY) {
38
+ console.error('PRIVATE_KEY not found');
39
+ process.exit(1);
40
+ }
41
+
42
+ const EXECUTE = process.argv.includes('--execute');
43
+
44
+ // NVIDIA market token IDs (from report)
45
+ const NVIDIA_MARKET = {
46
+ conditionId: '0x0b16eb7741855ca3d4c8293089b5a9bdf7e59a6cf7f86ab87eb59c60ca01bbcf',
47
+ yesTokenId: '94850533403292240972948844256810904078895883844462287088135166537739765648754',
48
+ noTokenId: '69263280792958981516606123639467754139758192236863611059536531765186180114584',
49
+ question: 'NVIDIA largest company by market cap on Dec 31',
50
+ };
51
+
52
+ // Conditional Tokens contract ABI
53
+ const CTF_ABI = [
54
+ 'function balanceOf(address account, uint256 positionId) view returns (uint256)',
55
+ ];
56
+ const CTF_ADDRESS = '0x4D97DCd97eC945f40cF65F87097ACe5EA0476045';
57
+
58
+ async function main() {
59
+ console.log('╔════════════════════════════════════════════════════════════════╗');
60
+ console.log('║ SELL NVIDIA POSITIONS ║');
61
+ console.log('╠════════════════════════════════════════════════════════════════╣');
62
+ console.log(`║ Mode: ${EXECUTE ? '🔥 EXECUTE' : '👀 DRY RUN'} ║`);
63
+ console.log('╚════════════════════════════════════════════════════════════════╝');
64
+ console.log('');
65
+
66
+ const rateLimiter = new RateLimiter();
67
+ const cache = new Cache();
68
+ const ctf = new CTFClient({ privateKey: PRIVATE_KEY });
69
+ const trading = new TradingClient(rateLimiter, { privateKey: PRIVATE_KEY });
70
+ const clobApi = new ClobApiClient(rateLimiter, cache);
71
+ const dataApi = new DataApiClient(rateLimiter, cache);
72
+
73
+ const address = ctf.getAddress();
74
+ console.log(`Wallet: ${address}`);
75
+ console.log(`Market: ${NVIDIA_MARKET.question}`);
76
+ console.log('');
77
+
78
+ // Create provider and CTF contract for balance queries
79
+ const provider = new ethers.providers.JsonRpcProvider('https://polygon-rpc.com');
80
+ const ctfContract = new ethers.Contract(CTF_ADDRESS, CTF_ABI, provider);
81
+
82
+ // Get current balances
83
+ const usdcBalance = await ctf.getUsdcBalance();
84
+ const yesBalanceRaw = await ctfContract.balanceOf(address, NVIDIA_MARKET.yesTokenId);
85
+ const noBalanceRaw = await ctfContract.balanceOf(address, NVIDIA_MARKET.noTokenId);
86
+ const yesBalance = ethers.utils.formatUnits(yesBalanceRaw, 6);
87
+ const noBalance = ethers.utils.formatUnits(noBalanceRaw, 6);
88
+
89
+ console.log('─── Current Balances ───');
90
+ console.log(`USDC: $${parseFloat(usdcBalance).toFixed(2)}`);
91
+ console.log(`YES: ${parseFloat(yesBalance).toFixed(2)} tokens`);
92
+ console.log(`NO: ${parseFloat(noBalance).toFixed(2)} tokens`);
93
+ console.log('');
94
+
95
+ // Get market prices
96
+ const orderbook = await clobApi.getOrderbook(NVIDIA_MARKET.yesTokenId);
97
+ const yesBid = orderbook?.bids?.[0]?.price ?? 0;
98
+ const yesAsk = orderbook?.asks?.[0]?.price ?? 1;
99
+
100
+ const noOrderbook = await clobApi.getOrderbook(NVIDIA_MARKET.noTokenId);
101
+ const noBid = noOrderbook?.bids?.[0]?.price ?? 0;
102
+ const noAsk = noOrderbook?.asks?.[0]?.price ?? 1;
103
+
104
+ console.log('─── Market Prices ───');
105
+ console.log(`YES: bid=$${yesBid.toFixed(3)} ask=$${yesAsk.toFixed(3)}`);
106
+ console.log(`NO: bid=$${noBid.toFixed(3)} ask=$${noAsk.toFixed(3)}`);
107
+ console.log('');
108
+
109
+ const yesValue = parseFloat(yesBalance) * yesBid;
110
+ const noValue = parseFloat(noBalance) * noBid;
111
+
112
+ console.log('─── Expected Value (at bid prices) ───');
113
+ console.log(`YES: ${parseFloat(yesBalance).toFixed(2)} × $${yesBid.toFixed(3)} = $${yesValue.toFixed(2)}`);
114
+ console.log(`NO: ${parseFloat(noBalance).toFixed(2)} × $${noBid.toFixed(3)} = $${noValue.toFixed(2)}`);
115
+ console.log(`Total: $${(yesValue + noValue).toFixed(2)}`);
116
+ console.log('');
117
+
118
+ if (!EXECUTE) {
119
+ console.log('💡 Run with --execute to sell positions:');
120
+ console.log(' npx tsx scripts/sell-nvidia-positions.ts --execute');
121
+ return;
122
+ }
123
+
124
+ // Initialize trading client
125
+ console.log('─── Executing Trades ───');
126
+ console.log('Initializing trading client...');
127
+ await trading.initialize();
128
+ console.log('Trading client ready.');
129
+ console.log('');
130
+
131
+ // Sell YES tokens if we have any
132
+ if (parseFloat(yesBalance) >= 1) {
133
+ console.log(`Selling ${parseFloat(yesBalance).toFixed(2)} YES tokens...`);
134
+ try {
135
+ const result = await trading.createMarketOrder({
136
+ tokenId: NVIDIA_MARKET.yesTokenId,
137
+ side: 'SELL',
138
+ amount: parseFloat(yesBalance),
139
+ orderType: 'FAK', // Fill-and-Kill allows partial fills
140
+ });
141
+
142
+ if (result.success) {
143
+ console.log(`✅ YES sold! Order ID: ${result.orderId || 'N/A'}`);
144
+ if (result.transactionHashes?.length) {
145
+ console.log(` TX: ${result.transactionHashes[0]}`);
146
+ }
147
+ } else {
148
+ console.log(`⚠️ YES order may not have filled: ${result.errorMsg || 'Unknown'}`);
149
+ }
150
+ } catch (error: any) {
151
+ console.log(`❌ YES sell failed: ${error.message}`);
152
+ }
153
+ } else {
154
+ console.log('No YES tokens to sell (balance < 1)');
155
+ }
156
+
157
+ console.log('');
158
+
159
+ // Sell NO tokens if we have any
160
+ if (parseFloat(noBalance) >= 1) {
161
+ console.log(`Selling ${parseFloat(noBalance).toFixed(2)} NO tokens...`);
162
+ try {
163
+ const result = await trading.createMarketOrder({
164
+ tokenId: NVIDIA_MARKET.noTokenId,
165
+ side: 'SELL',
166
+ amount: parseFloat(noBalance),
167
+ orderType: 'FAK', // Fill-and-Kill allows partial fills
168
+ });
169
+
170
+ if (result.success) {
171
+ console.log(`✅ NO sold! Order ID: ${result.orderId || 'N/A'}`);
172
+ if (result.transactionHashes?.length) {
173
+ console.log(` TX: ${result.transactionHashes[0]}`);
174
+ }
175
+ } else {
176
+ console.log(`⚠️ NO order may not have filled: ${result.errorMsg || 'Unknown'}`);
177
+ }
178
+ } catch (error: any) {
179
+ console.log(`❌ NO sell failed: ${error.message}`);
180
+ }
181
+ } else {
182
+ console.log('No NO tokens to sell (balance < 1)');
183
+ }
184
+
185
+ console.log('');
186
+
187
+ // Wait a bit for balances to update
188
+ await new Promise((r) => setTimeout(r, 2000));
189
+
190
+ // Get final balances
191
+ const finalUsdc = await ctf.getUsdcBalance();
192
+ const finalYesRaw = await ctfContract.balanceOf(address, NVIDIA_MARKET.yesTokenId);
193
+ const finalNoRaw = await ctfContract.balanceOf(address, NVIDIA_MARKET.noTokenId);
194
+ const finalYes = ethers.utils.formatUnits(finalYesRaw, 6);
195
+ const finalNo = ethers.utils.formatUnits(finalNoRaw, 6);
196
+
197
+ console.log('─── Final Balances ───');
198
+ console.log(`USDC: $${parseFloat(finalUsdc).toFixed(2)} (${parseFloat(finalUsdc) >= parseFloat(usdcBalance) ? '+' : ''}$${(parseFloat(finalUsdc) - parseFloat(usdcBalance)).toFixed(2)})`);
199
+ console.log(`YES: ${parseFloat(finalYes).toFixed(2)} tokens`);
200
+ console.log(`NO: ${parseFloat(finalNo).toFixed(2)} tokens`);
201
+ console.log('');
202
+
203
+ console.log('✅ Done!');
204
+ }
205
+
206
+ main().catch(console.error);
@@ -0,0 +1,172 @@
1
+ #!/usr/bin/env npx tsx
2
+ /**
3
+ * Test Order Script - 测试下单功能
4
+ *
5
+ * 测试 GTC 限价单 vs FOK 市价单的区别
6
+ *
7
+ * Usage:
8
+ * POLY_PRIVKEY=0x... npx tsx scripts/test-order.ts
9
+ */
10
+
11
+ import { TradingClient, RateLimiter, ClobApiClient } from '../src/index.js';
12
+ import * as fs from 'fs';
13
+ import * as path from 'path';
14
+ import { fileURLToPath } from 'url';
15
+
16
+ const __filename = fileURLToPath(import.meta.url);
17
+ const __dirname = path.dirname(__filename);
18
+
19
+ // Read private key from dashboard-api .env
20
+ const envPath = path.resolve(__dirname, '../../earning-engine/dashboard-api/.env');
21
+ const envContent = fs.readFileSync(envPath, 'utf8');
22
+ const match = envContent.match(/^PRIVATE_KEY=(.+)$/m);
23
+ const PRIVATE_KEY = process.env.POLY_PRIVKEY || (match ? match[1].trim() : '');
24
+
25
+ // 使用一个活跃的市场进行测试 - NVIDIA market cap
26
+ const TEST_MARKET = {
27
+ name: 'NVIDIA largest company by market cap on Dec 31',
28
+ conditionId: '0x0b16eb7741855ca3d4383fabb8b760c897c2165d603916497f484b87ba9826dc',
29
+ yesTokenId: '94850533403292240972948844256810904078895883844462287088135166537739765648754',
30
+ noTokenId: '69263280792958981516606123639467754139758192236863611059536531765186180114584',
31
+ };
32
+
33
+ const TEST_AMOUNT = 5; // 5 USDC 测试 (Polymarket 最小订单量是 5 份)
34
+
35
+ async function main() {
36
+ if (!PRIVATE_KEY) {
37
+ console.error('Error: Set POLY_PRIVKEY environment variable');
38
+ process.exit(1);
39
+ }
40
+
41
+ console.log('╔════════════════════════════════════════════════════════════════╗');
42
+ console.log('║ ORDER TYPE TEST - GTC vs FOK ║');
43
+ console.log('╚════════════════════════════════════════════════════════════════╝');
44
+ console.log('');
45
+ console.log(`Market: ${TEST_MARKET.name}`);
46
+ console.log(`Test Amount: $${TEST_AMOUNT} USDC`);
47
+ console.log('');
48
+
49
+ const rateLimiter = new RateLimiter();
50
+ const tradingClient = new TradingClient(rateLimiter, {
51
+ privateKey: PRIVATE_KEY,
52
+ chainId: 137,
53
+ });
54
+
55
+ await tradingClient.initialize();
56
+ console.log(`Wallet: ${tradingClient.getAddress()}`);
57
+
58
+ // 获取当前余额
59
+ const { balance, allowance } = await tradingClient.getBalanceAllowance('COLLATERAL');
60
+ console.log(`USDC Balance: ${(parseFloat(balance) / 1e6).toFixed(2)} USDC`);
61
+ console.log(`Allowance: ${allowance === 'unlimited' || parseFloat(allowance) / 1e6 > 1e12 ? 'Unlimited' : (parseFloat(allowance) / 1e6).toFixed(2)}`);
62
+ console.log('');
63
+
64
+ // 获取当前市场价格
65
+ const clobClient = new ClobApiClient(rateLimiter);
66
+ const orderbook = await clobClient.getOrderbook(TEST_MARKET.yesTokenId);
67
+ const bestBid = orderbook.bids[0]?.price || 0;
68
+ const bestAsk = orderbook.asks[0]?.price || 1;
69
+ console.log(`Current YES price: ${bestBid.toFixed(3)} / ${bestAsk.toFixed(3)}`);
70
+ console.log('');
71
+
72
+ // ═══════════════════════════════════════════════════════════════════════════
73
+ // TEST 1: GTC Limit Order (这是 Earning Engine 使用的方式)
74
+ // ═══════════════════════════════════════════════════════════════════════════
75
+ console.log('═══════════════════════════════════════════════════════════════');
76
+ console.log('TEST 1: GTC Limit Order (Earning Engine 方式)');
77
+ console.log('═══════════════════════════════════════════════════════════════');
78
+
79
+ // 尝试以低于市价的价格买入 (maker order)
80
+ const gtcBuyPrice = Math.max(0.01, bestBid - 0.05); // 低于最佳买价 5 cents
81
+ const gtcSize = Math.max(5, TEST_AMOUNT / gtcBuyPrice); // 最小 5 份
82
+
83
+ console.log(`Placing GTC BUY order: ${gtcSize.toFixed(2)} shares @ $${gtcBuyPrice.toFixed(3)}`);
84
+ console.log(`Expected cost: $${(gtcSize * gtcBuyPrice).toFixed(2)}`);
85
+
86
+ try {
87
+ const gtcResult = await tradingClient.createOrder({
88
+ tokenId: TEST_MARKET.yesTokenId,
89
+ side: 'BUY',
90
+ price: gtcBuyPrice,
91
+ size: gtcSize,
92
+ orderType: 'GTC',
93
+ });
94
+
95
+ if (gtcResult.success) {
96
+ console.log(`✅ GTC Order SUCCESS!`);
97
+ console.log(` Order ID: ${gtcResult.orderId}`);
98
+
99
+ // 立即取消订单
100
+ console.log(' Cancelling order...');
101
+ const cancelResult = await tradingClient.cancelOrder(gtcResult.orderId!);
102
+ console.log(` Cancel: ${cancelResult.success ? '✓' : '✗'}`);
103
+ } else {
104
+ console.log(`❌ GTC Order FAILED: ${gtcResult.errorMsg}`);
105
+ }
106
+ } catch (error: any) {
107
+ console.log(`❌ GTC Order ERROR: ${error.message}`);
108
+ }
109
+
110
+ console.log('');
111
+
112
+ // ═══════════════════════════════════════════════════════════════════════════
113
+ // TEST 2: FOK Market Order (这是套利脚本使用的方式)
114
+ // ═══════════════════════════════════════════════════════════════════════════
115
+ console.log('═══════════════════════════════════════════════════════════════');
116
+ console.log('TEST 2: FOK Market Order (套利脚本方式)');
117
+ console.log('═══════════════════════════════════════════════════════════════');
118
+
119
+ console.log(`Placing FOK BUY order: $${TEST_AMOUNT} USDC worth`);
120
+
121
+ try {
122
+ const fokResult = await tradingClient.createMarketOrder({
123
+ tokenId: TEST_MARKET.yesTokenId,
124
+ side: 'BUY',
125
+ amount: TEST_AMOUNT,
126
+ orderType: 'FOK',
127
+ });
128
+
129
+ if (fokResult.success) {
130
+ console.log(`✅ FOK Order SUCCESS!`);
131
+ console.log(` Order ID: ${fokResult.orderId}`);
132
+
133
+ // 等待一下让订单成交
134
+ await new Promise((r) => setTimeout(r, 2000));
135
+
136
+ // 检查持仓并卖出
137
+ console.log(' Selling back...');
138
+ const sellResult = await tradingClient.createMarketOrder({
139
+ tokenId: TEST_MARKET.yesTokenId,
140
+ side: 'SELL',
141
+ amount: TEST_AMOUNT * 0.95, // 卖出略少一点确保成功
142
+ orderType: 'FOK',
143
+ });
144
+ console.log(` Sell: ${sellResult.success ? '✓' : '✗'} ${sellResult.errorMsg || ''}`);
145
+ } else {
146
+ console.log(`❌ FOK Order FAILED: ${fokResult.errorMsg}`);
147
+ }
148
+ } catch (error: any) {
149
+ console.log(`❌ FOK Order ERROR: ${error.message}`);
150
+ }
151
+
152
+ console.log('');
153
+
154
+ // ═══════════════════════════════════════════════════════════════════════════
155
+ // SUMMARY
156
+ // ═══════════════════════════════════════════════════════════════════════════
157
+ console.log('═══════════════════════════════════════════════════════════════');
158
+ console.log('SUMMARY');
159
+ console.log('═══════════════════════════════════════════════════════════════');
160
+ console.log('');
161
+ console.log('如果 GTC 失败但 FOK 成功,说明:');
162
+ console.log(' - Polymarket 对 GTC 限价单有不同的余额要求');
163
+ console.log(' - 可能需要通过 Polymarket UI 存入资金');
164
+ console.log(' - 或者 Earning Engine 应该改用 FOK 市价单');
165
+ console.log('');
166
+ console.log('如果两个都失败,说明:');
167
+ console.log(' - 钱包配置可能有问题');
168
+ console.log(' - 需要检查 API Key 或签名');
169
+ console.log('');
170
+ }
171
+
172
+ main().catch(console.error);