mirror of
https://github.com/alchemy-fr/Phraseanet.git
synced 2025-10-16 22:43:19 +00:00

Concept queries on private fields where adding a « must » clause to restrict match on certain collections only. Boolean queries do not enforce one « should » clause match at least once a « must » clause is added to it. To make queries on private fields more robust & performants, these are now wrapped in a filtered query with a collection filter.
84 lines
2.8 KiB
PHP
84 lines
2.8 KiB
PHP
<?php
|
|
|
|
namespace Alchemy\Phrasea\SearchEngine\Elastic\AST;
|
|
|
|
use Alchemy\Phrasea\SearchEngine\Elastic\Search\QueryContext;
|
|
use Alchemy\Phrasea\SearchEngine\Elastic\Search\QueryHelper;
|
|
use Alchemy\Phrasea\SearchEngine\Elastic\Structure\Field;
|
|
use Alchemy\Phrasea\SearchEngine\Elastic\Thesaurus\Term;
|
|
|
|
class TextNode extends AbstractTermNode implements ContextAbleInterface
|
|
{
|
|
/**
|
|
* Merge two text nodes by concatenating their content.
|
|
*
|
|
* /!\ Text contexts are lost in the process
|
|
*
|
|
* @param TextNode $a First node
|
|
* @param TextNode $b The other one
|
|
* @return TextNode Merged text node
|
|
*/
|
|
public static function merge(TextNode $a, TextNode $b)
|
|
{
|
|
return new self(sprintf('%s%s', $a->getValue(), $b->getValue()));
|
|
}
|
|
|
|
/**
|
|
* Creates a new text node with the same content and the provided context.
|
|
*
|
|
* /!\ The original node context will not be preserved (ie. not merged).
|
|
*
|
|
* @param Context $context Context to add on the new node
|
|
* @return TextNode A text node with a context
|
|
*/
|
|
public function withContext(Context $context)
|
|
{
|
|
return new self($this->getValue(), $context);
|
|
}
|
|
|
|
public function buildQuery(QueryContext $context)
|
|
{
|
|
$query_builder = function (array $fields) use ($context) {
|
|
$index_fields = [];
|
|
foreach ($fields as $field) {
|
|
foreach ($context->localizeField($field) as $f) {
|
|
$index_fields[] = $f;
|
|
}
|
|
}
|
|
if (!$index_fields) {
|
|
return null;
|
|
}
|
|
return [
|
|
'multi_match' => [
|
|
'fields' => $index_fields,
|
|
'query' => $this->text,
|
|
'operator' => 'and',
|
|
'lenient' => true,
|
|
]
|
|
];
|
|
};
|
|
|
|
$unrestricted_fields = $context->getUnrestrictedFields();
|
|
$unrestricted_fields = Field::filterByValueCompatibility($unrestricted_fields, $this->text);
|
|
$query = $query_builder($unrestricted_fields);
|
|
|
|
$private_fields = $context->getPrivateFields();
|
|
$private_fields = Field::filterByValueCompatibility($private_fields, $this->text);
|
|
foreach (QueryHelper::wrapPrivateFieldQueries($private_fields, $query_builder) as $private_field_query) {
|
|
$query = QueryHelper::applyBooleanClause($query, 'should', $private_field_query);
|
|
}
|
|
|
|
$concept_query = $this->buildConceptQuery($context);
|
|
if ($concept_query !== null) {
|
|
$query = QueryHelper::applyBooleanClause($query, 'should', $concept_query);
|
|
}
|
|
|
|
return $query;
|
|
}
|
|
|
|
public function __toString()
|
|
{
|
|
return sprintf('<text:%s>', Term::dump($this));
|
|
}
|
|
}
|