How to avoid all database calls using Spatie Response Cache package

It’s no secret I stan Spatie packages. I’m convinced there must be something in the water over there in Belgium, because, all they seem to do is pump out quality package after quality package. And, the most impressive thing is that they’re all incredibly well maintained.

Fanboy-ing aside, I noticed a gotcha when using the Response Cache package, specifically some custom route binding rules:

// RouteServiceProvider.php

Route::bind('post', function($value) {
    return Post::where('slug', $value)->firstOrFail();
});

What the code above allows me to do is bind routes via the slug column in my database. so /posts/my-awesome-post would map to a post with my-awesome-post as the slug.

The problem

This is all well and good, however, despite caching this route, we will still incur a database call. This is because the route binding expressed above is run before Spatie’s cache middleware.

The workaround

The workaround for this is fetching the record from the controller action. This will allow the response to be truly cached.

// PostsController.php 

public function show(string $slug)
{
    return Post::where('slug', $value)->firstOrFail();
}

While we lose the spiffy custom route model binding above, we prevent a database hit for each request.

This solution isn’t perfect, and I’m open to other ways of doing it, but I wanted to put this out there should anyone else encounter the same quirk.

Happy coding!