@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,239 @@
1
+ /**
2
+ * Market Management Transaction Builders
3
+ *
4
+ * Builds unsigned transactions for:
5
+ * - Closing markets (stopping betting)
6
+ * - Extending market deadlines
7
+ */
8
+ import { Connection, PublicKey, Transaction, TransactionInstruction, } from '@solana/web3.js';
9
+ import { PROGRAM_ID, CONFIG_PDA, RPC_ENDPOINT, } from '../config.js';
10
+ // =============================================================================
11
+ // DISCRIMINATORS
12
+ // =============================================================================
13
+ const DISCRIMINATORS = {
14
+ close_market: Buffer.from([88, 154, 248, 186, 48, 14, 123, 244]),
15
+ close_race_market: Buffer.from([39, 189, 166, 118, 134, 37, 102, 41]),
16
+ extend_market: Buffer.from([105, 89, 206, 205, 57, 31, 153, 252]),
17
+ extend_race_market: Buffer.from([242, 176, 227, 152, 79, 116, 110, 168]),
18
+ cancel_market: Buffer.from([205, 121, 84, 210, 222, 71, 150, 11]),
19
+ cancel_race: Buffer.from([28, 31, 113, 29, 126, 206, 39, 119]),
20
+ };
21
+ // =============================================================================
22
+ // BOOLEAN MARKET MANAGEMENT
23
+ // =============================================================================
24
+ /**
25
+ * Build close_market transaction
26
+ * Stops betting on a market (usually done by creator before resolution)
27
+ */
28
+ export async function buildCloseMarketTransaction(params) {
29
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
30
+ const marketPubkey = new PublicKey(params.marketPda);
31
+ const callerPubkey = new PublicKey(params.callerWallet);
32
+ const data = DISCRIMINATORS.close_market;
33
+ const keys = [
34
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
35
+ { pubkey: marketPubkey, isSigner: false, isWritable: true },
36
+ { pubkey: callerPubkey, isSigner: true, isWritable: false },
37
+ ];
38
+ const instruction = new TransactionInstruction({
39
+ programId: PROGRAM_ID,
40
+ keys,
41
+ data,
42
+ });
43
+ const transaction = new Transaction().add(instruction);
44
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
45
+ transaction.recentBlockhash = blockhash;
46
+ transaction.feePayer = callerPubkey;
47
+ return {
48
+ transaction,
49
+ serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),
50
+ };
51
+ }
52
+ /**
53
+ * Build extend_market transaction
54
+ * Extends the closing time and/or resolution time
55
+ */
56
+ export async function buildExtendMarketTransaction(params) {
57
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
58
+ const marketPubkey = new PublicKey(params.marketPda);
59
+ const callerPubkey = new PublicKey(params.callerWallet);
60
+ // Instruction data: discriminator + new_closing_time (i64) + Option<i64> new_resolution_time
61
+ const hasResolutionTime = params.newResolutionTime !== undefined;
62
+ const dataSize = 8 + 8 + 1 + (hasResolutionTime ? 8 : 0);
63
+ const data = Buffer.alloc(dataSize);
64
+ let offset = 0;
65
+ DISCRIMINATORS.extend_market.copy(data, offset);
66
+ offset += 8;
67
+ data.writeBigInt64LE(BigInt(params.newClosingTime), offset);
68
+ offset += 8;
69
+ if (hasResolutionTime) {
70
+ data.writeUInt8(1, offset); // Some
71
+ offset += 1;
72
+ data.writeBigInt64LE(BigInt(params.newResolutionTime), offset);
73
+ }
74
+ else {
75
+ data.writeUInt8(0, offset); // None
76
+ }
77
+ const keys = [
78
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
79
+ { pubkey: marketPubkey, isSigner: false, isWritable: true },
80
+ { pubkey: callerPubkey, isSigner: true, isWritable: false },
81
+ ];
82
+ const instruction = new TransactionInstruction({
83
+ programId: PROGRAM_ID,
84
+ keys,
85
+ data,
86
+ });
87
+ const transaction = new Transaction().add(instruction);
88
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
89
+ transaction.recentBlockhash = blockhash;
90
+ transaction.feePayer = callerPubkey;
91
+ return {
92
+ transaction,
93
+ serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),
94
+ };
95
+ }
96
+ // =============================================================================
97
+ // RACE MARKET MANAGEMENT
98
+ // =============================================================================
99
+ /**
100
+ * Build close_race_market transaction
101
+ */
102
+ export async function buildCloseRaceMarketTransaction(params) {
103
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
104
+ const raceMarketPubkey = new PublicKey(params.raceMarketPda);
105
+ const callerPubkey = new PublicKey(params.callerWallet);
106
+ const data = DISCRIMINATORS.close_race_market;
107
+ const keys = [
108
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
109
+ { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },
110
+ { pubkey: callerPubkey, isSigner: true, isWritable: false },
111
+ ];
112
+ const instruction = new TransactionInstruction({
113
+ programId: PROGRAM_ID,
114
+ keys,
115
+ data,
116
+ });
117
+ const transaction = new Transaction().add(instruction);
118
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
119
+ transaction.recentBlockhash = blockhash;
120
+ transaction.feePayer = callerPubkey;
121
+ return {
122
+ transaction,
123
+ serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),
124
+ };
125
+ }
126
+ /**
127
+ * Build extend_race_market transaction
128
+ */
129
+ export async function buildExtendRaceMarketTransaction(params) {
130
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
131
+ const raceMarketPubkey = new PublicKey(params.raceMarketPda);
132
+ const callerPubkey = new PublicKey(params.callerWallet);
133
+ const hasResolutionTime = params.newResolutionTime !== undefined;
134
+ const dataSize = 8 + 8 + 1 + (hasResolutionTime ? 8 : 0);
135
+ const data = Buffer.alloc(dataSize);
136
+ let offset = 0;
137
+ DISCRIMINATORS.extend_race_market.copy(data, offset);
138
+ offset += 8;
139
+ data.writeBigInt64LE(BigInt(params.newClosingTime), offset);
140
+ offset += 8;
141
+ if (hasResolutionTime) {
142
+ data.writeUInt8(1, offset); // Some
143
+ offset += 1;
144
+ data.writeBigInt64LE(BigInt(params.newResolutionTime), offset);
145
+ }
146
+ else {
147
+ data.writeUInt8(0, offset); // None
148
+ }
149
+ const keys = [
150
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
151
+ { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },
152
+ { pubkey: callerPubkey, isSigner: true, isWritable: false },
153
+ ];
154
+ const instruction = new TransactionInstruction({
155
+ programId: PROGRAM_ID,
156
+ keys,
157
+ data,
158
+ });
159
+ const transaction = new Transaction().add(instruction);
160
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
161
+ transaction.recentBlockhash = blockhash;
162
+ transaction.feePayer = callerPubkey;
163
+ return {
164
+ transaction,
165
+ serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),
166
+ };
167
+ }
168
+ // =============================================================================
169
+ // CANCEL MARKET (ADMIN/CREATOR)
170
+ // =============================================================================
171
+ /**
172
+ * Build cancel_market transaction
173
+ * Cancels a market and allows all bettors to claim refunds.
174
+ * Only callable by admin or creator (depending on market status).
175
+ */
176
+ export async function buildCancelMarketTransaction(params) {
177
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
178
+ const marketPubkey = new PublicKey(params.marketPda);
179
+ const authorityPubkey = new PublicKey(params.authorityWallet);
180
+ // Instruction data: discriminator + reason (string)
181
+ // String encoding: 4-byte length prefix + UTF-8 bytes
182
+ const reasonBytes = Buffer.from(params.reason, 'utf8');
183
+ const data = Buffer.alloc(8 + 4 + reasonBytes.length);
184
+ DISCRIMINATORS.cancel_market.copy(data, 0);
185
+ data.writeUInt32LE(reasonBytes.length, 8);
186
+ reasonBytes.copy(data, 12);
187
+ const keys = [
188
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
189
+ { pubkey: marketPubkey, isSigner: false, isWritable: true },
190
+ { pubkey: authorityPubkey, isSigner: true, isWritable: false },
191
+ ];
192
+ const instruction = new TransactionInstruction({
193
+ programId: PROGRAM_ID,
194
+ keys,
195
+ data,
196
+ });
197
+ const transaction = new Transaction().add(instruction);
198
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
199
+ transaction.recentBlockhash = blockhash;
200
+ transaction.feePayer = authorityPubkey;
201
+ return {
202
+ transaction,
203
+ serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),
204
+ };
205
+ }
206
+ /**
207
+ * Build cancel_race transaction
208
+ * Cancels a race market and allows all bettors to claim refunds.
209
+ */
210
+ export async function buildCancelRaceTransaction(params) {
211
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
212
+ const raceMarketPubkey = new PublicKey(params.raceMarketPda);
213
+ const authorityPubkey = new PublicKey(params.authorityWallet);
214
+ // Instruction data: discriminator + reason (string)
215
+ const reasonBytes = Buffer.from(params.reason, 'utf8');
216
+ const data = Buffer.alloc(8 + 4 + reasonBytes.length);
217
+ DISCRIMINATORS.cancel_race.copy(data, 0);
218
+ data.writeUInt32LE(reasonBytes.length, 8);
219
+ reasonBytes.copy(data, 12);
220
+ const keys = [
221
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
222
+ { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },
223
+ { pubkey: authorityPubkey, isSigner: true, isWritable: false },
224
+ ];
225
+ const instruction = new TransactionInstruction({
226
+ programId: PROGRAM_ID,
227
+ keys,
228
+ data,
229
+ });
230
+ const transaction = new Transaction().add(instruction);
231
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
232
+ transaction.recentBlockhash = blockhash;
233
+ transaction.feePayer = authorityPubkey;
234
+ return {
235
+ transaction,
236
+ serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),
237
+ };
238
+ }
239
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"market-management-transaction.js","sourceRoot":"","sources":["../../src/builders/market-management-transaction.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Market Management Transaction Builders\n *\n * Builds unsigned transactions for:\n * - Closing markets (stopping betting)\n * - Extending market deadlines\n */\nimport {\n  Connection,\n  PublicKey,\n  Transaction,\n  TransactionInstruction,\n} from '@solana/web3.js';\nimport {\n  PROGRAM_ID,\n  CONFIG_PDA,\n  RPC_ENDPOINT,\n} from '../config.js';\n\n// =============================================================================\n// DISCRIMINATORS\n// =============================================================================\n\nconst DISCRIMINATORS = {\n  close_market: Buffer.from([88, 154, 248, 186, 48, 14, 123, 244]),\n  close_race_market: Buffer.from([39, 189, 166, 118, 134, 37, 102, 41]),\n  extend_market: Buffer.from([105, 89, 206, 205, 57, 31, 153, 252]),\n  extend_race_market: Buffer.from([242, 176, 227, 152, 79, 116, 110, 168]),\n  cancel_market: Buffer.from([205, 121, 84, 210, 222, 71, 150, 11]),\n  cancel_race: Buffer.from([28, 31, 113, 29, 126, 206, 39, 119]),\n};\n\n// =============================================================================\n// BOOLEAN MARKET MANAGEMENT\n// =============================================================================\n\n/**\n * Build close_market transaction\n * Stops betting on a market (usually done by creator before resolution)\n */\nexport async function buildCloseMarketTransaction(params: {\n  marketPda: string;\n  callerWallet: string;\n  connection?: Connection;\n}): Promise<{ transaction: Transaction; serializedTx: string }> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const marketPubkey = new PublicKey(params.marketPda);\n  const callerPubkey = new PublicKey(params.callerWallet);\n\n  const data = DISCRIMINATORS.close_market;\n\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: marketPubkey, isSigner: false, isWritable: true },\n    { pubkey: callerPubkey, isSigner: true, isWritable: false },\n  ];\n\n  const instruction = new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n\n  const transaction = new Transaction().add(instruction);\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = callerPubkey;\n\n  return {\n    transaction,\n    serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),\n  };\n}\n\n/**\n * Build extend_market transaction\n * Extends the closing time and/or resolution time\n */\nexport async function buildExtendMarketTransaction(params: {\n  marketPda: string;\n  newClosingTime: number; // Unix timestamp\n  newResolutionTime?: number; // Unix timestamp (optional)\n  callerWallet: string;\n  connection?: Connection;\n}): Promise<{ transaction: Transaction; serializedTx: string }> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const marketPubkey = new PublicKey(params.marketPda);\n  const callerPubkey = new PublicKey(params.callerWallet);\n\n  // Instruction data: discriminator + new_closing_time (i64) + Option<i64> new_resolution_time\n  const hasResolutionTime = params.newResolutionTime !== undefined;\n  const dataSize = 8 + 8 + 1 + (hasResolutionTime ? 8 : 0);\n  const data = Buffer.alloc(dataSize);\n\n  let offset = 0;\n  DISCRIMINATORS.extend_market.copy(data, offset);\n  offset += 8;\n\n  data.writeBigInt64LE(BigInt(params.newClosingTime), offset);\n  offset += 8;\n\n  if (hasResolutionTime) {\n    data.writeUInt8(1, offset); // Some\n    offset += 1;\n    data.writeBigInt64LE(BigInt(params.newResolutionTime!), offset);\n  } else {\n    data.writeUInt8(0, offset); // None\n  }\n\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: marketPubkey, isSigner: false, isWritable: true },\n    { pubkey: callerPubkey, isSigner: true, isWritable: false },\n  ];\n\n  const instruction = new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n\n  const transaction = new Transaction().add(instruction);\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = callerPubkey;\n\n  return {\n    transaction,\n    serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),\n  };\n}\n\n// =============================================================================\n// RACE MARKET MANAGEMENT\n// =============================================================================\n\n/**\n * Build close_race_market transaction\n */\nexport async function buildCloseRaceMarketTransaction(params: {\n  raceMarketPda: string;\n  callerWallet: string;\n  connection?: Connection;\n}): Promise<{ transaction: Transaction; serializedTx: string }> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const raceMarketPubkey = new PublicKey(params.raceMarketPda);\n  const callerPubkey = new PublicKey(params.callerWallet);\n\n  const data = DISCRIMINATORS.close_race_market;\n\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },\n    { pubkey: callerPubkey, isSigner: true, isWritable: false },\n  ];\n\n  const instruction = new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n\n  const transaction = new Transaction().add(instruction);\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = callerPubkey;\n\n  return {\n    transaction,\n    serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),\n  };\n}\n\n/**\n * Build extend_race_market transaction\n */\nexport async function buildExtendRaceMarketTransaction(params: {\n  raceMarketPda: string;\n  newClosingTime: number; // Unix timestamp\n  newResolutionTime?: number; // Unix timestamp (optional)\n  callerWallet: string;\n  connection?: Connection;\n}): Promise<{ transaction: Transaction; serializedTx: string }> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const raceMarketPubkey = new PublicKey(params.raceMarketPda);\n  const callerPubkey = new PublicKey(params.callerWallet);\n\n  const hasResolutionTime = params.newResolutionTime !== undefined;\n  const dataSize = 8 + 8 + 1 + (hasResolutionTime ? 8 : 0);\n  const data = Buffer.alloc(dataSize);\n\n  let offset = 0;\n  DISCRIMINATORS.extend_race_market.copy(data, offset);\n  offset += 8;\n\n  data.writeBigInt64LE(BigInt(params.newClosingTime), offset);\n  offset += 8;\n\n  if (hasResolutionTime) {\n    data.writeUInt8(1, offset); // Some\n    offset += 1;\n    data.writeBigInt64LE(BigInt(params.newResolutionTime!), offset);\n  } else {\n    data.writeUInt8(0, offset); // None\n  }\n\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },\n    { pubkey: callerPubkey, isSigner: true, isWritable: false },\n  ];\n\n  const instruction = new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n\n  const transaction = new Transaction().add(instruction);\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = callerPubkey;\n\n  return {\n    transaction,\n    serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),\n  };\n}\n\n// =============================================================================\n// CANCEL MARKET (ADMIN/CREATOR)\n// =============================================================================\n\n/**\n * Build cancel_market transaction\n * Cancels a market and allows all bettors to claim refunds.\n * Only callable by admin or creator (depending on market status).\n */\nexport async function buildCancelMarketTransaction(params: {\n  marketPda: string;\n  reason: string;\n  authorityWallet: string;\n  connection?: Connection;\n}): Promise<{ transaction: Transaction; serializedTx: string }> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const marketPubkey = new PublicKey(params.marketPda);\n  const authorityPubkey = new PublicKey(params.authorityWallet);\n\n  // Instruction data: discriminator + reason (string)\n  // String encoding: 4-byte length prefix + UTF-8 bytes\n  const reasonBytes = Buffer.from(params.reason, 'utf8');\n  const data = Buffer.alloc(8 + 4 + reasonBytes.length);\n  DISCRIMINATORS.cancel_market.copy(data, 0);\n  data.writeUInt32LE(reasonBytes.length, 8);\n  reasonBytes.copy(data, 12);\n\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: marketPubkey, isSigner: false, isWritable: true },\n    { pubkey: authorityPubkey, isSigner: true, isWritable: false },\n  ];\n\n  const instruction = new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n\n  const transaction = new Transaction().add(instruction);\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = authorityPubkey;\n\n  return {\n    transaction,\n    serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),\n  };\n}\n\n/**\n * Build cancel_race transaction\n * Cancels a race market and allows all bettors to claim refunds.\n */\nexport async function buildCancelRaceTransaction(params: {\n  raceMarketPda: string;\n  reason: string;\n  authorityWallet: string;\n  connection?: Connection;\n}): Promise<{ transaction: Transaction; serializedTx: string }> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const raceMarketPubkey = new PublicKey(params.raceMarketPda);\n  const authorityPubkey = new PublicKey(params.authorityWallet);\n\n  // Instruction data: discriminator + reason (string)\n  const reasonBytes = Buffer.from(params.reason, 'utf8');\n  const data = Buffer.alloc(8 + 4 + reasonBytes.length);\n  DISCRIMINATORS.cancel_race.copy(data, 0);\n  data.writeUInt32LE(reasonBytes.length, 8);\n  reasonBytes.copy(data, 12);\n\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },\n    { pubkey: authorityPubkey, isSigner: true, isWritable: false },\n  ];\n\n  const instruction = new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n\n  const transaction = new Transaction().add(instruction);\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = authorityPubkey;\n\n  return {\n    transaction,\n    serializedTx: transaction.serialize({ requireAllSignatures: false, verifySignatures: false }).toString('base64'),\n  };\n}\n"]}
@@ -0,0 +1,67 @@
1
+ /**
2
+ * Race Market Transaction Builders
3
+ *
4
+ * Builds unsigned transactions for:
5
+ * - Placing bets on race (multi-outcome) markets
6
+ * - Claiming race winnings
7
+ * - Claiming race refunds
8
+ */
9
+ import { Connection, Transaction } from '@solana/web3.js';
10
+ export interface RaceBetTransactionResult {
11
+ transaction: Transaction;
12
+ serializedTx: string;
13
+ positionPda: string;
14
+ marketId: bigint;
15
+ }
16
+ /**
17
+ * Build bet_on_race_outcome_sol transaction
18
+ */
19
+ export declare function buildRaceBetTransaction(params: {
20
+ raceMarketPda: string;
21
+ marketId: bigint;
22
+ outcomeIndex: number;
23
+ amountSol: number;
24
+ userWallet: string;
25
+ whitelistRequired?: boolean;
26
+ affiliatePda?: string;
27
+ connection?: Connection;
28
+ }): Promise<RaceBetTransactionResult>;
29
+ /**
30
+ * Fetch race market data and build bet transaction
31
+ */
32
+ export declare function fetchAndBuildRaceBetTransaction(params: {
33
+ raceMarketPda: string;
34
+ outcomeIndex: number;
35
+ amountSol: number;
36
+ userWallet: string;
37
+ affiliatePda?: string;
38
+ connection?: Connection;
39
+ }): Promise<{
40
+ transaction: RaceBetTransactionResult | null;
41
+ marketId: bigint;
42
+ error?: string;
43
+ }>;
44
+ /**
45
+ * Build claim_race_winnings_sol transaction
46
+ */
47
+ export declare function buildClaimRaceWinningsTransaction(params: {
48
+ raceMarketPda: string;
49
+ positionPda: string;
50
+ userWallet: string;
51
+ connection?: Connection;
52
+ }): Promise<{
53
+ transaction: Transaction;
54
+ serializedTx: string;
55
+ }>;
56
+ /**
57
+ * Build claim_race_refund transaction
58
+ */
59
+ export declare function buildClaimRaceRefundTransaction(params: {
60
+ raceMarketPda: string;
61
+ positionPda: string;
62
+ userWallet: string;
63
+ connection?: Connection;
64
+ }): Promise<{
65
+ transaction: Transaction;
66
+ serializedTx: string;
67
+ }>;
@@ -0,0 +1,242 @@
1
+ /**
2
+ * Race Market Transaction Builders
3
+ *
4
+ * Builds unsigned transactions for:
5
+ * - Placing bets on race (multi-outcome) markets
6
+ * - Claiming race winnings
7
+ * - Claiming race refunds
8
+ */
9
+ import { Connection, PublicKey, Transaction, TransactionInstruction, SystemProgram, } from '@solana/web3.js';
10
+ import { PROGRAM_ID, CONFIG_PDA, SOL_TREASURY_PDA, SEEDS, RPC_ENDPOINT, solToLamports, } from '../config.js';
11
+ // =============================================================================
12
+ // INSTRUCTION DISCRIMINATORS
13
+ // =============================================================================
14
+ // Correct discriminators: sha256("global:<instruction_name>")[0..8]
15
+ const BET_ON_RACE_SOL_DISCRIMINATOR = Buffer.from([195, 181, 151, 159, 105, 100, 234, 244]);
16
+ const BET_ON_RACE_SOL_WITH_AFFILIATE_DISCRIMINATOR = Buffer.from([26, 224, 14, 181, 67, 52, 24, 0]);
17
+ const CLAIM_RACE_WINNINGS_SOL_DISCRIMINATOR = Buffer.from([46, 120, 202, 194, 126, 72, 22, 52]);
18
+ const CLAIM_RACE_REFUND_DISCRIMINATOR = Buffer.from([174, 101, 101, 227, 171, 69, 173, 243]);
19
+ // =============================================================================
20
+ // RACE POSITION PDA
21
+ // =============================================================================
22
+ function deriveRacePositionPda(marketId, user) {
23
+ const marketIdBuffer = Buffer.alloc(8);
24
+ marketIdBuffer.writeBigUInt64LE(marketId);
25
+ const [pda] = PublicKey.findProgramAddressSync([SEEDS.RACE_POSITION, marketIdBuffer, user.toBuffer()], PROGRAM_ID);
26
+ return pda;
27
+ }
28
+ function deriveRaceWhitelistPda(raceMarketPda) {
29
+ const [pda] = PublicKey.findProgramAddressSync([SEEDS.RACE_WHITELIST, raceMarketPda.toBuffer()], PROGRAM_ID);
30
+ return pda;
31
+ }
32
+ function deriveRaceReferralPda(user) {
33
+ const [pda] = PublicKey.findProgramAddressSync([Buffer.from('race_referral'), user.toBuffer()], PROGRAM_ID);
34
+ return pda;
35
+ }
36
+ // =============================================================================
37
+ // BET ON RACE OUTCOME
38
+ // =============================================================================
39
+ /**
40
+ * Build bet_on_race_outcome_sol transaction
41
+ */
42
+ export async function buildRaceBetTransaction(params) {
43
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
44
+ const raceMarketPubkey = new PublicKey(params.raceMarketPda);
45
+ const userPubkey = new PublicKey(params.userWallet);
46
+ // Derive position PDA
47
+ const positionPda = deriveRacePositionPda(params.marketId, userPubkey);
48
+ // Optional whitelist PDA
49
+ const whitelistPda = params.whitelistRequired
50
+ ? deriveRaceWhitelistPda(raceMarketPubkey)
51
+ : null;
52
+ // Amount in lamports
53
+ const amountLamports = solToLamports(params.amountSol);
54
+ let instruction;
55
+ if (params.affiliatePda) {
56
+ // With affiliate
57
+ const affiliatePubkey = new PublicKey(params.affiliatePda);
58
+ const raceReferralPda = deriveRaceReferralPda(userPubkey);
59
+ // Instruction data: discriminator + outcome_index (u8) + amount (u64)
60
+ const data = Buffer.alloc(17);
61
+ BET_ON_RACE_SOL_WITH_AFFILIATE_DISCRIMINATOR.copy(data, 0);
62
+ data.writeUInt8(params.outcomeIndex, 8);
63
+ data.writeBigUInt64LE(amountLamports, 9);
64
+ const keys = [
65
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
66
+ { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },
67
+ { pubkey: positionPda, isSigner: false, isWritable: true },
68
+ { pubkey: affiliatePubkey, isSigner: false, isWritable: true },
69
+ { pubkey: raceReferralPda, isSigner: false, isWritable: true },
70
+ ];
71
+ if (whitelistPda) {
72
+ keys.push({ pubkey: whitelistPda, isSigner: false, isWritable: false });
73
+ }
74
+ keys.push({ pubkey: userPubkey, isSigner: true, isWritable: true }, { pubkey: SystemProgram.programId, isSigner: false, isWritable: false });
75
+ instruction = new TransactionInstruction({
76
+ programId: PROGRAM_ID,
77
+ keys,
78
+ data,
79
+ });
80
+ }
81
+ else {
82
+ // Without affiliate
83
+ const data = Buffer.alloc(17);
84
+ BET_ON_RACE_SOL_DISCRIMINATOR.copy(data, 0);
85
+ data.writeUInt8(params.outcomeIndex, 8);
86
+ data.writeBigUInt64LE(amountLamports, 9);
87
+ const keys = [
88
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
89
+ { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },
90
+ { pubkey: positionPda, isSigner: false, isWritable: true },
91
+ ];
92
+ if (whitelistPda) {
93
+ keys.push({ pubkey: whitelistPda, isSigner: false, isWritable: false });
94
+ }
95
+ keys.push({ pubkey: userPubkey, isSigner: true, isWritable: true }, { pubkey: SystemProgram.programId, isSigner: false, isWritable: false });
96
+ instruction = new TransactionInstruction({
97
+ programId: PROGRAM_ID,
98
+ keys,
99
+ data,
100
+ });
101
+ }
102
+ const transaction = new Transaction();
103
+ transaction.add(instruction);
104
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
105
+ transaction.recentBlockhash = blockhash;
106
+ transaction.feePayer = userPubkey;
107
+ const serializedTx = transaction.serialize({
108
+ requireAllSignatures: false,
109
+ verifySignatures: false,
110
+ }).toString('base64');
111
+ return {
112
+ transaction,
113
+ serializedTx,
114
+ positionPda: positionPda.toBase58(),
115
+ marketId: params.marketId,
116
+ };
117
+ }
118
+ // =============================================================================
119
+ // FETCH AND BUILD RACE BET
120
+ // =============================================================================
121
+ /**
122
+ * Fetch race market data and build bet transaction
123
+ */
124
+ export async function fetchAndBuildRaceBetTransaction(params) {
125
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
126
+ try {
127
+ const raceMarketPubkey = new PublicKey(params.raceMarketPda);
128
+ const accountInfo = await conn.getAccountInfo(raceMarketPubkey);
129
+ if (!accountInfo) {
130
+ return {
131
+ transaction: null,
132
+ marketId: 0n,
133
+ error: 'Race market not found',
134
+ };
135
+ }
136
+ const data = accountInfo.data;
137
+ // Extract market_id (first field after discriminator)
138
+ const marketId = data.readBigUInt64LE(8);
139
+ // Check access_gate (need to navigate through struct)
140
+ // For now, assume public
141
+ const whitelistRequired = false;
142
+ const result = await buildRaceBetTransaction({
143
+ raceMarketPda: params.raceMarketPda,
144
+ marketId,
145
+ outcomeIndex: params.outcomeIndex,
146
+ amountSol: params.amountSol,
147
+ userWallet: params.userWallet,
148
+ whitelistRequired,
149
+ affiliatePda: params.affiliatePda,
150
+ connection: conn,
151
+ });
152
+ return {
153
+ transaction: result,
154
+ marketId,
155
+ };
156
+ }
157
+ catch (err) {
158
+ return {
159
+ transaction: null,
160
+ marketId: 0n,
161
+ error: err instanceof Error ? err.message : 'Unknown error',
162
+ };
163
+ }
164
+ }
165
+ // =============================================================================
166
+ // CLAIM RACE WINNINGS
167
+ // =============================================================================
168
+ /**
169
+ * Build claim_race_winnings_sol transaction
170
+ */
171
+ export async function buildClaimRaceWinningsTransaction(params) {
172
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
173
+ const raceMarketPubkey = new PublicKey(params.raceMarketPda);
174
+ const positionPubkey = new PublicKey(params.positionPda);
175
+ const userPubkey = new PublicKey(params.userWallet);
176
+ const data = CLAIM_RACE_WINNINGS_SOL_DISCRIMINATOR;
177
+ const keys = [
178
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
179
+ { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },
180
+ { pubkey: positionPubkey, isSigner: false, isWritable: true },
181
+ { pubkey: SOL_TREASURY_PDA, isSigner: false, isWritable: true },
182
+ { pubkey: userPubkey, isSigner: true, isWritable: true },
183
+ { pubkey: SystemProgram.programId, isSigner: false, isWritable: false },
184
+ ];
185
+ const instruction = new TransactionInstruction({
186
+ programId: PROGRAM_ID,
187
+ keys,
188
+ data,
189
+ });
190
+ const transaction = new Transaction();
191
+ transaction.add(instruction);
192
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
193
+ transaction.recentBlockhash = blockhash;
194
+ transaction.feePayer = userPubkey;
195
+ const serializedTx = transaction.serialize({
196
+ requireAllSignatures: false,
197
+ verifySignatures: false,
198
+ }).toString('base64');
199
+ return {
200
+ transaction,
201
+ serializedTx,
202
+ };
203
+ }
204
+ // =============================================================================
205
+ // CLAIM RACE REFUND
206
+ // =============================================================================
207
+ /**
208
+ * Build claim_race_refund transaction
209
+ */
210
+ export async function buildClaimRaceRefundTransaction(params) {
211
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
212
+ const raceMarketPubkey = new PublicKey(params.raceMarketPda);
213
+ const positionPubkey = new PublicKey(params.positionPda);
214
+ const userPubkey = new PublicKey(params.userWallet);
215
+ const data = CLAIM_RACE_REFUND_DISCRIMINATOR;
216
+ // IDL Accounts: race_market, position, user, system_program (NO config!)
217
+ const keys = [
218
+ { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },
219
+ { pubkey: positionPubkey, isSigner: false, isWritable: true },
220
+ { pubkey: userPubkey, isSigner: true, isWritable: true },
221
+ { pubkey: SystemProgram.programId, isSigner: false, isWritable: false },
222
+ ];
223
+ const instruction = new TransactionInstruction({
224
+ programId: PROGRAM_ID,
225
+ keys,
226
+ data,
227
+ });
228
+ const transaction = new Transaction();
229
+ transaction.add(instruction);
230
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
231
+ transaction.recentBlockhash = blockhash;
232
+ transaction.feePayer = userPubkey;
233
+ const serializedTx = transaction.serialize({
234
+ requireAllSignatures: false,
235
+ verifySignatures: false,
236
+ }).toString('base64');
237
+ return {
238
+ transaction,
239
+ serializedTx,
240
+ };
241
+ }
242
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"race-transaction.js","sourceRoot":"","sources":["../../src/builders/race-transaction.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Race Market Transaction Builders\n *\n * Builds unsigned transactions for:\n * - Placing bets on race (multi-outcome) markets\n * - Claiming race winnings\n * - Claiming race refunds\n */\nimport {\n  Connection,\n  PublicKey,\n  Transaction,\n  TransactionInstruction,\n  SystemProgram,\n} from '@solana/web3.js';\nimport {\n  PROGRAM_ID,\n  CONFIG_PDA,\n  SOL_TREASURY_PDA,\n  SEEDS,\n  RPC_ENDPOINT,\n  solToLamports,\n} from '../config.js';\n\n// =============================================================================\n// INSTRUCTION DISCRIMINATORS\n// =============================================================================\n\n// Correct discriminators: sha256(\"global:<instruction_name>\")[0..8]\nconst BET_ON_RACE_SOL_DISCRIMINATOR = Buffer.from([195, 181, 151, 159, 105, 100, 234, 244]);\nconst BET_ON_RACE_SOL_WITH_AFFILIATE_DISCRIMINATOR = Buffer.from([26, 224, 14, 181, 67, 52, 24, 0]);\nconst CLAIM_RACE_WINNINGS_SOL_DISCRIMINATOR = Buffer.from([46, 120, 202, 194, 126, 72, 22, 52]);\nconst CLAIM_RACE_REFUND_DISCRIMINATOR = Buffer.from([174, 101, 101, 227, 171, 69, 173, 243]);\n\n// =============================================================================\n// TYPES\n// =============================================================================\n\nexport interface RaceBetTransactionResult {\n  transaction: Transaction;\n  serializedTx: string;\n  positionPda: string;\n  marketId: bigint;\n}\n\n// =============================================================================\n// RACE POSITION PDA\n// =============================================================================\n\nfunction deriveRacePositionPda(marketId: bigint, user: PublicKey): PublicKey {\n  const marketIdBuffer = Buffer.alloc(8);\n  marketIdBuffer.writeBigUInt64LE(marketId);\n  const [pda] = PublicKey.findProgramAddressSync(\n    [SEEDS.RACE_POSITION, marketIdBuffer, user.toBuffer()],\n    PROGRAM_ID\n  );\n  return pda;\n}\n\nfunction deriveRaceWhitelistPda(raceMarketPda: PublicKey): PublicKey {\n  const [pda] = PublicKey.findProgramAddressSync(\n    [SEEDS.RACE_WHITELIST, raceMarketPda.toBuffer()],\n    PROGRAM_ID\n  );\n  return pda;\n}\n\nfunction deriveRaceReferralPda(user: PublicKey): PublicKey {\n  const [pda] = PublicKey.findProgramAddressSync(\n    [Buffer.from('race_referral'), user.toBuffer()],\n    PROGRAM_ID\n  );\n  return pda;\n}\n\n// =============================================================================\n// BET ON RACE OUTCOME\n// =============================================================================\n\n/**\n * Build bet_on_race_outcome_sol transaction\n */\nexport async function buildRaceBetTransaction(params: {\n  raceMarketPda: string;\n  marketId: bigint;\n  outcomeIndex: number;\n  amountSol: number;\n  userWallet: string;\n  whitelistRequired?: boolean;\n  affiliatePda?: string;\n  connection?: Connection;\n}): Promise<RaceBetTransactionResult> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const raceMarketPubkey = new PublicKey(params.raceMarketPda);\n  const userPubkey = new PublicKey(params.userWallet);\n\n  // Derive position PDA\n  const positionPda = deriveRacePositionPda(params.marketId, userPubkey);\n\n  // Optional whitelist PDA\n  const whitelistPda = params.whitelistRequired\n    ? deriveRaceWhitelistPda(raceMarketPubkey)\n    : null;\n\n  // Amount in lamports\n  const amountLamports = solToLamports(params.amountSol);\n\n  let instruction: TransactionInstruction;\n\n  if (params.affiliatePda) {\n    // With affiliate\n    const affiliatePubkey = new PublicKey(params.affiliatePda);\n    const raceReferralPda = deriveRaceReferralPda(userPubkey);\n\n    // Instruction data: discriminator + outcome_index (u8) + amount (u64)\n    const data = Buffer.alloc(17);\n    BET_ON_RACE_SOL_WITH_AFFILIATE_DISCRIMINATOR.copy(data, 0);\n    data.writeUInt8(params.outcomeIndex, 8);\n    data.writeBigUInt64LE(amountLamports, 9);\n\n    const keys = [\n      { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n      { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },\n      { pubkey: positionPda, isSigner: false, isWritable: true },\n      { pubkey: affiliatePubkey, isSigner: false, isWritable: true },\n      { pubkey: raceReferralPda, isSigner: false, isWritable: true },\n    ];\n\n    if (whitelistPda) {\n      keys.push({ pubkey: whitelistPda, isSigner: false, isWritable: false });\n    }\n\n    keys.push(\n      { pubkey: userPubkey, isSigner: true, isWritable: true },\n      { pubkey: SystemProgram.programId, isSigner: false, isWritable: false }\n    );\n\n    instruction = new TransactionInstruction({\n      programId: PROGRAM_ID,\n      keys,\n      data,\n    });\n  } else {\n    // Without affiliate\n    const data = Buffer.alloc(17);\n    BET_ON_RACE_SOL_DISCRIMINATOR.copy(data, 0);\n    data.writeUInt8(params.outcomeIndex, 8);\n    data.writeBigUInt64LE(amountLamports, 9);\n\n    const keys = [\n      { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n      { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },\n      { pubkey: positionPda, isSigner: false, isWritable: true },\n    ];\n\n    if (whitelistPda) {\n      keys.push({ pubkey: whitelistPda, isSigner: false, isWritable: false });\n    }\n\n    keys.push(\n      { pubkey: userPubkey, isSigner: true, isWritable: true },\n      { pubkey: SystemProgram.programId, isSigner: false, isWritable: false }\n    );\n\n    instruction = new TransactionInstruction({\n      programId: PROGRAM_ID,\n      keys,\n      data,\n    });\n  }\n\n  const transaction = new Transaction();\n  transaction.add(instruction);\n\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = userPubkey;\n\n  const serializedTx = transaction.serialize({\n    requireAllSignatures: false,\n    verifySignatures: false,\n  }).toString('base64');\n\n  return {\n    transaction,\n    serializedTx,\n    positionPda: positionPda.toBase58(),\n    marketId: params.marketId,\n  };\n}\n\n// =============================================================================\n// FETCH AND BUILD RACE BET\n// =============================================================================\n\n/**\n * Fetch race market data and build bet transaction\n */\nexport async function fetchAndBuildRaceBetTransaction(params: {\n  raceMarketPda: string;\n  outcomeIndex: number;\n  amountSol: number;\n  userWallet: string;\n  affiliatePda?: string;\n  connection?: Connection;\n}): Promise<{\n  transaction: RaceBetTransactionResult | null;\n  marketId: bigint;\n  error?: string;\n}> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n\n  try {\n    const raceMarketPubkey = new PublicKey(params.raceMarketPda);\n    const accountInfo = await conn.getAccountInfo(raceMarketPubkey);\n\n    if (!accountInfo) {\n      return {\n        transaction: null,\n        marketId: 0n,\n        error: 'Race market not found',\n      };\n    }\n\n    const data = accountInfo.data;\n\n    // Extract market_id (first field after discriminator)\n    const marketId = data.readBigUInt64LE(8);\n\n    // Check access_gate (need to navigate through struct)\n    // For now, assume public\n    const whitelistRequired = false;\n\n    const result = await buildRaceBetTransaction({\n      raceMarketPda: params.raceMarketPda,\n      marketId,\n      outcomeIndex: params.outcomeIndex,\n      amountSol: params.amountSol,\n      userWallet: params.userWallet,\n      whitelistRequired,\n      affiliatePda: params.affiliatePda,\n      connection: conn,\n    });\n\n    return {\n      transaction: result,\n      marketId,\n    };\n  } catch (err) {\n    return {\n      transaction: null,\n      marketId: 0n,\n      error: err instanceof Error ? err.message : 'Unknown error',\n    };\n  }\n}\n\n// =============================================================================\n// CLAIM RACE WINNINGS\n// =============================================================================\n\n/**\n * Build claim_race_winnings_sol transaction\n */\nexport async function buildClaimRaceWinningsTransaction(params: {\n  raceMarketPda: string;\n  positionPda: string;\n  userWallet: string;\n  connection?: Connection;\n}): Promise<{\n  transaction: Transaction;\n  serializedTx: string;\n}> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const raceMarketPubkey = new PublicKey(params.raceMarketPda);\n  const positionPubkey = new PublicKey(params.positionPda);\n  const userPubkey = new PublicKey(params.userWallet);\n\n  const data = CLAIM_RACE_WINNINGS_SOL_DISCRIMINATOR;\n\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },\n    { pubkey: positionPubkey, isSigner: false, isWritable: true },\n    { pubkey: SOL_TREASURY_PDA, isSigner: false, isWritable: true },\n    { pubkey: userPubkey, isSigner: true, isWritable: true },\n    { pubkey: SystemProgram.programId, isSigner: false, isWritable: false },\n  ];\n\n  const instruction = new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n\n  const transaction = new Transaction();\n  transaction.add(instruction);\n\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = userPubkey;\n\n  const serializedTx = transaction.serialize({\n    requireAllSignatures: false,\n    verifySignatures: false,\n  }).toString('base64');\n\n  return {\n    transaction,\n    serializedTx,\n  };\n}\n\n// =============================================================================\n// CLAIM RACE REFUND\n// =============================================================================\n\n/**\n * Build claim_race_refund transaction\n */\nexport async function buildClaimRaceRefundTransaction(params: {\n  raceMarketPda: string;\n  positionPda: string;\n  userWallet: string;\n  connection?: Connection;\n}): Promise<{\n  transaction: Transaction;\n  serializedTx: string;\n}> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const raceMarketPubkey = new PublicKey(params.raceMarketPda);\n  const positionPubkey = new PublicKey(params.positionPda);\n  const userPubkey = new PublicKey(params.userWallet);\n\n  const data = CLAIM_RACE_REFUND_DISCRIMINATOR;\n\n  // IDL Accounts: race_market, position, user, system_program (NO config!)\n  const keys = [\n    { pubkey: raceMarketPubkey, isSigner: false, isWritable: true },\n    { pubkey: positionPubkey, isSigner: false, isWritable: true },\n    { pubkey: userPubkey, isSigner: true, isWritable: true },\n    { pubkey: SystemProgram.programId, isSigner: false, isWritable: false },\n  ];\n\n  const instruction = new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n\n  const transaction = new Transaction();\n  transaction.add(instruction);\n\n  const { blockhash } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = userPubkey;\n\n  const serializedTx = transaction.serialize({\n    requireAllSignatures: false,\n    verifySignatures: false,\n  }).toString('base64');\n\n  return {\n    transaction,\n    serializedTx,\n  };\n}\n"]}