@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.
- package/README.md +294 -0
- package/dist/__tests__/full-test.d.ts +1 -0
- package/dist/__tests__/full-test.js +291 -0
- package/dist/builders/affiliate-transaction.d.ts +41 -0
- package/dist/builders/affiliate-transaction.js +123 -0
- package/dist/builders/bet-transaction.d.ts +70 -0
- package/dist/builders/bet-transaction.js +323 -0
- package/dist/builders/claim-transaction.d.ts +57 -0
- package/dist/builders/claim-transaction.js +196 -0
- package/dist/builders/creator-transaction.d.ts +49 -0
- package/dist/builders/creator-transaction.js +177 -0
- package/dist/builders/dispute-transaction.d.ts +81 -0
- package/dist/builders/dispute-transaction.js +215 -0
- package/dist/builders/index.d.ts +14 -0
- package/dist/builders/index.js +15 -0
- package/dist/builders/market-creation-tx.d.ts +65 -0
- package/dist/builders/market-creation-tx.js +362 -0
- package/dist/builders/market-management-transaction.d.ts +85 -0
- package/dist/builders/market-management-transaction.js +239 -0
- package/dist/builders/race-transaction.d.ts +67 -0
- package/dist/builders/race-transaction.js +242 -0
- package/dist/builders/resolution-transaction.d.ts +108 -0
- package/dist/builders/resolution-transaction.js +250 -0
- package/dist/builders/whitelist-transaction.d.ts +72 -0
- package/dist/builders/whitelist-transaction.js +179 -0
- package/dist/config.d.ts +138 -0
- package/dist/config.js +307 -0
- package/dist/handlers/agent-network.d.ts +81 -0
- package/dist/handlers/agent-network.js +332 -0
- package/dist/handlers/claims.d.ts +47 -0
- package/dist/handlers/claims.js +218 -0
- package/dist/handlers/market-creation.d.ts +154 -0
- package/dist/handlers/market-creation.js +290 -0
- package/dist/handlers/markets.d.ts +41 -0
- package/dist/handlers/markets.js +319 -0
- package/dist/handlers/positions.d.ts +40 -0
- package/dist/handlers/positions.js +244 -0
- package/dist/handlers/quote.d.ts +33 -0
- package/dist/handlers/quote.js +144 -0
- package/dist/handlers/race-markets.d.ts +54 -0
- package/dist/handlers/race-markets.js +308 -0
- package/dist/handlers/resolution.d.ts +43 -0
- package/dist/handlers/resolution.js +194 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +109 -0
- package/dist/resources.d.ts +13 -0
- package/dist/resources.js +336 -0
- package/dist/tools.d.ts +3109 -0
- package/dist/tools.js +1956 -0
- package/dist/validation/bet-rules.d.ts +82 -0
- package/dist/validation/bet-rules.js +276 -0
- package/dist/validation/creation-rules.d.ts +69 -0
- package/dist/validation/creation-rules.js +302 -0
- package/dist/validation/index.d.ts +6 -0
- package/dist/validation/index.js +7 -0
- package/dist/validation/market-rules.d.ts +60 -0
- package/dist/validation/market-rules.js +237 -0
- package/dist/validation/parimutuel-rules.d.ts +117 -0
- package/dist/validation/parimutuel-rules.js +270 -0
- 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
|
+
}
|