Eloquent - Calculating totals using conditional aggregates
Something which looks like this:
data:image/s3,"s3://crabby-images/3e1a5/3e1a58f2df08cbd4d0038254bef14184c67b6e29" alt="Feature List"
If we add the totals, the appearance would change to something like this:
data:image/s3,"s3://crabby-images/0eddf/0eddf88fa3f9bb56cb47f5384a56dcafe752c133" alt="Feature List"
But what's the most efficient way of calculating the totals?
$statuses->requested = Feature::where('status', 'requested')->count();
$statuses->planned = Feature::where('status', 'planned')->count();
$statuses->completed = Feature::where('status', 'completed')->count();
This approach is simple enough and for this example, it's not a big deal as we've only got 3 statuses, but what if we had 5, or 10, or 20?
That's when this approach starts to become a problem!
data:image/s3,"s3://crabby-images/a95ed/a95ed458e67f7d4da53a909860f78e9a9ff9684e" alt="Option 1"
It is possible to fetch everything you need in one database query by writing count aggregate queries. The query would look like this:
SELECT
count(CASE WHEN status = 'requested' THEN 1 END) AS requested,
count(CASE WHEN status = 'planned' THEN 1 END) AS planned,
count(CASE WHEN status = 'completed' THEN 1 END) AS completed
from features;
So how do we do this with Eloquent?
$statuses = Feature::toBase()
->selectRaw("count(CASE WHEN status = 'requested' THEN 1 END) AS requested")
->selectRaw("count(CASE WHEN status = 'planned' THEN 1 END) AS planned")
->selectRaw("count(CASE WHEN status = 'completed' THEN 1 END) AS completed")
->first();
We've now executed one query (so we've got a scalable solution / no N+1 problem), not three and this query is even quicker than the first one ran above.
data:image/s3,"s3://crabby-images/2bbfd/2bbfde53dffb0cb60b04af64fd923007111d993b" alt="Option 2"
Ready to bring your vision to life?
We believe in excellence, empathy, integrity, and transparency throughout the process. Our goal is to build fast, responsive websites that not only perform but also reflect your values and vision.