GraphQL won’t solve your performance problems, but @defer might help

@defer has been long discussed within the GraphQL working group, and while not part of the specification yet, it’s an exciting new addition that may help with your application’s user experience and mask performance problems.

Rate this content
Bookmark
Slides
Video Summary and Transcription
The talk delves into the benefits and complexities of using the defer directive in GraphQL. The defer directive is a tool that allows clients to specify which parts of a query can be delivered incrementally, enhancing the user experience by enabling partial data rendering. This is particularly useful for handling slow fields in GraphQL, as it allows for a quicker time to first byte. Defer is not yet part of the official GraphQL specification, but it is already being used in production by several companies. It complements other solutions like query splitting and prefetching by addressing high latency fields more effectively. When using defer, it's important to consider error handling and nullability, as the behavior of errors can differ from traditional GraphQL queries. Apollo's ecosystem supports the defer directive, making it easier for developers to implement and test it within their applications. Additionally, the talk highlights the potential of using defer with GraphQL mutations, allowing for partial results during long-running operations.

FAQ

As of now, the defer directive is not part of the official GraphQL specification. However, it is being used in production by several companies and is under consideration by a working group with a publicly accessible RFC.

Yes, the defer directive can be used with GraphQL mutations. It allows for partial results to be returned during long-running operations, such as payment processing, enabling a better user experience by showing incremental updates.

Alternatives to the defer directive include query splitting and prefetching. Query splitting involves breaking a single query into multiple queries, while prefetching involves requesting data ahead of time based on user interactions. Both have their own trade-offs and limitations.

No, the defer directive does not improve server-side performance. It is designed to enhance the user experience by allowing partial rendering of data. Server-side performance issues should be addressed separately.

The defer directive is a tool used by clients to inform GraphQL servers that parts of a query can be delivered incrementally, allowing certain fields to be deferred and returned later. This can improve user experience by enabling partial rendering of data.

Apollo has introduced support for the defer directive in their React and Client libraries, with iOS support coming soon. Apollo Studio and Apollo Sandbox can be used to test defer. For federated graphs, Apollo Router can be used to manage asynchronous data fetching without requiring changes to subgraphs.

Prefetching in GraphQL involves requesting data ahead of time based on assumed user interactions. For example, fetching a user's payment information before they reach the checkout page. This can mitigate performance concerns but requires servers to handle additional load for data that may not be used immediately.

Query splitting involves breaking a single GraphQL query into multiple queries, each handled separately. Clients must orchestrate the responses, and servers need to manage multiple unique queries, which can add overhead and complexity.

The defer directive improves user experience by allowing partial rendering of data. This means users can start interacting with parts of the data immediately while waiting for the rest to load, reducing the overall latency perceived by the user.

When using defer, it's important to handle partial error states carefully. Errors in deferred fields should be codified in the schema whenever possible to provide a cleaner client experience. Clients should also be prepared to handle partial data and not fail entirely if one piece of data fails to load.

1. Introduction to the Defer Directive#

Short description:

Hello, everybody. My name is Lucas Ledbetter and I'm a solutions architect here at Apollo. I'm extremely excited to be here today, at least virtually, to talk about the defer directive and what it can mean for your clients. GraphQL requires the entire body be sent as a single response. Some fields require additional time to resolve. The client should be able to partially render data as it comes in.

Hello, everybody. My name is Lucas Ledbetter and I'm a solutions architect here at Apollo. I'm extremely excited to be here today, at least virtually, to talk about the defer directive and what it can mean for your clients.

To get the full picture, we'll need to cover a few background details before diving into defer actually is. Lastly, I'll give a few examples showing where defer can be an incredibly powerful tool in your arsenal to help your clients give a better user experience.

So, to take a step back, you might be aware that GraphQL requires the entire body be sent as a single response. This is great for most queries as your UI expects all data to be there. You certainly wouldn't want to return only some of the data for a user on a user profile page, for example. As your graph grows, you realize not all fields are built the same. Some fields require additional time to resolve for one reason or another. For many organizations, this is simply due to scale. But for others, it could be tied to a third-party service that they may not have full control over, thus upholding the other parties response times, or a number of other reasons. Regardless of why it takes more time, they realize that the request for those fields cause the entire response to slow down. As clients begin to access those fields, they start to see the overall graph latency increasing as a result, which isn't ideal and means the user experience is impacted. The client should be able to partially render data as it comes in. But it is unfortunately not possible with a single request.

2. Different Approaches to Address the Problem#

Short description:

Companies came up with two different approaches to solve this problem. The first solution is query splitting, which required clients to orchestrate multiple queries. This addressed the weakest link problem in traditional GraphQL but had performance and context limitations. Prefetching became another popular approach, making assumptions about user intention and requesting data ahead of time. However, this placed additional load on servers and relied on informed guesswork.

