Object Types

Object types are the building blocks of a GraphQL schema. Each object type has a name and a set of fields. Fields can return scalars like String and Int, or other object types, forming a graph that clients traverse through their queries.

GraphQL
type Product {
  name: String!
  price: Decimal!
  inStock: Boolean!
}

type Author {
  name: String!
  bio: String
  books: [Book!]!
}

type Book {
  title: String!
  author: Author!
}

Every field in a query resolves to a concrete value. Object types define the shape of that value. Understanding how to define and configure them is the foundation of building a Hot Chocolate schema.

Defining Object Types#

In the implementation-first approach, a C# class becomes a GraphQL object type automatically. The source generator picks up public properties and methods and maps them to fields. In the code-first approach, you create a class that inherits from ObjectType<T> and configure it explicitly.

Properties as Fields#

Public properties with a getter are automatically mapped to GraphQL fields. Hot Chocolate converts the property name to camelCase for the schema.

This produces the following schema:

GraphQL
type Product {
  id: Int!
  name: String!
  price: Decimal!
  inStock: Boolean!
}

Methods as Resolvers#

Public methods on a class become resolver fields. This is how you add computed fields or fields that fetch data from external sources. Method parameters that are registered services are injected automatically.

Both approaches produce this schema:

GraphQL
type Author {
  name: String!
  books: [Book!]!
}

The naming rules for methods are the same as for query fields: Get prefixes and Async suffixes are stripped, and the result is camelCased.

Field Configuration#

You can rename fields, ignore them, and add descriptions without changing the shape of your C# classes.

Renaming Fields#

You can also rename the type itself.

If only one client needs different names, prefer using aliases in that client's queries instead of changing the schema.

Ignoring Fields#

Descriptions#

Descriptions appear in GraphQL introspection and tooling like Nitro. They help consumers of your API understand the purpose of each type and field.

Explicit Binding#

By default, all public properties and methods are included as fields. You can switch to explicit binding, where you opt in to each field individually.

C#
public class ProductType : ObjectType<Product>
{
    protected override void Configure(IObjectTypeDescriptor<Product> descriptor)
    {
        descriptor.BindFieldsExplicitly();

        descriptor.Field(f => f.Name);
        descriptor.Field(f => f.Price);
    }
}

Only name and price appear in the schema. All other properties on Product are excluded.

You can also set this globally, which affects all types.

C#
builder
    .AddGraphQL()
    .ModifyOptions(options =>
    {
        options.DefaultBindingBehavior = BindingBehavior.Explicit;
    });

Nullability#

Hot Chocolate uses C# nullability to determine whether a GraphQL field is nullable or non-null. When nullable reference types are enabled in your project, the mapping is straightforward.

C# TypeGraphQL Type
stringString!
string?String
intInt!
int?Int
List<string>[String!]!
List<string?>[String]!
List<string>?[String!]

Value types (int, bool, decimal) are non-null by default. Their nullable counterpart (int?, bool?) maps to a nullable GraphQL field.

Reference types follow your project's nullable reference type settings. With nullable reference types enabled (recommended), string maps to String! and string? maps to String. Without nullable reference types enabled, all reference type fields are nullable by default.

You can override the inferred nullability when needed.

For full details on nullability, see Non-Null.

Dictionary Support#

Hot Chocolate automatically maps Dictionary<TKey, TValue> properties to a list of key-value pair objects. This eliminates the need for custom resolvers when exposing dictionary data.

This produces the following schema:

GraphQL
type Product {
  name: String!
  attributes: [KeyValuePairOfStringAndString!]!
}

type KeyValuePairOfStringAndString {
  key: String!
  value: String!
}

Clients query dictionary fields like any other list.

GraphQL
{
  product {
    name
    attributes {
      key
      value
    }
  }
}

This works with any key and value types. For example, Dictionary<string, int> produces KeyValuePairOfStringAndInt32 with the appropriate scalar types.

Next Steps#

Edit this page on GitHub
Last updated on by Tobias Tengler