@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
package/dist/index.js
ADDED
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* Baozi MCP Server V2.0.0
|
|
4
|
+
*
|
|
5
|
+
* Model Context Protocol server for Baozi prediction markets on Solana.
|
|
6
|
+
* Now with mainnet support and transaction building capabilities.
|
|
7
|
+
*
|
|
8
|
+
* Features:
|
|
9
|
+
* - Read markets, positions, quotes
|
|
10
|
+
* - Validate market timing (v6.2 rules)
|
|
11
|
+
* - Build unsigned bet transactions
|
|
12
|
+
* - Simulate transactions before signing
|
|
13
|
+
*
|
|
14
|
+
* Usage:
|
|
15
|
+
* npx @baozi/mcp-server
|
|
16
|
+
*
|
|
17
|
+
* Or add to Claude Desktop config:
|
|
18
|
+
* {
|
|
19
|
+
* "mcpServers": {
|
|
20
|
+
* "baozi": {
|
|
21
|
+
* "command": "npx",
|
|
22
|
+
* "args": ["@baozi/mcp-server"]
|
|
23
|
+
* }
|
|
24
|
+
* }
|
|
25
|
+
* }
|
|
26
|
+
*
|
|
27
|
+
* Environment Variables:
|
|
28
|
+
* HELIUS_RPC_URL - Helius RPC endpoint (recommended)
|
|
29
|
+
* SOLANA_RPC_URL - Alternative RPC endpoint
|
|
30
|
+
* SOLANA_NETWORK - Network: 'mainnet-beta' (default) or 'devnet'
|
|
31
|
+
* BAOZI_PROGRAM_ID - Override program ID (default: V4.7.6 mainnet)
|
|
32
|
+
*/
|
|
33
|
+
import { Server } from '@modelcontextprotocol/sdk/server/index.js';
|
|
34
|
+
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
|
|
35
|
+
import { CallToolRequestSchema, ListToolsRequestSchema, ListResourcesRequestSchema, ReadResourceRequestSchema, } from '@modelcontextprotocol/sdk/types.js';
|
|
36
|
+
import { TOOLS, handleTool } from './tools.js';
|
|
37
|
+
import { RESOURCES, handleResource } from './resources.js';
|
|
38
|
+
import { PROGRAM_ID, NETWORK, IS_MAINNET, RPC_ENDPOINT } from './config.js';
|
|
39
|
+
const VERSION = '2.0.0';
|
|
40
|
+
// Create MCP server
|
|
41
|
+
const server = new Server({
|
|
42
|
+
name: 'baozi-mcp',
|
|
43
|
+
version: VERSION,
|
|
44
|
+
}, {
|
|
45
|
+
capabilities: {
|
|
46
|
+
tools: {},
|
|
47
|
+
resources: {},
|
|
48
|
+
},
|
|
49
|
+
});
|
|
50
|
+
// List available tools
|
|
51
|
+
server.setRequestHandler(ListToolsRequestSchema, async () => {
|
|
52
|
+
return { tools: TOOLS };
|
|
53
|
+
});
|
|
54
|
+
// Handle tool calls
|
|
55
|
+
server.setRequestHandler(CallToolRequestSchema, async (request) => {
|
|
56
|
+
const { name, arguments: args } = request.params;
|
|
57
|
+
return handleTool(name, args || {});
|
|
58
|
+
});
|
|
59
|
+
// List available resources
|
|
60
|
+
server.setRequestHandler(ListResourcesRequestSchema, async () => {
|
|
61
|
+
return { resources: RESOURCES };
|
|
62
|
+
});
|
|
63
|
+
// Read resource content
|
|
64
|
+
server.setRequestHandler(ReadResourceRequestSchema, async (request) => {
|
|
65
|
+
const { uri } = request.params;
|
|
66
|
+
return handleResource(uri);
|
|
67
|
+
});
|
|
68
|
+
// Start server
|
|
69
|
+
async function main() {
|
|
70
|
+
const transport = new StdioServerTransport();
|
|
71
|
+
await server.connect(transport);
|
|
72
|
+
// Log startup info to stderr (stdout is for MCP protocol)
|
|
73
|
+
console.error('');
|
|
74
|
+
console.error('='.repeat(60));
|
|
75
|
+
console.error(`Baozi MCP Server v${VERSION}`);
|
|
76
|
+
console.error('='.repeat(60));
|
|
77
|
+
console.error('');
|
|
78
|
+
console.error(`Network: ${IS_MAINNET ? 'MAINNET' : 'Devnet'} (${NETWORK})`);
|
|
79
|
+
console.error(`Program ID: ${PROGRAM_ID.toBase58()}`);
|
|
80
|
+
console.error(`RPC: ${RPC_ENDPOINT.substring(0, 50)}...`);
|
|
81
|
+
console.error('');
|
|
82
|
+
console.error('Available Tools:');
|
|
83
|
+
console.error('-'.repeat(40));
|
|
84
|
+
TOOLS.forEach(tool => {
|
|
85
|
+
console.error(` ${tool.name}`);
|
|
86
|
+
console.error(` ${tool.description.substring(0, 70)}...`);
|
|
87
|
+
});
|
|
88
|
+
console.error('');
|
|
89
|
+
console.error('Available Resources:');
|
|
90
|
+
console.error('-'.repeat(40));
|
|
91
|
+
RESOURCES.forEach(resource => {
|
|
92
|
+
console.error(` ${resource.uri}`);
|
|
93
|
+
console.error(` ${resource.description}`);
|
|
94
|
+
});
|
|
95
|
+
console.error('');
|
|
96
|
+
console.error('Dynamic Resources:');
|
|
97
|
+
console.error('-'.repeat(40));
|
|
98
|
+
console.error(' baozi://portfolio/{wallet}');
|
|
99
|
+
console.error(' User portfolio with positions and statistics');
|
|
100
|
+
console.error('');
|
|
101
|
+
console.error('='.repeat(60));
|
|
102
|
+
console.error('Ready for connections...');
|
|
103
|
+
console.error('');
|
|
104
|
+
}
|
|
105
|
+
main().catch((error) => {
|
|
106
|
+
console.error('Fatal error:', error);
|
|
107
|
+
process.exit(1);
|
|
108
|
+
});
|
|
109
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8BG;AACH,OAAO,EAAE,MAAM,EAAE,MAAM,2CAA2C,CAAC;AACnE,OAAO,EAAE,oBAAoB,EAAE,MAAM,2CAA2C,CAAC;AACjF,OAAO,EACL,qBAAqB,EACrB,sBAAsB,EACtB,0BAA0B,EAC1B,yBAAyB,GAC1B,MAAM,oCAAoC,CAAC;AAE5C,OAAO,EAAE,KAAK,EAAE,UAAU,EAAE,MAAM,YAAY,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAE5E,MAAM,OAAO,GAAG,OAAO,CAAC;AAExB,oBAAoB;AACpB,MAAM,MAAM,GAAG,IAAI,MAAM,CACvB;IACE,IAAI,EAAE,WAAW;IACjB,OAAO,EAAE,OAAO;CACjB,EACD;IACE,YAAY,EAAE;QACZ,KAAK,EAAE,EAAE;QACT,SAAS,EAAE,EAAE;KACd;CACF,CACF,CAAC;AAEF,uBAAuB;AACvB,MAAM,CAAC,iBAAiB,CAAC,sBAAsB,EAAE,KAAK,IAAI,EAAE;IAC1D,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;AAC1B,CAAC,CAAC,CAAC;AAEH,oBAAoB;AACpB,MAAM,CAAC,iBAAiB,CAAC,qBAAqB,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE;IAChE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;IACjD,OAAO,UAAU,CAAC,IAAI,EAAE,IAAI,IAAI,EAAE,CAAC,CAAC;AACtC,CAAC,CAAC,CAAC;AAEH,2BAA2B;AAC3B,MAAM,CAAC,iBAAiB,CAAC,0BAA0B,EAAE,KAAK,IAAI,EAAE;IAC9D,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC;AAClC,CAAC,CAAC,CAAC;AAEH,wBAAwB;AACxB,MAAM,CAAC,iBAAiB,CAAC,yBAAyB,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE;IACpE,MAAM,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;IAC/B,OAAO,cAAc,CAAC,GAAG,CAAC,CAAC;AAC7B,CAAC,CAAC,CAAC;AAEH,eAAe;AACf,KAAK,UAAU,IAAI;IACjB,MAAM,SAAS,GAAG,IAAI,oBAAoB,EAAE,CAAC;IAC7C,MAAM,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;IAEhC,0DAA0D;IAC1D,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IAClB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9B,OAAO,CAAC,KAAK,CAAC,qBAAqB,OAAO,EAAE,CAAC,CAAC;IAC9C,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9B,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IAClB,OAAO,CAAC,KAAK,CAAC,YAAY,UAAU,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ,KAAK,OAAO,GAAG,CAAC,CAAC;IAC5E,OAAO,CAAC,KAAK,CAAC,eAAe,UAAU,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;IACtD,OAAO,CAAC,KAAK,CAAC,QAAQ,YAAY,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC;IAC1D,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IAClB,OAAO,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAC;IAClC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9B,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;QACnB,OAAO,CAAC,KAAK,CAAC,KAAK,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QAChC,OAAO,CAAC,KAAK,CAAC,OAAO,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC;IAC/D,CAAC,CAAC,CAAC;IACH,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IAClB,OAAO,CAAC,KAAK,CAAC,sBAAsB,CAAC,CAAC;IACtC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9B,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE;QAC3B,OAAO,CAAC,KAAK,CAAC,KAAK,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC;QACnC,OAAO,CAAC,KAAK,CAAC,OAAO,QAAQ,CAAC,WAAW,EAAE,CAAC,CAAC;IAC/C,CAAC,CAAC,CAAC;IACH,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IAClB,OAAO,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;IACpC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9B,OAAO,CAAC,KAAK,CAAC,8BAA8B,CAAC,CAAC;IAC9C,OAAO,CAAC,KAAK,CAAC,kDAAkD,CAAC,CAAC;IAClE,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IAClB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9B,OAAO,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAC;IAC1C,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;AACpB,CAAC;AAED,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;IACrB,OAAO,CAAC,KAAK,CAAC,cAAc,EAAE,KAAK,CAAC,CAAC;IACrC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC","sourcesContent":["#!/usr/bin/env node\n/**\n * Baozi MCP Server V2.0.0\n *\n * Model Context Protocol server for Baozi prediction markets on Solana.\n * Now with mainnet support and transaction building capabilities.\n *\n * Features:\n * - Read markets, positions, quotes\n * - Validate market timing (v6.2 rules)\n * - Build unsigned bet transactions\n * - Simulate transactions before signing\n *\n * Usage:\n *   npx @baozi/mcp-server\n *\n * Or add to Claude Desktop config:\n *   {\n *     \"mcpServers\": {\n *       \"baozi\": {\n *         \"command\": \"npx\",\n *         \"args\": [\"@baozi/mcp-server\"]\n *       }\n *     }\n *   }\n *\n * Environment Variables:\n *   HELIUS_RPC_URL    - Helius RPC endpoint (recommended)\n *   SOLANA_RPC_URL    - Alternative RPC endpoint\n *   SOLANA_NETWORK    - Network: 'mainnet-beta' (default) or 'devnet'\n *   BAOZI_PROGRAM_ID  - Override program ID (default: V4.7.6 mainnet)\n */\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\nimport {\n  CallToolRequestSchema,\n  ListToolsRequestSchema,\n  ListResourcesRequestSchema,\n  ReadResourceRequestSchema,\n} from '@modelcontextprotocol/sdk/types.js';\n\nimport { TOOLS, handleTool } from './tools.js';\nimport { RESOURCES, handleResource } from './resources.js';\nimport { PROGRAM_ID, NETWORK, IS_MAINNET, RPC_ENDPOINT } from './config.js';\n\nconst VERSION = '2.0.0';\n\n// Create MCP server\nconst server = new Server(\n  {\n    name: 'baozi-mcp',\n    version: VERSION,\n  },\n  {\n    capabilities: {\n      tools: {},\n      resources: {},\n    },\n  }\n);\n\n// List available tools\nserver.setRequestHandler(ListToolsRequestSchema, async () => {\n  return { tools: TOOLS };\n});\n\n// Handle tool calls\nserver.setRequestHandler(CallToolRequestSchema, async (request) => {\n  const { name, arguments: args } = request.params;\n  return handleTool(name, args || {});\n});\n\n// List available resources\nserver.setRequestHandler(ListResourcesRequestSchema, async () => {\n  return { resources: RESOURCES };\n});\n\n// Read resource content\nserver.setRequestHandler(ReadResourceRequestSchema, async (request) => {\n  const { uri } = request.params;\n  return handleResource(uri);\n});\n\n// Start server\nasync function main() {\n  const transport = new StdioServerTransport();\n  await server.connect(transport);\n\n  // Log startup info to stderr (stdout is for MCP protocol)\n  console.error('');\n  console.error('='.repeat(60));\n  console.error(`Baozi MCP Server v${VERSION}`);\n  console.error('='.repeat(60));\n  console.error('');\n  console.error(`Network: ${IS_MAINNET ? 'MAINNET' : 'Devnet'} (${NETWORK})`);\n  console.error(`Program ID: ${PROGRAM_ID.toBase58()}`);\n  console.error(`RPC: ${RPC_ENDPOINT.substring(0, 50)}...`);\n  console.error('');\n  console.error('Available Tools:');\n  console.error('-'.repeat(40));\n  TOOLS.forEach(tool => {\n    console.error(`  ${tool.name}`);\n    console.error(`    ${tool.description.substring(0, 70)}...`);\n  });\n  console.error('');\n  console.error('Available Resources:');\n  console.error('-'.repeat(40));\n  RESOURCES.forEach(resource => {\n    console.error(`  ${resource.uri}`);\n    console.error(`    ${resource.description}`);\n  });\n  console.error('');\n  console.error('Dynamic Resources:');\n  console.error('-'.repeat(40));\n  console.error('  baozi://portfolio/{wallet}');\n  console.error('    User portfolio with positions and statistics');\n  console.error('');\n  console.error('='.repeat(60));\n  console.error('Ready for connections...');\n  console.error('');\n}\n\nmain().catch((error) => {\n  console.error('Fatal error:', error);\n  process.exit(1);\n});\n"]}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
export declare const RESOURCES: {
|
|
2
|
+
uri: string;
|
|
3
|
+
name: string;
|
|
4
|
+
description: string;
|
|
5
|
+
mimeType: string;
|
|
6
|
+
}[];
|
|
7
|
+
export declare function handleResource(uri: string): Promise<{
|
|
8
|
+
contents: Array<{
|
|
9
|
+
uri: string;
|
|
10
|
+
mimeType: string;
|
|
11
|
+
text: string;
|
|
12
|
+
}>;
|
|
13
|
+
}>;
|
|
@@ -0,0 +1,336 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MCP Resource Definitions for Baozi Markets
|
|
3
|
+
* V2.0.0 - Mainnet + Extended Resources
|
|
4
|
+
*
|
|
5
|
+
* Resources provide read-only data that can be fetched by URI.
|
|
6
|
+
*/
|
|
7
|
+
import { listMarkets } from './handlers/markets.js';
|
|
8
|
+
import { getPositionsSummary } from './handlers/positions.js';
|
|
9
|
+
import { PROGRAM_ID, FEES, BET_LIMITS, TIMING, MARKET_STATUS_NAMES, MARKET_LAYER_NAMES, IS_MAINNET, NETWORK, } from './config.js';
|
|
10
|
+
// =============================================================================
|
|
11
|
+
// V6.2 MARKET RULES DOCUMENT
|
|
12
|
+
// =============================================================================
|
|
13
|
+
const MARKET_RULES_V6_2 = `
|
|
14
|
+
# Baozi Market Creation Rules (v6.2)
|
|
15
|
+
|
|
16
|
+
## Overview
|
|
17
|
+
Markets must follow timing rules to ensure fair betting and accurate resolution.
|
|
18
|
+
|
|
19
|
+
## Rule A: Event-Based Markets
|
|
20
|
+
For markets based on a single point-in-time event (e.g., "Will X win the election?"):
|
|
21
|
+
|
|
22
|
+
1. **Betting Close Buffer**: Betting must close AT LEAST 12 hours before the event
|
|
23
|
+
2. **Recommended Buffer**: 18-24 hours provides safety margin for timezone issues
|
|
24
|
+
3. **Resolution**: After event occurs, market is resolved based on outcome
|
|
25
|
+
|
|
26
|
+
### Example (Rule A)
|
|
27
|
+
- Event: "Super Bowl kickoff at 2025-02-09 18:30:00 UTC"
|
|
28
|
+
- Question: "Will the Chiefs win Super Bowl LIX?"
|
|
29
|
+
- Recommended betting close: 2025-02-08 18:30:00 UTC (24h before)
|
|
30
|
+
- Latest acceptable close: 2025-02-09 06:30:00 UTC (12h before)
|
|
31
|
+
|
|
32
|
+
## Rule B: Measurement-Period Markets
|
|
33
|
+
For markets based on outcomes over a time period (e.g., "Will BTC reach $100k in January?"):
|
|
34
|
+
|
|
35
|
+
1. **CRITICAL**: Betting must close BEFORE measurement period starts
|
|
36
|
+
2. **No Overlap**: Zero tolerance for betting during measurement period
|
|
37
|
+
3. **Information Advantage**: Allowing bets during measurement enables unfair advantage
|
|
38
|
+
4. **Period Length**: Prefer 2-7 days for optimal user experience
|
|
39
|
+
|
|
40
|
+
### Example (Rule B)
|
|
41
|
+
- Question: "Will ETH be above $4000 on Feb 1st 2025?"
|
|
42
|
+
- Measurement period: Feb 1st 00:00 - 23:59 UTC
|
|
43
|
+
- Betting must close: Before Feb 1st 00:00 UTC
|
|
44
|
+
- Recommended close: Jan 31st 22:00 UTC (2h buffer)
|
|
45
|
+
|
|
46
|
+
## Common Timing Mistakes
|
|
47
|
+
|
|
48
|
+
### INVALID Configurations:
|
|
49
|
+
- Betting closes AFTER event starts (Rule A violation)
|
|
50
|
+
- Betting overlaps with measurement period (Rule B violation)
|
|
51
|
+
- Buffer < 12 hours for event markets (too risky)
|
|
52
|
+
|
|
53
|
+
### WARNING Configurations:
|
|
54
|
+
- Buffer 12-18 hours (acceptable but tight)
|
|
55
|
+
- Measurement period > 7 days (poor UX)
|
|
56
|
+
- Very short buffer < 2 hours for measurement (risk of late bets)
|
|
57
|
+
|
|
58
|
+
## Validation Endpoint
|
|
59
|
+
Use the \`validate_market_params\` tool to check your market parameters before creation.
|
|
60
|
+
`;
|
|
61
|
+
// =============================================================================
|
|
62
|
+
// MARKET TEMPLATES
|
|
63
|
+
// =============================================================================
|
|
64
|
+
const EVENT_MARKET_TEMPLATE = {
|
|
65
|
+
type: 'event',
|
|
66
|
+
description: 'Template for event-based prediction markets (single point in time)',
|
|
67
|
+
example: {
|
|
68
|
+
question: 'Will [Team A] win the [Event Name]?',
|
|
69
|
+
closing_time: 'YYYY-MM-DDTHH:MM:SSZ (24 hours before event)',
|
|
70
|
+
event_time: 'YYYY-MM-DDTHH:MM:SSZ (when event occurs)',
|
|
71
|
+
market_type: 'event',
|
|
72
|
+
layer: 'Lab',
|
|
73
|
+
},
|
|
74
|
+
rules: [
|
|
75
|
+
'Question must be answerable with YES or NO',
|
|
76
|
+
'Event time must be after closing time',
|
|
77
|
+
'Minimum 12 hour buffer between close and event',
|
|
78
|
+
'Question max 200 characters',
|
|
79
|
+
],
|
|
80
|
+
examples: [
|
|
81
|
+
'Will the Chiefs win Super Bowl LIX?',
|
|
82
|
+
'Will Bitcoin reach $100,000 before March 2025?',
|
|
83
|
+
'Will SpaceX successfully launch Starship on [date]?',
|
|
84
|
+
],
|
|
85
|
+
};
|
|
86
|
+
const MEASUREMENT_MARKET_TEMPLATE = {
|
|
87
|
+
type: 'measurement',
|
|
88
|
+
description: 'Template for measurement-period prediction markets (outcome over time range)',
|
|
89
|
+
example: {
|
|
90
|
+
question: 'Will [Metric] be above [Value] on [Date]?',
|
|
91
|
+
closing_time: 'YYYY-MM-DDTHH:MM:SSZ (before measurement starts)',
|
|
92
|
+
measurement_start: 'YYYY-MM-DDTHH:MM:SSZ (when measurement period begins)',
|
|
93
|
+
measurement_end: 'YYYY-MM-DDTHH:MM:SSZ (when measurement period ends)',
|
|
94
|
+
market_type: 'measurement',
|
|
95
|
+
layer: 'Lab',
|
|
96
|
+
},
|
|
97
|
+
rules: [
|
|
98
|
+
'Betting MUST close before measurement period starts',
|
|
99
|
+
'Measurement period should be well-defined',
|
|
100
|
+
'Prefer 2-7 day periods for optimal UX',
|
|
101
|
+
'Data source for resolution must be clear',
|
|
102
|
+
],
|
|
103
|
+
examples: [
|
|
104
|
+
'Will ETH be above $4000 on Feb 1st 2025?',
|
|
105
|
+
'Will US inflation be below 3% in January 2025?',
|
|
106
|
+
'Will AAPL close above $200 on earnings day?',
|
|
107
|
+
],
|
|
108
|
+
};
|
|
109
|
+
// =============================================================================
|
|
110
|
+
// RESOURCE DEFINITIONS
|
|
111
|
+
// =============================================================================
|
|
112
|
+
export const RESOURCES = [
|
|
113
|
+
{
|
|
114
|
+
uri: 'baozi://markets/open',
|
|
115
|
+
name: 'Open Markets',
|
|
116
|
+
description: 'List of currently open prediction markets accepting bets on Solana mainnet',
|
|
117
|
+
mimeType: 'application/json',
|
|
118
|
+
},
|
|
119
|
+
{
|
|
120
|
+
uri: 'baozi://markets/all',
|
|
121
|
+
name: 'All Markets',
|
|
122
|
+
description: 'List of all prediction markets (open, closed, resolved) on Solana mainnet',
|
|
123
|
+
mimeType: 'application/json',
|
|
124
|
+
},
|
|
125
|
+
{
|
|
126
|
+
uri: 'baozi://config',
|
|
127
|
+
name: 'Program Config',
|
|
128
|
+
description: 'Baozi V4.7.6 program configuration, fees, and limits',
|
|
129
|
+
mimeType: 'application/json',
|
|
130
|
+
},
|
|
131
|
+
{
|
|
132
|
+
uri: 'baozi://rules',
|
|
133
|
+
name: 'Market Rules v6.2',
|
|
134
|
+
description: 'Documentation of market timing rules and validation requirements',
|
|
135
|
+
mimeType: 'text/markdown',
|
|
136
|
+
},
|
|
137
|
+
{
|
|
138
|
+
uri: 'baozi://templates/event',
|
|
139
|
+
name: 'Event Market Template',
|
|
140
|
+
description: 'Template for creating event-based prediction markets',
|
|
141
|
+
mimeType: 'application/json',
|
|
142
|
+
},
|
|
143
|
+
{
|
|
144
|
+
uri: 'baozi://templates/measurement',
|
|
145
|
+
name: 'Measurement Market Template',
|
|
146
|
+
description: 'Template for creating measurement-period prediction markets',
|
|
147
|
+
mimeType: 'application/json',
|
|
148
|
+
},
|
|
149
|
+
];
|
|
150
|
+
// =============================================================================
|
|
151
|
+
// RESOURCE HANDLERS
|
|
152
|
+
// =============================================================================
|
|
153
|
+
export async function handleResource(uri) {
|
|
154
|
+
try {
|
|
155
|
+
// Handle portfolio requests with wallet parameter
|
|
156
|
+
if (uri.startsWith('baozi://portfolio/')) {
|
|
157
|
+
const wallet = uri.replace('baozi://portfolio/', '');
|
|
158
|
+
const summary = await getPositionsSummary(wallet);
|
|
159
|
+
return {
|
|
160
|
+
contents: [
|
|
161
|
+
{
|
|
162
|
+
uri,
|
|
163
|
+
mimeType: 'application/json',
|
|
164
|
+
text: JSON.stringify({
|
|
165
|
+
type: 'portfolio',
|
|
166
|
+
network: NETWORK,
|
|
167
|
+
...summary,
|
|
168
|
+
fetchedAt: new Date().toISOString(),
|
|
169
|
+
}, null, 2),
|
|
170
|
+
},
|
|
171
|
+
],
|
|
172
|
+
};
|
|
173
|
+
}
|
|
174
|
+
switch (uri) {
|
|
175
|
+
case 'baozi://markets/open': {
|
|
176
|
+
const markets = await listMarkets('Active');
|
|
177
|
+
return {
|
|
178
|
+
contents: [
|
|
179
|
+
{
|
|
180
|
+
uri,
|
|
181
|
+
mimeType: 'application/json',
|
|
182
|
+
text: JSON.stringify({
|
|
183
|
+
type: 'open_markets',
|
|
184
|
+
network: NETWORK,
|
|
185
|
+
programId: PROGRAM_ID.toBase58(),
|
|
186
|
+
count: markets.length,
|
|
187
|
+
markets: markets.map(m => ({
|
|
188
|
+
publicKey: m.publicKey,
|
|
189
|
+
marketId: m.marketId,
|
|
190
|
+
question: m.question,
|
|
191
|
+
layer: m.layer,
|
|
192
|
+
yesPercent: m.yesPercent,
|
|
193
|
+
noPercent: m.noPercent,
|
|
194
|
+
totalPoolSol: m.totalPoolSol,
|
|
195
|
+
closingTime: m.closingTime,
|
|
196
|
+
isBettingOpen: m.isBettingOpen,
|
|
197
|
+
})),
|
|
198
|
+
fetchedAt: new Date().toISOString(),
|
|
199
|
+
}, null, 2),
|
|
200
|
+
},
|
|
201
|
+
],
|
|
202
|
+
};
|
|
203
|
+
}
|
|
204
|
+
case 'baozi://markets/all': {
|
|
205
|
+
const markets = await listMarkets();
|
|
206
|
+
const byStatus = {};
|
|
207
|
+
const byLayer = {};
|
|
208
|
+
for (const m of markets) {
|
|
209
|
+
byStatus[m.status] = (byStatus[m.status] || 0) + 1;
|
|
210
|
+
byLayer[m.layer] = (byLayer[m.layer] || 0) + 1;
|
|
211
|
+
}
|
|
212
|
+
return {
|
|
213
|
+
contents: [
|
|
214
|
+
{
|
|
215
|
+
uri,
|
|
216
|
+
mimeType: 'application/json',
|
|
217
|
+
text: JSON.stringify({
|
|
218
|
+
type: 'all_markets',
|
|
219
|
+
network: NETWORK,
|
|
220
|
+
programId: PROGRAM_ID.toBase58(),
|
|
221
|
+
count: markets.length,
|
|
222
|
+
byStatus,
|
|
223
|
+
byLayer,
|
|
224
|
+
markets: markets.map(m => ({
|
|
225
|
+
publicKey: m.publicKey,
|
|
226
|
+
marketId: m.marketId,
|
|
227
|
+
question: m.question,
|
|
228
|
+
status: m.status,
|
|
229
|
+
layer: m.layer,
|
|
230
|
+
winningOutcome: m.winningOutcome,
|
|
231
|
+
totalPoolSol: m.totalPoolSol,
|
|
232
|
+
})),
|
|
233
|
+
fetchedAt: new Date().toISOString(),
|
|
234
|
+
}, null, 2),
|
|
235
|
+
},
|
|
236
|
+
],
|
|
237
|
+
};
|
|
238
|
+
}
|
|
239
|
+
case 'baozi://config': {
|
|
240
|
+
return {
|
|
241
|
+
contents: [
|
|
242
|
+
{
|
|
243
|
+
uri,
|
|
244
|
+
mimeType: 'application/json',
|
|
245
|
+
text: JSON.stringify({
|
|
246
|
+
type: 'program_config',
|
|
247
|
+
program: {
|
|
248
|
+
id: PROGRAM_ID.toBase58(),
|
|
249
|
+
network: NETWORK,
|
|
250
|
+
version: '4.7.6',
|
|
251
|
+
isMainnet: IS_MAINNET,
|
|
252
|
+
},
|
|
253
|
+
fees: {
|
|
254
|
+
official: {
|
|
255
|
+
platformFeeBps: FEES.OFFICIAL_PLATFORM_FEE_BPS,
|
|
256
|
+
platformFeePercent: `${FEES.OFFICIAL_PLATFORM_FEE_BPS / 100}%`,
|
|
257
|
+
creationFeeSol: FEES.OFFICIAL_CREATION_FEE / 1e9,
|
|
258
|
+
},
|
|
259
|
+
lab: {
|
|
260
|
+
platformFeeBps: FEES.LAB_PLATFORM_FEE_BPS,
|
|
261
|
+
platformFeePercent: `${FEES.LAB_PLATFORM_FEE_BPS / 100}%`,
|
|
262
|
+
creationFeeSol: FEES.LAB_CREATION_FEE / 1e9,
|
|
263
|
+
},
|
|
264
|
+
private: {
|
|
265
|
+
platformFeeBps: FEES.PRIVATE_PLATFORM_FEE_BPS,
|
|
266
|
+
platformFeePercent: `${FEES.PRIVATE_PLATFORM_FEE_BPS / 100}%`,
|
|
267
|
+
creationFeeSol: FEES.PRIVATE_CREATION_FEE / 1e9,
|
|
268
|
+
},
|
|
269
|
+
affiliateFeeBps: FEES.AFFILIATE_FEE_BPS,
|
|
270
|
+
creatorFeeBps: FEES.CREATOR_FEE_BPS,
|
|
271
|
+
},
|
|
272
|
+
limits: {
|
|
273
|
+
minBetSol: BET_LIMITS.MIN_BET_SOL,
|
|
274
|
+
maxBetSol: BET_LIMITS.MAX_BET_SOL,
|
|
275
|
+
},
|
|
276
|
+
timing: {
|
|
277
|
+
bettingFreezeSeconds: TIMING.BETTING_FREEZE_SECONDS,
|
|
278
|
+
minEventBufferHours: TIMING.MIN_EVENT_BUFFER_HOURS,
|
|
279
|
+
recommendedEventBufferHours: TIMING.RECOMMENDED_EVENT_BUFFER_HOURS,
|
|
280
|
+
disputeWindowSeconds: TIMING.DISPUTE_WINDOW_SECONDS,
|
|
281
|
+
},
|
|
282
|
+
marketStatuses: MARKET_STATUS_NAMES,
|
|
283
|
+
marketLayers: MARKET_LAYER_NAMES,
|
|
284
|
+
links: {
|
|
285
|
+
website: 'https://baozi.bet',
|
|
286
|
+
api: 'https://baozi.ooo/api/v4',
|
|
287
|
+
explorer: `https://solscan.io/account/${PROGRAM_ID.toBase58()}${IS_MAINNET ? '' : '?cluster=devnet'}`,
|
|
288
|
+
},
|
|
289
|
+
fetchedAt: new Date().toISOString(),
|
|
290
|
+
}, null, 2),
|
|
291
|
+
},
|
|
292
|
+
],
|
|
293
|
+
};
|
|
294
|
+
}
|
|
295
|
+
case 'baozi://rules': {
|
|
296
|
+
return {
|
|
297
|
+
contents: [
|
|
298
|
+
{
|
|
299
|
+
uri,
|
|
300
|
+
mimeType: 'text/markdown',
|
|
301
|
+
text: MARKET_RULES_V6_2,
|
|
302
|
+
},
|
|
303
|
+
],
|
|
304
|
+
};
|
|
305
|
+
}
|
|
306
|
+
case 'baozi://templates/event': {
|
|
307
|
+
return {
|
|
308
|
+
contents: [
|
|
309
|
+
{
|
|
310
|
+
uri,
|
|
311
|
+
mimeType: 'application/json',
|
|
312
|
+
text: JSON.stringify(EVENT_MARKET_TEMPLATE, null, 2),
|
|
313
|
+
},
|
|
314
|
+
],
|
|
315
|
+
};
|
|
316
|
+
}
|
|
317
|
+
case 'baozi://templates/measurement': {
|
|
318
|
+
return {
|
|
319
|
+
contents: [
|
|
320
|
+
{
|
|
321
|
+
uri,
|
|
322
|
+
mimeType: 'application/json',
|
|
323
|
+
text: JSON.stringify(MEASUREMENT_MARKET_TEMPLATE, null, 2),
|
|
324
|
+
},
|
|
325
|
+
],
|
|
326
|
+
};
|
|
327
|
+
}
|
|
328
|
+
default:
|
|
329
|
+
throw new Error(`Unknown resource: ${uri}`);
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
catch (error) {
|
|
333
|
+
throw error;
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resources.js","sourceRoot":"","sources":["../src/resources.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MCP Resource Definitions for Baozi Markets\n * V2.0.0 - Mainnet + Extended Resources\n *\n * Resources provide read-only data that can be fetched by URI.\n */\nimport { listMarkets } from './handlers/markets.js';\nimport { getPositionsSummary } from './handlers/positions.js';\nimport {\n  PROGRAM_ID,\n  FEES,\n  BET_LIMITS,\n  TIMING,\n  MARKET_STATUS_NAMES,\n  MARKET_LAYER_NAMES,\n  IS_MAINNET,\n  NETWORK,\n} from './config.js';\n\n// =============================================================================\n// V6.2 MARKET RULES DOCUMENT\n// =============================================================================\n\nconst MARKET_RULES_V6_2 = `\n# Baozi Market Creation Rules (v6.2)\n\n## Overview\nMarkets must follow timing rules to ensure fair betting and accurate resolution.\n\n## Rule A: Event-Based Markets\nFor markets based on a single point-in-time event (e.g., \"Will X win the election?\"):\n\n1. **Betting Close Buffer**: Betting must close AT LEAST 12 hours before the event\n2. **Recommended Buffer**: 18-24 hours provides safety margin for timezone issues\n3. **Resolution**: After event occurs, market is resolved based on outcome\n\n### Example (Rule A)\n- Event: \"Super Bowl kickoff at 2025-02-09 18:30:00 UTC\"\n- Question: \"Will the Chiefs win Super Bowl LIX?\"\n- Recommended betting close: 2025-02-08 18:30:00 UTC (24h before)\n- Latest acceptable close: 2025-02-09 06:30:00 UTC (12h before)\n\n## Rule B: Measurement-Period Markets\nFor markets based on outcomes over a time period (e.g., \"Will BTC reach $100k in January?\"):\n\n1. **CRITICAL**: Betting must close BEFORE measurement period starts\n2. **No Overlap**: Zero tolerance for betting during measurement period\n3. **Information Advantage**: Allowing bets during measurement enables unfair advantage\n4. **Period Length**: Prefer 2-7 days for optimal user experience\n\n### Example (Rule B)\n- Question: \"Will ETH be above $4000 on Feb 1st 2025?\"\n- Measurement period: Feb 1st 00:00 - 23:59 UTC\n- Betting must close: Before Feb 1st 00:00 UTC\n- Recommended close: Jan 31st 22:00 UTC (2h buffer)\n\n## Common Timing Mistakes\n\n### INVALID Configurations:\n- Betting closes AFTER event starts (Rule A violation)\n- Betting overlaps with measurement period (Rule B violation)\n- Buffer < 12 hours for event markets (too risky)\n\n### WARNING Configurations:\n- Buffer 12-18 hours (acceptable but tight)\n- Measurement period > 7 days (poor UX)\n- Very short buffer < 2 hours for measurement (risk of late bets)\n\n## Validation Endpoint\nUse the \\`validate_market_params\\` tool to check your market parameters before creation.\n`;\n\n// =============================================================================\n// MARKET TEMPLATES\n// =============================================================================\n\nconst EVENT_MARKET_TEMPLATE = {\n  type: 'event',\n  description: 'Template for event-based prediction markets (single point in time)',\n  example: {\n    question: 'Will [Team A] win the [Event Name]?',\n    closing_time: 'YYYY-MM-DDTHH:MM:SSZ (24 hours before event)',\n    event_time: 'YYYY-MM-DDTHH:MM:SSZ (when event occurs)',\n    market_type: 'event',\n    layer: 'Lab',\n  },\n  rules: [\n    'Question must be answerable with YES or NO',\n    'Event time must be after closing time',\n    'Minimum 12 hour buffer between close and event',\n    'Question max 200 characters',\n  ],\n  examples: [\n    'Will the Chiefs win Super Bowl LIX?',\n    'Will Bitcoin reach $100,000 before March 2025?',\n    'Will SpaceX successfully launch Starship on [date]?',\n  ],\n};\n\nconst MEASUREMENT_MARKET_TEMPLATE = {\n  type: 'measurement',\n  description: 'Template for measurement-period prediction markets (outcome over time range)',\n  example: {\n    question: 'Will [Metric] be above [Value] on [Date]?',\n    closing_time: 'YYYY-MM-DDTHH:MM:SSZ (before measurement starts)',\n    measurement_start: 'YYYY-MM-DDTHH:MM:SSZ (when measurement period begins)',\n    measurement_end: 'YYYY-MM-DDTHH:MM:SSZ (when measurement period ends)',\n    market_type: 'measurement',\n    layer: 'Lab',\n  },\n  rules: [\n    'Betting MUST close before measurement period starts',\n    'Measurement period should be well-defined',\n    'Prefer 2-7 day periods for optimal UX',\n    'Data source for resolution must be clear',\n  ],\n  examples: [\n    'Will ETH be above $4000 on Feb 1st 2025?',\n    'Will US inflation be below 3% in January 2025?',\n    'Will AAPL close above $200 on earnings day?',\n  ],\n};\n\n// =============================================================================\n// RESOURCE DEFINITIONS\n// =============================================================================\n\nexport const RESOURCES = [\n  {\n    uri: 'baozi://markets/open',\n    name: 'Open Markets',\n    description: 'List of currently open prediction markets accepting bets on Solana mainnet',\n    mimeType: 'application/json',\n  },\n  {\n    uri: 'baozi://markets/all',\n    name: 'All Markets',\n    description: 'List of all prediction markets (open, closed, resolved) on Solana mainnet',\n    mimeType: 'application/json',\n  },\n  {\n    uri: 'baozi://config',\n    name: 'Program Config',\n    description: 'Baozi V4.7.6 program configuration, fees, and limits',\n    mimeType: 'application/json',\n  },\n  {\n    uri: 'baozi://rules',\n    name: 'Market Rules v6.2',\n    description: 'Documentation of market timing rules and validation requirements',\n    mimeType: 'text/markdown',\n  },\n  {\n    uri: 'baozi://templates/event',\n    name: 'Event Market Template',\n    description: 'Template for creating event-based prediction markets',\n    mimeType: 'application/json',\n  },\n  {\n    uri: 'baozi://templates/measurement',\n    name: 'Measurement Market Template',\n    description: 'Template for creating measurement-period prediction markets',\n    mimeType: 'application/json',\n  },\n];\n\n// =============================================================================\n// RESOURCE HANDLERS\n// =============================================================================\n\nexport async function handleResource(uri: string): Promise<{\n  contents: Array<{ uri: string; mimeType: string; text: string }>;\n}> {\n  try {\n    // Handle portfolio requests with wallet parameter\n    if (uri.startsWith('baozi://portfolio/')) {\n      const wallet = uri.replace('baozi://portfolio/', '');\n      const summary = await getPositionsSummary(wallet);\n      return {\n        contents: [\n          {\n            uri,\n            mimeType: 'application/json',\n            text: JSON.stringify({\n              type: 'portfolio',\n              network: NETWORK,\n              ...summary,\n              fetchedAt: new Date().toISOString(),\n            }, null, 2),\n          },\n        ],\n      };\n    }\n\n    switch (uri) {\n      case 'baozi://markets/open': {\n        const markets = await listMarkets('Active');\n        return {\n          contents: [\n            {\n              uri,\n              mimeType: 'application/json',\n              text: JSON.stringify({\n                type: 'open_markets',\n                network: NETWORK,\n                programId: PROGRAM_ID.toBase58(),\n                count: markets.length,\n                markets: markets.map(m => ({\n                  publicKey: m.publicKey,\n                  marketId: m.marketId,\n                  question: m.question,\n                  layer: m.layer,\n                  yesPercent: m.yesPercent,\n                  noPercent: m.noPercent,\n                  totalPoolSol: m.totalPoolSol,\n                  closingTime: m.closingTime,\n                  isBettingOpen: m.isBettingOpen,\n                })),\n                fetchedAt: new Date().toISOString(),\n              }, null, 2),\n            },\n          ],\n        };\n      }\n\n      case 'baozi://markets/all': {\n        const markets = await listMarkets();\n        const byStatus: Record<string, number> = {};\n        const byLayer: Record<string, number> = {};\n\n        for (const m of markets) {\n          byStatus[m.status] = (byStatus[m.status] || 0) + 1;\n          byLayer[m.layer] = (byLayer[m.layer] || 0) + 1;\n        }\n\n        return {\n          contents: [\n            {\n              uri,\n              mimeType: 'application/json',\n              text: JSON.stringify({\n                type: 'all_markets',\n                network: NETWORK,\n                programId: PROGRAM_ID.toBase58(),\n                count: markets.length,\n                byStatus,\n                byLayer,\n                markets: markets.map(m => ({\n                  publicKey: m.publicKey,\n                  marketId: m.marketId,\n                  question: m.question,\n                  status: m.status,\n                  layer: m.layer,\n                  winningOutcome: m.winningOutcome,\n                  totalPoolSol: m.totalPoolSol,\n                })),\n                fetchedAt: new Date().toISOString(),\n              }, null, 2),\n            },\n          ],\n        };\n      }\n\n      case 'baozi://config': {\n        return {\n          contents: [\n            {\n              uri,\n              mimeType: 'application/json',\n              text: JSON.stringify({\n                type: 'program_config',\n                program: {\n                  id: PROGRAM_ID.toBase58(),\n                  network: NETWORK,\n                  version: '4.7.6',\n                  isMainnet: IS_MAINNET,\n                },\n                fees: {\n                  official: {\n                    platformFeeBps: FEES.OFFICIAL_PLATFORM_FEE_BPS,\n                    platformFeePercent: `${FEES.OFFICIAL_PLATFORM_FEE_BPS / 100}%`,\n                    creationFeeSol: FEES.OFFICIAL_CREATION_FEE / 1e9,\n                  },\n                  lab: {\n                    platformFeeBps: FEES.LAB_PLATFORM_FEE_BPS,\n                    platformFeePercent: `${FEES.LAB_PLATFORM_FEE_BPS / 100}%`,\n                    creationFeeSol: FEES.LAB_CREATION_FEE / 1e9,\n                  },\n                  private: {\n                    platformFeeBps: FEES.PRIVATE_PLATFORM_FEE_BPS,\n                    platformFeePercent: `${FEES.PRIVATE_PLATFORM_FEE_BPS / 100}%`,\n                    creationFeeSol: FEES.PRIVATE_CREATION_FEE / 1e9,\n                  },\n                  affiliateFeeBps: FEES.AFFILIATE_FEE_BPS,\n                  creatorFeeBps: FEES.CREATOR_FEE_BPS,\n                },\n                limits: {\n                  minBetSol: BET_LIMITS.MIN_BET_SOL,\n                  maxBetSol: BET_LIMITS.MAX_BET_SOL,\n                },\n                timing: {\n                  bettingFreezeSeconds: TIMING.BETTING_FREEZE_SECONDS,\n                  minEventBufferHours: TIMING.MIN_EVENT_BUFFER_HOURS,\n                  recommendedEventBufferHours: TIMING.RECOMMENDED_EVENT_BUFFER_HOURS,\n                  disputeWindowSeconds: TIMING.DISPUTE_WINDOW_SECONDS,\n                },\n                marketStatuses: MARKET_STATUS_NAMES,\n                marketLayers: MARKET_LAYER_NAMES,\n                links: {\n                  website: 'https://baozi.bet',\n                  api: 'https://baozi.ooo/api/v4',\n                  explorer: `https://solscan.io/account/${PROGRAM_ID.toBase58()}${IS_MAINNET ? '' : '?cluster=devnet'}`,\n                },\n                fetchedAt: new Date().toISOString(),\n              }, null, 2),\n            },\n          ],\n        };\n      }\n\n      case 'baozi://rules': {\n        return {\n          contents: [\n            {\n              uri,\n              mimeType: 'text/markdown',\n              text: MARKET_RULES_V6_2,\n            },\n          ],\n        };\n      }\n\n      case 'baozi://templates/event': {\n        return {\n          contents: [\n            {\n              uri,\n              mimeType: 'application/json',\n              text: JSON.stringify(EVENT_MARKET_TEMPLATE, null, 2),\n            },\n          ],\n        };\n      }\n\n      case 'baozi://templates/measurement': {\n        return {\n          contents: [\n            {\n              uri,\n              mimeType: 'application/json',\n              text: JSON.stringify(MEASUREMENT_MARKET_TEMPLATE, null, 2),\n            },\n          ],\n        };\n      }\n\n      default:\n        throw new Error(`Unknown resource: ${uri}`);\n    }\n  } catch (error) {\n    throw error;\n  }\n}\n"]}
|