So to solve this growing problem, companies came up with two different approaches. Both addressed some of the problems, but each came with tradeoffs. The first solution we'll cover is query splitting. This solution was to split the single query into multiple queries and required clients to orchestrate the responses. This sounds a lot like one of the problems that GraphQL was built to solve for. However, this addressed the weakest link problem that plagued traditional GraphQL, until performance problems can be mitigated or addressed on the server.

In addition to having to handle the responses on the client, it also required the server to handle multiple unique queries from the clients, each with traditional HTTP and GraphQL overhead, as well as being unable to use already fetched data such as apparent data type. Losing that additional context, especially for fields needing to make many calls to other services, could be quite expensive.

On the other hand, prefetching became another popular way to address this problem. For requesting data ahead of time, it is possible to mitigate performance concerns by making assumptions about the user's intention. For example, grabbing the user's payment information prior to hitting the checkout page. This had the added benefit of it still being a single request. However, the user's experience in the client doesn't change much, if at all, when it works. As might be obvious, servers are required to handle this additional load for assumed interaction, even if the user never interacted with the returned data, making this a mostly informed guesswork.

3. Understanding the Defer Directive in GraphQL#

Short description:

Both of these solutions made progress towards the end goal of a good user experience, but never got there all the way. That takes us to the defer directive. As you might have guessed, defer is a possible solution to the ongoing problem of higher latency fields while still having a single response. Defer is a directive used by clients to inform GraphQL servers that the requests can be delivered incrementally by specifying parts of a query that were returned later. As of today, this directive is not part of the official GraphQL specification. However, it's been used by a number of companies in production already. It is currently under the working group with a publicly accessible RFC which covers ins and outs of the directive and I'll have a link to the RFC at the end of the presentation. By making defer indicated by the client instead of the server, we can now let the clients inform the server of what fields can be responded to later and which ones cannot. So how does this directive actually work? Is it magic? Well, it's not magic, but it might seem like it after working with it for the first time. We'll be covering this at a high level today for the sake of time, but there are excellent talks and posts covering the more technical aspects of the directive, as well as the RFC from the Working Group. So consider the basic schema for a fictional social media site, which has a query to return a list of users, as well as a user type with a list of friends. In this example, the company running this GraphQL schema notices that the friends field isn't as performant as it probably should be. Now, given the query on the left to fetch the list of users along with their friends, we send the requests and it takes a second for the entire response. This isn't great, and as a client developer, we could render the user data partially before rendering the list of friends for each user.

Both of these solutions made progress towards the end goal of a good user experience, but never got there all the way. That takes us to the defer directive. As you might have guessed, defer is a possible solution to the ongoing problem of higher latency fields while still having a single response.

The first question you might ask yourself is, what is this defer directive thing? And it's a great question. Defer is a directive used by clients to inform GraphQL servers that the requests can be delivered incrementally by specifying parts of a query that were returned later. Or in other words, by deferring the response.

As of today, this directive is not part of the official GraphQL specification. However, it's been used by a number of companies in production already. It is currently under the working group with a publicly accessible RFC which covers ins and outs of the directive and I'll have a link to the RFC at the end of the presentation.

As a quick sidebar, we've all seen directives used in server-side applications. From formatting information, to Apollo Federation notation, and more. But the GraphQL specification also calls out the ability to use them in the query itself. In general, I'm a big proponent of executable directives or client-side directives. They allow for clients to provide additional context to defer use and processing, and not simply notation in the schema about expected behavior as with server-side directives. We've seen client-side directives in the past with things like skip and include as defined in the specification and defer is much the same.

So why do I bring this up? By making defer indicated by the client instead of the server, we can now let the clients inform the server of what fields can be responded to later and which ones cannot. Now the requests are the same. It can be located for portions of a query in one component, but that same query may be returned as one response in another component. That context is hard to convey without defer. So now back to regular scheduled programming.

So how does this directive actually work? Is it magic? Well, it's not magic, but it might seem like it after working with it for the first time. We'll be covering this at a high level today for the sake of time, but there are excellent talks and posts covering the more technical aspects of the directive, as well as the RFC from the Working Group.

So consider the basic schema for a fictional social media site, which has a query to return a list of users, as well as a user type with a list of friends. In this example, the company running this GraphQL schema notices that the friends field isn't as performant as it probably should be. I want to explicitly call out something here, however. The friends field, while returning a list of non-null users, is itself nullable. This is important and is something we'll touch on in a bit as to why that's the case.

Now, given the query on the left to fetch the list of users along with their friends, we send the requests and it takes a second for the entire response. This isn't great, and as a client developer, we could render the user data partially before rendering the list of friends for each user. On a user profile, we probably render their username, contact information, and more.

