Satsuj1n

portfolio-mcp

Community Satsuj1n
Updated

Local-first, read-only MCP server to talk to your XP Investimentos portfolio through Claude Desktop. TypeScript + SQLite. Zero scraping, zero credentials.

portfolio-mcp

Talk to your XP Investimentos portfolio through Claude โ€” 100% local, read-only, zero scraping.

TypeScriptNode.jsMCPLicensenpmStatus

What is this?

portfolio-mcp is a Model Context Protocol server that exposes your XP Investimentos portfolio to Claude Desktop (or any MCP-compatible client). You feed it the official PDF/CSV exports XP already gives you โ€” the server parses them, normalizes everything into a local SQLite database, and answers questions through MCP tools.

  • ๐Ÿ”’ Local-first โ€” data lives in ~/.xp-mcp/data.db. Core tools make zero outbound HTTP calls; advisor tools (v0.4+) are opt-in.
  • ๐Ÿ“ต No credentials, no scraping โ€” you control what goes in (PDF/CSV exports).
  • ๐Ÿ‘ Read-only by design โ€” no tool can place orders or modify anything at XP.
  • ๐Ÿงฑ Stdio transport โ€” Claude spawns the process; nothing listens on a port.

Built for personal use, but the architecture works for any custodian that lets you export reports (Rico, NuInvest, Inter, Avenue, ...). PRs welcome.

Why does this exist?

Custodian apps are great at "show me what I have" but terrible at:

  • "Compare my current allocation against my target portfolio."
  • "Which of my CDBs mature in the next 12 months?"
  • "What's my real yield-to-maturity weighted by position size?"
  • "How concentrated am I in a single issuer's FGC limit?"

MCP lets you bolt that intelligence onto Claude without giving the LLM provider your credentials, sending your positions to a third-party SaaS, or trusting browser-scraping that breaks every quarter.

Demo

After connecting the server, just chat with Claude:

You: Liste minhas posiรงรตes agrupadas por classe de ativo.

Claude (using portfolio-mcp.get_positions):
  TESOURO (53.28%)        R$ 16,644.31
    โ€ข Tesouro Selic 2031  R$ 14,556.80
    โ€ข Tesouro Selic 2028  R$  2,087.51

  RENDA_FIXA_PRIVADA (20.98%)  R$  6,554.32
    โ€ข CDB BANCO XP - AGO/2026 - 15,00%       R$ 2,203.85
    โ€ข CDB BANCO XP - SET/2026 - 14,95%       R$ 2,197.10
    โ€ข CDB BANCO XP - OUT/2026 - 14,55%       R$ 1,083.63
    โ€ข CDB BANCO XP - NOV/2027 - 100,00% CDI  R$ 1,069.74

  FUNDO  (10.48%)  R$  3,273.71
  ACAO   ( 7.44%)  R$  2,325.12  (BBAS3)
  FII    ( 7.81%)  R$  2,439.20  (MXRF11, GTWR11)

  Total: R$ 31,236.66

You: Quais CDBs vencem em 2026?

Claude (filters by asset_class + maturity_date):
  3 CDBs maturing in 2026:
    โ€ข Aug/2026 โ€” 15.00% pre-fixed     R$ 2,203.85
    โ€ข Sep/2026 โ€” 14.95% pre-fixed     R$ 2,197.10
    โ€ข Oct/2026 โ€” 14.55% pre-fixed     R$ 1,083.63
  All issued by BANCO XP S.A. (within single-issuer FGC limit โœ“)

You: Estou bem alocado em relaรงรฃo ao meu target?

Claude (using portfolio-mcp.calculate_allocation_drift):
  Lendo ~/.xp-mcp/allocation.json (target: 40% Tesouro, 20% RF, 15% FII,
  15% Aรงรตes, 5% ETF, 5% Fundos, tolerรขncia ยฑ2pp).

  Mais fora do alvo:
    โ€ข TESOURO:  53.28%  (target 40%)  โ†’ vender ~R$ 4.148
    โ€ข ACAO:      7.44%  (target 15%)  โ†’ comprar ~R$ 2.362
    โ€ข FII:       7.81%  (target 15%)  โ†’ comprar ~R$ 2.247

  Dentro da banda:
    โ€ข RENDA_FIXA_PRIVADA: 20.98% (target 20%) โœ“

  Fora da banda, magnitude menor:
    โ€ข FUNDO: 10.48% (target  5%) โ€” vender ~R$ 1.712
    โ€ข ETF:    0.00% (target  5%) โ€” comprar ~R$ 1.562

  Net rebalance: -R$ 152 (aporte pequeno sugerido pra fechar).

