OK Mauro, but I want to do paging AND sorting
It’s a kind of tornado of souls here. Coronavirus is all around where I live, and smart working, as Italians call remote working, is all the rage. Articles, posts, and twitter threads are all about how to be great at remote working. One way or the other, I’ve been working remotely for the last ten years or so, and in the last six years, I’ve been 100% remote working. Don’t worry. I don’t want to talk “remote working,” at least not today.
Looking for a ViewModel Composition framework? Take a look at the open source ServiceComposer.AspNetCore.
This post assumes that you’ve read “Paging and sorting in distributed systems, oh my!” article. If this is not the case, go and read it. I’ll wait here.
As you probably noticed, the approaches described to page data and sort data work in isolation, but immediately fall apart if we need to use them together. If the need is to apply “paging and sorting” in the same request, then the proposed solutions cannot be applied. Or, if implemented, can only result in a very chatty and non-performant design.
Paging and sorting is misleading
Data sorting needs to happen before paging can be applied. If we page and then sort, we’re going to sort only the current page, which means that if the user transitions to another page, the sort doesn’t make sense anymore. It should be “sorting and paging,” but that sounds bad.
The problem is that if we have to sort first and the request coming in is like /products?orderBy=price,description
then, as we already said, sorting is not so easy. The request reveals a relationship between the service owning prices and the service owning descriptions. If data are not in the same storage, the only option is first to load all the prices and sort them, then hand the result set over to the service that can load all the descriptions and sort the result set based on price and description. Only at this point paging can be applied.
That’s a join, Mauro
If we were to apply the outlined logic manually, it’s like we’re reinventing the wheel. And we’ve already discussed something very similar recently.
When we talked about searching in distributed systems, the suggestion was to ship data into a common place best suited to perform a search. The same rule applies here. If the need is to sort data across multiple services and apply paging to the sorted result set, it’s way simpler to perform all these steps in a single place, for example, a SQL database or an Elasticsearch instance.
Tomek and I discussed at length how to do that in The Quest for Better Search and the follow-up Search is a Team Effort. It’s just a matter of applying the same logic in this case.
Conclusion
When designing distributed systems, one of the goals is to keep coupling as low as possible. Zero coupling is an illusion, and we want to have in our tool belt all the tools that we might need to keep coupling low. Do we need to apply paging? We understood how to do that with almost no coupling. Do we need to sort data? It can be done. Do we need to sort and page data? Sure, there is a solution to that as well, but this time comes with a little coupling. As for searching, we want to be very careful and make sure that only data required to apply the sorting end up being stored in the storage. We don’t want to pay the price of unneeded coupling.
Header image: Photo by Kai Pilger on Unsplash