2017-02-17 12:03:51 +13:00
|
|
|
<?php
|
|
|
|
namespace SilverStripe\FullTextSearch\Solr\Tasks;
|
2017-04-23 06:21:06 +12:00
|
|
|
use ReflectionClass;
|
|
|
|
use SilverStripe\Core\ClassInfo;
|
|
|
|
use SilverStripe\Core\Injector\Injector;
|
|
|
|
use SilverStripe\FullTextSearch\Search\Variants\SearchVariant;
|
|
|
|
use SilverStripe\ORM\DataList;
|
2017-04-25 21:11:29 +12:00
|
|
|
use SilverStripe\FullTextSearch\Solr\Reindex\Handlers\SolrReindexHandler;
|
|
|
|
use SilverStripe\FullTextSearch\Solr\SolrIndex;
|
2017-04-23 06:21:06 +12:00
|
|
|
|
2017-02-17 12:03:51 +13:00
|
|
|
/**
|
|
|
|
* Task used for both initiating a new reindex, as well as for processing incremental batches
|
|
|
|
* within a reindex.
|
|
|
|
*
|
|
|
|
* When running a complete reindex you can provide any of the following
|
|
|
|
* - class (to limit to a single class)
|
|
|
|
* - verbose (optional)
|
|
|
|
*
|
|
|
|
* When running with a single batch, provide the following querystring arguments:
|
|
|
|
* - start
|
|
|
|
* - index
|
|
|
|
* - class
|
|
|
|
* - variantstate
|
|
|
|
* - verbose (optional)
|
|
|
|
*/
|
|
|
|
class Solr_Reindex extends Solr_BuildTask
|
|
|
|
{
|
|
|
|
private static $segment = 'Solr_Reindex';
|
|
|
|
|
|
|
|
protected $enabled = true;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Number of records to load and index per request
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
* @config
|
|
|
|
*/
|
|
|
|
private static $recordsPerRequest = 200;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the reindex handler
|
|
|
|
*
|
|
|
|
* @return SolrReindexHandler
|
|
|
|
*/
|
|
|
|
protected function getHandler()
|
|
|
|
{
|
2017-04-26 20:58:14 +12:00
|
|
|
|
2017-02-17 12:03:51 +13:00
|
|
|
//@todo: this needs to determine the best class from a Factory implementation
|
|
|
|
//@todo: it was 'SolrReindexHandler' but that doesn't work on 4.0
|
2017-04-26 20:58:14 +12:00
|
|
|
//@todo left commented after a confusing merge conflict. Revisit if further investigation /testing is needed
|
|
|
|
//return Injector::inst()->get('SolrReindexImmediateHandler');
|
2017-04-25 21:11:29 +12:00
|
|
|
return Injector::inst()->get(SolrReindexHandler::class);
|
2017-02-17 12:03:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param SS_HTTPRequest $request
|
|
|
|
*/
|
|
|
|
public function run($request)
|
|
|
|
{
|
|
|
|
parent::run($request);
|
|
|
|
|
|
|
|
// Reset state
|
|
|
|
$originalState = SearchVariant::current_state();
|
|
|
|
$this->doReindex($request);
|
|
|
|
SearchVariant::activate_state($originalState);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param SS_HTTPRequest $request
|
|
|
|
*/
|
|
|
|
protected function doReindex($request)
|
|
|
|
{
|
|
|
|
$class = $request->getVar('class');
|
|
|
|
|
|
|
|
$index = $request->getVar('index');
|
|
|
|
|
|
|
|
//find the index classname by IndexName
|
|
|
|
// this is for when index names do not match the class name (this can be done by overloading getIndexName() on
|
|
|
|
// indexes
|
|
|
|
if ($index && !ClassInfo::exists($index)) {
|
2017-04-25 21:11:29 +12:00
|
|
|
foreach(ClassInfo::subclassesFor(SolrIndex::class) as $solrIndexClass) {
|
2017-02-17 12:03:51 +13:00
|
|
|
$reflection = new ReflectionClass($solrIndexClass);
|
|
|
|
//skip over abstract classes
|
|
|
|
if (!$reflection->isInstantiable()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
//check the indexname matches the index passed to the request
|
|
|
|
if (!strcasecmp(singleton($solrIndexClass)->getIndexName(), $index)) {
|
|
|
|
//if we match, set the correct index name and move on
|
|
|
|
$index = $solrIndexClass;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deprecated reindex mechanism
|
|
|
|
$start = $request->getVar('start');
|
|
|
|
if ($start !== null) {
|
|
|
|
// Run single batch directly
|
|
|
|
$indexInstance = singleton($index);
|
|
|
|
$state = json_decode($request->getVar('variantstate'), true);
|
|
|
|
$this->runFrom($indexInstance, $class, $start, $state);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if we are re-indexing a single group
|
|
|
|
// If not using queuedjobs, we need to invoke Solr_Reindex as a separate process
|
|
|
|
// Otherwise each group is processed via a SolrReindexGroupJob
|
|
|
|
$groups = $request->getVar('groups');
|
|
|
|
|
|
|
|
$handler = $this->getHandler();
|
|
|
|
if ($groups) {
|
|
|
|
// Run grouped batches (id % groups = group)
|
|
|
|
$group = $request->getVar('group');
|
|
|
|
$indexInstance = singleton($index);
|
|
|
|
$state = json_decode($request->getVar('variantstate'), true);
|
|
|
|
|
|
|
|
$handler->runGroup($this->getLogger(), $indexInstance, $state, $class, $groups, $group);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If run at the top level, delegate to appropriate handler
|
|
|
|
$self = get_class($this);
|
|
|
|
$handler->triggerReindex($this->getLogger(), $this->config()->recordsPerRequest, $self, $class);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @deprecated since version 2.0.0
|
|
|
|
*/
|
|
|
|
protected function runFrom($index, $class, $start, $variantstate)
|
|
|
|
{
|
|
|
|
DeprecationTest_Deprecation::notice('2.0.0', 'Solr_Reindex now uses a new grouping mechanism');
|
|
|
|
|
|
|
|
// Set time limit and state
|
|
|
|
increase_time_limit_to();
|
|
|
|
SearchVariant::activate_state($variantstate);
|
|
|
|
|
|
|
|
// Generate filtered list
|
|
|
|
$items = DataList::create($class)
|
|
|
|
->limit($this->config()->recordsPerRequest, $start);
|
|
|
|
|
|
|
|
// Add child filter
|
|
|
|
$classes = $index->getClasses();
|
|
|
|
$options = $classes[$class];
|
|
|
|
if (!$options['include_children']) {
|
|
|
|
$items = $items->filter('ClassName', $class);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process selected records in this class
|
|
|
|
$this->getLogger()->info("Adding $class");
|
|
|
|
foreach ($items->sort("ID") as $item) {
|
|
|
|
$this->getLogger()->debug($item->ID);
|
|
|
|
|
|
|
|
// See SearchUpdater_ObjectHandler::triggerReindex
|
|
|
|
$item->triggerReindex();
|
|
|
|
$item->destroy();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->getLogger()->info("Done");
|
|
|
|
}
|
|
|
|
}
|