EVM MCP Server

by mcpdotdirect
Verified
import { formatEther, formatUnits, type Address, type Abi, getContract } from 'viem'; import { getPublicClient } from './clients.js'; import { readContract } from './contracts.js'; import { resolveAddress } from './ens.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' } ] as const; // 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' } ] as const; // 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' } ] as const; /** * Get the ETH balance for an address * @param addressOrEns Ethereum address or ENS name * @param network Network name or chain ID * @returns Balance in wei and ether */ export async function getETHBalance( addressOrEns: string, network = 'ethereum' ): Promise<{ wei: bigint; ether: string }> { // Resolve ENS name to address if needed const address = await resolveAddress(addressOrEns, network); const client = getPublicClient(network); const balance = await client.getBalance({ address }); return { wei: balance, ether: formatEther(balance) }; } /** * Get the balance of an ERC20 token for an address * @param tokenAddressOrEns Token contract address or ENS name * @param ownerAddressOrEns Owner address or ENS name * @param network Network name or chain ID * @returns Token balance with formatting information */ export async function getERC20Balance( tokenAddressOrEns: string, ownerAddressOrEns: string, network = 'ethereum' ): Promise<{ raw: bigint; formatted: string; token: { symbol: string; decimals: number; } }> { // Resolve ENS names to addresses if needed const tokenAddress = await resolveAddress(tokenAddressOrEns, network); const ownerAddress = await resolveAddress(ownerAddressOrEns, network); const publicClient = getPublicClient(network); const contract = getContract({ address: tokenAddress, abi: erc20Abi, client: publicClient, }); const [balance, symbol, decimals] = await Promise.all([ contract.read.balanceOf([ownerAddress]), 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 tokenAddressOrEns NFT contract address or ENS name * @param ownerAddressOrEns Owner address or ENS name * @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( tokenAddressOrEns: string, ownerAddressOrEns: string, tokenId: bigint, network = 'ethereum' ): Promise<boolean> { // Resolve ENS names to addresses if needed const tokenAddress = await resolveAddress(tokenAddressOrEns, network); const ownerAddress = await resolveAddress(ownerAddressOrEns, network); try { const actualOwner = await readContract({ address: tokenAddress, abi: erc721Abi, functionName: 'ownerOf', args: [tokenId] }, network) as Address; return actualOwner.toLowerCase() === ownerAddress.toLowerCase(); } catch (error: any) { console.error(`Error checking NFT ownership: ${error.message}`); return false; } } /** * Get the number of NFTs owned by an address for a specific collection * @param tokenAddressOrEns NFT contract address or ENS name * @param ownerAddressOrEns Owner address or ENS name * @param network Network name or chain ID * @returns Number of NFTs owned */ export async function getERC721Balance( tokenAddressOrEns: string, ownerAddressOrEns: string, network = 'ethereum' ): Promise<bigint> { // Resolve ENS names to addresses if needed const tokenAddress = await resolveAddress(tokenAddressOrEns, network); const ownerAddress = await resolveAddress(ownerAddressOrEns, network); return readContract({ address: tokenAddress, abi: erc721Abi, functionName: 'balanceOf', args: [ownerAddress] }, network) as Promise<bigint>; } /** * Get the balance of an ERC1155 token for an address * @param tokenAddressOrEns ERC1155 contract address or ENS name * @param ownerAddressOrEns Owner address or ENS name * @param tokenId Token ID to check * @param network Network name or chain ID * @returns Token balance */ export async function getERC1155Balance( tokenAddressOrEns: string, ownerAddressOrEns: string, tokenId: bigint, network = 'ethereum' ): Promise<bigint> { // Resolve ENS names to addresses if needed const tokenAddress = await resolveAddress(tokenAddressOrEns, network); const ownerAddress = await resolveAddress(ownerAddressOrEns, network); return readContract({ address: tokenAddress, abi: erc1155Abi, functionName: 'balanceOf', args: [ownerAddress, tokenId] }, network) as Promise<bigint>; }