@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,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,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
|
+
};
|