529 lines
14 KiB
PHP
529 lines
14 KiB
PHP
|
<?php
|
||
|
namespace WapplerSystems\Meilisearch\IndexQueue;
|
||
|
|
||
|
/***************************************************************
|
||
|
* Copyright notice
|
||
|
*
|
||
|
* (c) 2009-2015 Ingo Renner <ingo@typo3.org>
|
||
|
* All rights reserved
|
||
|
*
|
||
|
* This script is part of the TYPO3 project. The TYPO3 project 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 3 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* The GNU General Public License can be found at
|
||
|
* http://www.gnu.org/copyleft/gpl.html.
|
||
|
*
|
||
|
* This script 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.
|
||
|
*
|
||
|
* This copyright notice MUST APPEAR in all copies of the script!
|
||
|
***************************************************************/
|
||
|
|
||
|
use WapplerSystems\Meilisearch\Domain\Index\Queue\IndexQueueIndexingPropertyRepository;
|
||
|
use WapplerSystems\Meilisearch\Domain\Index\Queue\QueueItemRepository;
|
||
|
use WapplerSystems\Meilisearch\Domain\Site\SiteRepository;
|
||
|
use WapplerSystems\Meilisearch\Domain\Site\Site;
|
||
|
use TYPO3\CMS\Backend\Utility\BackendUtility;
|
||
|
use TYPO3\CMS\Core\Utility\GeneralUtility;
|
||
|
|
||
|
/**
|
||
|
* Representation of an index queue item, carrying meta data and the record to be
|
||
|
* indexed.
|
||
|
*
|
||
|
* @todo: Loose coupling from Repos
|
||
|
*
|
||
|
* @author Ingo Renner <ingo@typo3.org>
|
||
|
*/
|
||
|
class Item
|
||
|
{
|
||
|
const STATE_BLOCKED = -1;
|
||
|
|
||
|
const STATE_PENDING = 0;
|
||
|
|
||
|
const STATE_INDEXED = 1;
|
||
|
|
||
|
/**
|
||
|
* The item's uid in the index queue (tx_meilisearch_indexqueue_item.uid)
|
||
|
*
|
||
|
* @var int
|
||
|
*/
|
||
|
protected $indexQueueUid;
|
||
|
|
||
|
/**
|
||
|
* The root page uid of the tree the item is located in (tx_meilisearch_indexqueue_item.root)
|
||
|
*
|
||
|
* @var int
|
||
|
*/
|
||
|
protected $rootPageUid;
|
||
|
|
||
|
/**
|
||
|
* The record's type, usually a table name, but could also be a file type (tx_meilisearch_indexqueue_item.item_type)
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $type;
|
||
|
|
||
|
/**
|
||
|
* The name of the indexing configuration that should be used when indexing (tx_meilisearch_indexqueue_item.indexing_configuration)
|
||
|
* the item.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $indexingConfigurationName;
|
||
|
|
||
|
/**
|
||
|
* The unix timestamp when the record was last changed (tx_meilisearch_indexqueue_item.changed)
|
||
|
*
|
||
|
* @var int
|
||
|
*/
|
||
|
protected $changed;
|
||
|
|
||
|
/**
|
||
|
* The unix timestamp when the record was last indexed (tx_meilisearch_indexqueue_item.indexed)
|
||
|
*
|
||
|
* @var int
|
||
|
*/
|
||
|
protected $indexed;
|
||
|
|
||
|
/**
|
||
|
* Indexing properties to provide additional information for the item's
|
||
|
* indexer / how to index the item.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $indexingProperties = [];
|
||
|
|
||
|
/**
|
||
|
* Flag for lazy loading indexing properties.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
protected $indexingPropertiesLoaded = false;
|
||
|
|
||
|
/**
|
||
|
* Flag, whether indexing properties exits for this item.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
protected $hasIndexingProperties = false;
|
||
|
|
||
|
/**
|
||
|
* The record's uid.
|
||
|
*
|
||
|
* @var int
|
||
|
*/
|
||
|
protected $recordUid = 0;
|
||
|
|
||
|
/**
|
||
|
* The record itself
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $record;
|
||
|
|
||
|
/**
|
||
|
* Moint point identifier.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $mountPointIdentifier;
|
||
|
|
||
|
/**
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $errors = '';
|
||
|
|
||
|
/**
|
||
|
* @var IndexQueueIndexingPropertyRepository
|
||
|
*/
|
||
|
protected $indexQueueIndexingPropertyRepository;
|
||
|
|
||
|
/**
|
||
|
* @var QueueItemRepository
|
||
|
*/
|
||
|
protected $queueItemRepository;
|
||
|
|
||
|
/**
|
||
|
* Constructor, takes item meta data information and resolves that to the full record.
|
||
|
*
|
||
|
* @param array $itemMetaData Metadata describing the item to index using the index queue. Is expected to contain a record from table tx_meilisearch_indexqueue_item
|
||
|
* @param array $fullRecord Optional full record for the item. If provided, can save some SQL queries.
|
||
|
* @param IndexQueueIndexingPropertyRepository|null $indexQueueIndexingPropertyRepository
|
||
|
*/
|
||
|
public function __construct(array $itemMetaData, array $fullRecord = [], IndexQueueIndexingPropertyRepository $indexQueueIndexingPropertyRepository = null, QueueItemRepository $queueItemRepository = null)
|
||
|
{
|
||
|
$this->indexQueueUid = $itemMetaData['uid'];
|
||
|
$this->rootPageUid = $itemMetaData['root'];
|
||
|
$this->type = $itemMetaData['item_type'];
|
||
|
$this->recordUid = $itemMetaData['item_uid'];
|
||
|
$this->mountPointIdentifier = (string) empty($itemMetaData['pages_mountidentifier']) ? '' : $itemMetaData['pages_mountidentifier'];
|
||
|
$this->changed = $itemMetaData['changed'];
|
||
|
$this->indexed = $itemMetaData['indexed'];
|
||
|
$this->errors = (string) empty($itemMetaData['errors']) ? '' : $itemMetaData['errors'];
|
||
|
|
||
|
$this->indexingConfigurationName = $itemMetaData['indexing_configuration'];
|
||
|
$this->hasIndexingProperties = (boolean)$itemMetaData['has_indexing_properties'];
|
||
|
|
||
|
if (!empty($fullRecord)) {
|
||
|
$this->record = $fullRecord;
|
||
|
}
|
||
|
|
||
|
$this->indexQueueIndexingPropertyRepository = $indexQueueIndexingPropertyRepository ?? GeneralUtility::makeInstance(IndexQueueIndexingPropertyRepository::class);
|
||
|
$this->queueItemRepository = $queueItemRepository ?? GeneralUtility::makeInstance(QueueItemRepository::class);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Getter for Index Queue UID
|
||
|
*
|
||
|
* @return integer
|
||
|
*/
|
||
|
public function getIndexQueueUid()
|
||
|
{
|
||
|
return $this->indexQueueUid;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the item's root page ID (uid)
|
||
|
*
|
||
|
* @return int root page ID
|
||
|
*/
|
||
|
public function getRootPageUid()
|
||
|
{
|
||
|
return $this->rootPageUid;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns mount point identifier
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function getMountPointIdentifier()
|
||
|
{
|
||
|
return $this->mountPointIdentifier;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param integer $uid
|
||
|
*/
|
||
|
public function setRootPageUid($uid)
|
||
|
{
|
||
|
$this->rootPageUid = intval($uid);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return string
|
||
|
*/
|
||
|
public function getErrors()
|
||
|
{
|
||
|
return $this->errors;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return boolean
|
||
|
*/
|
||
|
public function getHasErrors()
|
||
|
{
|
||
|
return trim($this->errors) !== '';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return int
|
||
|
*/
|
||
|
public function getState()
|
||
|
{
|
||
|
if ($this->getHasErrors()) {
|
||
|
return self::STATE_BLOCKED;
|
||
|
}
|
||
|
|
||
|
if ($this->getIndexed() > $this->getChanged()) {
|
||
|
return self::STATE_INDEXED;
|
||
|
}
|
||
|
|
||
|
return self::STATE_PENDING;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the site the item belongs to.
|
||
|
*
|
||
|
* @return Site Site instance the item belongs to.
|
||
|
*/
|
||
|
public function getSite()
|
||
|
{
|
||
|
$siteRepository = GeneralUtility::makeInstance(SiteRepository::class);
|
||
|
return $siteRepository->getSiteByRootPageId($this->rootPageUid);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the type/tablename of the queue record.
|
||
|
*
|
||
|
* @return mixed|string
|
||
|
*/
|
||
|
public function getType()
|
||
|
{
|
||
|
return $this->type;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param $type
|
||
|
*/
|
||
|
public function setType($type)
|
||
|
{
|
||
|
$this->type = $type;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the name of the index configuration that was used to create this record.
|
||
|
*
|
||
|
* @return mixed|string
|
||
|
*/
|
||
|
public function getIndexingConfigurationName()
|
||
|
{
|
||
|
return $this->indexingConfigurationName;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $indexingConfigurationName
|
||
|
*/
|
||
|
public function setIndexingConfigurationName($indexingConfigurationName)
|
||
|
{
|
||
|
$this->indexingConfigurationName = $indexingConfigurationName;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the timestamp when this queue item was changed.
|
||
|
*
|
||
|
* @return int|mixed
|
||
|
*/
|
||
|
public function getChanged()
|
||
|
{
|
||
|
return $this->changed;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the timestamp when this queue item was indexed.
|
||
|
*
|
||
|
* @return int|mixed
|
||
|
*/
|
||
|
public function getIndexed()
|
||
|
{
|
||
|
return $this->indexed;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Used to set the timestamp when the related item was changed.
|
||
|
*
|
||
|
* @param int $changed
|
||
|
*/
|
||
|
public function setChanged($changed)
|
||
|
{
|
||
|
$this->changed = intval($changed);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the uid of related record (item_uid).
|
||
|
*
|
||
|
* @return mixed
|
||
|
*/
|
||
|
public function getRecordUid()
|
||
|
{
|
||
|
$this->getRecord();
|
||
|
|
||
|
return $this->record['uid'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the item's full record.
|
||
|
*
|
||
|
* Uses lazy loading.
|
||
|
*
|
||
|
* @return array The item's DB record.
|
||
|
*/
|
||
|
public function getRecord()
|
||
|
{
|
||
|
if (empty($this->record)) {
|
||
|
$this->record = (array)BackendUtility::getRecord(
|
||
|
$this->type,
|
||
|
$this->recordUid,
|
||
|
'*',
|
||
|
'',
|
||
|
false
|
||
|
);
|
||
|
}
|
||
|
|
||
|
return $this->record;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Can be used to set the related record.
|
||
|
*
|
||
|
* @param array $record
|
||
|
*/
|
||
|
public function setRecord(array $record)
|
||
|
{
|
||
|
$this->record = $record;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieves the page id where the related record is stored.
|
||
|
*
|
||
|
* @return int
|
||
|
*/
|
||
|
public function getRecordPageId()
|
||
|
{
|
||
|
$this->getRecord();
|
||
|
|
||
|
return $this->record['pid'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Stores the indexing properties.
|
||
|
*
|
||
|
*/
|
||
|
public function storeIndexingProperties()
|
||
|
{
|
||
|
$this->indexQueueIndexingPropertyRepository->removeByRootPidAndIndexQueueUid(intval($this->rootPageUid), intval($this->indexQueueUid));
|
||
|
|
||
|
if ($this->hasIndexingProperties()) {
|
||
|
$this->writeIndexingProperties();
|
||
|
}
|
||
|
|
||
|
$this->queueItemRepository->updateHasIndexingPropertiesFlagByItemUid($this->indexQueueUid, $this->hasIndexingProperties);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function hasIndexingProperties()
|
||
|
{
|
||
|
return $this->hasIndexingProperties;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Writes all indexing properties.
|
||
|
*/
|
||
|
protected function writeIndexingProperties()
|
||
|
{
|
||
|
$properties = [];
|
||
|
foreach ($this->indexingProperties as $propertyKey => $propertyValue) {
|
||
|
$properties[] = [
|
||
|
'root' => $this->rootPageUid,
|
||
|
'item_id' => $this->indexQueueUid,
|
||
|
'property_key' => $propertyKey,
|
||
|
'property_value' => $propertyValue
|
||
|
];
|
||
|
}
|
||
|
if (empty($properties)) {
|
||
|
return;
|
||
|
}
|
||
|
$this->indexQueueIndexingPropertyRepository->bulkInsert($properties);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $key
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function hasIndexingProperty($key)
|
||
|
{
|
||
|
$this->loadIndexingProperties();
|
||
|
|
||
|
return array_key_exists($key, $this->indexingProperties);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Loads the indexing properties for the item - if not already loaded.
|
||
|
*/
|
||
|
public function loadIndexingProperties()
|
||
|
{
|
||
|
if ($this->indexingPropertiesLoaded) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$indexingProperties = $this->indexQueueIndexingPropertyRepository->findAllByIndexQueueUid(intval($this->indexQueueUid));
|
||
|
$this->indexingPropertiesLoaded = true;
|
||
|
if (empty($indexingProperties)) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
foreach ($indexingProperties as $indexingProperty) {
|
||
|
$this->indexingProperties[$indexingProperty['property_key']] = $indexingProperty['property_value'];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Sets an indexing property for the item.
|
||
|
*
|
||
|
* @param string $key Indexing property name
|
||
|
* @param string|int|float $value Indexing property value
|
||
|
* @throws \InvalidArgumentException when $value is not string, integer or float
|
||
|
*/
|
||
|
public function setIndexingProperty($key, $value)
|
||
|
{
|
||
|
// make sure to not interfere with existing indexing properties
|
||
|
$this->loadIndexingProperties();
|
||
|
|
||
|
$key = (string)$key; // Scalar typehints now!
|
||
|
|
||
|
if (!is_string($value) && !is_int($value) && !is_float($value)) {
|
||
|
throw new \InvalidArgumentException(
|
||
|
'Cannot set indexing property "' . $key
|
||
|
. '", its value must be string, integer or float, '
|
||
|
. 'type given was "' . gettype($value) . '"',
|
||
|
1323173209
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$this->indexingProperties[$key] = $value;
|
||
|
$this->hasIndexingProperties = true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets a specific indexing property by its name/key.
|
||
|
*
|
||
|
* @param string $key Indexing property name/key.
|
||
|
* @throws \InvalidArgumentException when the given $key does not exist.
|
||
|
* @return string
|
||
|
*/
|
||
|
public function getIndexingProperty($key)
|
||
|
{
|
||
|
$this->loadIndexingProperties();
|
||
|
|
||
|
if (!array_key_exists($key, $this->indexingProperties)) {
|
||
|
throw new \InvalidArgumentException(
|
||
|
'No indexing property "' . $key . '".',
|
||
|
1323174143
|
||
|
);
|
||
|
}
|
||
|
|
||
|
return $this->indexingProperties[$key];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets all indexing properties set for this item.
|
||
|
*
|
||
|
* @return array Array of indexing properties.
|
||
|
*/
|
||
|
public function getIndexingProperties()
|
||
|
{
|
||
|
$this->loadIndexingProperties();
|
||
|
|
||
|
return $this->indexingProperties;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the names/keys of the item's indexing properties.
|
||
|
*
|
||
|
* @return array Array of indexing property names/keys
|
||
|
*/
|
||
|
public function getIndexingPropertyKeys()
|
||
|
{
|
||
|
$this->loadIndexingProperties();
|
||
|
|
||
|
return array_keys($this->indexingProperties);
|
||
|
}
|
||
|
}
|