4. Using the Defer Directive in Queries#

Short description:

We've modified the query to include the new deferred directive. The multi-part response includes a has-next field to indicate more responses and an incremental field for additional data. The response is shown in a gif, with an immediate response in about 8 milliseconds and the remainder coming later.

Well, the friends list can load when it arrives, instead of blocking the entire rendering process. So, we've now modified the query to include the new deferred directive. You first might note that the directive is on an inline fragment. This is because the current RFC states it must be included only on fragments or inline fragments. Now, if you execute this query, we get back a multi-part response shown on the left. The first response has a new field but otherwise contains the same structure as the normal GraphQL response, subsequent responses have a new format. While it's not critical to understand all of this, there are two parts I'd like to highlight. First, each response has the has-next field indicated in blue which indicates to the client that are more responses coming. Once it's false, the entire response has been sent. Each subsequent response has an incremental field which contains the Delta and path to the additional data. As you can see, the additional info here is inserted into the user's array at index 0. Now, to put it together, here's a short gif of the results. You can see we get back an immediate response in about 8 milliseconds and then a second later we get the remainder. If you're curious, the gif is from a Apollo sandbox and supports Defer out of the box now including a nice timeline to scroll through the various states of the response and could be a helpful tool to understand how your deferred response is happening.

5. Understanding the Importance of Performance#

Short description:

So as you begin to think about defer, you might notice that it doesn't actually improve performance. But to really dive into how it can help, you must first ask yourself, Why do you care about performance? This is a question I have to ask a number of companies to answer for one reason or another. For some, it's measuring how third parties are performing within their graph. For others, it's about how the user experience is going. Most of the time, however, the reality is that users are the most impacted by performing graphs.

So as you begin to think about defer, you might notice that it doesn't actually improve performance. You're still waiting for a response after all. But to really dive into how it can help, you must first ask yourself, Why do you care about performance? This is a question I have to ask a number of companies to answer for one reason or another. For some, it's measuring how third parties are performing within their graph. For others, it's about how the user experience is going. Most of the time, however, the reality is that users are the most impacted by performing graphs. Many client teams measure against time-to-first-byte, time-to-interactivity, payload size, or any number of other metrics. But Defer specifically targets time to first byte and can intersist with time-to-interactivity, depending on how it's implemented.

6. Examples of Defer in Action#

Short description:

Let's explore examples of Defer in action and its impact on user experience. We'll start with a straightforward way to use Defer for partial rendering or lazy loading. By adding additional fields to the schema, we can render the user's profile information first and load friends later.

So let's look at a few examples of Defer in action and talk about how they can improve the user experience. We'll first want to look at a relatively straightforward way to leverage Defer as a way to do partial rendering or lazy loading. Going back to our previous schema, we have now added a few more fields to the fictional social media site. As mentioned earlier, we can properly render the user's profile information quickly before adding in friends later. Now, we'll see the two queries side by side. Note that we've added the avatar URL, username, and title to the base query in both since we need that right away. Otherwise, we'll defer the friends list, same as before.

7. Using Defer Directive for UX Improvements#

Short description:

Now, excuse my terrible mock-up. In the normal query, we're blocked by the lowest fields, which is the friends field, and we can't render any other data until it's finished. But you can see right away that we get the profile information in the deferred requests and can clearly see that we get our data quickly. Our time to first byte has dropped significantly, and probably even more importantly, time to error activity has decreased as well. Now, going a step further, we've added parameters to our friends field to specify the number of friends to show and an offset. Using defer and GraphQL aliasing, we can load the first 10 friends quickly and then offset the remaining friends in the deferred statement. These optimizations are enabled by using defer. It's also possible to have more than one defer or even nesting defers. In this example, we have a user with a company profile associated with it. We can defer each step to start returning data along the path, rendering data as it comes in. Now let's consider some other potential UX improvements. Fetching data is straightforward, but what about changing data? Mutations are in GraphQL as well.

Now, excuse my terrible mock-up. I'm most certainly not a UX designer. In the normal query, we're blocked by the lowest fields, which is in the case the friends field, and we can't render any other data until it's finished. But you can see right away that we get the profile information in the deferred requests and can clearly see that we get our data quickly. Our time to first byte has dropped significantly, and probably even more importantly, time to error activity has decreased as well. The user can see and engage with parts of the data before the full response is sent.

