Trading Simulator MCP Server

by recallnet
Verified
import * as crypto from 'crypto'; /** * Trading Simulator API Client * * This client handles authentication, request signing, and provides methods for interacting * with the Trading Simulator API. It's designed primarily for teams participating in trading * competitions to execute trades, check balances, and view competition status. * * Required configuration: * - API key: Your team's unique API key provided during registration * - API secret: Your team's secret key for request signing (keep this secure!) * - Base URL: The endpoint of the Trading Simulator server * * @example * // Basic setup * const client = new TradingSimulatorClient( * "sk_7b550f528ba35cfb50b9de65b63e27e4", // Your API key * "a56229f71f5a2a42f93197fb32159916d1ff7796433c133d00b90097a0bbf12f", // Your API secret * "https://trading-simulator.example.com" // API base URL * ); * * // Get team balances * const balances = await client.getBalances(); * * // Execute a trade on Base chain (within-chain trade) * const tradeResult = await client.executeTrade({ * fromToken: "0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913", // USDC on Base * toToken: "0x0b3e328455c4059EEb9e3f84b5543F74E24e7E1b", // TOSHI on Base * amount: "50", * fromChain: BlockchainType.EVM, * toChain: BlockchainType.EVM, * fromSpecificChain: SpecificChain.BASE, * toSpecificChain: SpecificChain.BASE * }); */ // Define blockchain types export enum BlockchainType { SVM = 'svm', // Solana Virtual Machine EVM = 'evm' // Ethereum Virtual Machine } // Define specific EVM chains export enum SpecificChain { ETH = 'eth', POLYGON = 'polygon', BSC = 'bsc', ARBITRUM = 'arbitrum', BASE = 'base', OPTIMISM = 'optimism', AVALANCHE = 'avalanche', LINEA = 'linea', SVM = 'svm' } // Common token addresses export const COMMON_TOKENS = { // Solana tokens SVM: { USDC: 'EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v', SOL: 'So11111111111111111111111111111111111111112' }, // Ethereum tokens EVM: { USDC: '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48', ETH: '0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2', // WETH LINK: '0x514910771af9ca656af840dff83e8264ecf986ca', // Chainlink ARB: '0x912CE59144191C1204E64559FE8253a0e49E6548', // Arbitrum TOSHI: '0x532f27101965dd16442E59d40670FaF5eBB142E4' // Toshi token on Base } }; // Map tokens to their known chains for quick lookups export const TOKEN_CHAINS: Record<string, SpecificChain> = { // EVM tokens with their specific chains [COMMON_TOKENS.EVM.ETH]: SpecificChain.ETH, [COMMON_TOKENS.EVM.USDC]: SpecificChain.ETH, [COMMON_TOKENS.EVM.LINK]: SpecificChain.ETH, [COMMON_TOKENS.EVM.ARB]: SpecificChain.ARBITRUM, [COMMON_TOKENS.EVM.TOSHI]: SpecificChain.BASE, // SVM tokens [COMMON_TOKENS.SVM.SOL]: SpecificChain.SVM, [COMMON_TOKENS.SVM.USDC]: SpecificChain.SVM }; export class TradingSimulatorClient { private apiKey: string; private apiSecret: string; private baseUrl: string; private testMode: boolean; /** * Create a new instance of the Trading Simulator client * * @param apiKey The API key for your team * @param apiSecret The API secret for your team * @param baseUrl The base URL of the Trading Simulator API (default: http://localhost:3001) * @param testMode Should ONLY be set to true when used in automated tests with NODE_ENV=test. * When true, uses a timestamp 2 years in the future which is only accepted by * the server in test environments. For both production and development, * this must be false to use current timestamps, which are validated * within a 5-minute window. (default: false) */ constructor( apiKey: string, apiSecret: string, baseUrl: string = 'http://localhost:3001', testMode: boolean = false ) { this.apiKey = apiKey; this.apiSecret = apiSecret; this.baseUrl = baseUrl; this.testMode = testMode; } /** * Generate the required headers for API authentication * * @param method The HTTP method (GET, POST, etc.) * @param path The API endpoint path (e.g., /api/account/balances) * @param body The request body (if any) * @returns An object containing the required headers */ private generateHeaders(method: string, path: string, body: string = '{}'): Record<string, string> { // Use current timestamp for production or future timestamp for testing let timestamp: string; if (this.testMode) { // Use timestamp 2 years in the future for e2e tests (to avoid expiration) timestamp = new Date(Date.now() + 2 * 365 * 24 * 60 * 60 * 1000).toISOString(); } else { // Use current timestamp for production use timestamp = new Date().toISOString(); } // Important: Use '{}' for empty bodies to match the test utility implementation const bodyString = body || '{}'; const data = method + path + timestamp + bodyString; const signature = crypto .createHmac('sha256', this.apiSecret) .update(data) .digest('hex'); return { 'X-API-Key': this.apiKey, 'X-Timestamp': timestamp, 'X-Signature': signature, 'Content-Type': 'application/json' }; } /** * Make a request to the API * * @param method The HTTP method * @param path The API endpoint path * @param body The request body (if any) * @returns A promise that resolves to the API response */ private async request<T>(method: string, path: string, body: any = null): Promise<T> { const url = `${this.baseUrl}${path}`; // IMPORTANT: Always use '{}' for empty bodies instead of empty string ('') to match server expectations // This is crucial for signature validation to work correctly const bodyString = body ? JSON.stringify(body) : '{}'; const headers = this.generateHeaders(method, path, bodyString); const options: RequestInit = { method, headers, body: body ? bodyString : undefined // Only include body if it exists }; try { const response = await fetch(url, options); const data = await response.json(); if (!response.ok) { throw new Error(data.error?.message || 'API request failed'); } return data as T; } catch (error) { console.error('API request error:', error); throw error; } } /** * Detect blockchain type from token address format * * @param token The token address * @returns The detected blockchain type (SVM or EVM) */ public detectChain(token: string): BlockchainType { // Ethereum addresses start with '0x' followed by 40 hex characters if (/^0x[a-fA-F0-9]{40}$/.test(token)) { return BlockchainType.EVM; } // Solana addresses are base58 encoded, typically around 44 characters // This is a simplified detection, could be more robust return BlockchainType.SVM; } /** * Get your team's token balances across all supported chains * * @returns Balance information including tokens on all chains (EVM and SVM) * * @example * const balances = await client.getBalances(); * console.log('My ETH balance on Base:', balances.balance['0x4200000000000000000000000000000000000006']); * console.log('My USDC balance on Base:', balances.balance['0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913']); */ async getBalances(): Promise<any> { return this.request<any>('GET', '/api/account/balances'); } /** * Get the trade history for your team * * @param options Optional filtering parameters * @returns A promise that resolves to the trade history response */ async getTrades(options?: { limit?: number; offset?: number; token?: string; chain?: BlockchainType; }): Promise<any> { let query = ''; if (options) { const params = new URLSearchParams(); if (options.limit) params.append('limit', options.limit.toString()); if (options.offset) params.append('offset', options.offset.toString()); if (options.token) params.append('token', options.token); if (options.chain) params.append('chain', options.chain); query = `?${params.toString()}`; } return this.request<any>('GET', `/api/account/trades${query}`); } /** * Get the current price for a token * * @param token The token address to get the price for * @param chain Optional blockchain type (auto-detected if not provided) * @param specificChain Optional specific chain for EVM tokens (like eth, polygon, base, etc.) * @returns A promise that resolves to the price response */ async getPrice( token: string, chain?: BlockchainType, specificChain?: SpecificChain ): Promise<any> { let query = `?token=${encodeURIComponent(token)}`; // Add chain parameter if explicitly provided if (chain) { query += `&chain=${chain}`; } // Add specificChain parameter if provided (for EVM tokens) if (specificChain) { query += `&specificChain=${specificChain}`; } return this.request<any>('GET', `/api/price${query}`); } /** * Get detailed token information including specific chain * * @param token The token address * @param chain Optional blockchain type (auto-detected if not provided) * @param specificChain Optional specific chain for EVM tokens * @returns A promise that resolves to the token info response */ async getTokenInfo( token: string, chain?: BlockchainType, specificChain?: SpecificChain ): Promise<any> { let query = `?token=${encodeURIComponent(token)}`; // Add chain parameter if explicitly provided if (chain) { query += `&chain=${chain}`; } // Add specificChain parameter if provided if (specificChain) { query += `&specificChain=${specificChain}`; } return this.request<any>('GET', `/api/price/token-info${query}`); } /** * @deprecated The provider endpoint is no longer available * This method has been deprecated as the system now uses only the DexScreener provider. * Please use the getPrice() method instead with optional chain and specificChain parameters. */ async getPriceFromProvider( token: string, provider: string, chain?: BlockchainType, specificChain?: SpecificChain ): Promise<any> { console.warn('This method is deprecated. Please use getPrice() instead.'); return this.getPrice(token, chain, specificChain); } /** * Execute a token trade on the trading simulator * * This method allows you to trade between tokens on the same chain, * which is the default supported behavior. * * @param params - Trade parameters * @param params.fromToken - Source token address to sell * @param params.toToken - Destination token address to buy * @param params.amount - Amount of fromToken to sell (as string) * @param params.slippageTolerance - Optional slippage tolerance percentage (e.g., "0.5" for 0.5%) * @param params.fromChain - Blockchain type of source token (BlockchainType.EVM or BlockchainType.SVM) * @param params.fromSpecificChain - Specific chain for source token (e.g., SpecificChain.ETH, SpecificChain.BASE) * Providing this greatly improves performance for EVM tokens * @param params.toChain - Blockchain type of destination token (should match fromChain for within-chain trades) * @param params.toSpecificChain - Specific chain for destination token (should match fromSpecificChain) * * @returns Trade result with transaction ID, amounts, and updated balances * * @example * // Trade USDC for TOSHI on Base chain (within-chain trade) * const tradeResult = await client.executeTrade({ * fromToken: "0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913", // USDC on Base * toToken: "0x0b3e328455c4059EEb9e3f84b5543F74E24e7E1b", // TOSHI on Base * amount: "50", * fromChain: BlockchainType.EVM, * toChain: BlockchainType.EVM, * fromSpecificChain: SpecificChain.BASE, * toSpecificChain: SpecificChain.BASE * }); */ async executeTrade(params: { fromToken: string; toToken: string; amount: string; price?: string; slippageTolerance?: string; fromChain?: BlockchainType; toChain?: BlockchainType; fromSpecificChain?: SpecificChain; toSpecificChain?: SpecificChain; }): Promise<any> { // Create the request payload const payload: any = { fromToken: params.fromToken, toToken: params.toToken, amount: params.amount }; // Add optional parameters if they exist if (params.price) payload.price = params.price; if (params.slippageTolerance) payload.slippageTolerance = params.slippageTolerance; if (params.fromChain) payload.fromChain = params.fromChain; if (params.toChain) payload.toChain = params.toChain; if (params.fromSpecificChain) payload.fromSpecificChain = params.fromSpecificChain; if (params.toSpecificChain) payload.toSpecificChain = params.toSpecificChain; // If chain parameters are not provided, try to detect them if (!params.fromChain) { payload.fromChain = this.detectChain(params.fromToken); } if (!params.toChain) { payload.toChain = this.detectChain(params.toToken); } // Make the API request return this.request<any>('POST', '/api/trade/execute', payload); } /** * Get the status of the current competition * * @returns A promise that resolves to the competition status response */ async getCompetitionStatus(): Promise<any> { return this.request<any>('GET', '/api/competition/status'); } /** * Get the leaderboard for the current competition * * @returns A promise that resolves to the leaderboard response */ async getLeaderboard(): Promise<any> { return this.request<any>('GET', '/api/competition/leaderboard'); } /** * Get your team's profile information * * @returns A promise that resolves to the team profile */ async getProfile(): Promise<any> { return this.request<any>('GET', '/api/account/profile'); } /** * Update your team's profile information * * @param profileData Profile data to update * @returns A promise that resolves to the updated profile */ async updateProfile(profileData: any): Promise<any> { return this.request<any>('PUT', '/api/account/profile', profileData); } } // Example usage async function example() { // For security, use environment variables for credentials in production // This assumes you have dotenv installed and .env file configured // import dotenv from 'dotenv'; // dotenv.config(); // Create client with environment variables (recommended for production) // const client = new TradingSimulatorClient( // process.env.API_KEY || '', // process.env.API_SECRET || '', // process.env.API_BASE_URL || 'http://localhost:3001', // false // testMode=false uses current timestamps (required for production/development) // ); // Example with hardcoded values (for demonstration only, not recommended) const client = new TradingSimulatorClient( 'your-api-key', 'your-api-secret', 'http://localhost:3001', false // testMode=false is correct for both production and development // Only use testMode=true for automated tests with NODE_ENV=test ); try { // Get balances (shows all tokens across all chains) const balances = await client.getBalances(); console.log('Balances:', balances); // Get team profile const profile = await client.getProfile(); console.log('Team Profile:', profile); // Get price for SOL (Solana) const solPrice = await client.getPrice(COMMON_TOKENS.SVM.SOL); console.log('SOL Price:', solPrice); // Get price for ETH (Ethereum) const ethPrice = await client.getPrice(COMMON_TOKENS.EVM.ETH); console.log('ETH Price:', ethPrice); // Execute a trade to buy SOL on Solana const solTrade = await client.executeTrade({ fromToken: COMMON_TOKENS.SVM.USDC, toToken: COMMON_TOKENS.SVM.SOL, amount: '10', fromChain: BlockchainType.SVM, toChain: BlockchainType.SVM }); console.log('SOL Trade Result:', solTrade); // Execute a trade to buy ETH on Ethereum const ethTrade = await client.executeTrade({ fromToken: COMMON_TOKENS.EVM.USDC, toToken: COMMON_TOKENS.EVM.ETH, amount: '10', fromChain: BlockchainType.EVM, toChain: BlockchainType.EVM, fromSpecificChain: SpecificChain.ETH, toSpecificChain: SpecificChain.ETH }); console.log('ETH Trade Result:', ethTrade); // Execute a cross-chain trade (Solana USDC to Ethereum ETH) const crossChainTrade = await client.executeTrade({ fromToken: COMMON_TOKENS.SVM.USDC, toToken: COMMON_TOKENS.EVM.ETH, amount: '100', fromChain: BlockchainType.SVM, toChain: BlockchainType.EVM, fromSpecificChain: SpecificChain.SVM, toSpecificChain: SpecificChain.ETH }); console.log('Cross-Chain Trade Result:', crossChainTrade); // Get trade history (filtered by chain) const solTrades = await client.getTrades({ chain: BlockchainType.SVM }); console.log('Solana Trade History:', solTrades); // Get competition status const status = await client.getCompetitionStatus(); console.log('Competition Status:', status); } catch (error) { console.error('Error:', error); } } // Uncomment to run the example // example();