The type system that powers most of the GiraphQL type checking has 2 components. The first is the TypeInfo type param passed into the SchemaBuilder. This allows a shared set of types to be reused throughout the schema, and is responsible for providing type information for shared types like the Context object, and any Object, Interface, or Scalar types that you want to reference by name (as a string). Having all type information in a single object can be convenient at times, but with large schemas, can become unwieldy.
To support a number of additional use cases, including Unions and Enums, large schemas, and plugins that use extract type information from other sources (eg the Prisma, or the simple-objects plugin), GiraphQL has another way of passing around type information. This system is based in
Ref objects that contain the type information it represents. Every builder method for creating a type or a field returns a
Using Ref objects allows us to separate the type information from the implementation, and allows for a more modular design.
GiraphQL aims to be extensible in a way that makes features from plugins feel like part of the core experience. To achieve this, every options object used in GiraphQL can be extended with custom properties. You may notice that there are places throughout the documentation where there are empty options objects passed into functions. While this may seem inconvenient at first glance, the reason behind is that it should be easy for plugins to add new required options, and having optional options objects would break that goal.
Implementing plugins for GiraphQL is currently not well documented, and requires a deep understanding of how the internal types in GiraphQL function.
There are 3 parts to the GiraphQL plugin API.
GiraphQLSchemaTypes is a global typescript namespace that contains interfaces for all the object types used by the GiraphQL, as well as types describing the core builder and field builder classes. By putting these things in a shared global namespace, plugins can freely extend these types and take advantage of all the same generics. This enables things like adding authorization check function into field options, and allowing those check functions to correctly infer the parent, and args for the field.
BasePlugin class describes the hooks that are available for plugins, and enables plugins to hook into the configuration of types and fields.
BaseFieldWrapper class provides functionality for wrapping fields with additional runtime logic. There are a lot of use cases for wrapping fields, and when too many things need to wrap fields to add their own logic, things can get complicated, and there are a lot of complex edge cases to consider. The BaseFieldWrapper simplifies to process of hooking into the execution lifecycle of fields, allowing for running code before the field executes, after a field resolves, when the concrete type of a returned object/interface/union is determined, as well as a way to pass information down through the tree as a request is resolved.