@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,302 @@
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
+ import { TIMING, FEES } from '../config.js';
11
+ // =============================================================================
12
+ // CONSTANTS
13
+ // =============================================================================
14
+ const MAX_QUESTION_LENGTH = 200;
15
+ const MIN_OUTCOMES = 2;
16
+ const MAX_OUTCOMES = 10;
17
+ const MAX_OUTCOME_LABEL_LENGTH = 50;
18
+ const MIN_RESOLUTION_BUFFER_SECONDS = 600; // 10 minutes
19
+ const MAX_MARKET_DURATION_DAYS = 365;
20
+ // Rent estimates (lamports)
21
+ const MARKET_RENT = 5_000_000; // ~0.005 SOL
22
+ const RACE_MARKET_BASE_RENT = 8_000_000; // ~0.008 SOL
23
+ const RACE_OUTCOME_RENT = 500_000; // ~0.0005 SOL per outcome
24
+ // =============================================================================
25
+ // MAIN VALIDATION
26
+ // =============================================================================
27
+ /**
28
+ * Validate market creation parameters
29
+ */
30
+ export function validateMarketCreation(params) {
31
+ const errors = [];
32
+ const warnings = [];
33
+ const suggestions = [];
34
+ // Determine rule type
35
+ let ruleType = 'unknown';
36
+ if (params.marketType === 'event' || params.eventTime) {
37
+ ruleType = 'A';
38
+ }
39
+ else if (params.marketType === 'measurement' || params.measurementStart) {
40
+ ruleType = 'B';
41
+ }
42
+ // =========================================================================
43
+ // Question Validation
44
+ // =========================================================================
45
+ if (!params.question || params.question.trim().length === 0) {
46
+ errors.push('Question is required');
47
+ }
48
+ else if (params.question.length > MAX_QUESTION_LENGTH) {
49
+ errors.push(`Question exceeds ${MAX_QUESTION_LENGTH} characters (got ${params.question.length})`);
50
+ }
51
+ if (!params.question.endsWith('?')) {
52
+ warnings.push('Question should end with a question mark for clarity');
53
+ }
54
+ // =========================================================================
55
+ // Timing Validation
56
+ // =========================================================================
57
+ const now = new Date();
58
+ // Closing time must be in future
59
+ if (params.closingTime <= now) {
60
+ errors.push('Closing time must be in the future');
61
+ }
62
+ // Maximum duration check
63
+ const durationDays = (params.closingTime.getTime() - now.getTime()) / (1000 * 60 * 60 * 24);
64
+ if (durationDays > MAX_MARKET_DURATION_DAYS) {
65
+ errors.push(`Market duration exceeds ${MAX_MARKET_DURATION_DAYS} days`);
66
+ }
67
+ // Resolution time must be after closing time
68
+ if (params.resolutionTime <= params.closingTime) {
69
+ errors.push('Resolution time must be after closing time');
70
+ }
71
+ // Minimum resolution buffer
72
+ const resolutionBufferSec = (params.resolutionTime.getTime() - params.closingTime.getTime()) / 1000;
73
+ if (resolutionBufferSec < MIN_RESOLUTION_BUFFER_SECONDS) {
74
+ errors.push(`Resolution buffer too short: ${resolutionBufferSec}s (min ${MIN_RESOLUTION_BUFFER_SECONDS}s)`);
75
+ }
76
+ // =========================================================================
77
+ // Rule A: Event-Based Validation
78
+ // =========================================================================
79
+ let bufferHours;
80
+ let recommendedClosingTime;
81
+ if (ruleType === 'A') {
82
+ if (!params.eventTime) {
83
+ errors.push('Event-based markets require event_time');
84
+ }
85
+ else {
86
+ // Event must be after closing
87
+ if (params.eventTime <= params.closingTime) {
88
+ errors.push('Event time must be after closing time');
89
+ }
90
+ // Calculate buffer
91
+ bufferHours = (params.eventTime.getTime() - params.closingTime.getTime()) / (1000 * 60 * 60);
92
+ if (bufferHours < TIMING.MIN_EVENT_BUFFER_HOURS) {
93
+ errors.push(`Event buffer too short: ${bufferHours.toFixed(1)}h. ` +
94
+ `Minimum ${TIMING.MIN_EVENT_BUFFER_HOURS}h required (v6.2 Rule A).`);
95
+ // Suggest corrected closing time
96
+ const suggestedClose = new Date(params.eventTime.getTime() - (TIMING.RECOMMENDED_EVENT_BUFFER_HOURS * 60 * 60 * 1000));
97
+ recommendedClosingTime = suggestedClose.toISOString();
98
+ suggestions.push(`Recommended closing time: ${recommendedClosingTime}`);
99
+ }
100
+ else if (bufferHours < 18) {
101
+ warnings.push(`Buffer is ${bufferHours.toFixed(1)}h. ` +
102
+ `Recommend 18-24h for safety margin (v6.2 Rule A).`);
103
+ }
104
+ // Event must be in future
105
+ if (params.eventTime <= now) {
106
+ errors.push('Event time must be in the future');
107
+ }
108
+ }
109
+ }
110
+ // =========================================================================
111
+ // Rule B: Measurement-Period Validation
112
+ // =========================================================================
113
+ if (ruleType === 'B') {
114
+ if (!params.measurementStart) {
115
+ errors.push('Measurement-period markets require measurement_start');
116
+ }
117
+ else {
118
+ // CRITICAL: Betting must close BEFORE measurement starts
119
+ if (params.closingTime >= params.measurementStart) {
120
+ const overlapHours = (params.closingTime.getTime() - params.measurementStart.getTime()) / (1000 * 60 * 60);
121
+ errors.push(`INVALID: Betting closes ${overlapHours.toFixed(1)}h AFTER measurement starts. ` +
122
+ `This allows information advantage! (v6.2 Rule B)`);
123
+ // Suggest corrected closing time
124
+ const suggestedClose = new Date(params.measurementStart.getTime() - (60 * 60 * 1000)); // 1h before
125
+ recommendedClosingTime = suggestedClose.toISOString();
126
+ suggestions.push(`Recommended closing time: ${recommendedClosingTime}`);
127
+ }
128
+ // Measurement end validation
129
+ if (params.measurementEnd) {
130
+ if (params.measurementEnd <= params.measurementStart) {
131
+ errors.push('Measurement end must be after measurement start');
132
+ }
133
+ const periodDays = (params.measurementEnd.getTime() - params.measurementStart.getTime()) / (1000 * 60 * 60 * 24);
134
+ if (periodDays > 7) {
135
+ warnings.push(`Long measurement period: ${periodDays.toFixed(0)} days. ` +
136
+ `Prefer 2-7 days for better UX (v6.2 guidance).`);
137
+ }
138
+ }
139
+ }
140
+ }
141
+ // =========================================================================
142
+ // Race Market Validation
143
+ // =========================================================================
144
+ if (params.outcomes) {
145
+ if (params.outcomes.length < MIN_OUTCOMES) {
146
+ errors.push(`Race markets require at least ${MIN_OUTCOMES} outcomes`);
147
+ }
148
+ if (params.outcomes.length > MAX_OUTCOMES) {
149
+ errors.push(`Race markets limited to ${MAX_OUTCOMES} outcomes (got ${params.outcomes.length})`);
150
+ }
151
+ // Check outcome labels
152
+ for (let i = 0; i < params.outcomes.length; i++) {
153
+ const outcome = params.outcomes[i];
154
+ if (!outcome || outcome.trim().length === 0) {
155
+ errors.push(`Outcome ${i} is empty`);
156
+ }
157
+ else if (outcome.length > MAX_OUTCOME_LABEL_LENGTH) {
158
+ errors.push(`Outcome ${i} exceeds ${MAX_OUTCOME_LABEL_LENGTH} characters`);
159
+ }
160
+ }
161
+ // Check for duplicates
162
+ const uniqueOutcomes = new Set(params.outcomes.map(o => o.toLowerCase().trim()));
163
+ if (uniqueOutcomes.size !== params.outcomes.length) {
164
+ errors.push('Outcome labels must be unique');
165
+ }
166
+ }
167
+ // =========================================================================
168
+ // Layer-Specific Validation
169
+ // =========================================================================
170
+ let creationFeeSol;
171
+ let platformFeeBps;
172
+ switch (params.layer) {
173
+ case 'official':
174
+ creationFeeSol = FEES.OFFICIAL_CREATION_FEE / 1e9;
175
+ platformFeeBps = FEES.OFFICIAL_PLATFORM_FEE_BPS;
176
+ warnings.push('Official markets require admin approval');
177
+ break;
178
+ case 'private':
179
+ creationFeeSol = FEES.PRIVATE_CREATION_FEE / 1e9;
180
+ platformFeeBps = FEES.PRIVATE_PLATFORM_FEE_BPS;
181
+ if (!params.inviteHash) {
182
+ warnings.push('Private markets can use invite_hash for restricted access');
183
+ }
184
+ break;
185
+ case 'lab':
186
+ default:
187
+ creationFeeSol = FEES.LAB_CREATION_FEE / 1e9;
188
+ platformFeeBps = FEES.LAB_PLATFORM_FEE_BPS;
189
+ break;
190
+ }
191
+ // =========================================================================
192
+ // Rent Estimation
193
+ // =========================================================================
194
+ let estimatedRentSol;
195
+ if (params.outcomes) {
196
+ estimatedRentSol = (RACE_MARKET_BASE_RENT + (params.outcomes.length * RACE_OUTCOME_RENT)) / 1e9;
197
+ }
198
+ else {
199
+ estimatedRentSol = MARKET_RENT / 1e9;
200
+ }
201
+ return {
202
+ valid: errors.length === 0,
203
+ errors,
204
+ warnings,
205
+ suggestions,
206
+ computed: {
207
+ ruleType,
208
+ bufferHours,
209
+ recommendedClosingTime,
210
+ creationFeeSol,
211
+ platformFeeBps,
212
+ estimatedRentSol,
213
+ },
214
+ };
215
+ }
216
+ // =============================================================================
217
+ // HELPER FUNCTIONS
218
+ // =============================================================================
219
+ /**
220
+ * Calculate recommended resolution time from closing time
221
+ */
222
+ export function calculateResolutionTime(closingTime, marketType, eventTime) {
223
+ if (marketType === 'event' && eventTime) {
224
+ // Resolution = event time + 1 hour buffer
225
+ return new Date(eventTime.getTime() + (60 * 60 * 1000));
226
+ }
227
+ // Default: closing time + 1 day
228
+ return new Date(closingTime.getTime() + (24 * 60 * 60 * 1000));
229
+ }
230
+ /**
231
+ * Calculate recommended closing time for event
232
+ */
233
+ export function calculateRecommendedClosingTime(eventTime, bufferHours = TIMING.RECOMMENDED_EVENT_BUFFER_HOURS) {
234
+ return new Date(eventTime.getTime() - (bufferHours * 60 * 60 * 1000));
235
+ }
236
+ /**
237
+ * Get creation fee for layer
238
+ */
239
+ export function getCreationFee(layer) {
240
+ let lamports;
241
+ switch (layer) {
242
+ case 'official':
243
+ lamports = FEES.OFFICIAL_CREATION_FEE;
244
+ break;
245
+ case 'private':
246
+ lamports = FEES.PRIVATE_CREATION_FEE;
247
+ break;
248
+ case 'lab':
249
+ default:
250
+ lamports = FEES.LAB_CREATION_FEE;
251
+ break;
252
+ }
253
+ return { lamports, sol: lamports / 1e9 };
254
+ }
255
+ /**
256
+ * Validate question format
257
+ */
258
+ export function validateQuestion(question) {
259
+ const errors = [];
260
+ const suggestions = [];
261
+ if (!question || question.trim().length === 0) {
262
+ errors.push('Question is required');
263
+ }
264
+ else {
265
+ if (question.length > MAX_QUESTION_LENGTH) {
266
+ errors.push(`Question exceeds ${MAX_QUESTION_LENGTH} characters`);
267
+ }
268
+ if (!question.endsWith('?')) {
269
+ suggestions.push('Add a question mark at the end');
270
+ }
271
+ if (question.length < 10) {
272
+ suggestions.push('Consider a more descriptive question');
273
+ }
274
+ }
275
+ return { valid: errors.length === 0, errors, suggestions };
276
+ }
277
+ /**
278
+ * Validate race outcomes
279
+ */
280
+ export function validateRaceOutcomes(outcomes) {
281
+ const errors = [];
282
+ if (outcomes.length < MIN_OUTCOMES) {
283
+ errors.push(`Minimum ${MIN_OUTCOMES} outcomes required`);
284
+ }
285
+ if (outcomes.length > MAX_OUTCOMES) {
286
+ errors.push(`Maximum ${MAX_OUTCOMES} outcomes allowed`);
287
+ }
288
+ for (let i = 0; i < outcomes.length; i++) {
289
+ if (!outcomes[i] || outcomes[i].trim().length === 0) {
290
+ errors.push(`Outcome ${i + 1} is empty`);
291
+ }
292
+ else if (outcomes[i].length > MAX_OUTCOME_LABEL_LENGTH) {
293
+ errors.push(`Outcome ${i + 1} exceeds ${MAX_OUTCOME_LABEL_LENGTH} chars`);
294
+ }
295
+ }
296
+ const unique = new Set(outcomes.map(o => o.toLowerCase().trim()));
297
+ if (unique.size !== outcomes.length) {
298
+ errors.push('Outcomes must be unique');
299
+ }
300
+ return { valid: errors.length === 0, errors };
301
+ }
302
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"creation-rules.js","sourceRoot":"","sources":["../../src/validation/creation-rules.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Market Creation Validation Rules (v6.2 Compliant)\n *\n * Implements validation for:\n * - Rule A: Event-based markets (12-24h buffer before event)\n * - Rule B: Measurement-period markets (close before measurement starts)\n * - Race market outcome validation\n * - Question and timing constraints\n */\n\nimport { TIMING, FEES, MARKET_LAYER } from '../config.js';\n\n// =============================================================================\n// TYPES\n// =============================================================================\n\nexport interface CreateMarketParams {\n  question: string;\n  closingTime: Date;\n  resolutionTime: Date;\n  layer: 'official' | 'lab' | 'private';\n\n  // Event-based (Rule A)\n  marketType?: 'event' | 'measurement';\n  eventTime?: Date;\n\n  // Measurement-period (Rule B)\n  measurementStart?: Date;\n  measurementEnd?: Date;\n\n  // Race markets\n  outcomes?: string[];\n\n  // Private market\n  inviteHash?: string;\n}\n\nexport interface CreationValidationResult {\n  valid: boolean;\n  errors: string[];\n  warnings: string[];\n  suggestions: string[];\n\n  // Computed values\n  computed: {\n    ruleType: 'A' | 'B' | 'unknown';\n    bufferHours?: number;\n    recommendedClosingTime?: string;\n    creationFeeSol: number;\n    platformFeeBps: number;\n    estimatedRentSol: number;\n  };\n}\n\n// =============================================================================\n// CONSTANTS\n// =============================================================================\n\nconst MAX_QUESTION_LENGTH = 200;\nconst MIN_OUTCOMES = 2;\nconst MAX_OUTCOMES = 10;\nconst MAX_OUTCOME_LABEL_LENGTH = 50;\nconst MIN_RESOLUTION_BUFFER_SECONDS = 600; // 10 minutes\nconst MAX_MARKET_DURATION_DAYS = 365;\n\n// Rent estimates (lamports)\nconst MARKET_RENT = 5_000_000; // ~0.005 SOL\nconst RACE_MARKET_BASE_RENT = 8_000_000; // ~0.008 SOL\nconst RACE_OUTCOME_RENT = 500_000; // ~0.0005 SOL per outcome\n\n// =============================================================================\n// MAIN VALIDATION\n// =============================================================================\n\n/**\n * Validate market creation parameters\n */\nexport function validateMarketCreation(params: CreateMarketParams): CreationValidationResult {\n  const errors: string[] = [];\n  const warnings: string[] = [];\n  const suggestions: string[] = [];\n\n  // Determine rule type\n  let ruleType: 'A' | 'B' | 'unknown' = 'unknown';\n  if (params.marketType === 'event' || params.eventTime) {\n    ruleType = 'A';\n  } else if (params.marketType === 'measurement' || params.measurementStart) {\n    ruleType = 'B';\n  }\n\n  // =========================================================================\n  // Question Validation\n  // =========================================================================\n\n  if (!params.question || params.question.trim().length === 0) {\n    errors.push('Question is required');\n  } else if (params.question.length > MAX_QUESTION_LENGTH) {\n    errors.push(`Question exceeds ${MAX_QUESTION_LENGTH} characters (got ${params.question.length})`);\n  }\n\n  if (!params.question.endsWith('?')) {\n    warnings.push('Question should end with a question mark for clarity');\n  }\n\n  // =========================================================================\n  // Timing Validation\n  // =========================================================================\n\n  const now = new Date();\n\n  // Closing time must be in future\n  if (params.closingTime <= now) {\n    errors.push('Closing time must be in the future');\n  }\n\n  // Maximum duration check\n  const durationDays = (params.closingTime.getTime() - now.getTime()) / (1000 * 60 * 60 * 24);\n  if (durationDays > MAX_MARKET_DURATION_DAYS) {\n    errors.push(`Market duration exceeds ${MAX_MARKET_DURATION_DAYS} days`);\n  }\n\n  // Resolution time must be after closing time\n  if (params.resolutionTime <= params.closingTime) {\n    errors.push('Resolution time must be after closing time');\n  }\n\n  // Minimum resolution buffer\n  const resolutionBufferSec = (params.resolutionTime.getTime() - params.closingTime.getTime()) / 1000;\n  if (resolutionBufferSec < MIN_RESOLUTION_BUFFER_SECONDS) {\n    errors.push(`Resolution buffer too short: ${resolutionBufferSec}s (min ${MIN_RESOLUTION_BUFFER_SECONDS}s)`);\n  }\n\n  // =========================================================================\n  // Rule A: Event-Based Validation\n  // =========================================================================\n\n  let bufferHours: number | undefined;\n  let recommendedClosingTime: string | undefined;\n\n  if (ruleType === 'A') {\n    if (!params.eventTime) {\n      errors.push('Event-based markets require event_time');\n    } else {\n      // Event must be after closing\n      if (params.eventTime <= params.closingTime) {\n        errors.push('Event time must be after closing time');\n      }\n\n      // Calculate buffer\n      bufferHours = (params.eventTime.getTime() - params.closingTime.getTime()) / (1000 * 60 * 60);\n\n      if (bufferHours < TIMING.MIN_EVENT_BUFFER_HOURS) {\n        errors.push(\n          `Event buffer too short: ${bufferHours.toFixed(1)}h. ` +\n          `Minimum ${TIMING.MIN_EVENT_BUFFER_HOURS}h required (v6.2 Rule A).`\n        );\n\n        // Suggest corrected closing time\n        const suggestedClose = new Date(\n          params.eventTime.getTime() - (TIMING.RECOMMENDED_EVENT_BUFFER_HOURS * 60 * 60 * 1000)\n        );\n        recommendedClosingTime = suggestedClose.toISOString();\n        suggestions.push(`Recommended closing time: ${recommendedClosingTime}`);\n      } else if (bufferHours < 18) {\n        warnings.push(\n          `Buffer is ${bufferHours.toFixed(1)}h. ` +\n          `Recommend 18-24h for safety margin (v6.2 Rule A).`\n        );\n      }\n\n      // Event must be in future\n      if (params.eventTime <= now) {\n        errors.push('Event time must be in the future');\n      }\n    }\n  }\n\n  // =========================================================================\n  // Rule B: Measurement-Period Validation\n  // =========================================================================\n\n  if (ruleType === 'B') {\n    if (!params.measurementStart) {\n      errors.push('Measurement-period markets require measurement_start');\n    } else {\n      // CRITICAL: Betting must close BEFORE measurement starts\n      if (params.closingTime >= params.measurementStart) {\n        const overlapHours = (params.closingTime.getTime() - params.measurementStart.getTime()) / (1000 * 60 * 60);\n        errors.push(\n          `INVALID: Betting closes ${overlapHours.toFixed(1)}h AFTER measurement starts. ` +\n          `This allows information advantage! (v6.2 Rule B)`\n        );\n\n        // Suggest corrected closing time\n        const suggestedClose = new Date(params.measurementStart.getTime() - (60 * 60 * 1000)); // 1h before\n        recommendedClosingTime = suggestedClose.toISOString();\n        suggestions.push(`Recommended closing time: ${recommendedClosingTime}`);\n      }\n\n      // Measurement end validation\n      if (params.measurementEnd) {\n        if (params.measurementEnd <= params.measurementStart) {\n          errors.push('Measurement end must be after measurement start');\n        }\n\n        const periodDays = (params.measurementEnd.getTime() - params.measurementStart.getTime()) / (1000 * 60 * 60 * 24);\n        if (periodDays > 7) {\n          warnings.push(\n            `Long measurement period: ${periodDays.toFixed(0)} days. ` +\n            `Prefer 2-7 days for better UX (v6.2 guidance).`\n          );\n        }\n      }\n    }\n  }\n\n  // =========================================================================\n  // Race Market Validation\n  // =========================================================================\n\n  if (params.outcomes) {\n    if (params.outcomes.length < MIN_OUTCOMES) {\n      errors.push(`Race markets require at least ${MIN_OUTCOMES} outcomes`);\n    }\n    if (params.outcomes.length > MAX_OUTCOMES) {\n      errors.push(`Race markets limited to ${MAX_OUTCOMES} outcomes (got ${params.outcomes.length})`);\n    }\n\n    // Check outcome labels\n    for (let i = 0; i < params.outcomes.length; i++) {\n      const outcome = params.outcomes[i];\n      if (!outcome || outcome.trim().length === 0) {\n        errors.push(`Outcome ${i} is empty`);\n      } else if (outcome.length > MAX_OUTCOME_LABEL_LENGTH) {\n        errors.push(`Outcome ${i} exceeds ${MAX_OUTCOME_LABEL_LENGTH} characters`);\n      }\n    }\n\n    // Check for duplicates\n    const uniqueOutcomes = new Set(params.outcomes.map(o => o.toLowerCase().trim()));\n    if (uniqueOutcomes.size !== params.outcomes.length) {\n      errors.push('Outcome labels must be unique');\n    }\n  }\n\n  // =========================================================================\n  // Layer-Specific Validation\n  // =========================================================================\n\n  let creationFeeSol: number;\n  let platformFeeBps: number;\n\n  switch (params.layer) {\n    case 'official':\n      creationFeeSol = FEES.OFFICIAL_CREATION_FEE / 1e9;\n      platformFeeBps = FEES.OFFICIAL_PLATFORM_FEE_BPS;\n      warnings.push('Official markets require admin approval');\n      break;\n    case 'private':\n      creationFeeSol = FEES.PRIVATE_CREATION_FEE / 1e9;\n      platformFeeBps = FEES.PRIVATE_PLATFORM_FEE_BPS;\n      if (!params.inviteHash) {\n        warnings.push('Private markets can use invite_hash for restricted access');\n      }\n      break;\n    case 'lab':\n    default:\n      creationFeeSol = FEES.LAB_CREATION_FEE / 1e9;\n      platformFeeBps = FEES.LAB_PLATFORM_FEE_BPS;\n      break;\n  }\n\n  // =========================================================================\n  // Rent Estimation\n  // =========================================================================\n\n  let estimatedRentSol: number;\n  if (params.outcomes) {\n    estimatedRentSol = (RACE_MARKET_BASE_RENT + (params.outcomes.length * RACE_OUTCOME_RENT)) / 1e9;\n  } else {\n    estimatedRentSol = MARKET_RENT / 1e9;\n  }\n\n  return {\n    valid: errors.length === 0,\n    errors,\n    warnings,\n    suggestions,\n    computed: {\n      ruleType,\n      bufferHours,\n      recommendedClosingTime,\n      creationFeeSol,\n      platformFeeBps,\n      estimatedRentSol,\n    },\n  };\n}\n\n// =============================================================================\n// HELPER FUNCTIONS\n// =============================================================================\n\n/**\n * Calculate recommended resolution time from closing time\n */\nexport function calculateResolutionTime(\n  closingTime: Date,\n  marketType: 'event' | 'measurement',\n  eventTime?: Date\n): Date {\n  if (marketType === 'event' && eventTime) {\n    // Resolution = event time + 1 hour buffer\n    return new Date(eventTime.getTime() + (60 * 60 * 1000));\n  }\n  // Default: closing time + 1 day\n  return new Date(closingTime.getTime() + (24 * 60 * 60 * 1000));\n}\n\n/**\n * Calculate recommended closing time for event\n */\nexport function calculateRecommendedClosingTime(\n  eventTime: Date,\n  bufferHours: number = TIMING.RECOMMENDED_EVENT_BUFFER_HOURS\n): Date {\n  return new Date(eventTime.getTime() - (bufferHours * 60 * 60 * 1000));\n}\n\n/**\n * Get creation fee for layer\n */\nexport function getCreationFee(layer: 'official' | 'lab' | 'private'): {\n  lamports: number;\n  sol: number;\n} {\n  let lamports: number;\n  switch (layer) {\n    case 'official':\n      lamports = FEES.OFFICIAL_CREATION_FEE;\n      break;\n    case 'private':\n      lamports = FEES.PRIVATE_CREATION_FEE;\n      break;\n    case 'lab':\n    default:\n      lamports = FEES.LAB_CREATION_FEE;\n      break;\n  }\n  return { lamports, sol: lamports / 1e9 };\n}\n\n/**\n * Validate question format\n */\nexport function validateQuestion(question: string): {\n  valid: boolean;\n  errors: string[];\n  suggestions: string[];\n} {\n  const errors: string[] = [];\n  const suggestions: string[] = [];\n\n  if (!question || question.trim().length === 0) {\n    errors.push('Question is required');\n  } else {\n    if (question.length > MAX_QUESTION_LENGTH) {\n      errors.push(`Question exceeds ${MAX_QUESTION_LENGTH} characters`);\n    }\n    if (!question.endsWith('?')) {\n      suggestions.push('Add a question mark at the end');\n    }\n    if (question.length < 10) {\n      suggestions.push('Consider a more descriptive question');\n    }\n  }\n\n  return { valid: errors.length === 0, errors, suggestions };\n}\n\n/**\n * Validate race outcomes\n */\nexport function validateRaceOutcomes(outcomes: string[]): {\n  valid: boolean;\n  errors: string[];\n} {\n  const errors: string[] = [];\n\n  if (outcomes.length < MIN_OUTCOMES) {\n    errors.push(`Minimum ${MIN_OUTCOMES} outcomes required`);\n  }\n  if (outcomes.length > MAX_OUTCOMES) {\n    errors.push(`Maximum ${MAX_OUTCOMES} outcomes allowed`);\n  }\n\n  for (let i = 0; i < outcomes.length; i++) {\n    if (!outcomes[i] || outcomes[i].trim().length === 0) {\n      errors.push(`Outcome ${i + 1} is empty`);\n    } else if (outcomes[i].length > MAX_OUTCOME_LABEL_LENGTH) {\n      errors.push(`Outcome ${i + 1} exceeds ${MAX_OUTCOME_LABEL_LENGTH} chars`);\n    }\n  }\n\n  const unique = new Set(outcomes.map(o => o.toLowerCase().trim()));\n  if (unique.size !== outcomes.length) {\n    errors.push('Outcomes must be unique');\n  }\n\n  return { valid: errors.length === 0, errors };\n}\n"]}
@@ -0,0 +1,6 @@
1
+ /**
2
+ * Validation Engine Index
3
+ * Exports all validation functions and types
4
+ */
5
+ export * from './market-rules.js';
6
+ export * from './bet-rules.js';
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Validation Engine Index
3
+ * Exports all validation functions and types
4
+ */
5
+ export * from './market-rules.js';
6
+ export * from './bet-rules.js';
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdmFsaWRhdGlvbi9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxjQUFjLG1CQUFtQixDQUFDO0FBQ2xDLGNBQWMsZ0JBQWdCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFZhbGlkYXRpb24gRW5naW5lIEluZGV4XG4gKiBFeHBvcnRzIGFsbCB2YWxpZGF0aW9uIGZ1bmN0aW9ucyBhbmQgdHlwZXNcbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL21hcmtldC1ydWxlcy5qcyc7XG5leHBvcnQgKiBmcm9tICcuL2JldC1ydWxlcy5qcyc7XG4iXX0=
@@ -0,0 +1,60 @@
1
+ /**
2
+ * Market Validation Rules (v6.2)
3
+ *
4
+ * Implements timing validation for market creation based on:
5
+ * - Rule A: Event-Based Markets (single point in time)
6
+ * - Rule B: Measurement-Period Markets (outcome over time range)
7
+ */
8
+ export interface MarketTimingParams {
9
+ question: string;
10
+ closingTime: Date;
11
+ marketType: 'event' | 'measurement';
12
+ eventTime?: Date;
13
+ measurementStart?: Date;
14
+ measurementEnd?: Date;
15
+ }
16
+ export interface MarketValidation {
17
+ valid: boolean;
18
+ ruleType: 'A' | 'B';
19
+ errors: string[];
20
+ warnings: string[];
21
+ suggestions: string[];
22
+ timing?: {
23
+ bufferHours?: number;
24
+ recommendedClose?: Date;
25
+ measurementDays?: number;
26
+ };
27
+ }
28
+ /**
29
+ * Validate market timing parameters against v6.2 rules
30
+ *
31
+ * Rule A (Event-Based):
32
+ * - Betting closes BEFORE the event occurs
33
+ * - Minimum 12h buffer between close and event
34
+ * - Recommended 18-24h buffer for safety
35
+ *
36
+ * Rule B (Measurement-Period):
37
+ * - Betting must close BEFORE measurement starts
38
+ * - Measurement period should be well-defined
39
+ * - Prefer 2-7 day measurement periods for UX
40
+ */
41
+ export declare function validateMarketTiming(params: MarketTimingParams): MarketValidation;
42
+ /**
43
+ * Generate timing suggestions based on market parameters
44
+ */
45
+ export declare function generateTimingSuggestions(params: MarketTimingParams): string[];
46
+ /**
47
+ * Check if a market question follows best practices
48
+ */
49
+ export declare function validateQuestionFormat(question: string): {
50
+ valid: boolean;
51
+ issues: string[];
52
+ };
53
+ /**
54
+ * Calculate recommended times for a market
55
+ */
56
+ export declare function calculateRecommendedTimes(eventOrMeasurementStart: Date, marketType: 'event' | 'measurement'): {
57
+ recommendedClose: Date;
58
+ latestClose: Date;
59
+ earliestClose: Date;
60
+ };