graphql-schema▌
apollographql/skills · updated Apr 8, 2026
Industry best practices guide for designing intuitive, performant, and maintainable GraphQL schemas.
- ›Covers core design principles including client-centric type organization, explicit nullability patterns, and backwards-compatible evolution strategies
- ›Provides reference documentation on types, naming conventions, cursor-based pagination, error modeling, and security considerations
- ›Includes practical patterns for interfaces, unions, input types, mutations, and ID strategies with code
GraphQL Schema Design Guide
This guide covers best practices for designing GraphQL schemas that are intuitive, performant, and maintainable. Schema design is primarily a server-side concern that directly impacts API usability.
Schema Design Principles
1. Design for Client Needs
- Think about what queries clients will write
- Organize types around use cases, not database tables
- Expose capabilities, not implementation details
2. Be Explicit
- Use clear, descriptive names
- Make nullability intentional
- Document with descriptions
3. Design for Evolution
- Plan for backwards compatibility
- Use deprecation before removal
- Avoid breaking changes
Quick Reference
Type Definition Syntax
"""
A user in the system.
"""
type User {
id: ID!
email: String!
name: String
posts(first: Int = 10, after: String): PostConnection!
createdAt: DateTime!
}
Nullability Rules
| Pattern | Meaning |
|---|---|
| String | Nullable - may be null |
| String! | Non-null - always has value |
| [String] | Nullable list, nullable items |
| [String!] | Nullable list, non-null items |
| [String]! | Non-null list, nullable items |
| [String!]! | Non-null list, non-null items |
Best Practice: Use [Type!]! for lists - empty list over null, no null items.
Input vs Output Types
# Output type - what clients receive
type User {
id: ID!
email: String!
createdAt: DateTime!
}
# Input type - what clients send
input CreateUserInput {
email: String!
name: String
}
# Mutation using input type
type Mutation {
createUser(input: CreateUserInput!): User!
}
Interface Pattern
interface Node {
id: ID!
}
type User implements Node {
id: ID!
email: String!
}
type Post implements Node {
id: ID!
title: String!
}
Union Pattern
union SearchResult = User | Post | Comment
type Query {
search(query: String!): [SearchResult!]!
}
Reference Files
Detailed documentation for specific topics:
- Types - Type design patterns, interfaces, unions, and custom scalars
- Naming - Naming conventions for types, fields, and arguments
- Pagination - Connection pattern and cursor-based pagination
- Errors - Error modeling and result types
- Security - Security best practices for schema design
Key Rules
Type Design
- Define types based on domain concepts, not data storage
- Use interfaces for shared fields across types
- Use unions for mutually exclusive types
- Keep types focused (single responsibility)
- Avoid deep nesting - flatten when possible
Field Design
- Fields should be named from client's perspective
- Return the most specific type possible
- Make expensive fields explicit (consider arguments)
- Use arguments for filtering, sorting, pagination
Mutation Design
- Use single input argument pattern:
mutation(input: InputType!) - Return affected objects in mutation responses
- Model mutations around business operations, not CRUD
- Consider returning a union of success/error types
ID Strategy
- Use globally unique IDs when possible
- Implement
Nodeinterface for refetchability - Base64-encode compound IDs if needed
Ground Rules
- ALWAYS add descriptions to types and fields
- ALWAYS use non-null (!) for fields that cannot be null
- ALWAYS use [Type!]! pattern for lists
- NEVER expose database internals in schema
- NEVER break backwards compatibility without deprecation
- PREFER dedicated input types over many arguments
- PREFER enums over arbitrary strings for fixed values
- USE
IDtype for identifiers, notStringorInt - USE custom scalars for domain-specific values (DateTime, Email, URL)
Ratings
4.5★★★★★38 reviews- ★★★★★Pratham Ware· Dec 28, 2024
Solid pick for teams standardizing on skills: graphql-schema is focused, and the summary matches what you get after install.
- ★★★★★Isabella Garcia· Dec 28, 2024
graphql-schema has been reliable in day-to-day use. Documentation quality is above average for community skills.
- ★★★★★Yash Thakker· Nov 19, 2024
We added graphql-schema from the explainx registry; install was straightforward and the SKILL.md answered most questions upfront.
- ★★★★★Lucas Choi· Nov 19, 2024
graphql-schema fits our agent workflows well — practical, well scoped, and easy to wire into existing repos.
- ★★★★★Dhruvi Jain· Oct 10, 2024
graphql-schema fits our agent workflows well — practical, well scoped, and easy to wire into existing repos.
- ★★★★★Amina Anderson· Oct 10, 2024
We added graphql-schema from the explainx registry; install was straightforward and the SKILL.md answered most questions upfront.
- ★★★★★Isabella Yang· Sep 5, 2024
Useful defaults in graphql-schema — fewer surprises than typical one-off scripts, and it plays nicely with `npx skills` flows.
- ★★★★★Oshnikdeep· Sep 1, 2024
Registry listing for graphql-schema matched our evaluation — installs cleanly and behaves as described in the markdown.
- ★★★★★Tariq Khan· Sep 1, 2024
graphql-schema reduced setup friction for our internal harness; good balance of opinion and flexibility.
- ★★★★★Yuki Brown· Sep 1, 2024
graphql-schema has been reliable in day-to-day use. Documentation quality is above average for community skills.
showing 1-10 of 38