蜜豆视频

GraphQL queries

This is part 2 of the series for GraphQL and 蜜豆视频 Commerce. In this tutorial and video, learn about GraphQL queries and how to perform them against 蜜豆视频 Commerce.

video poster

Transcript
The easiest way to understand the basics of GraphQL is to look at some simple examples. So what I have here are some examples that I鈥檓 looking at in the graphical GraphQL client called Altair, A-L-T-A-I-R, and this provides a lot of good tools for testing and prototyping GraphQL queries against your endpoints. And we鈥檙e specifically going to be looking at some example queries that match schema found in 蜜豆视频 Commerce. I am issuing requests to the vinia.magento.com example implementation. So the Vinia theme is the reference implementation of a PWA build with PWA Studio tools, and I鈥檓 hitting the GraphQL endpoint here. The query text that you see here in this pane is being sent in the body of my HTTP request, or alternately can be encoded into the query string. And this provides a good example of what a GraphQL query looks like and the syntax involved. So the first thing to see here is that I鈥檓 querying a data type called categories, or this could also be thought of as a field on the query type in whatever data graph exists on the server that I鈥檓 querying. And within the definition of categories in my query here, I have a sub-selection of fields, and I鈥檓 selecting one field called items, and then I have sub-selections of fields within that. So this is a whole tree of nested fields wherein I am expressing exactly the data that I want to get back from the server. So within items, for each item that I get back, I say that I want the name and an item called products. Within products, I have another sub-selection saying that I want a field called items, and then for each item in there, I want a field called skew. There鈥檚 a lot more data, a lot more fields available on the server that I could be querying, but this syntax lets me express exactly the fields that I want to get back, and I鈥檓 only going to get those fields back in the response. What we see here for categories also shows arguments that I鈥檓 passing in. So to the categories query, I鈥檓 passing in a filters argument that, again, has an expression of some nested fields where I鈥檓 passing in the search term that I want to match in my categories query. And then I鈥檓 also passing in arguments to one of the sub-fields, one of the fields in my sub-selection here, the products field, passing in arguments about the page size and the page of results that I want back of the categories products. Now a very significant detail if we scroll down here is that in the very same request, I鈥檓 also making a request for this query or data type or top-level field called products. So quite separately from my request for a list of categories, I am also requesting a list of products matching a certain search term and also expressing the nested fields that I want from that. And I can include any available queries in the data graph or the schema that exists on the 蜜豆视频 Commerce endpoint that I鈥檓 hitting. I can include any queries in this same request and get back information for those different data types. They don鈥檛 have to be related in any way. Now I鈥檓 wrapping all of this in braces that I鈥檓 also attaching an operation name to. I have this query keyword and I鈥檓 just giving an arbitrary operation name that can be anything here. And I鈥檓 doing that for the purposes of including variables. So the arguments that I鈥檓 passing at various points in my query here, I鈥檓 actually just passing in placeholders matching variables in this wrapper. And that is because GraphQL has first-class support for separating user input from the actual syntax or the actual text of the GraphQL query itself. And here in the Altair client, I have a pane for variables where I can actually include the values that I want substituted in for those different variable names in my query. So if I send this request and look at the response that I get back, the important thing is how similar the format of the response is to the format of my query. So I was able to be very expressive with saying the fields that I wanted back and then the format of the data that I get back very closely mirrors that format and makes it very easy for me to see what I鈥檓 getting here. So I said that for categories I wanted to get items and within that I wanted name and products. Then we see that exact same hierarchy here in the return data categories, items, and the list of items with each one having names and products. And then the tree, the nested tree, just continues to go from there. We see for the list of products in that category I have the items with the skew of each. And if I come all the way down here I also have the results of my products query with the list of products and the specific fields that I requested for those. So this is kind of the expressive syntax that GraphQL allows that let us just get any data that we want in the same request and allows us to express the fields and the specific points of data that we want to get back so that we鈥檙e not wasting any resources or wasting anything in the size of our requests and our responses.

GraphQL Syntax example

