@baozi.bet/mcp-server 4.0.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 (60) hide show
  1. package/README.md +294 -0
  2. package/dist/__tests__/full-test.d.ts +1 -0
  3. package/dist/__tests__/full-test.js +291 -0
  4. package/dist/builders/affiliate-transaction.d.ts +41 -0
  5. package/dist/builders/affiliate-transaction.js +123 -0
  6. package/dist/builders/bet-transaction.d.ts +70 -0
  7. package/dist/builders/bet-transaction.js +323 -0
  8. package/dist/builders/claim-transaction.d.ts +57 -0
  9. package/dist/builders/claim-transaction.js +196 -0
  10. package/dist/builders/creator-transaction.d.ts +49 -0
  11. package/dist/builders/creator-transaction.js +177 -0
  12. package/dist/builders/dispute-transaction.d.ts +81 -0
  13. package/dist/builders/dispute-transaction.js +215 -0
  14. package/dist/builders/index.d.ts +14 -0
  15. package/dist/builders/index.js +15 -0
  16. package/dist/builders/market-creation-tx.d.ts +65 -0
  17. package/dist/builders/market-creation-tx.js +362 -0
  18. package/dist/builders/market-management-transaction.d.ts +85 -0
  19. package/dist/builders/market-management-transaction.js +239 -0
  20. package/dist/builders/race-transaction.d.ts +67 -0
  21. package/dist/builders/race-transaction.js +242 -0
  22. package/dist/builders/resolution-transaction.d.ts +108 -0
  23. package/dist/builders/resolution-transaction.js +250 -0
  24. package/dist/builders/whitelist-transaction.d.ts +72 -0
  25. package/dist/builders/whitelist-transaction.js +179 -0
  26. package/dist/config.d.ts +138 -0
  27. package/dist/config.js +307 -0
  28. package/dist/handlers/agent-network.d.ts +81 -0
  29. package/dist/handlers/agent-network.js +332 -0
  30. package/dist/handlers/claims.d.ts +47 -0
  31. package/dist/handlers/claims.js +218 -0
  32. package/dist/handlers/market-creation.d.ts +154 -0
  33. package/dist/handlers/market-creation.js +290 -0
  34. package/dist/handlers/markets.d.ts +41 -0
  35. package/dist/handlers/markets.js +319 -0
  36. package/dist/handlers/positions.d.ts +40 -0
  37. package/dist/handlers/positions.js +244 -0
  38. package/dist/handlers/quote.d.ts +33 -0
  39. package/dist/handlers/quote.js +144 -0
  40. package/dist/handlers/race-markets.d.ts +54 -0
  41. package/dist/handlers/race-markets.js +308 -0
  42. package/dist/handlers/resolution.d.ts +43 -0
  43. package/dist/handlers/resolution.js +194 -0
  44. package/dist/index.d.ts +2 -0
  45. package/dist/index.js +109 -0
  46. package/dist/resources.d.ts +13 -0
  47. package/dist/resources.js +336 -0
  48. package/dist/tools.d.ts +3109 -0
  49. package/dist/tools.js +1956 -0
  50. package/dist/validation/bet-rules.d.ts +82 -0
  51. package/dist/validation/bet-rules.js +276 -0
  52. package/dist/validation/creation-rules.d.ts +69 -0
  53. package/dist/validation/creation-rules.js +302 -0
  54. package/dist/validation/index.d.ts +6 -0
  55. package/dist/validation/index.js +7 -0
  56. package/dist/validation/market-rules.d.ts +60 -0
  57. package/dist/validation/market-rules.js +237 -0
  58. package/dist/validation/parimutuel-rules.d.ts +117 -0
  59. package/dist/validation/parimutuel-rules.js +270 -0
  60. package/package.json +52 -0
