Field Usage in GraphOS
Understand field usage metrics including which clients and operations request which fields and how often
You can analyze the field metrics sent by your graph to GraphOS from any variant's Insights page in GraphOS Studio
:The Insights page presents operation and field metrics. It has two main sections:
The collapsible left sidebar, where you can search, filter, and sort operations and fields
The main insights view, which displays overall operations metrics by default, or field information and usage metrics for a single field once you select it from the left sidebar
This guide focuses on field metrics and information. Refer to Operation metrics in GraphOS for more information on the other metrics on this page, including resolver-level traces.
Field details
When you click a field name, you open detailed information for that field in the main insights view. This includes when the field first and last received traffic, any applied tags, and the subgraphs that provide the field.
Below that, you can view the field definitions within your subgraph schema.
Next, you see which clients and operations contribute to the field's usage. Each row in the Clients & Operations table shows a client that requested the field, the number of operations that used the field the client made, and the total number of requests the client made. Selecting a client shows more details about the operations that requested and executed the field.
Below Clients & Operations, you can view visualizations for the field's request rate, latency distribution, and error metrics.
Field requests and executions
Each row in the Clients & Operations table displays some combination of the following metrics for each field, depending on which data you report to GraphOS:
Metric | Description |
---|---|
Client versions | How many different client versions have sent the operation |
Requests | How many operations sent by clients over a given time period included the field, according to metrics provided by your servers |
Executions | How many times your servers have executed the resolver for the field over a given time period |
For each of these columns to appear on the Insights page, your GraphQL servers must report the associated metrics to GraphOS. If some but not all your GraphQL servers report this data, the Insights page presents an incomplete picture of your graph's field usage.
Take a look at the Requests and Executions for an example field:
As you can see, they can differ significantly. There are many possible reasons for this, described below.
Objects in lists
Let's say a client executes the following query one time:
1query GetBooks {
2 books {
3 title
4 }
5}
If Query.books
returns a list of ten Book
objects, then Book.title
is resolved ten times. This query, therefore contributes just one request but ten executions to the Book.title
field.
Multiple references to a field
Let's say a client executes the following query one time:
1query GetTwoBooks {
2 firstBook: book(id: "123") {
3 title
4 }
5 secondBook: book(id: "345") {
6 title
7 }
8}
This operation includes two references to the fields Query.book
and Book.title
. Therefore, the resolvers for these fields each execute twice (assuming Query.book
doesn't return null
). However, these multiple references are all part of a single operation.
Therefore, this query contributes just one request but two executions for each of the Query.book
and Book.title
fields.
Fields that return interfaces
Let's say our GraphQL server's schema defines the following interface and object types:
1interface Media {
2 title: String!
3}
4
5type Book implements Media {
6 title: String!
7 author: String!
8}
9
10type Query {
11 favoriteMedia: Media!
12}
Now, let's say a client executes the following query:
1query GetFavoriteMedia {
2 favoriteMedia {
3 title
4 }
5}
If Query.favoriteMedia
returns a Book
object here, then Book.title
is resolved one time. However, the original query does not reference Book.title
. Instead, it references Media.title
, because Query.favoriteMedia
has a return type of Media
.
Therefore, this query contributes zero requests and one execution for Book.title
. It also contributes one request for Media.title
. Note that interface fields always have zero executions.
Requested fields that aren't resolved
Let's say a client executes the following query one time:
1query GetLoggedInUser {
2 loggedInUser {
3 name
4 }
5}
Now, let's say Query.loggedInUser
returns null
because no user is logged in. In this case, the resolver for User.name
never executes, because its parent returns null
. Therefore, this query contributes one request and zero executions for User.name
.
A requested field might not be resolved for any of these reasons:
The field is nested under a field that returns
null
, as shown above.The field is nested under a field that returns a list, but the list is empty.
The field is part of a fragment that doesn't apply to a particular object.
The resolver is skipped due to a
@skip
or@include
directive.
@key
and @requires
fields in a federated graph
Let's say our federated graph includes these two subgraph schemas:
1type Product @key(fields: "id") {
2 id: ID!
3 name: String!
4}
1extend type Product @key(fields: "id") {
2 id: ID! @external
3}
4
5type Review {
6 id: ID!
7 score: Int!
8 product: Product!
9}
10
11type Query {
12 reviews: [Review!]!
13}
Now, let's say a client executes the following query:
1query GetAllReviews {
2 reviews {
3 score
4 product {
5 name
6 }
7 }
8}
This query's execution starts in the Reviews subgraph, but it needs to obtain each Product
's name
from the Products subgraph. As part of this process, the Products subgraph must resolve references to Product
s provided by the Reviews subgraph.
To help resolve these references, the Reviews subgraph must return each Product
's id
field, even though that field isn't included in the original query. This is because id
is a @key
field for Product
.
Therefore, this query contributes zero requests and one execution for Product.id
. Similar logic applies to fields that use the @requires
directive.