Docs
Guides
Batching for N+1 Problem

Batching for N+1 problem

A Mesh Gateway that does be not properly configured will face the same famous issue as any other GraphQL servers: the N+1 Query problem.

Fortunately, GraphQL Mesh provides a way to batch requests with specific parameters.

Enable requests batching

Our multiple-sources (opens in a new tab) example Gateway:

Will emit multiple requests to the "Authors" API when resolving the nested Book.author field:

query bestSellersByStore {
  stores {
    id
    name
    bookSells {
      sellsCount
      book {
        id
        title
        author {
          id
          name
        }
      }
    }
  }
}

Fortunately, Mesh allows an extra directive-based configuration to provide a "batching" query that will help resolve many record of the same type.

Our current resolver configuration for Book.author is the following:

.meshrc.yaml (opens in a new tab)

.meshrc.yaml
sources:
  # …
transforms:
  # …
additionalTypeDefs: |
  # …
  extend type Book {
    author: authors_v1_Author @resolveTo(
      sourceName: "Authors", # Which source does the target field belong to?
      sourceTypeName: "Query", # Which root type does the target field belong to?
      sourceFieldName: "authors_v1_AuthorsService_GetAuthor", # What is the source field name?
      requiredSelectionSet: "{ authorId }",
      # What args does this need to take?
      sourceArgs: {
        "input.id": "{root.authorId}"
      }
    )
  }

Assuming that the "Authors" is exposing a authors_v1_AuthorsService_GetAuthors(input: authors_v1_GetAuthorsRequest_Input) with authors_v1_GetAuthorsRequest_Input being:

input authors_v1_GetAuthorsRequest_Input {
  ids: [String!]!
}

We could update our .meshrc.yaml configuration as follows:

.meshrc.yaml
sources:
  # …
transforms:
  # …
 
# Create a resolver with batching to solve N+1 problem
additionalTypeDefs: |
  # …
  extend type Book {
    author: authors_v1_Author @resolveTo(
      targetTypeName: "Book",
      targetFieldName: "author",
      sourceFieldName: "authors_v1_AuthorsService_GetAuthors",
      keyField: "authorId",
      keysArg: "input.ids"
    )
  }

requiredSelectionSet and sourceArgs got replaced by keyField and keysArg:

  • keysArg provides the name of the batching primary key argument (input.ids from authors_v1_GetAuthorsRequest_Input)
  • keyField indicates which Book selection-set field should be used to provide the ids value

Now, our Mesh Gateway will try to batch calls to the "Authors" API when resolving Book.author: