Semantic Nullability
At a glanceβ
- Identifier: SemanticNullability
- Stage: RFC0: Strawman
- Champion: -
- PR: -
- Related:
Timelineβ
- RFC document updated on 2025-03-06 by Benjie
- RFC document updated on 2025-02-22 by Benjie
- RFC document updated on 2025-02-22 by Benjie
- RFC document updated on 2025-02-20 by Benjie
- RFC document updated on 2025-02-20 by Benjie
- RFC document updated on 2025-02-20 by Martin Bonnin
- RFC document updated on 2025-02-17 by Benjie
- RFC document updated on 2025-02-17 by Alex Reilly
- RFC document updated on 2025-02-17 by Martin Bonnin
- RFC document updated on 2025-02-17 by Benjie
- RFC document updated on 2025-02-17 by Benjie
- RFC document updated on 2025-02-16 by Benjie
- RFC document updated on 2025-02-16 by Benjie
- RFC document created on 2025-02-15 by Benjie
RFC: Semantic Nullability
π Problem History
One of GraphQL's early decisions was to allow "partial success"; this was a critical feature for Facebook - if one part of their backend infrastructure became degraded they wouldn't want to just render an error page, instead they wanted to serve the user a page with as much working data as they could.
Error propagationβ
To accomplish this, if an error occured within a resolver, the resolver's value would be replaced with a
null
, and an error would be added to theerrors
array in the response. GraphQL thus adopted the non-traditional stance of all types being "nullable by default" (since an error could happen anywhere at any time for any reason).However, null-checking is exhausting and in some positions errors are extremely unlikely (and null is not an expected value according to the business logic) so GraphQL allowed a position to be marked non-nullable by following the type with a
!
marker - this would guarantee that that position in the data could not contain anull
.What if a non-null field were to throw an error, or incorrectly return
null
, then? To solve that apparent contradiction, GraphQL introduced the "error propagation" behavior (also known colloquially as "null bubbling") - when anull
(from an error or otherwise) occurs in a non-nullable position, the parent position (either a field or a list item) is madenull
instead. This behavior would repeat if the parent position was also non-nullable, and this could propagate (or "bubble") all the way up to the root of the operation if everything in the path is non-nullable.Thus the
!
non-null marker has also been known as "kills parent on exception" due to this destructive error propagation behavior.This solved the issue, and meant that GraphQL's nullability promises were still honoured; but it wasn't without complications.
Complication 1: partial successβ
We want to be resilient to systems failing; but errors that occur in non-nullable positions cascade to surrounding parts of the query, making less and less data available to be rendered.
This seems contrary to our "partial success" aim, but it's easy to solve - we just make sure that the positions where we expect errors to occur are nullable so that errors don't propagate further. Clients now need to handle
null
in these positions.Complication 2: nullable epidemicβ
Almost any field in your GraphQL schema could raise an error - errors might not only be caused by backend services becoming unavailable or responding in unexpected ways; they can also be caused by simple programming errors in your business logic, data consistency errors (e.g. expecting a boolean but receiving a float), or any other cause.
Since we don't want to "blow up" the entire response if any such issue occurred, we've moved to strongly encourage nullable usage throughout a schema, only adding the non-nullable
!
marker to positions where we're truly sure that field is extremely unlikely to error. This has the effect of meaning that developers consuming the GraphQL API have to handle potential nulls in more positions than they would expect, making for additional work.Complication 3: normalized cachingβ
Many modern GraphQL clients use a "normalized" cache, such that updates pulled down from the API in one query can automatically update all the previously rendered data across the application. This helps ensure consistency for users, and is a powerful feature.
However, if an error occurs in a non-nullable position, it's no longer safe to store the data to the normalized cache. Again, the solution is to make more of your schema nullable.
The Nullability Working Groupβ
At first, we thought the solution to this was to give clients control over the nullability of a response, so we set up the Client-Controlled Nullability (CCN) Working Group. Later, we renamed the working group to the Nullability WG to show that it encompassed all potential solutions to this problem.
Client-controlled nullabilityβ
The first Nullability WG proposal came from a collaboration between Yelp and Netflix, with contributions from GraphQL WG regulars Alex Reilly, Mark Larah, and Stephen Spalding among others. They proposed we could adorn the queries we issue to the server with sigils indicating our desired nullability overrides for the given fields - client-controlled nullability.
A
?
would be added to fields where we don't mind if they're null, but we definitely want errors to stop there; and add a!
to fields where we definitely don't want a null to occur (whether or not there is an error). This would give consumers control over where errors/nulls were handled.However, after much exploration of the topic over years we found numerous issues that traded one set of concerns for another. We kept iterating whilst we looked for a solution to these tradeoffs.
True nullability schemaβ
Jordan Eldredge proposed that making fields nullable to handle error propagation was hiding the "true" nullability of the data. Instead, he suggested, we should have the schema represent the true nullability, and put the responsibility on clients to use the
?
CCN operator to handle errors in the relevant places.However, this would mean that clients such as Relay would want to add
?
in every position, causing an "explosion" of question marks, because really what Relay desired was to disable error propagation entirely.Disabling error propagationβ
It became clear that disabling error propagation was desired by advanced GraphQL clients and vital for ensuring that normalized caches were as useful as possible and that we could live up to the promise of GraphQL's partial success without compromise. But that was only part of the problem - the other part was that we want to see the "true" nullability of fields, the nullability if we were to exclude errors.
Note: this RFC assumes that clients may opt out of error propagation via some mechanism that is outside the scope of this RFC and will be handled in a separate RFC (e.g. via a directive such as
@noErrorPropagation
or@behavior(onError: NULL)
; or via a request-level flag) - in general the specific mechanism is unimportant and thus solutions are not expected to comment on it unless the choice is significant to the proposal.Semantic nullabilityβ
We realised that if we were to do this, we would need two schemas: one for when null bubbling is disabled, where the true nullability of fields could be represented; and one for the traditional error handling behavior, where nullability would need to factor in that errors can occur.
However, maintaining two nearly-identical-except-for-nullability schemas is a chore... and it felt like it was solveable if we could teach GraphQL to understand this need... What we ultimately realised is that GraphQL is missing a type.
Ignoring errors, if we look at our business logic we can determine if a field is either semantically nullable (it's meaningful for this field to be null - for example an
Animal
might not have anowner
currently) or semantically non-nullable (this field will never be null - for example everyPost
must belong to atopic
). However GraphQL muddied the waters here by factoring errors into the mix... "what if the "topics" service went down?" it would ask; "we might want to render the post!" And thus, we would makePost.topic
nullable, even though we know it should always exist, because we don't want error propagation to destroy the entire response.So we actually have three types:
Value Error null Semantically nullable β β β Semantically non-nullable β β β Strictly non-nullable β β β We could already express a position that could never error and never be null (we called this non-nullable, e.g.
Int!
), and we could express a position that could be null or have an error (we called this nullable, e.g.Int
), but what we lacked was the ability to say "this position can be null, but that will only happen if an error has occurred" - a "null only on error" or "semantically non-null" type.π Problem Statement
GraphQL needs to be able to represent semantically nullable and semantically non-nullable types as such when error propagation is disabled.
π Solution Criteria
This section sketches out the potential goals that a solution might attempt to fulfill. These goals will be evaluated with the GraphQL Spec Guiding Principles in mind:
- Backwards compatibility
- Performance is a feature
- Favor no change
- Enable new capabilities motivated by real use cases
- Simplicity and consistency over expressiveness and terseness
- Preserve option value
- Understandability is just as important as correctness
Each criteria is identified with a
Letter
so they can be referenced in the rest of the document. New criteria must be added to the end of the list.Solutions are evaluated and scored using a simple 3 part scale. A solution may have multiple evaluations based on variations present in the solution.
- β Pass. The solution clearly meets the criteria
- β οΈ Warning. The solution doesn't clearly meet or fail the criteria, or there is an important caveat to passing the criteria
- π« Fail. The solution clearly fails the criteria
- β The criteria hasn't been evaluated yet
Passing or failing a specific criteria is NOT the final word. Both the Criteria and the Solutions are up for debate.
Criteria have been given a "score" according to their relative importance in solving the problem laid out in this RFC while adhering to the GraphQL Spec Guiding Principles. The scores are:
- π₯ Gold - A must-have
- π₯ Silver - A nice-to-have
- π₯ Bronze - Not necessary
π― A. GraphQL should be able to indicate which nullable fields should become non-nullable when error propagation is disabledβ
The promise of this RFC - the reflection of the semantic nullability of the fields without compromising requests with error propagation enabled via the differentiation of a "null if and only if an error occurs" type.
With error propagation enabled (the traditional GraphQL behavior), it's recommended that fields are marked nullable if errors may happen there, even if the underlying value is semantically non-nullable. If we allow error-handling clients to disable error propagation, then these traditionally nullable positions can be marked (semantically) non-nullable in that mode, since with error propagation disabled the selection sets are no longer destroyed.
Note: Traditional non-nullable types will effectively become semantically non-nullable when error propagation is disabled no matter which solution is chosen, so this criteria is only concerned with traditionally nullable types.
1 2 3 4 5 6 β β β β π«π β Criteria score: π₯
π― B. Existing executable documents should retain validity and meaningβ
Users should be able to adopt semantic nullability into an existing schema, and when doing so all existing operations should remain valid, and should have the same meaning as they always did.
1 2 3 4 5 6 β π« β β β β Criteria score: π₯
π― C. Unadorned type should mean nullableβ
GraphQL has been public for 10 years and there's a lot of content out there noting that GraphQL types are nullable by default (unadorned type is nullable) and our changes should not invalidate this content.
1 2 3 4 5 6 β π« β π« β β Criteria score: π₯
π― D. Syntax should be obvious to programmersβ
The GraphQL languages similarity to JSON is one of its strengths, making it immediately feel familiar. Syntax used should feel obvious to developers new to GraphQL.
1 2 3 4 5 6 π« β β β β οΈ β Criteria score: π₯
π― E. Syntax used in SDL and in executable documents should be consistent with SDLβ
When a user wishes to replace the value for an input field or argument with a variable in their GraphQL operation, the type syntax should be either identical or similar, and should carry the same meaning.
1 2 3 4 5 6 β β β π« β β Criteria score: π₯
π― F. Alternative syntaxes should not cause confusionβ
Where a proposal allows alternative syntaxes to be used, the two syntaxes should not cause confusion.
1 2 3 4 5 6 β β β π« β β Criteria score: π₯
π― G. Error propagation boundaries should not change in existing executable documentsβ
An expansion of B, this states that the proposal will not change where errors propagate to when error propagation is enabled (i.e. existing documents will still keep errors local to the same positions that they did when they were published), allowing for the "partial success" feature of GraphQL to continue to shine and not compromising the resiliency of legacy deployed app versions.
1 2 3 4 5 6 β β β β π« β Criteria score: π₯
- βοΈ Objection: proposal to lower the score to π₯. With enough advance notice and a clear upgrade path for legacy apps, the tradeoff might be acceptable.
π― H. Implementation and spec simplicityβ
The implementation required to make the proposal work should be simple.
1 2 3 4 5 6 β π« π« π« β β Criteria score: π₯
π― I. Syntax used in executable documents should be unchangedβ
Executable documents do not differentiate between semantic and strict non-null since inputs never handle "errors" ("null only on error" is the same as "not null" on input). As such, there's no benefit to clients for the syntax of executable documents to change.
1 2 3 4 5 6 β β β π« β β Criteria score: π₯
π― J. Type reasoning should remain localβ
The type of a field (
foo: Int
) can be determined by looking at the field and its type; the reader should not have to read a document or schema directive to determine how the type should be interpreted.
1 2 3 4 5 6 β β β οΈ π« β β οΈ Criteria score: π₯
π― K. Introspection must be backwards compatibleβ
We do not want to break existing tooling.
1 2 3 4 5 6 β β β β β β Criteria score: π₯
π― L. General GraphQL consumers should only need to think about nullable vs non-nullableβ
Schema authors and client frameworks can handle different types of nullability based around error handling and error propagation, but consumers (frontend developers) should only need to deal with nullable or non-nullable as presented to them by their client framework of choice.
May contradict: M
1 2 3 4 5 6 β β β β β β οΈ Criteria score: π₯
π― M. The SDL should have exactly one form used by all producers and consumersβ
The SDL should not be influenced by client features such as local extensions and error propagation mechanics, and should always represent the true full source schema SDL.
May contradict: L
1 2 3 4 5 6 β οΈ β β οΈ β β β Criteria score: π₯
π― N. The solution should add value even with error propagation enabledβ
Even when error propagation is enabled, it's valuable to be able to tell the difference between a field that is truly (semantically) nullable, and one that's only nullable because errors may occur. GraphQL-TOE can be used in such situations so that codegen can safely use non-nullable types in semantically non-nullable positions.
1 2 3 4 5 6 β β β β π« β Criteria score: π₯
π― O. Should not have breaking changes for existing executable documentsβ
It should be possible to enable the solution without negatively impacting existing deployed clients.
Per Lee:
A breaking change is a client observable change in behavior. The decade old GraphQL query should work in the same way as it always has. (We sometimes allow inconsequential changes in behavior, but bubbling the error up isn't inconsequential.)
1 2 3 4 5 6 β β β β π« β Note: though this criteria is currently not considered due to overlap with B and G, it acts as a reminder to look for other forms of breaking change, and helps to reason why B and G are important.
Criteria score: X (not considered - covered by B and G)
π― P. The solution should result in users marking all semantically non-null fields as suchβ
When a field returns data that the business logic dictates does not and will never return a legitimate (non-error) null, the schema authors should have no hesitation over marking it as semantically non-nullable - and thus all semantically non-nullable fields should be marked as such.
Per Benoit:
Not sure how to express it well, but I feel there should be a criteria to mean something like βthe solution encourages that eventually most fields in most schemas are semantically non nullβ. As a client developer thatβs kind of an outcome of this whole effort Iβd like to see happening.
1 2 3 4 5 6 β β β β π« β οΈ Criteria score: π₯
π― Q. Migrating the unadorned output type to other forms of nullability should be non-breakingβ
The default (unadorned) type should be a type that you can migrate away from, once nullability expectations become more concrete, without breaking existing client queries.
1 2 3 4 5 6 β π« β π« β β Note: this is not necessarily a duplicate of C as it doesn't specifically requireβ the unadorned type be nullable, however no proposal currently proposes a mechanism for moving from any non-nullable type to a nullable type in a non-breaking way, and thus this criteria is currently discounted.
Criteria score: X (not considered)
π― R. Semantic nullability should only impact outputs, not inputsβ
There's no meaningful difference between semantic non-null and strict non-null on input, since inputs do not handle errors (and thus "null only on error" describes a situation that cannot occur).
Inputs include: field arguments, directive arguments, and input fields.
As such:
- the syntax used to represent input nullability in SDL (
Int
= nullable, andInt!
= non-nullable) should be unchanged- the representation in introspection for inputs (namely the
NON_NULL
type wrapper) should be unchanged
1 2 3 4 5 6 β β β π« β β Criteria score: π₯
<!--
Template for new items:
π― X. Titleβ
DESCRIPTION
1 2 3 4 5 ? ? ? ? ? Criteria score: β
-->
π§ Possible Solutions
The community has imagined a variety of possible solutions, synthesized here.
Each solution is identified with a
Number
so they can be referenced in the rest of the document. New solutions must be added to the end of the list.Semantic nullability is only relevant to output positions, so when comparing syntax we will look for changes versus the current syntax used to represent these types:
Input syntax Output syntax Semantically nullable Int
Int
Semantically non-nullable - Int
Strictly non-nullable Int!
Int!
π‘ 1. New "Semantic Non-Null" type, represented by
*
βChampion: @benjie
- Spec edits: https://github.com/graphql/graphql-spec/pull/1065
- GraphQL.js implementation: https://github.com/graphql/graphql-js/pull/4192
- Additional tools:
- GraphQL-TOE - throw on error when reading from an errored field
- GraphQL-SOCK - Semantic Output Conversion Kit - takes a schema involving semanantic non-null type and converts it to a schema using traditional syntax (respecting the client's error handling behaviors) for compatibility with existing (semantic non-null unaware) tooling
This proposal introduces a new Semantic Non-Null type using a prefix or postfix symbol (currently
*
postfix) to indicate a field that will be null only on error. Existing types and operations are unaffected, and usage can be migrated on a per-type-position basis. Moving from a nullable type to a semantic non-nullable type (on output) is a non-breaking change. Semantic non-nullable is meaningless on input.type Post {
# Every post belongs to a topic, however don't blow the post up if retrieval of the topic fails.
topic: Topic*
}Querying a semantic non-null field is the same as querying any other field.
Input syntax Output syntax Semantically nullable Int
Int
Semantically non-nullable - Int
βInt*
Strictly non-nullable Int!
Int!
π² Variationsβ
Various options for the syntax have been discussed the choice of symbol comes down mostly to aesthetics.
βοΈ Evaluationβ
- A
- β
- B
- β Existing symbology unchanged.
- C
- β Existing symbology unchanged.
- D
- π«
Int*
syntax is not immediately obvious.- E
- β Same syntax.
- F
- β Same syntax.
- G
- β Error capture positions unchanged when error propagation enabled
- H
- β Implementation and spec simplicity.
- I
- β
*
doesn't apply on input, so syntax is unchanged.- J
- β Local syntax only
- K
- β Introspection backwards compatible via
__Field.type(includeSemanticNonNull: Boolean! = false)
- L
- β Proposal encourages consumers to use client-produced SDL which only uses traditional nullability (
Type
/Type!
)- M
- β οΈ You can use the same SDL everywhere, but that's not what this solution encourages.
- N
- β Indicates semantically non-null and strictly non-null types separately
- O
- β Client syntax unchanged
- P
- β There are no drawbacks to adding semantically non-nullable fields
- Q
- β
- R
- β
*
only needed in output positions, input positions unchangedπ‘ 2. "Strict Semantic Nullability"β
Champion: @leebyron
This proposal introduces a
@strictNullability
directive on the schema. Types in schemas using this directive would now be semantically non-nullable by default, and a new semantically nullable type is introduced (using the?
symbol) to indicate that a position may semantically be null.
Input syntax Output syntax Semantically nullable Int
β ???Int
βInt?
Semantically non-nullable - Int
Strictly non-nullable Int!
β ???Int!
βοΈ Evaluationβ
- A
- β
- B
- π« Though existing documents remain valid, input variables using the unadorned type now mean "semantically non-nullable" and will no longer accept
null
values? {Confirmation by Lee pending.}- C
- π« GraphQL is no longer "nullable by default", and
Int
no longer represents a nullable integer.- D
- β
Int?
is commonly used to indicate nullablility in programming languages andInt!
indicating non-nullable or danger is common.Int
is less obvious when?
and!
variants exist.- E
- β The same syntax is used on input and output.
- F
- β There is no alternative syntax.
- G
- β Error capture positions unchanged when error propagation enabled
- H
- π« Implementation and spec simplicity.
- I
- β
- J
- β
- K
- β
- L
- β
- M
- β
- N
- β Indicates semantically non-null and strictly non-null types separately
- O
- β
- P
- β
- Q
- π«
Int
βInt?
is breaking- R
β
π‘ 3. New "Semantic Non-Null" type, usurping
!
syntaxβChampion: @benjie
This proposal is similar to proposal 1, but:
It introduces a document-level directive,
@semanticNullability
, which when present on a document allows the!
suffix to be used to represent semantically non-nullable output types, and a new!!
suffix to be used to represent strictly non-nullable output types.The
Int!
syntax simply means "non-nullable" on input, as it always has. (Note: input types are always either semantically nullable or strictly non-nullable.)Syntax only changes when
@semanticNullability
directive is present:
Input syntax Output syntax Semantically nullable Int
Int
Semantically non-nullable - Int
βInt!
Strictly non-nullable Int!
Int!
βInt!!
All documents (both SDL and executable documents) retain their current meaning, and the semantically non-null type can be adopted in output positions on a per-document basis by adding the document directive.
Since there's no difference between whether a type is "semantically" or "strictly" non-nullable on input (input does not represent errors), executable documents will retain their existing syntax in perpetuity and never need to use this new directive - it's only used in the SDL.
Further, it's proposed that the SDL production responsibility be pushed to the client framework (Relay, Apollo, URQL, etc), which can reflect their own SDL for the schema that honours their error behavior (e.g. throw on error), null handling, and any client-local modifications (e.g. additional client-side fields/types). This client-produced SDL can use the traditional syntax and should be used by tooling such as code generation - this further limits the
@semanticNullability
directive to only be used by schema and tooling authors, meaning the vast majority of GraphQL consumers do not need to see it in their day-to-day work.βοΈ Evaluationβ
- A
- β
- B
- β Executable documents are not impacted by this proposal.
- C
- β
Int
means nullable still.- D
- β
Int!
to indicate non-nullable is common in programming languages; andInt!!
looks like it indicates "danger".- E
- β Executable documents do not use
!!
, and!
means non-nullable on both input and output (the difference between semantic an strict non-null does not occur on input)- F
- β
Int
reatains its meaning across both modes, andInt!
means non-nullable in both modes. Only the SDL ever usesInt!!
and it still means non-null, just with the additional "kills parent on exception" behavior.- G
- β Error capture positions unchanged when error propagation enabled
- H
- π« Implementation and spec simplicity.
- I
- β Semantic non-null not relevant to inputs, so no reason to use directive in executable documents -> syntax unchanged.
- J
- β οΈ Local reasoning holds for all but the schema authors; this is enabled through the use of client-generated SDL reflecting client extensions and error propagation behavior. For schema authors, local reasoning in the source SDL returns whether a field is nullable or non-nullable, but does not differentiate between semantically non-nullable and strictly non-nullable.
- K
- β Introspection backwards compatible via
__Field.type(includeSemanticNonNull: Boolean! = false)
- L
- β Proposal encourages consumers to use client-produced SDL which only uses traditional nullability (
Type
/Type!
)- M
- β οΈ You can use the same SDL everywhere, but that's not what this solution encourages.
- N
- β Indicates semantically non-null and strictly non-null types separately
- O
- β Client syntax unchanged
- P
- β There are no drawbacks to adding semantically non-nullable fields
- Q
- β
- R
β Syntax used for inputs is unchanged with or without the directive.
π‘ 4. New "Semantic Non-Null" type, with
?
used for nullable typesβChampion: @twof
This proposal builds on solution 3, but with a syntactic shuffle such that the unadorned type may be used as the semantically non-nullable type when the directive is present, and a
?
symbol is used to indicate a nullable position.Syntax only changes when
@semanticNullability
directive is present:
Input syntax Output syntax Semantically nullable Int
βInt?
Int
βInt?
Semantically non-nullable - Int
Strictly non-nullable Int!
βInt
Int!
βοΈ Evaluationβ
- A
- β
- B
- β Existing documents don't use the directive, and thus are not impacted.
- C
- π« With the directive present, GraphQL is no longer "nullable by default", and
Int
no longer represents a nullable integer.- D
- β
Int?
is commonly used to indicate nullablility in programming languages andInt!
indicating non-nullable or danger is common.Int
is less obvious when?
and!
variants exist.- E
- π« If the schema uses
@semanticNullability
but an operation document does not,Int
has vastly different meanings: nullable on input but non-nullable on output.- F
- π«
Int
being nullable in one mode and non-nullable in the other mode is unexpected and will likely lead to confusion.- G
- β Error capture positions unchanged when error propagation enabled
- H
- π« Implementation and spec simplicity.
- I
- π« Clients will need to move to using new syntax (
Type?
/Type
) or have syntax incongruent with schema SDL- J
- π« The nullability of
Type
cannot be determined without checking for a document directive- K
- β Introspection backwards compatible via
__Field.type(includeSemanticNonNull: Boolean! = false)
- L
- β
- M
- β
- N
- β Indicates semantically non-null and strictly non-null types separately
- O
- β Clients must opt in to new syntax with document directive
- P
- β There are no drawbacks to adding semantically non-nullable fields
- Q
- π«
Int
βInt?
is breaking- R
π« Input positions have changed
Int
βInt?
,Int!
βInt
π‘ 5. Use non-null in semantically non-nullable places and encourage disabling error propagationβ
Champion: @martinbonnin
This proposal relies on the ability of clients to opt out of error propagation; instead of introducing a new type it instructs schema authors to optimize for error-handling clients and use the traditional non-null type (
!
) on all semantically non-null fields.
Input syntax Output syntax Semantically nullable Int
Int
Semantically non-nullable - Int
βInt!
Strictly non-nullable Int!
Int!
βοΈ Evaluationβ
- A
- π«π The nullability used in both error-propagation and no-error-propagation modes are the same. This is a feature, not a bug!
- B
- β The change from nullable to non-nullable on output is backwards compatible from a type perspective; for impact on error boundaries see G.
- C
- β
Int
means nullable still.- D
- β οΈ Adding
@onError
to operations is not immediately intuitive but most error-handling clients should add it automatically, making it transparent to end users.- E
- β Same syntax.
- F
- β Same syntax.
- G
- π« Using non-null in more positions will change the error boundary positions when error propagation is enabled.
- H
- β Implementation and spec simplicity.
- I
- β No change
- J
- β No change
- K
- β No change
- L
- β No change
- M
- β No change
- N
- π« Solution actually decreases value when error propagation is enabled due to lowered resilience to errors.
- O
- π« Changing fields to strictly non-null causes errors to propagate further, a breaking change. (Duplicate of G.)
- P
- π« Though the solution states it encourages the adoption of non-null, doing so is a breaking change for existing clients and so adopters are likely to hesitate when marking some semantically non-nullable positions as such
- Q
- β Same syntax.
- R
- β Same syntax.
π‘ 6.
@semanticNonNull
directiveβChampion: -
Outline: https://specs.apollo.dev/nullability/v0.4/#@semanticNonNull
This proposal (which is already adopted in a few places!) introduces a directive that can be added to fields to indicate their semantic nullability (and that of their nested list positions).
directive @semanticNonNull(levels: [Int!]! = [0]) on FIELD_DEFINITION
type Query {
nonNullListOfNonNullInt: [Int] @semanticNonNull(levels: [0, 1])
}The proposal is broadly similar to solution 1, but avoids introducing new syntax. Interestingly, since the directive only applies on
FIELD_DEFINITION
it explicitly limits semantic nullability to output positions.
Input syntax Output syntax Semantically nullable Int
Int
Semantically non-nullable - Int
βInt @semanticNotNull
Strictly non-nullable Int!
Int!
βοΈ Evaluationβ
- A
- β
- B
- β Existing symbology unchanged.
- C
- β Existing symbology unchanged.
- D
- β No syntax change (directive syntax already exists).
- E
- β Same syntax.
- F
- β Same syntax.
- G
- β Error capture positions unchanged when error propagation enabled
- H
- β Implementation and spec simplicity.
- I
- β Directive does not apply to input positions.
- J
- β οΈ Though the directives are local to the field, the reader must still correlate the directive and the passed indexes with the types specified to conclude what the final type is.
- K
- β Introspection backwards compatible via
__Field.type(includeSemanticNonNull: Boolean! = false)
- L
- β οΈ Depends whether we advise using client-generated SDL or not.
- M
- β Same SDL everywhere.
- N
- β Indicates semantically non-null and strictly non-null types separately.
- O
- β
- P
- β οΈ Though there's no technical reason not to do so, the mechanics of adding the directive (particularly when referencing positions inside lists) are tiresome in SDL-first schemas, decreasing likeliness that positions will be updated. (Code-first schemas are unaffected.) Further, the directives are likely to have a significant impact on the formatting of the SDL (
@semanticNonNull
is 16 characters, almost quarter of a line if wrap at 80), so designers may wish to only add them in the most critical of locations.- Q
- β
- R
- β Directive is only valid on output positions.