BUG fix issues with search variants applying to more than one class

This commit is contained in:
Damian Mooyman 2016-04-15 17:59:10 +12:00
parent afe7af18d2
commit e5fbdf9d42
12 changed files with 437 additions and 133 deletions

View File

@ -1,36 +1,35 @@
# See https://github.com/silverstripe-labs/silverstripe-travis-support for setup details # See https://github.com/silverstripe-labs/silverstripe-travis-support for setup details
sudo: false
language: php language: php
sudo: false
php: php:
- 5.3
- 5.4 - 5.4
- 5.5 - 5.5
- 5.6 - 5.6
- 7.0
env: env:
- DB=MYSQL CORE_RELEASE=3.2 - DB=MYSQL CORE_RELEASE=3.2
matrix: matrix:
include: include:
- php: 5.3
env: DB=PGSQL CORE_RELEASE=3.1
- php: 5.6 - php: 5.6
env: DB=MYSQL CORE_RELEASE=3 env: DB=MYSQL CORE_RELEASE=3.2
- php: 5.6 - php: 5.6
env: DB=MYSQL CORE_RELEASE=3.1 env: DB=MYSQL CORE_RELEASE=3.3 SUBSITES=1
- php: 5.6 - php: 5.6
env: DB=PGSQL CORE_RELEASE=3.2 env: DB=MYSQL CORE_RELEASE=3.3 QUEUEDJOBS=1
allow_failures:
- php: 7.0
before_script: before_script:
- composer self-update || true - composer self-update || true
- git clone git://github.com/silverstripe-labs/silverstripe-travis-support.git ~/travis-support - git clone git://github.com/silverstripe-labs/silverstripe-travis-support.git ~/travis-support
- php ~/travis-support/travis_setup.php --source `pwd` --target ~/builds/ss - "if [ \"$SUBSITES\" = \"\" -a \"$QUEUEDJOBS\" = \"\" ]; then php ~/travis-support/travis_setup.php --source `pwd` --target ~/builds/ss; fi"
- "if [ \"$SUBSITES\" = \"1\" ]; then php ~/travis-support/travis_setup.php --source `pwd` --target ~/builds/ss --require silverstripe/subsites; fi"
- "if [ \"$QUEUEDJOBS\" = \"1\" ]; then php ~/travis-support/travis_setup.php --source `pwd` --target ~/builds/ss --require silverstripe/queuedjobs; fi"
- cd ~/builds/ss - cd ~/builds/ss
- composer install
script: script:
- vendor/bin/phpunit fulltextsearch/tests - vendor/bin/phpunit fulltextsearch/tests/

View File

