Skip to content

Commit 39e593c

Browse files
committed
@OneOf fields
1 parent f6bd659 commit 39e593c

3 files changed

+67
-3
lines changed

spec/Section 3 -- Type System.md

+50-3
Original file line numberDiff line numberDiff line change
@@ -818,6 +818,9 @@ of rules must be adhered to by every Object type in a GraphQL schema.
818818
characters {"__"} (two underscores).
819819
2. The argument must accept a type where {IsInputType(argumentType)}
820820
returns {true}.
821+
3. If the field is a Oneof Field:
822+
1. The field must be nullable.
823+
2. The field must not have a default value.
821824
3. An object type may declare that it implements one or more unique interfaces.
822825
4. An object type must be a super-set of all interfaces it implements:
823826
1. Let this object type be {objectType}.
@@ -845,6 +848,8 @@ IsValidImplementation(type, implementedType):
845848
2. Let {implementedFieldType} be the return type of {implementedField}.
846849
3. {IsValidImplementationFieldType(fieldType, implementedFieldType)}
847850
must be {true}.
851+
6. {field} must be a Oneof Field if and only if {implementedField} is a
852+
Oneof Field.
848853

849854
IsValidImplementationFieldType(fieldType, implementedFieldType):
850855
1. If {fieldType} is a Non-Null type:
@@ -917,6 +922,30 @@ May yield the result:
917922
The type of an object field argument must be an input type (any type except an
918923
Object, Interface, or Union type).
919924

925+
**Oneof Fields**
926+
927+
Oneof Fields are a special variant of Object Type fields where the type system
928+
asserts that exactly one of the field's arguments must be set and non-null, all
929+
others being omitted. This is useful for representing situations where an input
930+
may be one of many different options.
931+
932+
When using the type system definition language, the `@oneOf` directive is used
933+
to indicate that a Field is a Oneof Field (and thus requires exactly one of its
934+
arguments be provided):
935+
936+
```graphql
937+
type Query {
938+
findUser(
939+
byID: ID
940+
byUsername: String
941+
byEmail: String
942+
byRegistrationNumber: Int
943+
): User @oneOf
944+
}
945+
```
946+
947+
In schema introspection, the `__Field.oneArgument` field will return {true} for
948+
Oneof Fields, and {false} for all other Fields.
920949

921950
### Field Deprecation
922951

@@ -1160,6 +1189,9 @@ Interface types have the potential to be invalid if incorrectly defined.
11601189
characters {"__"} (two underscores).
11611190
2. The argument must accept a type where {IsInputType(argumentType)}
11621191
returns {true}.
1192+
3. If the field is a Oneof Field:
1193+
1. The field must be nullable.
1194+
2. The field must not have a default value.
11631195
3. An interface type may declare that it implements one or more unique
11641196
interfaces, but may not implement itself.
11651197
4. An interface type must be a super-set of all interfaces it implements:
@@ -1880,7 +1912,8 @@ provide:
18801912
- the `@deprecated` directive if representing deprecated portions of the
18811913
schema;
18821914
- the `@oneOf` directive if representing types that require exactly one field
1883-
(i.e. Oneof Input Objects).
1915+
(i.e. Oneof Input Objects) or fields that require exactly one argument (i.e.
1916+
Oneof Fields).
18841917

18851918
**Custom Directives**
18861919

@@ -2048,11 +2081,14 @@ type ExampleType {
20482081
### @oneOf
20492082

20502083
```graphql
2051-
directive @oneOf on INPUT_OBJECT
2084+
directive @oneOf on INPUT_OBJECT | FIELD_DEFINITION
20522085
```
20532086

20542087
The `@oneOf` directive is used within the type system definition language
2055-
to indicate an Input Object is a Oneof Input Object.
2088+
to indicate:
2089+
2090+
- an Input Object is a Oneof Input Object, or
2091+
- an Object Type's Field is a Oneof Field.
20562092

20572093
```graphql example
20582094
input UserUniqueCondition @oneOf {
@@ -2061,3 +2097,14 @@ input UserUniqueCondition @oneOf {
20612097
organizationAndEmail: OrganizationAndEmailInput
20622098
}
20632099
```
2100+
2101+
```graphql example
2102+
type Query {
2103+
findUser(
2104+
byID: ID
2105+
byUsername: String
2106+
byEmail: String
2107+
byRegistrationNumber: Int
2108+
): User @oneOf
2109+
}
2110+
```

spec/Section 4 -- Introspection.md

+3
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,7 @@ type __Field {
159159
type: __Type!
160160
isDeprecated: Boolean!
161161
deprecationReason: String
162+
oneArgument: Boolean!
162163
}
163164

164165
type __InputValue {
@@ -385,6 +386,8 @@ Fields
385386
* `isDeprecated` returns {true} if this field should no longer be used,
386387
otherwise {false}.
387388
* `deprecationReason` optionally provides a reason why this field is deprecated.
389+
* `oneArgument` must return {true} for Oneof Fields, {false} for all other
390+
Fields.
388391

389392

390393
### The __InputValue Type

spec/Section 5 -- Validation.md

+14
Original file line numberDiff line numberDiff line change
@@ -780,6 +780,20 @@ fragment missingRequiredArg on Arguments {
780780
}
781781
```
782782

783+
#### Oneof Fields Have Exactly One Argument
784+
785+
* For each Oneof Field in the document:
786+
* Let {arguments} be the arguments provided by the Field.
787+
* {arguments} must contain exactly one entry.
788+
* For the sole {argument} in {arguments}:
789+
* Let {value} be the value of {argument}.
790+
* {value} must not be the {null} literal.
791+
792+
**Explanatory Text**
793+
794+
Oneof Fields require that exactly one argument must be supplied and that
795+
argument must not be null.
796+
783797
## Fragments
784798

785799
### Fragment Declarations

0 commit comments

Comments
 (0)