You: Me dรก um panorama do meu portfรณlio.

Claude (using portfolio-mcp.get_portfolio_summary):
  Total: R$ 31.236,66 (12 posiรงรตes, ref. 2026-05-21)
  P&L: +R$ 2.736,66 (+9,60% sobre R$ 28.500 investidos ยท 8/12 com P&L computรกvel)

  Por classe:
    TESOURO              53,28%  R$ 16.644,31
    RENDA_FIXA_PRIVADA   20,98%  R$  6.554,32  (+4,04%)
    FUNDO                10,48%  R$  3.273,71
    ACAO                  7,44%  R$  2.325,12
    FII                   7,81%  R$  2.439,20

  FGC: R$ 6.554,32 (20,98%) cobertos
  Vencimentos: curto R$ 5.484 ยท mรฉdio R$ 16.625 ยท longo R$ 1.070 ยท sem maturity R$ 8.038

  Reconciliaรงรฃo: declarado R$ 31.250,00 vs computado R$ 31.236,66
    โ†’ gap โˆ’R$ 13,34 (โˆ’0,04%, dentro de tolerรขncia)

Architecture

โ”Œโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”
โ”‚  Claude Desktop โ”‚
โ””โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ฌโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜
         โ”‚ stdio (JSON-RPC 2.0)
         โ–ผ
โ”Œโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”       PDF / CSV       โ”Œโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”
โ”‚   portfolio-mcp     โ”‚โ—€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”‚   XP exports         โ”‚
โ”‚  TypeScript + MCP   โ”‚                       โ”‚   (your machine)     โ”‚
โ””โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ฌโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜                       โ””โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜
         โ”‚ SQL (better-sqlite3, WAL mode)
         โ–ผ
โ”Œโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”
โ”‚      SQLite         โ”‚
โ”‚   ~/.xp-mcp/        โ”‚
โ””โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”˜

      No network. No credentials. Read-only at XP.

Stack: TypeScript ยท @modelcontextprotocol/sdk ยท better-sqlite3 ยท pdf-parse ยท papaparse ยท zod

Tools

Tool Purpose Status
import_xperformance_pdf Parse XP's official portfolio PDF (XPerformance). Idempotent on re-import. โœ…
import_extract_csv Parse a Posiรงรฃo Consolidada / Extrato CSV. Auto-detects delimiter and column aliases. โœ…
get_positions List positions with quantity, market value, indexer, maturity. Optional class filter. โœ…
get_portfolio_summary Aggregate stats + declared-vs-computed reconciliation gap โœ…
get_transactions History of buys/sells โณ
get_dividends Income / proventos โณ
calculate_allocation_drift Compare current vs target allocation from ~/.xp-mcp/allocation.json. Returns drift %, BRL delta, and BUY/SELL suggestions per class. โœ…
set_advisor_profile Save the advisor profile (risk, horizon, objective, exclusions, outbound gate, brapi token). โœ…
get_advisor_profile Read the saved profile. Returns exists: false if not configured. โœ…
get_market_data Quotes / fundamentals from brapi.dev for 1-50 tickers, SQLite-cached. Opt-in. โœ…
screen_assets Rank B3 FIIs / stocks / ETFs by DY, P/VP, P/L, ROE, market cap. Opt-in. โœ…
import_nota_corretagem Parse broker-note PDFs for transaction history โณ
import_bank_extract_pdf Import a PDF exported from XP's Conta Digital Extrato. Filters for investment-account transfers only (APORTE/RESGATE). Idempotent. โœ…
get_cash_flows List cash flows with optional date/kind filters; returns aggregate totals (aporte/resgate/net) over all matching rows. โœ…
suggest_buys Suggest BUYs per underweight class using profile objective ร— asset class matrix (FII/ACAO/ETF). Non-screenable classes (TESOURO/RF/FUNDO) reported in skipped_classes. Requires outbound_enabled=true. โœ…

Quick Start

Add this to your claude_desktop_config.json:

{
  "mcpServers": {
    "portfolio": {
      "command": "npx",
      "args": ["-y", "portfolio-mcp"]
    }
  }
}

Config file location:

  • macOS: ~/Library/Application Support/Claude/claude_desktop_config.json
  • Windows: %APPDATA%\Claude\claude_desktop_config.json
  • Linux: ~/.config/Claude/claude_desktop_config.json

Restart Claude Desktop. Done.

See Development below for clone + build instructions if you want to hack on the server.

How to export from XP

XP has no single "export everything" button. Try in this order:

  1. XPerformance PDF (recommended) โ€” Investimentos โ†’ XPerformance โ†’ โฌ‡ PDF. The richest single-file snapshot.
  2. Portal XP web โ€” Investimentos โ†’ Posiรงรฃo Consolidada โ†’ Exportar CSV
  3. Hub XP / Assessor Digital โ€” Meus Investimentos โ†’ Posiรงรฃo โ†’ Exportar
  4. Notas de corretagem (PDF) โ€” Conta โ†’ Documentos โ†’ Notas de Corretagem (phase 2)
  5. Extrato de movimentaรงรฃo โ€” Conta โ†’ Extrato โ†’ filtrar perรญodo โ†’ Exportar

The CSV parser auto-detects ;, ,, \t, | delimiters and fuzzy-matches common XP column names (Quantidade, Qtd, Preรงo mรฉdio, Valor aplicado, ...). If a column is missed, extend HEADER_ALIASES in src/parsers/csv-extract.ts.

Configuring your target allocation

calculate_allocation_drift reads ~/.xp-mcp/allocation.json. A starter file is included at examples/allocation.example.json. Copy it once and edit the percentages:

mkdir -p ~/.xp-mcp
cp examples/allocation.example.json ~/.xp-mcp/allocation.json

The six valid keys are TESOURO, RENDA_FIXA_PRIVADA, FII, ETF, ACAO, FUNDO. Values must sum to 1.00 (ยฑ0.001). tolerance_pp is optional โ€” when set, drifts within the band are reported as "ok" with no action.

Try it without Claude

The server speaks JSON-RPC 2.0 over stdio. Smoke test from any terminal:

node dist/index.js <<'EOF'
{"jsonrpc":"2.0","id":1,"method":"initialize","params":{"protocolVersion":"2024-11-05","capabilities":{},"clientInfo":{"name":"smoke","version":"0.0.0"}}}
{"jsonrpc":"2.0","method":"notifications/initialized"}
{"jsonrpc":"2.0","id":2,"method":"tools/list"}
EOF

You should see the server announce itself and list the registered tools.

Advisor (opt-in)

v0.4 adds four MCP tools that turn portfolio-mcp into an investment-analysis advisor. These tools are opt-in: they only make outbound HTTP calls when the user has set outbound_enabled: true in their profile and accepted the disclaimer.

Configure your profile

The profile lives at ~/.xp-mcp/advisor-profile.json. Create or update it with set_advisor_profile:

{
  "risk_tolerance": 6,
  "horizon_years": 15,
  "objective": "balanced",
  "monthly_income_target_brl": 3000,
  "excluded_classes": ["FUNDO"],
  "excluded_tickers": ["XYZW3"],
  "notes": "evitar tabaco, prefiro logรญstica e shoppings",
  "outbound_enabled": true,
  "brapi_token": "..."
}

When you flip outbound_enabled to true, the tool requires accept_disclaimer: true in the same call and stamps accepted_disclaimer_at on the saved profile.

Disclaimer

Anรกlise educacional baseada em dados pรบblicos. Nรฃo constitui recomendaรงรฃo de investimento. Decisรตes financeiras sรฃo de sua responsabilidade.

The disclaimer is also returned in warnings[0] of every advisor tool that ranks or suggests assets.

Source: brapi.dev

v0.4 uses brapi.dev as the sole market-data source. Anonymous calls work for personal use; if you hit rate limits, set brapi_token in the profile. Quotes are cached for 60 minutes, fundamentals for 24 hours, universe lists for 7 days. Override per call via cache_ttl_minutes.

Example flow

  1. set_advisor_profile โ€” saves the profile, enables outbound
  2. get_market_data { tickers: ["BBAS3","MXRF11"] } โ€” quotes
  3. screen_assets { asset_class: "FII", criteria: { sort_by: "dividend_yield", order: "desc", filters: { min_dividend_yield_pct: 8 }, limit: 10 } }