@ -25,7 +25,7 @@
* - Specifying which classes and fields this index contains * - Specifying which classes and fields this index contains
* *
* - Specifying update rules that are not extractable from metadata (because the values come from functions for instance) * - Specifying update rules that are not extractable from metadata (because the values come from functions for instance)
* *
*/ */
abstract class SearchIndex extends ViewableData abstract class SearchIndex extends ViewableData
{ {
@ -354,7 +354,7 @@ abstract class SearchIndex extends ViewableData
/** /**
* Returns an array where each member is all the fields and the classes that are at the end of some * Returns an array where each member is all the fields and the classes that are at the end of some
* specific lookup chain from one of the base classes * specific lookup chain from one of the base classes
*/ */
public function getDerivedFields() public function getDerivedFields()
{ {
@ -391,7 +391,7 @@ abstract class SearchIndex extends ViewableData
/** /**
* Get the "document ID" (a database & variant unique id) given some "Base" class, DataObject ID and state array * Get the "document ID" (a database & variant unique id) given some "Base" class, DataObject ID and state array
* *
* @param String $base - The base class of the object * @param String $base - The base class of the object
* @param Integer $id - The ID of the object * @param Integer $id - The ID of the object
* @param Array $state - The variant state of the object * @param Array $state - The variant state of the object
@ -465,7 +465,7 @@ abstract class SearchIndex extends ViewableData
$method = $step['method']; $method = $step['method'];
$object = $object->$method(); $object = $object->$method();
} elseif ($step['call'] == 'variant') { } elseif ($step['call'] == 'variant') {
$variants = SearchVariant::variants($field['base'], true); $variants = SearchVariant::variants();
$variant = $variants[$step['variant']]; $variant = $variants[$step['variant']];
$method = $step['method']; $method = $step['method'];
$object = $variant->$method($object); $object = $variant->$method($object);
@ -476,6 +476,7 @@ abstract class SearchIndex extends ViewableData
} }
} }
} catch (Exception $e) { } catch (Exception $e) {
static::warn($e);
$object = null; $object = null;
} }
@ -483,6 +484,20 @@ abstract class SearchIndex extends ViewableData
return $object; return $object;
} }
/**
* Log non-fatal errors
*
* @param Exception $e
* @throws Exception
*/
public static function warn($e) {
// Noisy errors during testing
if(class_exists('SapphireTest', false) && SapphireTest::is_running_test()) {
throw $e;
}
SS_Log::log($e, SS_Log::WARN);
}
/** /**
* Given a class, object id, set of stateful ids and a list of changed fields (in a special format), * Given a class, object id, set of stateful ids and a list of changed fields (in a special format),
* return what statefulids need updating in this index * return what statefulids need updating in this index
@ -620,7 +635,7 @@ abstract class SearchIndex_Recording extends SearchIndex
$res = array(); $res = array();
$res['ID'] = $object->ID; $res['ID'] = $object->ID;
foreach ($this->getFieldsIterator() as $name => $field) { foreach ($this->getFieldsIterator() as $name => $field) {
$val = $this->_getFieldValue($object, $field); $val = $this->_getFieldValue($object, $field);
$res[$name] = $val; $res[$name] = $val;
@ -655,7 +670,7 @@ abstract class SearchIndex_Recording extends SearchIndex
{ {
$this->committed = true; $this->committed = true;
} }
public function getIndexName() public function getIndexName()
{ {
return get_class($this); return get_class($this);

View File

@ -123,7 +123,7 @@ abstract class SearchVariant
* @param bool $includeSubclasses - (Optional) If false, only variants that apply strictly to the passed class or its super-classes * @param bool $includeSubclasses - (Optional) If false, only variants that apply strictly to the passed class or its super-classes
* will be checked. If true (the default), variants that apply to any sub-class of the passed class with also be checked * will be checked. If true (the default), variants that apply to any sub-class of the passed class with also be checked
* *
* @return An object with one method, call() * @return SearchVariant_Caller An object with one method, call()
*/ */
public static function with($class = null, $includeSubclasses = true) public static function with($class = null, $includeSubclasses = true)
{ {
@ -197,6 +197,59 @@ abstract class SearchVariant
return $allstates ? new CombinationsArrayIterator($allstates) : array(array()); return $allstates ? new CombinationsArrayIterator($allstates) : array(array());
} }
/**
* Add new filter field to index safely.
*
* This method will respect existing filters with the same field name that
* correspond to multiple classes
*
* @param SearchIndex $index
* @param string $name Field name
* @param array $field Field spec
*/
protected function addFilterField($index, $name, $field) {
// If field already exists, make sure to merge origin / base fields
if(isset($index->filterFields[$name])) {
$field['base'] = $this->mergeClasses(
$index->filterFields[$name]['base'],
$field['base']
);
$field['origin'] = $this->mergeClasses(
$index->filterFields[$name]['origin'],
$field['origin']
);
}
$index->filterFields[$name] = $field;
}
/**
* Merge sets of (or individual) class names together for a search index field.
*
* If there is only one unique class name, then just return it as a string instead of array.
*
* @param array|string $left Left class(es)
* @param array|string $right Right class(es)
* @return array|string List of classes, or single class
*/
protected function mergeClasses($left, $right) {
// Merge together and remove dupes
if(!is_array($left)) {
$left = array($left);
}
if(!is_array($right)) {
$right = array($right);
}
$merged = array_values(array_unique(array_merge($left, $right)));
// If there is only one item, return it as a single string
if(count($merged) === 1) {
return reset($merged);
}
return $merged;
}
} }
/** /**

View File

@ -40,19 +40,19 @@ class SearchVariantSiteTreeSubsitesPolyhome extends SearchVariant
} }
} }
public function alterDefinition($base, $index) public function alterDefinition($class, $index)
{ {
$self = get_class($this); $self = get_class($this);
$index->filterFields['_subsite'] = array( $this->addFilterField($index, '_subsite', array(
'name' => '_subsite', 'name' => '_subsite',
'field' => '_subsite', 'field' => '_subsite',
'fullfield' => '_subsite', 'fullfield' => '_subsite',
'base' => $base, 'base' => ClassInfo::baseDataClass($class),
'origin' => $base, 'origin' => $class,
'type' => 'Int', 'type' => 'Int',
'lookup_chain' => array(array('call' => 'variant', 'variant' => $self, 'method' => 'currentState')) 'lookup_chain' => array(array('call' => 'variant', 'variant' => $self, 'method' => 'currentState'))
); ));
} }
public function alterQuery($query, $index) public function alterQuery($query, $index)

View File

@ -44,29 +44,29 @@ class SearchVariantSubsites extends SearchVariant
Permission::flush_permission_cache(); Permission::flush_permission_cache();
} }
public function alterDefinition($base, $index) public function alterDefinition($class, $index)
{ {
$self = get_class($this); $self = get_class($this);
$index->filterFields['_subsite'] = array( // Add field to root
$this->addFilterField($index, '_subsite', array(
'name' => '_subsite', 'name' => '_subsite',
'field' => '_subsite', 'field' => '_subsite',
'fullfield' => '_subsite', 'fullfield' => '_subsite',
'base' => $base, 'base' => ClassInfo::baseDataClass($class),
'origin' => $base, 'origin' => $class,
'type' => 'Int', 'type' => 'Int',
'lookup_chain' => array(array('call' => 'variant', 'variant' => $self, 'method' => 'currentState')) 'lookup_chain' => array(array('call' => 'variant', 'variant' => $self, 'method' => 'currentState'))
); ));
} }
public function alterQuery($query, $index) public function alterQuery($query, $index)
{ {
$subsite = Subsite::currentSubsiteID(); $subsite = Subsite::currentSubsiteID();
$query->filter('_subsite', array($subsite, SearchQuery::$missing)); $query->filter('_subsite', array($subsite, SearchQuery::$missing));
} }
public static $subsites = null;
/** /**
* We need _really_ complicated logic to find just the changed subsites (because we use versions there's no explicit * We need _really_ complicated logic to find just the changed subsites (because we use versions there's no explicit
* deletes, just new versions with different members) so just always use all of them * deletes, just new versions with different members) so just always use all of them
@ -74,25 +74,27 @@ class SearchVariantSubsites extends SearchVariant
public function extractManipulationWriteState(&$writes) public function extractManipulationWriteState(&$writes)
{ {
$self = get_class($this); $self = get_class($this);
$query = new SQLQuery('"ID"', '"Subsite"');
$subsites = array_merge(array('0'), $query->execute()->column());
foreach ($writes as $key => $write) { foreach ($writes as $key => $write) {
if (!$this->appliesTo($write['class'], true)) { $applies = $this->appliesTo($write['class'], true);
if (!$applies) {
continue; continue;
} }
if (self::$subsites === null) {
$query = new SQLQuery('"ID"', '"Subsite"');
self::$subsites = array_merge(array('0'), $query->execute()->column());
}
$next = array(); $next = array();
foreach ($write['statefulids'] as $i => $statefulid) { foreach ($write['statefulids'] as $i => $statefulid) {
foreach (self::$subsites as $subsiteID) { foreach ($subsites as $subsiteID) {
$next[] = array('id' => $statefulid['id'], 'state' => array_merge($statefulid['state'], array($self => (string)$subsiteID))); $next[] = array(
'id' => $statefulid['id'],
'state' => array_merge(
$statefulid['state'],
array($self => (string)$subsiteID)
)
);
} }
} }
$writes[$key]['statefulids'] = $next; $writes[$key]['statefulids'] = $next;
} }
} }

View File

@ -25,19 +25,19 @@ class SearchVariantVersioned extends SearchVariant
Versioned::reading_stage($state); Versioned::reading_stage($state);
} }
public function alterDefinition($base, $index) public function alterDefinition($class, $index)
{ {
$self = get_class($this); $self = get_class($this);
$index->filterFields['_versionedstage'] = array( $this->addFilterField($index, '_versionedstage', array(
'name' => '_versionedstage', 'name' => '_versionedstage',
'field' => '_versionedstage', 'field' => '_versionedstage',
'fullfield' => '_versionedstage', 'fullfield' => '_versionedstage',
'base' => $base, 'base' => ClassInfo::baseDataClass($class),
'origin' => $base, 'origin' => $class,
'type' => 'String', 'type' => 'String',
'lookup_chain' => array(array('call' => 'variant', 'variant' => $self, 'method' => 'currentState')) 'lookup_chain' => array(array('call' => 'variant', 'variant' => $self, 'method' => 'currentState'))
); ));
} }
public function alterQuery($query, $index) public function alterQuery($query, $index)
@ -45,11 +45,11 @@ class SearchVariantVersioned extends SearchVariant
$stage = Versioned::current_stage(); $stage = Versioned::current_stage();
$query->filter('_versionedstage', array($stage, SearchQuery::$missing)); $query->filter('_versionedstage', array($stage, SearchQuery::$missing));
} }
public function extractManipulationState(&$manipulation) public function extractManipulationState(&$manipulation)
{ {
$self = get_class($this); $self = get_class($this);
foreach ($manipulation as $table => $details) { foreach ($manipulation as $table => $details) {
$class = $details['class']; $class = $details['class'];
$stage = 'Stage'; $stage = 'Stage';

View File

@ -173,7 +173,7 @@ class Solr_BuildTask extends BuildTask
/** /**
* Get the current logger * Get the current logger
* *
* @return LoggerInterface * @return LoggerInterface
*/ */
public function getLogger() public function getLogger()
@ -225,7 +225,7 @@ class Solr_Configure extends Solr_BuildTask
public function run($request) public function run($request)
{ {
parent::run($request); parent::run($request);
// Find the IndexStore handler, which will handle uploading config files to Solr // Find the IndexStore handler, which will handle uploading config files to Solr
$store = $this->getSolrConfigStore(); $store = $this->getSolrConfigStore();
$indexes = Solr::get_indexes(); $indexes = Solr::get_indexes();
@ -240,10 +240,10 @@ class Solr_Configure extends Solr_BuildTask
} }
} }
} }
/** /**
* Update the index on the given store * Update the index on the given store
* *
* @param SolrIndex $instance Instance * @param SolrIndex $instance Instance
* @param SolrConfigStore $store * @param SolrConfigStore $store
*/ */
@ -251,11 +251,11 @@ class Solr_Configure extends Solr_BuildTask
{ {
$index = $instance->getIndexName(); $index = $instance->getIndexName();
$this->getLogger()->info("Configuring $index."); $this->getLogger()->info("Configuring $index.");
// Upload the config files for this index // Upload the config files for this index
$this->getLogger()->info("Uploading configuration ..."); $this->getLogger()->info("Uploading configuration ...");
$instance->uploadConfig($store); $instance->uploadConfig($store);
// Then tell Solr to use those config files // Then tell Solr to use those config files
$service = Solr::service(); $service = Solr::service();
if ($service->coreIsActive($index)) { if ($service->coreIsActive($index)) {
@ -265,23 +265,23 @@ class Solr_Configure extends Solr_BuildTask
$this->getLogger()->info("Creating core ..."); $this->getLogger()->info("Creating core ...");
$service->coreCreate($index, $store->instanceDir($index)); $service->coreCreate($index, $store->instanceDir($index));
} }
$this->getLogger()->info("Done"); $this->getLogger()->info("Done");
} }
/** /**
* Get config store * Get config store
* *
* @return SolrConfigStore * @return SolrConfigStore
*/ */
protected function getSolrConfigStore() protected function getSolrConfigStore()
{ {
$options = Solr::solr_options(); $options = Solr::solr_options();
if (!isset($options['indexstore']) || !($indexstore = $options['indexstore'])) { if (!isset($options['indexstore']) || !($indexstore = $options['indexstore'])) {
user_error('No index configuration for Solr provided', E_USER_ERROR); user_error('No index configuration for Solr provided', E_USER_ERROR);
} }
// Find the IndexStore handler, which will handle uploading config files to Solr // Find the IndexStore handler, which will handle uploading config files to Solr
$mode = $indexstore['mode']; $mode = $indexstore['mode'];
@ -340,7 +340,7 @@ class Solr_Reindex extends Solr_BuildTask
public function run($request) public function run($request)
{ {
parent::run($request); parent::run($request);
// Reset state // Reset state
$originalState = SearchVariant::current_state(); $originalState = SearchVariant::current_state();
$this->doReindex($request); $this->doReindex($request);
@ -411,7 +411,7 @@ class Solr_Reindex extends Solr_BuildTask
protected function runFrom($index, $class, $start, $variantstate) protected function runFrom($index, $class, $start, $variantstate)
{ {
DeprecationTest_Deprecation::notice('2.0.0', 'Solr_Reindex now uses a new grouping mechanism'); DeprecationTest_Deprecation::notice('2.0.0', 'Solr_Reindex now uses a new grouping mechanism');
// Set time limit and state // Set time limit and state
increase_time_limit_to(); increase_time_limit_to();
SearchVariant::activate_state($variantstate); SearchVariant::activate_state($variantstate);

View File

@ -31,7 +31,7 @@ abstract class SolrIndex extends SearchIndex
protected $extrasPath = null; protected $extrasPath = null;
protected $templatesPath = null; protected $templatesPath = null;
/** /**
* List of boosted fields * List of boosted fields
* *
@ -54,7 +54,7 @@ abstract class SolrIndex extends SearchIndex
* @var array * @var array
*/ */
private static $copy_fields = array(); private static $copy_fields = array();
/** /**
* @return String Absolute path to the folder containing * @return String Absolute path to the folder containing
* templates which are used for generating the schema and field definitions. * templates which are used for generating the schema and field definitions.
@ -93,11 +93,11 @@ abstract class SolrIndex extends SearchIndex
/** /**
* Index-time analyzer which is applied to a specific field. * Index-time analyzer which is applied to a specific field.
* Can be used to remove HTML tags, apply stemming, etc. * Can be used to remove HTML tags, apply stemming, etc.
* *
* @see http://wiki.apache.org/solr/AnalyzersTokenizersTokenFilters#solr.WhitespaceTokenizerFactory * @see http://wiki.apache.org/solr/AnalyzersTokenizersTokenFilters#solr.WhitespaceTokenizerFactory
* *
* @param String $field * @param String $field
* @param String $type * @param String $type
* @param Array $params Parameters for the analyzer, usually at least a "class" * @param Array $params Parameters for the analyzer, usually at least a "class"
*/ */
public function addAnalyzer($field, $type, $params) public function addAnalyzer($field, $type, $params)
@ -179,13 +179,13 @@ abstract class SolrIndex extends SearchIndex
} }
$xml[] = $this->getFieldDefinition($name, $field); $xml[] = $this->getFieldDefinition($name, $field);
} }
return implode("\n\t\t", $xml); return implode("\n\t\t", $xml);
} }
/** /**
* Extract first suggestion text from collated values * Extract first suggestion text from collated values
* *
* @param mixed $collation * @param mixed $collation
* @return string * @return string
*/ */
@ -246,10 +246,10 @@ abstract class SolrIndex extends SearchIndex
$options = array_merge($extraOptions, array('stored' => 'true')); $options = array_merge($extraOptions, array('stored' => 'true'));
$this->addFulltextField($field, $forceType, $options); $this->addFulltextField($field, $forceType, $options);
} }
/** /**
* Add a fulltext field with a boosted value * Add a fulltext field with a boosted value
* *
* @param string $field The field to add * @param string $field The field to add
* @param string $forceType The type to force this field as (required in some cases, when not * @param string $forceType The type to force this field as (required in some cases, when not
* detectable from metadata) * detectable from metadata)
@ -261,7 +261,7 @@ abstract class SolrIndex extends SearchIndex
$options = array_merge($extraOptions, array('boost' => $boost)); $options = array_merge($extraOptions, array('boost' => $boost));
$this->addFulltextField($field, $forceType, $options); $this->addFulltextField($field, $forceType, $options);
} }
public function fieldData($field, $forceType = null, $extraOptions = array()) public function fieldData($field, $forceType = null, $extraOptions = array())
{ {
@ -272,7 +272,7 @@ abstract class SolrIndex extends SearchIndex
unset($extraOptions['boost']); unset($extraOptions['boost']);
} }
$data = parent::fieldData($field, $forceType, $extraOptions); $data = parent::fieldData($field, $forceType, $extraOptions);
// Boost all fields with this name // Boost all fields with this name
if (isset($boost)) { if (isset($boost)) {
foreach ($data as $fieldName => $fieldInfo) { foreach ($data as $fieldName => $fieldInfo) {
@ -281,14 +281,14 @@ abstract class SolrIndex extends SearchIndex
} }
return $data; return $data;
} }
/** /**
* Set the default boosting level for a specific field. * Set the default boosting level for a specific field.
* Will control the default value for qf param (Query Fields), but will not * Will control the default value for qf param (Query Fields), but will not
* override a query-specific value. * override a query-specific value.
* *
* Fields must be added before having a field boosting specified * Fields must be added before having a field boosting specified
* *
* @param string $field Full field key (Model_Field) * @param string $field Full field key (Model_Field)
* @param float|null $level Numeric boosting value. Set to null to clear boost * @param float|null $level Numeric boosting value. Set to null to clear boost
*/ */
@ -303,20 +303,20 @@ abstract class SolrIndex extends SearchIndex
$this->boostedFields[$field] = $level; $this->boostedFields[$field] = $level;
} }
} }
/** /**
* Get all boosted fields * Get all boosted fields
* *
* @return array * @return array
*/ */
public function getBoostedFields() public function getBoostedFields()
{ {
return $this->boostedFields; return $this->boostedFields;
} }
/** /**
* Determine the best default value for the 'qf' parameter * Determine the best default value for the 'qf' parameter
* *
* @return array|null List of query fields, or null if not specified * @return array|null List of query fields, or null if not specified
*/ */
public function getQueryFields() public function getQueryFields()
@ -335,7 +335,7 @@ abstract class SolrIndex extends SearchIndex
if ($queryFields && !isset($this->boostedFields[$df])) { if ($queryFields && !isset($this->boostedFields[$df])) {
$queryFields[] = $df; $queryFields[] = $df;
} }
return $queryFields; return $queryFields;
} }
@ -390,7 +390,7 @@ abstract class SolrIndex extends SearchIndex
/** /**
* Convert definition to XML tag * Convert definition to XML tag
* *
* @param String $tag * @param String $tag
* @param String $attrs Map of attributes * @param String $attrs Map of attributes
* @param String $content Inner content * @param String $content Inner content
@ -451,32 +451,53 @@ abstract class SolrIndex extends SearchIndex
return implode("\n\t", $xml); return implode("\n\t", $xml);
} }
/**
* Determine if the given object is one of the given type
*
* @param string $class
* @param array|string $base Class or list of base classes
* @return bool
*/
protected function classIs($class, $base) {
if(is_array($base)) {
foreach($base as $nextBase) {
if($this->classIs($class, $nextBase)) {
return true;
}
}
return false;
}
// Check single origin
return $class === $base || is_subclass_of($class, $base);
}
protected function _addField($doc, $object, $field) protected function _addField($doc, $object, $field)
{ {
$class = get_class($object); $class = get_class($object);
if ($class != $field['origin'] && !is_subclass_of($class, $field['origin'])) { if(!$this->classIs($class, $field['origin'])) {
return; return;
} }
$value = $this->_getFieldValue($object, $field); $value = $this->_getFieldValue($object, $field);
$type = isset(self::$filterTypeMap[$field['type']]) ? self::$filterTypeMap[$field['type']] : self::$filterTypeMap['*']; $type = isset(self::$filterTypeMap[$field['type']]) ? self::$filterTypeMap[$field['type']] : self::$filterTypeMap['*'];
if (is_array($value)) { if (is_array($value)) {
foreach ($value as $sub) { foreach ($value as $sub) {
/* Solr requires dates in the form 1995-12-31T23:59:59Z */ /* Solr requires dates in the form 1995-12-31T23:59:59Z */
if ($type == 'tdate') { if ($type == 'tdate') {
if (!$sub) { if (!$sub) {
continue;
}
$sub = gmdate('Y-m-d\TH:i:s\Z', strtotime($sub));
}
/* Solr requires numbers to be valid if presented, not just empty */
if (($type == 'tint' || $type == 'tfloat' || $type == 'tdouble') && !is_numeric($sub)) {
continue; continue;
} }
$sub = gmdate('Y-m-d\TH:i:s\Z', strtotime($sub));
}
/* Solr requires numbers to be valid if presented, not just empty */
if (($type == 'tint' || $type == 'tfloat' || $type == 'tdouble') && !is_numeric($sub)) {
continue;
}
$doc->addField($field['name'], $sub); $doc->addField($field['name'], $sub);
} }
} else { } else {
@ -516,7 +537,7 @@ abstract class SolrIndex extends SearchIndex
// Add the user-specified fields // Add the user-specified fields
foreach ($this->getFieldsIterator() as $name => $field) { foreach ($this->getFieldsIterator() as $name => $field) {
if ($field['base'] == $base) { if ($field['base'] === $base || (is_array($field['base']) && in_array($base, $field['base']))) {
$this->_addField($doc, $object, $field); $this->_addField($doc, $object, $field);
} }
} }
@ -524,7 +545,7 @@ abstract class SolrIndex extends SearchIndex
try { try {
$this->getService()->addDocument($doc); $this->getService()->addDocument($doc);
} catch (Exception $e) { } catch (Exception $e) {
SS_Log::log($e, SS_Log::WARN); static::warn($e);
return false; return false;
} }
@ -564,7 +585,7 @@ abstract class SolrIndex extends SearchIndex
try { try {
$this->getService()->deleteById($documentID); $this->getService()->deleteById($documentID);
} catch (Exception $e) { } catch (Exception $e) {
SS_Log::log($e, SS_Log::WARN); static::warn($e);
return false; return false;
} }
} }
@ -608,7 +629,7 @@ abstract class SolrIndex extends SearchIndex
try { try {
$this->getService()->commit(false, false, false); $this->getService()->commit(false, false, false);
} catch (Exception $e) { } catch (Exception $e) {
SS_Log::log($e, SS_Log::WARN); static::warn($e);
return false; return false;
} }
} }
@ -618,7 +639,7 @@ abstract class SolrIndex extends SearchIndex
* @param integer $offset * @param integer $offset
* @param integer $limit * @param integer $limit
* @param array $params Extra request parameters passed through to Solr * @param array $params Extra request parameters passed through to Solr
* @return ArrayData Map with the following keys: * @return ArrayData Map with the following keys:
* - 'Matches': ArrayList of the matched object instances * - 'Matches': ArrayList of the matched object instances
*/ */
public function search(SearchQuery $query, $offset = -1, $limit = -1, $params = array()) public function search(SearchQuery $query, $offset = -1, $limit = -1, $params = array())
@ -638,7 +659,7 @@ abstract class SolrIndex extends SearchIndex
// Build the search itself // Build the search itself
$q = $this->getQueryComponent($query, $hlq); $q = $this->getQueryComponent($query, $hlq);
// If using boosting, set the clean term separately for highlighting. // If using boosting, set the clean term separately for highlighting.
// See https://issues.apache.org/jira/browse/SOLR-2632 // See https://issues.apache.org/jira/browse/SOLR-2632
if (array_key_exists('hl', $params) && !array_key_exists('hl.q', $params)) { if (array_key_exists('hl', $params) && !array_key_exists('hl.q', $params)) {
@ -657,10 +678,10 @@ abstract class SolrIndex extends SearchIndex
if ($classq) { if ($classq) {
$fq[] = '+('.implode(' ', $classq).')'; $fq[] = '+('.implode(' ', $classq).')';
} }
// Filter by filters // Filter by filters
$fq = array_merge($fq, $this->getFiltersComponent($query)); $fq = array_merge($fq, $this->getFiltersComponent($query));
// Prepare query fields unless specified explicitly // Prepare query fields unless specified explicitly
if (isset($params['qf'])) { if (isset($params['qf'])) {
$qf = $params['qf']; $qf = $params['qf'];
@ -697,7 +718,7 @@ abstract class SolrIndex extends SearchIndex
} }
$params = array_merge($params, array('fq' => implode(' ', $fq))); $params = array_merge($params, array('fq' => implode(' ', $fq)));
$res = $service->search( $res = $service->search(
$q ? implode(' ', $q) : '*:*', $q ? implode(' ', $q) : '*:*',
$offset, $offset,
@ -751,7 +772,7 @@ abstract class SolrIndex extends SearchIndex
$ret['Matches']->setPageStart($offset); $ret['Matches']->setPageStart($offset);
// Results per page // Results per page
$ret['Matches']->setPageLength($limit); $ret['Matches']->setPageLength($limit);
// Include spellcheck and suggestion data. Requires spellcheck=true in $params // Include spellcheck and suggestion data. Requires spellcheck=true in $params
if (isset($res->spellcheck)) { if (isset($res->spellcheck)) {
// Expose all spellcheck data, for custom handling. // Expose all spellcheck data, for custom handling.
@ -761,7 +782,7 @@ abstract class SolrIndex extends SearchIndex
if (isset($res->spellcheck->suggestions->collation)) { if (isset($res->spellcheck->suggestions->collation)) {
// Extract string suggestion // Extract string suggestion
$suggestion = $this->getCollatedSuggestion($res->spellcheck->suggestions->collation); $suggestion = $this->getCollatedSuggestion($res->spellcheck->suggestions->collation);
// The collation, including advanced query params (e.g. +), suitable for making another query programmatically. // The collation, including advanced query params (e.g. +), suitable for making another query programmatically.
$ret['Suggestion'] = $suggestion; $ret['Suggestion'] = $suggestion;
@ -922,10 +943,10 @@ abstract class SolrIndex extends SearchIndex
$this->service = $service; $this->service = $service;
return $this; return $this;
} }
/** /**
* Upload config for this index to the given store * Upload config for this index to the given store
* *
* @param SolrConfigStore $store * @param SolrConfigStore $store
*/ */
public function uploadConfig($store) public function uploadConfig($store)

View File

@ -0,0 +1,145 @@
<?php
if (class_exists('Phockito')) {
Phockito::include_hamcrest(false);
}
/**
* Subsite specific solr testing
*/
class SolrIndexSubsitesTest extends SapphireTest {
public static $fixture_file = 'SolrIndexSubsitesTest.yml';
/**
* @var SolrIndexSubsitesTest_Index
*/
private static $index = null;
protected $server = null;
public function setUp()
{
// Prevent parent::setUp() crashing on db build
if (!class_exists('Subsite')) {
$this->skipTest = true;
}
parent::setUp();
$this->server = $_SERVER;
if (!class_exists('Phockito')) {
$this->skipTest = true;
$this->markTestSkipped("These tests need the Phockito module installed to run");
return;
}
// Check versioned available
if (!class_exists('Subsite')) {
$this->skipTest = true;
$this->markTestSkipped('The subsite module is not installed');
return;
}
if (self::$index === null) {
self::$index = singleton('SolrIndexSubsitesTest_Index');
}
SearchUpdater::bind_manipulation_capture();
Config::inst()->update('Injector', 'SearchUpdateProcessor', array(
'class' => 'SearchUpdateImmediateProcessor'
));
FullTextSearch::force_index_list(self::$index);
SearchUpdater::clear_dirty_indexes();
}
public function tearDown()
{
if($this->server) {
$_SERVER = $this->server;
$this->server = null;
}
parent::tearDown();
}
protected function getServiceMock()
{
return Phockito::mock('Solr4Service');
}
/**
* @param DataObject $object Item being added
* @param int $subsiteID
* @param string $stage
* @return string
*/
protected function getExpectedDocumentId($object, $subsiteID, $stage = null)
{
$id = $object->ID;
$class = ClassInfo::baseDataClass($object);
$variants = array();
// Check subsite
if(class_exists('Subsite') && $object->hasOne('Subsite')) {
$variants[] = '"SearchVariantSubsites":"' . $subsiteID. '"';
}
// Check versioned
if($stage) {
$variants[] = '"SearchVariantVersioned":"' . $stage . '"';
}
return $id.'-'.$class.'-{'.implode(',',$variants).'}';
}
public function testPublishing()
{
// Setup mocks
$serviceMock = $this->getServiceMock();
self::$index->setService($serviceMock);
$subsite1 = $this->objFromFixture('Subsite', 'subsite1');
// Add records to first subsite
Versioned::reading_stage('Stage');
$_SERVER['HTTP_HOST'] = 'www.subsite1.com';
Phockito::reset($serviceMock);
$file = new File();
$file->Title = 'My File';
$file->SubsiteID = $subsite1->ID;
$file->write();
$page = new Page();
$page->Title = 'My Page';
$page->SubsiteID = $subsite1->ID;
$page->write();
SearchUpdater::flush_dirty_indexes();
$doc1 = new SolrDocumentMatcher(array(
'_documentid' => $this->getExpectedDocumentId($page, $subsite1->ID, 'Stage'),
'ClassName' => 'Page',
'SiteTree_Title' => 'My Page',
'_versionedstage' => 'Stage',
'_subsite' => $subsite1->ID
));
$doc2 = new SolrDocumentMatcher(array(
'_documentid' => $this->getExpectedDocumentId($file, $subsite1->ID),
'ClassName' => 'File',
'File_Title' => 'My File',
'_subsite' => $subsite1->ID
));
Phockito::verify($serviceMock)->addDocument($doc1);
Phockito::verify($serviceMock)->addDocument($doc2);
}
}
class SolrIndexSubsitesTest_Index extends SolrIndex
{
public function init()
{
$this->addClass('File');
$this->addClass('SiteTree');
$this->addAllFulltextFields();
}
}

View File

@ -0,0 +1,16 @@
Subsite:
main:
Title: Template
subsite1:
Title: 'Subsite1 Template'
subsite2:
Title: 'Subsite2 Template'
SubsiteDomain:
subsite1:
SubsiteID: =>Subsite.subsite1
Domain: www.subsite1.com
Protocol: automatic
subsite2:
SubsiteID: =>Subsite.subsite2
Domain: www.subsite2.com
Protocol: automatic

View File

@ -11,7 +11,8 @@ class SolrIndexVersionedTest extends SapphireTest
protected static $index = null; protected static $index = null;
protected $extraDataObjects = array( protected $extraDataObjects = array(
'SearchVariantVersionedTest_Item' 'SearchVariantVersionedTest_Item',
'SolrIndexVersionedTest_Object',
); );
public function setUp() public function setUp()
@ -20,13 +21,15 @@ class SolrIndexVersionedTest extends SapphireTest
if (!class_exists('Phockito')) { if (!class_exists('Phockito')) {
$this->skipTest = true; $this->skipTest = true;
return $this->markTestSkipped("These tests need the Phockito module installed to run"); $this->markTestSkipped("These tests need the Phockito module installed to run");
return;
} }
// Check versioned available // Check versioned available
if (!class_exists('Versioned')) { if (!class_exists('Versioned')) {
$this->skipTest = true; $this->skipTest = true;
return $this->markTestSkipped('The versioned decorator is not installed'); $this->markTestSkipped('The versioned decorator is not installed');
return;
} }
if (self::$index === null) { if (self::$index === null) {
@ -57,11 +60,21 @@ class SolrIndexVersionedTest extends SapphireTest
return Phockito::mock('Solr3Service'); return Phockito::mock('Solr3Service');
} }
protected function getExpectedDocumentId($id, $stage) /**
* @param DataObject $object Item being added
* @param string $stage
* @return string
*/
protected function getExpectedDocumentId($object, $stage)
{ {
$id = $object->ID;
$class = ClassInfo::baseDataClass($object);
// Prevent subsites from breaking tests // Prevent subsites from breaking tests
$subsites = class_exists('Subsite') ? '"SearchVariantSubsites":"0",' : ''; $subsites = '';
return $id.'-SiteTree-{'.$subsites.'"SearchVariantVersioned":"'.$stage.'"}'; if(class_exists('Subsite') && $object->hasOne('Subsite')) {
$subsites = '"SearchVariantSubsites":"0",';
}
return $id.'-'.$class.'-{'.$subsites.'"SearchVariantVersioned":"'.$stage.'"}';
} }
public function testPublishing() public function testPublishing()
@ -74,32 +87,54 @@ class SolrIndexVersionedTest extends SapphireTest
// Check that write updates Stage // Check that write updates Stage
Versioned::reading_stage('Stage'); Versioned::reading_stage('Stage');
Phockito::reset($serviceMock); Phockito::reset($serviceMock);
$item = new SearchVariantVersionedTest_Item(array('Title' => 'Foo')); $item = new SearchVariantVersionedTest_Item(array('TestText' => 'Foo'));
$item->write(); $item->write();
$object = new SolrIndexVersionedTest_Object(array('TestText' => 'Bar'));
$object->write();
SearchUpdater::flush_dirty_indexes(); SearchUpdater::flush_dirty_indexes();
$doc = new SolrDocumentMatcher(array( $doc1 = new SolrDocumentMatcher(array(
'_documentid' => $this->getExpectedDocumentId($item->ID, 'Stage'), '_documentid' => $this->getExpectedDocumentId($item, 'Stage'),
'ClassName' => 'SearchVariantVersionedTest_Item' 'ClassName' => 'SearchVariantVersionedTest_Item',
'SearchVariantVersionedTest_Item_TestText' => 'Foo',
'_versionedstage' => 'Stage'
)); ));
Phockito::verify($serviceMock)->addDocument($doc); $doc2 = new SolrDocumentMatcher(array(
'_documentid' => $this->getExpectedDocumentId($object, 'Stage'),
'ClassName' => 'SolrIndexVersionedTest_Object',
'SolrIndexVersionedTest_Object_TestText' => 'Bar',
'_versionedstage' => 'Stage'
));
Phockito::verify($serviceMock)->addDocument($doc1);
Phockito::verify($serviceMock)->addDocument($doc2);
// Check that write updates Live // Check that write updates Live
Versioned::reading_stage('Stage'); Versioned::reading_stage('Stage');
Phockito::reset($serviceMock); Phockito::reset($serviceMock);
$item = new SearchVariantVersionedTest_Item(array('Title' => 'Bar')); $item = new SearchVariantVersionedTest_Item(array('TestText' => 'Foo'));
$item->write(); $item->write();
$item->publish('Stage', 'Live'); $item->publish('Stage', 'Live');
$object = new SolrIndexVersionedTest_Object(array('TestText' => 'Bar'));
$object->write();
$object->publish('Stage', 'Live');
SearchUpdater::flush_dirty_indexes(); SearchUpdater::flush_dirty_indexes();
$doc = new SolrDocumentMatcher(array( $doc = new SolrDocumentMatcher(array(
'_documentid' => $this->getExpectedDocumentId($item->ID, 'Live'), '_documentid' => $this->getExpectedDocumentId($item, 'Live'),
'ClassName' => 'SearchVariantVersionedTest_Item' 'ClassName' => 'SearchVariantVersionedTest_Item',
'SearchVariantVersionedTest_Item_TestText' => 'Foo',
'_versionedstage' => 'Live'
));
$doc2 = new SolrDocumentMatcher(array(
'_documentid' => $this->getExpectedDocumentId($object, 'Live'),
'ClassName' => 'SolrIndexVersionedTest_Object',
'SolrIndexVersionedTest_Object_TestText' => 'Bar',
'_versionedstage' => 'Live'
)); ));
Phockito::verify($serviceMock)->addDocument($doc); Phockito::verify($serviceMock)->addDocument($doc);
Phockito::verify($serviceMock)->addDocument($doc2);
} }
public function testDelete() public function testDelete()
{ {
// Setup mocks // Setup mocks
$serviceMock = $this->getServiceMock(); $serviceMock = $this->getServiceMock();
self::$index->setService($serviceMock); self::$index->setService($serviceMock);
@ -107,11 +142,11 @@ class SolrIndexVersionedTest extends SapphireTest
// Delete the live record (not the stage) // Delete the live record (not the stage)
Versioned::reading_stage('Stage'); Versioned::reading_stage('Stage');
Phockito::reset($serviceMock); Phockito::reset($serviceMock);
$item = new SearchVariantVersionedTest_Item(array('Title' => 'Too')); $item = new SearchVariantVersionedTest_Item(array('TestText' => 'Too'));
$item->write(); $item->write();
$item->publish('Stage', 'Live'); $item->publish('Stage', 'Live');
Versioned::reading_stage('Live'); Versioned::reading_stage('Live');
$id = $item->ID; $id = clone $item;
$item->delete(); $item->delete();
SearchUpdater::flush_dirty_indexes(); SearchUpdater::flush_dirty_indexes();
Phockito::verify($serviceMock, 1) Phockito::verify($serviceMock, 1)
@ -122,10 +157,10 @@ class SolrIndexVersionedTest extends SapphireTest
// Delete the stage record // Delete the stage record
Versioned::reading_stage('Stage'); Versioned::reading_stage('Stage');
Phockito::reset($serviceMock); Phockito::reset($serviceMock);
$item = new SearchVariantVersionedTest_Item(array('Title' => 'Too')); $item = new SearchVariantVersionedTest_Item(array('TestText' => 'Too'));
$item->write(); $item->write();
$item->publish('Stage', 'Live'); $item->publish('Stage', 'Live');
$id = $item->ID; $id = clone $item;
$item->delete(); $item->delete();
SearchUpdater::flush_dirty_indexes(); SearchUpdater::flush_dirty_indexes();
Phockito::verify($serviceMock, 1) Phockito::verify($serviceMock, 1)
@ -141,7 +176,9 @@ class SolrVersionedTest_Index extends SolrIndex
public function init() public function init()
{ {
$this->addClass('SearchVariantVersionedTest_Item'); $this->addClass('SearchVariantVersionedTest_Item');
$this->addClass('SolrIndexVersionedTest_Object');
$this->addFilterField('TestText'); $this->addFilterField('TestText');
$this->addFulltextField('Content');
} }
} }
@ -178,3 +215,19 @@ class SolrDocumentMatcher extends Hamcrest_BaseMatcher
return true; return true;
} }
} }
/**
* Non-sitetree versioned dataobject
*/
class SolrIndexVersionedTest_Object extends DataObject implements TestOnly {
private static $extensions = array(
'Versioned'
);
private static $db = array(
'Title' => 'Varchar',
'Content' => 'Text',
'TestText' => 'Varchar',
);
}

