Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

Optimize taxonomy <> collection queries #376

Merged
merged 17 commits into from
Nov 13, 2024
30 changes: 29 additions & 1 deletion src/Taxonomies/TermQueryBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@

namespace Statamic\Eloquent\Taxonomies;

use Illuminate\Database\Query\Expression;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Str;
use Statamic\Contracts\Taxonomies\Term as TermContract;
use Statamic\Facades\Blink;
Expand Down Expand Up @@ -222,13 +224,39 @@ private function applyCollectionAndTaxonomyWheres()
return [];
}

// workaround to handle potential n+1 queries in the database
// if/when Statamic core supports relationships in a meaningful way this should be removed
if (config('statamic.eloquent-driver.entries.driver', 'file') == 'eloquent') {
$entryClass = app('statamic.eloquent.entries.model');
$termClass = app('statamic.eloquent.terms.model');

$entriesTable = (new $entryClass)->getTable();
$termsTable = (new $termClass)->getTable();

return TermModel::where('taxonomy', $taxonomy)
->whereExists(function ($query) use ($entriesTable, $taxonomy, $termsTable) {
$wrappedColumn = $query->getGrammar()->wrap("{$termsTable}.slug");
$value = match ($query->getConnection()->getDriverName()) {
'sqlite' => new Expression($wrappedColumn),
'pgsql' => new Expression("to_jsonb({$wrappedColumn}::text)"),
default => DB::raw("concat('\"', {$wrappedColumn}, '\"')"),
};

$query->select(DB::raw(1))
->from($entriesTable)
->whereIn('collection', $this->collections)
->whereJsonContains(Entry::query()->column($taxonomy->handle()), $value);
})
->pluck('slug');
}

return TermModel::where('taxonomy', $taxonomy)
->select('slug')
->get()
->map(function ($term) use ($taxonomy) {
return Entry::query()
->whereIn('collection', $this->collections)
->whereJsonContains('data->'.$taxonomy, [$term->slug])
->whereJsonContains($taxonomy->handle(), [$term->slug])
->count() > 0 ? $term->slug : null;
})
->filter()
Expand Down
10 changes: 10 additions & 0 deletions tests/Data/Taxonomies/TermQueryBuilderTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -207,6 +207,16 @@ public function terms_are_found_using_where_column()
$this->assertEquals(['a', 'b', 'e'], $terms->map->slug()->all());
}

#[Test]
public function it_filters_usage_in_collections_using_file_based_entries()
{
config()->set('statamic.eloquent-driver.entries.driver', 'file');
$this->it_filters_usage_in_collections();

// revert config to original
config()->set('statamic.eloquent-driver.entries.driver', 'eloquent');
}

#[Test]
public function it_filters_usage_in_collections()
{
Expand Down
Loading