Project layout

src/
โ”œโ”€โ”€ index.ts                       # MCP server entrypoint (stdio transport)
โ”œโ”€โ”€ tools/
โ”‚   โ”œโ”€โ”€ import-xperformance-pdf.ts # XPerformance PDF โ†’ positions
โ”‚   โ”œโ”€โ”€ import-extract-csv.ts      # XP CSV โ†’ positions
โ”‚   โ””โ”€โ”€ get-positions.ts           # SQL โ†’ positions list with P&L
โ”œโ”€โ”€ parsers/
โ”‚   โ”œโ”€โ”€ pdf-xperformance.ts        # PDF text โ†’ rows
โ”‚   โ”œโ”€โ”€ csv-extract.ts             # papaparse + fuzzy header matching
โ”‚   โ”œโ”€โ”€ classify.ts                # asset-class heuristics + name metadata
โ”‚   โ””โ”€โ”€ normalize.ts               # BRL / date / quantity normalizers
โ”œโ”€โ”€ storage/
โ”‚   โ”œโ”€โ”€ db.ts                      # better-sqlite3 singleton, WAL, env override
โ”‚   โ”œโ”€โ”€ schema.ts                  # CREATE TABLE + AssetClass enum
โ”‚   โ””โ”€โ”€ positions-repo.ts          # UPSERT, listPositions, import records
โ””โ”€โ”€ util/
    โ””โ”€โ”€ zod-to-json-schema.ts      # minimal zod โ†’ JSON Schema for MCP

Data model

imports        every parse attempt with row counts, timestamps, source path
positions      one row per (asset_class, external_id); upserted on re-import
transactions   buys/sells from broker notes (phase 2)
dividends      income / proventos (phase 2)

All monetary values are stored as INTEGER cents. Float math + currency is a well-known source of off-by-a-cent bugs; integers make every aggregation exact.

The UNIQUE (asset_class, external_id) constraint + ON CONFLICT DO UPDATE makes re-importing the same file idempotent โ€” no duplicates, no manual deduping.

Design decisions worth calling out

  • Wide positions table with nullable per-class columns. A table-per-asset-class would be cleaner in theory but adds JOINs for every read, and the column set is small. The wide table fits the access pattern (Claude almost always wants "all positions, maybe filtered by class").
  • Stdio over HTTP. Claude Desktop spawns the process directly. Zero ports listening, no auth surface, no CORS to misconfigure. Trade-off: no remote clients without a wrapper.
  • stderr-only logging. stdout is the JSON-RPC wire. One stray console.log silently corrupts every response. All logs go to stderr, where Claude Desktop captures them into ~/Library/Logs/Claude/mcp-server-xp.log.
  • No price fetching in MVP. Adding it means a network dependency and a rate-limit problem. When it lands (phase 2), it'll be behind an opt-in env var with a clearly-documented data source.

Privacy

  • The SQLite file (~/.xp-mcp/data.db) is the only place your position data lives. .gitignore blocks *.db, *.sqlite, and data/ from ever being committed.
  • The server makes no outbound HTTP calls for the core import/inspection tools (import_xperformance_pdf, import_extract_csv, get_positions, calculate_allocation_drift). No telemetry. No analytics.
  • The v0.4 advisor tools (get_market_data, screen_assets) call brapi.dev only when outbound_enabled: true in ~/.xp-mcp/advisor-profile.json and the user has accepted the disclaimer. Disabled by default.

