mirror of
https://github.com/alchemy-fr/Phraseanet.git
synced 2025-10-23 18:03:17 +00:00
Same grammar path for native keys than « IN » expressions
- Replace Database/Collection/Type/RecordIdentifier specific AST with a generic KeyValueExpression - « IN » queries in regular fields are still using InExpression right now - Add some tests for « type:XXXX » queries
This commit is contained in:
@@ -31,10 +31,11 @@
|
||||
%token collection collection
|
||||
%token type type
|
||||
%token id id|recordid
|
||||
%token field_prefix field.
|
||||
%token flag_prefix flag.
|
||||
%token true true|1
|
||||
%token false false|0
|
||||
%token word [^\s()\[\]:<>≤≥=]+
|
||||
%token word [^\s\(\)\[\]:<>≤≥=]+
|
||||
|
||||
// relative order of precedence is NOT > XOR > AND > OR
|
||||
|
||||
@@ -75,7 +76,18 @@ boolean:
|
||||
// Field narrowing
|
||||
|
||||
quinary:
|
||||
senary() ( ::space:: ::in:: ::space:: field() #in )?
|
||||
senary() ( ::space:: ::in:: ::space:: key() #in )?
|
||||
|
||||
key:
|
||||
native_key() #native_key
|
||||
| ::field_prefix:: field()
|
||||
| field()
|
||||
|
||||
native_key:
|
||||
<database>
|
||||
| <collection>
|
||||
| <type>
|
||||
| <id>
|
||||
|
||||
#field:
|
||||
word_or_keyword()+
|
||||
@@ -151,6 +163,7 @@ keyword:
|
||||
| <collection>
|
||||
| <type>
|
||||
| <id>
|
||||
| <field_prefix>
|
||||
| <flag_prefix>
|
||||
| <true>
|
||||
| <false>
|
||||
|
@@ -4,7 +4,6 @@ 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\Concept;
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\Thesaurus\TermInterface;
|
||||
|
||||
|
@@ -1,34 +0,0 @@
|
||||
<?php
|
||||
|
||||
namespace Alchemy\Phrasea\SearchEngine\Elastic\AST;
|
||||
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\Search\QueryContext;
|
||||
|
||||
class CollectionExpression extends Node
|
||||
{
|
||||
private $collectionName;
|
||||
|
||||
public function __construct($collectionName)
|
||||
{
|
||||
$this->collectionName = $collectionName;
|
||||
}
|
||||
|
||||
public function buildQuery(QueryContext $context)
|
||||
{
|
||||
return [
|
||||
'term' => [
|
||||
'collection_name' => $this->collectionName
|
||||
]
|
||||
];
|
||||
}
|
||||
|
||||
public function getTermNodes()
|
||||
{
|
||||
return array();
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return sprintf('<collection:%s>', $this->collectionName);
|
||||
}
|
||||
}
|
@@ -1,34 +0,0 @@
|
||||
<?php
|
||||
|
||||
namespace Alchemy\Phrasea\SearchEngine\Elastic\AST;
|
||||
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\Search\QueryContext;
|
||||
|
||||
class DatabaseExpression extends Node
|
||||
{
|
||||
private $database;
|
||||
|
||||
public function __construct($database)
|
||||
{
|
||||
$this->database = $database;
|
||||
}
|
||||
|
||||
public function buildQuery(QueryContext $context)
|
||||
{
|
||||
return [
|
||||
'term' => [
|
||||
'databox_name' => $this->database
|
||||
]
|
||||
];
|
||||
}
|
||||
|
||||
public function getTermNodes()
|
||||
{
|
||||
return array();
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return sprintf('<database:%s>', $this->database);
|
||||
}
|
||||
}
|
50
lib/Alchemy/Phrasea/SearchEngine/Elastic/AST/Key.php
Normal file
50
lib/Alchemy/Phrasea/SearchEngine/Elastic/AST/Key.php
Normal file
@@ -0,0 +1,50 @@
|
||||
<?php
|
||||
|
||||
namespace Alchemy\Phrasea\SearchEngine\Elastic\AST;
|
||||
|
||||
class Key
|
||||
{
|
||||
const TYPE_DATABASE = 'database';
|
||||
const TYPE_COLLECTION = 'collection';
|
||||
const TYPE_MEDIA_TYPE = 'media_type';
|
||||
const TYPE_RECORD_IDENTIFIER = 'record_identifier';
|
||||
|
||||
private $type;
|
||||
private $key;
|
||||
|
||||
public static function database()
|
||||
{
|
||||
return new self(self::TYPE_DATABASE, 'databox_name');
|
||||
}
|
||||
|
||||
public static function collection()
|
||||
{
|
||||
return new self(self::TYPE_COLLECTION, 'collection_name');
|
||||
}
|
||||
|
||||
public static function mediaType()
|
||||
{
|
||||
return new self(self::TYPE_MEDIA_TYPE, 'type');
|
||||
}
|
||||
|
||||
public static function recordIdentifier()
|
||||
{
|
||||
return new self(self::TYPE_RECORD_IDENTIFIER, 'record_id');
|
||||
}
|
||||
|
||||
private function __construct($type, $key)
|
||||
{
|
||||
$this->type = $type;
|
||||
$this->key = $key;
|
||||
}
|
||||
|
||||
public function getIndexField()
|
||||
{
|
||||
return $this->key;
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return $this->type;
|
||||
}
|
||||
}
|
@@ -0,0 +1,38 @@
|
||||
<?php
|
||||
|
||||
namespace Alchemy\Phrasea\SearchEngine\Elastic\AST;
|
||||
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\Search\QueryContext;
|
||||
use Assert\Assertion;
|
||||
|
||||
class KeyValueExpression extends Node
|
||||
{
|
||||
protected $key;
|
||||
protected $value;
|
||||
|
||||
public function __construct(Key $key, $value)
|
||||
{
|
||||
Assertion::string($value);
|
||||
$this->key = $key;
|
||||
$this->value = $value;
|
||||
}
|
||||
|
||||
public function buildQuery(QueryContext $context)
|
||||
{
|
||||
return [
|
||||
'term' => [
|
||||
$this->key->getIndexField() => $this->value
|
||||
]
|
||||
];
|
||||
}
|
||||
|
||||
public function getTermNodes()
|
||||
{
|
||||
return [];
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return sprintf('<%s:%s>', $this->key, $this->value);
|
||||
}
|
||||
}
|
@@ -1,34 +0,0 @@
|
||||
<?php
|
||||
|
||||
namespace Alchemy\Phrasea\SearchEngine\Elastic\AST;
|
||||
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\Search\QueryContext;
|
||||
|
||||
class RecordIdentifierExpression extends Node
|
||||
{
|
||||
private $record_id;
|
||||
|
||||
public function __construct($record_id)
|
||||
{
|
||||
$this->record_id = $record_id;
|
||||
}
|
||||
|
||||
public function buildQuery(QueryContext $context)
|
||||
{
|
||||
return [
|
||||
'term' => [
|
||||
'record_id' => $this->record_id
|
||||
]
|
||||
];
|
||||
}
|
||||
|
||||
public function getTermNodes()
|
||||
{
|
||||
return array();
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return sprintf('<record_identifier:%s>', $this->record_id);
|
||||
}
|
||||
}
|
@@ -1,34 +0,0 @@
|
||||
<?php
|
||||
|
||||
namespace Alchemy\Phrasea\SearchEngine\Elastic\AST;
|
||||
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\Search\QueryContext;
|
||||
|
||||
class TypeExpression extends Node
|
||||
{
|
||||
private $typeName;
|
||||
|
||||
public function __construct($typeName)
|
||||
{
|
||||
$this->typeName = $typeName;
|
||||
}
|
||||
|
||||
public function buildQuery(QueryContext $context)
|
||||
{
|
||||
return [
|
||||
'term' => [
|
||||
'type' => $this->typeName
|
||||
]
|
||||
];
|
||||
}
|
||||
|
||||
public function getTermNodes()
|
||||
{
|
||||
return array();
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return sprintf('<type:%s>', $this->typeName);
|
||||
}
|
||||
}
|
@@ -23,6 +23,7 @@ class NodeTypes
|
||||
const CONTEXT = '#context';
|
||||
const FLAG_STATEMENT = '#flag_statement';
|
||||
const FLAG = '#flag';
|
||||
const NATIVE_KEY = '#native_key';
|
||||
const COLLECTION = '#collection';
|
||||
const TYPE = '#type';
|
||||
const DATABASE = '#database';
|
||||
@@ -31,6 +32,10 @@ class NodeTypes
|
||||
const TOKEN_WORD = 'word';
|
||||
const TOKEN_QUOTED_STRING = 'quoted';
|
||||
const TOKEN_RAW_STRING = 'raw_quoted';
|
||||
const TOKEN_DATABASE = 'database';
|
||||
const TOKEN_COLLECTION = 'collection';
|
||||
const TOKEN_MEDIA_TYPE = 'type';
|
||||
const TOKEN_RECORD_ID = 'id';
|
||||
const TOKEN_TRUE = 'true';
|
||||
const TOKEN_FALSE = 'false';
|
||||
}
|
||||
|
@@ -88,6 +88,9 @@ class QueryVisitor implements Visit
|
||||
case NodeTypes::FLAG:
|
||||
return $this->visitString($element);
|
||||
|
||||
case NodeTypes::NATIVE_KEY:
|
||||
return $this->visitNativeKeyNode($element);
|
||||
|
||||
case NodeTypes::DATABASE:
|
||||
return $this->visitDatabaseNode($element);
|
||||
|
||||
@@ -119,9 +122,18 @@ class QueryVisitor implements Visit
|
||||
if ($element->getChildrenNumber() !== 2) {
|
||||
throw new \Exception('IN expression can only have 2 childs.');
|
||||
}
|
||||
$expression = $element->getChild(0)->accept($this);
|
||||
$expression = $element->getChild(0);
|
||||
$field = $this->visit($element->getChild(1));
|
||||
return new AST\InExpression($field, $expression);
|
||||
if ($field instanceof AST\Field) {
|
||||
return new AST\InExpression($field, $this->visit($expression));
|
||||
} elseif ($field instanceof AST\Key) {
|
||||
return new AST\KeyValueExpression(
|
||||
$field,
|
||||
$this->visitString($expression)
|
||||
);
|
||||
} else {
|
||||
throw new \Exception(sprintf('Unexpected key node type "%s".', is_object($field) ? get_class($field) : gettype($field)));
|
||||
}
|
||||
}
|
||||
|
||||
private function visitAndNode(Element $element)
|
||||
@@ -299,6 +311,26 @@ class QueryVisitor implements Visit
|
||||
}
|
||||
}
|
||||
|
||||
private function visitNativeKeyNode(Element $element)
|
||||
{
|
||||
if ($element->getChildrenNumber() !== 1) {
|
||||
throw new \Exception('Native key node can only have a single child.');
|
||||
}
|
||||
$type = $element->getChild(0)->getValue()['token'];
|
||||
switch ($type) {
|
||||
case NodeTypes::TOKEN_DATABASE:
|
||||
return AST\Key::database();
|
||||
case NodeTypes::TOKEN_COLLECTION:
|
||||
return AST\Key::collection();
|
||||
case NodeTypes::TOKEN_MEDIA_TYPE:
|
||||
return AST\Key::mediaType();
|
||||
case NodeTypes::TOKEN_RECORD_ID:
|
||||
return AST\Key::recordIdentifier();
|
||||
default:
|
||||
throw new InvalidArgumentException(sprintf('Unexpected token type "%s" for native key.', $type));
|
||||
}
|
||||
}
|
||||
|
||||
private function visitDatabaseNode(Element $element)
|
||||
{
|
||||
if ($element->getChildrenNumber() !== 1) {
|
||||
@@ -306,7 +338,7 @@ class QueryVisitor implements Visit
|
||||
}
|
||||
$baseName = $element->getChild(0)->getValue()['value'];
|
||||
|
||||
return new AST\DatabaseExpression($baseName);
|
||||
return new AST\KeyValueExpression(AST\Key::database(), $baseName);
|
||||
}
|
||||
|
||||
private function visitCollectionNode(Element $element)
|
||||
@@ -316,7 +348,7 @@ class QueryVisitor implements Visit
|
||||
}
|
||||
$collectionName = $element->getChild(0)->getValue()['value'];
|
||||
|
||||
return new AST\CollectionExpression($collectionName);
|
||||
return new AST\KeyValueExpression(AST\Key::collection(), $collectionName);
|
||||
}
|
||||
|
||||
private function visitTypeNode(Element $element)
|
||||
@@ -326,7 +358,7 @@ class QueryVisitor implements Visit
|
||||
}
|
||||
$typeName = $element->getChild(0)->getValue()['value'];
|
||||
|
||||
return new AST\TypeExpression($typeName);
|
||||
return new AST\KeyValueExpression(AST\Key::mediaType(), $typeName);
|
||||
}
|
||||
|
||||
private function visitIdentifierNode(Element $element)
|
||||
@@ -336,6 +368,6 @@ class QueryVisitor implements Visit
|
||||
}
|
||||
$identifier = $element->getChild(0)->getValue()['value'];
|
||||
|
||||
return new AST\RecordIdentifierExpression($identifier);
|
||||
return new AST\KeyValueExpression(AST\Key::recordIdentifier(), $identifier);
|
||||
}
|
||||
}
|
||||
|
@@ -0,0 +1,40 @@
|
||||
<?php
|
||||
|
||||
namespace Alchemy\Tests\Phrasea\SearchEngine\AST;
|
||||
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\AST\Key;
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\AST\KeyValueExpression;
|
||||
use Alchemy\Phrasea\SearchEngine\Elastic\Search\QueryContext;
|
||||
|
||||
/**
|
||||
* @group unit
|
||||
* @group searchengine
|
||||
* @group ast
|
||||
*/
|
||||
class KeyValueExpressionTest extends \PHPUnit_Framework_TestCase
|
||||
{
|
||||
public function testSerialization()
|
||||
{
|
||||
$this->assertTrue(method_exists(KeyValueExpression::class, '__toString'), 'Class does not have method __toString');
|
||||
$node = new KeyValueExpression(Key::database(), 'bar');
|
||||
$this->assertEquals('<database:bar>', (string) $node);
|
||||
}
|
||||
|
||||
public function testQueryBuild()
|
||||
{
|
||||
$query_context = $this->prophesize(QueryContext::class);
|
||||
$key = $this->prophesize(Key::class);
|
||||
$key->getIndexField()->willReturn('foo');
|
||||
|
||||
$node = new KeyValueExpression($key->reveal(), 'bar');
|
||||
$query = $node->buildQuery($query_context->reveal());
|
||||
|
||||
$expected = '{
|
||||
"term": {
|
||||
"foo": "bar"
|
||||
}
|
||||
}';
|
||||
|
||||
$this->assertEquals(json_decode($expected, true), $query);
|
||||
}
|
||||
}
|
@@ -66,6 +66,27 @@ foo bar IN baz|(<text:"foo bar"> IN <field:baz>)
|
||||
foo IN bar baz|<text:"foo IN bar baz">
|
||||
fooINbar|<text:"fooINbar">
|
||||
|
||||
# Native fields with IN syntax (temporary)
|
||||
foo IN collection|<collection:foo>
|
||||
foo IN collection AND bar|(<collection:foo> AND <text:"bar">)
|
||||
foo IN collection bar|<text:"foo IN collection bar">
|
||||
foo IN database|<database:foo>
|
||||
foo IN database AND bar|(<database:foo> AND <text:"bar">)
|
||||
foo IN database bar|<text:"foo IN database bar">
|
||||
foo IN type|<media_type:foo>
|
||||
foo IN type AND bar|(<media_type:foo> AND <text:"bar">)
|
||||
foo IN type bar|<text:"foo IN type bar">
|
||||
90 IN id|<record_identifier:90>
|
||||
90 IN id AND foo|(<record_identifier:90> AND <text:"foo">)
|
||||
90 IN id foo|<text:"90 IN id foo">
|
||||
90 IN recordid|<record_identifier:90>
|
||||
|
||||
# Regular field with name colliding with a native key
|
||||
foo IN field.collection|(<text:"foo"> IN <field:collection>)
|
||||
foo IN field.database|(<text:"foo"> IN <field:database>)
|
||||
foo IN field.type|(<text:"foo"> IN <field:type>)
|
||||
foo IN field.id|(<text:"foo"> IN <field:id>)
|
||||
|
||||
# Matchers
|
||||
collection:foo|<collection:foo>
|
||||
collection:foo AND bar|(<collection:foo> AND <text:"bar">)
|
||||
@@ -73,6 +94,9 @@ collection:foo bar|<text:"collection:foo bar">
|
||||
database:foo|<database:foo>
|
||||
database:foo AND bar|(<database:foo> AND <text:"bar">)
|
||||
database:foo bar|<text:"database:foo bar">
|
||||
type:foo|<media_type:foo>
|
||||
type:foo AND bar|(<media_type:foo> AND <text:"bar">)
|
||||
type:foo bar|<text:"type:foo bar">
|
||||
id:90|<record_identifier:90>
|
||||
id:90 AND foo|(<record_identifier:90> AND <text:"foo">)
|
||||
id:90 foo|<text:"id:90 foo">
|
||||
|
Can't render this file because it contains an unexpected character in line 1 and column 11.
|
Reference in New Issue
Block a user