Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -130,6 +130,14 @@ $searchQuery->sumSortBy('field', 'asc');
$searchQuery->medianSortBy('field', 'asc');
```

### Pinned query

Promotes selected documents to rank higher than those matching a given query. This feature is typically used to guide searchers to curated documents that are promoted over and above any "organic" matches for a search. The promoted or "pinned" documents are identified using the document IDs stored in the _id field.

```php
$searchQuery->pinned(['doc-3', 'doc-1', 'doc-2']);
```

### Pagination

#### Offset Pagination
Expand Down
7 changes: 7 additions & 0 deletions src/Concerns/DecoratesBoolQuery.php
Original file line number Diff line number Diff line change
Expand Up @@ -172,4 +172,11 @@ public function addFunctionScore(array $functions, ?DSLAware $query = null, ?Fun

return $this;
}

public function pinned(array $ids, ?DSLAware $query = null): static
{
$this->forwardCallTo($this->boolQuery(), __FUNCTION__, func_get_args());

return $this;
}
}
2 changes: 2 additions & 0 deletions src/Contracts/BoolQuery.php
Original file line number Diff line number Diff line change
Expand Up @@ -53,4 +53,6 @@ public function whereBetween(string $field, mixed $from, mixed $to): static;
* @param ?FunctionScoreOptions $options
*/
public function addFunctionScore(array $functions, ?DSLAware $query = null, ?FunctionScoreOptions $options = null): static;

public function pinned(array $ids, ?DSLAware $query = null): static;
}
8 changes: 8 additions & 0 deletions src/Filtering/BoolQueryBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
use Ensi\LaravelElasticQuery\Filtering\Criterias\MultiMatch;
use Ensi\LaravelElasticQuery\Filtering\Criterias\Nested;
use Ensi\LaravelElasticQuery\Filtering\Criterias\OneMatch;
use Ensi\LaravelElasticQuery\Filtering\Criterias\Pinned;
use Ensi\LaravelElasticQuery\Filtering\Criterias\RangeBound;
use Ensi\LaravelElasticQuery\Filtering\Criterias\Term;
use Ensi\LaravelElasticQuery\Filtering\Criterias\Terms;
Expand Down Expand Up @@ -279,6 +280,13 @@ public function addFunctionScore(array $functions, ?DSLAware $query = null, ?Fun
return $this;
}

public function pinned(array $ids, ?DSLAware $query = null): static
{
$this->must->add(new Pinned($ids, $query));

return $this;
}

public function addMustBool(callable $fn): static
{
$this->must->add(static::make(builder: $fn));
Expand Down
32 changes: 32 additions & 0 deletions src/Filtering/Criterias/Pinned.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
<?php

namespace Ensi\LaravelElasticQuery\Filtering\Criterias;

use Ensi\LaravelElasticQuery\Contracts\Criteria;
use Ensi\LaravelElasticQuery\Contracts\DSLAware;
use stdClass;
use Webmozart\Assert\Assert;

class Pinned implements Criteria
{
public function __construct(
private array $ids,
private ?DSLAware $query = null,
) {
Assert::minCount($ids, 1);
}

public function toDSL(): array
{
$body = [
'ids' => $this->ids,
'organic' => ['match_all' => new stdClass()],
];

if ($this->query) {
$body['organic'] = $this->query->toDSL();
}

return ['pinned' => $body];
}
}
17 changes: 1 addition & 16 deletions src/Search/SearchQuery.php
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,6 @@ class SearchQuery implements SortableQuery, CollapsibleQuery, HighlightingQuery
protected ?int $from = null;
protected array $fields = [];
protected array $include = [];
protected ?array $pinnedIds = null;
protected array $exclude = [];
protected ?string $searchType = null;

Expand Down Expand Up @@ -150,19 +149,12 @@ protected function execute(
$dsl = [
'size' => $size,
'from' => $from,
'query' => [],
'query' => $this->boolQuery->toDSL(),
'track_total_hits' => $totals,
'_source' => $this->sourceToDSL($source),
'fields' => $source && $this->fields ? $this->fields : null,
];

if ($this->pinnedIds) {
$dsl['query']['pinned']['ids'] = $this->pinnedIds;
$dsl['query']['pinned']['organic'] = $this->boolQuery->toDSL();
} else {
$dsl['query'] = $this->boolQuery->toDSL();
}

$sorts ??= $this->sorts;
if (!$sorts->isEmpty()) {
$dsl['sort'] = $sorts->toDSL();
Expand Down Expand Up @@ -253,13 +245,6 @@ public function setPostFilter(BoolQueryBuilder $boolQueryBuilder): static
return $this;
}

public function pinned(array $ids): static
{
$this->pinnedIds = $ids;

return $this;
}

public function addAggregations(Aggregation $aggregation): static
{
$this->aggregations ??= new AggregationCollection();
Expand Down
Loading