Back to Graphql Ruby

Introspection

guides/schema/introspection.md

2.6.16.2 KB
Original Source

A GraphQL schema has a built-in introspection system that publishes the schema's structure. In fact, the introspection system can be queried using GraphQL, for example:

graphql
{
  __schema {
    queryType {
      name
    }
  }
}
# Returns:
# {
#   "data": {
#     "__schema": {
#       "queryType": {
#         "name": "Query"
#       }
#     }
#   }
# }

This system is used for GraphQL tooling like the GraphiQL editor.

Here are the default parts of the introspection system:

  • __schema is a root-level field that contains data about the schema: its entry points, types, and directives.
  • __type(name: String!) is a root-level field that returns data about a type with the given name, if there is a type with that name.
  • __typename works a bit differently: it can be added to any selection, and it will return the type of object being queried.

Here are some __typename examples:

graphql
{
  user(id: "1") {
    handle
    __typename
  }
}
# Returns:
# {
#   "data": {
#     "user": {
#       "handle": "rmosolgo",
#       "__typename": "User"
#     }
#   }
# }

For unions and interfaces, __typename returns the object type for the current object, for example:

graphql
{
  search(term: "puppies") {
    title
    __typename
  }
}
# Returns:
# {
#   "data": {
#     "search": [
#       {
#         "title": "Sound of Dogs Barking",
#         "__typename": "AudioClip",
#       },
#       {
#         "title": "Cute puppies playing with a stick",
#         "__typename": "VideoClip",
#       },
#       {
#         "title": "The names of my favorite pets",
#         "__typename": "TextSnippet"
#       },
#     ]
#   }
# }

Customizing Introspection

You can use custom introspection types.

ruby
# create a module namespace for your custom types:
module Introspection
  # described below ...
end

class MySchema < GraphQL::Schema
  # ...
  # then pass the module as `introspection`
  introspection Introspection
end

Keep in mind that off-the-shelf tooling may not support your custom introspection fields. You may have to modify existing tooling or create your own tools to make use of your extensions.

Introspection Namespace

The introspection namespace may contain a few different customizations:

  • Class-based {% internal_link "object definitions", "/type_definitions/objects" %} which replace the built-in introspection types (such as __Schema and __Type)
  • EntryPoints, A class-based {% internal_link "object definition", "/type_definitions/objects" %} containing introspection entry points (like __schema and __type(name:)).
  • DynamicFields, A class-based {% internal_link "object definition", "/type_definitions/objects" %} containing dynamic, globally-available fields (like __typename.)

Custom Introspection Types

The module passed as introspection may contain classes with the following names, which replace the built-in introspection types:

Custom class nameGraphQL typeBuilt-in class name
SchemaType__Schema{{ "GraphQL::Introspection::SchemaType"
TypeType__Type{{ "GraphQL::Introspection::TypeType"
DirectiveType__Directive{{ "GraphQL::Introspection::DirectiveType"
DirectiveLocationType__DirectiveLocation{{ "GraphQL::Introspection::DirectiveLocationEnum"
EnumValueType__EnumValue{{ "GraphQL::Introspection::EnumValueType"
FieldType__Field{{ "GraphQL::Introspection::FieldType"
InputValueType__InputValue{{ "GraphQL::Introspection::InputValueType"
TypeKindType__TypeKind{{ "GraphQL::Introspection::TypeKindEnum"

The class-based definitions' names must match the names of the types they replace.

Extending a Built-in Type

The built-in classes listed above may be extended:

ruby
module Introspection
  class SchemaType < GraphQL::Introspection::SchemaType
    # ...
  end
end

Inside the class definition, you may:

  • add new fields by calling field(...) and providing implementations
  • redefine field structure by calling field(...)
  • provide new field implementations by defining methods
  • provide new descriptions by calling description(...)

Introspection Entry Points

The GraphQL spec describes two entry points to the introspection system:

  • __schema returns data about the schema (as type __Schema)
  • __type(name:) returns data about a type, if one is found by name (as type __Type)

You can re-implement these fields or create new ones by creating a custom EntryPoints class in your introspection namespace:

ruby
module Introspection
  class EntryPoints < GraphQL::Introspection::EntryPoints
    # ...
  end
end

This class is an object type definition, so you can override existing fields or add new ones here. They'll be available on the root query object, but ignored in introspection (just like __schema and __type).

Dynamic Fields

The GraphQL spec describes a field which may be added to any selection: __typename. It returns the name of the current GraphQL type.

You can add fields like this (or override __typename) by creating a custom DynamicFields definition:

ruby
module Introspection
  class DynamicFields < GraphQL::Introspection::DynamicFields
    # ...
  end
end

Any fields defined there will be available in any selection, but ignored in introspection (just like __typename).

Disabling Introspection

In case you want to turn off introspection entry points __schema and __type (for instance in the production environment) you can use a #disable_introspection_entry_points shorthand method:

ruby
class MySchema < GraphQL::Schema
  disable_introspection_entry_points if Rails.env.production?
end

Where disable_introspection_entry_points will disable both the __schema and __type introspection entry points, you can also individually disable the introspection entry points using the disable_schema_introspection_entry_point and disable_type_introspection_entry_point shorthand methods:

ruby
class MySchema < GraphQL::Schema
  disable_schema_introspection_entry_point
  disable_type_introspection_entry_point
end