@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.
- package/.env +0 -0
- package/README.md +803 -0
- package/dist/__tests__/clob-api.test.d.ts +5 -0
- package/dist/__tests__/clob-api.test.d.ts.map +1 -0
- package/dist/__tests__/clob-api.test.js +240 -0
- package/dist/__tests__/clob-api.test.js.map +1 -0
- package/dist/__tests__/integration/bridge-client.integration.test.d.ts +11 -0
- package/dist/__tests__/integration/bridge-client.integration.test.d.ts.map +1 -0
- package/dist/__tests__/integration/bridge-client.integration.test.js +260 -0
- package/dist/__tests__/integration/bridge-client.integration.test.js.map +1 -0
- package/dist/__tests__/integration/clob-api.integration.test.d.ts +13 -0
- package/dist/__tests__/integration/clob-api.integration.test.d.ts.map +1 -0
- package/dist/__tests__/integration/clob-api.integration.test.js +170 -0
- package/dist/__tests__/integration/clob-api.integration.test.js.map +1 -0
- package/dist/__tests__/integration/ctf-client.integration.test.d.ts +17 -0
- package/dist/__tests__/integration/ctf-client.integration.test.d.ts.map +1 -0
- package/dist/__tests__/integration/ctf-client.integration.test.js +234 -0
- package/dist/__tests__/integration/ctf-client.integration.test.js.map +1 -0
- package/dist/__tests__/integration/data-api.integration.test.d.ts +9 -0
- package/dist/__tests__/integration/data-api.integration.test.d.ts.map +1 -0
- package/dist/__tests__/integration/data-api.integration.test.js +161 -0
- package/dist/__tests__/integration/data-api.integration.test.js.map +1 -0
- package/dist/__tests__/integration/gamma-api.integration.test.d.ts +9 -0
- package/dist/__tests__/integration/gamma-api.integration.test.d.ts.map +1 -0
- package/dist/__tests__/integration/gamma-api.integration.test.js +170 -0
- package/dist/__tests__/integration/gamma-api.integration.test.js.map +1 -0
- package/dist/__tests__/test-utils.d.ts +92 -0
- package/dist/__tests__/test-utils.d.ts.map +1 -0
- package/dist/__tests__/test-utils.js +143 -0
- package/dist/__tests__/test-utils.js.map +1 -0
- package/dist/clients/bridge-client.d.ts +388 -0
- package/dist/clients/bridge-client.d.ts.map +1 -0
- package/dist/clients/bridge-client.js +587 -0
- package/dist/clients/bridge-client.js.map +1 -0
- package/dist/clients/clob-api.d.ts +318 -0
- package/dist/clients/clob-api.d.ts.map +1 -0
- package/dist/clients/clob-api.js +388 -0
- package/dist/clients/clob-api.js.map +1 -0
- package/dist/clients/ctf-client.d.ts +473 -0
- package/dist/clients/ctf-client.d.ts.map +1 -0
- package/dist/clients/ctf-client.js +915 -0
- package/dist/clients/ctf-client.js.map +1 -0
- package/dist/clients/data-api.d.ts +134 -0
- package/dist/clients/data-api.d.ts.map +1 -0
- package/dist/clients/data-api.js +265 -0
- package/dist/clients/data-api.js.map +1 -0
- package/dist/clients/gamma-api.d.ts +401 -0
- package/dist/clients/gamma-api.d.ts.map +1 -0
- package/dist/clients/gamma-api.js +352 -0
- package/dist/clients/gamma-api.js.map +1 -0
- package/dist/clients/trading-client.d.ts +252 -0
- package/dist/clients/trading-client.d.ts.map +1 -0
- package/dist/clients/trading-client.js +543 -0
- package/dist/clients/trading-client.js.map +1 -0
- package/dist/clients/websocket-manager.d.ts +100 -0
- package/dist/clients/websocket-manager.d.ts.map +1 -0
- package/dist/clients/websocket-manager.js +193 -0
- package/dist/clients/websocket-manager.js.map +1 -0
- package/dist/core/cache-adapter-bridge.d.ts +36 -0
- package/dist/core/cache-adapter-bridge.d.ts.map +1 -0
- package/dist/core/cache-adapter-bridge.js +81 -0
- package/dist/core/cache-adapter-bridge.js.map +1 -0
- package/dist/core/cache.d.ts +40 -0
- package/dist/core/cache.d.ts.map +1 -0
- package/dist/core/cache.js +71 -0
- package/dist/core/cache.js.map +1 -0
- package/dist/core/errors.d.ts +38 -0
- package/dist/core/errors.d.ts.map +1 -0
- package/dist/core/errors.js +84 -0
- package/dist/core/errors.js.map +1 -0
- package/dist/core/rate-limiter.d.ts +31 -0
- package/dist/core/rate-limiter.d.ts.map +1 -0
- package/dist/core/rate-limiter.js +70 -0
- package/dist/core/rate-limiter.js.map +1 -0
- package/dist/core/types.d.ts +314 -0
- package/dist/core/types.d.ts.map +1 -0
- package/dist/core/types.js +19 -0
- package/dist/core/types.js.map +1 -0
- package/dist/core/unified-cache.d.ts +63 -0
- package/dist/core/unified-cache.d.ts.map +1 -0
- package/dist/core/unified-cache.js +114 -0
- package/dist/core/unified-cache.js.map +1 -0
- package/dist/index.d.ts +94 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +258 -0
- package/dist/index.js.map +1 -0
- package/dist/mcp/errors.d.ts +33 -0
- package/dist/mcp/errors.d.ts.map +1 -0
- package/dist/mcp/errors.js +86 -0
- package/dist/mcp/errors.js.map +1 -0
- package/dist/mcp/index.d.ts +62 -0
- package/dist/mcp/index.d.ts.map +1 -0
- package/dist/mcp/index.js +173 -0
- package/dist/mcp/index.js.map +1 -0
- package/dist/mcp/server.d.ts +17 -0
- package/dist/mcp/server.d.ts.map +1 -0
- package/dist/mcp/server.js +155 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/tools/guide.d.ts +12 -0
- package/dist/mcp/tools/guide.d.ts.map +1 -0
- package/dist/mcp/tools/guide.js +801 -0
- package/dist/mcp/tools/guide.js.map +1 -0
- package/dist/mcp/tools/index.d.ts +11 -0
- package/dist/mcp/tools/index.d.ts.map +1 -0
- package/dist/mcp/tools/index.js +27 -0
- package/dist/mcp/tools/index.js.map +1 -0
- package/dist/mcp/tools/market.d.ts +11 -0
- package/dist/mcp/tools/market.d.ts.map +1 -0
- package/dist/mcp/tools/market.js +314 -0
- package/dist/mcp/tools/market.js.map +1 -0
- package/dist/mcp/tools/order.d.ts +10 -0
- package/dist/mcp/tools/order.d.ts.map +1 -0
- package/dist/mcp/tools/order.js +258 -0
- package/dist/mcp/tools/order.js.map +1 -0
- package/dist/mcp/tools/trade.d.ts +38 -0
- package/dist/mcp/tools/trade.d.ts.map +1 -0
- package/dist/mcp/tools/trade.js +314 -0
- package/dist/mcp/tools/trade.js.map +1 -0
- package/dist/mcp/tools/trader.d.ts +11 -0
- package/dist/mcp/tools/trader.d.ts.map +1 -0
- package/dist/mcp/tools/trader.js +277 -0
- package/dist/mcp/tools/trader.js.map +1 -0
- package/dist/mcp/tools/wallet.d.ts +274 -0
- package/dist/mcp/tools/wallet.d.ts.map +1 -0
- package/dist/mcp/tools/wallet.js +579 -0
- package/dist/mcp/tools/wallet.js.map +1 -0
- package/dist/mcp/types.d.ts +413 -0
- package/dist/mcp/types.d.ts.map +1 -0
- package/dist/mcp/types.js +5 -0
- package/dist/mcp/types.js.map +1 -0
- package/dist/services/authorization-service.d.ts +97 -0
- package/dist/services/authorization-service.d.ts.map +1 -0
- package/dist/services/authorization-service.js +279 -0
- package/dist/services/authorization-service.js.map +1 -0
- package/dist/services/market-service.d.ts +108 -0
- package/dist/services/market-service.d.ts.map +1 -0
- package/dist/services/market-service.js +458 -0
- package/dist/services/market-service.js.map +1 -0
- package/dist/services/realtime-service.d.ts +82 -0
- package/dist/services/realtime-service.d.ts.map +1 -0
- package/dist/services/realtime-service.js +150 -0
- package/dist/services/realtime-service.js.map +1 -0
- package/dist/services/swap-service.d.ts +217 -0
- package/dist/services/swap-service.d.ts.map +1 -0
- package/dist/services/swap-service.js +695 -0
- package/dist/services/swap-service.js.map +1 -0
- package/dist/services/wallet-service.d.ts +94 -0
- package/dist/services/wallet-service.d.ts.map +1 -0
- package/dist/services/wallet-service.js +173 -0
- package/dist/services/wallet-service.js.map +1 -0
- package/dist/utils/price-utils.d.ts +153 -0
- package/dist/utils/price-utils.d.ts.map +1 -0
- package/dist/utils/price-utils.js +236 -0
- package/dist/utils/price-utils.js.map +1 -0
- package/docs/00-design.md +760 -0
- package/docs/01-mcp.md +2041 -0
- package/docs/02-API.md +1148 -0
- package/docs/e2e/01-trader-tools.md +159 -0
- package/docs/e2e/02-market-tools.md +180 -0
- package/docs/e2e/03-order-tools.md +166 -0
- package/docs/e2e/04-wallet-tools.md +224 -0
- package/docs/e2e/05-trading-tools.md +327 -0
- package/docs/e2e/06-integration-scenarios.md +481 -0
- package/docs/e2e/coordinator.md +376 -0
- package/examples/01-basic-usage.ts +68 -0
- package/examples/02-smart-money.ts +95 -0
- package/examples/03-market-analysis.ts +108 -0
- package/examples/04-kline-aggregation.ts +158 -0
- package/examples/05-follow-wallet-strategy.ts +156 -0
- package/examples/06-services-demo.ts +124 -0
- package/examples/07-realtime-websocket.ts +117 -0
- package/examples/08-trading-orders.ts +278 -0
- package/examples/09-rewards-tracking.ts +187 -0
- package/examples/10-ctf-operations.ts +336 -0
- package/examples/11-live-arbitrage-scan.ts +221 -0
- package/examples/12-trending-arb-monitor.ts +406 -0
- package/examples/README.md +179 -0
- package/package.json +62 -0
- package/scripts/README.md +163 -0
- package/scripts/approvals/approve-erc1155.ts +129 -0
- package/scripts/approvals/approve-neg-risk-erc1155.ts +149 -0
- package/scripts/approvals/approve-neg-risk.ts +102 -0
- package/scripts/approvals/check-all-allowances.ts +150 -0
- package/scripts/approvals/check-allowance.ts +129 -0
- package/scripts/approvals/check-ctf-approval.ts +158 -0
- package/scripts/datas/001-report.md +486 -0
- package/scripts/datas/clone-modal-screenshot.png +0 -0
- package/scripts/deposit/deposit-native-usdc.ts +179 -0
- package/scripts/deposit/deposit-usdc.ts +155 -0
- package/scripts/deposit/swap-usdc-to-usdce.ts +375 -0
- package/scripts/research/research-markets.ts +166 -0
- package/scripts/trading/check-orders.ts +50 -0
- package/scripts/trading/sell-nvidia-positions.ts +206 -0
- package/scripts/trading/test-order.ts +172 -0
- package/scripts/truth.md +440 -0
- package/scripts/verify/test-approve-trading.ts +98 -0
- package/scripts/verify/test-provider-fix.ts +43 -0
- package/scripts/verify/test-search-mcp.ts +113 -0
- package/scripts/verify/verify-all-apis.ts +160 -0
- package/scripts/wallet/check-wallet-balances.ts +75 -0
- package/scripts/wallet/test-wallet-operations.ts +191 -0
- package/scripts/wallet/verify-wallet-tools.ts +124 -0
- package/src/__tests__/clob-api.test.ts +301 -0
- package/src/__tests__/integration/bridge-client.integration.test.ts +314 -0
- package/src/__tests__/integration/clob-api.integration.test.ts +218 -0
- package/src/__tests__/integration/ctf-client.integration.test.ts +331 -0
- package/src/__tests__/integration/data-api.integration.test.ts +194 -0
- package/src/__tests__/integration/gamma-api.integration.test.ts +206 -0
- package/src/__tests__/test-utils.ts +170 -0
- package/src/clients/bridge-client.ts +841 -0
- package/src/clients/clob-api.ts +629 -0
- package/src/clients/ctf-client.ts +1216 -0
- package/src/clients/data-api.ts +469 -0
- package/src/clients/gamma-api.ts +597 -0
- package/src/clients/trading-client.ts +749 -0
- package/src/clients/websocket-manager.ts +267 -0
- package/src/core/cache-adapter-bridge.ts +94 -0
- package/src/core/cache.ts +85 -0
- package/src/core/errors.ts +117 -0
- package/src/core/rate-limiter.ts +74 -0
- package/src/core/types.ts +360 -0
- package/src/core/unified-cache.ts +153 -0
- package/src/index.ts +455 -0
- package/src/mcp/README.md +380 -0
- package/src/mcp/errors.ts +124 -0
- package/src/mcp/index.ts +309 -0
- package/src/mcp/server.ts +183 -0
- package/src/mcp/tools/guide.ts +821 -0
- package/src/mcp/tools/index.ts +73 -0
- package/src/mcp/tools/market.ts +363 -0
- package/src/mcp/tools/order.ts +326 -0
- package/src/mcp/tools/trade.ts +417 -0
- package/src/mcp/tools/trader.ts +322 -0
- package/src/mcp/tools/wallet.ts +683 -0
- package/src/mcp/types.ts +472 -0
- package/src/services/authorization-service.ts +357 -0
- package/src/services/market-service.ts +544 -0
- package/src/services/realtime-service.ts +196 -0
- package/src/services/swap-service.ts +896 -0
- package/src/services/wallet-service.ts +259 -0
- package/src/utils/price-utils.ts +307 -0
- package/tsconfig.json +8 -0
- package/vitest.config.ts +19 -0
- package/vitest.integration.config.ts +18 -0
|
@@ -0,0 +1,388 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CLOB API Client for Polymarket
|
|
3
|
+
*
|
|
4
|
+
* The Central Limit Order Book (CLOB) API provides access to Polymarket's
|
|
5
|
+
* trading infrastructure. This client handles market information retrieval
|
|
6
|
+
* and orderbook data.
|
|
7
|
+
*
|
|
8
|
+
* @remarks
|
|
9
|
+
* - Base URL: https://clob.polymarket.com
|
|
10
|
+
* - Rate limits are automatically handled by the RateLimiter
|
|
11
|
+
* - Market data is cached to reduce API calls
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```typescript
|
|
15
|
+
* import { ClobApiClient, RateLimiter, Cache } from '@prediction-router/poly-sdk';
|
|
16
|
+
*
|
|
17
|
+
* const client = new ClobApiClient(new RateLimiter(), new Cache());
|
|
18
|
+
*
|
|
19
|
+
* // Get market info
|
|
20
|
+
* const market = await client.getMarket('0x123...');
|
|
21
|
+
* console.log(market.question);
|
|
22
|
+
*
|
|
23
|
+
* // Get orderbook
|
|
24
|
+
* const orderbook = await client.getOrderbook(market.tokens[0].tokenId);
|
|
25
|
+
* console.log('Best bid:', orderbook.bids[0]);
|
|
26
|
+
* ```
|
|
27
|
+
*
|
|
28
|
+
* @see {@link https://docs.polymarket.com/#clob-api CLOB API Documentation}
|
|
29
|
+
*
|
|
30
|
+
* @module clients/clob-api
|
|
31
|
+
*/
|
|
32
|
+
import { ApiType } from '../core/rate-limiter.js';
|
|
33
|
+
import { CACHE_TTL } from '../core/unified-cache.js';
|
|
34
|
+
import { PolymarketError, ErrorCode } from '../core/errors.js';
|
|
35
|
+
/** CLOB API base URL */
|
|
36
|
+
const CLOB_API_BASE = 'https://clob.polymarket.com';
|
|
37
|
+
// ===== Client =====
|
|
38
|
+
/**
|
|
39
|
+
* CLOB API client for interacting with Polymarket's orderbook
|
|
40
|
+
*
|
|
41
|
+
* @remarks
|
|
42
|
+
* This client provides read-only access to market data and orderbooks.
|
|
43
|
+
* For trading operations, use {@link TradingClient} instead.
|
|
44
|
+
*
|
|
45
|
+
* @example
|
|
46
|
+
* ```typescript
|
|
47
|
+
* const client = new ClobApiClient(rateLimiter, cache);
|
|
48
|
+
*
|
|
49
|
+
* // Get market details
|
|
50
|
+
* const market = await client.getMarket('0x123...');
|
|
51
|
+
*
|
|
52
|
+
* // Get processed orderbook with analytics
|
|
53
|
+
* const processed = await client.getProcessedOrderbook('0x123...');
|
|
54
|
+
* console.log('Long arb profit:', processed.summary.longArbProfit);
|
|
55
|
+
* ```
|
|
56
|
+
*/
|
|
57
|
+
export class ClobApiClient {
|
|
58
|
+
rateLimiter;
|
|
59
|
+
cache;
|
|
60
|
+
config;
|
|
61
|
+
/**
|
|
62
|
+
* Creates a new CLOB API client
|
|
63
|
+
*
|
|
64
|
+
* @param rateLimiter - Rate limiter instance for API throttling
|
|
65
|
+
* @param cache - Cache instance for storing market data (supports both legacy Cache and CacheAdapter)
|
|
66
|
+
* @param config - Optional configuration for trading capabilities
|
|
67
|
+
* @param config.chainId - Polygon chain ID (137 for mainnet, 80002 for Amoy testnet)
|
|
68
|
+
* @param config.signer - Ethers signer for authenticated requests
|
|
69
|
+
* @param config.creds - API credentials for L2 authentication
|
|
70
|
+
*/
|
|
71
|
+
constructor(rateLimiter, cache, config) {
|
|
72
|
+
this.rateLimiter = rateLimiter;
|
|
73
|
+
this.cache = cache;
|
|
74
|
+
this.config = config;
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Get the signer if configured
|
|
78
|
+
* @returns The signer or undefined if not configured
|
|
79
|
+
*/
|
|
80
|
+
get signer() {
|
|
81
|
+
return this.config?.signer;
|
|
82
|
+
}
|
|
83
|
+
// ===== Market Info =====
|
|
84
|
+
/**
|
|
85
|
+
* Get market information by condition ID
|
|
86
|
+
*
|
|
87
|
+
* @param conditionId - The unique condition identifier for the market
|
|
88
|
+
* @returns Market information including tokens and status
|
|
89
|
+
*
|
|
90
|
+
* @throws {@link PolymarketError} If the market is not found or API fails
|
|
91
|
+
*
|
|
92
|
+
* @example
|
|
93
|
+
* ```typescript
|
|
94
|
+
* const market = await client.getMarket('0x82ace55...');
|
|
95
|
+
* console.log(market.question); // "Will BTC reach $100k?"
|
|
96
|
+
* console.log(market.tokens[0].tokenId); // YES token ID
|
|
97
|
+
* console.log(market.tokens[1].tokenId); // NO token ID
|
|
98
|
+
* ```
|
|
99
|
+
*/
|
|
100
|
+
async getMarket(conditionId) {
|
|
101
|
+
const cacheKey = `clob:market:${conditionId}`;
|
|
102
|
+
return this.cache.getOrSet(cacheKey, CACHE_TTL.MARKET_INFO, async () => {
|
|
103
|
+
return this.rateLimiter.execute(ApiType.CLOB_API, async () => {
|
|
104
|
+
const response = await fetch(`${CLOB_API_BASE}/markets/${conditionId}`);
|
|
105
|
+
if (!response.ok)
|
|
106
|
+
throw PolymarketError.fromHttpError(response.status, await response.json().catch(() => null));
|
|
107
|
+
const data = (await response.json());
|
|
108
|
+
return this.normalizeMarket(data);
|
|
109
|
+
});
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
// ===== Orderbook =====
|
|
113
|
+
/**
|
|
114
|
+
* Get raw orderbook for a specific token
|
|
115
|
+
*
|
|
116
|
+
* @param tokenId - The ERC-1155 token ID (either YES or NO token)
|
|
117
|
+
* @returns Orderbook with sorted bids and asks
|
|
118
|
+
*
|
|
119
|
+
* @remarks
|
|
120
|
+
* - Bids are sorted descending (highest bid first)
|
|
121
|
+
* - Asks are sorted ascending (lowest ask first)
|
|
122
|
+
* - This returns the raw orderbook for ONE outcome token
|
|
123
|
+
* - For complete market analysis, use {@link getProcessedOrderbook}
|
|
124
|
+
*
|
|
125
|
+
* @throws {@link PolymarketError} If the token is not found or API fails
|
|
126
|
+
*
|
|
127
|
+
* @example
|
|
128
|
+
* ```typescript
|
|
129
|
+
* const orderbook = await client.getOrderbook('21742633...');
|
|
130
|
+
*
|
|
131
|
+
* console.log('Best bid:', orderbook.bids[0]?.price); // e.g., 0.55
|
|
132
|
+
* console.log('Best ask:', orderbook.asks[0]?.price); // e.g., 0.57
|
|
133
|
+
* console.log('Spread:', orderbook.asks[0]?.price - orderbook.bids[0]?.price);
|
|
134
|
+
* ```
|
|
135
|
+
*/
|
|
136
|
+
async getOrderbook(tokenId) {
|
|
137
|
+
return this.rateLimiter.execute(ApiType.CLOB_API, async () => {
|
|
138
|
+
const response = await fetch(`${CLOB_API_BASE}/book?token_id=${tokenId}`);
|
|
139
|
+
if (!response.ok)
|
|
140
|
+
throw PolymarketError.fromHttpError(response.status, await response.json().catch(() => null));
|
|
141
|
+
const data = (await response.json());
|
|
142
|
+
// Sort bids descending (highest bid first)
|
|
143
|
+
// Sort asks ascending (lowest ask first)
|
|
144
|
+
const bids = (data.bids || [])
|
|
145
|
+
.map((l) => ({
|
|
146
|
+
price: Number(l.price),
|
|
147
|
+
size: Number(l.size),
|
|
148
|
+
}))
|
|
149
|
+
.sort((a, b) => b.price - a.price);
|
|
150
|
+
const asks = (data.asks || [])
|
|
151
|
+
.map((l) => ({
|
|
152
|
+
price: Number(l.price),
|
|
153
|
+
size: Number(l.size),
|
|
154
|
+
}))
|
|
155
|
+
.sort((a, b) => a.price - b.price);
|
|
156
|
+
return {
|
|
157
|
+
bids,
|
|
158
|
+
asks,
|
|
159
|
+
timestamp: data.timestamp ? Number(data.timestamp) : Date.now(),
|
|
160
|
+
market: data.market,
|
|
161
|
+
assetId: data.asset_id,
|
|
162
|
+
hash: data.hash,
|
|
163
|
+
minOrderSize: data.min_order_size,
|
|
164
|
+
tickSize: data.tick_size,
|
|
165
|
+
negRisk: data.neg_risk,
|
|
166
|
+
};
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Get processed orderbook with complete market analysis
|
|
171
|
+
*
|
|
172
|
+
* @param conditionId - The unique condition identifier for the market
|
|
173
|
+
* @returns Processed orderbook with both YES/NO books and arbitrage analysis
|
|
174
|
+
*
|
|
175
|
+
* @remarks
|
|
176
|
+
* This method fetches both YES and NO orderbooks and calculates:
|
|
177
|
+
* - Effective prices (accounting for order book mirroring)
|
|
178
|
+
* - Arbitrage opportunities (long and short)
|
|
179
|
+
* - Depth and liquidity metrics
|
|
180
|
+
*
|
|
181
|
+
* **Important**: Polymarket orderbooks have a mirroring property:
|
|
182
|
+
* - Buying YES @ P = Selling NO @ (1-P)
|
|
183
|
+
* - The same order appears in both books
|
|
184
|
+
*
|
|
185
|
+
* Therefore, correct arbitrage calculation must use "effective prices":
|
|
186
|
+
* - effectiveBuyYes = min(YES.ask, 1 - NO.bid)
|
|
187
|
+
* - effectiveBuyNo = min(NO.ask, 1 - YES.bid)
|
|
188
|
+
*
|
|
189
|
+
* @throws {@link PolymarketError} If market not found or missing tokens
|
|
190
|
+
*
|
|
191
|
+
* @example
|
|
192
|
+
* ```typescript
|
|
193
|
+
* const processed = await client.getProcessedOrderbook('0x82ace55...');
|
|
194
|
+
*
|
|
195
|
+
* // Check for arbitrage
|
|
196
|
+
* if (processed.summary.longArbProfit > 0.003) {
|
|
197
|
+
* console.log('Long arb opportunity!');
|
|
198
|
+
* console.log('Buy YES @', processed.summary.effectivePrices.effectiveBuyYes);
|
|
199
|
+
* console.log('Buy NO @', processed.summary.effectivePrices.effectiveBuyNo);
|
|
200
|
+
* console.log('Profit:', processed.summary.longArbProfit * 100, '%');
|
|
201
|
+
* }
|
|
202
|
+
* ```
|
|
203
|
+
*/
|
|
204
|
+
async getProcessedOrderbook(conditionId) {
|
|
205
|
+
const market = await this.getMarket(conditionId);
|
|
206
|
+
const yesToken = market.tokens.find((t) => t.outcome === 'Yes');
|
|
207
|
+
const noToken = market.tokens.find((t) => t.outcome === 'No');
|
|
208
|
+
if (!yesToken || !noToken) {
|
|
209
|
+
throw new PolymarketError(ErrorCode.INVALID_RESPONSE, 'Missing tokens in market');
|
|
210
|
+
}
|
|
211
|
+
const [yesBook, noBook] = await Promise.all([
|
|
212
|
+
this.getOrderbook(yesToken.tokenId),
|
|
213
|
+
this.getOrderbook(noToken.tokenId),
|
|
214
|
+
]);
|
|
215
|
+
return this.processOrderbooks(yesBook, noBook, yesToken.tokenId, noToken.tokenId);
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Process orderbooks and calculate analytics
|
|
219
|
+
*
|
|
220
|
+
* 关键概念:Polymarket 订单簿的镜像特性
|
|
221
|
+
*
|
|
222
|
+
* 买 YES @ P = 卖 NO @ (1-P)
|
|
223
|
+
* 因此同一订单会在 YES 和 NO 订单簿中同时出现
|
|
224
|
+
*
|
|
225
|
+
* 正确的套利计算必须使用"有效价格":
|
|
226
|
+
* - effectiveBuyYes = min(YES.ask, 1 - NO.bid)
|
|
227
|
+
* - effectiveBuyNo = min(NO.ask, 1 - YES.bid)
|
|
228
|
+
* - effectiveSellYes = max(YES.bid, 1 - NO.ask)
|
|
229
|
+
* - effectiveSellNo = max(NO.bid, 1 - YES.ask)
|
|
230
|
+
*
|
|
231
|
+
* 详细文档见: docs/01-polymarket-orderbook-arbitrage.md
|
|
232
|
+
*/
|
|
233
|
+
processOrderbooks(yesBook, noBook, yesTokenId, noTokenId) {
|
|
234
|
+
const yesBestBid = yesBook.bids[0]?.price || 0;
|
|
235
|
+
const yesBestAsk = yesBook.asks[0]?.price || 1;
|
|
236
|
+
const noBestBid = noBook.bids[0]?.price || 0;
|
|
237
|
+
const noBestAsk = noBook.asks[0]?.price || 1;
|
|
238
|
+
const yesBidDepth = yesBook.bids.reduce((sum, l) => sum + l.price * l.size, 0);
|
|
239
|
+
const yesAskDepth = yesBook.asks.reduce((sum, l) => sum + l.price * l.size, 0);
|
|
240
|
+
const noBidDepth = noBook.bids.reduce((sum, l) => sum + l.price * l.size, 0);
|
|
241
|
+
const noAskDepth = noBook.asks.reduce((sum, l) => sum + l.price * l.size, 0);
|
|
242
|
+
// 原始价格和(仅供参考,可能包含重复计算)
|
|
243
|
+
const askSum = yesBestAsk + noBestAsk;
|
|
244
|
+
const bidSum = yesBestBid + noBestBid;
|
|
245
|
+
// ===== 计算有效价格(考虑镜像订单)=====
|
|
246
|
+
// 这是正确的套利计算方式
|
|
247
|
+
const effectivePrices = {
|
|
248
|
+
// 买 YES: 直接买 YES.ask 或 通过卖 NO (成本 = 1 - NO.bid)
|
|
249
|
+
effectiveBuyYes: Math.min(yesBestAsk, 1 - noBestBid),
|
|
250
|
+
// 买 NO: 直接买 NO.ask 或 通过卖 YES (成本 = 1 - YES.bid)
|
|
251
|
+
effectiveBuyNo: Math.min(noBestAsk, 1 - yesBestBid),
|
|
252
|
+
// 卖 YES: 直接卖 YES.bid 或 通过买 NO (收入 = 1 - NO.ask)
|
|
253
|
+
effectiveSellYes: Math.max(yesBestBid, 1 - noBestAsk),
|
|
254
|
+
// 卖 NO: 直接卖 NO.bid 或 通过买 YES (收入 = 1 - YES.ask)
|
|
255
|
+
effectiveSellNo: Math.max(noBestBid, 1 - yesBestAsk),
|
|
256
|
+
};
|
|
257
|
+
// 有效套利成本/收入
|
|
258
|
+
const effectiveLongCost = effectivePrices.effectiveBuyYes + effectivePrices.effectiveBuyNo;
|
|
259
|
+
const effectiveShortRevenue = effectivePrices.effectiveSellYes + effectivePrices.effectiveSellNo;
|
|
260
|
+
// 套利利润(基于有效价格)
|
|
261
|
+
const longArbProfit = 1 - effectiveLongCost;
|
|
262
|
+
const shortArbProfit = effectiveShortRevenue - 1;
|
|
263
|
+
// YES spread(由于镜像,这也能反映整体市场效率)
|
|
264
|
+
const yesSpread = yesBestAsk - yesBestBid;
|
|
265
|
+
return {
|
|
266
|
+
yes: {
|
|
267
|
+
bid: yesBestBid,
|
|
268
|
+
ask: yesBestAsk,
|
|
269
|
+
bidSize: yesBook.bids[0]?.size || 0,
|
|
270
|
+
askSize: yesBook.asks[0]?.size || 0,
|
|
271
|
+
bidDepth: yesBidDepth,
|
|
272
|
+
askDepth: yesAskDepth,
|
|
273
|
+
spread: yesSpread,
|
|
274
|
+
tokenId: yesTokenId,
|
|
275
|
+
},
|
|
276
|
+
no: {
|
|
277
|
+
bid: noBestBid,
|
|
278
|
+
ask: noBestAsk,
|
|
279
|
+
bidSize: noBook.bids[0]?.size || 0,
|
|
280
|
+
askSize: noBook.asks[0]?.size || 0,
|
|
281
|
+
bidDepth: noBidDepth,
|
|
282
|
+
askDepth: noAskDepth,
|
|
283
|
+
spread: noBestAsk - noBestBid,
|
|
284
|
+
tokenId: noTokenId,
|
|
285
|
+
},
|
|
286
|
+
summary: {
|
|
287
|
+
// 原始价格和(仅供参考)
|
|
288
|
+
askSum,
|
|
289
|
+
bidSum,
|
|
290
|
+
// 有效价格
|
|
291
|
+
effectivePrices,
|
|
292
|
+
// 有效成本/收入
|
|
293
|
+
effectiveLongCost,
|
|
294
|
+
effectiveShortRevenue,
|
|
295
|
+
// 套利利润(基于有效价格,这才是正确的计算)
|
|
296
|
+
longArbProfit, // > 0 means long arbitrage opportunity
|
|
297
|
+
shortArbProfit, // > 0 means short arbitrage opportunity
|
|
298
|
+
// 其他指标
|
|
299
|
+
totalBidDepth: yesBidDepth + noBidDepth,
|
|
300
|
+
totalAskDepth: yesAskDepth + noAskDepth,
|
|
301
|
+
imbalanceRatio: (yesBidDepth + noBidDepth) / (yesAskDepth + noAskDepth + 0.001),
|
|
302
|
+
yesSpread,
|
|
303
|
+
},
|
|
304
|
+
};
|
|
305
|
+
}
|
|
306
|
+
// ===== Trading (requires authentication) =====
|
|
307
|
+
/**
|
|
308
|
+
* Check if this client has trading capabilities
|
|
309
|
+
*
|
|
310
|
+
* @returns True if a signer or API credentials are configured
|
|
311
|
+
*
|
|
312
|
+
* @remarks
|
|
313
|
+
* Trading requires either:
|
|
314
|
+
* - A signer (for L1 authentication)
|
|
315
|
+
* - API credentials (for L2 authentication)
|
|
316
|
+
*
|
|
317
|
+
* For actual trading, use the {@link TradingClient} instead.
|
|
318
|
+
*
|
|
319
|
+
* @example
|
|
320
|
+
* ```typescript
|
|
321
|
+
* if (client.hasTradingCapabilities()) {
|
|
322
|
+
* console.log('Client can execute trades');
|
|
323
|
+
* } else {
|
|
324
|
+
* console.log('Read-only mode - use TradingClient for trading');
|
|
325
|
+
* }
|
|
326
|
+
* ```
|
|
327
|
+
*/
|
|
328
|
+
hasTradingCapabilities() {
|
|
329
|
+
return !!(this.config?.signer || this.config?.creds);
|
|
330
|
+
}
|
|
331
|
+
// ===== Data Normalization =====
|
|
332
|
+
normalizeMarket(m) {
|
|
333
|
+
const tokens = m.tokens;
|
|
334
|
+
const rewards = m.rewards;
|
|
335
|
+
return {
|
|
336
|
+
// Core identifiers
|
|
337
|
+
conditionId: String(m.condition_id || ''),
|
|
338
|
+
questionId: m.question_id ? String(m.question_id) : undefined,
|
|
339
|
+
marketSlug: String(m.market_slug || ''),
|
|
340
|
+
// Market content
|
|
341
|
+
question: String(m.question || ''),
|
|
342
|
+
description: m.description ? String(m.description) : undefined,
|
|
343
|
+
image: m.image ? String(m.image) : undefined,
|
|
344
|
+
icon: m.icon ? String(m.icon) : undefined,
|
|
345
|
+
// Tokens
|
|
346
|
+
tokens: Array.isArray(tokens)
|
|
347
|
+
? tokens.map((t) => ({
|
|
348
|
+
tokenId: String(t.token_id || ''),
|
|
349
|
+
outcome: String(t.outcome || ''),
|
|
350
|
+
price: Number(t.price),
|
|
351
|
+
winner: t.winner !== undefined ? Boolean(t.winner) : undefined,
|
|
352
|
+
}))
|
|
353
|
+
: [],
|
|
354
|
+
tags: Array.isArray(m.tags) ? m.tags : undefined,
|
|
355
|
+
// Status flags
|
|
356
|
+
active: Boolean(m.active),
|
|
357
|
+
closed: Boolean(m.closed),
|
|
358
|
+
archived: m.archived !== undefined ? Boolean(m.archived) : undefined,
|
|
359
|
+
acceptingOrders: Boolean(m.accepting_orders),
|
|
360
|
+
acceptingOrderTimestamp: m.accepting_order_timestamp ? String(m.accepting_order_timestamp) : undefined,
|
|
361
|
+
enableOrderBook: m.enable_order_book !== undefined ? Boolean(m.enable_order_book) : undefined,
|
|
362
|
+
// Trading parameters
|
|
363
|
+
minimumOrderSize: m.minimum_order_size !== undefined ? Number(m.minimum_order_size) : undefined,
|
|
364
|
+
minimumTickSize: m.minimum_tick_size !== undefined ? Number(m.minimum_tick_size) : undefined,
|
|
365
|
+
makerBaseFee: m.maker_base_fee !== undefined ? Number(m.maker_base_fee) : undefined,
|
|
366
|
+
takerBaseFee: m.taker_base_fee !== undefined ? Number(m.taker_base_fee) : undefined,
|
|
367
|
+
// Timing
|
|
368
|
+
endDateIso: m.end_date_iso !== undefined ? (m.end_date_iso === null ? null : String(m.end_date_iso)) : undefined,
|
|
369
|
+
gameStartTime: m.game_start_time !== undefined ? (m.game_start_time === null ? null : String(m.game_start_time)) : undefined,
|
|
370
|
+
secondsDelay: m.seconds_delay !== undefined ? Number(m.seconds_delay) : undefined,
|
|
371
|
+
// Neg risk
|
|
372
|
+
negRisk: m.neg_risk !== undefined ? Boolean(m.neg_risk) : undefined,
|
|
373
|
+
negRiskMarketId: m.neg_risk_market_id ? String(m.neg_risk_market_id) : undefined,
|
|
374
|
+
negRiskRequestId: m.neg_risk_request_id ? String(m.neg_risk_request_id) : undefined,
|
|
375
|
+
// Rewards
|
|
376
|
+
rewards: rewards ? {
|
|
377
|
+
rates: rewards.rates,
|
|
378
|
+
minSize: rewards.min_size !== undefined ? Number(rewards.min_size) : undefined,
|
|
379
|
+
maxSpread: rewards.max_spread !== undefined ? Number(rewards.max_spread) : undefined,
|
|
380
|
+
} : undefined,
|
|
381
|
+
// Other
|
|
382
|
+
fpmm: m.fpmm ? String(m.fpmm) : undefined,
|
|
383
|
+
notificationsEnabled: m.notifications_enabled !== undefined ? Boolean(m.notifications_enabled) : undefined,
|
|
384
|
+
is5050Outcome: m.is_50_50_outcome !== undefined ? Boolean(m.is_50_50_outcome) : undefined,
|
|
385
|
+
};
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
//# sourceMappingURL=clob-api.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"clob-api.js","sourceRoot":"","sources":["../../src/clients/clob-api.ts"],"names":[],"mappings":"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"}
|