Roadmap

  • PDF parser for XPerformance (XP's official portfolio report)
  • CSV parser for Posiรงรฃo Consolidada / Extrato
  • get_positions with class filter, P&L when invested-value is known
  • import_nota_corretagem (broker-note PDF โ†’ transactions)
  • CSV parser for proventos export
  • calculate_allocation_drift against ~/.xp-mcp/allocation.json
  • v0.3 โ€” npm publish + Smithery + awesome-mcp PR
  • v0.4 โ€” Investment advisor foundations: set_advisor_profile, get_advisor_profile, get_market_data, screen_assets (brapi.dev, SQLite cache)
  • v0.5 โ€” Portfolio summary + reconciliation gap: get_portfolio_summary (aggregate stats, declared-vs-computed gap)
  • v0.6 โ€” Bank extract import + cash flows: import_bank_extract_pdf (XP Conta Digital Extrato โ†’ cash_flows table, schema v3), get_cash_flows (filterable query with aggregate totals)
  • v0.7 โ€” Suggest Buys: suggest_buys (composes profile + drift + screening into deterministic BUY suggestions per underweight class; non-screenable classes surface in skipped_classes)
  • Crypto adapter โ€” new MarketDataSource for Binance / Mercado Bitcoin / Foxbit
  • Adapters for other Brazilian brokers (Rico, NuInvest, Inter, Avenue)
  • Open Finance Brasil investment module when the standard matures

Changelog

v0.7.0 โ€” Suggest Buys (2026-05-23)

  • New tool: suggest_buys โ€” composes advisor profile + allocation drift + asset screening into a deterministic list of BUY suggestions per underweight asset class.
  • Encodes a fixed objective ร— asset_class screening matrix (income/growth/balanced ร— FII/ACAO/ETF).
  • Non-screenable underweight classes (TESOURO, RENDA_FIXA_PRIVADA, FUNDO) surface in a dedicated skipped_classes field rather than being silently dropped.
  • Each suggestion includes amount_brl (= class gap / actual_returned), score, rationale, and already_owned.
  • 12 tools total (was 11). No schema migration.

Development

For contributors or anyone who wants to run a local checkout instead of the published npm package:

git clone https://github.com/Satsuj1n/portfolio-mcp
cd portfolio-mcp
npm install
npm run build
npm run test

Then point claude_desktop_config.json at the local build:

{
  "mcpServers": {
    "portfolio": {
      "command": "node",
      "args": ["/absolute/path/to/portfolio-mcp/dist/index.js"]
    }
  }
}

Run the type checker with npm run typecheck. Smoke-test the MCP server end-to-end with:

node dist/index.js <<'EOF'
{"jsonrpc":"2.0","id":1,"method":"initialize","params":{"protocolVersion":"2024-11-05","capabilities":{},"clientInfo":{"name":"local","version":"0.0.0"}}}
{"jsonrpc":"2.0","method":"notifications/initialized"}
{"jsonrpc":"2.0","id":2,"method":"tools/list"}
EOF

Contributing

PRs welcome โ€” especially:

  • New parsers (other brokers, other export formats)
  • Heuristics for classifyAsset covering edge cases your portfolio exposes
  • Additional tools (calculate_allocation_drift, get_dividends, ...)
  • Tests with anonymized fixtures (no real position values, no real account numbers)

Please never include real portfolio data, account numbers, or PDFs in your PR. Use the fixture pattern in tests/fixtures/ (synthetic data only).

License

MIT โ€” see LICENSE.

Built by Felipe Lima ยท Powered by Model Context Protocol.

MCP Server ยท Populars

MCP Server ยท New

    TickDB

    TickDB โ€” Unified Real-time Market Data API for Forex, Stocks, Crypto

    ็ปŸไธ€็š„ๅฎžๆ—ถ้‡‘่ž่กŒๆƒ…ๆ•ฐๆฎAPI๏ผŒ่ฆ†็›–ๅค–ๆฑ‡ใ€่ดต้‡‘ๅฑžใ€ๆŒ‡ๆ•ฐใ€็พŽ่‚กใ€ๆธฏ่‚กใ€A่‚กๅ’ŒๅŠ ๅฏ†่ดงๅธ๏ผŒๆ”ฏๆŒ WebSocket ๅฎžๆ—ถๆŽจ้€ไธŽ REST ๆŽฅๅฃ่ฎฟ้—ฎ | Unified real-time market data API covering forex, commodities, indices, US stocks, HK stocks, A-shares and cryptocurrencies, with WebSocket streaming and REST access

    Community TickDB
    Patdolitse

    piia-engram

    One memory. Every AI tool. Yours to keep. Local-first, MCP-compatible, Apache 2.0.

    Community Patdolitse
    ada20204

    antigravity-sync

    MCP Server

    Community ada20204
    SepineTam

    mcp-for-stata

    Integrate Stata into your agent.

    Community SepineTam
    Keesan12

    martin-loop

    The control plane for autonomous work and coding agent teams.

    Community Keesan12