@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,82 @@
1
+ /**
2
+ * Bet Validation Rules
3
+ *
4
+ * Validates bet parameters against protocol constraints:
5
+ * - Amount limits (0.01-100 SOL)
6
+ * - Market state (active, not frozen, not paused)
7
+ * - Whitelist access (for private markets)
8
+ * - Timing constraints (betting freeze)
9
+ */
10
+ export interface BetValidationParams {
11
+ amountSol: number;
12
+ marketStatus: number;
13
+ closingTime: Date;
14
+ isPaused: boolean;
15
+ accessGate: number;
16
+ userWhitelisted?: boolean;
17
+ layer?: number;
18
+ }
19
+ export interface BetValidation {
20
+ valid: boolean;
21
+ error?: string;
22
+ warnings: string[];
23
+ details: {
24
+ amountValid: boolean;
25
+ marketStateValid: boolean;
26
+ timingValid: boolean;
27
+ accessValid: boolean;
28
+ };
29
+ }
30
+ export interface ClaimValidationParams {
31
+ marketStatus: number;
32
+ marketOutcome: number;
33
+ positionSide: 'Yes' | 'No';
34
+ positionAmount: number;
35
+ alreadyClaimed: boolean;
36
+ }
37
+ export interface ClaimValidation {
38
+ valid: boolean;
39
+ error?: string;
40
+ canClaim: boolean;
41
+ isWinner: boolean;
42
+ }
43
+ /**
44
+ * Validate bet parameters before building transaction
45
+ */
46
+ export declare function validateBet(params: BetValidationParams): BetValidation;
47
+ /**
48
+ * Validate claim parameters before building claim transaction
49
+ */
50
+ export declare function validateClaim(params: ClaimValidationParams): ClaimValidation;
51
+ /**
52
+ * Calculate quote for a bet (pari-mutuel)
53
+ */
54
+ export declare function calculateBetQuote(params: {
55
+ betAmountSol: number;
56
+ side: 'Yes' | 'No';
57
+ currentYesPool: number;
58
+ currentNoPool: number;
59
+ platformFeeBps: number;
60
+ }): {
61
+ expectedPayoutSol: number;
62
+ potentialProfitSol: number;
63
+ feeSol: number;
64
+ impliedOdds: number;
65
+ decimalOdds: number;
66
+ newYesPool: number;
67
+ newNoPool: number;
68
+ };
69
+ /**
70
+ * Estimate claim amount for a winning position
71
+ */
72
+ export declare function estimateClaimAmount(params: {
73
+ positionAmount: number;
74
+ positionSide: 'Yes' | 'No';
75
+ totalYesPool: number;
76
+ totalNoPool: number;
77
+ platformFeeBps: number;
78
+ }): {
79
+ grossPayout: number;
80
+ fee: number;
81
+ netPayout: number;
82
+ };
@@ -0,0 +1,276 @@
1
+ /**
2
+ * Bet Validation Rules
3
+ *
4
+ * Validates bet parameters against protocol constraints:
5
+ * - Amount limits (0.01-100 SOL)
6
+ * - Market state (active, not frozen, not paused)
7
+ * - Whitelist access (for private markets)
8
+ * - Timing constraints (betting freeze)
9
+ */
10
+ import { BET_LIMITS, TIMING, MARKET_STATUS, ACCESS_GATE, MARKET_LAYER_NAMES, } from '../config.js';
11
+ // =============================================================================
12
+ // BET VALIDATION
13
+ // =============================================================================
14
+ /**
15
+ * Validate bet parameters before building transaction
16
+ */
17
+ export function validateBet(params) {
18
+ const warnings = [];
19
+ const details = {
20
+ amountValid: true,
21
+ marketStateValid: true,
22
+ timingValid: true,
23
+ accessValid: true,
24
+ };
25
+ // ==========================================================================
26
+ // Amount Validation
27
+ // ==========================================================================
28
+ if (params.amountSol < BET_LIMITS.MIN_BET_SOL) {
29
+ details.amountValid = false;
30
+ return {
31
+ valid: false,
32
+ error: `Minimum bet is ${BET_LIMITS.MIN_BET_SOL} SOL`,
33
+ warnings,
34
+ details,
35
+ };
36
+ }
37
+ if (params.amountSol > BET_LIMITS.MAX_BET_SOL) {
38
+ details.amountValid = false;
39
+ return {
40
+ valid: false,
41
+ error: `Maximum bet is ${BET_LIMITS.MAX_BET_SOL} SOL`,
42
+ warnings,
43
+ details,
44
+ };
45
+ }
46
+ // Amount warnings
47
+ if (params.amountSol > 50) {
48
+ warnings.push('Large bet amount. Ensure you understand the odds before placing.');
49
+ }
50
+ // ==========================================================================
51
+ // Market State Validation
52
+ // ==========================================================================
53
+ if (params.marketStatus !== MARKET_STATUS.ACTIVE) {
54
+ details.marketStateValid = false;
55
+ const statusName = getStatusName(params.marketStatus);
56
+ return {
57
+ valid: false,
58
+ error: `Market is ${statusName}, not accepting bets`,
59
+ warnings,
60
+ details,
61
+ };
62
+ }
63
+ if (params.isPaused) {
64
+ details.marketStateValid = false;
65
+ return {
66
+ valid: false,
67
+ error: 'Market is paused',
68
+ warnings,
69
+ details,
70
+ };
71
+ }
72
+ // ==========================================================================
73
+ // Timing Validation
74
+ // ==========================================================================
75
+ const now = new Date();
76
+ const freezeTime = new Date(params.closingTime.getTime() - TIMING.BETTING_FREEZE_SECONDS * 1000);
77
+ if (now >= params.closingTime) {
78
+ details.timingValid = false;
79
+ return {
80
+ valid: false,
81
+ error: 'Betting has closed',
82
+ warnings,
83
+ details,
84
+ };
85
+ }
86
+ if (now >= freezeTime) {
87
+ details.timingValid = false;
88
+ const remainingMs = params.closingTime.getTime() - now.getTime();
89
+ const remainingMin = Math.ceil(remainingMs / 60000);
90
+ return {
91
+ valid: false,
92
+ error: `Betting is frozen (${remainingMin} minutes until close)`,
93
+ warnings,
94
+ details,
95
+ };
96
+ }
97
+ // Timing warnings
98
+ const timeToFreezeMs = freezeTime.getTime() - now.getTime();
99
+ const timeToFreezeMin = Math.floor(timeToFreezeMs / 60000);
100
+ if (timeToFreezeMin < 30) {
101
+ warnings.push(`Betting freezes in ${timeToFreezeMin} minutes`);
102
+ }
103
+ // ==========================================================================
104
+ // Access Validation (Whitelist)
105
+ // ==========================================================================
106
+ if (params.accessGate === ACCESS_GATE.WHITELIST) {
107
+ if (params.userWhitelisted !== true) {
108
+ details.accessValid = false;
109
+ return {
110
+ valid: false,
111
+ error: 'You are not whitelisted for this private market',
112
+ warnings,
113
+ details,
114
+ };
115
+ }
116
+ }
117
+ // Layer-specific warnings
118
+ if (params.layer !== undefined) {
119
+ const layerName = MARKET_LAYER_NAMES[params.layer] || 'Unknown';
120
+ if (params.layer === 1) { // Lab
121
+ warnings.push(`This is a ${layerName} market (community-created). DYOR.`);
122
+ }
123
+ }
124
+ return {
125
+ valid: true,
126
+ warnings,
127
+ details,
128
+ };
129
+ }
130
+ // =============================================================================
131
+ // CLAIM VALIDATION
132
+ // =============================================================================
133
+ /**
134
+ * Validate claim parameters before building claim transaction
135
+ */
136
+ export function validateClaim(params) {
137
+ // Already claimed
138
+ if (params.alreadyClaimed) {
139
+ return {
140
+ valid: false,
141
+ error: 'Position already claimed',
142
+ canClaim: false,
143
+ isWinner: false,
144
+ };
145
+ }
146
+ // Market not resolved
147
+ if (params.marketStatus !== MARKET_STATUS.RESOLVED &&
148
+ params.marketStatus !== MARKET_STATUS.CANCELLED) {
149
+ const statusName = getStatusName(params.marketStatus);
150
+ return {
151
+ valid: false,
152
+ error: `Market is ${statusName}, cannot claim yet`,
153
+ canClaim: false,
154
+ isWinner: false,
155
+ };
156
+ }
157
+ // Cancelled market - everyone can claim refund
158
+ if (params.marketStatus === MARKET_STATUS.CANCELLED) {
159
+ return {
160
+ valid: true,
161
+ canClaim: true,
162
+ isWinner: false, // Refund, not winning
163
+ };
164
+ }
165
+ // Check if position is on winning side
166
+ const isWinner = (params.positionSide === 'Yes' && params.marketOutcome === 2) ||
167
+ (params.positionSide === 'No' && params.marketOutcome === 3);
168
+ if (!isWinner) {
169
+ return {
170
+ valid: false,
171
+ error: 'Position is on losing side, nothing to claim',
172
+ canClaim: false,
173
+ isWinner: false,
174
+ };
175
+ }
176
+ // Nothing to claim if amount is 0
177
+ if (params.positionAmount <= 0) {
178
+ return {
179
+ valid: false,
180
+ error: 'No position amount to claim',
181
+ canClaim: false,
182
+ isWinner: true,
183
+ };
184
+ }
185
+ return {
186
+ valid: true,
187
+ canClaim: true,
188
+ isWinner: true,
189
+ };
190
+ }
191
+ // =============================================================================
192
+ // HELPER FUNCTIONS
193
+ // =============================================================================
194
+ function getStatusName(status) {
195
+ switch (status) {
196
+ case MARKET_STATUS.ACTIVE: return 'Active';
197
+ case MARKET_STATUS.CLOSED: return 'Closed';
198
+ case MARKET_STATUS.RESOLVED: return 'Resolved';
199
+ case MARKET_STATUS.CANCELLED: return 'Cancelled';
200
+ case MARKET_STATUS.PAUSED: return 'Paused';
201
+ case MARKET_STATUS.RESOLVED_PENDING: return 'Pending Resolution';
202
+ case MARKET_STATUS.DISPUTED: return 'Disputed';
203
+ default: return 'Unknown';
204
+ }
205
+ }
206
+ /**
207
+ * Calculate quote for a bet (pari-mutuel)
208
+ */
209
+ export function calculateBetQuote(params) {
210
+ // Calculate new pools after bet
211
+ const newYesPool = params.side === 'Yes'
212
+ ? params.currentYesPool + params.betAmountSol
213
+ : params.currentYesPool;
214
+ const newNoPool = params.side === 'No'
215
+ ? params.currentNoPool + params.betAmountSol
216
+ : params.currentNoPool;
217
+ const newTotalPool = newYesPool + newNoPool;
218
+ // Calculate expected payout (pari-mutuel)
219
+ const sidePool = params.side === 'Yes' ? newYesPool : newNoPool;
220
+ const expectedPayoutSol = sidePool > 0
221
+ ? (params.betAmountSol / sidePool) * newTotalPool
222
+ : 0;
223
+ // Calculate profit and fee
224
+ const grossProfit = expectedPayoutSol - params.betAmountSol;
225
+ const feeSol = grossProfit > 0
226
+ ? (grossProfit * params.platformFeeBps) / 10000
227
+ : 0;
228
+ const potentialProfitSol = grossProfit - feeSol;
229
+ // Calculate odds
230
+ const impliedOdds = newTotalPool > 0
231
+ ? (sidePool / newTotalPool) * 100
232
+ : 50;
233
+ const decimalOdds = sidePool > 0
234
+ ? newTotalPool / sidePool
235
+ : 2;
236
+ return {
237
+ expectedPayoutSol: round4(expectedPayoutSol),
238
+ potentialProfitSol: round4(potentialProfitSol),
239
+ feeSol: round4(feeSol),
240
+ impliedOdds: round2(impliedOdds),
241
+ decimalOdds: round2(decimalOdds),
242
+ newYesPool: round4(newYesPool),
243
+ newNoPool: round4(newNoPool),
244
+ };
245
+ }
246
+ /**
247
+ * Estimate claim amount for a winning position
248
+ */
249
+ export function estimateClaimAmount(params) {
250
+ const totalPool = params.totalYesPool + params.totalNoPool;
251
+ const winningPool = params.positionSide === 'Yes'
252
+ ? params.totalYesPool
253
+ : params.totalNoPool;
254
+ // Calculate share of winning pool
255
+ const shareOfPool = winningPool > 0
256
+ ? params.positionAmount / winningPool
257
+ : 0;
258
+ // Gross payout from total pool
259
+ const grossPayout = shareOfPool * totalPool;
260
+ // Calculate fee on profit only
261
+ const profit = grossPayout - params.positionAmount;
262
+ const fee = profit > 0 ? (profit * params.platformFeeBps) / 10000 : 0;
263
+ return {
264
+ grossPayout: round4(grossPayout),
265
+ fee: round4(fee),
266
+ netPayout: round4(grossPayout - fee),
267
+ };
268
+ }
269
+ // Rounding helpers
270
+ function round2(n) {
271
+ return Math.round(n * 100) / 100;
272
+ }
273
+ function round4(n) {
274
+ return Math.round(n * 10000) / 10000;
275
+ }
276
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bet-rules.js","sourceRoot":"","sources":["../../src/validation/bet-rules.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Bet Validation Rules\n *\n * Validates bet parameters against protocol constraints:\n * - Amount limits (0.01-100 SOL)\n * - Market state (active, not frozen, not paused)\n * - Whitelist access (for private markets)\n * - Timing constraints (betting freeze)\n */\n\nimport {\n  BET_LIMITS,\n  TIMING,\n  MARKET_STATUS,\n  ACCESS_GATE,\n  MARKET_LAYER_NAMES,\n} from '../config.js';\n\n// =============================================================================\n// TYPES\n// =============================================================================\n\nexport interface BetValidationParams {\n  amountSol: number;\n  marketStatus: number;\n  closingTime: Date;\n  isPaused: boolean;\n  accessGate: number;\n  userWhitelisted?: boolean;\n  layer?: number;\n}\n\nexport interface BetValidation {\n  valid: boolean;\n  error?: string;\n  warnings: string[];\n  details: {\n    amountValid: boolean;\n    marketStateValid: boolean;\n    timingValid: boolean;\n    accessValid: boolean;\n  };\n}\n\nexport interface ClaimValidationParams {\n  marketStatus: number;\n  marketOutcome: number;\n  positionSide: 'Yes' | 'No';\n  positionAmount: number;\n  alreadyClaimed: boolean;\n}\n\nexport interface ClaimValidation {\n  valid: boolean;\n  error?: string;\n  canClaim: boolean;\n  isWinner: boolean;\n}\n\n// =============================================================================\n// BET VALIDATION\n// =============================================================================\n\n/**\n * Validate bet parameters before building transaction\n */\nexport function validateBet(params: BetValidationParams): BetValidation {\n  const warnings: string[] = [];\n  const details = {\n    amountValid: true,\n    marketStateValid: true,\n    timingValid: true,\n    accessValid: true,\n  };\n\n  // ==========================================================================\n  // Amount Validation\n  // ==========================================================================\n  if (params.amountSol < BET_LIMITS.MIN_BET_SOL) {\n    details.amountValid = false;\n    return {\n      valid: false,\n      error: `Minimum bet is ${BET_LIMITS.MIN_BET_SOL} SOL`,\n      warnings,\n      details,\n    };\n  }\n\n  if (params.amountSol > BET_LIMITS.MAX_BET_SOL) {\n    details.amountValid = false;\n    return {\n      valid: false,\n      error: `Maximum bet is ${BET_LIMITS.MAX_BET_SOL} SOL`,\n      warnings,\n      details,\n    };\n  }\n\n  // Amount warnings\n  if (params.amountSol > 50) {\n    warnings.push('Large bet amount. Ensure you understand the odds before placing.');\n  }\n\n  // ==========================================================================\n  // Market State Validation\n  // ==========================================================================\n  if (params.marketStatus !== MARKET_STATUS.ACTIVE) {\n    details.marketStateValid = false;\n    const statusName = getStatusName(params.marketStatus);\n    return {\n      valid: false,\n      error: `Market is ${statusName}, not accepting bets`,\n      warnings,\n      details,\n    };\n  }\n\n  if (params.isPaused) {\n    details.marketStateValid = false;\n    return {\n      valid: false,\n      error: 'Market is paused',\n      warnings,\n      details,\n    };\n  }\n\n  // ==========================================================================\n  // Timing Validation\n  // ==========================================================================\n  const now = new Date();\n  const freezeTime = new Date(\n    params.closingTime.getTime() - TIMING.BETTING_FREEZE_SECONDS * 1000\n  );\n\n  if (now >= params.closingTime) {\n    details.timingValid = false;\n    return {\n      valid: false,\n      error: 'Betting has closed',\n      warnings,\n      details,\n    };\n  }\n\n  if (now >= freezeTime) {\n    details.timingValid = false;\n    const remainingMs = params.closingTime.getTime() - now.getTime();\n    const remainingMin = Math.ceil(remainingMs / 60000);\n    return {\n      valid: false,\n      error: `Betting is frozen (${remainingMin} minutes until close)`,\n      warnings,\n      details,\n    };\n  }\n\n  // Timing warnings\n  const timeToFreezeMs = freezeTime.getTime() - now.getTime();\n  const timeToFreezeMin = Math.floor(timeToFreezeMs / 60000);\n  if (timeToFreezeMin < 30) {\n    warnings.push(`Betting freezes in ${timeToFreezeMin} minutes`);\n  }\n\n  // ==========================================================================\n  // Access Validation (Whitelist)\n  // ==========================================================================\n  if (params.accessGate === ACCESS_GATE.WHITELIST) {\n    if (params.userWhitelisted !== true) {\n      details.accessValid = false;\n      return {\n        valid: false,\n        error: 'You are not whitelisted for this private market',\n        warnings,\n        details,\n      };\n    }\n  }\n\n  // Layer-specific warnings\n  if (params.layer !== undefined) {\n    const layerName = MARKET_LAYER_NAMES[params.layer] || 'Unknown';\n    if (params.layer === 1) { // Lab\n      warnings.push(`This is a ${layerName} market (community-created). DYOR.`);\n    }\n  }\n\n  return {\n    valid: true,\n    warnings,\n    details,\n  };\n}\n\n// =============================================================================\n// CLAIM VALIDATION\n// =============================================================================\n\n/**\n * Validate claim parameters before building claim transaction\n */\nexport function validateClaim(params: ClaimValidationParams): ClaimValidation {\n  // Already claimed\n  if (params.alreadyClaimed) {\n    return {\n      valid: false,\n      error: 'Position already claimed',\n      canClaim: false,\n      isWinner: false,\n    };\n  }\n\n  // Market not resolved\n  if (params.marketStatus !== MARKET_STATUS.RESOLVED &&\n      params.marketStatus !== MARKET_STATUS.CANCELLED) {\n    const statusName = getStatusName(params.marketStatus);\n    return {\n      valid: false,\n      error: `Market is ${statusName}, cannot claim yet`,\n      canClaim: false,\n      isWinner: false,\n    };\n  }\n\n  // Cancelled market - everyone can claim refund\n  if (params.marketStatus === MARKET_STATUS.CANCELLED) {\n    return {\n      valid: true,\n      canClaim: true,\n      isWinner: false, // Refund, not winning\n    };\n  }\n\n  // Check if position is on winning side\n  const isWinner = (params.positionSide === 'Yes' && params.marketOutcome === 2) ||\n                   (params.positionSide === 'No' && params.marketOutcome === 3);\n\n  if (!isWinner) {\n    return {\n      valid: false,\n      error: 'Position is on losing side, nothing to claim',\n      canClaim: false,\n      isWinner: false,\n    };\n  }\n\n  // Nothing to claim if amount is 0\n  if (params.positionAmount <= 0) {\n    return {\n      valid: false,\n      error: 'No position amount to claim',\n      canClaim: false,\n      isWinner: true,\n    };\n  }\n\n  return {\n    valid: true,\n    canClaim: true,\n    isWinner: true,\n  };\n}\n\n// =============================================================================\n// HELPER FUNCTIONS\n// =============================================================================\n\nfunction getStatusName(status: number): string {\n  switch (status) {\n    case MARKET_STATUS.ACTIVE: return 'Active';\n    case MARKET_STATUS.CLOSED: return 'Closed';\n    case MARKET_STATUS.RESOLVED: return 'Resolved';\n    case MARKET_STATUS.CANCELLED: return 'Cancelled';\n    case MARKET_STATUS.PAUSED: return 'Paused';\n    case MARKET_STATUS.RESOLVED_PENDING: return 'Pending Resolution';\n    case MARKET_STATUS.DISPUTED: return 'Disputed';\n    default: return 'Unknown';\n  }\n}\n\n/**\n * Calculate quote for a bet (pari-mutuel)\n */\nexport function calculateBetQuote(params: {\n  betAmountSol: number;\n  side: 'Yes' | 'No';\n  currentYesPool: number;\n  currentNoPool: number;\n  platformFeeBps: number;\n}): {\n  expectedPayoutSol: number;\n  potentialProfitSol: number;\n  feeSol: number;\n  impliedOdds: number;\n  decimalOdds: number;\n  newYesPool: number;\n  newNoPool: number;\n} {\n  // Calculate new pools after bet\n  const newYesPool = params.side === 'Yes'\n    ? params.currentYesPool + params.betAmountSol\n    : params.currentYesPool;\n  const newNoPool = params.side === 'No'\n    ? params.currentNoPool + params.betAmountSol\n    : params.currentNoPool;\n  const newTotalPool = newYesPool + newNoPool;\n\n  // Calculate expected payout (pari-mutuel)\n  const sidePool = params.side === 'Yes' ? newYesPool : newNoPool;\n  const expectedPayoutSol = sidePool > 0\n    ? (params.betAmountSol / sidePool) * newTotalPool\n    : 0;\n\n  // Calculate profit and fee\n  const grossProfit = expectedPayoutSol - params.betAmountSol;\n  const feeSol = grossProfit > 0\n    ? (grossProfit * params.platformFeeBps) / 10000\n    : 0;\n  const potentialProfitSol = grossProfit - feeSol;\n\n  // Calculate odds\n  const impliedOdds = newTotalPool > 0\n    ? (sidePool / newTotalPool) * 100\n    : 50;\n  const decimalOdds = sidePool > 0\n    ? newTotalPool / sidePool\n    : 2;\n\n  return {\n    expectedPayoutSol: round4(expectedPayoutSol),\n    potentialProfitSol: round4(potentialProfitSol),\n    feeSol: round4(feeSol),\n    impliedOdds: round2(impliedOdds),\n    decimalOdds: round2(decimalOdds),\n    newYesPool: round4(newYesPool),\n    newNoPool: round4(newNoPool),\n  };\n}\n\n/**\n * Estimate claim amount for a winning position\n */\nexport function estimateClaimAmount(params: {\n  positionAmount: number;\n  positionSide: 'Yes' | 'No';\n  totalYesPool: number;\n  totalNoPool: number;\n  platformFeeBps: number;\n}): {\n  grossPayout: number;\n  fee: number;\n  netPayout: number;\n} {\n  const totalPool = params.totalYesPool + params.totalNoPool;\n  const winningPool = params.positionSide === 'Yes'\n    ? params.totalYesPool\n    : params.totalNoPool;\n\n  // Calculate share of winning pool\n  const shareOfPool = winningPool > 0\n    ? params.positionAmount / winningPool\n    : 0;\n\n  // Gross payout from total pool\n  const grossPayout = shareOfPool * totalPool;\n\n  // Calculate fee on profit only\n  const profit = grossPayout - params.positionAmount;\n  const fee = profit > 0 ? (profit * params.platformFeeBps) / 10000 : 0;\n\n  return {\n    grossPayout: round4(grossPayout),\n    fee: round4(fee),\n    netPayout: round4(grossPayout - fee),\n  };\n}\n\n// Rounding helpers\nfunction round2(n: number): number {\n  return Math.round(n * 100) / 100;\n}\n\nfunction round4(n: number): number {\n  return Math.round(n * 10000) / 10000;\n}\n"]}
@@ -0,0 +1,69 @@
1
+ /**
2
+ * Market Creation Validation Rules (v6.2 Compliant)
3
+ *
4
+ * Implements validation for:
5
+ * - Rule A: Event-based markets (12-24h buffer before event)
6
+ * - Rule B: Measurement-period markets (close before measurement starts)
7
+ * - Race market outcome validation
8
+ * - Question and timing constraints
9
+ */
10
+ export interface CreateMarketParams {
11
+ question: string;
12
+ closingTime: Date;
13
+ resolutionTime: Date;
14
+ layer: 'official' | 'lab' | 'private';
15
+ marketType?: 'event' | 'measurement';
16
+ eventTime?: Date;
17
+ measurementStart?: Date;
18
+ measurementEnd?: Date;
19
+ outcomes?: string[];
20
+ inviteHash?: string;
21
+ }
22
+ export interface CreationValidationResult {
23
+ valid: boolean;
24
+ errors: string[];
25
+ warnings: string[];
26
+ suggestions: string[];
27
+ computed: {
28
+ ruleType: 'A' | 'B' | 'unknown';
29
+ bufferHours?: number;
30
+ recommendedClosingTime?: string;
31
+ creationFeeSol: number;
32
+ platformFeeBps: number;
33
+ estimatedRentSol: number;
34
+ };
35
+ }
36
+ /**
37
+ * Validate market creation parameters
38
+ */
39
+ export declare function validateMarketCreation(params: CreateMarketParams): CreationValidationResult;
40
+ /**
41
+ * Calculate recommended resolution time from closing time
42
+ */
43
+ export declare function calculateResolutionTime(closingTime: Date, marketType: 'event' | 'measurement', eventTime?: Date): Date;
44
+ /**
45
+ * Calculate recommended closing time for event
46
+ */
47
+ export declare function calculateRecommendedClosingTime(eventTime: Date, bufferHours?: number): Date;
48
+ /**
49
+ * Get creation fee for layer
50
+ */
51
+ export declare function getCreationFee(layer: 'official' | 'lab' | 'private'): {
52
+ lamports: number;
53
+ sol: number;
54
+ };
55
+ /**
56
+ * Validate question format
57
+ */
58
+ export declare function validateQuestion(question: string): {
59
+ valid: boolean;
60
+ errors: string[];
61
+ suggestions: string[];
62
+ };
63
+ /**
64
+ * Validate race outcomes
65
+ */
66
+ export declare function validateRaceOutcomes(outcomes: string[]): {
67
+ valid: boolean;
68
+ errors: string[];
69
+ };