Now, going a step further, perhaps our company identified that the friend field wasn't, or slowed down based on the number of friends selected for whatever reason. There's another potential path we could take. We've now added two new parameters to our friends field in the schema shown on the left to allow us to specify the number of friends to show and an offset. Using defer as well as GraphQL aliasing, we can now load the first 10 friends quickly and then offset the remaining friends in the deferred statement. Traditionally, this would consist of multiple queries to fetch the other friends to render, allowing the server to run these queries in parallel gives other performance benefits beyond simple UX improvements. In fact, this is quite similar to the stream directive, which is a part of the same working group as the defer directive, but falls a bit out of scope for today's discussion. In summary, it uses much of the same patterns as defer to send large lists incrementally with an initial list of items sent across. Each subsequent chunk in the multi-part response would then contain additional entries to the list until finished. There are a number of great talks and documents about Stream and I highly recommend taking a look at it if taking a look at the RFC if for that directive if sending large lists is a concern for your organization. These sorts of optimizations are enabled by using defer. And so as you go down the path of considering defer as a solution, consider how you might also leverage other aspects of the GraphQL to layer features together. You'll likely realize you can create solutions with deferred returns that weren't previously possible and can help with improving fight side performance. Note that's also possible to have more than one defer or even nesting defers. In this example, we have a user which has a company profile associated with it. That company also lists their owner. In this case, it's quite possible that the user's service communicates to the company service, which then goes back to the user service to resolve the owner information. These hops take time. And in this case, we can defer each step to start returning data along the path, rendering data as it comes in. It's quite possible that the company name isn't immediately critical on a profile and the list of owner isn't probably important at all.

Now let's consider some other potential UX improvements. Fetching data is pretty straightforward. We can all envision how to do partial loading since there was a solve prior to defer. We'd simply use multiple queries or queries per component in React. But what about changing data? Mutations are in GraphQL as well.

8. Defer Directive and Mutations#

Short description:

Defer supports mutations, allowing for the return of a transactional ID that can be used later. Errors are a fact of life, and when they occur during a deferred statement, they are similar to JavaScript promises. Partial error states should be considered, and retry strategies should be evaluated on a case-by-case basis. Codifying expected error states in the schema can create a cleaner experience for client teams.

And thankfully, defer supports mutations as well. As an example, here's a sample mutation for a fictional payment service. Breaking this down, it returns a payment ID and a user as well as payment status interface. This interface either indicates in success or failure states. As a side note, I personally love codifying errors and schemas for things just like this. And we'll actually touch on errors in just a moment.

Diving a bit deeper into this for folks that work with e-commerce, you know that payments can take a while. Defer affords us the ability to return a transactional ID, which we can use for later, such as the user refreshing the processing and otherwise show a loading screen. Once we've gotten a result from the payment status field, we can then render appropriately. Either six success in which case we have the build amount or it's a failure in which case we have failure. Or in which case we have the message or reason in this case. From a user perspective however, we could probably render the right result without subscription or polling to get an updated status. Instead, we use the existing HTTP connection to forward the information to our clients for use later.

And as much as I love talking about codifying errors in your schema and shout out to Saucer Solomon for her excellent talk on the subject. Errors are a fact of life. As much as we strive to build error-free code codify all the various error states in our schema and make tests to avoid them, and as we start discussing performance, errors are a natural topic of discussion since they're also a measurement of success and user experience. With that in mind, you might wonder what happens when an error occurs during a deferred statement? Well, to see we have this certainly not at all suspect query to return definitely not an error in a deferred state. I mean, who could have seen that coming? But in all seriousness, the first errors on defers are strange. Best analogy I have is that they're much like JavaScript promises. You're pushing the result to come at some undefined time in the future and have to check to make sure that's not rejected or errored in this case. So for the client teams watching this, it requires you to think about partial error states as well more carefully. Previously, you might retry the whole request if you saw an error, but in the world of FUR you might have a success state initially, but an error state laying requests. So you should consider your retry strategy on a case by case basis. To demonstrate this, on the right, we have a GiF of a policy inbox once again, you notice immediately we don't have any errors and then one comes through and matches the format we already saw. Looking through the timeline, we can see the first response was excellent, no errors, but the second reply does have an error and six fans with multiple defers. If you have two defers certain statements error, you get multiple errors in the error key as you would in a normal graphical response. And to briefly touch on some considerations you should have as you use defer around errors, we recommend codifying any expected error, failure states such as user not found into your schema whenever possible. This creates a cleaner experience for your client teams when they get failure states and it a first fragment, avoiding them having to parse through errors. When you can't however, expect everything in the kitchen sink. You already handle errors within your client by checking throughout your deferred responses to see if you run into errors.

9. Handling Partial Data and Error Behavior#

Short description:

Lastly, if you have a partial data, use that partial data and don't kick a user off the page if one piece fails to load. Defers supports both null and non-null fields, but the behavior of errors differs depending on how you use defer and how the field is typed.

