Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
295 changes: 295 additions & 0 deletions skills/relay-best-practices/SKILL.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,295 @@
---
name: relay-best-practices
description: >-
Best practices for writing idiomatic Relay code. ALWAYS use this skill when
writing or modifying React components that use Relay for data fetching. Covers
fragments, queries, mutations, pagination, and common anti-patterns. Use when
you see `useFragment`, `useLazyLoadQuery`, `usePreloadedQuery`, `useMutation`,
`usePaginationFragment`, `graphql` template literals, `react-relay` imports,
or `__generated__/*.graphql` files. Also use when asked to explain Relay
concepts, debug Relay issues, or review Relay code.
---

# Relay 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 in `package.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.

```tsx
// 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.

```tsx
// 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.

```tsx
// 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:

```graphql
// 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 |
8 changes: 8 additions & 0 deletions website/docs/getting-started/quick-start.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,14 @@ npm install --save-dev babel-plugin-relay graphql relay-compiler
npm install --save-dev @types/relay-runtime @types/react-relay
```

:::tip
If you're using an AI coding assistant, install the Relay best practices skill to make your agent a Relay expert:

```bash
npx skills install facebook/relay
```
:::

## Configure Vite to use Relay

Relay uses a [Babel plugin](./babel-plugin.mdx) to insert code generated by the Relay compiler into your bundle. We can enable the Relay Babel plugin we installed earlier by configuring the React Vite plugin.
Expand Down
Loading