120 lines
4.5 KiB
PHP
120 lines
4.5 KiB
PHP
<?php
|
|
|
|
declare(strict_types = 1);
|
|
|
|
// {{{ License
|
|
|
|
// This file is part of GNU social - https://www.gnu.org/software/social
|
|
//
|
|
// GNU social is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// GNU social is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with GNU social. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
// }}}
|
|
|
|
namespace Component\Search\Util;
|
|
|
|
use App\Core\Event;
|
|
use App\Util\Exception\ServerException;
|
|
use Doctrine\Common\Collections\Criteria;
|
|
|
|
abstract class Parser
|
|
{
|
|
/**
|
|
* Merge $parts into $criteria_arr
|
|
*/
|
|
private static function connectParts(array &$parts, array &$criteria_arr, string $last_op, mixed $eb, bool $force = false): void
|
|
{
|
|
foreach ([' ' => 'orX', '|' => 'orX', '&' => 'andX'] as $op => $func) {
|
|
if ($last_op === $op || $force) {
|
|
$criteria_arr[] = $eb->{$func}(...$parts);
|
|
$note_parts = [];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse $input string into a Doctrine query Criteria
|
|
*
|
|
* Currently doesn't support nesting with parenthesis and
|
|
* recognises either spaces (currently `or`, should be fuzzy match), `OR` or `|` (`or`) and `AND` or `&` (`and`)
|
|
*
|
|
* TODO Better fuzzy match, implement exact match with quotes and nesting with parens
|
|
*
|
|
* @return Criteria[]
|
|
*/
|
|
public static function parse(string $input, ?string $language = null, int $level = 0): array
|
|
{
|
|
if ($level === 0) {
|
|
$input = trim(preg_replace(['/\s+/', '/\s+AND\s+/', '/\s+OR\s+/'], [' ', '&', '|'], $input), ' |&');
|
|
}
|
|
|
|
$left = $right = 0;
|
|
$lenght = mb_strlen($input);
|
|
$stack = [];
|
|
$eb = Criteria::expr();
|
|
$note_criteria_arr = [];
|
|
$actor_criteria_arr = [];
|
|
$note_parts = [];
|
|
$actor_parts = [];
|
|
$last_op = null;
|
|
|
|
for ($index = 0; $index < $lenght; ++$index) {
|
|
$end = false;
|
|
$match = false;
|
|
|
|
foreach (['&', '|', ' '] as $delimiter) {
|
|
if ($input[$index] === $delimiter || $end = ($index === $lenght - 1)) {
|
|
$term = mb_substr($input, $left, $end ? null : $right - $left);
|
|
$note_res = $actor_res = null;
|
|
$ret = Event::handle('SearchCreateExpression', [$eb, $term, $language, &$note_res, &$actor_res]);
|
|
if (\is_null($note_res) && \is_null($actor_res)) {
|
|
throw new ServerException("No one claimed responsibility for a match term: {$term}");
|
|
} elseif (!\is_null($note_res)) {
|
|
$note_parts[] = $note_res;
|
|
} elseif (!\is_null($actor_res)) {
|
|
$actor_parts[] = $actor_res;
|
|
} else {
|
|
throw new ServerException('Unexpected state in Search parser');
|
|
}
|
|
|
|
$right = $left = $index + 1;
|
|
|
|
if (!\is_null($last_op) && $last_op !== $delimiter) {
|
|
self::connectParts($note_parts, $note_criteria_arr, $last_op, $eb, force: false);
|
|
self::connectParts($actor_parts, $actor_criteria_arr, $last_op, $eb, force: false);
|
|
} else {
|
|
$last_op = $delimiter;
|
|
}
|
|
$match = true;
|
|
continue 2;
|
|
}
|
|
}
|
|
if (!$match) {
|
|
++$right;
|
|
}
|
|
}
|
|
|
|
$note_criteria = $actor_criteria = null;
|
|
if (!empty($note_parts)) {
|
|
self::connectParts($note_parts, $note_criteria_arr, $last_op, $eb, force: true);
|
|
$note_criteria = new Criteria($eb->orX(...$note_criteria_arr));
|
|
} elseif (!empty($actor_parts)) {
|
|
self::connectParts($actor_parts, $actor_criteria_arr, $last_op, $eb, force: true);
|
|
$actor_criteria = new Criteria($eb->orX(...$actor_criteria_arr));
|
|
}
|
|
|
|
return [$note_criteria, $actor_criteria];
|
|
}
|
|
}
|