View File

@ -39,7 +39,7 @@ class SolrReindexTest extends SapphireTest
$this->skipTest = true; $this->skipTest = true;
return $this->markTestSkipped("These tests need the Phockito module installed to run"); return $this->markTestSkipped("These tests need the Phockito module installed to run");
} }
// Set test handler for reindex // Set test handler for reindex
Config::inst()->update('Injector', 'SolrReindexHandler', array( Config::inst()->update('Injector', 'SolrReindexHandler', array(
'class' => 'SolrReindexTest_TestHandler' 'class' => 'SolrReindexTest_TestHandler'
@ -416,19 +416,19 @@ class SolrReindexTest_Variant extends SearchVariant implements TestOnly
} }
} }
public function alterDefinition($base, $index) public function alterDefinition($class, $index)
{ {
$self = get_class($this); $self = get_class($this);
$index->filterFields['_testvariant'] = array( $this->addFilterField($index, '_testvariant', array(
'name' => '_testvariant', 'name' => '_testvariant',
'field' => '_testvariant', 'field' => '_testvariant',
'fullfield' => '_testvariant', 'fullfield' => '_testvariant',
'base' => $base, 'base' => ClassInfo::baseDataClass($class),
'origin' => $base, 'origin' => $class,
'type' => 'Int', 'type' => 'Int',
'lookup_chain' => array(array('call' => 'variant', 'variant' => $self, 'method' => 'currentState')) 'lookup_chain' => array(array('call' => 'variant', 'variant' => $self, 'method' => 'currentState'))
); ));
} }
public function alterQuery($query, $index) public function alterQuery($query, $index)