@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,41 @@
1
+ /**
2
+ * Affiliate Transaction Builders
3
+ *
4
+ * Builds unsigned transactions for:
5
+ * - Registering as an affiliate
6
+ * - Toggling affiliate active status
7
+ */
8
+ import { Connection, Transaction } from '@solana/web3.js';
9
+ export interface RegisterAffiliateResult {
10
+ transaction: Transaction;
11
+ serializedTx: string;
12
+ affiliatePda: string;
13
+ code: string;
14
+ }
15
+ /**
16
+ * Build register_affiliate transaction
17
+ *
18
+ * Registers a new affiliate with a unique code.
19
+ * The code must be 3-16 alphanumeric characters.
20
+ */
21
+ export declare function buildRegisterAffiliateTransaction(params: {
22
+ code: string;
23
+ userWallet: string;
24
+ connection?: Connection;
25
+ }): Promise<RegisterAffiliateResult>;
26
+ /**
27
+ * Build toggle_affiliate transaction
28
+ *
29
+ * Toggles an affiliate's active status (active/inactive).
30
+ */
31
+ export declare function buildToggleAffiliateTransaction(params: {
32
+ code: string;
33
+ active: boolean;
34
+ userWallet: string;
35
+ connection?: Connection;
36
+ }): Promise<{
37
+ transaction: Transaction;
38
+ serializedTx: string;
39
+ affiliatePda: string;
40
+ newStatus: boolean;
41
+ }>;
@@ -0,0 +1,123 @@
1
+ /**
2
+ * Affiliate Transaction Builders
3
+ *
4
+ * Builds unsigned transactions for:
5
+ * - Registering as an affiliate
6
+ * - Toggling affiliate active status
7
+ */
8
+ import { Connection, PublicKey, Transaction, TransactionInstruction, SystemProgram, } from '@solana/web3.js';
9
+ import { PROGRAM_ID, CONFIG_PDA, SEEDS, RPC_ENDPOINT, } from '../config.js';
10
+ // =============================================================================
11
+ // INSTRUCTION DISCRIMINATORS
12
+ // =============================================================================
13
+ // Correct discriminators: sha256("global:<instruction_name>")[0..8]
14
+ const REGISTER_AFFILIATE_DISCRIMINATOR = Buffer.from([87, 121, 99, 184, 126, 63, 103, 217]);
15
+ const TOGGLE_AFFILIATE_DISCRIMINATOR = Buffer.from([47, 161, 133, 19, 172, 44, 43, 194]);
16
+ // =============================================================================
17
+ // REGISTER AFFILIATE
18
+ // =============================================================================
19
+ /**
20
+ * Build register_affiliate transaction
21
+ *
22
+ * Registers a new affiliate with a unique code.
23
+ * The code must be 3-16 alphanumeric characters.
24
+ */
25
+ export async function buildRegisterAffiliateTransaction(params) {
26
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
27
+ const userPubkey = new PublicKey(params.userWallet);
28
+ // Validate code
29
+ if (!isValidCode(params.code)) {
30
+ throw new Error('Invalid affiliate code. Must be 3-16 alphanumeric characters.');
31
+ }
32
+ // Derive affiliate PDA
33
+ const [affiliatePda] = PublicKey.findProgramAddressSync([SEEDS.AFFILIATE, Buffer.from(params.code)], PROGRAM_ID);
34
+ // Instruction data: discriminator + code (as string)
35
+ // String encoding: 4-byte length prefix + UTF-8 bytes
36
+ const codeBytes = Buffer.from(params.code, 'utf8');
37
+ const data = Buffer.alloc(8 + 4 + codeBytes.length);
38
+ REGISTER_AFFILIATE_DISCRIMINATOR.copy(data, 0);
39
+ data.writeUInt32LE(codeBytes.length, 8);
40
+ codeBytes.copy(data, 12);
41
+ // Accounts for register_affiliate:
42
+ // config, affiliate, owner, system_program
43
+ const keys = [
44
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
45
+ { pubkey: affiliatePda, isSigner: false, isWritable: true },
46
+ { pubkey: userPubkey, isSigner: true, isWritable: true },
47
+ { pubkey: SystemProgram.programId, isSigner: false, isWritable: false },
48
+ ];
49
+ const instruction = new TransactionInstruction({
50
+ programId: PROGRAM_ID,
51
+ keys,
52
+ data,
53
+ });
54
+ const transaction = new Transaction();
55
+ transaction.add(instruction);
56
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
57
+ transaction.recentBlockhash = blockhash;
58
+ transaction.feePayer = userPubkey;
59
+ const serializedTx = transaction.serialize({
60
+ requireAllSignatures: false,
61
+ verifySignatures: false,
62
+ }).toString('base64');
63
+ return {
64
+ transaction,
65
+ serializedTx,
66
+ affiliatePda: affiliatePda.toBase58(),
67
+ code: params.code,
68
+ };
69
+ }
70
+ // =============================================================================
71
+ // TOGGLE AFFILIATE
72
+ // =============================================================================
73
+ /**
74
+ * Build toggle_affiliate transaction
75
+ *
76
+ * Toggles an affiliate's active status (active/inactive).
77
+ */
78
+ export async function buildToggleAffiliateTransaction(params) {
79
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
80
+ const userPubkey = new PublicKey(params.userWallet);
81
+ // Derive affiliate PDA
82
+ const [affiliatePda] = PublicKey.findProgramAddressSync([SEEDS.AFFILIATE, Buffer.from(params.code)], PROGRAM_ID);
83
+ // Instruction data: discriminator + is_active (bool)
84
+ const data = Buffer.alloc(9);
85
+ TOGGLE_AFFILIATE_DISCRIMINATOR.copy(data, 0);
86
+ data.writeUInt8(params.active ? 1 : 0, 8);
87
+ // Accounts for toggle_affiliate:
88
+ // config, affiliate, owner
89
+ const keys = [
90
+ { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },
91
+ { pubkey: affiliatePda, isSigner: false, isWritable: true },
92
+ { pubkey: userPubkey, isSigner: true, isWritable: false },
93
+ ];
94
+ const instruction = new TransactionInstruction({
95
+ programId: PROGRAM_ID,
96
+ keys,
97
+ data,
98
+ });
99
+ const transaction = new Transaction();
100
+ transaction.add(instruction);
101
+ const { blockhash } = await conn.getLatestBlockhash('finalized');
102
+ transaction.recentBlockhash = blockhash;
103
+ transaction.feePayer = userPubkey;
104
+ const serializedTx = transaction.serialize({
105
+ requireAllSignatures: false,
106
+ verifySignatures: false,
107
+ }).toString('base64');
108
+ return {
109
+ transaction,
110
+ serializedTx,
111
+ affiliatePda: affiliatePda.toBase58(),
112
+ newStatus: params.active,
113
+ };
114
+ }
115
+ // =============================================================================
116
+ // HELPERS
117
+ // =============================================================================
118
+ function isValidCode(code) {
119
+ if (code.length < 3 || code.length > 16)
120
+ return false;
121
+ return /^[a-zA-Z0-9_]+$/.test(code);
122
+ }
123
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"affiliate-transaction.js","sourceRoot":"","sources":["../../src/builders/affiliate-transaction.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Affiliate Transaction Builders\n *\n * Builds unsigned transactions for:\n * - Registering as an affiliate\n * - Toggling affiliate active status\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  SEEDS,\n  RPC_ENDPOINT,\n} from '../config.js';\n\n// =============================================================================\n// INSTRUCTION DISCRIMINATORS\n// =============================================================================\n\n// Correct discriminators: sha256(\"global:<instruction_name>\")[0..8]\nconst REGISTER_AFFILIATE_DISCRIMINATOR = Buffer.from([87, 121, 99, 184, 126, 63, 103, 217]);\nconst TOGGLE_AFFILIATE_DISCRIMINATOR = Buffer.from([47, 161, 133, 19, 172, 44, 43, 194]);\n\n// =============================================================================\n// TYPES\n// =============================================================================\n\nexport interface RegisterAffiliateResult {\n  transaction: Transaction;\n  serializedTx: string;\n  affiliatePda: string;\n  code: string;\n}\n\n// =============================================================================\n// REGISTER AFFILIATE\n// =============================================================================\n\n/**\n * Build register_affiliate transaction\n *\n * Registers a new affiliate with a unique code.\n * The code must be 3-16 alphanumeric characters.\n */\nexport async function buildRegisterAffiliateTransaction(params: {\n  code: string;\n  userWallet: string;\n  connection?: Connection;\n}): Promise<RegisterAffiliateResult> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const userPubkey = new PublicKey(params.userWallet);\n\n  // Validate code\n  if (!isValidCode(params.code)) {\n    throw new Error('Invalid affiliate code. Must be 3-16 alphanumeric characters.');\n  }\n\n  // Derive affiliate PDA\n  const [affiliatePda] = PublicKey.findProgramAddressSync(\n    [SEEDS.AFFILIATE, Buffer.from(params.code)],\n    PROGRAM_ID\n  );\n\n  // Instruction data: discriminator + code (as string)\n  // String encoding: 4-byte length prefix + UTF-8 bytes\n  const codeBytes = Buffer.from(params.code, 'utf8');\n  const data = Buffer.alloc(8 + 4 + codeBytes.length);\n  REGISTER_AFFILIATE_DISCRIMINATOR.copy(data, 0);\n  data.writeUInt32LE(codeBytes.length, 8);\n  codeBytes.copy(data, 12);\n\n  // Accounts for register_affiliate:\n  // config, affiliate, owner, system_program\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: affiliatePda, 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    affiliatePda: affiliatePda.toBase58(),\n    code: params.code,\n  };\n}\n\n// =============================================================================\n// TOGGLE AFFILIATE\n// =============================================================================\n\n/**\n * Build toggle_affiliate transaction\n *\n * Toggles an affiliate's active status (active/inactive).\n */\nexport async function buildToggleAffiliateTransaction(params: {\n  code: string;\n  active: boolean;\n  userWallet: string;\n  connection?: Connection;\n}): Promise<{\n  transaction: Transaction;\n  serializedTx: string;\n  affiliatePda: string;\n  newStatus: boolean;\n}> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n  const userPubkey = new PublicKey(params.userWallet);\n\n  // Derive affiliate PDA\n  const [affiliatePda] = PublicKey.findProgramAddressSync(\n    [SEEDS.AFFILIATE, Buffer.from(params.code)],\n    PROGRAM_ID\n  );\n\n  // Instruction data: discriminator + is_active (bool)\n  const data = Buffer.alloc(9);\n  TOGGLE_AFFILIATE_DISCRIMINATOR.copy(data, 0);\n  data.writeUInt8(params.active ? 1 : 0, 8);\n\n  // Accounts for toggle_affiliate:\n  // config, affiliate, owner\n  const keys = [\n    { pubkey: CONFIG_PDA, isSigner: false, isWritable: false },\n    { pubkey: affiliatePda, isSigner: false, isWritable: true },\n    { pubkey: userPubkey, 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();\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    affiliatePda: affiliatePda.toBase58(),\n    newStatus: params.active,\n  };\n}\n\n// =============================================================================\n// HELPERS\n// =============================================================================\n\nfunction isValidCode(code: string): boolean {\n  if (code.length < 3 || code.length > 16) return false;\n  return /^[a-zA-Z0-9_]+$/.test(code);\n}\n"]}
@@ -0,0 +1,70 @@
1
+ /**
2
+ * Bet Transaction Builder
3
+ *
4
+ * Builds unsigned transactions for placing bets on Baozi markets.
5
+ * Agent builds, user signs. No private keys in agent.
6
+ */
7
+ import { Connection, PublicKey, Transaction } from '@solana/web3.js';
8
+ export interface BuildBetTransactionParams {
9
+ marketPda: PublicKey;
10
+ marketId: bigint;
11
+ userWallet: PublicKey;
12
+ outcome: 'yes' | 'no';
13
+ amountSol: number;
14
+ affiliatePda?: PublicKey;
15
+ affiliateOwner?: PublicKey;
16
+ whitelistRequired?: boolean;
17
+ }
18
+ export interface BuildBetTransactionResult {
19
+ transaction: Transaction;
20
+ positionPda: PublicKey;
21
+ serializedTx: string;
22
+ }
23
+ /**
24
+ * Build an unsigned bet transaction
25
+ *
26
+ * @param params - Transaction parameters
27
+ * @param connection - Optional connection (will create if not provided)
28
+ * @returns Unsigned transaction ready for user signing
29
+ */
30
+ export declare function buildBetTransaction(params: BuildBetTransactionParams, connection?: Connection): Promise<BuildBetTransactionResult>;
31
+ /**
32
+ * Simulate a bet transaction
33
+ */
34
+ export declare function simulateBetTransaction(transaction: Transaction, userWallet: PublicKey, connection?: Connection): Promise<{
35
+ success: boolean;
36
+ logs: string[];
37
+ unitsConsumed?: number;
38
+ error?: string;
39
+ }>;
40
+ /**
41
+ * Extract market_id from market account data
42
+ * V4.7.6 Market struct layout:
43
+ * - discriminator (8 bytes)
44
+ * - market_id (u64, 8 bytes) <-- First field!
45
+ * - question (string: 4 byte len + content)
46
+ * - ...rest of fields
47
+ */
48
+ export declare function extractMarketIdFromData(data: Buffer): bigint;
49
+ /**
50
+ * Extract access_gate from market data to determine if whitelist is needed
51
+ * This requires parsing through the struct to find access_gate field
52
+ */
53
+ export declare function extractAccessGateFromData(data: Buffer): number;
54
+ /**
55
+ * Fetch market data and build bet transaction
56
+ * Convenience function that handles market fetching and market_id extraction
57
+ */
58
+ export declare function fetchAndBuildBetTransaction(params: {
59
+ marketPda: string;
60
+ userWallet: string;
61
+ outcome: 'yes' | 'no';
62
+ amountSol: number;
63
+ affiliatePda?: string;
64
+ affiliateOwner?: string;
65
+ connection?: Connection;
66
+ }): Promise<{
67
+ transaction: BuildBetTransactionResult | null;
68
+ marketId: bigint;
69
+ error?: string;
70
+ }>;
@@ -0,0 +1,323 @@
1
+ /**
2
+ * Bet Transaction Builder
3
+ *
4
+ * Builds unsigned transactions for placing bets on Baozi markets.
5
+ * Agent builds, user signs. No private keys in agent.
6
+ */
7
+ import { Connection, PublicKey, Transaction, TransactionInstruction, SystemProgram, } from '@solana/web3.js';
8
+ import { PROGRAM_ID, CONFIG_PDA, SEEDS, RPC_ENDPOINT, solToLamports, } from '../config.js';
9
+ // =============================================================================
10
+ // INSTRUCTION DISCRIMINATORS
11
+ // =============================================================================
12
+ // place_bet_sol discriminator: [137, 137, 247, 253, 233, 243, 48, 170]
13
+ const PLACE_BET_SOL_DISCRIMINATOR = Buffer.from([137, 137, 247, 253, 233, 243, 48, 170]);
14
+ // place_bet_sol_with_affiliate discriminator: [197, 186, 187, 145, 252, 239, 101, 96]
15
+ const PLACE_BET_SOL_WITH_AFFILIATE_DISCRIMINATOR = Buffer.from([197, 186, 187, 145, 252, 239, 101, 96]);
16
+ // =============================================================================
17
+ // PDA DERIVATION
18
+ // =============================================================================
19
+ /**
20
+ * Derive position PDA from market ID and user
21
+ */
22
+ function derivePositionPda(marketId, user) {
23
+ const marketIdBuffer = Buffer.alloc(8);
24
+ marketIdBuffer.writeBigUInt64LE(marketId);
25
+ const [pda] = PublicKey.findProgramAddressSync([SEEDS.POSITION, marketIdBuffer, user.toBuffer()], PROGRAM_ID);
26
+ return pda;
27
+ }
28
+ /**
29
+ * Derive whitelist PDA from market ID
30
+ */
31
+ function deriveWhitelistPda(marketId) {
32
+ const marketIdBuffer = Buffer.alloc(8);
33
+ marketIdBuffer.writeBigUInt64LE(marketId);
34
+ const [pda] = PublicKey.findProgramAddressSync([SEEDS.WHITELIST, marketIdBuffer], PROGRAM_ID);
35
+ return pda;
36
+ }
37
+ /**
38
+ * Derive referred user PDA
39
+ */
40
+ function deriveReferredUserPda(user) {
41
+ const [pda] = PublicKey.findProgramAddressSync([Buffer.from('referred'), user.toBuffer()], PROGRAM_ID);
42
+ return pda;
43
+ }
44
+ // =============================================================================
45
+ // INSTRUCTION BUILDERS
46
+ // =============================================================================
47
+ /**
48
+ * Create place_bet_sol instruction
49
+ */
50
+ function createPlaceBetSolInstruction(params) {
51
+ // Serialize instruction data
52
+ // [discriminator(8)] [outcome(1)] [amount(8)]
53
+ const data = Buffer.alloc(17);
54
+ PLACE_BET_SOL_DISCRIMINATOR.copy(data, 0);
55
+ data.writeUInt8(params.outcome ? 1 : 0, 8);
56
+ data.writeBigUInt64LE(params.amount, 9);
57
+ const keys = [
58
+ { pubkey: params.config, isSigner: false, isWritable: false },
59
+ { pubkey: params.market, isSigner: false, isWritable: true },
60
+ { pubkey: params.position, isSigner: false, isWritable: true },
61
+ ];
62
+ // Add optional whitelist
63
+ if (params.whitelist) {
64
+ keys.push({ pubkey: params.whitelist, isSigner: false, isWritable: false });
65
+ }
66
+ keys.push({ pubkey: params.user, isSigner: true, isWritable: true }, { pubkey: SystemProgram.programId, isSigner: false, isWritable: false });
67
+ return new TransactionInstruction({
68
+ programId: PROGRAM_ID,
69
+ keys,
70
+ data,
71
+ });
72
+ }
73
+ /**
74
+ * Create place_bet_sol_with_affiliate instruction
75
+ */
76
+ function createPlaceBetSolWithAffiliateInstruction(params) {
77
+ // Serialize instruction data
78
+ // [discriminator(8)] [outcome(1)] [amount(8)]
79
+ const data = Buffer.alloc(17);
80
+ PLACE_BET_SOL_WITH_AFFILIATE_DISCRIMINATOR.copy(data, 0);
81
+ data.writeUInt8(params.outcome ? 1 : 0, 8);
82
+ data.writeBigUInt64LE(params.amount, 9);
83
+ const keys = [
84
+ { pubkey: params.config, isSigner: false, isWritable: false },
85
+ { pubkey: params.market, isSigner: false, isWritable: true },
86
+ { pubkey: params.position, isSigner: false, isWritable: true },
87
+ { pubkey: params.affiliate, isSigner: false, isWritable: true },
88
+ { pubkey: params.referredUser, isSigner: false, isWritable: true },
89
+ ];
90
+ // Add optional whitelist
91
+ if (params.whitelist) {
92
+ keys.push({ pubkey: params.whitelist, isSigner: false, isWritable: false });
93
+ }
94
+ keys.push({ pubkey: params.user, isSigner: true, isWritable: true }, { pubkey: SystemProgram.programId, isSigner: false, isWritable: false });
95
+ return new TransactionInstruction({
96
+ programId: PROGRAM_ID,
97
+ keys,
98
+ data,
99
+ });
100
+ }
101
+ // =============================================================================
102
+ // MAIN BUILDER FUNCTION
103
+ // =============================================================================
104
+ /**
105
+ * Build an unsigned bet transaction
106
+ *
107
+ * @param params - Transaction parameters
108
+ * @param connection - Optional connection (will create if not provided)
109
+ * @returns Unsigned transaction ready for user signing
110
+ */
111
+ export async function buildBetTransaction(params, connection) {
112
+ const conn = connection || new Connection(RPC_ENDPOINT, 'confirmed');
113
+ // Derive PDAs
114
+ const positionPda = derivePositionPda(params.marketId, params.userWallet);
115
+ const whitelistPda = params.whitelistRequired
116
+ ? deriveWhitelistPda(params.marketId)
117
+ : null;
118
+ // Convert amount to lamports
119
+ const amountLamports = solToLamports(params.amountSol);
120
+ // Create instruction
121
+ let instruction;
122
+ if (params.affiliatePda && params.affiliateOwner) {
123
+ const referredUserPda = deriveReferredUserPda(params.userWallet);
124
+ instruction = createPlaceBetSolWithAffiliateInstruction({
125
+ config: CONFIG_PDA,
126
+ market: params.marketPda,
127
+ position: positionPda,
128
+ affiliate: params.affiliatePda,
129
+ referredUser: referredUserPda,
130
+ whitelist: whitelistPda,
131
+ user: params.userWallet,
132
+ outcome: params.outcome === 'yes',
133
+ amount: amountLamports,
134
+ });
135
+ }
136
+ else {
137
+ instruction = createPlaceBetSolInstruction({
138
+ config: CONFIG_PDA,
139
+ market: params.marketPda,
140
+ position: positionPda,
141
+ whitelist: whitelistPda,
142
+ user: params.userWallet,
143
+ outcome: params.outcome === 'yes',
144
+ amount: amountLamports,
145
+ });
146
+ }
147
+ // Build transaction
148
+ const transaction = new Transaction();
149
+ transaction.add(instruction);
150
+ // Get recent blockhash
151
+ const { blockhash, lastValidBlockHeight } = await conn.getLatestBlockhash('finalized');
152
+ transaction.recentBlockhash = blockhash;
153
+ transaction.feePayer = params.userWallet;
154
+ // Serialize without signatures (returns Buffer)
155
+ const serializedTx = transaction.serialize({
156
+ requireAllSignatures: false,
157
+ verifySignatures: false,
158
+ }).toString('base64');
159
+ return {
160
+ transaction,
161
+ positionPda,
162
+ serializedTx,
163
+ };
164
+ }
165
+ // =============================================================================
166
+ // SIMULATION
167
+ // =============================================================================
168
+ /**
169
+ * Simulate a bet transaction
170
+ */
171
+ export async function simulateBetTransaction(transaction, userWallet, connection) {
172
+ const conn = connection || new Connection(RPC_ENDPOINT, 'confirmed');
173
+ try {
174
+ // Use the legacy simulation API for Transaction objects
175
+ const simulation = await conn.simulateTransaction(transaction);
176
+ if (simulation.value.err) {
177
+ return {
178
+ success: false,
179
+ logs: simulation.value.logs || [],
180
+ unitsConsumed: simulation.value.unitsConsumed,
181
+ error: JSON.stringify(simulation.value.err),
182
+ };
183
+ }
184
+ return {
185
+ success: true,
186
+ logs: simulation.value.logs || [],
187
+ unitsConsumed: simulation.value.unitsConsumed,
188
+ };
189
+ }
190
+ catch (err) {
191
+ return {
192
+ success: false,
193
+ logs: [],
194
+ error: err instanceof Error ? err.message : 'Unknown simulation error',
195
+ };
196
+ }
197
+ }
198
+ // =============================================================================
199
+ // MARKET DATA EXTRACTION
200
+ // =============================================================================
201
+ /**
202
+ * Extract market_id from market account data
203
+ * V4.7.6 Market struct layout:
204
+ * - discriminator (8 bytes)
205
+ * - market_id (u64, 8 bytes) <-- First field!
206
+ * - question (string: 4 byte len + content)
207
+ * - ...rest of fields
208
+ */
209
+ export function extractMarketIdFromData(data) {
210
+ // market_id is at offset 8 (right after discriminator)
211
+ return data.readBigUInt64LE(8);
212
+ }
213
+ /**
214
+ * Extract access_gate from market data to determine if whitelist is needed
215
+ * This requires parsing through the struct to find access_gate field
216
+ */
217
+ export function extractAccessGateFromData(data) {
218
+ // V4.7.6 Market struct layout after market_id:
219
+ // market_id (8) + question (4+len) + closing_time (8) + resolution_time (8) +
220
+ // auto_stop_buffer (8) + yes_pool (8) + no_pool (8) + snapshot_yes_pool (8) +
221
+ // snapshot_no_pool (8) + status (1) + winning_outcome (1+1 option) +
222
+ // currency_type (1) + _reserved_usdc_vault (33) + creator_bond (8) +
223
+ // total_claimed (8) + platform_fee_collected (8) + last_bet_time (8) +
224
+ // bump (1) + layer (1) + resolution_mode (1) + access_gate (1)
225
+ let offset = 8; // Skip discriminator
226
+ // market_id
227
+ offset += 8;
228
+ // question (string: 4 byte len + content)
229
+ const questionLen = data.readUInt32LE(offset);
230
+ offset += 4 + questionLen;
231
+ // closing_time, resolution_time, auto_stop_buffer (3 * 8 = 24)
232
+ offset += 24;
233
+ // yes_pool, no_pool, snapshot_yes_pool, snapshot_no_pool (4 * 8 = 32)
234
+ offset += 32;
235
+ // status (enum, 1 byte)
236
+ offset += 1;
237
+ // winning_outcome (Option<bool>: 1 byte discriminant + 1 byte value if Some)
238
+ const hasWinningOutcome = data.readUInt8(offset);
239
+ offset += 1;
240
+ if (hasWinningOutcome === 1) {
241
+ offset += 1;
242
+ }
243
+ // currency_type (enum, 1 byte)
244
+ offset += 1;
245
+ // _reserved_usdc_vault (33 bytes)
246
+ offset += 33;
247
+ // creator_bond (8)
248
+ offset += 8;
249
+ // total_claimed (8)
250
+ offset += 8;
251
+ // platform_fee_collected (8)
252
+ offset += 8;
253
+ // last_bet_time (8)
254
+ offset += 8;
255
+ // bump (1)
256
+ offset += 1;
257
+ // layer (enum, 1 byte)
258
+ offset += 1;
259
+ // resolution_mode (enum, 1 byte)
260
+ offset += 1;
261
+ // access_gate (enum, 1 byte)
262
+ const accessGate = data.readUInt8(offset);
263
+ return accessGate;
264
+ }
265
+ // =============================================================================
266
+ // HELPER: FETCH MARKET AND BUILD
267
+ // =============================================================================
268
+ /**
269
+ * Fetch market data and build bet transaction
270
+ * Convenience function that handles market fetching and market_id extraction
271
+ */
272
+ export async function fetchAndBuildBetTransaction(params) {
273
+ const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');
274
+ try {
275
+ const marketPubkey = new PublicKey(params.marketPda);
276
+ const userPubkey = new PublicKey(params.userWallet);
277
+ // Fetch market account to get market_id
278
+ const accountInfo = await conn.getAccountInfo(marketPubkey);
279
+ if (!accountInfo) {
280
+ return {
281
+ transaction: null,
282
+ marketId: 0n,
283
+ error: 'Market not found',
284
+ };
285
+ }
286
+ const data = accountInfo.data;
287
+ // Extract market_id (first field after discriminator)
288
+ const marketId = extractMarketIdFromData(data);
289
+ // Check if whitelist is required
290
+ const accessGate = extractAccessGateFromData(data);
291
+ const whitelistRequired = accessGate === 1; // 1 = Whitelist
292
+ // Build affiliate PDAs if provided
293
+ let affiliatePda;
294
+ let affiliateOwner;
295
+ if (params.affiliatePda && params.affiliateOwner) {
296
+ affiliatePda = new PublicKey(params.affiliatePda);
297
+ affiliateOwner = new PublicKey(params.affiliateOwner);
298
+ }
299
+ // Build the transaction
300
+ const result = await buildBetTransaction({
301
+ marketPda: marketPubkey,
302
+ marketId,
303
+ userWallet: userPubkey,
304
+ outcome: params.outcome,
305
+ amountSol: params.amountSol,
306
+ affiliatePda,
307
+ affiliateOwner,
308
+ whitelistRequired,
309
+ }, conn);
310
+ return {
311
+ transaction: result,
312
+ marketId,
313
+ };
314
+ }
315
+ catch (err) {
316
+ return {
317
+ transaction: null,
318
+ marketId: 0n,
319
+ error: err instanceof Error ? err.message : 'Unknown error',
320
+ };
321
+ }
322
+ }
323
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bet-transaction.js","sourceRoot":"","sources":["../../src/builders/bet-transaction.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Bet Transaction Builder\n *\n * Builds unsigned transactions for placing bets on Baozi markets.\n * Agent builds, user signs. No private keys in agent.\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  SEEDS,\n  RPC_ENDPOINT,\n  solToLamports,\n} from '../config.js';\n\n// =============================================================================\n// INSTRUCTION DISCRIMINATORS\n// =============================================================================\n\n// place_bet_sol discriminator: [137, 137, 247, 253, 233, 243, 48, 170]\nconst PLACE_BET_SOL_DISCRIMINATOR = Buffer.from([137, 137, 247, 253, 233, 243, 48, 170]);\n\n// place_bet_sol_with_affiliate discriminator: [197, 186, 187, 145, 252, 239, 101, 96]\nconst PLACE_BET_SOL_WITH_AFFILIATE_DISCRIMINATOR = Buffer.from([197, 186, 187, 145, 252, 239, 101, 96]);\n\n// =============================================================================\n// TYPES\n// =============================================================================\n\nexport interface BuildBetTransactionParams {\n  marketPda: PublicKey;\n  marketId: bigint;\n  userWallet: PublicKey;\n  outcome: 'yes' | 'no';\n  amountSol: number;\n  affiliatePda?: PublicKey;\n  affiliateOwner?: PublicKey;\n  whitelistRequired?: boolean;\n}\n\nexport interface BuildBetTransactionResult {\n  transaction: Transaction;\n  positionPda: PublicKey;\n  serializedTx: string;\n}\n\n// =============================================================================\n// PDA DERIVATION\n// =============================================================================\n\n/**\n * Derive position PDA from market ID and user\n */\nfunction derivePositionPda(marketId: bigint, user: PublicKey): PublicKey {\n  const marketIdBuffer = Buffer.alloc(8);\n  marketIdBuffer.writeBigUInt64LE(marketId);\n  const [pda] = PublicKey.findProgramAddressSync(\n    [SEEDS.POSITION, marketIdBuffer, user.toBuffer()],\n    PROGRAM_ID\n  );\n  return pda;\n}\n\n/**\n * Derive whitelist PDA from market ID\n */\nfunction deriveWhitelistPda(marketId: bigint): PublicKey {\n  const marketIdBuffer = Buffer.alloc(8);\n  marketIdBuffer.writeBigUInt64LE(marketId);\n  const [pda] = PublicKey.findProgramAddressSync(\n    [SEEDS.WHITELIST, marketIdBuffer],\n    PROGRAM_ID\n  );\n  return pda;\n}\n\n/**\n * Derive referred user PDA\n */\nfunction deriveReferredUserPda(user: PublicKey): PublicKey {\n  const [pda] = PublicKey.findProgramAddressSync(\n    [Buffer.from('referred'), user.toBuffer()],\n    PROGRAM_ID\n  );\n  return pda;\n}\n\n// =============================================================================\n// INSTRUCTION BUILDERS\n// =============================================================================\n\n/**\n * Create place_bet_sol instruction\n */\nfunction createPlaceBetSolInstruction(params: {\n  config: PublicKey;\n  market: PublicKey;\n  position: PublicKey;\n  whitelist: PublicKey | null;\n  user: PublicKey;\n  outcome: boolean;\n  amount: bigint;\n}): TransactionInstruction {\n  // Serialize instruction data\n  // [discriminator(8)] [outcome(1)] [amount(8)]\n  const data = Buffer.alloc(17);\n  PLACE_BET_SOL_DISCRIMINATOR.copy(data, 0);\n  data.writeUInt8(params.outcome ? 1 : 0, 8);\n  data.writeBigUInt64LE(params.amount, 9);\n\n  const keys = [\n    { pubkey: params.config, isSigner: false, isWritable: false },\n    { pubkey: params.market, isSigner: false, isWritable: true },\n    { pubkey: params.position, isSigner: false, isWritable: true },\n  ];\n\n  // Add optional whitelist\n  if (params.whitelist) {\n    keys.push({ pubkey: params.whitelist, isSigner: false, isWritable: false });\n  }\n\n  keys.push(\n    { pubkey: params.user, isSigner: true, isWritable: true },\n    { pubkey: SystemProgram.programId, isSigner: false, isWritable: false }\n  );\n\n  return new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n}\n\n/**\n * Create place_bet_sol_with_affiliate instruction\n */\nfunction createPlaceBetSolWithAffiliateInstruction(params: {\n  config: PublicKey;\n  market: PublicKey;\n  position: PublicKey;\n  affiliate: PublicKey;\n  referredUser: PublicKey;\n  whitelist: PublicKey | null;\n  user: PublicKey;\n  outcome: boolean;\n  amount: bigint;\n}): TransactionInstruction {\n  // Serialize instruction data\n  // [discriminator(8)] [outcome(1)] [amount(8)]\n  const data = Buffer.alloc(17);\n  PLACE_BET_SOL_WITH_AFFILIATE_DISCRIMINATOR.copy(data, 0);\n  data.writeUInt8(params.outcome ? 1 : 0, 8);\n  data.writeBigUInt64LE(params.amount, 9);\n\n  const keys = [\n    { pubkey: params.config, isSigner: false, isWritable: false },\n    { pubkey: params.market, isSigner: false, isWritable: true },\n    { pubkey: params.position, isSigner: false, isWritable: true },\n    { pubkey: params.affiliate, isSigner: false, isWritable: true },\n    { pubkey: params.referredUser, isSigner: false, isWritable: true },\n  ];\n\n  // Add optional whitelist\n  if (params.whitelist) {\n    keys.push({ pubkey: params.whitelist, isSigner: false, isWritable: false });\n  }\n\n  keys.push(\n    { pubkey: params.user, isSigner: true, isWritable: true },\n    { pubkey: SystemProgram.programId, isSigner: false, isWritable: false }\n  );\n\n  return new TransactionInstruction({\n    programId: PROGRAM_ID,\n    keys,\n    data,\n  });\n}\n\n// =============================================================================\n// MAIN BUILDER FUNCTION\n// =============================================================================\n\n/**\n * Build an unsigned bet transaction\n *\n * @param params - Transaction parameters\n * @param connection - Optional connection (will create if not provided)\n * @returns Unsigned transaction ready for user signing\n */\nexport async function buildBetTransaction(\n  params: BuildBetTransactionParams,\n  connection?: Connection\n): Promise<BuildBetTransactionResult> {\n  const conn = connection || new Connection(RPC_ENDPOINT, 'confirmed');\n\n  // Derive PDAs\n  const positionPda = derivePositionPda(params.marketId, params.userWallet);\n  const whitelistPda = params.whitelistRequired\n    ? deriveWhitelistPda(params.marketId)\n    : null;\n\n  // Convert amount to lamports\n  const amountLamports = solToLamports(params.amountSol);\n\n  // Create instruction\n  let instruction: TransactionInstruction;\n\n  if (params.affiliatePda && params.affiliateOwner) {\n    const referredUserPda = deriveReferredUserPda(params.userWallet);\n    instruction = createPlaceBetSolWithAffiliateInstruction({\n      config: CONFIG_PDA,\n      market: params.marketPda,\n      position: positionPda,\n      affiliate: params.affiliatePda,\n      referredUser: referredUserPda,\n      whitelist: whitelistPda,\n      user: params.userWallet,\n      outcome: params.outcome === 'yes',\n      amount: amountLamports,\n    });\n  } else {\n    instruction = createPlaceBetSolInstruction({\n      config: CONFIG_PDA,\n      market: params.marketPda,\n      position: positionPda,\n      whitelist: whitelistPda,\n      user: params.userWallet,\n      outcome: params.outcome === 'yes',\n      amount: amountLamports,\n    });\n  }\n\n  // Build transaction\n  const transaction = new Transaction();\n  transaction.add(instruction);\n\n  // Get recent blockhash\n  const { blockhash, lastValidBlockHeight } = await conn.getLatestBlockhash('finalized');\n  transaction.recentBlockhash = blockhash;\n  transaction.feePayer = params.userWallet;\n\n  // Serialize without signatures (returns Buffer)\n  const serializedTx = transaction.serialize({\n    requireAllSignatures: false,\n    verifySignatures: false,\n  }).toString('base64');\n\n  return {\n    transaction,\n    positionPda,\n    serializedTx,\n  };\n}\n\n// =============================================================================\n// SIMULATION\n// =============================================================================\n\n/**\n * Simulate a bet transaction\n */\nexport async function simulateBetTransaction(\n  transaction: Transaction,\n  userWallet: PublicKey,\n  connection?: Connection\n): Promise<{\n  success: boolean;\n  logs: string[];\n  unitsConsumed?: number;\n  error?: string;\n}> {\n  const conn = connection || new Connection(RPC_ENDPOINT, 'confirmed');\n\n  try {\n    // Use the legacy simulation API for Transaction objects\n    const simulation = await conn.simulateTransaction(transaction);\n\n    if (simulation.value.err) {\n      return {\n        success: false,\n        logs: simulation.value.logs || [],\n        unitsConsumed: simulation.value.unitsConsumed,\n        error: JSON.stringify(simulation.value.err),\n      };\n    }\n\n    return {\n      success: true,\n      logs: simulation.value.logs || [],\n      unitsConsumed: simulation.value.unitsConsumed,\n    };\n  } catch (err) {\n    return {\n      success: false,\n      logs: [],\n      error: err instanceof Error ? err.message : 'Unknown simulation error',\n    };\n  }\n}\n\n// =============================================================================\n// MARKET DATA EXTRACTION\n// =============================================================================\n\n/**\n * Extract market_id from market account data\n * V4.7.6 Market struct layout:\n * - discriminator (8 bytes)\n * - market_id (u64, 8 bytes) <-- First field!\n * - question (string: 4 byte len + content)\n * - ...rest of fields\n */\nexport function extractMarketIdFromData(data: Buffer): bigint {\n  // market_id is at offset 8 (right after discriminator)\n  return data.readBigUInt64LE(8);\n}\n\n/**\n * Extract access_gate from market data to determine if whitelist is needed\n * This requires parsing through the struct to find access_gate field\n */\nexport function extractAccessGateFromData(data: Buffer): number {\n  // V4.7.6 Market struct layout after market_id:\n  // market_id (8) + question (4+len) + closing_time (8) + resolution_time (8) +\n  // auto_stop_buffer (8) + yes_pool (8) + no_pool (8) + snapshot_yes_pool (8) +\n  // snapshot_no_pool (8) + status (1) + winning_outcome (1+1 option) +\n  // currency_type (1) + _reserved_usdc_vault (33) + creator_bond (8) +\n  // total_claimed (8) + platform_fee_collected (8) + last_bet_time (8) +\n  // bump (1) + layer (1) + resolution_mode (1) + access_gate (1)\n\n  let offset = 8; // Skip discriminator\n\n  // market_id\n  offset += 8;\n\n  // question (string: 4 byte len + content)\n  const questionLen = data.readUInt32LE(offset);\n  offset += 4 + questionLen;\n\n  // closing_time, resolution_time, auto_stop_buffer (3 * 8 = 24)\n  offset += 24;\n\n  // yes_pool, no_pool, snapshot_yes_pool, snapshot_no_pool (4 * 8 = 32)\n  offset += 32;\n\n  // status (enum, 1 byte)\n  offset += 1;\n\n  // winning_outcome (Option<bool>: 1 byte discriminant + 1 byte value if Some)\n  const hasWinningOutcome = data.readUInt8(offset);\n  offset += 1;\n  if (hasWinningOutcome === 1) {\n    offset += 1;\n  }\n\n  // currency_type (enum, 1 byte)\n  offset += 1;\n\n  // _reserved_usdc_vault (33 bytes)\n  offset += 33;\n\n  // creator_bond (8)\n  offset += 8;\n\n  // total_claimed (8)\n  offset += 8;\n\n  // platform_fee_collected (8)\n  offset += 8;\n\n  // last_bet_time (8)\n  offset += 8;\n\n  // bump (1)\n  offset += 1;\n\n  // layer (enum, 1 byte)\n  offset += 1;\n\n  // resolution_mode (enum, 1 byte)\n  offset += 1;\n\n  // access_gate (enum, 1 byte)\n  const accessGate = data.readUInt8(offset);\n\n  return accessGate;\n}\n\n// =============================================================================\n// HELPER: FETCH MARKET AND BUILD\n// =============================================================================\n\n/**\n * Fetch market data and build bet transaction\n * Convenience function that handles market fetching and market_id extraction\n */\nexport async function fetchAndBuildBetTransaction(params: {\n  marketPda: string;\n  userWallet: string;\n  outcome: 'yes' | 'no';\n  amountSol: number;\n  affiliatePda?: string;\n  affiliateOwner?: string;\n  connection?: Connection;\n}): Promise<{\n  transaction: BuildBetTransactionResult | null;\n  marketId: bigint;\n  error?: string;\n}> {\n  const conn = params.connection || new Connection(RPC_ENDPOINT, 'confirmed');\n\n  try {\n    const marketPubkey = new PublicKey(params.marketPda);\n    const userPubkey = new PublicKey(params.userWallet);\n\n    // Fetch market account to get market_id\n    const accountInfo = await conn.getAccountInfo(marketPubkey);\n    if (!accountInfo) {\n      return {\n        transaction: null,\n        marketId: 0n,\n        error: 'Market not found',\n      };\n    }\n\n    const data = accountInfo.data;\n\n    // Extract market_id (first field after discriminator)\n    const marketId = extractMarketIdFromData(data);\n\n    // Check if whitelist is required\n    const accessGate = extractAccessGateFromData(data);\n    const whitelistRequired = accessGate === 1; // 1 = Whitelist\n\n    // Build affiliate PDAs if provided\n    let affiliatePda: PublicKey | undefined;\n    let affiliateOwner: PublicKey | undefined;\n\n    if (params.affiliatePda && params.affiliateOwner) {\n      affiliatePda = new PublicKey(params.affiliatePda);\n      affiliateOwner = new PublicKey(params.affiliateOwner);\n    }\n\n    // Build the transaction\n    const result = await buildBetTransaction(\n      {\n        marketPda: marketPubkey,\n        marketId,\n        userWallet: userPubkey,\n        outcome: params.outcome,\n        amountSol: params.amountSol,\n        affiliatePda,\n        affiliateOwner,\n        whitelistRequired,\n      },\n      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"]}