Co je GraphQL a proč vznikl
GraphQL je dotazovací jazyk pro API a runtime prostředí pro plnění těchto dotazů pomocí dat z vašich zdrojů. Vznikl jako reakce na omezení tradičního REST přístupu, zejména na problémy over-fetchingu (přenášíme více dat, než klient potřebuje) a under-fetchingu (musíme volat více endpointů, abychom sestavili požadovaný tvar dat). GraphQL dává klientovi možnost deklarativně specifikovat přesný tvar výsledku, čímž zjednodušuje vývoj front-endu, optimalizuje přenosy a urychluje iterace.
Jádro GraphQL: schéma, typy a resolvery
- Schéma definuje kontrakt mezi klientem a serverem. Popisuje typy, pole, vztahy a operace Query, Mutation a Subscription.
- Typy: skalární (
Int,Float,String,Boolean,ID), Object, Interface, Union, Enum, Input a List. Vše je striktně typované a introspektivní. - Resolver je funkce, která naplní dané pole daty. Resolver může číst z databáze, volat REST/GRPC služby, cache či jiné zdroje. Skládáním resolverů vzniká datový graf.
Operace: Query, Mutation, Subscription
- Query: čtení bez vedlejších efektů. Klient přesně určí strukturu, včetně vnořených vazeb a argumentů.
- Mutation: operace, které mění stav (vytvoření, aktualizace, smazání). Vrací stav po změně (například nově vytvořený objekt), což usnadňuje synchronizaci uživatelského rozhraní.
- Subscription: stream událostí v reálném čase (typicky přes WebSocket). Užitečné pro notifikace, dashboardy, kolaborativní scénáře.
Transport a protokol
GraphQL je agnostický vůči transportu, ale nejběžnější je HTTP s POST metodou (lze i GET pro cacheovatelné dotazy). Subscriptions typicky využívají WebSocket. V praxi se uplatňují konvence jako GraphQL-over-HTTP (využití hlaviček, status kódů) a operationName pro více operací v jednom dokumentu.
Výhody oproti REST
- Přesné dotazy: klient obdrží pouze požadovaná data (méně dat, méně požadavků).
- Jednoendpointové API: místo mnoha REST endpointů je typicky jedno
/graphql. Snižuje to režii správy verzí a dokumentace. - Silné typování + introspekce: nástroje mohou generovat typy a klientská SDK, zlepšuje se vývojářská zkušenost (DX).
- Evoluce bez verzí: rozšiřováním schématu o nová pole se vyhneme verzování URL; odstranění polí je řešeno jejich vyřazením (deprecací).
- Kompozice dat: agregace dat z více zdrojů v jednom dotazu bez orchestrace na klientovi.
Kdy REST dává stále smysl
- Jednoduché CRUD služby s jasně definovanými zdroji a silnou závislostí na HTTP cache a status kódech.
- Vysoká cacheovatelnost na CDN (GET na konkrétní URL s ETag/Last-Modified), zejména statické API.
- Integrace s legacy systémy, auditní a bezpečnostní kontroly vázané na REST rozhraní.
Modelování schématu a doménový návrh
Dobré schéma je zrcadlem domény, nikoli databázového modelu. Doporučení:
- Doménové typy místo generických „DTO“ objektů.
- Input typy pro mutace (jasně definují, co lze měnit).
- Rozhraní (Interface) a Unie (Union) pro polymorfismus a variantní struktury.
- Deprecation pro řízené odstraňování starých polí a operací.
Pagination, filtrování a třídění
Dvě dominantní strategie:
- Offset/limit: snadné, ale náchylné ke změnám v datech a snižování výkonu na velkých offsetech.
- Cursor-based (Relay): stabilnější, výkonnější, využívá edges, node, cursor, pageInfo.
Filtrace a třídění se modelují jako argumenty polí; složitější dotazy mohou používat input objekty.
N+1 problém a DataLoader
Protože resolvery běží po jednotlivých polích, hrozí mnoho malých dotazů do databáze (N+1 problém). Řešení:
- Batching a caching na úrovni požadavku (například DataLoader) pro sloučení několika
findByIddo jednohofindByIds. - Projekce a joiny na databázové vrstvě, případně materializované pohledy.
Cache a výkon v GraphQL
- HTTP cache je složitější než u REST.
- Dotazy nejsou vázány na URL – proto se používají Persisted Queries a Automatic Persisted Queries (APQ): klient posílá hash dotazu, který CDN nebo API Gateway může cacheovat.
- Response caching na úrovni polí či resolverů (krátká TTL, stale-while-revalidate), případně fragment-level caching na klientovi.
- CDN na edge s pravidly podle operationName a proměnných.
Bezpečnost: limity, autorizace, ochrana schématu
- Rate limiting a throttling na vrstvě gateway; omezování hloubky a komplexity dotazu (maximální hloubka dotazu, skórování polí).
- Autorizace: kontrola na úrovni polí (field-level) pomocí direktiv a middleware, pravidla na úrovni záznamů (record-level) a přístup založený na atributech (attribute-based access).
- Zakázání nebo omezení introspekce v produkčním prostředí (nebo jen pro privilegované klienty) pro ochranu před introspection leaks.
- Validace vstupů a allow-list dotazů (persisted queries) pro veřejná API.
Chyby a návratové kódy
GraphQL vždy vrací HTTP 200 u syntakticky validní odpovědi; chyby jsou vráceny v poli errors s informacemi o path a extensions. Doporučuje se standardizovat error codes v extensions.code a mapovat doménové chyby jako FORBIDDEN, NOT_FOUND, CONFLICT.
Verzování vs. evoluce schématu
- „No versioning“: přidáváme nová pole, stará označíme jako
@deprecateda po stanovené době odstraníme. - Breaking changes se plánují s dostatečným předstihem a komunikací; využívají se automatické kontroly kompatibility (schema diff v CI).
Federace a modulární monorepo schémat
Ve větších organizacích je praktické rozdělit schéma mezi doménové týmy. Federace umožňuje publikovat dílčí subgraphy a skládat supergraph na gateway (entity klíče, resolvery přes hranice týmů). Alternativou je schema stitching nebo BFF (Backend for Frontend) pro každou aplikaci zvlášť.
Nástroje a ekosystém
- Server: Apollo Server, GraphQL Yoga, Mercurius (Fastify), Helix; pro jiné jazyky graphql-java, Sangria (Scala), graphql-go, Strawberry/Graphene (Python).
- Klient: Apollo Client, Relay, urql; generování typů (GraphQL Code Generator), fragmenty, cache normalizace.
- Vývoj: GraphiQL/GraphQL Playground, Explorer, lint a validace schémat, mocking resolverů.
Testování a kvalita
- Jednotkové testy resolverů a datových přístupů.
- Integrační testy s lokálním serverem a seedovanými daty.
- Contract testing vůči schématu (operation registry, kontrola breaking changes v CI/CD).
Observabilita a monitoring
- Tracing na úrovni resolveru (OpenTelemetry) – latence polí, detekce N+1 problémů, tepelné mapy nejnáročnějších dotazů.
- Analýza využití: kdo volá jaké operace a fragmenty (pomáhá s deprecací a úklidem schématu).
- Operation registry a allow-list pro zabezpečení a optimalizaci výkonu na CDN.
Integrace s REST a existujícími systémy
GraphQL často funguje jako orchestrace nad REST/GRPC – nevyměňuje všechny systémy, ale sjednocuje přístup k datům. Výhodou je evoluční migrace: nejprve zabalíme existující endpointy pomocí resolverů, později refaktorujeme zdroje.
Best practices pro produkci
- Governance schématu: vlastnictví domén, review proces, automatické kontroly kompatibility.
- Bezpečnost a limity: omezení hloubky a komplexity dotazu, modelování nákladů dotazu, rate limiting, Row-Level Security na úrovni databáze.
- Výkon: DataLoader, caching, APQ, jemnozrnná autorizace (nikoli v databázových cyklech), vhodné indexy.
- Dokumentace: descriptions ve schématu, generované reference, interaktivní playgroundy s ukázkami.
Příklady typických use-case
- Komplexní front-endy (mobilní aplikace/SPA), kde jeden dotaz sestavuje dashboard z více domén.
- Produktové katalogy s bohatými vazbami (produkty, varianty, recenze, sklad, ceny).
- Multi-tenant SaaS se silnou personalizací a řízením přístupů na úrovni polí.
Antivzory a časté chyby
- „Jedna mega-mutace“: obtížná autorizace a audit; preferujte menší, konzistentní mutace.
- Expozice databázového modelu: schéma by nemělo rovnat se tabulka; skrývejte technické detaily.
- Ignorování N+1 problému: vždy plánujte DataLoader/batching.
- Chybějící limity na dotazy: otevřená cesta k DoS útokům – nastavte omezení hloubky a komplexity.
GraphQL vs. REST: shrnutí
GraphQL brilantně řeší flexibilitu klienta, agregaci dat a rychlou evoluci kontraktu. REST exceluje v jednoduchosti, HTTP cache a jasném mapování zdrojů. V praxi se přístupy často kombinují: veřejné read-heavy API zůstává REST/HTTP cache friendly, interní BFF nebo federované API jsou GraphQL.
Závěr
GraphQL není jen „rychlejší REST“, ale jiný model interakce – klient si specifikuje přesná data a server se stará o jejich složení. Díky silnému typování, introspekci, federaci a bohatému ekosystému výrazně zlepšuje produktivitu týmů a uživatelskou zkušenost. Úspěch však stojí na promyšleném návrhu schématu, sledování výkonu, bezpečnostních limitech a disciplinované správě změn.