Lastly, if you have a partial data, use that partial data and don't kick a user off the page if one piece fails to load. As obvious as this sounds, defer is best used for lower priority data, things that can wait to be returned and as such retrying or other mechanisms can help make sure the user experience is still excellent. And if you recall earlier, I explicitly called out their social media sites friends field was nullable. Defers supports both null and non-null fields, but the behavior of errors differs depending on how you use defer and how the field is typed. This differs from traditional GraphQL. So it's important information to know. Our friends list was a nullable field with a non-null list of users. The query used before is great. It likely would, it's likely that the request would be an all or nothing request where if we fail to get the friends for one reason or another, the field will remain null as you can see here.

10. Understanding Defer Behavior and Implementation#

Short description:

In a normal query, if the friends field is null, we get an empty data object. However, with deferred statements, we receive a partial response without the friends field. The same applies when deferring specific fields. Be careful with nullability as defer introduces additional complexity. Apollo supports defer in their clients, making it easy to use in practice. Apollo Studio and Apollo Sandbox are great tools for testing defer. Apollo has also released a version of defer for federated graphs, which doesn't require additional support. If you're using federation, consider checking out defer. The implementation is agnostic of the library. Adding federation and entity declaration can enable defer without much code change on your server.

Great, now what happens if the friends field was null, non-null? In a normal query, we get back an empty data object since a non-null field returned null. Now how about deferred statements? You might expect that the same would happen. However, we get back a partial response without the friends field but including the ID field. This is an important aspect to consider when dealing with deferring clients. The error behavior changes from traditional GraphQL and you must be aware of this change when migrating to using defer in your queries.

The same also applies when deferring specific fields. For example, you have a query to get a user and their avatar URL, which comes from a service that isn't 100% reliable. This field needs a response, however. So we marked it as non-null so clients can expect it to exist. Same as before, query on the left is not deferred. And as expected, we get back an empty data object. Now for the defer, we get back the partial object again. So all of this in mind, my recommendation is this, be careful with nullability as defer introduces additional complexity when it comes to the response. Just be aware that this exists as it currently stands in the RFC and you'll be okay.

And while we talk about what defer is and how you can use it in principle, to use it in practice is thankfully fairly straightforward. Apollo recently announced support for defer in both their React and Clon clients with iOS support coming soon. For those considering defer, this can be a massive help if you're currently using Apollo's clients since defer can plug into existing applications without headache-inducing rewrites. And once you have a client that supports defer, you simply need to rewrite your queries with defer in mind. As mentioned earlier, Apollo Studio and Apollo Sandbox can be great ways to test defer and see how it works on your graph and ensuring what it does what you expect. In fact, that's what you're seeing on the right. It's the nested defer example from earlier. Apollo has also released a version of defer for organizations using a federated graph and this is especially unique since it doesn't require any support beyond being able to resolve federated entities. I won't have time to dive in deep here on this, but the short version is this. If you're using federation and have even a remote interest in defer and what can afford for your clients, check it out. The implementation is agnostic of library. So even if your subgraph library doesn't support it out of the box, this can enable it so long as you're using Apollo router to serve your super graph. To summarize how it does this, the router leverages the existing entity relationships and manages the asynchronous process of getting the data for you. So subgraphs are not needed to handle sending the responses as multi-part as all. If you don't currently have a federated environment, adding federation and entity declaration, which is just syntax is a simple way to utilize the router as a way to get defer without much code change on your server. We've seen how great defer can be.

11. Considerations and Resources#

Short description:

Consider the experiences you drive currently and what use cases might benefit from being deferred. Defer is driven by the client and is best used for lower priority data. It won't solve your server-side performance concerns. Defer is a powerful tool for your consumers and should be leveraged if possible. Get involved in the graph community and provide feedback. Resources: RFC, social solomon's talk, 200 okay talk, Apollo React client docs, Apollo Router Federation defer support. Thank you for being here.

It's also a lot of fairly complex problems in a declarative manner playing to the strengths of GraphQL. But at the same time, consider the experiences you drive currently and what use cases might benefit from being deferred. We've seen examples of how to do this. One, whereas drop-in fixed for the team and another was schema needed to accommodate defer to provide a better client experience. Keeping in mind those experiences can ensure your schema remains expressive while still enabling defer as an option to improve user experience.

Finally, as mentioned before, defer is driven by the client. They will be most aware of what use cases can be driven by defer. We've seen that defer is best used for lower priority data. So simply stuffing a query full of defers will provide little benefit. Make sure defer is used carefully and ensure it makes sense to do so when you do use it. And while you can improve the user experience and clients and performance defer isn't a panacea. It won't solve your server side performance concerns and it should be treated as such. Identifying and resolving server side performance issues can also remove some of the need for defer. So doing both can help your graph seem and be more performant for clients and servers.

