Compare commits
18 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 3f57c78d01 | |||
| e88da82267 | |||
| 319334834b | |||
| 9bad1cb2b9 | |||
| df927a8197 | |||
| 5c98989631 | |||
| 373087a5d5 | |||
| 205b29e07c | |||
| e1d5f094de | |||
| 8145b87242 | |||
| d027933ad9 | |||
| cbad86a01c | |||
| 15bf7cb324 | |||
| 1b9787aa49 | |||
| 9d4297659d | |||
| 92842c552b | |||
| 80e584e29f | |||
| c03d2ae9ad |
5
config/packages/doctrine.yaml
Normal file
5
config/packages/doctrine.yaml
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
doctrine:
|
||||||
|
orm:
|
||||||
|
dql:
|
||||||
|
string_functions:
|
||||||
|
match: DoctrineExtensions\Query\Mysql\MatchAgainst
|
||||||
@@ -3,6 +3,11 @@ services:
|
|||||||
autowire: true
|
autowire: true
|
||||||
autoconfigure: true
|
autoconfigure: true
|
||||||
|
|
||||||
|
pcm_search.command.index:
|
||||||
|
class: Pcm\SearchBundle\Command\SearchIndexCommand
|
||||||
|
tags:
|
||||||
|
- { name: 'console.command', command: 'pcm:search:reindex' }
|
||||||
|
|
||||||
pcm_search.searchable_subscriber:
|
pcm_search.searchable_subscriber:
|
||||||
class: Pcm\SearchBundle\EventSubscriber\SearchableSubscriber
|
class: Pcm\SearchBundle\EventSubscriber\SearchableSubscriber
|
||||||
public: true
|
public: true
|
||||||
@@ -13,3 +18,8 @@ services:
|
|||||||
alias: Pcm\SearchBundle\Service\SearchService
|
alias: Pcm\SearchBundle\Service\SearchService
|
||||||
public: true
|
public: true
|
||||||
|
|
||||||
|
Pcm\SearchBundle\Service\SearchService: ~
|
||||||
|
|
||||||
|
Pcm\SearchBundle\Repository\SearchIndexRepository:
|
||||||
|
autowire: true
|
||||||
|
tags: ['doctrine.repository_service']
|
||||||
|
|||||||
@@ -18,5 +18,6 @@ class PcmSearchExtension extends Extension
|
|||||||
new FileLocator(__DIR__.'/../../config')
|
new FileLocator(__DIR__.'/../../config')
|
||||||
);
|
);
|
||||||
$loader->load('services.yaml');
|
$loader->load('services.yaml');
|
||||||
|
$loader->load('packages/doctrine.yaml');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
124
src/Repository/SearchIndexRepository.php
Normal file
124
src/Repository/SearchIndexRepository.php
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pcm\SearchBundle\Repository;
|
||||||
|
|
||||||
|
use Pcm\SearchBundle\Entity\SearchIndex;
|
||||||
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
|
use Doctrine\ORM\Query;
|
||||||
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
|
||||||
|
class SearchIndexRepository extends ServiceEntityRepository
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param ManagerRegistry $registry
|
||||||
|
*/
|
||||||
|
public function __construct(ManagerRegistry $registry)
|
||||||
|
{
|
||||||
|
parent::__construct($registry, SearchIndex::class);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Saves a SearchIndex entity
|
||||||
|
*
|
||||||
|
* @param SearchIndex $entity
|
||||||
|
* @param boolean $flush
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function add(SearchIndex $entity, bool $flush = false): void
|
||||||
|
{
|
||||||
|
$this->getEntityManager()->persist($entity);
|
||||||
|
|
||||||
|
if ($flush) {
|
||||||
|
$this->getEntityManager()->flush();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes a SearchIndex entity
|
||||||
|
*
|
||||||
|
* @param SearchIndex $entity
|
||||||
|
* @param boolean $flush
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function remove(SearchIndex $entity, bool $flush = false): void
|
||||||
|
{
|
||||||
|
$this->getEntityManager()->remove($entity);
|
||||||
|
|
||||||
|
if ($flush) {
|
||||||
|
$this->getEntityManager()->flush();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a Query object ready to be paginated or used to present results.
|
||||||
|
*
|
||||||
|
* @param string $query
|
||||||
|
* @param integer $minScore
|
||||||
|
* @return Query
|
||||||
|
*/
|
||||||
|
public function findAllPagination(string $query, int $minScore = 0): Query
|
||||||
|
{
|
||||||
|
$qb = $this->createQueryBuilder('r')
|
||||||
|
->addSelect('MATCH(r.data) AGAINST(:searchText boolean) AS score')
|
||||||
|
->where(
|
||||||
|
sprintf(
|
||||||
|
'MATCH(r.data) AGAINST(:searchText boolean) > %f',
|
||||||
|
$minScore
|
||||||
|
)
|
||||||
|
)->orderBy('score', 'DESC')
|
||||||
|
->setParameter(
|
||||||
|
'searchText',
|
||||||
|
$this->convertSearchTerm($query)
|
||||||
|
);
|
||||||
|
return $qb->getQuery();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Takes a string $query and explodes into individual words. Each word
|
||||||
|
* is then prefixed with + and ends with *, making the full text search
|
||||||
|
* operate as wildcard on all words
|
||||||
|
*
|
||||||
|
* @param string $query
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
private function convertSearchTerm(string $query): string
|
||||||
|
{
|
||||||
|
$extractedWords = [];
|
||||||
|
$sanitisedString = preg_replace('/[^\w^\d]/', ' ', $query);
|
||||||
|
$words = mb_split('\s', preg_replace(['/([^\w+])/','/(\s+)/'], ' ', $sanitisedString));
|
||||||
|
|
||||||
|
foreach ($words as $word) {
|
||||||
|
if (strlen($word)< 1) {
|
||||||
|
//
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$word = strtoupper($word);
|
||||||
|
$extractedWords[$word] = $word;
|
||||||
|
}
|
||||||
|
|
||||||
|
array_walk(
|
||||||
|
$extractedWords,
|
||||||
|
function(&$word) {
|
||||||
|
// require every word but allow matching just the start
|
||||||
|
$word = '+' . $word . '*';
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return implode(' ', $extractedWords);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clears the index table of all results
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function clearIndex(): void
|
||||||
|
{
|
||||||
|
$this
|
||||||
|
->createQueryBuilder('s')
|
||||||
|
->delete()
|
||||||
|
->getQuery()
|
||||||
|
->execute();
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user