@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,218 @@
1
+ /**
2
+ * Claims Handler - Winnings, Refunds, Affiliate & Creator Claims
3
+ */
4
+ import { Connection, PublicKey } from '@solana/web3.js';
5
+ import bs58 from 'bs58';
6
+ import { PROGRAM_ID, RPC_ENDPOINT, DISCRIMINATORS, lamportsToSol, SEEDS, } from '../config.js';
7
+ import { getMarket } from './markets.js';
8
+ import { getPositions } from './positions.js';
9
+ // =============================================================================
10
+ // HELPERS
11
+ // =============================================================================
12
+ /**
13
+ * Derive market PDA from market_id
14
+ */
15
+ function deriveMarketPda(marketId) {
16
+ const marketIdBigInt = BigInt(marketId);
17
+ const marketIdBuffer = Buffer.alloc(8);
18
+ marketIdBuffer.writeBigUInt64LE(marketIdBigInt);
19
+ const [pda] = PublicKey.findProgramAddressSync([Buffer.from('market'), marketIdBuffer], PROGRAM_ID);
20
+ return pda.toBase58();
21
+ }
22
+ // =============================================================================
23
+ // CLAIMABLE POSITIONS
24
+ // =============================================================================
25
+ /**
26
+ * Get all claimable positions for a wallet
27
+ * Checks which positions can be claimed (winnings or refunds)
28
+ */
29
+ export async function getClaimablePositions(walletAddress) {
30
+ const positions = await getPositions(walletAddress);
31
+ const claimable = [];
32
+ let winningsTotal = 0;
33
+ let refundsTotal = 0;
34
+ let alreadyClaimed = 0;
35
+ for (const position of positions) {
36
+ if (position.claimed) {
37
+ alreadyClaimed++;
38
+ continue;
39
+ }
40
+ // Derive market PDA from market_id
41
+ const marketPda = deriveMarketPda(position.marketId);
42
+ // Fetch market to check status
43
+ const market = await getMarket(marketPda);
44
+ if (!market)
45
+ continue;
46
+ // Get the bet amount for the winning side
47
+ const yesAmount = position.yesAmountSol;
48
+ const noAmount = position.noAmountSol;
49
+ const totalBet = yesAmount + noAmount;
50
+ let claimType = null;
51
+ let estimatedPayout = 0;
52
+ let winningSide = null;
53
+ if (market.status === 'Resolved') {
54
+ // Check if user bet on the winning side
55
+ if (market.winningOutcome === 'Yes' && yesAmount > 0) {
56
+ winningSide = 'Yes';
57
+ claimType = 'winnings';
58
+ const totalPool = market.yesPoolSol + market.noPoolSol;
59
+ if (market.yesPoolSol > 0) {
60
+ const share = yesAmount / market.yesPoolSol;
61
+ const grossPayout = share * totalPool;
62
+ const profit = grossPayout - yesAmount;
63
+ const fee = profit > 0 ? (profit * market.platformFeeBps) / 10000 : 0;
64
+ estimatedPayout = grossPayout - fee;
65
+ }
66
+ winningsTotal += estimatedPayout;
67
+ }
68
+ else if (market.winningOutcome === 'No' && noAmount > 0) {
69
+ winningSide = 'No';
70
+ claimType = 'winnings';
71
+ const totalPool = market.yesPoolSol + market.noPoolSol;
72
+ if (market.noPoolSol > 0) {
73
+ const share = noAmount / market.noPoolSol;
74
+ const grossPayout = share * totalPool;
75
+ const profit = grossPayout - noAmount;
76
+ const fee = profit > 0 ? (profit * market.platformFeeBps) / 10000 : 0;
77
+ estimatedPayout = grossPayout - fee;
78
+ }
79
+ winningsTotal += estimatedPayout;
80
+ }
81
+ else if (market.winningOutcome === null) {
82
+ // Invalid/Draw - refund all bets
83
+ claimType = 'refund';
84
+ estimatedPayout = totalBet;
85
+ winningSide = yesAmount > noAmount ? 'Yes' : 'No';
86
+ refundsTotal += estimatedPayout;
87
+ }
88
+ // Loser - nothing to claim
89
+ }
90
+ else if (market.status === 'Cancelled') {
91
+ // Cancelled - full refund of all bets
92
+ claimType = 'cancelled';
93
+ estimatedPayout = totalBet;
94
+ winningSide = yesAmount > noAmount ? 'Yes' : 'No';
95
+ refundsTotal += estimatedPayout;
96
+ }
97
+ if (claimType && winningSide) {
98
+ claimable.push({
99
+ positionPda: position.publicKey,
100
+ marketPda,
101
+ marketQuestion: market.question,
102
+ side: winningSide,
103
+ betAmountSol: winningSide === 'Yes' ? yesAmount : noAmount,
104
+ claimType,
105
+ estimatedPayoutSol: round4(estimatedPayout),
106
+ marketStatus: market.status,
107
+ marketOutcome: market.winningOutcome,
108
+ });
109
+ }
110
+ }
111
+ return {
112
+ wallet: walletAddress,
113
+ totalClaimableSol: round4(winningsTotal + refundsTotal),
114
+ winningsClaimableSol: round4(winningsTotal),
115
+ refundsClaimableSol: round4(refundsTotal),
116
+ claimablePositions: claimable,
117
+ alreadyClaimedCount: alreadyClaimed,
118
+ };
119
+ }
120
+ // =============================================================================
121
+ // AFFILIATE INFO
122
+ // =============================================================================
123
+ /**
124
+ * Decode Affiliate account
125
+ * Affiliate struct:
126
+ * - discriminator (8)
127
+ * - owner (Pubkey, 32)
128
+ * - code (String: 4 + len)
129
+ * - total_earned (u64, 8)
130
+ * - total_claimed (u64, 8)
131
+ * - referral_count (u64, 8)
132
+ * - is_active (bool, 1)
133
+ * - bump (u8, 1)
134
+ */
135
+ function decodeAffiliate(data, pubkey) {
136
+ try {
137
+ let offset = 8; // Skip discriminator
138
+ const owner = new PublicKey(data.slice(offset, offset + 32));
139
+ offset += 32;
140
+ const codeLen = data.readUInt32LE(offset);
141
+ offset += 4;
142
+ const code = data.slice(offset, offset + codeLen).toString('utf8');
143
+ offset += codeLen;
144
+ const totalEarned = data.readBigUInt64LE(offset);
145
+ offset += 8;
146
+ const totalClaimed = data.readBigUInt64LE(offset);
147
+ offset += 8;
148
+ const referralCount = data.readBigUInt64LE(offset);
149
+ offset += 8;
150
+ const isActive = data.readUInt8(offset) === 1;
151
+ return {
152
+ affiliatePda: pubkey.toBase58(),
153
+ owner: owner.toBase58(),
154
+ code,
155
+ totalEarnedSol: round4(lamportsToSol(totalEarned)),
156
+ unclaimedSol: round4(lamportsToSol(totalEarned - totalClaimed)),
157
+ referralCount: Number(referralCount),
158
+ isActive,
159
+ };
160
+ }
161
+ catch {
162
+ return null;
163
+ }
164
+ }
165
+ /**
166
+ * Get affiliate info by code
167
+ */
168
+ export async function getAffiliateByCode(code) {
169
+ const connection = new Connection(RPC_ENDPOINT, 'confirmed');
170
+ // Derive affiliate PDA from code
171
+ const [affiliatePda] = PublicKey.findProgramAddressSync([SEEDS.AFFILIATE, Buffer.from(code)], PROGRAM_ID);
172
+ try {
173
+ const account = await connection.getAccountInfo(affiliatePda);
174
+ if (!account)
175
+ return null;
176
+ return decodeAffiliate(account.data, affiliatePda);
177
+ }
178
+ catch {
179
+ return null;
180
+ }
181
+ }
182
+ /**
183
+ * Get affiliate info by owner wallet
184
+ */
185
+ export async function getAffiliateByOwner(walletAddress) {
186
+ const connection = new Connection(RPC_ENDPOINT, 'confirmed');
187
+ const accounts = await connection.getProgramAccounts(PROGRAM_ID, {
188
+ filters: [
189
+ {
190
+ memcmp: {
191
+ offset: 0,
192
+ bytes: bs58.encode(DISCRIMINATORS.AFFILIATE),
193
+ },
194
+ },
195
+ {
196
+ memcmp: {
197
+ offset: 8, // After discriminator
198
+ bytes: walletAddress,
199
+ },
200
+ },
201
+ ],
202
+ });
203
+ const affiliates = [];
204
+ for (const { account, pubkey } of accounts) {
205
+ const affiliate = decodeAffiliate(account.data, pubkey);
206
+ if (affiliate) {
207
+ affiliates.push(affiliate);
208
+ }
209
+ }
210
+ return affiliates;
211
+ }
212
+ // =============================================================================
213
+ // HELPERS
214
+ // =============================================================================
215
+ function round4(n) {
216
+ return Math.round(n * 10000) / 10000;
217
+ }
218
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"claims.js","sourceRoot":"","sources":["../../src/handlers/claims.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Claims Handler - Winnings, Refunds, Affiliate & Creator Claims\n */\nimport { Connection, PublicKey } from '@solana/web3.js';\nimport bs58 from 'bs58';\nimport {\n  PROGRAM_ID,\n  RPC_ENDPOINT,\n  DISCRIMINATORS,\n  lamportsToSol,\n  SEEDS,\n} from '../config.js';\nimport { getMarket, Market } from './markets.js';\nimport { getPositions, Position } from './positions.js';\n\n// =============================================================================\n// TYPES\n// =============================================================================\n\nexport interface ClaimablePosition {\n  positionPda: string;\n  marketPda: string;\n  marketQuestion: string;\n  side: 'Yes' | 'No';\n  betAmountSol: number;\n  claimType: 'winnings' | 'refund' | 'cancelled';\n  estimatedPayoutSol: number;\n  marketStatus: string;\n  marketOutcome: string | null;\n}\n\nexport interface ClaimSummary {\n  wallet: string;\n  totalClaimableSol: number;\n  winningsClaimableSol: number;\n  refundsClaimableSol: number;\n  claimablePositions: ClaimablePosition[];\n  alreadyClaimedCount: number;\n}\n\nexport interface AffiliateInfo {\n  affiliatePda: string;\n  owner: string;\n  code: string;\n  totalEarnedSol: number;\n  unclaimedSol: number;\n  referralCount: number;\n  isActive: boolean;\n}\n\nexport interface CreatorEarnings {\n  wallet: string;\n  totalCreatorFeesSol: number;\n  unclaimedSol: number;\n  marketsCreated: number;\n}\n\n// =============================================================================\n// HELPERS\n// =============================================================================\n\n/**\n * Derive market PDA from market_id\n */\nfunction deriveMarketPda(marketId: string): string {\n  const marketIdBigInt = BigInt(marketId);\n  const marketIdBuffer = Buffer.alloc(8);\n  marketIdBuffer.writeBigUInt64LE(marketIdBigInt);\n  const [pda] = PublicKey.findProgramAddressSync(\n    [Buffer.from('market'), marketIdBuffer],\n    PROGRAM_ID\n  );\n  return pda.toBase58();\n}\n\n// =============================================================================\n// CLAIMABLE POSITIONS\n// =============================================================================\n\n/**\n * Get all claimable positions for a wallet\n * Checks which positions can be claimed (winnings or refunds)\n */\nexport async function getClaimablePositions(walletAddress: string): Promise<ClaimSummary> {\n  const positions = await getPositions(walletAddress);\n\n  const claimable: ClaimablePosition[] = [];\n  let winningsTotal = 0;\n  let refundsTotal = 0;\n  let alreadyClaimed = 0;\n\n  for (const position of positions) {\n    if (position.claimed) {\n      alreadyClaimed++;\n      continue;\n    }\n\n    // Derive market PDA from market_id\n    const marketPda = deriveMarketPda(position.marketId);\n\n    // Fetch market to check status\n    const market = await getMarket(marketPda);\n    if (!market) continue;\n\n    // Get the bet amount for the winning side\n    const yesAmount = position.yesAmountSol;\n    const noAmount = position.noAmountSol;\n    const totalBet = yesAmount + noAmount;\n\n    let claimType: 'winnings' | 'refund' | 'cancelled' | null = null;\n    let estimatedPayout = 0;\n    let winningSide: 'Yes' | 'No' | null = null;\n\n    if (market.status === 'Resolved') {\n      // Check if user bet on the winning side\n      if (market.winningOutcome === 'Yes' && yesAmount > 0) {\n        winningSide = 'Yes';\n        claimType = 'winnings';\n        const totalPool = market.yesPoolSol + market.noPoolSol;\n        if (market.yesPoolSol > 0) {\n          const share = yesAmount / market.yesPoolSol;\n          const grossPayout = share * totalPool;\n          const profit = grossPayout - yesAmount;\n          const fee = profit > 0 ? (profit * market.platformFeeBps) / 10000 : 0;\n          estimatedPayout = grossPayout - fee;\n        }\n        winningsTotal += estimatedPayout;\n      } else if (market.winningOutcome === 'No' && noAmount > 0) {\n        winningSide = 'No';\n        claimType = 'winnings';\n        const totalPool = market.yesPoolSol + market.noPoolSol;\n        if (market.noPoolSol > 0) {\n          const share = noAmount / market.noPoolSol;\n          const grossPayout = share * totalPool;\n          const profit = grossPayout - noAmount;\n          const fee = profit > 0 ? (profit * market.platformFeeBps) / 10000 : 0;\n          estimatedPayout = grossPayout - fee;\n        }\n        winningsTotal += estimatedPayout;\n      } else if (market.winningOutcome === null) {\n        // Invalid/Draw - refund all bets\n        claimType = 'refund';\n        estimatedPayout = totalBet;\n        winningSide = yesAmount > noAmount ? 'Yes' : 'No';\n        refundsTotal += estimatedPayout;\n      }\n      // Loser - nothing to claim\n    } else if (market.status === 'Cancelled') {\n      // Cancelled - full refund of all bets\n      claimType = 'cancelled';\n      estimatedPayout = totalBet;\n      winningSide = yesAmount > noAmount ? 'Yes' : 'No';\n      refundsTotal += estimatedPayout;\n    }\n\n    if (claimType && winningSide) {\n      claimable.push({\n        positionPda: position.publicKey,\n        marketPda,\n        marketQuestion: market.question,\n        side: winningSide,\n        betAmountSol: winningSide === 'Yes' ? yesAmount : noAmount,\n        claimType,\n        estimatedPayoutSol: round4(estimatedPayout),\n        marketStatus: market.status,\n        marketOutcome: market.winningOutcome,\n      });\n    }\n  }\n\n  return {\n    wallet: walletAddress,\n    totalClaimableSol: round4(winningsTotal + refundsTotal),\n    winningsClaimableSol: round4(winningsTotal),\n    refundsClaimableSol: round4(refundsTotal),\n    claimablePositions: claimable,\n    alreadyClaimedCount: alreadyClaimed,\n  };\n}\n\n// =============================================================================\n// AFFILIATE INFO\n// =============================================================================\n\n/**\n * Decode Affiliate account\n * Affiliate struct:\n * - discriminator (8)\n * - owner (Pubkey, 32)\n * - code (String: 4 + len)\n * - total_earned (u64, 8)\n * - total_claimed (u64, 8)\n * - referral_count (u64, 8)\n * - is_active (bool, 1)\n * - bump (u8, 1)\n */\nfunction decodeAffiliate(data: Buffer, pubkey: PublicKey): AffiliateInfo | null {\n  try {\n    let offset = 8; // Skip discriminator\n\n    const owner = new PublicKey(data.slice(offset, offset + 32));\n    offset += 32;\n\n    const codeLen = data.readUInt32LE(offset);\n    offset += 4;\n    const code = data.slice(offset, offset + codeLen).toString('utf8');\n    offset += codeLen;\n\n    const totalEarned = data.readBigUInt64LE(offset);\n    offset += 8;\n\n    const totalClaimed = data.readBigUInt64LE(offset);\n    offset += 8;\n\n    const referralCount = data.readBigUInt64LE(offset);\n    offset += 8;\n\n    const isActive = data.readUInt8(offset) === 1;\n\n    return {\n      affiliatePda: pubkey.toBase58(),\n      owner: owner.toBase58(),\n      code,\n      totalEarnedSol: round4(lamportsToSol(totalEarned)),\n      unclaimedSol: round4(lamportsToSol(totalEarned - totalClaimed)),\n      referralCount: Number(referralCount),\n      isActive,\n    };\n  } catch {\n    return null;\n  }\n}\n\n/**\n * Get affiliate info by code\n */\nexport async function getAffiliateByCode(code: string): Promise<AffiliateInfo | null> {\n  const connection = new Connection(RPC_ENDPOINT, 'confirmed');\n\n  // Derive affiliate PDA from code\n  const [affiliatePda] = PublicKey.findProgramAddressSync(\n    [SEEDS.AFFILIATE, Buffer.from(code)],\n    PROGRAM_ID\n  );\n\n  try {\n    const account = await connection.getAccountInfo(affiliatePda);\n    if (!account) return null;\n    return decodeAffiliate(account.data as Buffer, affiliatePda);\n  } catch {\n    return null;\n  }\n}\n\n/**\n * Get affiliate info by owner wallet\n */\nexport async function getAffiliateByOwner(walletAddress: string): Promise<AffiliateInfo[]> {\n  const connection = new Connection(RPC_ENDPOINT, 'confirmed');\n\n  const accounts = await connection.getProgramAccounts(PROGRAM_ID, {\n    filters: [\n      {\n        memcmp: {\n          offset: 0,\n          bytes: bs58.encode(DISCRIMINATORS.AFFILIATE),\n        },\n      },\n      {\n        memcmp: {\n          offset: 8, // After discriminator\n          bytes: walletAddress,\n        },\n      },\n    ],\n  });\n\n  const affiliates: AffiliateInfo[] = [];\n  for (const { account, pubkey } of accounts) {\n    const affiliate = decodeAffiliate(account.data as Buffer, pubkey);\n    if (affiliate) {\n      affiliates.push(affiliate);\n    }\n  }\n\n  return affiliates;\n}\n\n// =============================================================================\n// HELPERS\n// =============================================================================\n\nfunction round4(n: number): number {\n  return Math.round(n * 10000) / 10000;\n}\n"]}
@@ -0,0 +1,154 @@
1
+ /**
2
+ * Market Creation Handler
3
+ *
4
+ * Provides high-level functions for creating markets:
5
+ * - Validation against v6.2 rules
6
+ * - Fee calculation
7
+ * - Transaction building
8
+ * - PDA derivation helpers
9
+ */
10
+ import { Connection } from '@solana/web3.js';
11
+ import { CreationValidationResult } from '../validation/creation-rules.js';
12
+ export interface MarketCreationPreview {
13
+ validation: CreationValidationResult;
14
+ marketPda?: string;
15
+ marketId?: string;
16
+ creationFeeSol: number;
17
+ platformFeeBps: number;
18
+ estimatedRentSol: number;
19
+ totalCostSol: number;
20
+ recommendedTiming?: {
21
+ closingTime: string;
22
+ resolutionTime: string;
23
+ };
24
+ }
25
+ export interface CreateMarketRequest {
26
+ question: string;
27
+ layer: 'lab' | 'private';
28
+ closingTime: string;
29
+ resolutionTime?: string;
30
+ marketType?: 'event' | 'measurement';
31
+ eventTime?: string;
32
+ measurementStart?: string;
33
+ measurementEnd?: string;
34
+ inviteHash?: string;
35
+ creatorWallet: string;
36
+ }
37
+ export interface CreateRaceMarketRequest {
38
+ question: string;
39
+ outcomes: string[];
40
+ closingTime: string;
41
+ resolutionTime?: string;
42
+ creatorWallet: string;
43
+ }
44
+ /**
45
+ * Preview market creation - validates and returns costs without building tx
46
+ */
47
+ export declare function previewMarketCreation(params: CreateMarketRequest, connection?: Connection): Promise<MarketCreationPreview>;
48
+ /**
49
+ * Preview race market creation
50
+ */
51
+ export declare function previewRaceMarketCreation(params: CreateRaceMarketRequest, connection?: Connection): Promise<MarketCreationPreview>;
52
+ /**
53
+ * Build lab market creation transaction with full validation
54
+ */
55
+ export declare function createLabMarket(params: CreateMarketRequest, connection?: Connection): Promise<{
56
+ success: boolean;
57
+ error?: string;
58
+ validation: CreationValidationResult;
59
+ transaction?: {
60
+ serialized: string;
61
+ marketPda: string;
62
+ marketId: string;
63
+ };
64
+ simulation?: {
65
+ success: boolean;
66
+ error?: string;
67
+ unitsConsumed?: number;
68
+ };
69
+ }>;
70
+ /**
71
+ * Build private market creation transaction
72
+ */
73
+ export declare function createPrivateMarket(params: CreateMarketRequest, connection?: Connection): Promise<{
74
+ success: boolean;
75
+ error?: string;
76
+ validation: CreationValidationResult;
77
+ transaction?: {
78
+ serialized: string;
79
+ marketPda: string;
80
+ marketId: string;
81
+ };
82
+ simulation?: {
83
+ success: boolean;
84
+ error?: string;
85
+ };
86
+ }>;
87
+ /**
88
+ * Build race market creation transaction
89
+ */
90
+ export declare function createRaceMarket(params: CreateRaceMarketRequest, connection?: Connection): Promise<{
91
+ success: boolean;
92
+ error?: string;
93
+ validation: CreationValidationResult;
94
+ transaction?: {
95
+ serialized: string;
96
+ raceMarketPda: string;
97
+ marketId: string;
98
+ };
99
+ simulation?: {
100
+ success: boolean;
101
+ error?: string;
102
+ };
103
+ }>;
104
+ /**
105
+ * Get creation fees for all layers
106
+ */
107
+ export declare function getAllCreationFees(): {
108
+ official: {
109
+ sol: number;
110
+ lamports: number;
111
+ };
112
+ lab: {
113
+ sol: number;
114
+ lamports: number;
115
+ };
116
+ private: {
117
+ sol: number;
118
+ lamports: number;
119
+ };
120
+ };
121
+ /**
122
+ * Get platform fees for all layers
123
+ */
124
+ export declare function getAllPlatformFees(): {
125
+ official: {
126
+ bps: number;
127
+ percent: string;
128
+ };
129
+ lab: {
130
+ bps: number;
131
+ percent: string;
132
+ };
133
+ private: {
134
+ bps: number;
135
+ percent: string;
136
+ };
137
+ };
138
+ /**
139
+ * Get timing constraints
140
+ */
141
+ export declare function getTimingConstraints(): {
142
+ minEventBufferHours: number;
143
+ recommendedEventBufferHours: number;
144
+ bettingFreezeSeconds: number;
145
+ maxMarketDurationDays: number;
146
+ };
147
+ /**
148
+ * Generate invite hash for private market
149
+ */
150
+ export declare function generateInviteHash(): string;
151
+ /**
152
+ * Derive invite link from hash
153
+ */
154
+ export declare function getInviteLink(marketPda: string, inviteHash: string): string;