And so a few parting thoughts around this. First defer is a powerful tool for your consumers to your graph and should be leveraged if possible. There are a menagerie of ways to improve your user experience using it. And to that end while specification is new there are a few libraries already supported and Apollo recently announced support for organizations using a federated graph as mentioned previously. And my last point today is this. If you are interested in defer but it doesn't meet your requirements or you feel like it can be improved, get involved. The graph community is we dot are deep and wide and the working groups can always use more feedback about specific use cases concerns or more. Get actively included and if you can. Lastly, just some resources as mentioned, we have the link to the RFC social solomon's talk the 200 okay talk about codifying errors in your schema docs for both Apollo react client which was used to demo defer earlier as well as the Apollo router Federation defer support. With that, thank you. You can reach me at elevator on Twitter and that's my email Lucas at Apollo graphical.com. Hi everyone. Nice to meet everyone. Thanks for joining today's topic. Awesome. Thank you so much for being here.

QnA

Using Defer Directive and Other Approaches#

Short description:

So just a reminder to everyone you can join the Andromeda Q and a channel on discord to ask your questions for Lucas here in this Q and a, but let's go ahead and get started by taking a look at the answers to the poll question. It looks like the majority of companies currently handle slow fields by waiting for the response or splitting the query and prefetching when possible. Defer doesn't necessarily replace these approaches, as there are use cases where split queries or prefetching may be more suitable. Implementing defer is an incremental process and requires server support. It's important to measure the success and benefits of defer for your specific use case. Tools like the router and React's profiling tool can help with implementation and measuring rendering time. Other approaches like the stream directive can also address performance problems.

So just a reminder to everyone you can join the Andromeda Q and a channel on discord to ask your questions for Lucas here in this Q and a, but let's go ahead and get started by taking a look at the answers to the poll question.

So again, the question that we had was how do you currently handle slow fields within your graph? And it looks like now a pretty overwhelming and strong response for just wait for the response. A little bit behind that is split the query and prefetch when possible, but Lucas what do you think about the results? Is that aligned with what you expected? Yeah, for sure. I think in general, we see a lot of companies doing this where it's not currently feasible or possible to orchestrate the responses to handle split queries or making assumptions about user interaction may not have data there. And so in general, it's really just, we wait for those slow fields to show up and then render when we get it.

Yeah, and just to answer a question here from the audience, does defer replace things like query splitting or prefetching? It doesn't necessarily replace a hundred percent of the time there are certainly use cases for handling both of the situations. For example, you may not wanna have the data requested all the time. There may be an extensive field that the user is needing and it may be necessary on the page at some point. For example, if they scroll down. So split queries or prefetching might be a more appropriate solution, but in general, defer can help with most of these.

Yeah, and when you're talking about implementing something like defer, are there certain areas that you may want to try it out first, or is there a way that you can do more of like a proof of concept or implement it into a code base? Or do you typically see people going on and on it right at once? Yeah, it's definitely an incremental process. We don't see people will just all of a sudden add a defer to every single query that they do for a number of reasons. Primarily the fact that defer requires server support, and so service support can vary across languages. I know we heard that GraphQL Yoga, for example, supports it now. However, on the Apollo side of things, we just introduced support for entity-based defer. So if you're running Federation, this has just become available. And it's me to be something that you're not aware that exists. But as you continue to look at it, it's also part of the thing, a part of it is you need to measure what success means to you. Whether that's time to first buy or whether it's when the user can start interacting with it. And that's going to be something that you have to kind of measure and see how beneficial defer can be, because you may realize that it's not providing the benefit that you expect and may not be worth the continued investment and engineering investment towards that specific use case. And so with all that being said, I definitely think taking a look at it and doing a proof of concept, if you haven't already looked, the router is an easy way to do this through kind of a Federation. Like model where you add entity declaration to interior existing GraphQL server. That can be an easy way to get started and measure whether defer can be a good tool for you. And are there a certain best practices or tools that you would use to measure or stress test in order to determine what may be a good place to apply this first? Yeah, I would definitely look at seeing where it takes most time to render on your site. This can vary based on UI framework you're using, but understanding where your rendering time comes from most can be usually where you can start to drill down and find out why is this happening. And oftentimes it's related to your data service. React, for example, has a really good profiling tool, which you can use to get an understanding of how long it takes to render a specific component. And it could be a good place to start to see how people or how your data is being rendered within your UI and seeing, can we delay this and partially render this component until we get the rest of the data? And you talked about performance problems in general and some of those issues. Are you seeing any kind of newer approaches or best practices to performance problems aside from defer that are of interest to you? Yeah, I think obviously there's things like the stream directive which obviously didn't fall in today's discussion but I think it's definitely worth talking about because Stream also solves a lot of the problems that defer may not fully address. Things like sending full lists or arrays of data across.

