@graphql-ts/schema is a thin wrapper around
GraphQL.js providing type-safety for
constructing GraphQL Schemas while avoiding type-generation, declaration
merging
and
decorators.
The gWithContext function accepts a Context type parameter which binds
the returned functions so they can be used to compose GraphQL types into a
GraphQL schema.
A simple schema with only a query type looks like this:
You can use pass the schema to ApolloServer and other GraphQL servers.
You can also create a more advanced schema with other object types, circular
types, args, and mutations. See GWithContext for what the other
functions on g do.
The gWithContext.infer type is useful particularly when defining circular
types to resolve errors from TypeScript because of the circularity.
We recommend aliasing gWithContext.infer to your g like this to make it
easier to use:
Creates a GraphQL object type.
Note this is an output type, if you want an input object, use
g.inputObject.
When calling g.object, you must provide a type parameter that is the
source of the object type. The source is what you receive as the first
argument of resolvers on this type and what you must return from resolvers
of fields that return this type.
To do anything other than just return a field from the source type, you need to provide a resolver.
Note: TypeScript will force you to provide a resolve function if the field in the source type and the GraphQL field don't match
GraphQL types will often contain references to themselves and to make
TypeScript allow that, you need have an explicit type annotation of
g<typeof g.object<Source>> along with making fields a function that
returns the object.
Create a GraphQL union type.
A union type represents an object that could be one of a list of types. Note it is similar to an interface type except that a union doesn't imply having a common set of fields among the member types.
Creates a GraphQL field.
These will generally be passed directly to the fields object in a
g.object call.
A helper to declare fields while providing the source type a single time rather than in every resolver.
Creates a GraphQL interface field.
These will generally be passed directly to the fields object in a
call. Interfaces fields are similar to
regular fields except that they don't define how the field
is resolved.
Note that regular fields are assignable to interface fields but the opposite is not true. This means that you can use a regular field in an interface type.
Creates a GraphQL interface type that can be implemented by other GraphQL object and interface types.
When using GraphQL interface and union types, there needs to a way to
determine which concrete object type has been returned from a resolver.
With graphql-js and @graphql-ts/schema, this is done with isTypeOf on
object types and resolveType on interface and union types. Note
@graphql-ts/schema does not aim to strictly type the implementation of
resolveType and isTypeOf. If you don't provide resolveType or
isTypeOf, a __typename property on the source type will be used, if
that fails, an error will be thrown at runtime.
You might have noticed that g.interfaceField was used instead of
g.field for the fields on the interfaces. This is because interfaces
aren't defining implementation of fields which means that fields on an
interface don't need define resolvers.
Even though interfaces don't contain field implementations, you may still
want to share field implementations between interface implementations. You
can use g.fields to do that. See g.fields for more information about
why you should use g.fields instead of just defining an object the fields
and spreading that.
A shorthand to easily create enum values to pass to .
If you need to set a description or deprecationReason for an enum
variant, you should pass values directly to g.enum without using
g.enumValues.
Creates an enum type with a number of enum values.
Creates a GraphQL argument.
Args can can be used as arguments on output fields:
Or as fields on input objects:
Creates an input object type
Circular input objects require explicitly specifying the fields on the object in the type because of TypeScript's limits with circularity.
You can specify all of your non-circular fields outside of the fields
object and then use typeof to get the type to avoid writing the
non-circular fields as types again.
Wraps any GraphQL type in a list type.
When used as an input type, you will recieve an array of the inner type.
When used as an output type, you can return an iterable of the inner type
that also matches typeof val === 'object' so for example, you'll probably
return an Array most of the time but you could also return a Set you
couldn't return a string though, even though a string is an iterable, it
doesn't match typeof val === 'object'.
Wraps a nullable type with a non-nullable type.
Types in GraphQL are always nullable by default so if you want to enforce that a type must always be there, you can use the non-null type.
When using a non-null type as an input type, your resolver will never recieve null and consumers of your GraphQL API must provide a value for it unless you provide a default value.
When using a non-null type as an output type, your resolver must never
return null. If you do return null(which unless you do
type-casting/ts-ignore/etc. @graphql-ts/schema will not let you do)
graphql-js will return an error to consumers of your GraphQL API.
Non-null types should be used very carefully on output types. If you have to do a fallible operation like a network request or etc. to get the value, it probably shouldn't be non-null. If you make a field non-null and doing the fallible operation fails, consumers of your GraphQL API will be unable to see any of the other fields on the object that the non-null field was on. For example, an id on some type is a good candidate for being non-null because if you have the item, you will already have the id so getting the id will never fail but fetching a related item from a database would be fallible so even if it will never be null in the success case, you should make it nullable.
If you try to wrap another non-null type in a non-null type again, you will get a type error.
Creates a scalar type.
Note, while graphql-js allows you to express scalar types like the ID
type which accepts integers and strings as both input values and return
values from resolvers which are transformed into strings before calling
resolvers and returning the query respectively, the type you use should be
string for ID since that is what it is transformed into.
@graphql-ts/schema doesn't currently express the coercion of scalars, you
should instead convert values to the canonical form yourself before
returning from resolvers.
A GraphQL object type. This should generally be constructed with object.
Note this is an output type, if you want an input object, use GInputObjectType.
If you use the GObjectType constructor directly, all fields will need
explicit resolvers so you should use g.object instead.
A GraphQL enum type. This should generally be constructed with enum.
Unlike some other constructors in this module, this constructor functions
exactly the same as it's counterpart g.enum so it is safe to use directly
if desired.
A GraphQL enum type. This should generally be constructed with scalar.
Unlike some other constructors in this module, this constructor functions
exactly the same as it's counterpart g.scalar so it is safe to use directly
if desired.
Also unlike some other types in this module, this type is exactly equivalent
to the original GraphQLScalarType type from the
graphql package.
A GraphQL input object type. This should generally be constructed with inputObject.
Unlike some other constructors in this module, this constructor functions
exactly the same as it's counterpart g.inputObject so it is safe to use
directly if desired.
A GraphQL interface type that can be implemented by other GraphQL object and interface types. This should generally be constructed with interface.
If you use the GInterfaceType constructor directly, all fields will need
explicit resolvers so you should use g.interface instead.
A GraphQL union type. This should generally be constructed with union.
A union type represents an object that could be one of a list of types. Note it is similar to an GInterfaceType except that a union doesn't imply having a common set of fields among the member types.
While this constructor will work, you should generally use g.union because
you will need to explicitly provide the source type parameter as TypeScript
is unable to infer it correctly. Note this is only required for this
constructor, this is not required when using g.union.
A GraphQL list type. This should generally be constructed with list.
Unlike some other constructors in this module, this constructor functions
exactly the same as it's counterpart g.list so it is safe to use directly
if desired.
Also unlike the named types in this module, the original
GraphQLList type from the graphql package cannot be
assigned to a variable of type GList. Though GList is assignable to
GraphQLList.
For example, the following code will not compile:
But the following code will compile:
This is due to the lack of a discriminating property between the
GraphQLNonNull and GraphQLList types.
A GraphQL non-null type. This should generally be constructed with nonNull.
Unlike some other constructors in this module, this constructor functions
exactly the same as it's counterpart g.nonNull so it is safe to use
directly if desired.
Also unlike the named types in this module, the original
GraphQLNonNull type from the graphql package cannot
be assigned to a variable of type GNonNull. Though GNonNull is
assignable to GraphQLNonNull.
For example, the following code will not compile:
But the following code will compile:
This is due to the lack of a discriminating property between the
GraphQLNonNull and GraphQLList types.
A GraphQL argument. These should be created with arg
Args can can be used as arguments on output fields:
Or as fields on input objects:
When the type of an arg is non-null, the value will always exist.
A GraphQL output field for an object type which should be created using field.
export type GField<Source, Args extends { }, Type extends GOutputType<Context>, SourceAtKey, Context> = {