@@ -0,0 +1,308 @@
1
+ /**
2
+ * Race Markets Handler - Multi-Outcome Prediction Markets
3
+ */
4
+ import { Connection, PublicKey } from '@solana/web3.js';
5
+ import bs58 from 'bs58';
6
+ import { PROGRAM_ID, RPC_ENDPOINT, DISCRIMINATORS, MARKET_STATUS_NAMES, MARKET_LAYER_NAMES, lamportsToSol, } from '../config.js';
7
+ // =============================================================================
8
+ // RACE MARKET DECODER
9
+ // =============================================================================
10
+ /**
11
+ * Decode RaceMarket account data
12
+ *
13
+ * RaceMarket struct layout (from IDL v4.7.6):
14
+ * - discriminator (8)
15
+ * - market_id (u64, 8)
16
+ * - question (String: 4 + len)
17
+ * - closing_time (i64, 8)
18
+ * - resolution_time (i64, 8)
19
+ * - auto_stop_buffer (i64, 8)
20
+ * - outcome_count (u8, 1)
21
+ * - outcome_labels ([[u8; 32]; 10], 320 bytes FIXED)
22
+ * - outcome_pools ([u64; 10], 80 bytes FIXED)
23
+ * - total_pool (u64, 8)
24
+ * - snapshot_pools ([u64; 10], 80 bytes)
25
+ * - snapshot_total (u64, 8)
26
+ * - status (enum, 1)
27
+ * - winning_outcome (Option<u8>: 1 + 0/1)
28
+ * - currency_type (enum, 1)
29
+ * - platform_fee_collected (u64, 8)
30
+ * - creator_fee_collected (u64, 8)
31
+ * - total_claimed (u64, 8)
32
+ * - last_bet_time (i64, 8)
33
+ * - bump (u8, 1)
34
+ * - layer (enum, 1)
35
+ * - resolution_mode (enum, 1)
36
+ * - access_gate (enum, 1)
37
+ * - creator (Pubkey, 32)
38
+ * - platform_fee_bps_at_creation (u16, 2)
39
+ * - affiliate_fee_bps_at_creation (u16, 2)
40
+ * - betting_freeze_seconds_at_creation (i64, 8)
41
+ * - dust_swept (bool, 1)
42
+ * - reserved ([u8; 19], 19)
43
+ */
44
+ function decodeRaceMarket(data, pubkey) {
45
+ try {
46
+ // Minimum expected size for a RaceMarket account
47
+ // 8 (disc) + 8 (id) + 4 (qlen) + 8*3 (times) + 1 (count) + 320 (labels) + 80 (pools) + ...
48
+ if (data.length < 500) {
49
+ return null; // Account too small to be a RaceMarket
50
+ }
51
+ let offset = 8; // Skip discriminator
52
+ // market_id (u64)
53
+ const marketId = data.readBigUInt64LE(offset);
54
+ offset += 8;
55
+ // question (String: 4 bytes length + UTF-8 bytes)
56
+ const questionLen = data.readUInt32LE(offset);
57
+ offset += 4;
58
+ // Sanity check: question length should be reasonable (max 200 chars)
59
+ if (questionLen > 500 || questionLen + offset > data.length) {
60
+ return null; // Invalid question length - not a valid RaceMarket
61
+ }
62
+ const question = data.slice(offset, offset + questionLen).toString('utf8');
63
+ offset += questionLen;
64
+ // closing_time (i64)
65
+ const closingTime = data.readBigInt64LE(offset);
66
+ offset += 8;
67
+ // resolution_time (i64)
68
+ const resolutionTime = data.readBigInt64LE(offset);
69
+ offset += 8;
70
+ // auto_stop_buffer (i64)
71
+ offset += 8; // Skip for now
72
+ // outcome_count (u8)
73
+ const outcomeCount = data.readUInt8(offset);
74
+ offset += 1;
75
+ // outcome_labels: [[u8; 32]; 10] = 320 bytes FIXED
76
+ // Each label is 32 bytes, padded with zeros
77
+ const outcomeLabels = [];
78
+ for (let i = 0; i < 10; i++) {
79
+ const labelBytes = data.slice(offset, offset + 32);
80
+ // Find null terminator or end of string
81
+ let labelEnd = 32;
82
+ for (let j = 0; j < 32; j++) {
83
+ if (labelBytes[j] === 0) {
84
+ labelEnd = j;
85
+ break;
86
+ }
87
+ }
88
+ if (i < outcomeCount) {
89
+ outcomeLabels.push(labelBytes.slice(0, labelEnd).toString('utf8'));
90
+ }
91
+ offset += 32;
92
+ }
93
+ // outcome_pools: [u64; 10] = 80 bytes FIXED
94
+ const outcomePools = [];
95
+ for (let i = 0; i < 10; i++) {
96
+ const pool = data.readBigUInt64LE(offset);
97
+ if (i < outcomeCount) {
98
+ outcomePools.push(pool);
99
+ }
100
+ offset += 8;
101
+ }
102
+ // total_pool (u64)
103
+ const totalPoolLamports = data.readBigUInt64LE(offset);
104
+ offset += 8;
105
+ // snapshot_pools: [u64; 10] = 80 bytes - skip
106
+ offset += 80;
107
+ // snapshot_total (u64) - skip
108
+ offset += 8;
109
+ // status (enum, 1 byte)
110
+ const statusCode = data.readUInt8(offset);
111
+ offset += 1;
112
+ // winning_outcome (Option<u8>: 1 byte discriminant + optional 1 byte value)
113
+ const hasWinningOutcome = data.readUInt8(offset);
114
+ offset += 1;
115
+ let winningOutcomeIndex = null;
116
+ if (hasWinningOutcome === 1) {
117
+ winningOutcomeIndex = data.readUInt8(offset);
118
+ offset += 1;
119
+ }
120
+ // currency_type (enum, 1 byte)
121
+ offset += 1;
122
+ // platform_fee_collected (u64)
123
+ offset += 8;
124
+ // creator_fee_collected (u64)
125
+ offset += 8;
126
+ // total_claimed (u64)
127
+ offset += 8;
128
+ // last_bet_time (i64)
129
+ offset += 8;
130
+ // bump (u8)
131
+ offset += 1;
132
+ // layer (enum, 1 byte)
133
+ const layerCode = data.readUInt8(offset);
134
+ offset += 1;
135
+ // resolution_mode (enum, 1 byte)
136
+ offset += 1;
137
+ // access_gate (enum, 1 byte)
138
+ const accessGateCode = data.readUInt8(offset);
139
+ offset += 1;
140
+ // creator (Pubkey, 32 bytes)
141
+ const creator = new PublicKey(data.slice(offset, offset + 32));
142
+ offset += 32;
143
+ // oracle_host (Option<Pubkey>: 1 + 0/32)
144
+ const hasOracleHost = data.readUInt8(offset);
145
+ offset += 1;
146
+ if (hasOracleHost === 1) {
147
+ offset += 32;
148
+ }
149
+ // council: [Pubkey; 5] = 160 bytes - skip
150
+ offset += 160;
151
+ // council_size (u8)
152
+ offset += 1;
153
+ // council_votes: [u8; 10] = 10 bytes - skip
154
+ offset += 10;
155
+ // council_threshold (u8)
156
+ offset += 1;
157
+ // creator_fee_bps (u16)
158
+ offset += 2;
159
+ // creator_profile (Option<Pubkey>: 1 + 0/32)
160
+ const hasCreatorProfile = data.readUInt8(offset);
161
+ offset += 1;
162
+ if (hasCreatorProfile === 1) {
163
+ offset += 32;
164
+ }
165
+ // platform_fee_bps_at_creation (u16)
166
+ const platformFeeBps = data.readUInt16LE(offset);
167
+ offset += 2;
168
+ // Calculate derived fields
169
+ const totalPoolSol = lamportsToSol(totalPoolLamports);
170
+ const outcomes = outcomeLabels.map((label, i) => {
171
+ const poolSol = lamportsToSol(outcomePools[i] || 0n);
172
+ const percent = totalPoolSol > 0 ? (poolSol / totalPoolSol) * 100 : 100 / outcomeLabels.length;
173
+ return {
174
+ index: i,
175
+ label,
176
+ poolSol: round4(poolSol),
177
+ percent: round2(percent),
178
+ };
179
+ });
180
+ // Betting open check
181
+ const now = BigInt(Math.floor(Date.now() / 1000));
182
+ const freezeTime = closingTime - 300n; // 5 min freeze
183
+ const isBettingOpen = statusCode === 0 && now < freezeTime;
184
+ return {
185
+ publicKey: pubkey.toBase58(),
186
+ marketId: marketId.toString(),
187
+ question,
188
+ outcomes,
189
+ closingTime: new Date(Number(closingTime) * 1000).toISOString(),
190
+ resolutionTime: new Date(Number(resolutionTime) * 1000).toISOString(),
191
+ status: MARKET_STATUS_NAMES[statusCode] || 'Unknown',
192
+ statusCode,
193
+ winningOutcomeIndex,
194
+ totalPoolSol: round4(totalPoolSol),
195
+ layer: MARKET_LAYER_NAMES[layerCode] || 'Unknown',
196
+ layerCode,
197
+ accessGate: accessGateCode === 0 ? 'Public' : 'Whitelist',
198
+ creator: creator.toBase58(),
199
+ platformFeeBps,
200
+ isBettingOpen,
201
+ };
202
+ }
203
+ catch (err) {
204
+ console.error('Error decoding race market:', err);
205
+ return null;
206
+ }
207
+ }
208
+ // =============================================================================
209
+ // PUBLIC API
210
+ // =============================================================================
211
+ /**
212
+ * List all race markets
213
+ */
214
+ export async function listRaceMarkets(status) {
215
+ const connection = new Connection(RPC_ENDPOINT, 'confirmed');
216
+ const accounts = await connection.getProgramAccounts(PROGRAM_ID, {
217
+ filters: [
218
+ {
219
+ memcmp: {
220
+ offset: 0,
221
+ bytes: bs58.encode(DISCRIMINATORS.RACE_MARKET),
222
+ },
223
+ },
224
+ ],
225
+ });
226
+ const markets = [];
227
+ for (const { account, pubkey } of accounts) {
228
+ const market = decodeRaceMarket(account.data, pubkey);
229
+ if (market) {
230
+ if (!status || market.status.toLowerCase() === status.toLowerCase()) {
231
+ markets.push(market);
232
+ }
233
+ }
234
+ }
235
+ // Sort by closing time
236
+ markets.sort((a, b) => {
237
+ if (a.status === 'Active' && b.status !== 'Active')
238
+ return -1;
239
+ if (a.status !== 'Active' && b.status === 'Active')
240
+ return 1;
241
+ return new Date(a.closingTime).getTime() - new Date(b.closingTime).getTime();
242
+ });
243
+ return markets;
244
+ }
245
+ /**
246
+ * Get a specific race market
247
+ */
248
+ export async function getRaceMarket(publicKey) {
249
+ const connection = new Connection(RPC_ENDPOINT, 'confirmed');
250
+ try {
251
+ const pubkey = new PublicKey(publicKey);
252
+ const account = await connection.getAccountInfo(pubkey);
253
+ if (!account)
254
+ return null;
255
+ return decodeRaceMarket(account.data, pubkey);
256
+ }
257
+ catch {
258
+ return null;
259
+ }
260
+ }
261
+ /**
262
+ * Get race quote for a potential bet
263
+ */
264
+ export function getRaceQuote(market, outcomeIndex, betAmountSol) {
265
+ if (outcomeIndex < 0 || outcomeIndex >= market.outcomes.length) {
266
+ return {
267
+ valid: false,
268
+ error: `Invalid outcome index. Must be 0-${market.outcomes.length - 1}`,
269
+ outcomeLabel: '',
270
+ betAmountSol,
271
+ expectedPayoutSol: 0,
272
+ impliedOdds: 0,
273
+ newOutcomePercent: 0,
274
+ };
275
+ }
276
+ const outcome = market.outcomes[outcomeIndex];
277
+ const currentPool = outcome.poolSol;
278
+ const totalPool = market.totalPoolSol;
279
+ // New pools after bet
280
+ const newOutcomePool = currentPool + betAmountSol;
281
+ const newTotalPool = totalPool + betAmountSol;
282
+ // Share of outcome pool
283
+ const share = betAmountSol / newOutcomePool;
284
+ const grossPayout = share * newTotalPool;
285
+ const profit = grossPayout - betAmountSol;
286
+ const fee = profit > 0 ? (profit * market.platformFeeBps) / 10000 : 0;
287
+ const expectedPayout = grossPayout - fee;
288
+ // Implied odds
289
+ const impliedOdds = newOutcomePool / newTotalPool * 100;
290
+ return {
291
+ valid: true,
292
+ outcomeLabel: outcome.label,
293
+ betAmountSol,
294
+ expectedPayoutSol: round4(expectedPayout),
295
+ impliedOdds: round2(impliedOdds),
296
+ newOutcomePercent: round2((newOutcomePool / newTotalPool) * 100),
297
+ };
298
+ }
299
+ // =============================================================================
300
+ // HELPERS
301
+ // =============================================================================
302
+ function round2(n) {
303
+ return Math.round(n * 100) / 100;
304
+ }
305
+ function round4(n) {
306
+ return Math.round(n * 10000) / 10000;
307
+ }
308
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"race-markets.js","sourceRoot":"","sources":["../../src/handlers/race-markets.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Race Markets Handler - Multi-Outcome Prediction Markets\n */\nimport { Connection, PublicKey } from '@solana/web3.js';\nimport bs58 from 'bs58';\nimport {\n  PROGRAM_ID,\n  RPC_ENDPOINT,\n  DISCRIMINATORS,\n  MARKET_STATUS_NAMES,\n  MARKET_LAYER_NAMES,\n  lamportsToSol,\n} from '../config.js';\n\n// =============================================================================\n// TYPES\n// =============================================================================\n\nexport interface RaceOutcome {\n  index: number;\n  label: string;\n  poolSol: number;\n  percent: number;\n}\n\nexport interface RaceMarket {\n  publicKey: string;\n  marketId: string;\n  question: string;\n  outcomes: RaceOutcome[];\n  closingTime: string;\n  resolutionTime: string;\n  status: string;\n  statusCode: number;\n  winningOutcomeIndex: number | null;\n  totalPoolSol: number;\n  layer: string;\n  layerCode: number;\n  accessGate: string;\n  creator: string;\n  platformFeeBps: number;\n  isBettingOpen: boolean;\n}\n\nexport interface RacePosition {\n  publicKey: string;\n  user: string;\n  raceMarketPda: string;\n  marketId: string;\n  outcomeIndex: number;\n  amountSol: number;\n  claimed: boolean;\n  createdAt: string;\n}\n\n// =============================================================================\n// RACE MARKET DECODER\n// =============================================================================\n\n/**\n * Decode RaceMarket account data\n *\n * RaceMarket struct layout (from IDL v4.7.6):\n * - discriminator (8)\n * - market_id (u64, 8)\n * - question (String: 4 + len)\n * - closing_time (i64, 8)\n * - resolution_time (i64, 8)\n * - auto_stop_buffer (i64, 8)\n * - outcome_count (u8, 1)\n * - outcome_labels ([[u8; 32]; 10], 320 bytes FIXED)\n * - outcome_pools ([u64; 10], 80 bytes FIXED)\n * - total_pool (u64, 8)\n * - snapshot_pools ([u64; 10], 80 bytes)\n * - snapshot_total (u64, 8)\n * - status (enum, 1)\n * - winning_outcome (Option<u8>: 1 + 0/1)\n * - currency_type (enum, 1)\n * - platform_fee_collected (u64, 8)\n * - creator_fee_collected (u64, 8)\n * - total_claimed (u64, 8)\n * - last_bet_time (i64, 8)\n * - bump (u8, 1)\n * - layer (enum, 1)\n * - resolution_mode (enum, 1)\n * - access_gate (enum, 1)\n * - creator (Pubkey, 32)\n * - platform_fee_bps_at_creation (u16, 2)\n * - affiliate_fee_bps_at_creation (u16, 2)\n * - betting_freeze_seconds_at_creation (i64, 8)\n * - dust_swept (bool, 1)\n * - reserved ([u8; 19], 19)\n */\nfunction decodeRaceMarket(data: Buffer, pubkey: PublicKey): RaceMarket | null {\n  try {\n    // Minimum expected size for a RaceMarket account\n    // 8 (disc) + 8 (id) + 4 (qlen) + 8*3 (times) + 1 (count) + 320 (labels) + 80 (pools) + ...\n    if (data.length < 500) {\n      return null; // Account too small to be a RaceMarket\n    }\n\n    let offset = 8; // Skip discriminator\n\n    // market_id (u64)\n    const marketId = data.readBigUInt64LE(offset);\n    offset += 8;\n\n    // question (String: 4 bytes length + UTF-8 bytes)\n    const questionLen = data.readUInt32LE(offset);\n    offset += 4;\n\n    // Sanity check: question length should be reasonable (max 200 chars)\n    if (questionLen > 500 || questionLen + offset > data.length) {\n      return null; // Invalid question length - not a valid RaceMarket\n    }\n\n    const question = data.slice(offset, offset + questionLen).toString('utf8');\n    offset += questionLen;\n\n    // closing_time (i64)\n    const closingTime = data.readBigInt64LE(offset);\n    offset += 8;\n\n    // resolution_time (i64)\n    const resolutionTime = data.readBigInt64LE(offset);\n    offset += 8;\n\n    // auto_stop_buffer (i64)\n    offset += 8; // Skip for now\n\n    // outcome_count (u8)\n    const outcomeCount = data.readUInt8(offset);\n    offset += 1;\n\n    // outcome_labels: [[u8; 32]; 10] = 320 bytes FIXED\n    // Each label is 32 bytes, padded with zeros\n    const outcomeLabels: string[] = [];\n    for (let i = 0; i < 10; i++) {\n      const labelBytes = data.slice(offset, offset + 32);\n      // Find null terminator or end of string\n      let labelEnd = 32;\n      for (let j = 0; j < 32; j++) {\n        if (labelBytes[j] === 0) {\n          labelEnd = j;\n          break;\n        }\n      }\n      if (i < outcomeCount) {\n        outcomeLabels.push(labelBytes.slice(0, labelEnd).toString('utf8'));\n      }\n      offset += 32;\n    }\n\n    // outcome_pools: [u64; 10] = 80 bytes FIXED\n    const outcomePools: bigint[] = [];\n    for (let i = 0; i < 10; i++) {\n      const pool = data.readBigUInt64LE(offset);\n      if (i < outcomeCount) {\n        outcomePools.push(pool);\n      }\n      offset += 8;\n    }\n\n    // total_pool (u64)\n    const totalPoolLamports = data.readBigUInt64LE(offset);\n    offset += 8;\n\n    // snapshot_pools: [u64; 10] = 80 bytes - skip\n    offset += 80;\n\n    // snapshot_total (u64) - skip\n    offset += 8;\n\n    // status (enum, 1 byte)\n    const statusCode = data.readUInt8(offset);\n    offset += 1;\n\n    // winning_outcome (Option<u8>: 1 byte discriminant + optional 1 byte value)\n    const hasWinningOutcome = data.readUInt8(offset);\n    offset += 1;\n    let winningOutcomeIndex: number | null = null;\n    if (hasWinningOutcome === 1) {\n      winningOutcomeIndex = data.readUInt8(offset);\n      offset += 1;\n    }\n\n    // currency_type (enum, 1 byte)\n    offset += 1;\n\n    // platform_fee_collected (u64)\n    offset += 8;\n\n    // creator_fee_collected (u64)\n    offset += 8;\n\n    // total_claimed (u64)\n    offset += 8;\n\n    // last_bet_time (i64)\n    offset += 8;\n\n    // bump (u8)\n    offset += 1;\n\n    // layer (enum, 1 byte)\n    const layerCode = data.readUInt8(offset);\n    offset += 1;\n\n    // resolution_mode (enum, 1 byte)\n    offset += 1;\n\n    // access_gate (enum, 1 byte)\n    const accessGateCode = data.readUInt8(offset);\n    offset += 1;\n\n    // creator (Pubkey, 32 bytes)\n    const creator = new PublicKey(data.slice(offset, offset + 32));\n    offset += 32;\n\n    // oracle_host (Option<Pubkey>: 1 + 0/32)\n    const hasOracleHost = data.readUInt8(offset);\n    offset += 1;\n    if (hasOracleHost === 1) {\n      offset += 32;\n    }\n\n    // council: [Pubkey; 5] = 160 bytes - skip\n    offset += 160;\n\n    // council_size (u8)\n    offset += 1;\n\n    // council_votes: [u8; 10] = 10 bytes - skip\n    offset += 10;\n\n    // council_threshold (u8)\n    offset += 1;\n\n    // creator_fee_bps (u16)\n    offset += 2;\n\n    // creator_profile (Option<Pubkey>: 1 + 0/32)\n    const hasCreatorProfile = data.readUInt8(offset);\n    offset += 1;\n    if (hasCreatorProfile === 1) {\n      offset += 32;\n    }\n\n    // platform_fee_bps_at_creation (u16)\n    const platformFeeBps = data.readUInt16LE(offset);\n    offset += 2;\n\n    // Calculate derived fields\n    const totalPoolSol = lamportsToSol(totalPoolLamports);\n\n    const outcomes: RaceOutcome[] = outcomeLabels.map((label, i) => {\n      const poolSol = lamportsToSol(outcomePools[i] || 0n);\n      const percent = totalPoolSol > 0 ? (poolSol / totalPoolSol) * 100 : 100 / outcomeLabels.length;\n      return {\n        index: i,\n        label,\n        poolSol: round4(poolSol),\n        percent: round2(percent),\n      };\n    });\n\n    // Betting open check\n    const now = BigInt(Math.floor(Date.now() / 1000));\n    const freezeTime = closingTime - 300n; // 5 min freeze\n    const isBettingOpen = statusCode === 0 && now < freezeTime;\n\n    return {\n      publicKey: pubkey.toBase58(),\n      marketId: marketId.toString(),\n      question,\n      outcomes,\n      closingTime: new Date(Number(closingTime) * 1000).toISOString(),\n      resolutionTime: new Date(Number(resolutionTime) * 1000).toISOString(),\n      status: MARKET_STATUS_NAMES[statusCode] || 'Unknown',\n      statusCode,\n      winningOutcomeIndex,\n      totalPoolSol: round4(totalPoolSol),\n      layer: MARKET_LAYER_NAMES[layerCode] || 'Unknown',\n      layerCode,\n      accessGate: accessGateCode === 0 ? 'Public' : 'Whitelist',\n      creator: creator.toBase58(),\n      platformFeeBps,\n      isBettingOpen,\n    };\n  } catch (err) {\n    console.error('Error decoding race market:', err);\n    return null;\n  }\n}\n\n// =============================================================================\n// PUBLIC API\n// =============================================================================\n\n/**\n * List all race markets\n */\nexport async function listRaceMarkets(status?: string): Promise<RaceMarket[]> {\n  const connection = new Connection(RPC_ENDPOINT, 'confirmed');\n\n  const accounts = await connection.getProgramAccounts(PROGRAM_ID, {\n    filters: [\n      {\n        memcmp: {\n          offset: 0,\n          bytes: bs58.encode(DISCRIMINATORS.RACE_MARKET),\n        },\n      },\n    ],\n  });\n\n  const markets: RaceMarket[] = [];\n\n  for (const { account, pubkey } of accounts) {\n    const market = decodeRaceMarket(account.data as Buffer, pubkey);\n    if (market) {\n      if (!status || market.status.toLowerCase() === status.toLowerCase()) {\n        markets.push(market);\n      }\n    }\n  }\n\n  // Sort by closing time\n  markets.sort((a, b) => {\n    if (a.status === 'Active' && b.status !== 'Active') return -1;\n    if (a.status !== 'Active' && b.status === 'Active') return 1;\n    return new Date(a.closingTime).getTime() - new Date(b.closingTime).getTime();\n  });\n\n  return markets;\n}\n\n/**\n * Get a specific race market\n */\nexport async function getRaceMarket(publicKey: string): Promise<RaceMarket | null> {\n  const connection = new Connection(RPC_ENDPOINT, 'confirmed');\n\n  try {\n    const pubkey = new PublicKey(publicKey);\n    const account = await connection.getAccountInfo(pubkey);\n    if (!account) return null;\n    return decodeRaceMarket(account.data as Buffer, pubkey);\n  } catch {\n    return null;\n  }\n}\n\n/**\n * Get race quote for a potential bet\n */\nexport function getRaceQuote(\n  market: RaceMarket,\n  outcomeIndex: number,\n  betAmountSol: number\n): {\n  valid: boolean;\n  error?: string;\n  outcomeLabel: string;\n  betAmountSol: number;\n  expectedPayoutSol: number;\n  impliedOdds: number;\n  newOutcomePercent: number;\n} {\n  if (outcomeIndex < 0 || outcomeIndex >= market.outcomes.length) {\n    return {\n      valid: false,\n      error: `Invalid outcome index. Must be 0-${market.outcomes.length - 1}`,\n      outcomeLabel: '',\n      betAmountSol,\n      expectedPayoutSol: 0,\n      impliedOdds: 0,\n      newOutcomePercent: 0,\n    };\n  }\n\n  const outcome = market.outcomes[outcomeIndex];\n  const currentPool = outcome.poolSol;\n  const totalPool = market.totalPoolSol;\n\n  // New pools after bet\n  const newOutcomePool = currentPool + betAmountSol;\n  const newTotalPool = totalPool + betAmountSol;\n\n  // Share of outcome pool\n  const share = betAmountSol / newOutcomePool;\n  const grossPayout = share * newTotalPool;\n  const profit = grossPayout - betAmountSol;\n  const fee = profit > 0 ? (profit * market.platformFeeBps) / 10000 : 0;\n  const expectedPayout = grossPayout - fee;\n\n  // Implied odds\n  const impliedOdds = newOutcomePool / newTotalPool * 100;\n\n  return {\n    valid: true,\n    outcomeLabel: outcome.label,\n    betAmountSol,\n    expectedPayoutSol: round4(expectedPayout),\n    impliedOdds: round2(impliedOdds),\n    newOutcomePercent: round2((newOutcomePool / newTotalPool) * 100),\n  };\n}\n\n// =============================================================================\n// HELPERS\n// =============================================================================\n\nfunction round2(n: number): number {\n  return Math.round(n * 100) / 100;\n}\n\nfunction round4(n: number): number {\n  return Math.round(n * 10000) / 10000;\n}\n"]}
@@ -0,0 +1,43 @@
1
+ import { Market } from './markets.js';
2
+ export interface ResolutionStatus {
3
+ marketPda: string;
4
+ marketQuestion: string;
5
+ status: string;
6
+ isResolved: boolean;
7
+ winningOutcome: string | null;
8
+ proposedOutcome: string | null;
9
+ closingTime: string;
10
+ resolutionTime: string;
11
+ canBeResolved: boolean;
12
+ resolutionWindowOpen: boolean;
13
+ isDisputed: boolean;
14
+ disputeDeadline: string | null;
15
+ disputeReason: string | null;
16
+ councilSize: number;
17
+ councilVotesYes: number;
18
+ councilVotesNo: number;
19
+ councilThreshold: number;
20
+ resolutionMode: 'Creator' | 'Oracle' | 'Council' | 'Admin';
21
+ }
22
+ export interface DisputeMeta {
23
+ publicKey: string;
24
+ marketPda: string;
25
+ disputer: string;
26
+ reason: string;
27
+ proposedOutcome: boolean | null;
28
+ createdAt: string;
29
+ deadline: string;
30
+ resolved: boolean;
31
+ }
32
+ /**
33
+ * Get detailed resolution status for a market
34
+ */
35
+ export declare function getResolutionStatus(marketPda: string): Promise<ResolutionStatus | null>;
36
+ /**
37
+ * Get all disputed markets
38
+ */
39
+ export declare function getDisputedMarkets(): Promise<DisputeMeta[]>;
40
+ /**
41
+ * Get markets pending resolution (closed but not resolved)
42
+ */
43
+ export declare function getMarketsAwaitingResolution(): Promise<Market[]>;
@@ -0,0 +1,194 @@
1
+ /**
2
+ * Resolution Handler - Market Resolution Status & Disputes
3
+ */
4
+ import { Connection, PublicKey } from '@solana/web3.js';
5
+ import bs58 from 'bs58';
6
+ import { PROGRAM_ID, RPC_ENDPOINT, DISCRIMINATORS, SEEDS, } from '../config.js';
7
+ import { getMarket } from './markets.js';
8
+ // =============================================================================
9
+ // RESOLUTION STATUS
10
+ // =============================================================================
11
+ /**
12
+ * Get detailed resolution status for a market
13
+ */
14
+ export async function getResolutionStatus(marketPda) {
15
+ const market = await getMarket(marketPda);
16
+ if (!market)
17
+ return null;
18
+ const now = new Date();
19
+ const closingTime = new Date(market.closingTime);
20
+ const resolutionTime = new Date(market.resolutionTime);
21
+ // Check resolution window
22
+ const canBeResolved = now > closingTime && market.status === 'Closed';
23
+ const resolutionWindowOpen = now > closingTime && now < resolutionTime;
24
+ // Check dispute status
25
+ const disputeMeta = await getDisputeMeta(marketPda);
26
+ const isDisputed = disputeMeta !== null && !disputeMeta.resolved;
27
+ // Determine resolution mode from market data
28
+ let resolutionMode = 'Creator';
29
+ // This would need to be parsed from market data
30
+ return {
31
+ marketPda,
32
+ marketQuestion: market.question,
33
+ status: market.status,
34
+ isResolved: market.status === 'Resolved',
35
+ winningOutcome: market.winningOutcome,
36
+ proposedOutcome: null, // Would need to track proposed but not finalized
37
+ closingTime: market.closingTime,
38
+ resolutionTime: market.resolutionTime,
39
+ canBeResolved,
40
+ resolutionWindowOpen,
41
+ isDisputed,
42
+ disputeDeadline: disputeMeta?.deadline || null,
43
+ disputeReason: disputeMeta?.reason || null,
44
+ councilSize: 0, // Would parse from market
45
+ councilVotesYes: 0,
46
+ councilVotesNo: 0,
47
+ councilThreshold: 0,
48
+ resolutionMode,
49
+ };
50
+ }
51
+ /**
52
+ * Get dispute meta for a market (if exists)
53
+ */
54
+ async function getDisputeMeta(marketPda) {
55
+ const connection = new Connection(RPC_ENDPOINT, 'confirmed');
56
+ try {
57
+ const marketPubkey = new PublicKey(marketPda);
58
+ // Derive dispute_meta PDA
59
+ const [disputeMetaPda] = PublicKey.findProgramAddressSync([SEEDS.DISPUTE_META, marketPubkey.toBuffer()], PROGRAM_ID);
60
+ const account = await connection.getAccountInfo(disputeMetaPda);
61
+ if (!account)
62
+ return null;
63
+ // Decode DisputeMeta
64
+ // DisputeMeta struct:
65
+ // - discriminator (8)
66
+ // - market (Pubkey, 32)
67
+ // - disputer (Pubkey, 32)
68
+ // - reason (String: 4 + len)
69
+ // - proposed_outcome (Option<bool>: 1 + 0/1)
70
+ // - created_at (i64, 8)
71
+ // - deadline (i64, 8)
72
+ // - resolved (bool, 1)
73
+ const data = account.data;
74
+ let offset = 8;
75
+ const market = new PublicKey(data.slice(offset, offset + 32));
76
+ offset += 32;
77
+ const disputer = new PublicKey(data.slice(offset, offset + 32));
78
+ offset += 32;
79
+ const reasonLen = data.readUInt32LE(offset);
80
+ offset += 4;
81
+ const reason = data.slice(offset, offset + reasonLen).toString('utf8');
82
+ offset += reasonLen;
83
+ const hasProposedOutcome = data.readUInt8(offset);
84
+ offset += 1;
85
+ let proposedOutcome = null;
86
+ if (hasProposedOutcome === 1) {
87
+ proposedOutcome = data.readUInt8(offset) === 1;
88
+ offset += 1;
89
+ }
90
+ const createdAt = data.readBigInt64LE(offset);
91
+ offset += 8;
92
+ const deadline = data.readBigInt64LE(offset);
93
+ offset += 8;
94
+ const resolved = data.readUInt8(offset) === 1;
95
+ return {
96
+ publicKey: disputeMetaPda.toBase58(),
97
+ marketPda: market.toBase58(),
98
+ disputer: disputer.toBase58(),
99
+ reason,
100
+ proposedOutcome,
101
+ createdAt: new Date(Number(createdAt) * 1000).toISOString(),
102
+ deadline: new Date(Number(deadline) * 1000).toISOString(),
103
+ resolved,
104
+ };
105
+ }
106
+ catch {
107
+ return null;
108
+ }
109
+ }
110
+ /**
111
+ * Get all disputed markets
112
+ */
113
+ export async function getDisputedMarkets() {
114
+ const connection = new Connection(RPC_ENDPOINT, 'confirmed');
115
+ const accounts = await connection.getProgramAccounts(PROGRAM_ID, {
116
+ filters: [
117
+ {
118
+ memcmp: {
119
+ offset: 0,
120
+ bytes: bs58.encode(DISCRIMINATORS.DISPUTE_META),
121
+ },
122
+ },
123
+ ],
124
+ });
125
+ const disputes = [];
126
+ for (const { account, pubkey } of accounts) {
127
+ try {
128
+ const data = account.data;
129
+ let offset = 8;
130
+ const market = new PublicKey(data.slice(offset, offset + 32));
131
+ offset += 32;
132
+ const disputer = new PublicKey(data.slice(offset, offset + 32));
133
+ offset += 32;
134
+ const reasonLen = data.readUInt32LE(offset);
135
+ offset += 4;
136
+ const reason = data.slice(offset, offset + reasonLen).toString('utf8');
137
+ offset += reasonLen;
138
+ const hasProposedOutcome = data.readUInt8(offset);
139
+ offset += 1;
140
+ let proposedOutcome = null;
141
+ if (hasProposedOutcome === 1) {
142
+ proposedOutcome = data.readUInt8(offset) === 1;
143
+ offset += 1;
144
+ }
145
+ const createdAt = data.readBigInt64LE(offset);
146
+ offset += 8;
147
+ const deadline = data.readBigInt64LE(offset);
148
+ offset += 8;
149
+ const resolved = data.readUInt8(offset) === 1;
150
+ if (!resolved) {
151
+ disputes.push({
152
+ publicKey: pubkey.toBase58(),
153
+ marketPda: market.toBase58(),
154
+ disputer: disputer.toBase58(),
155
+ reason,
156
+ proposedOutcome,
157
+ createdAt: new Date(Number(createdAt) * 1000).toISOString(),
158
+ deadline: new Date(Number(deadline) * 1000).toISOString(),
159
+ resolved,
160
+ });
161
+ }
162
+ }
163
+ catch {
164
+ // Skip malformed
165
+ }
166
+ }
167
+ return disputes;
168
+ }
169
+ /**
170
+ * Get markets pending resolution (closed but not resolved)
171
+ */
172
+ export async function getMarketsAwaitingResolution() {
173
+ const connection = new Connection(RPC_ENDPOINT, 'confirmed');
174
+ // Get all markets with status = Closed (1)
175
+ const accounts = await connection.getProgramAccounts(PROGRAM_ID, {
176
+ filters: [
177
+ {
178
+ memcmp: {
179
+ offset: 0,
180
+ bytes: bs58.encode(DISCRIMINATORS.MARKET),
181
+ },
182
+ },
183
+ ],
184
+ });
185
+ const markets = [];
186
+ for (const { account, pubkey } of accounts) {
187
+ const market = await getMarket(pubkey.toBase58());
188
+ if (market && market.status === 'Closed') {
189
+ markets.push(market);
190
+ }
191
+ }
192
+ return markets;
193
+ }
194
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resolution.js","sourceRoot":"","sources":["../../src/handlers/resolution.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AACxD,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,EACL,UAAU,EACV,YAAY,EACZ,cAAc,EACd,KAAK,GACN,MAAM,cAAc,CAAC;AACtB,OAAO,EAAE,SAAS,EAAU,MAAM,cAAc,CAAC;AAgDjD,gFAAgF;AAChF,oBAAoB;AACpB,gFAAgF;AAEhF;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,mBAAmB,CAAC,SAAiB;IACzD,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,SAAS,CAAC,CAAC;IAC1C,IAAI,CAAC,MAAM;QAAE,OAAO,IAAI,CAAC;IAEzB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;IACvB,MAAM,WAAW,GAAG,IAAI,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;IACjD,MAAM,cAAc,GAAG,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;IAEvD,0BAA0B;IAC1B,MAAM,aAAa,GAAG,GAAG,GAAG,WAAW,IAAI,MAAM,CAAC,MAAM,KAAK,QAAQ,CAAC;IACtE,MAAM,oBAAoB,GAAG,GAAG,GAAG,WAAW,IAAI,GAAG,GAAG,cAAc,CAAC;IAEvE,uBAAuB;IACvB,MAAM,WAAW,GAAG,MAAM,cAAc,CAAC,SAAS,CAAC,CAAC;IACpD,MAAM,UAAU,GAAG,WAAW,KAAK,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC;IAEjE,6CAA6C;IAC7C,IAAI,cAAc,GAA+C,SAAS,CAAC;IAC3E,gDAAgD;IAEhD,OAAO;QACL,SAAS;QACT,cAAc,EAAE,MAAM,CAAC,QAAQ;QAC/B,MAAM,EAAE,MAAM,CAAC,MAAM;QAErB,UAAU,EAAE,MAAM,CAAC,MAAM,KAAK,UAAU;QACxC,cAAc,EAAE,MAAM,CAAC,cAAc;QACrC,eAAe,EAAE,IAAI,EAAE,iDAAiD;QAExE,WAAW,EAAE,MAAM,CAAC,WAAW;QAC/B,cAAc,EAAE,MAAM,CAAC,cAAc;QACrC,aAAa;QACb,oBAAoB;QAEpB,UAAU;QACV,eAAe,EAAE,WAAW,EAAE,QAAQ,IAAI,IAAI;QAC9C,aAAa,EAAE,WAAW,EAAE,MAAM,IAAI,IAAI;QAE1C,WAAW,EAAE,CAAC,EAAE,0BAA0B;QAC1C,eAAe,EAAE,CAAC;QAClB,cAAc,EAAE,CAAC;QACjB,gBAAgB,EAAE,CAAC;QAEnB,cAAc;KACf,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,KAAK,UAAU,cAAc,CAAC,SAAiB;IAC7C,MAAM,UAAU,GAAG,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAE7D,IAAI,CAAC;QACH,MAAM,YAAY,GAAG,IAAI,SAAS,CAAC,SAAS,CAAC,CAAC;QAE9C,0BAA0B;QAC1B,MAAM,CAAC,cAAc,CAAC,GAAG,SAAS,CAAC,sBAAsB,CACvD,CAAC,KAAK,CAAC,YAAY,EAAE,YAAY,CAAC,QAAQ,EAAE,CAAC,EAC7C,UAAU,CACX,CAAC;QAEF,MAAM,OAAO,GAAG,MAAM,UAAU,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;QAChE,IAAI,CAAC,OAAO;YAAE,OAAO,IAAI,CAAC;QAE1B,qBAAqB;QACrB,sBAAsB;QACtB,sBAAsB;QACtB,wBAAwB;QACxB,0BAA0B;QAC1B,6BAA6B;QAC7B,6CAA6C;QAC7C,wBAAwB;QACxB,sBAAsB;QACtB,uBAAuB;QAEvB,MAAM,IAAI,GAAG,OAAO,CAAC,IAAc,CAAC;QACpC,IAAI,MAAM,GAAG,CAAC,CAAC;QAEf,MAAM,MAAM,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC;QAC9D,MAAM,IAAI,EAAE,CAAC;QAEb,MAAM,QAAQ,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC;QAChE,MAAM,IAAI,EAAE,CAAC;QAEb,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;QAC5C,MAAM,IAAI,CAAC,CAAC;QACZ,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QACvE,MAAM,IAAI,SAAS,CAAC;QAEpB,MAAM,kBAAkB,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QAClD,MAAM,IAAI,CAAC,CAAC;QACZ,IAAI,eAAe,GAAmB,IAAI,CAAC;QAC3C,IAAI,kBAAkB,KAAK,CAAC,EAAE,CAAC;YAC7B,eAAe,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAC/C,MAAM,IAAI,CAAC,CAAC;QACd,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;QAC9C,MAAM,IAAI,CAAC,CAAC;QAEZ,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;QAC7C,MAAM,IAAI,CAAC,CAAC;QAEZ,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAE9C,OAAO;YACL,SAAS,EAAE,cAAc,CAAC,QAAQ,EAAE;YACpC,SAAS,EAAE,MAAM,CAAC,QAAQ,EAAE;YAC5B,QAAQ,EAAE,QAAQ,CAAC,QAAQ,EAAE;YAC7B,MAAM;YACN,eAAe;YACf,SAAS,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC,CAAC,WAAW,EAAE;YAC3D,QAAQ,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,CAAC,WAAW,EAAE;YACzD,QAAQ;SACT,CAAC;IACJ,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,kBAAkB;IACtC,MAAM,UAAU,GAAG,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAE7D,MAAM,QAAQ,GAAG,MAAM,UAAU,CAAC,kBAAkB,CAAC,UAAU,EAAE;QAC/D,OAAO,EAAE;YACP;gBACE,MAAM,EAAE;oBACN,MAAM,EAAE,CAAC;oBACT,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,YAAY,CAAC;iBAChD;aACF;SACF;KACF,CAAC,CAAC;IAEH,MAAM,QAAQ,GAAkB,EAAE,CAAC;IAEnC,KAAK,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,QAAQ,EAAE,CAAC;QAC3C,IAAI,CAAC;YACH,MAAM,IAAI,GAAG,OAAO,CAAC,IAAc,CAAC;YACpC,IAAI,MAAM,GAAG,CAAC,CAAC;YAEf,MAAM,MAAM,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC;YAC9D,MAAM,IAAI,EAAE,CAAC;YAEb,MAAM,QAAQ,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC;YAChE,MAAM,IAAI,EAAE,CAAC;YAEb,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;YAC5C,MAAM,IAAI,CAAC,CAAC;YACZ,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;YACvE,MAAM,IAAI,SAAS,CAAC;YAEpB,MAAM,kBAAkB,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;YAClD,MAAM,IAAI,CAAC,CAAC;YACZ,IAAI,eAAe,GAAmB,IAAI,CAAC;YAC3C,IAAI,kBAAkB,KAAK,CAAC,EAAE,CAAC;gBAC7B,eAAe,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBAC/C,MAAM,IAAI,CAAC,CAAC;YACd,CAAC;YAED,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;YAC9C,MAAM,IAAI,CAAC,CAAC;YAEZ,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;YAC7C,MAAM,IAAI,CAAC,CAAC;YAEZ,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAE9C,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACd,QAAQ,CAAC,IAAI,CAAC;oBACZ,SAAS,EAAE,MAAM,CAAC,QAAQ,EAAE;oBAC5B,SAAS,EAAE,MAAM,CAAC,QAAQ,EAAE;oBAC5B,QAAQ,EAAE,QAAQ,CAAC,QAAQ,EAAE;oBAC7B,MAAM;oBACN,eAAe;oBACf,SAAS,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC,CAAC,WAAW,EAAE;oBAC3D,QAAQ,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,CAAC,WAAW,EAAE;oBACzD,QAAQ;iBACT,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAAC,MAAM,CAAC;YACP,iBAAiB;QACnB,CAAC;IACH,CAAC;IAED,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,4BAA4B;IAChD,MAAM,UAAU,GAAG,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAE7D,2CAA2C;IAC3C,MAAM,QAAQ,GAAG,MAAM,UAAU,CAAC,kBAAkB,CAAC,UAAU,EAAE;QAC/D,OAAO,EAAE;YACP;gBACE,MAAM,EAAE;oBACN,MAAM,EAAE,CAAC;oBACT,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC;iBAC1C;aACF;SACF;KACF,CAAC,CAAC;IAEH,MAAM,OAAO,GAAa,EAAE,CAAC;IAE7B,KAAK,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,QAAQ,EAAE,CAAC;QAC3C,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;QAClD,IAAI,MAAM,IAAI,MAAM,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;YACzC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACvB,CAAC;IACH,CAAC;IAED,OAAO,OAAO,CAAC;AACjB,CAAC","sourcesContent":["/**\n * Resolution Handler - Market Resolution Status & Disputes\n */\nimport { Connection, PublicKey } from '@solana/web3.js';\nimport bs58 from 'bs58';\nimport {\n  PROGRAM_ID,\n  RPC_ENDPOINT,\n  DISCRIMINATORS,\n  SEEDS,\n} from '../config.js';\nimport { getMarket, Market } from './markets.js';\n\n// =============================================================================\n// TYPES\n// =============================================================================\n\nexport interface ResolutionStatus {\n  marketPda: string;\n  marketQuestion: string;\n  status: string;\n\n  // Resolution state\n  isResolved: boolean;\n  winningOutcome: string | null;\n  proposedOutcome: string | null;\n\n  // Timing\n  closingTime: string;\n  resolutionTime: string;\n  canBeResolved: boolean;\n  resolutionWindowOpen: boolean;\n\n  // Dispute state\n  isDisputed: boolean;\n  disputeDeadline: string | null;\n  disputeReason: string | null;\n\n  // Council voting (if disputed)\n  councilSize: number;\n  councilVotesYes: number;\n  councilVotesNo: number;\n  councilThreshold: number;\n\n  // Resolution mode\n  resolutionMode: 'Creator' | 'Oracle' | 'Council' | 'Admin';\n}\n\nexport interface DisputeMeta {\n  publicKey: string;\n  marketPda: string;\n  disputer: string;\n  reason: string;\n  proposedOutcome: boolean | null;\n  createdAt: string;\n  deadline: string;\n  resolved: boolean;\n}\n\n// =============================================================================\n// RESOLUTION STATUS\n// =============================================================================\n\n/**\n * Get detailed resolution status for a market\n */\nexport async function getResolutionStatus(marketPda: string): Promise<ResolutionStatus | null> {\n  const market = await getMarket(marketPda);\n  if (!market) return null;\n\n  const now = new Date();\n  const closingTime = new Date(market.closingTime);\n  const resolutionTime = new Date(market.resolutionTime);\n\n  // Check resolution window\n  const canBeResolved = now > closingTime && market.status === 'Closed';\n  const resolutionWindowOpen = now > closingTime && now < resolutionTime;\n\n  // Check dispute status\n  const disputeMeta = await getDisputeMeta(marketPda);\n  const isDisputed = disputeMeta !== null && !disputeMeta.resolved;\n\n  // Determine resolution mode from market data\n  let resolutionMode: 'Creator' | 'Oracle' | 'Council' | 'Admin' = 'Creator';\n  // This would need to be parsed from market data\n\n  return {\n    marketPda,\n    marketQuestion: market.question,\n    status: market.status,\n\n    isResolved: market.status === 'Resolved',\n    winningOutcome: market.winningOutcome,\n    proposedOutcome: null, // Would need to track proposed but not finalized\n\n    closingTime: market.closingTime,\n    resolutionTime: market.resolutionTime,\n    canBeResolved,\n    resolutionWindowOpen,\n\n    isDisputed,\n    disputeDeadline: disputeMeta?.deadline || null,\n    disputeReason: disputeMeta?.reason || null,\n\n    councilSize: 0, // Would parse from market\n    councilVotesYes: 0,\n    councilVotesNo: 0,\n    councilThreshold: 0,\n\n    resolutionMode,\n  };\n}\n\n/**\n * Get dispute meta for a market (if exists)\n */\nasync function getDisputeMeta(marketPda: string): Promise<DisputeMeta | null> {\n  const connection = new Connection(RPC_ENDPOINT, 'confirmed');\n\n  try {\n    const marketPubkey = new PublicKey(marketPda);\n\n    // Derive dispute_meta PDA\n    const [disputeMetaPda] = PublicKey.findProgramAddressSync(\n      [SEEDS.DISPUTE_META, marketPubkey.toBuffer()],\n      PROGRAM_ID\n    );\n\n    const account = await connection.getAccountInfo(disputeMetaPda);\n    if (!account) return null;\n\n    // Decode DisputeMeta\n    // DisputeMeta struct:\n    // - discriminator (8)\n    // - market (Pubkey, 32)\n    // - disputer (Pubkey, 32)\n    // - reason (String: 4 + len)\n    // - proposed_outcome (Option<bool>: 1 + 0/1)\n    // - created_at (i64, 8)\n    // - deadline (i64, 8)\n    // - resolved (bool, 1)\n\n    const data = account.data as Buffer;\n    let offset = 8;\n\n    const market = new PublicKey(data.slice(offset, offset + 32));\n    offset += 32;\n\n    const disputer = new PublicKey(data.slice(offset, offset + 32));\n    offset += 32;\n\n    const reasonLen = data.readUInt32LE(offset);\n    offset += 4;\n    const reason = data.slice(offset, offset + reasonLen).toString('utf8');\n    offset += reasonLen;\n\n    const hasProposedOutcome = data.readUInt8(offset);\n    offset += 1;\n    let proposedOutcome: boolean | null = null;\n    if (hasProposedOutcome === 1) {\n      proposedOutcome = data.readUInt8(offset) === 1;\n      offset += 1;\n    }\n\n    const createdAt = data.readBigInt64LE(offset);\n    offset += 8;\n\n    const deadline = data.readBigInt64LE(offset);\n    offset += 8;\n\n    const resolved = data.readUInt8(offset) === 1;\n\n    return {\n      publicKey: disputeMetaPda.toBase58(),\n      marketPda: market.toBase58(),\n      disputer: disputer.toBase58(),\n      reason,\n      proposedOutcome,\n      createdAt: new Date(Number(createdAt) * 1000).toISOString(),\n      deadline: new Date(Number(deadline) * 1000).toISOString(),\n      resolved,\n    };\n  } catch {\n    return null;\n  }\n}\n\n/**\n * Get all disputed markets\n */\nexport async function getDisputedMarkets(): Promise<DisputeMeta[]> {\n  const connection = new Connection(RPC_ENDPOINT, 'confirmed');\n\n  const accounts = await connection.getProgramAccounts(PROGRAM_ID, {\n    filters: [\n      {\n        memcmp: {\n          offset: 0,\n          bytes: bs58.encode(DISCRIMINATORS.DISPUTE_META),\n        },\n      },\n    ],\n  });\n\n  const disputes: DisputeMeta[] = [];\n\n  for (const { account, pubkey } of accounts) {\n    try {\n      const data = account.data as Buffer;\n      let offset = 8;\n\n      const market = new PublicKey(data.slice(offset, offset + 32));\n      offset += 32;\n\n      const disputer = new PublicKey(data.slice(offset, offset + 32));\n      offset += 32;\n\n      const reasonLen = data.readUInt32LE(offset);\n      offset += 4;\n      const reason = data.slice(offset, offset + reasonLen).toString('utf8');\n      offset += reasonLen;\n\n      const hasProposedOutcome = data.readUInt8(offset);\n      offset += 1;\n      let proposedOutcome: boolean | null = null;\n      if (hasProposedOutcome === 1) {\n        proposedOutcome = data.readUInt8(offset) === 1;\n        offset += 1;\n      }\n\n      const createdAt = data.readBigInt64LE(offset);\n      offset += 8;\n\n      const deadline = data.readBigInt64LE(offset);\n      offset += 8;\n\n      const resolved = data.readUInt8(offset) === 1;\n\n      if (!resolved) {\n        disputes.push({\n          publicKey: pubkey.toBase58(),\n          marketPda: market.toBase58(),\n          disputer: disputer.toBase58(),\n          reason,\n          proposedOutcome,\n          createdAt: new Date(Number(createdAt) * 1000).toISOString(),\n          deadline: new Date(Number(deadline) * 1000).toISOString(),\n          resolved,\n        });\n      }\n    } catch {\n      // Skip malformed\n    }\n  }\n\n  return disputes;\n}\n\n/**\n * Get markets pending resolution (closed but not resolved)\n */\nexport async function getMarketsAwaitingResolution(): Promise<Market[]> {\n  const connection = new Connection(RPC_ENDPOINT, 'confirmed');\n\n  // Get all markets with status = Closed (1)\n  const accounts = await connection.getProgramAccounts(PROGRAM_ID, {\n    filters: [\n      {\n        memcmp: {\n          offset: 0,\n          bytes: bs58.encode(DISCRIMINATORS.MARKET),\n        },\n      },\n    ],\n  });\n\n  const markets: Market[] = [];\n\n  for (const { account, pubkey } of accounts) {\n    const market = await getMarket(pubkey.toBase58());\n    if (market && market.status === 'Closed') {\n      markets.push(market);\n    }\n  }\n\n  return markets;\n}\n"]}
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};