Alternative Approaches to Improve Performance#

Short description:

Defer may not be appropriate for all cases, especially when constraints like offsets are involved. Another interesting area to explore alongside defer is Stream. Leveraging tools like subscriptions or live queries can also improve data responsiveness.

This is a really common problem where you have a gigantic array of, for example, statistic information that you don't wanna request all of it at once, which may slow down your request endpoint. But in this case, defer may not be appropriate given constraints like the ability to do offsets like we talked about during the talk. And so Stream is one area that I think is really interesting and something that is also alongside defer, so once that becomes official will be interesting to follow. I also think that there are other areas that the tools you can leverage to improve performance. I know a lot of people use subscriptions or effectively live queries, and those are also good utilities to help with some of this data responsiveness effectively.

Lucas Leadbetter
Lucas Leadbetter
26 min
08 Dec, 2022

Comments

Sign in or register to post your comment.

Check out more articles and videos

We constantly think of articles and videos that might spark Git people interest / skill us up or help building a stellar career

From GraphQL Zero to GraphQL Hero with RedwoodJS
GraphQL Galaxy 2021GraphQL Galaxy 2021
32 min
From GraphQL Zero to GraphQL Hero with RedwoodJS
Top Content
Tom Pressenwurter introduces Redwood.js, a full stack app framework for building GraphQL APIs easily and maintainably. He demonstrates a Redwood.js application with a React-based front end and a Node.js API. Redwood.js offers a simplified folder structure and schema for organizing the application. It provides easy data manipulation and CRUD operations through GraphQL functions. Redwood.js allows for easy implementation of new queries and directives, including authentication and limiting access to data. It is a stable and production-ready framework that integrates well with other front-end technologies.
Local State and Server Cache: Finding a Balance
Vue.js London Live 2021Vue.js London Live 2021
24 min
Local State and Server Cache: Finding a Balance
Top Content
This Talk discusses handling local state in software development, particularly when dealing with asynchronous behavior and API requests. It explores the challenges of managing global state and the need for actions when handling server data. The Talk also highlights the issue of fetching data not in Vuex and the challenges of keeping data up-to-date in Vuex. It mentions alternative tools like Apollo Client and React Query for handling local state. The Talk concludes with a discussion on GitLab going public and the celebration that followed.
Batteries Included Reimagined - The Revival of GraphQL Yoga
GraphQL Galaxy 2021GraphQL Galaxy 2021
33 min
Batteries Included Reimagined - The Revival of GraphQL Yoga
Envelope is a powerful GraphQL plugin system that simplifies server development and allows for powerful plugin integration. It provides conformity for large corporations with multiple GraphQL servers and can be used with various frameworks. Envelope acts as the Babel of GraphQL, allowing the use of non-spec features. The Guild offers GraphQL Hive, a service similar to Apollo Studio, and encourages collaboration with other frameworks and languages.
Rock Solid React and GraphQL Apps for People in a Hurry
GraphQL Galaxy 2022GraphQL Galaxy 2022
29 min
Rock Solid React and GraphQL Apps for People in a Hurry
The Talk discusses the challenges and advancements in using GraphQL and React together. It introduces RedwoodJS, a framework that simplifies frontend-backend integration and provides features like code generation, scaffolding, and authentication. The Talk demonstrates how to set up a Redwood project, generate layouts and models, and perform CRUD operations. Redwood automates many GraphQL parts and provides an easy way for developers to get started with GraphQL. It also highlights the benefits of Redwood and suggests checking out RedwoodJS.com for more information.
Adopting GraphQL in an Enterprise
GraphQL Galaxy 2021GraphQL Galaxy 2021
32 min
Adopting GraphQL in an Enterprise
Today's Talk is about adopting GraphQL in an enterprise. It discusses the challenges of using REST APIs and the benefits of GraphQL. The Talk explores different approaches to adopting GraphQL, including coexistence with REST APIs. It emphasizes the power of GraphQL and provides tips for successful adoption. Overall, the Talk highlights the advantages of GraphQL in terms of efficiency, collaboration, and control over APIs.
Step aside resolvers: a new approach to GraphQL execution
GraphQL Galaxy 2022GraphQL Galaxy 2022
16 min
Step aside resolvers: a new approach to GraphQL execution
GraphQL has made a huge impact in the way we build client applications, websites, and mobile apps. Despite the dominance of resolvers, the GraphQL specification does not mandate their use. Introducing Graphast, a new project that compiles GraphQL operations into execution and output plans, providing advanced optimizations. In GraphFast, instead of resolvers, we have plan resolvers that deal with future data. Graphfast plan resolvers are short and efficient, supporting all features of modern GraphQL.

