zapper-mcp

zapper-mcp

An MCP server that exposes the Zapper DeFi portfolio API, enabling LLMs to query wallet balances, token holdings, and DeFi positions using natural language.

Category
Visit Server

README

zapper-mcp

An MCP server that exposes the Zapper DeFi portfolio API as a thoughtfully designed tool surface for LLM clients. Connect it to Claude Desktop or any MCP-compatible host and ask natural-language questions about any wallet — "what is this wallet worth?", "does it have any Aave positions?", "show me the top holdings on Base."

Built on Day 9 of a 21-day AI engineering sprint. Day 10 wires this server into a Mastra agent.


Tool surface

The design rationale for each primitive is in DESIGN.md. The short version:

Primitive Name Why this placement
Tool get_portfolio Model-invoked, dynamic per address, returns full token + DeFi breakdown
Tool get_token_balances Focused tool for spot-token questions; avoids making the model parse a full portfolio when it only needs token holdings
Tool get_app_positions Focused tool for DeFi questions; separate from get_portfolio so the model can express precise intent and receive a focused schema
Resource zapper://supported-networks Static network list — host injects it as ambient context at prompt-assembly time so the model knows valid network names without burning a tool-call turn
Prompt analyze-wallet User-invoked workflow that pre-seeds a multi-turn portfolio analysis conversation with analyst persona, tool inventory, and wallet address

Why not one big get_everything tool? Collapsing the tools would force the model to receive and parse a large mixed-schema response for every question, even focused ones. A tool boundary is a declaration of scope — the right tool returns exactly what the reasoning step needs.

Why is the API key in server config, not a tool argument? Credentials belong in the host layer (env vars injected at process spawn), not in the MCP protocol. If api_key were a tool parameter, it would flow through the LLM's reasoning and appear in conversation history. For a multi-tenant deploy the right mechanism is transport-layer auth (Bearer token over Streamable HTTP) or per-user OAuth — both out of scope here. See Known limitations.


Requirements


Install

git clone https://github.com/mehdi-loup/zapper-mcp
cd zapper-mcp
pnpm install
pnpm build

Configuration

Copy .env.example to .env and add your key:

cp .env.example .env
# edit .env and set ZAPPER_API_KEY=your_key_here

The server fails fast at boot if ZAPPER_API_KEY is missing — you'll see the error immediately, not on the first tool call.


Run

Standalone smoke test (confirms everything works without Claude Desktop):

ZAPPER_API_KEY=your_key pnpm client

Output: lists tools/resources/prompts, then calls each tool against vitalik.eth.

Direct server start:

ZAPPER_API_KEY=your_key pnpm start

Claude Desktop wiring

Add to ~/Library/Application Support/Claude/claude_desktop_config.json:

{
  "mcpServers": {
    "zapper-mcp": {
      "command": "node",
      "args": ["/absolute/path/to/zapper-mcp/build/server.js"],
      "env": {
        "ZAPPER_API_KEY": "your_key_here"
      }
    }
  }
}

Restart Claude Desktop. The three tools, the zapper://supported-networks resource, and the analyze-wallet prompt will be available.

Logs (if the server fails to load):

~/Library/Logs/Claude/mcp-server-zapper-mcp.log

Mastra integration (Day 10)

To wire this server into a Mastra agent via Mastra's MCP client:

  1. Start the server: node /path/to/build/server.js
  2. Configure the Mastra MCP client with stdio transport, server name zapper-mcp
  3. The agent consumes Zapper data exclusively through MCP — lib/zapper.ts in the agent repo becomes unused

Not all tools need to be exposed to the Mastra agent; that's a Day 10 design call.


Tool reference

get_portfolio(address, networks?)

Full portfolio breakdown: total USD, all token holdings, all DeFi positions.

address   — wallet address or ENS name
networks  — optional array: ["ethereum", "base", "arbitrum", ...]

get_token_balances(address, networks?)

Spot token balances only (no DeFi positions).

get_app_positions(address, networks?, app_slug?)

DeFi app positions only (Aave, Uniswap, Sablier, etc.).

app_slug  — optional filter: "aave-v3", "uniswap-v3", ...

Resource: zapper://supported-networks

JSON array of { name, chainId } for all indexed networks. Read by host at context-assembly time.

Prompt: analyze-wallet

Pre-seeds a portfolio analysis conversation. Takes an address argument.


Error handling

Every tool returns isError: true with a model-actionable message on:

  • HTTP 401 / invalid API key
  • HTTP 429 / rate limited
  • HTTP 5xx / Zapper server error
  • Network timeout (15s)
  • Malformed response

An empty wallet (totalUSD: 0, tokens: []) returns isError: false — empty is not an error.


Known limitations

  • Single-key trust model: the server holds one ZAPPER_API_KEY and serves one owner. A multi-tenant deploy needs per-user OAuth or transport-layer auth (Streamable HTTP with Bearer tokens).
  • No caching: every tool call hits the Zapper API. A production server would add a short TTL cache (positions change slowly) and respect rate limits proactively.
  • No resources/subscribe: zapper://supported-networks is a static list. Live updates would require the server to advertise subscribe capability and emit notifications/resources/updated.
  • stdio transport only: Streamable HTTP transport deferred to a future iteration.
  • Pagination ceiling: tools return up to 50 tokens and 20 app positions per request.

What's next

Day 10: wire this server into the Mastra wallet agent at ../day1-wallet-agent/ via Mastra's MCP client. The agent will consume Zapper data exclusively through MCP, validating that the tool surface actually decouples the capability from the agent framework.

Recommended Servers

playwright-mcp

playwright-mcp

A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.

Official
Featured
TypeScript
Magic Component Platform (MCP)

Magic Component Platform (MCP)

An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.

Official
Featured
Local
TypeScript
Audiense Insights MCP Server

Audiense Insights MCP Server

Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.

Official
Featured
Local
TypeScript
VeyraX MCP

VeyraX MCP

Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.

Official
Featured
Local
graphlit-mcp-server

graphlit-mcp-server

The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.

Official
Featured
TypeScript
Kagi MCP Server

Kagi MCP Server

An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.

Official
Featured
Python
E2B

E2B

Using MCP to run code via e2b.

Official
Featured
Neon Database

Neon Database

MCP server for interacting with Neon Management API and databases

Official
Featured
Qdrant Server

Qdrant Server

This repository is an example of how to create a MCP server for Qdrant, a vector search engine.

Official
Featured
Exa Search

Exa Search

A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.

Official
Featured