search_core/Classes/Connection/Elasticsearch.php

237 lines
7.4 KiB
PHP
Raw Normal View History

2016-12-09 13:19:35 +01:00
<?php
namespace Codappix\SearchCore\Connection;
2016-12-09 13:19:35 +01:00
/*
* Copyright (C) 2016 Daniel Siepmann <coding@daniel-siepmann.de>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301, USA.
*/
use Codappix\SearchCore\Connection\Elasticsearch\SearchResult;
use Codappix\SearchCore\Domain\Search\QueryFactory;
use Elastica\Query;
use Elastica\Type;
use TYPO3\CMS\Core\SingletonInterface as Singleton;
use TYPO3\CMS\Extbase\Object\ObjectManagerInterface;
2016-12-09 13:19:35 +01:00
/**
* Outer wrapper to elasticsearch.
*/
class Elasticsearch implements Singleton, ConnectionInterface
{
/**
* @var Elasticsearch\Connection
*/
protected $connection;
/**
* @var Elasticsearch\IndexFactory
2016-12-09 13:19:35 +01:00
*/
protected $indexFactory;
/**
* @var Elasticsearch\TypeFactory
2016-12-09 13:19:35 +01:00
*/
protected $typeFactory;
/**
* @var Elasticsearch\MappingFactory
*/
protected $mappingFactory;
2016-12-09 13:19:35 +01:00
/**
* @var Elasticsearch\DocumentFactory
2016-12-09 13:19:35 +01:00
*/
protected $documentFactory;
/**
* @var QueryFactory
*/
protected $queryFactory;
2016-12-09 13:19:35 +01:00
/**
* @var \TYPO3\CMS\Core\Log\Logger
*/
protected $logger;
/**
* @var ObjectManagerInterface
*/
protected $objectManager;
2016-12-09 13:19:35 +01:00
/**
* Inject log manager to get concrete logger from it.
*
* @param \TYPO3\CMS\Core\Log\LogManager $logManager
*/
public function injectLogger(\TYPO3\CMS\Core\Log\LogManager $logManager)
{
$this->logger = $logManager->getLogger(__CLASS__);
}
/**
* @param ObjectManagerInterface $objectManager
*/
public function injectObjectManager(ObjectManagerInterface $objectManager)
{
$this->objectManager = $objectManager;
}
2016-12-09 13:19:35 +01:00
/**
* @param Elasticsearch\Connection $connection
* @param Elasticsearch\IndexFactory $indexFactory
* @param Elasticsearch\TypeFactory $typeFactory
* @param Elasticsearch\MappingFactory $mappingFactory
2016-12-09 13:19:35 +01:00
* @param Elasticsearch\DocumentFactory $documentFactory
* @param QueryFactory $queryFactory
2016-12-09 13:19:35 +01:00
*/
public function __construct(
Elasticsearch\Connection $connection,
Elasticsearch\IndexFactory $indexFactory,
Elasticsearch\TypeFactory $typeFactory,
Elasticsearch\MappingFactory $mappingFactory,
Elasticsearch\DocumentFactory $documentFactory,
QueryFactory $queryFactory
2016-12-09 13:19:35 +01:00
) {
$this->connection = $connection;
$this->indexFactory = $indexFactory;
$this->typeFactory = $typeFactory;
$this->mappingFactory = $mappingFactory;
2016-12-09 13:19:35 +01:00
$this->documentFactory = $documentFactory;
$this->queryFactory = $queryFactory;
2016-12-09 13:19:35 +01:00
}
public function addDocument(string $documentType, array $document)
2016-12-09 13:19:35 +01:00
{
$this->withType(
$documentType,
function (Type $type, string $documentType) use ($document) {
$type->addDocument($this->documentFactory->getDocument($documentType, $document));
}
);
2016-12-09 13:19:35 +01:00
}
public function deleteDocument(string $documentType, string $identifier)
2016-12-09 13:19:35 +01:00
{
try {
$this->withType(
$documentType,
function (Type $type, string $documentType) use ($identifier) {
$type->deleteById($identifier);
}
);
} catch (\Elastica\Exception\NotFoundException $exception) {
$this->logger->debug(
'Tried to delete document in index, which does not exist.',
[$documentType, $identifier]
);
}
2016-12-09 13:19:35 +01:00
}
public function updateDocument(string $documentType, array $document)
2016-12-09 13:19:35 +01:00
{
$this->withType(
$documentType,
function (Type $type, string $documentType) use ($document) {
$type->updateDocument($this->documentFactory->getDocument($documentType, $document));
}
);
2016-12-09 13:19:35 +01:00
}
public function addDocuments(string $documentType, array $documents)
2016-12-09 13:19:35 +01:00
{
$this->withType(
$documentType,
function (Type $type, string $documentType) use ($documents) {
$type->addDocuments($this->documentFactory->getDocuments($documentType, $documents));
}
2016-12-09 13:19:35 +01:00
);
}
2016-12-09 13:19:35 +01:00
public function deleteIndex()
{
$index = $this->connection->getClient()->getIndex($this->indexFactory->getIndexName());
if (!$index->exists()) {
$this->logger->notice(
'Index did not exist, therefore was not deleted.',
[$this->indexFactory->getIndexName()]
);
return;
}
$index->delete();
}
public function deleteIndexByDocumentType(string $documentType)
{
$query = Query::create([
'query' => [
'term' => [
'search_document_type' => $documentType
]
]
]);
$this->deleteIndexByQuery($query);
}
public function deleteIndexByQuery(Query $query)
{
$index = $this->connection->getClient()->getIndex($this->indexFactory->getIndexName());
if (!$index->exists()) {
$this->logger->notice(
'Index did not exist, therefore items can not be deleted by query.',
[$this->indexFactory->getIndexName(), $query->getQuery()]
);
return;
}
$response = $index->deleteByQuery($query);
if ($response->getData()['deleted'] > 0) {
// Refresh index when delete query is invoked
$index->refresh();
}
}
public function search(SearchRequestInterface $searchRequest): SearchResultInterface
2016-12-09 13:19:35 +01:00
{
$this->logger->debug('Search for', [$searchRequest->getSearchTerm()]);
$search = new \Elastica\Search($this->connection->getClient());
$search->addIndex($this->indexFactory->getIndexName());
$search->setQuery($this->queryFactory->create($searchRequest));
return $this->objectManager->get(SearchResult::class, $searchRequest, $search->search());
2016-12-09 13:19:35 +01:00
}
/**
* Execute given callback with Elastica Type based on provided documentType
*/
private function withType(string $documentType, callable $callback)
{
$type = $this->typeFactory->getType($documentType);
// TODO: Check whether it's to heavy to send it so often e.g. for every single document.
// Perhaps add command controller to submit mapping?!
// Also it's not possible to change mapping without deleting index first.
// Mattes told about a solution.
// So command looks like the best way so far, except we manage mattes solution.
// Still then this should be done once. So perhaps singleton which tracks state and does only once?
$this->mappingFactory->getMapping($documentType)->send();
$callback($type, $documentType);
$type->getIndex()->refresh();
}
2016-12-09 13:19:35 +01:00
}