Disable Error Propagation Directive
At a glanceβ
- Identifier: DisableErrorPropagationDirective
- Stage: RFC0: Strawman
- Champion: -
- PR: -
- Related:
- #1050 (Directive proposal for opting out of null bubbling)
Timelineβ
- RFC document created on 2025-03-19 by Alex Reilly
RFC: Disable Error Propagation Directive
Proposed by: Martin Bonnin
See also: Original proposal/spec edits by Benjie
Implementation PR: https://github.com/graphql/graphql-js/pull/4348
This RFC proposes adding a new directive
@disableErrorPropagation
that allows clients to disable error propagation for specific operations in their GraphQL queries.π Problem Statementβ
In GraphQL, nullability serves two distinct purposes:
- Semantic null: Indicating that a field can have a legitimate "null" value (e.g., a user without an avatar)
- Error handling: Allowing errors to propagate up through nullable parent fields
This coupling of nullability and errors makes it difficult for clients to distinguish between semantic nulls and error states by looking at the schema. When a field resolver throws an error and the field is non-nullable, the error propagates up through parent fields until it reaches a nullable field, potentially nullifying a large portion of the response.
While this behavior helps maintain data consistency guarantees, there are cases where clients may want more granular control over error propagation, particularly when partial data is preferable to no data.
Current Behaviorβ
Consider this schema:
type User {
id: ID!
name: String
posts: [Post!]!
optionalPosts: [Post!]
}
type Post {
id: ID!
title: String!
content: String
}If a resolver for
Post.title
throws an error:
- The error is added to the
errors
array in the response- Since
title
is non-nullable (String!
), its parentPost
object must be null- Since the array items are non-nullable (
[Post!]
), the entireposts
array must be null- Since
posts
is non-nullable ([Post!]!
), its parentUser
object must be null- This continues up the response tree until reaching a nullable field
For the
optionalPosts
field, which is nullable ([Post!]
), the propagation would stop at that field, setting it to null.This behavior ensures type safety but can lead to situations where a single error in a deeply nested non-nullable field causes a large portion of the response to be nullified, even when the remaining data might still be useful to the client.
Use Casesβ
Normalized Cache Protection: When clients like Relay maintain a normalized cache, error propagation can cause cache corruption. For example, if two different queries fetch the same entity but one query errors on a non-nullable field, the error propagation can cause valid data from the other query to be nullified in the cache. Disabling error propagation allows clients to preserve valid data in their normalized caches while still handling errors appropriately.
Partial Data Acceptance: In some applications, receiving partial data with errors is preferable to receiving null. For example, in a feed-style application, if one post fails to load, the client might still want to display the other posts.
Fine-grained Error Control: Clients may want to specify different error handling behaviors for different parts of their queries based on their application requirements.
β RFC Goalsβ
- Provide a way for clients to disable error propagation for specific operations
- Help uncouple nullability and error handling in GraphQL
- Support the transition to more semantically accurate nullability in schemas
- Maintain backward compatibility with existing GraphQL behavior
- Keep the implementation simple and focused
π« RFC Non-goalsβ
- This is not intended to be a general-purpose error handling solution
π§βπ» Proposed Solutionβ
Add a new directive
@disableErrorPropagation
that can be applied to operations in executable documents:directive @disableErrorPropagation on QUERY | MUTATION | SUBSCRIPTION
query GetUserPosts @disableErrorPropagation {
user {
id
name
posts {
id
title
content
}
}
}When this directive is present on an operation:
- Errors thrown during execution will still be added to the
errors
array- The errors will not cause nullability violations to propagate up through parent fields
Exampleβ
Given these types:
type User {
id: ID!
name: String
posts: [Post!]!
}
type Post {
id: ID!
title: String!
content: String
}And this query:
query GetUserPosts @disableErrorPropagation {
user {
id
name
posts {
id
title
content
}
}
}If the
title
resolver for one of the posts throws an error:Current behavior (without directive):
{
"data": null,
"errors": [
{
"message": "Failed to load title",
"path": ["user", "posts", 0, "title"]
}
]
}With @disableErrorPropagation:
{
"data": {
"user": {
"id": "123",
"name": "Alice",
"posts": [
{
"id": "post1",
"title": null,
"content": "Some content"
}
]
}
},
"errors": [
{
"message": "Failed to load title",
"path": ["user", "posts", 0, "title"]
}
]
}π οΈ Implementation Considerationsβ
The implementation requires changes to the execution algorithm in graphql-js:
- During operation execution, check if the operation has the
@disableErrorPropagation
directive- If present, modify error handling to prevent propagation while still collecting errors
- Return field values as-is, even if errors occurred
While this proposal focuses on a simple boolean directive, future extensions might consider additional error behaviors:
PROPAGATE
: Current default behavior (errors propagate up)NULL
: Replace errored positions with nullABORT
: Abort the entire request on any errorThese additional behaviors are not part of this proposal but may be considered in future iterations.
πΊοΈ Migration Pathβ
For client authors wishing to adopt this feature:
- Ensure your GraphQL server implementation supports
@disableErrorPropagation
. Clients can check for directive support by looking for the existence of the directive on introspection.- Update client code to handle both nulls and errors appropriately.
- Add the directive to operations where you want to prevent error propagation. Many clients, especially those with normalized caches, will wish to apply
@disableErrorPropagation
to all operations.For schema authors wishing to adopt this feature:
- Update servers to a version that has support for
@disableErrorPropagation
Once upgraded, schema authors may feel more comfortable using non-nullable fields. We'll want to keep an eye on that and how it affects developers using older or simpler clients.
π€ Alternatives Consideredβ
Configurable Error Behaviorβ
Instead of a simple boolean directive, we could provide an enum of error behaviors:
enum ErrorBehavior {
"""
Non-nullable positions that error cause the error to propagate to the nearest nullable
ancestor position. The error is added to the "errors" list.
"""
PROPAGATE
"""
Positions that error are replaced with a `null` and an error is added to the "errors"
list.
"""
NULL
"""
If any error occurs, abort the entire request and just return the error in the "errors"
list. (No partial success.)
"""
ABORT
}
directive @errorBehavior(behavior: ErrorBehavior!) on QUERY | MUTATION | SUBSCRIPTIONThe simple boolean
@disableErrorPropagation
directive was chosen as it:
- Addresses the most common use case (wanting partial data)
- Maintains GraphQL's existing error representation
- Is simple to implement and understand
- Leaves room for future extensions if more sophisticated error handling patterns prove necessary
π Conclusionβ
The
@disableErrorPropagation
directive provides a simple but powerful way for clients to control error propagation behavior. While it doesn't solve all error handling challenges, it represents an important step toward uncoupling nullability and errors in GraphQL. This allows for more precise schema design while maintaining backwards compatibility for existing clients.