relay-best-practices作成者: facebook
Relay is a GraphQL client for React that enforces colocated, composable, and type-safe data fetching. Its core insight is that each component should declare exactly what data it needs via GraphQL fragments, and Relay handles the rest — fetching, caching, consistency, and updates.
npx skills add https://github.com/facebook/relay --skill relay-best-practicesRelay Best Practices
Relay is a GraphQL client for React that enforces colocated, composable, and type-safe data fetching. Its core insight is that each component should declare exactly what data it needs via GraphQL fragments, and Relay handles the rest — fetching, caching, consistency, and updates.
This skill provides opinionated guidance on which patterns to prefer. For detailed API documentation, read the relevant page from the doc map below.
Documentation
Relay ships LLM-friendly docs in node_modules/relay-runtime/llm-docs/
(available after v20.1.1). For older versions, fetch the same files from
https://raw.githubusercontent.com/facebook/relay/main/website/docs/.
Paths below are relative to this directory (<llm-docs>/). Read the relevant
page before writing Relay code. Key docs:
| Topic | Path |
|---|---|
| Core concepts & philosophy | principles-and-architecture/thinking-in-relay.mdx |
| Fragments | guided-tour/rendering/fragments.mdx |
| Queries | guided-tour/rendering/queries.mdx |
| Mutations | guided-tour/updating-data/graphql-mutations.mdx |
| Pagination | guided-tour/list-data/pagination.mdx |
| Refetching | guided-tour/refetching/refetching-queries-with-different-data.mdx |
useFragment | api-reference/hooks/use-fragment.mdx |
usePreloadedQuery | api-reference/hooks/use-preloaded-query.mdx |
useQueryLoader / loadQuery | api-reference/hooks/load-query.mdx |
useMutation | api-reference/hooks/use-mutation.mdx |
usePaginationFragment | api-reference/hooks/use-pagination-fragment.mdx |
@throwOnFieldError | guides/throw-on-field-error-directive.mdx |
@catch directive | guides/catch-directive.mdx |
| Semantic nullability | guides/semantic-nullability.mdx |
| Relay Resolvers | guides/relay-resolvers/introduction.mdx |
| Testing | guides/testing-relay-components.mdx |
| Compiler setup | getting-started/compiler.mdx |
| Compiler configuration | getting-started/compiler-config.mdx |
| Lint rules (ESLint plugin) | getting-started/lint-rules.mdx |
Core Philosophy
These principles are the foundation of every decision below. When in doubt, refer back to them.
- Co-location: Each component declares its own data requirements via a GraphQL fragment, right next to the rendering code. Data needs travel with the component, not separately.
- Data masking: A component can only access the fields it explicitly selected in its own fragment. Parents cannot see child fragment data, and vice versa. This prevents implicit coupling between components.
- Composition: Fragments compose into parent fragments and ultimately into queries, forming a tree that mirrors the component tree. The compiler flattens this into a single network request per query.
- Render-as-you-fetch: Start fetching data before the component that needs it renders. This avoids sequential request waterfalls.
- Normalized store: Relay maintains a flat, ID-keyed cache. When a mutation returns updated data, every component reading that data re-renders automatically. You do not need to manually propagate changes.
Compiler Workflow
Relay uses an ahead-of-time compiler that reads graphql tagged template
literals in your code and generates runtime artifacts and TypeScript/Flow types.
Finding the config
The compiler looks for its config in these locations (checked in order):
relay.config.{json,js,mjs,ts}in the project root- A
"relay"key inpackage.json
See <llm-docs>/getting-started/compiler-config.mdx for the full config schema.
You can also emit a JSON Schema for the config by running
npx relay-compiler config-json-schema.
Running the compiler
Run npx relay-compiler after any change to the contents of a graphql tagged
template literal or the docblock of a Relay Resolver. Some projects add this as
a script in package.json (e.g., yarn relay). The compiler also supports watch mode (--watch), but avoid using it in
non-interactive contexts since the process never exits.
Generated files go into __generated__/ directories next to the source files.
Never edit these files — they are overwritten on every compiler run. If you see
type errors about missing generated types, run the compiler first — the
types are likely just out of date.
Lint Rules
Relay's ESLint plugin (eslint-plugin-relay) is a key part of the developer
experience. Two rules are especially important:
relay/unused-fields— detects GraphQL fields that are selected in a fragment but never read in the component. This prevents the "append-only query" problem where fragments accumulate unused fields over time, fetching data no component actually needs.relay/no-future-added-value— prevents explicitly handling the"%future added value"enum placeholder that Relay inserts to ensure you handle the possibility of new enum variants being added by the server.
See <llm-docs>/getting-started/lint-rules.mdx for installation and configuration.
Decision Rules
Queries: Never use useLazyLoadQuery
Use usePreloadedQuery + useQueryLoader (or loadQuery). Start the fetch in
an event handler, route transition, or during app initialization — before the
component renders. useLazyLoadQuery does not start fetching until render,
creating waterfalls. See <llm-docs>/guided-tour/rendering/queries.mdx for the full pattern.
Data flow: Always use fragments
Every component that displays server data should declare a fragment and receive
a fragment reference (the $key type) as a prop. The parent spreads the
child's fragment in its own query or fragment and passes the result down. See
the "Maintain fragment co-location" anti-pattern below for an example.
Mutations: Spread fragments into responses
Spread the consuming component's fragment into the mutation response rather than selecting fields individually. This keeps them in sync automatically. See the anti-pattern example below.
Error handling: Use @throwOnFieldError and @catch
The recommended approach for handling field errors and nullability is to add
@throwOnFieldError to your fragment or query. This causes Relay to throw a
JavaScript exception if a field error is encountered, which can be caught by a
React error boundary. It also enables non-null types for @semanticNonNull
fields, eliminating unnecessary null checks. Note that this pattern depends on
React error boundaries being configured in your application — proceed with
caution if error boundaries are not set up robustly.
For fields where you want to handle errors locally instead of throwing, use
@catch to receive errors inline as { ok: true, value: T } | { ok: false, errors: [...] }.
@required is also available for declaring that specific fields must be
non-null, but @throwOnFieldError + @catch is the preferred pattern for new
code.
See <llm-docs>/guides/throw-on-field-error-directive.mdx, <llm-docs>/guides/catch-directive.mdx,
and <llm-docs>/guides/semantic-nullability.mdx.
Pagination: Use the three-directive pattern
Always use @argumentDefinitions (for cursor/count variables),
@refetchable (to auto-generate the pagination query), and @connection (to
identify the connection for store management) together. Never write manual
pagination queries. See <llm-docs>/guided-tour/list-data/pagination.mdx.
Client state: Prefer Relay Resolvers
When multiple components need to read client-side data, use Relay Resolvers to
define client-only fields on the GraphQL schema rather than prop-drilling or
React context. This gives client state the same composability and caching
guarantees as server data. Use useClientQuery for queries that read only
resolver-defined fields.
See <llm-docs>/guides/relay-resolvers/introduction.mdx for how to define resolvers.
Critical Anti-Patterns
Never copy Relay data into React state
This is the single most important rule. Do not read data from useFragment and
copy it into useState, and do not update that state manually in mutation
onCompleted callbacks.
// WRONG: Copying Relay data into React state
function UserProfile({userKey}) {
const data = useFragment(UserProfileFragment, userKey);
const [name, setName] = useState(data.name); // broken
const [commit] = useMutation(UpdateNameMutation);
const handleSave = (newName) => {
commit({
variables: {name: newName},
onCompleted: (response) => {
setName(response.updateName.user.name); // broken
},
});
};
return <span>{name}</span>;
}
Why this is wrong: Relay's normalized store is the single source of truth. When
a mutation returns updated data with a matching id, Relay automatically
updates every component reading that data via useFragment. By copying into
useState, you create a second source of truth that Relay cannot update. The
component will show stale data whenever the record is updated by another
mutation, subscription, or refetch elsewhere in the app.
// CORRECT: Read directly from the fragment
function UserProfile({userKey}) {
const data = useFragment(UserProfileFragment, userKey);
const [commit, isInFlight] = useMutation(UpdateNameMutation);
const handleSave = (newName) => {
commit({variables: {name: newName}});
// No onCompleted needed — Relay updates the store automatically,
// and useFragment re-renders this component with the new data.
};
return <span>{data.name}</span>;
}
Similarly, do not store a fragment key (the $key prop) in React state. Relay
garbage collects data that is no longer retained by a mounted query component —
if the component that originally fetched the data unmounts, a stashed key may
point to data that is no longer in the store.
Maintain fragment co-location
Do not fetch all data in a parent's query and pass raw data objects as props to
children. This defeats data masking and creates tight coupling — adding a field
to a child component requires editing the parent's query. Note that the
relay/unused-fields lint rule will flag fields selected in the parent that
are only used by children — this is a good signal that you need to extract a
fragment.
// WRONG: Parent fetches everything, passes raw data
function Parent({queryRef}) {
const data = usePreloadedQuery(graphql`
query ParentQuery {
user {
name
email
avatarUrl
}
}
`, queryRef);
return <UserCard name={data.user.name} avatarUrl={data.user.avatarUrl} />;
}
// CORRECT: Child declares its own fragment
function Parent({queryRef}) {
const data = usePreloadedQuery(graphql`
query ParentQuery {
user {
...UserCard_user
}
}
`, queryRef);
return <UserCard user={data.user} />;
}
Spread fragments into mutation responses
Do not select fields individually in both a fragment and a mutation response — they will drift out of sync. Spread the fragment instead:
// WRONG
mutation UpdateUserMutation($input: UpdateUserInput!) {
updateUser(input: $input) {
user { id, name, email, avatarUrl }
}
}
// CORRECT
mutation UpdateUserMutation($input: UpdateUserInput!) {
updateUser(input: $input) {
user { ...UserCard_user }
}
}
Naming Conventions
| Element | Convention | Example |
|---|---|---|
| Fragment | ComponentName_propName | UserCard_user |
| Query | ComponentNameQuery | HomePageQuery |
| Mutation | ComponentNameMutation | LikeButtonMutation |
| Generated files | __generated__/*.graphql | Never edit these |