Let鈥檚 dive right into GraphQL query syntax with a full-fledged example. (Remember, you can try this yourself against https://venia.magento.com/graphql.)

Observe the following GraphQL query, which is broken down piece by piece:

{
    country (
        id: "US"
    ) {
        id
        full_name_english
    }

    categories(
        filters: {
            name: {
                match: "Tops"
            }
        }
    ) {
        items {
            name
            products(
                pageSize: 10,
                currentPage: 2
            ) {
                items {
                    sku
                }
            }
        }
    }
}

A plausible response from a GraphQL server for the above query could be:

{
  "data": {
    "country": {
      "id": "US",
      "full_name_english": "United States"
    },
    "categories": {
      "items": [
        {
          "name": "Tops",
          "products": {
            "items": [
              {
                "sku": "VSW06"
              },
              {
                "sku": "VT06"
              },
              {
                "sku": "VSW07"
              },
              {
                "sku": "VT07"
              },
              {
                "sku": "VSW08"
              },
              {
                "sku": "VT08"
              },
              {
                "sku": "VSW09"
              },
              {
                "sku": "VT09"
              },
              {
                "sku": "VSW10"
              },
              {
                "sku": "VT10"
              }
            ]
          }
        }
      ]
    }
  }
}

The above example relies on the out-of-the-box GraphQL schema for 蜜豆视频 Commerce, defined at the server. In this request, you
query multiple types of data at once. The query expresses exactly the fields that you want, and the returned data is formatted
similarly to the query itself.

NOTE
GraphQL clients obfuscate the form of the actual HTTP request being sent, but this is easy to discover. If you鈥檙e using a browser-based client, observe the Network tab when a query is sent. You see that the request contains a raw body consisting of 鈥渜uery: {string}鈥, where {string} is simply the raw string of your entire query. If the request is being sent as a GET, the query might be encoded in the query string parameter 鈥渜uery鈥 instead. Unlike with REST, the HTTP request type doesn鈥檛 matter, only the contents of the query.

Querying for what you want

country and categories in the example represent two different 鈥渜ueries,鈥 for two different kinds of data. Unlike a traditional API paradigm like REST, which would define separate and explicit endpoints for each data type. GraphQL gives you the flexibility to query a single endpoint with an expression that can fetch many types of data at once.

Likewise, the query specifies exactly the fields that are desired for both country (id and full_name_english) and categories (items, which itself has a subselection of fields), and the data you receive back mirrors that field specification. There are presumably many more fields available for these data types, but you get back only what you requested.

NOTE
You may notice that the return value for items is actually an array of values, but you are nevertheless directly selecting subfields for it. When a field鈥檚 type is a list, GraphQL implicitly understands subselections to apply to each item in the list.

Arguments

While the fields you want returned are specified within the braces of each type, named arguments and values for them are specified within parentheses after the type name. Arguments are often optional and often affect the way query results are filtered, formatted, or otherwise transformed.

You are passing an id argument to country, specifying the particular country to query, and a filters argument for categories.

Fields all the way down

While you might tend to think of country and categories as separate queries or entities, the entire tree expressed in your query actually consists of nothing but fields. The expression of products is syntactically no different from that of categories. Both are fields, and there is no difference between their construction.

Any GraphQL data graph has a single 鈥渞oot鈥 type (typically referred to Query) to start the tree, and the types often considered to be entities are assigned to fields on this root. The example query is actually making one generic query for the root type and selecting the fields country and categories. It is then selecting subfields of those fields, and so on, potentially several levels deep. Wherever the return type of a field is a complex type (for example, one with its own fields, rather than a scalar type), continue to select the fields you want.

This concept of nested fields is also why you can pass arguments for products (pageSize and currentPage) in the same way you did for the top-level categories field.

GraphQL Field Tree

Variables

Let鈥檚 try a different query:

query getProducts(
    $search: String
) {
    products(
        search: $search
    ) {
        items {
            ...productDetails
            related_products {
                ...productDetails
            }
        }
    }
}

fragment productDetails on ProductInterface {
    sku
    name
}

The first thing to note is the added the keyword query before the opening brace of the query, along with an operation name (getProducts). This operation name is arbitrary; it doesn鈥檛 correspond to anything in the server schema. This syntax was added to support the introduction of variables.

In the previous query, you hard-coded values for the arguments of your fields directly, as strings or integers. The GraphQL specification, however, has first-class support for separating user input from the main query using variables.

In the new query, you are using parentheses before the opening brace of the entire query to define a $search variable (variables always use the dollar sign prefix syntax). It is this variable that is being provided to the search argument for products.

When a query contains variables, the GraphQL request is expected to include a separate JSON-encoded dictionary of values alongside the query itself. For the query above, you might send the following JSON of variable values in addition to the query body:

{
    "search": "VT01"
}
NOTE
If you鈥檙e trying out these queries against the Venia example site rather than your own 蜜豆视频 Commerce instance, the returned results are likely to be empty for related_products.

In any GraphQL-aware client that you are using for testing (such as Altair and GraphiQL), the UI supports entering the variables JSON separately from the query.

Just as you saw that the actual HTTP request for a GraphQL query contains 鈥渜uery: {string}鈥 in its body, any request containing a variables dictionary simply includes an additional 鈥渧ariables: {json}鈥 in that same body, where {json} is the JSON string with the variable values.

The new query also uses a fragment (productDetails) to reuse the same field selection in multiple places. in the GraphQL documentation.

Useful GraphQL resources

recommendation-more-help
3a5f7e19-f383-4af8-8983-d01154c1402f