2011-05-02 06:33:05 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Base class to manage active search indexes.
|
|
|
|
*/
|
|
|
|
class FullTextSearch {
|
|
|
|
|
|
|
|
static protected $all_indexes = null;
|
|
|
|
|
|
|
|
static protected $indexes_by_subclass = array();
|
|
|
|
|
2015-05-07 05:16:27 +02:00
|
|
|
/**
|
|
|
|
* Optional list of index names to limit to. If left empty, all subclasses of SearchIndex
|
|
|
|
* will be used
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
* @config
|
|
|
|
*/
|
|
|
|
private static $indexes = array();
|
|
|
|
|
2011-05-02 06:33:05 +02:00
|
|
|
/**
|
|
|
|
* Get all the instantiable search indexes (so all the user created indexes, but not the connector or library level
|
|
|
|
* abstract indexes). Can optionally be filtered to only return indexes that are subclasses of some class
|
|
|
|
*
|
|
|
|
* @static
|
|
|
|
* @param String $class - Class name to filter indexes by, so that all returned indexes are subclasses of provided class
|
|
|
|
* @param bool $rebuild - If true, don't use cached values
|
|
|
|
*/
|
|
|
|
static function get_indexes($class = null, $rebuild = false) {
|
2015-05-07 05:16:27 +02:00
|
|
|
if ($rebuild) {
|
|
|
|
self::$all_indexes = null;
|
|
|
|
self::$indexes_by_subclass = array();
|
|
|
|
}
|
2011-05-02 06:33:05 +02:00
|
|
|
|
|
|
|
if (!$class) {
|
|
|
|
if (self::$all_indexes === null) {
|
2015-05-07 05:16:27 +02:00
|
|
|
// Get declared indexes, or otherwise default to all subclasses of SearchIndex
|
|
|
|
$classes = Config::inst()->get(__CLASS__, 'indexes')
|
|
|
|
?: ClassInfo::subclassesFor('SearchIndex');
|
2011-05-02 06:33:05 +02:00
|
|
|
|
2015-05-07 05:16:27 +02:00
|
|
|
$hidden = array();
|
|
|
|
$candidates = array();
|
2011-05-02 06:33:05 +02:00
|
|
|
foreach ($classes as $class) {
|
2015-05-07 05:16:27 +02:00
|
|
|
// Check if this index is disabled
|
|
|
|
$hides = $class::config()->hide_ancestor;
|
|
|
|
if($hides) {
|
|
|
|
$hidden[] = $hides;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if this index is abstract
|
2011-05-02 06:33:05 +02:00
|
|
|
$ref = new ReflectionClass($class);
|
2015-05-07 05:16:27 +02:00
|
|
|
if (!$ref->isInstantiable()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$candidates[] = $class;
|
|
|
|
}
|
|
|
|
|
|
|
|
if($hidden) {
|
|
|
|
$candidates = array_diff($candidates, $hidden);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create all indexes
|
|
|
|
$concrete = array();
|
|
|
|
foreach($candidates as $class) {
|
|
|
|
$concrete[$class] = singleton($class);
|
2011-05-02 06:33:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
self::$all_indexes = $concrete;
|
|
|
|
}
|
|
|
|
|
|
|
|
return self::$all_indexes;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (!isset(self::$indexes_by_subclass[$class])) {
|
|
|
|
$all = self::get_indexes();
|
|
|
|
|
|
|
|
$valid = array();
|
|
|
|
foreach ($all as $indexclass => $instance) {
|
2012-07-19 02:22:13 +02:00
|
|
|
if (is_subclass_of($indexclass, $class)) $valid[$indexclass] = $instance;
|
2011-05-02 06:33:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
self::$indexes_by_subclass[$class] = $valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
return self::$indexes_by_subclass[$class];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sometimes, like when in tests, you want to restrain the actual indexes to a subset
|
|
|
|
*
|
|
|
|
* Call with one argument - an array of class names, index instances or classname => indexinstance pairs (can be mixed).
|
|
|
|
* Alternatively call with multiple arguments, each of which is a class name or index instance
|
|
|
|
*
|
|
|
|
* From then on, fulltext search system will only see those indexes passed in this most recent call.
|
|
|
|
*
|
|
|
|
* Passing in no arguments resets back to automatic index list
|
2015-05-07 05:16:27 +02:00
|
|
|
*
|
|
|
|
* Alternatively you can use `FullTextSearch.indexes` to configure a list of indexes via config.
|
2011-05-02 06:33:05 +02:00
|
|
|
*/
|
|
|
|
static function force_index_list() {
|
|
|
|
$indexes = func_get_args();
|
|
|
|
|
|
|
|
// No arguments = back to automatic
|
|
|
|
if (!$indexes) {
|
|
|
|
self::get_indexes(null, true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Arguments can be a single array
|
|
|
|
if (is_array($indexes[0])) $indexes = $indexes[0];
|
|
|
|
|
|
|
|
// Reset to empty first
|
|
|
|
self::$all_indexes = array(); self::$indexes_by_subclass = array();
|
|
|
|
|
|
|
|
// And parse out alternative type combos for arguments and add to allIndexes
|
|
|
|
foreach ($indexes as $class => $index) {
|
|
|
|
if (is_string($index)) { $class = $index; $index = singleton($class); }
|
|
|
|
if (is_numeric($class)) $class = get_class($index);
|
|
|
|
|
|
|
|
self::$all_indexes[$class] = $index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|