Workshops on related topic

Build with SvelteKit and GraphQL
GraphQL Galaxy 2021GraphQL Galaxy 2021
140 min
Build with SvelteKit and GraphQL
Top Content
Featured WorkshopFree
Scott Spence
Scott Spence
Have you ever thought about building something that doesn't require a lot of boilerplate with a tiny bundle size? In this workshop, Scott Spence will go from hello world to covering routing and using endpoints in SvelteKit. You'll set up a backend GraphQL API then use GraphQL queries with SvelteKit to display the GraphQL API data. You'll build a fast secure project that uses SvelteKit's features, then deploy it as a fully static site. This course is for the Svelte curious who haven't had extensive experience with SvelteKit and want a deeper understanding of how to use it in practical applications.

Table of contents:
- Kick-off and Svelte introduction
- Initialise frontend project
- Tour of the SvelteKit skeleton project
- Configure backend project
- Query Data with GraphQL
- Fetching data to the frontend with GraphQL
- Styling
- Svelte directives
- Routing in SvelteKit
- Endpoints in SvelteKit
- Deploying to Netlify
- Navigation
- Mutations in GraphCMS
- Sending GraphQL Mutations via SvelteKit
- Q&A
Build Modern Applications Using GraphQL and Javascript
Node Congress 2024Node Congress 2024
152 min
Build Modern Applications Using GraphQL and Javascript
Featured Workshop
Emanuel Scirlet
Miguel Henriques
2 authors
Come and learn how you can supercharge your modern and secure applications using GraphQL and Javascript. In this workshop we will build a GraphQL API and we will demonstrate the benefits of the query language for APIs and what use cases that are fit for it. Basic Javascript knowledge required.
End-To-End Type Safety with React, GraphQL & Prisma
React Advanced 2022React Advanced 2022
95 min
End-To-End Type Safety with React, GraphQL & Prisma
Featured WorkshopFree
Sabin Adams
Sabin Adams
In this workshop, you will get a first-hand look at what end-to-end type safety is and why it is important. To accomplish this, you’ll be building a GraphQL API using modern, relevant tools which will be consumed by a React client.
Prerequisites: - Node.js installed on your machine (12.2.X / 14.X)- It is recommended (but not required) to use VS Code for the practical tasks- An IDE installed (VSCode recommended)- (Good to have)*A basic understanding of Node.js, React, and TypeScript
GraphQL for React Developers
GraphQL Galaxy 2022GraphQL Galaxy 2022
112 min
GraphQL for React Developers
Featured Workshop
Roy Derks
Roy Derks
There are many advantages to using GraphQL as a datasource for frontend development, compared to REST APIs. We developers in example need to write a lot of imperative code to retrieve data to display in our applications and handle state. With GraphQL you cannot only decrease the amount of code needed around data fetching and state-management you'll also get increased flexibility, better performance and most of all an improved developer experience. In this workshop you'll learn how GraphQL can improve your work as a frontend developer and how to handle GraphQL in your frontend React application.
Build a Headless WordPress App with Next.js and WPGraphQL
React Summit 2022React Summit 2022
173 min
Build a Headless WordPress App with Next.js and WPGraphQL
Top Content
WorkshopFree
Kellen Mace
Kellen Mace
In this workshop, you’ll learn how to build a Next.js app that uses Apollo Client to fetch data from a headless WordPress backend and use it to render the pages of your app. You’ll learn when you should consider a headless WordPress architecture, how to turn a WordPress backend into a GraphQL server, how to compose queries using the GraphiQL IDE, how to colocate GraphQL fragments with your components, and more.
Relational Database Modeling for GraphQL
GraphQL Galaxy 2020GraphQL Galaxy 2020
106 min
Relational Database Modeling for GraphQL
Top Content
WorkshopFree
Adron Hall
Adron Hall
In this workshop we'll dig deeper into data modeling. We'll start with a discussion about various database types and how they map to GraphQL. Once that groundwork is laid out, the focus will shift to specific types of databases and how to build data models that work best for GraphQL within various scenarios.
Table of contentsPart 1 - Hour 1      a. Relational Database Data Modeling      b. Comparing Relational and NoSQL Databases      c. GraphQL with the Database in mindPart 2 - Hour 2      a. Designing Relational Data Models      b. Relationship, Building MultijoinsTables      c. GraphQL & Relational Data Modeling Query Complexities
Prerequisites      a. Data modeling tool. The trainer will be using dbdiagram      b. Postgres, albeit no need to install this locally, as I'll be using a Postgres Dicker image, from Docker Hub for all examples      c. Hasura