UNPKG

@sei-js/mcp-server

Version:

Model Context Protocol (MCP) server for interacting with EVM-compatible networks

161 lines (160 loc) 5.5 kB
import { formatEther, formatUnits, getContract } from 'viem'; import { DEFAULT_NETWORK } from '../chains.js'; import { getPublicClient } from './clients.js'; import { readContract } from './contracts.js'; import * as services from './index.js'; // Standard ERC20 ABI (minimal for reading) const erc20Abi = [ { inputs: [], name: 'symbol', outputs: [{ type: 'string' }], stateMutability: 'view', type: 'function' }, { inputs: [], name: 'decimals', outputs: [{ type: 'uint8' }], stateMutability: 'view', type: 'function' }, { inputs: [{ type: 'address', name: 'account' }], name: 'balanceOf', outputs: [{ type: 'uint256' }], stateMutability: 'view', type: 'function' } ]; // Standard ERC721 ABI (minimal for reading) const erc721Abi = [ { inputs: [{ type: 'address', name: 'owner' }], name: 'balanceOf', outputs: [{ type: 'uint256' }], stateMutability: 'view', type: 'function' }, { inputs: [{ type: 'uint256', name: 'tokenId' }], name: 'ownerOf', outputs: [{ type: 'address' }], stateMutability: 'view', type: 'function' } ]; // Standard ERC1155 ABI (minimal for reading) const erc1155Abi = [ { inputs: [ { type: 'address', name: 'account' }, { type: 'uint256', name: 'id' } ], name: 'balanceOf', outputs: [{ type: 'uint256' }], stateMutability: 'view', type: 'function' } ]; /** * Get the Sei balance for an address * @param address Sei address * @param network Network name or chain ID * @returns Balance in wei and sei */ export async function getBalance(address, network = DEFAULT_NETWORK) { const validatedAddress = services.helpers.validateAddress(address); const client = getPublicClient(network); const balance = await client.getBalance({ address: validatedAddress }); return { wei: balance, sei: formatEther(balance) }; } /** * Get the balance of an ERC20 token for an address * @param tokenAddress Token contract address * @param ownerAddress Owner address * @param network Network name or chain ID * @returns Token balance with formatting information */ export async function getERC20Balance(tokenAddress, ownerAddress, network = DEFAULT_NETWORK) { const validatedTokenAddress = services.helpers.validateAddress(tokenAddress); const validatedOwnerAddress = services.helpers.validateAddress(ownerAddress); const publicClient = getPublicClient(network); const contract = getContract({ address: validatedTokenAddress, abi: erc20Abi, client: publicClient }); const [balance, symbol, decimals] = await Promise.all([contract.read.balanceOf([validatedOwnerAddress]), contract.read.symbol(), contract.read.decimals()]); return { raw: balance, formatted: formatUnits(balance, decimals), token: { symbol, decimals } }; } /** * Check if an address owns a specific NFT * @param tokenAddress NFT contract address * @param ownerAddress Owner address * @param tokenId Token ID to check * @param network Network name or chain ID * @returns True if the address owns the NFT */ export async function isNFTOwner(tokenAddress, ownerAddress, tokenId, network = DEFAULT_NETWORK) { const validatedTokenAddress = services.helpers.validateAddress(tokenAddress); const validatedOwnerAddress = services.helpers.validateAddress(ownerAddress); try { const actualOwner = (await readContract({ address: validatedTokenAddress, abi: erc721Abi, functionName: 'ownerOf', args: [tokenId] }, network)); return actualOwner.toLowerCase() === validatedOwnerAddress.toLowerCase(); } catch (error) { console.error(`Error checking NFT ownership: ${error instanceof Error ? error.message : String(error)}`); return false; } } /** * Get the number of NFTs owned by an address for a specific collection * @param tokenAddress NFT contract address * @param ownerAddress Owner address * @param network Network name or chain ID * @returns Number of NFTs owned */ export async function getERC721Balance(tokenAddress, ownerAddress, network = DEFAULT_NETWORK) { const validatedTokenAddress = services.helpers.validateAddress(tokenAddress); const validatedOwnerAddress = services.helpers.validateAddress(ownerAddress); return (await readContract({ address: validatedTokenAddress, abi: erc721Abi, functionName: 'balanceOf', args: [validatedOwnerAddress] }, network)); } /** * Get the balance of an ERC1155 token for an address * @param tokenAddress ERC1155 contract address * @param ownerAddress Owner address * @param tokenId Token ID to check * @param network Network name or chain ID * @returns Token balance */ export async function getERC1155Balance(tokenAddress, ownerAddress, tokenId, network = DEFAULT_NETWORK) { const validatedTokenAddress = services.helpers.validateAddress(tokenAddress); const validatedOwnerAddress = services.helpers.validateAddress(ownerAddress); return (await readContract({ address: validatedTokenAddress, abi: erc1155Abi, functionName: 'balanceOf', args: [validatedOwnerAddress, tokenId] }, network)); }