Stránkování a velké sady výsledků v MCP
February 8, 2026 · View on GitHub
Když váš MCP server zpracovává velké datové sady – ať už se jedná o tisíce souborů, záznamů v databázi nebo výsledků vyhledávání – potřebujete stránkování, abyste efektivně spravovali paměť a poskytli rychlou reakci uživatelského rozhraní. Tento průvodce pokrývá, jak implementovat a používat stránkování v MCP.
Proč je stránkování důležité
Bez stránkování mohou velké odpovědi způsobit:
- Vyčerpání paměti – Načítání milionů záznamů najednou
- Pomalé reakční časy – Uživatelé čekají, zatímco se načtou všechna data
- Chyby časového limitu – Požadavky překročí limity čekání
- Špatný výkon AI – LLM mají problém s masivním kontextem
MCP používá stránkování založené na kurzoru pro spolehlivé a konzistentní procházení sad výsledků.
Jak MCP stránkování funguje
Koncept kurzoru
Kurzor je neprůhledný řetězec, který označuje vaši pozici v sadě výsledků. Představte si ho jako záložku v dlouhé knize.
sequenceDiagram
participant Client
participant Server
Client->>Server: tools/list (bez kurzoru)
Server-->>Client: nástroje [1-10], nextCursor: "abc123"
Client->>Server: tools/list (kurzor: "abc123")
Server-->>Client: nástroje [11-20], nextCursor: "def456"
Client->>Server: tools/list (kurzor: "def456")
Server-->>Client: nástroje [21-25], nextCursor: null (konec)
Stránkování v MCP metodách
Tyto MCP metody podporují stránkování:
| Metoda | Vrací | Podpora kurzoru |
|---|---|---|
tools/list | Definice nástrojů | ✅ |
resources/list | Definice zdrojů | ✅ |
prompts/list | Definice promptů | ✅ |
resources/templates/list | Šablony zdrojů | ✅ |
Implementace na serveru
Python (FastMCP)
from mcp.server import Server
from mcp.types import Tool, ListToolsResult
import math
app = Server("paginated-server")
# Simulovaný velký dataset
ALL_TOOLS = [
Tool(name=f"tool_{i}", description=f"Tool number {i}", inputSchema={})
for i in range(100)
]
PAGE_SIZE = 10
@app.list_tools()
async def list_tools(cursor: str | None = None) -> ListToolsResult:
"""List tools with pagination support."""
# Dekódujte kurzor pro získání počátečního indexu
start_index = 0
if cursor:
try:
start_index = int(cursor)
except ValueError:
start_index = 0
# Získat stránku výsledků
end_index = min(start_index + PAGE_SIZE, len(ALL_TOOLS))
page_tools = ALL_TOOLS[start_index:end_index]
# Vypočítat další kurzor
next_cursor = None
if end_index < len(ALL_TOOLS):
next_cursor = str(end_index)
return ListToolsResult(
tools=page_tools,
nextCursor=next_cursor
)
TypeScript
import { Server } from "@modelcontextprotocol/sdk/server/index.js";
import { ListToolsResultSchema } from "@modelcontextprotocol/sdk/types.js";
const server = new Server({
name: "paginated-server",
version: "1.0.0"
});
// Simulovaný velký dataset
const ALL_TOOLS = Array.from({ length: 100 }, (_, i) => ({
name: `tool_${i}`,
description: `Tool number ${i}`,
inputSchema: { type: "object", properties: {} }
}));
const PAGE_SIZE = 10;
server.setRequestHandler(ListToolsResultSchema, async (request) => {
// Dekódovat kurzor
let startIndex = 0;
if (request.params?.cursor) {
startIndex = parseInt(request.params.cursor, 10) || 0;
}
// Získat stránku výsledků
const endIndex = Math.min(startIndex + PAGE_SIZE, ALL_TOOLS.length);
const pageTools = ALL_TOOLS.slice(startIndex, endIndex);
// Vypočítat další kurzor
const nextCursor = endIndex < ALL_TOOLS.length ? String(endIndex) : undefined;
return {
tools: pageTools,
nextCursor
};
});
Java (Spring MCP)
@Service
public class PaginatedToolService {
private static final int PAGE_SIZE = 10;
private final List<Tool> allTools;
public PaginatedToolService() {
// Inicializovat velkou datovou sadu
this.allTools = IntStream.range(0, 100)
.mapToObj(i -> new Tool("tool_" + i, "Tool number " + i, Map.of()))
.collect(Collectors.toList());
}
@McpMethod("tools/list")
public ListToolsResult listTools(@Param("cursor") String cursor) {
// Dekódovat kurzor
int startIndex = 0;
if (cursor != null && !cursor.isEmpty()) {
try {
startIndex = Integer.parseInt(cursor);
} catch (NumberFormatException e) {
startIndex = 0;
}
}
// Získat stránku výsledků
int endIndex = Math.min(startIndex + PAGE_SIZE, allTools.size());
List<Tool> pageTools = allTools.subList(startIndex, endIndex);
// Vypočítat další kurzor
String nextCursor = endIndex < allTools.size() ? String.valueOf(endIndex) : null;
return new ListToolsResult(pageTools, nextCursor);
}
}
Implementace na klientovi
Python klient
from mcp import ClientSession
async def get_all_tools(session: ClientSession) -> list:
"""Fetch all tools using pagination."""
all_tools = []
cursor = None
while True:
result = await session.list_tools(cursor=cursor)
all_tools.extend(result.tools)
if result.nextCursor is None:
break
cursor = result.nextCursor
return all_tools
# Použití
async with client_session as session:
tools = await get_all_tools(session)
print(f"Found {len(tools)} tools")
TypeScript klient
import { Client } from "@modelcontextprotocol/sdk/client/index.js";
async function getAllTools(client: Client): Promise<Tool[]> {
const allTools: Tool[] = [];
let cursor: string | undefined = undefined;
do {
const result = await client.listTools({ cursor });
allTools.push(...result.tools);
cursor = result.nextCursor;
} while (cursor);
return allTools;
}
// Použití
const tools = await getAllTools(client);
console.log(`Found ${tools.length} tools`);
Vzor Lazy Loading
Pro velmi velké datové sady načítejte stránky na vyžádání:
class PaginatedToolIterator:
"""Lazily iterate through paginated tools."""
def __init__(self, session: ClientSession):
self.session = session
self.cursor = None
self.buffer = []
self.exhausted = False
async def __anext__(self):
# Vrátit z bufferu, pokud je k dispozici
if self.buffer:
return self.buffer.pop(0)
# Zkontrolovat, zda jsme prozkoumali všechny stránky
if self.exhausted:
raise StopAsyncIteration
# Načíst další stránku
result = await self.session.list_tools(cursor=self.cursor)
self.buffer = list(result.tools)
self.cursor = result.nextCursor
if self.cursor is None:
self.exhausted = True
if not self.buffer:
raise StopAsyncIteration
return self.buffer.pop(0)
def __aiter__(self):
return self
# Použití - úsporné na paměť pro velké datové sady
async for tool in PaginatedToolIterator(session):
process_tool(tool)
Stránkování pro zdroje
Zdroje často potřebují stránkování pro adresáře nebo velké datové sady:
from mcp.server import Server
from mcp.types import Resource, ListResourcesResult
import os
app = Server("file-server")
@app.list_resources()
async def list_resources(cursor: str | None = None) -> ListResourcesResult:
"""List files in directory with pagination."""
directory = "/data/files"
all_files = sorted(os.listdir(directory))
# Dekódujte kurzor (index souboru)
start_index = int(cursor) if cursor else 0
page_size = 20
end_index = min(start_index + page_size, len(all_files))
# Vytvořte seznam zdrojů pro tuto stránku
resources = []
for filename in all_files[start_index:end_index]:
filepath = os.path.join(directory, filename)
resources.append(Resource(
uri=f"file://{filepath}",
name=filename,
mimeType="application/octet-stream"
))
# Vypočítejte další kurzor
next_cursor = str(end_index) if end_index < len(all_files) else None
return ListResourcesResult(
resources=resources,
nextCursor=next_cursor
)
Strategie návrhu kurzoru
Strategie 1: Indexové (jednoduché)
# Kurzor je pouze index
cursor = "50" # Začít u položky 50
Klady: Jednoduché, bezstavové
Zápory: Výsledky se mohou změnit, pokud jsou položky přidány/odebrány
Strategie 2: ID založené (stabilní)
# Kurzor je poslední viděné ID
cursor = "item_abc123" # Začněte po této položce
Klady: Stabilní i při změně položek
Zápory: Vyžaduje seřazená ID
Strategie 3: Zakódovaný stav (komplexní)
import base64
import json
def encode_cursor(state: dict) -> str:
return base64.b64encode(json.dumps(state).encode()).decode()
def decode_cursor(cursor: str) -> dict:
return json.loads(base64.b64decode(cursor).decode())
# Kurzor obsahuje více stavových polí
cursor = encode_cursor({
"offset": 50,
"filter": "active",
"sort": "name"
})
Klady: Může kódovat složitý stav
Zápory: Složitější, větší řetězce kurzoru
Nejlepší postupy
1. Vyberte vhodnou velikost stránky
# Zvažte velikost dat
PAGE_SIZE_SMALL_ITEMS = 100 # Jednoduché metadata
PAGE_SIZE_MEDIUM_ITEMS = 20 # Bohatší objekty
PAGE_SIZE_LARGE_ITEMS = 5 # Složitý obsah
2. Ošetřete neplatné kurzory elegantně
@app.list_tools()
async def list_tools(cursor: str | None = None) -> ListToolsResult:
try:
start_index = int(cursor) if cursor else 0
if start_index < 0 or start_index >= len(ALL_TOOLS):
start_index = 0 # Resetovat na začátek
except (ValueError, TypeError):
start_index = 0 # Neplatný kurzor, začněte znovu
# ...
3. Zahrňte celkový počet (volitelné)
return ListToolsResult(
tools=page_tools,
nextCursor=next_cursor,
# Některé implementace zahrnují celkový počet pro pokrok uživatelského rozhraní
_meta={"total": len(ALL_TOOLS)}
)
4. Testujte hraniční případy
async def test_pagination():
# Prázdná množina výsledků
result = await session.list_tools()
assert result.tools == []
assert result.nextCursor is None
# Jedna stránka
result = await session.list_tools()
assert len(result.tools) <= PAGE_SIZE
# Neplatný kurzor
result = await session.list_tools(cursor="invalid")
assert result.tools # Měla by vrátit první stránku
Časté chyby
❌ Vrácení všech výsledků a následné stránkování na klientovi
# ŠPATNĚ: Načítá všechno do paměti
@app.list_tools()
async def list_tools() -> ListToolsResult:
all_tools = load_all_tools() # 1 milion nástrojů!
return ListToolsResult(tools=all_tools)
✅ Stránkování přímo u zdroje dat
# DOBRÉ: Načítá pouze to, co je potřeba
@app.list_tools()
async def list_tools(cursor: str | None = None) -> ListToolsResult:
offset = int(cursor) if cursor else 0
tools = await db.query_tools(offset=offset, limit=PAGE_SIZE)
return ListToolsResult(tools=tools, nextCursor=...)
Co dál
Další zdroje
- Specifikace MCP - Stránkování
- Vysvětlení stránkování založeného na kurzoru
- Testy stránkování v Python SDK
Prohlášení:
Tento dokument byl přeložen pomocí AI překladatelské služby Co-op Translator. Snažíme se o přesnost, ale mějte prosím na paměti, že automatické překlady mohou obsahovat chyby nebo nepřesnosti. Originální dokument v jeho rodném jazyce by měl být považován za závazný zdroj. Pro důležité informace se doporučuje využít profesionální lidský překlad. Nejsme zodpovědní za jakékoliv nedorozumění nebo chybné interpretace vyplývající z použití tohoto překladu.