@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,270 @@
1
+ /**
2
+ * BAOZI PARIMUTUEL MARKET RULES v6.2
3
+ *
4
+ * STRICT ENFORCEMENT - All Lab markets MUST comply with these rules.
5
+ * AI agents creating markets through MCP MUST validate against these rules.
6
+ * Markets that don't comply will be BLOCKED from creation.
7
+ */
8
+ // =============================================================================
9
+ // MANDATORY RULES - MARKETS WILL BE BLOCKED IF VIOLATED
10
+ // =============================================================================
11
+ export const PARIMUTUEL_RULES = {
12
+ version: '6.2',
13
+ /**
14
+ * RULE A: Event-Based Markets
15
+ *
16
+ * For markets about specific events (sports, elections, announcements):
17
+ * - Betting must close AT LEAST 12 hours BEFORE the event
18
+ * - Recommended buffer: 18-24 hours
19
+ * - Event time must be explicitly specified
20
+ *
21
+ * RATIONALE: Prevents late-breaking information from giving unfair advantage
22
+ *
23
+ * Examples:
24
+ * ✅ "Will Team A win vs Team B?" + event_time = game_start
25
+ * ✅ "Will Company X announce earnings above $1B?" + event_time = announcement
26
+ * ❌ "Will it rain tomorrow?" (no clear event time)
27
+ */
28
+ RULE_A: {
29
+ name: 'Event-Based Markets',
30
+ minBufferHours: 12,
31
+ recommendedBufferHours: 24,
32
+ requirement: 'Betting must close 12+ hours BEFORE the event',
33
+ rationale: 'Prevents information advantage from late-breaking news',
34
+ },
35
+ /**
36
+ * RULE B: Measurement-Period Markets
37
+ *
38
+ * For markets about measured values (prices, temperatures, metrics):
39
+ * - Betting must close BEFORE the measurement period starts
40
+ * - measurement_start must be explicitly specified
41
+ * - Recommended: betting closes 1-2 hours before measurement
42
+ *
43
+ * RATIONALE: Prevents anyone from betting with foreknowledge of measurements
44
+ *
45
+ * Examples:
46
+ * ✅ "Will BTC be above $100k at 00:00 UTC Feb 1?" + measurement_start = Feb 1 00:00
47
+ * ✅ "Will Tokyo have snowfall on Feb 4?" + measurement_start = Feb 4 00:00
48
+ * ❌ "Will BTC close above $100k on Feb 2?" + betting_closes = Feb 2 23:59 (VIOLATION!)
49
+ */
50
+ RULE_B: {
51
+ name: 'Measurement-Period Markets',
52
+ requirement: 'Betting must close BEFORE measurement period starts',
53
+ rationale: 'Prevents betting with foreknowledge of measured values',
54
+ recommendedBufferHours: 2,
55
+ },
56
+ /**
57
+ * MANDATORY: Verifiable Data Source
58
+ *
59
+ * Every market question MUST specify or clearly imply a verifiable data source.
60
+ * This ensures objective resolution and prevents disputes.
61
+ *
62
+ * Examples:
63
+ * ✅ "Will BTC be above $100k? (Source: CoinGecko)"
64
+ * ✅ "Will it snow in Tokyo? (JMA official record)"
65
+ * ✅ "Will Real Madrid win?" (Implied: official UEFA result)
66
+ * ❌ "Will the economy improve?" (No verifiable source)
67
+ * ❌ "Will Claude be the best AI?" (Subjective, no source)
68
+ */
69
+ DATA_SOURCE: {
70
+ name: 'Verifiable Data Source',
71
+ requirement: 'Question must specify or clearly imply a verifiable data source',
72
+ rationale: 'Ensures objective resolution and prevents disputes',
73
+ },
74
+ /**
75
+ * MANDATORY: Clear YES/NO Criteria
76
+ *
77
+ * The market question must have clear, unambiguous YES/NO criteria.
78
+ * There should be no room for interpretation in the resolution.
79
+ *
80
+ * Examples:
81
+ * ✅ "Will BTC be above $100,000 at 00:00 UTC Feb 1, 2026?"
82
+ * ✅ "Will Team A score 3+ goals?"
83
+ * ❌ "Will BTC perform well?" (Subjective)
84
+ * ❌ "Will the game be exciting?" (Subjective)
85
+ */
86
+ CLEAR_CRITERIA: {
87
+ name: 'Clear Resolution Criteria',
88
+ requirement: 'Question must have unambiguous YES/NO criteria',
89
+ rationale: 'Prevents disputes and ensures fair resolution',
90
+ },
91
+ };
92
+ /**
93
+ * Validate market against parimutuel rules
94
+ * Returns BLOCKED=true if market violates mandatory rules
95
+ */
96
+ export function validateParimutuelRules(params) {
97
+ const errors = [];
98
+ const warnings = [];
99
+ const ruleViolations = [];
100
+ const rulesChecked = [];
101
+ // Only strictly enforce for Lab markets
102
+ const isLabMarket = params.layer === 'lab';
103
+ // =========================================================================
104
+ // CHECK: Market Type Classification
105
+ // =========================================================================
106
+ rulesChecked.push('Market Type Classification');
107
+ const hasEventTime = !!params.eventTime;
108
+ const hasMeasurementStart = !!params.measurementStart;
109
+ const isEventBased = params.marketType === 'event' || hasEventTime;
110
+ const isMeasurementBased = params.marketType === 'measurement' || hasMeasurementStart;
111
+ if (isLabMarket && !isEventBased && !isMeasurementBased) {
112
+ ruleViolations.push({
113
+ rule: 'Market Classification',
114
+ description: 'Lab markets MUST specify either event_time (Rule A) or measurement_start (Rule B). ' +
115
+ 'Without this, the market cannot be validated for fair betting windows.',
116
+ severity: 'CRITICAL',
117
+ });
118
+ errors.push('BLOCKED: Market must be classified as event-based (with event_time) or measurement-based (with measurement_start)');
119
+ }
120
+ // =========================================================================
121
+ // CHECK: Rule A - Event Buffer
122
+ // =========================================================================
123
+ if (isEventBased && params.eventTime) {
124
+ rulesChecked.push('Rule A: Event Buffer');
125
+ const bufferMs = params.eventTime.getTime() - params.closingTime.getTime();
126
+ const bufferHours = bufferMs / (1000 * 60 * 60);
127
+ if (bufferHours < PARIMUTUEL_RULES.RULE_A.minBufferHours) {
128
+ ruleViolations.push({
129
+ rule: 'Rule A',
130
+ description: `Event buffer is ${bufferHours.toFixed(1)}h but minimum is ${PARIMUTUEL_RULES.RULE_A.minBufferHours}h. ` +
131
+ `Betting must close at least 12 hours BEFORE the event to prevent information advantage.`,
132
+ severity: 'CRITICAL',
133
+ });
134
+ errors.push(`BLOCKED: Betting must close ${PARIMUTUEL_RULES.RULE_A.minBufferHours}+ hours before event (currently ${bufferHours.toFixed(1)}h)`);
135
+ }
136
+ else if (bufferHours < 18) {
137
+ warnings.push(`Event buffer is ${bufferHours.toFixed(1)}h. Recommend 18-24h for safety margin.`);
138
+ }
139
+ }
140
+ // =========================================================================
141
+ // CHECK: Rule B - Measurement Period
142
+ // =========================================================================
143
+ if (isMeasurementBased && params.measurementStart) {
144
+ rulesChecked.push('Rule B: Measurement Period');
145
+ if (params.closingTime >= params.measurementStart) {
146
+ const overlapMs = params.closingTime.getTime() - params.measurementStart.getTime();
147
+ const overlapHours = overlapMs / (1000 * 60 * 60);
148
+ ruleViolations.push({
149
+ rule: 'Rule B',
150
+ description: `CRITICAL VIOLATION: Betting closes ${overlapHours.toFixed(1)}h AFTER measurement starts! ` +
151
+ `This allows bettors to bet with foreknowledge of the outcome. ` +
152
+ `Betting MUST close BEFORE the measurement period begins.`,
153
+ severity: 'CRITICAL',
154
+ });
155
+ errors.push(`BLOCKED: Betting must close BEFORE measurement starts (currently closes ${overlapHours.toFixed(1)}h AFTER)`);
156
+ }
157
+ }
158
+ // =========================================================================
159
+ // CHECK: Data Source
160
+ // =========================================================================
161
+ rulesChecked.push('Verifiable Data Source');
162
+ const questionLower = params.question.toLowerCase();
163
+ const hasDataSource = questionLower.includes('source:') ||
164
+ questionLower.includes('coingecko') ||
165
+ questionLower.includes('coinmarketcap') ||
166
+ questionLower.includes('official') ||
167
+ questionLower.includes('nws') ||
168
+ questionLower.includes('jma') ||
169
+ questionLower.includes('ufc') ||
170
+ questionLower.includes('uefa') ||
171
+ questionLower.includes('fifa') ||
172
+ questionLower.includes('nba') ||
173
+ questionLower.includes('nfl') ||
174
+ questionLower.includes('mlb') ||
175
+ // Sports/events typically have implied official sources
176
+ questionLower.includes(' win ') ||
177
+ questionLower.includes(' defeat ') ||
178
+ questionLower.includes(' advance ') ||
179
+ questionLower.includes('championship') ||
180
+ questionLower.includes('election');
181
+ if (isLabMarket && !hasDataSource) {
182
+ warnings.push('Recommended: Include data source in question (e.g., "(Source: CoinGecko)" or "(Official: UEFA)"). ' +
183
+ 'This ensures objective resolution.');
184
+ }
185
+ // =========================================================================
186
+ // CHECK: Clear Criteria
187
+ // =========================================================================
188
+ rulesChecked.push('Clear Resolution Criteria');
189
+ const hasNumericThreshold = /\$[\d,]+/.test(params.question) || // Dollar amounts
190
+ /\d+%/.test(params.question) || // Percentages
191
+ /above|below|over|under|at least|more than|less than/i.test(params.question);
192
+ const hasClearBinaryOutcome = /will .+ (win|lose|defeat|advance|qualify|score|achieve)/i.test(params.question) ||
193
+ /will .+ (snow|rain|happen|occur)/i.test(params.question);
194
+ if (isLabMarket && !hasNumericThreshold && !hasClearBinaryOutcome) {
195
+ warnings.push('Question should have clear numeric threshold or binary outcome. ' +
196
+ 'Example: "above $X", "at least Y goals", "will Team A win"');
197
+ }
198
+ // =========================================================================
199
+ // RESULT
200
+ // =========================================================================
201
+ const hasCriticalViolation = ruleViolations.some(v => v.severity === 'CRITICAL');
202
+ return {
203
+ valid: errors.length === 0,
204
+ blocked: isLabMarket && hasCriticalViolation,
205
+ errors,
206
+ warnings,
207
+ ruleViolations,
208
+ rulesChecked,
209
+ };
210
+ }
211
+ // =============================================================================
212
+ // RULES DOCUMENTATION (for AI agents)
213
+ // =============================================================================
214
+ export const PARIMUTUEL_RULES_DOCUMENTATION = `
215
+ # BAOZI PARIMUTUEL MARKET RULES v6.2
216
+
217
+ ## MANDATORY FOR ALL LAB MARKETS
218
+
219
+ ### Rule A: Event-Based Markets
220
+ Markets about specific events (sports, elections, announcements):
221
+ - Betting MUST close AT LEAST 12 hours BEFORE the event
222
+ - You MUST specify event_time parameter
223
+ - Recommended buffer: 18-24 hours
224
+
225
+ Example:
226
+ - Question: "Will Team A win vs Team B?"
227
+ - Event time: Game start (e.g., 2026-02-15T20:00:00Z)
228
+ - Closing time: At least 12h before (e.g., 2026-02-15T08:00:00Z)
229
+
230
+ ### Rule B: Measurement-Period Markets
231
+ Markets about measured values (prices, temperatures, metrics):
232
+ - Betting MUST close BEFORE the measurement period starts
233
+ - You MUST specify measurement_start parameter
234
+ - Recommended: betting closes 1-2 hours before measurement
235
+
236
+ Example:
237
+ - Question: "Will BTC be above $100k at 00:00 UTC Feb 1?"
238
+ - Measurement start: 2026-02-01T00:00:00Z
239
+ - Closing time: BEFORE measurement (e.g., 2026-01-31T22:00:00Z)
240
+
241
+ ### Data Source Requirement
242
+ - Include verifiable data source in question
243
+ - Examples: "(Source: CoinGecko)", "(Official: UEFA)", "(NWS Los Angeles)"
244
+
245
+ ### Clear Criteria
246
+ - Question must have unambiguous YES/NO resolution
247
+ - Include specific thresholds: "$100,000", "3+ goals", etc.
248
+
249
+ ## VIOLATIONS WILL BLOCK MARKET CREATION
250
+
251
+ If you try to create a market that violates these rules, the MCP will:
252
+ 1. Return blocked: true
253
+ 2. List specific rule violations
254
+ 3. Refuse to build the transaction
255
+
256
+ ## CLASSIFICATION REQUIRED
257
+
258
+ Every Lab market MUST be classified as either:
259
+ 1. Event-based (provide event_time) - Rule A applies
260
+ 2. Measurement-based (provide measurement_start) - Rule B applies
261
+
262
+ Unclassified markets will be BLOCKED.
263
+ `;
264
+ /**
265
+ * Get rules summary for AI agents
266
+ */
267
+ export function getParimutuelRulesSummary() {
268
+ return PARIMUTUEL_RULES_DOCUMENTATION;
269
+ }
270
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parimutuel-rules.js","sourceRoot":"","sources":["../../src/validation/parimutuel-rules.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * BAOZI PARIMUTUEL MARKET RULES v6.2\n *\n * STRICT ENFORCEMENT - All Lab markets MUST comply with these rules.\n * AI agents creating markets through MCP MUST validate against these rules.\n * Markets that don't comply will be BLOCKED from creation.\n */\n\n// =============================================================================\n// MANDATORY RULES - MARKETS WILL BE BLOCKED IF VIOLATED\n// =============================================================================\n\nexport const PARIMUTUEL_RULES = {\n  version: '6.2',\n\n  /**\n   * RULE A: Event-Based Markets\n   *\n   * For markets about specific events (sports, elections, announcements):\n   * - Betting must close AT LEAST 12 hours BEFORE the event\n   * - Recommended buffer: 18-24 hours\n   * - Event time must be explicitly specified\n   *\n   * RATIONALE: Prevents late-breaking information from giving unfair advantage\n   *\n   * Examples:\n   * ✅ \"Will Team A win vs Team B?\" + event_time = game_start\n   * ✅ \"Will Company X announce earnings above $1B?\" + event_time = announcement\n   * ❌ \"Will it rain tomorrow?\" (no clear event time)\n   */\n  RULE_A: {\n    name: 'Event-Based Markets',\n    minBufferHours: 12,\n    recommendedBufferHours: 24,\n    requirement: 'Betting must close 12+ hours BEFORE the event',\n    rationale: 'Prevents information advantage from late-breaking news',\n  },\n\n  /**\n   * RULE B: Measurement-Period Markets\n   *\n   * For markets about measured values (prices, temperatures, metrics):\n   * - Betting must close BEFORE the measurement period starts\n   * - measurement_start must be explicitly specified\n   * - Recommended: betting closes 1-2 hours before measurement\n   *\n   * RATIONALE: Prevents anyone from betting with foreknowledge of measurements\n   *\n   * Examples:\n   * ✅ \"Will BTC be above $100k at 00:00 UTC Feb 1?\" + measurement_start = Feb 1 00:00\n   * ✅ \"Will Tokyo have snowfall on Feb 4?\" + measurement_start = Feb 4 00:00\n   * ❌ \"Will BTC close above $100k on Feb 2?\" + betting_closes = Feb 2 23:59 (VIOLATION!)\n   */\n  RULE_B: {\n    name: 'Measurement-Period Markets',\n    requirement: 'Betting must close BEFORE measurement period starts',\n    rationale: 'Prevents betting with foreknowledge of measured values',\n    recommendedBufferHours: 2,\n  },\n\n  /**\n   * MANDATORY: Verifiable Data Source\n   *\n   * Every market question MUST specify or clearly imply a verifiable data source.\n   * This ensures objective resolution and prevents disputes.\n   *\n   * Examples:\n   * ✅ \"Will BTC be above $100k? (Source: CoinGecko)\"\n   * ✅ \"Will it snow in Tokyo? (JMA official record)\"\n   * ✅ \"Will Real Madrid win?\" (Implied: official UEFA result)\n   * ❌ \"Will the economy improve?\" (No verifiable source)\n   * ❌ \"Will Claude be the best AI?\" (Subjective, no source)\n   */\n  DATA_SOURCE: {\n    name: 'Verifiable Data Source',\n    requirement: 'Question must specify or clearly imply a verifiable data source',\n    rationale: 'Ensures objective resolution and prevents disputes',\n  },\n\n  /**\n   * MANDATORY: Clear YES/NO Criteria\n   *\n   * The market question must have clear, unambiguous YES/NO criteria.\n   * There should be no room for interpretation in the resolution.\n   *\n   * Examples:\n   * ✅ \"Will BTC be above $100,000 at 00:00 UTC Feb 1, 2026?\"\n   * ✅ \"Will Team A score 3+ goals?\"\n   * ❌ \"Will BTC perform well?\" (Subjective)\n   * ❌ \"Will the game be exciting?\" (Subjective)\n   */\n  CLEAR_CRITERIA: {\n    name: 'Clear Resolution Criteria',\n    requirement: 'Question must have unambiguous YES/NO criteria',\n    rationale: 'Prevents disputes and ensures fair resolution',\n  },\n};\n\n// =============================================================================\n// STRICT VALIDATION FOR LAB MARKETS\n// =============================================================================\n\nexport interface ParimutuelValidationResult {\n  valid: boolean;\n  blocked: boolean;  // If true, market creation is BLOCKED\n  errors: string[];\n  warnings: string[];\n  ruleViolations: {\n    rule: string;\n    description: string;\n    severity: 'CRITICAL' | 'ERROR' | 'WARNING';\n  }[];\n  rulesChecked: string[];\n}\n\n/**\n * Validate market against parimutuel rules\n * Returns BLOCKED=true if market violates mandatory rules\n */\nexport function validateParimutuelRules(params: {\n  question: string;\n  closingTime: Date;\n  marketType?: 'event' | 'measurement';\n  eventTime?: Date;\n  measurementStart?: Date;\n  layer: 'official' | 'lab' | 'private';\n}): ParimutuelValidationResult {\n  const errors: string[] = [];\n  const warnings: string[] = [];\n  const ruleViolations: ParimutuelValidationResult['ruleViolations'] = [];\n  const rulesChecked: string[] = [];\n\n  // Only strictly enforce for Lab markets\n  const isLabMarket = params.layer === 'lab';\n\n  // =========================================================================\n  // CHECK: Market Type Classification\n  // =========================================================================\n  rulesChecked.push('Market Type Classification');\n\n  const hasEventTime = !!params.eventTime;\n  const hasMeasurementStart = !!params.measurementStart;\n  const isEventBased = params.marketType === 'event' || hasEventTime;\n  const isMeasurementBased = params.marketType === 'measurement' || hasMeasurementStart;\n\n  if (isLabMarket && !isEventBased && !isMeasurementBased) {\n    ruleViolations.push({\n      rule: 'Market Classification',\n      description: 'Lab markets MUST specify either event_time (Rule A) or measurement_start (Rule B). ' +\n        'Without this, the market cannot be validated for fair betting windows.',\n      severity: 'CRITICAL',\n    });\n    errors.push('BLOCKED: Market must be classified as event-based (with event_time) or measurement-based (with measurement_start)');\n  }\n\n  // =========================================================================\n  // CHECK: Rule A - Event Buffer\n  // =========================================================================\n  if (isEventBased && params.eventTime) {\n    rulesChecked.push('Rule A: Event Buffer');\n\n    const bufferMs = params.eventTime.getTime() - params.closingTime.getTime();\n    const bufferHours = bufferMs / (1000 * 60 * 60);\n\n    if (bufferHours < PARIMUTUEL_RULES.RULE_A.minBufferHours) {\n      ruleViolations.push({\n        rule: 'Rule A',\n        description: `Event buffer is ${bufferHours.toFixed(1)}h but minimum is ${PARIMUTUEL_RULES.RULE_A.minBufferHours}h. ` +\n          `Betting must close at least 12 hours BEFORE the event to prevent information advantage.`,\n        severity: 'CRITICAL',\n      });\n      errors.push(`BLOCKED: Betting must close ${PARIMUTUEL_RULES.RULE_A.minBufferHours}+ hours before event (currently ${bufferHours.toFixed(1)}h)`);\n    } else if (bufferHours < 18) {\n      warnings.push(`Event buffer is ${bufferHours.toFixed(1)}h. Recommend 18-24h for safety margin.`);\n    }\n  }\n\n  // =========================================================================\n  // CHECK: Rule B - Measurement Period\n  // =========================================================================\n  if (isMeasurementBased && params.measurementStart) {\n    rulesChecked.push('Rule B: Measurement Period');\n\n    if (params.closingTime >= params.measurementStart) {\n      const overlapMs = params.closingTime.getTime() - params.measurementStart.getTime();\n      const overlapHours = overlapMs / (1000 * 60 * 60);\n\n      ruleViolations.push({\n        rule: 'Rule B',\n        description: `CRITICAL VIOLATION: Betting closes ${overlapHours.toFixed(1)}h AFTER measurement starts! ` +\n          `This allows bettors to bet with foreknowledge of the outcome. ` +\n          `Betting MUST close BEFORE the measurement period begins.`,\n        severity: 'CRITICAL',\n      });\n      errors.push(`BLOCKED: Betting must close BEFORE measurement starts (currently closes ${overlapHours.toFixed(1)}h AFTER)`);\n    }\n  }\n\n  // =========================================================================\n  // CHECK: Data Source\n  // =========================================================================\n  rulesChecked.push('Verifiable Data Source');\n\n  const questionLower = params.question.toLowerCase();\n  const hasDataSource =\n    questionLower.includes('source:') ||\n    questionLower.includes('coingecko') ||\n    questionLower.includes('coinmarketcap') ||\n    questionLower.includes('official') ||\n    questionLower.includes('nws') ||\n    questionLower.includes('jma') ||\n    questionLower.includes('ufc') ||\n    questionLower.includes('uefa') ||\n    questionLower.includes('fifa') ||\n    questionLower.includes('nba') ||\n    questionLower.includes('nfl') ||\n    questionLower.includes('mlb') ||\n    // Sports/events typically have implied official sources\n    questionLower.includes(' win ') ||\n    questionLower.includes(' defeat ') ||\n    questionLower.includes(' advance ') ||\n    questionLower.includes('championship') ||\n    questionLower.includes('election');\n\n  if (isLabMarket && !hasDataSource) {\n    warnings.push(\n      'Recommended: Include data source in question (e.g., \"(Source: CoinGecko)\" or \"(Official: UEFA)\"). ' +\n      'This ensures objective resolution.'\n    );\n  }\n\n  // =========================================================================\n  // CHECK: Clear Criteria\n  // =========================================================================\n  rulesChecked.push('Clear Resolution Criteria');\n\n  const hasNumericThreshold =\n    /\\$[\\d,]+/.test(params.question) ||  // Dollar amounts\n    /\\d+%/.test(params.question) ||      // Percentages\n    /above|below|over|under|at least|more than|less than/i.test(params.question);\n\n  const hasClearBinaryOutcome =\n    /will .+ (win|lose|defeat|advance|qualify|score|achieve)/i.test(params.question) ||\n    /will .+ (snow|rain|happen|occur)/i.test(params.question);\n\n  if (isLabMarket && !hasNumericThreshold && !hasClearBinaryOutcome) {\n    warnings.push(\n      'Question should have clear numeric threshold or binary outcome. ' +\n      'Example: \"above $X\", \"at least Y goals\", \"will Team A win\"'\n    );\n  }\n\n  // =========================================================================\n  // RESULT\n  // =========================================================================\n\n  const hasCriticalViolation = ruleViolations.some(v => v.severity === 'CRITICAL');\n\n  return {\n    valid: errors.length === 0,\n    blocked: isLabMarket && hasCriticalViolation,\n    errors,\n    warnings,\n    ruleViolations,\n    rulesChecked,\n  };\n}\n\n// =============================================================================\n// RULES DOCUMENTATION (for AI agents)\n// =============================================================================\n\nexport const PARIMUTUEL_RULES_DOCUMENTATION = `\n# BAOZI PARIMUTUEL MARKET RULES v6.2\n\n## MANDATORY FOR ALL LAB MARKETS\n\n### Rule A: Event-Based Markets\nMarkets about specific events (sports, elections, announcements):\n- Betting MUST close AT LEAST 12 hours BEFORE the event\n- You MUST specify event_time parameter\n- Recommended buffer: 18-24 hours\n\nExample:\n- Question: \"Will Team A win vs Team B?\"\n- Event time: Game start (e.g., 2026-02-15T20:00:00Z)\n- Closing time: At least 12h before (e.g., 2026-02-15T08:00:00Z)\n\n### Rule B: Measurement-Period Markets\nMarkets about measured values (prices, temperatures, metrics):\n- Betting MUST close BEFORE the measurement period starts\n- You MUST specify measurement_start parameter\n- Recommended: betting closes 1-2 hours before measurement\n\nExample:\n- Question: \"Will BTC be above $100k at 00:00 UTC Feb 1?\"\n- Measurement start: 2026-02-01T00:00:00Z\n- Closing time: BEFORE measurement (e.g., 2026-01-31T22:00:00Z)\n\n### Data Source Requirement\n- Include verifiable data source in question\n- Examples: \"(Source: CoinGecko)\", \"(Official: UEFA)\", \"(NWS Los Angeles)\"\n\n### Clear Criteria\n- Question must have unambiguous YES/NO resolution\n- Include specific thresholds: \"$100,000\", \"3+ goals\", etc.\n\n## VIOLATIONS WILL BLOCK MARKET CREATION\n\nIf you try to create a market that violates these rules, the MCP will:\n1. Return blocked: true\n2. List specific rule violations\n3. Refuse to build the transaction\n\n## CLASSIFICATION REQUIRED\n\nEvery Lab market MUST be classified as either:\n1. Event-based (provide event_time) - Rule A applies\n2. Measurement-based (provide measurement_start) - Rule B applies\n\nUnclassified markets will be BLOCKED.\n`;\n\n/**\n * Get rules summary for AI agents\n */\nexport function getParimutuelRulesSummary(): string {\n  return PARIMUTUEL_RULES_DOCUMENTATION;\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,52 @@
1
+ {
2
+ "name": "@baozi.bet/mcp-server",
3
+ "version": "4.0.0",
4
+ "description": "MCP server for Baozi prediction markets on Solana - full protocol coverage with market creation and AI agent network",
5
+ "type": "module",
6
+ "main": "dist/index.js",
7
+ "types": "dist/index.d.ts",
8
+ "bin": {
9
+ "baozi-mcp": "./dist/index.js"
10
+ },
11
+ "scripts": {
12
+ "build": "tsc",
13
+ "start": "node dist/index.js",
14
+ "dev": "tsx src/index.ts",
15
+ "prepublishOnly": "npm run build",
16
+ "test": "tsx src/__tests__/index.test.ts"
17
+ },
18
+ "dependencies": {
19
+ "@modelcontextprotocol/sdk": "^1.0.0",
20
+ "@solana/web3.js": "^1.98.0",
21
+ "bs58": "^6.0.0"
22
+ },
23
+ "devDependencies": {
24
+ "@types/bs58": "^4.0.4",
25
+ "@types/node": "^20.0.0",
26
+ "tsx": "^4.0.0",
27
+ "typescript": "^5.0.0"
28
+ },
29
+ "keywords": [
30
+ "mcp",
31
+ "baozi",
32
+ "solana",
33
+ "prediction-markets",
34
+ "claude",
35
+ "ai-agent",
36
+ "transaction-builder",
37
+ "mainnet"
38
+ ],
39
+ "author": "Baozi",
40
+ "license": "MIT",
41
+ "repository": {
42
+ "type": "git",
43
+ "url": "https://github.com/bolivian-peru/baozi-mcp"
44
+ },
45
+ "engines": {
46
+ "node": ">=18.0.0"
47
+ },
48
+ "files": [
49
+ "dist",
50
+ "README.md"
51
+ ]
52
+ }