@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.
- package/README.md +294 -0
- package/dist/__tests__/full-test.d.ts +1 -0
- package/dist/__tests__/full-test.js +291 -0
- package/dist/builders/affiliate-transaction.d.ts +41 -0
- package/dist/builders/affiliate-transaction.js +123 -0
- package/dist/builders/bet-transaction.d.ts +70 -0
- package/dist/builders/bet-transaction.js +323 -0
- package/dist/builders/claim-transaction.d.ts +57 -0
- package/dist/builders/claim-transaction.js +196 -0
- package/dist/builders/creator-transaction.d.ts +49 -0
- package/dist/builders/creator-transaction.js +177 -0
- package/dist/builders/dispute-transaction.d.ts +81 -0
- package/dist/builders/dispute-transaction.js +215 -0
- package/dist/builders/index.d.ts +14 -0
- package/dist/builders/index.js +15 -0
- package/dist/builders/market-creation-tx.d.ts +65 -0
- package/dist/builders/market-creation-tx.js +362 -0
- package/dist/builders/market-management-transaction.d.ts +85 -0
- package/dist/builders/market-management-transaction.js +239 -0
- package/dist/builders/race-transaction.d.ts +67 -0
- package/dist/builders/race-transaction.js +242 -0
- package/dist/builders/resolution-transaction.d.ts +108 -0
- package/dist/builders/resolution-transaction.js +250 -0
- package/dist/builders/whitelist-transaction.d.ts +72 -0
- package/dist/builders/whitelist-transaction.js +179 -0
- package/dist/config.d.ts +138 -0
- package/dist/config.js +307 -0
- package/dist/handlers/agent-network.d.ts +81 -0
- package/dist/handlers/agent-network.js +332 -0
- package/dist/handlers/claims.d.ts +47 -0
- package/dist/handlers/claims.js +218 -0
- package/dist/handlers/market-creation.d.ts +154 -0
- package/dist/handlers/market-creation.js +290 -0
- package/dist/handlers/markets.d.ts +41 -0
- package/dist/handlers/markets.js +319 -0
- package/dist/handlers/positions.d.ts +40 -0
- package/dist/handlers/positions.js +244 -0
- package/dist/handlers/quote.d.ts +33 -0
- package/dist/handlers/quote.js +144 -0
- package/dist/handlers/race-markets.d.ts +54 -0
- package/dist/handlers/race-markets.js +308 -0
- package/dist/handlers/resolution.d.ts +43 -0
- package/dist/handlers/resolution.js +194 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +109 -0
- package/dist/resources.d.ts +13 -0
- package/dist/resources.js +336 -0
- package/dist/tools.d.ts +3109 -0
- package/dist/tools.js +1956 -0
- package/dist/validation/bet-rules.d.ts +82 -0
- package/dist/validation/bet-rules.js +276 -0
- package/dist/validation/creation-rules.d.ts +69 -0
- package/dist/validation/creation-rules.js +302 -0
- package/dist/validation/index.d.ts +6 -0
- package/dist/validation/index.js +7 -0
- package/dist/validation/market-rules.d.ts +60 -0
- package/dist/validation/market-rules.js +237 -0
- package/dist/validation/parimutuel-rules.d.ts +117 -0
- package/dist/validation/parimutuel-rules.js +270 -0
- 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":"AAAA;;;;;;GAMG;AACH,OAAO,EACL,UAAU,EACV,SAAS,EACT,WAAW,EACX,sBAAsB,GACvB,MAAM,iBAAiB,CAAC;AACzB,OAAO,EACL,UAAU,EACV,UAAU,EACV,YAAY,GACb,MAAM,cAAc,CAAC;AAEtB,gFAAgF;AAChF,iBAAiB;AACjB,gFAAgF;AAEhF,MAAM,cAAc,GAAG;IACrB,YAAY,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC;IAChE,iBAAiB,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,GAAG,EAAE,EAAE,CAAC,CAAC;IACrE,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC;IACjE,kBAAkB,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC;IACxE,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,GAAG,EAAE,EAAE,CAAC,CAAC;IACjE,WAAW,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,EAAE,EAAE,GAAG,EAAE,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,GAAG,CAAC,CAAC;CAC/D,CAAC;AAEF,gFAAgF;AAChF,4BAA4B;AAC5B,gFAAgF;AAEhF;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,2BAA2B,CAAC,MAIjD;IACC,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,IAAI,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAC5E,MAAM,YAAY,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IACrD,MAAM,YAAY,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;IAExD,MAAM,IAAI,GAAG,cAAc,CAAC,YAAY,CAAC;IAEzC,MAAM,IAAI,GAAG;QACX,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE;QAC1D,EAAE,MAAM,EAAE,YAAY,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE;QAC3D,EAAE,MAAM,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE;KAC5D,CAAC;IAEF,MAAM,WAAW,GAAG,IAAI,sBAAsB,CAAC;QAC7C,SAAS,EAAE,UAAU;QACrB,IAAI;QACJ,IAAI;KACL,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,IAAI,WAAW,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;IACvD,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;IACjE,WAAW,CAAC,eAAe,GAAG,SAAS,CAAC;IACxC,WAAW,CAAC,QAAQ,GAAG,YAAY,CAAC;IAEpC,OAAO;QACL,WAAW;QACX,YAAY,EAAE,WAAW,CAAC,SAAS,CAAC,EAAE,oBAAoB,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;KACjH,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,4BAA4B,CAAC,MAMlD;IACC,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,IAAI,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAC5E,MAAM,YAAY,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IACrD,MAAM,YAAY,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;IAExD,6FAA6F;IAC7F,MAAM,iBAAiB,GAAG,MAAM,CAAC,iBAAiB,KAAK,SAAS,CAAC;IACjE,MAAM,QAAQ,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACzD,MAAM,IAAI,GAAG,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;IAEpC,IAAI,MAAM,GAAG,CAAC,CAAC;IACf,cAAc,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;IAChD,MAAM,IAAI,CAAC,CAAC;IAEZ,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,MAAM,CAAC,cAAc,CAAC,EAAE,MAAM,CAAC,CAAC;IAC5D,MAAM,IAAI,CAAC,CAAC;IAEZ,IAAI,iBAAiB,EAAE,CAAC;QACtB,IAAI,CAAC,UAAU,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,OAAO;QACnC,MAAM,IAAI,CAAC,CAAC;QACZ,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,MAAM,CAAC,iBAAkB,CAAC,EAAE,MAAM,CAAC,CAAC;IAClE,CAAC;SAAM,CAAC;QACN,IAAI,CAAC,UAAU,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,OAAO;IACrC,CAAC;IAED,MAAM,IAAI,GAAG;QACX,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE;QAC1D,EAAE,MAAM,EAAE,YAAY,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE;QAC3D,EAAE,MAAM,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE;KAC5D,CAAC;IAEF,MAAM,WAAW,GAAG,IAAI,sBAAsB,CAAC;QAC7C,SAAS,EAAE,UAAU;QACrB,IAAI;QACJ,IAAI;KACL,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,IAAI,WAAW,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;IACvD,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;IACjE,WAAW,CAAC,eAAe,GAAG,SAAS,CAAC;IACxC,WAAW,CAAC,QAAQ,GAAG,YAAY,CAAC;IAEpC,OAAO;QACL,WAAW;QACX,YAAY,EAAE,WAAW,CAAC,SAAS,CAAC,EAAE,oBAAoB,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;KACjH,CAAC;AACJ,CAAC;AAED,gFAAgF;AAChF,yBAAyB;AACzB,gFAAgF;AAEhF;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,+BAA+B,CAAC,MAIrD;IACC,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,IAAI,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAC5E,MAAM,gBAAgB,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;IAC7D,MAAM,YAAY,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;IAExD,MAAM,IAAI,GAAG,cAAc,CAAC,iBAAiB,CAAC;IAE9C,MAAM,IAAI,GAAG;QACX,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE;QAC1D,EAAE,MAAM,EAAE,gBAAgB,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE;QAC/D,EAAE,MAAM,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE;KAC5D,CAAC;IAEF,MAAM,WAAW,GAAG,IAAI,sBAAsB,CAAC;QAC7C,SAAS,EAAE,UAAU;QACrB,IAAI;QACJ,IAAI;KACL,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,IAAI,WAAW,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;IACvD,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;IACjE,WAAW,CAAC,eAAe,GAAG,SAAS,CAAC;IACxC,WAAW,CAAC,QAAQ,GAAG,YAAY,CAAC;IAEpC,OAAO;QACL,WAAW;QACX,YAAY,EAAE,WAAW,CAAC,SAAS,CAAC,EAAE,oBAAoB,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;KACjH,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,gCAAgC,CAAC,MAMtD;IACC,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,IAAI,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAC5E,MAAM,gBAAgB,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;IAC7D,MAAM,YAAY,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;IAExD,MAAM,iBAAiB,GAAG,MAAM,CAAC,iBAAiB,KAAK,SAAS,CAAC;IACjE,MAAM,QAAQ,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACzD,MAAM,IAAI,GAAG,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;IAEpC,IAAI,MAAM,GAAG,CAAC,CAAC;IACf,cAAc,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;IACrD,MAAM,IAAI,CAAC,CAAC;IAEZ,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,MAAM,CAAC,cAAc,CAAC,EAAE,MAAM,CAAC,CAAC;IAC5D,MAAM,IAAI,CAAC,CAAC;IAEZ,IAAI,iBAAiB,EAAE,CAAC;QACtB,IAAI,CAAC,UAAU,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,OAAO;QACnC,MAAM,IAAI,CAAC,CAAC;QACZ,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,MAAM,CAAC,iBAAkB,CAAC,EAAE,MAAM,CAAC,CAAC;IAClE,CAAC;SAAM,CAAC;QACN,IAAI,CAAC,UAAU,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,OAAO;IACrC,CAAC;IAED,MAAM,IAAI,GAAG;QACX,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE;QAC1D,EAAE,MAAM,EAAE,gBAAgB,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE;QAC/D,EAAE,MAAM,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE;KAC5D,CAAC;IAEF,MAAM,WAAW,GAAG,IAAI,sBAAsB,CAAC;QAC7C,SAAS,EAAE,UAAU;QACrB,IAAI;QACJ,IAAI;KACL,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,IAAI,WAAW,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;IACvD,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;IACjE,WAAW,CAAC,eAAe,GAAG,SAAS,CAAC;IACxC,WAAW,CAAC,QAAQ,GAAG,YAAY,CAAC;IAEpC,OAAO;QACL,WAAW;QACX,YAAY,EAAE,WAAW,CAAC,SAAS,CAAC,EAAE,oBAAoB,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;KACjH,CAAC;AACJ,CAAC;AAED,gFAAgF;AAChF,gCAAgC;AAChC,gFAAgF;AAEhF;;;;GAIG;AACH,MAAM,CAAC,KAAK,UAAU,4BAA4B,CAAC,MAKlD;IACC,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,IAAI,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAC5E,MAAM,YAAY,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IACrD,MAAM,eAAe,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;IAE9D,oDAAoD;IACpD,sDAAsD;IACtD,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IACvD,MAAM,IAAI,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC;IACtD,cAAc,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;IAC3C,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAC1C,WAAW,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;IAE3B,MAAM,IAAI,GAAG;QACX,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE;QAC1D,EAAE,MAAM,EAAE,YAAY,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE;QAC3D,EAAE,MAAM,EAAE,eAAe,EAAE,QAAQ,EAAE,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE;KAC/D,CAAC;IAEF,MAAM,WAAW,GAAG,IAAI,sBAAsB,CAAC;QAC7C,SAAS,EAAE,UAAU;QACrB,IAAI;QACJ,IAAI;KACL,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,IAAI,WAAW,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;IACvD,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;IACjE,WAAW,CAAC,eAAe,GAAG,SAAS,CAAC;IACxC,WAAW,CAAC,QAAQ,GAAG,eAAe,CAAC;IAEvC,OAAO;QACL,WAAW;QACX,YAAY,EAAE,WAAW,CAAC,SAAS,CAAC,EAAE,oBAAoB,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;KACjH,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,0BAA0B,CAAC,MAKhD;IACC,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,IAAI,IAAI,UAAU,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAC5E,MAAM,gBAAgB,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;IAC7D,MAAM,eAAe,GAAG,IAAI,SAAS,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;IAE9D,oDAAoD;IACpD,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IACvD,MAAM,IAAI,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC;IACtD,cAAc,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;IACzC,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAC1C,WAAW,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;IAE3B,MAAM,IAAI,GAAG;QACX,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,KAAK,EAAE;QAC1D,EAAE,MAAM,EAAE,gBAAgB,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE;QAC/D,EAAE,MAAM,EAAE,eAAe,EAAE,QAAQ,EAAE,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE;KAC/D,CAAC;IAEF,MAAM,WAAW,GAAG,IAAI,sBAAsB,CAAC;QAC7C,SAAS,EAAE,UAAU;QACrB,IAAI;QACJ,IAAI;KACL,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,IAAI,WAAW,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;IACvD,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;IACjE,WAAW,CAAC,eAAe,GAAG,SAAS,CAAC;IACxC,WAAW,CAAC,QAAQ,GAAG,eAAe,CAAC;IAEvC,OAAO;QACL,WAAW;QACX,YAAY,EAAE,WAAW,CAAC,SAAS,CAAC,EAAE,oBAAoB,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;KACjH,CAAC;AACJ,CAAC","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"]}
|