2011-05-02 16:33:05 +12:00
|
|
|
<?php
|
|
|
|
|
|
|
|
class Solr {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Configuration on where to find the solr server and how to get new index configurations into it.
|
|
|
|
*
|
|
|
|
* Required fields:
|
|
|
|
* host (default: localhost) - The host or IP Solr is listening on
|
|
|
|
* port (default: 8983) - The port Solr is listening on
|
|
|
|
* path (default: /solr) - The suburl the solr service is available on
|
|
|
|
*
|
2012-09-05 18:09:45 +02:00
|
|
|
* Optional fields:
|
2013-11-08 13:12:43 +13:00
|
|
|
* version (default: 4) - The Solr server version. Currently supports 3 and 4 (you can add a sub-version like 4.5 if
|
|
|
|
* you like, but currently it has no effect)
|
|
|
|
* service (default: depends on version, Solr3Service for 3, Solr4Service for 4)
|
|
|
|
* the class that provides actual communcation to the Solr server
|
|
|
|
* extraspath (default: <basefolder>/fulltextsearch/conf/solr/{version}/extras/) - Absolute path to
|
2012-09-05 18:09:45 +02:00
|
|
|
* the folder containing templates which are used for generating the schema and field definitions.
|
2013-11-08 13:12:43 +13:00
|
|
|
* templates (default: <basefolder>/fulltextsearch/conf/solr/{version}/templates/) - Absolute path to
|
2012-09-05 18:09:45 +02:00
|
|
|
* the configuration default files, e.g. solrconfig.xml.
|
|
|
|
*
|
2011-05-02 16:33:05 +12:00
|
|
|
* indexstore => an array with
|
|
|
|
*
|
2013-08-29 10:26:55 +12:00
|
|
|
* mode - a classname which implements SolrConfigStore, or 'file' or 'webdav'
|
2011-05-02 16:33:05 +12:00
|
|
|
*
|
2013-08-29 10:26:55 +12:00
|
|
|
* When mode == SolrConfigStore_File or file (indexes should be written on a local filesystem)
|
2011-05-02 16:33:05 +12:00
|
|
|
* path - The (locally accessible) path to write the index configurations to.
|
|
|
|
* remotepath (default: the same as indexpath) - The path that the Solr server will read the index configurations from
|
|
|
|
*
|
2013-08-29 10:26:55 +12:00
|
|
|
* When mode == SolrConfigStore_WebDAV or webdav (indexes should stored on a remote Solr server via webdav)
|
2011-05-02 16:33:05 +12:00
|
|
|
* auth (default: none) - A username:password pair string to use to auth against the webdav server
|
|
|
|
* path (default: /solrindex) - The suburl on the solr host that is set up to accept index configurations via webdav
|
|
|
|
* remotepath - The path that the Solr server will read the index configurations from
|
|
|
|
*/
|
2013-11-08 13:12:43 +13:00
|
|
|
protected static $solr_options = array();
|
2011-05-02 16:33:05 +12:00
|
|
|
|
2013-11-08 13:12:43 +13:00
|
|
|
/** A cache of solr_options with the defaults all merged in */
|
|
|
|
protected static $merged_solr_options = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update the configuration for Solr. See $solr_options for a discussion of the accepted array keys
|
|
|
|
* @param array $options - The options to update
|
|
|
|
*/
|
2011-05-02 16:33:05 +12:00
|
|
|
static function configure_server($options = array()) {
|
2013-11-08 13:12:43 +13:00
|
|
|
self::$solr_options = array_merge(self::$solr_options, $options);
|
|
|
|
self::$merged_solr_options = null;
|
|
|
|
|
|
|
|
self::$service_singleton = null;
|
|
|
|
self::$service_core_singletons = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the configured Solr options with the defaults all merged in
|
|
|
|
* @return array - The merged options
|
|
|
|
*/
|
|
|
|
static function solr_options() {
|
|
|
|
if (self::$merged_solr_options) return self::$merged_solr_options;
|
|
|
|
|
|
|
|
$defaults = array(
|
2011-05-02 16:33:05 +12:00
|
|
|
'host' => 'localhost',
|
|
|
|
'port' => 8983,
|
2012-09-05 18:09:45 +02:00
|
|
|
'path' => '/solr',
|
2013-11-08 13:12:43 +13:00
|
|
|
'version' => '4'
|
|
|
|
);
|
|
|
|
|
|
|
|
// Build some by-version defaults
|
|
|
|
$version = isset(self::$solr_options['version']) ? self::$solr_options['version'] : $defaults['version'];
|
|
|
|
|
|
|
|
if (version_compare($version, '4', '>=')){
|
|
|
|
$versionDefaults = array(
|
|
|
|
'service' => 'Solr4Service',
|
|
|
|
'extraspath' => Director::baseFolder().'/fulltextsearch/conf/solr/4/extras/',
|
|
|
|
'templatespath' => Director::baseFolder().'/fulltextsearch/conf/solr/4/templates/',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$versionDefaults = array(
|
|
|
|
'service' => 'Solr3Service',
|
|
|
|
'extraspath' => Director::baseFolder().'/fulltextsearch/conf/solr/3/extras/',
|
|
|
|
'templatespath' => Director::baseFolder().'/fulltextsearch/conf/solr/3/templates/',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (self::$merged_solr_options = array_merge($defaults, $versionDefaults, self::$solr_options));
|
2011-05-02 16:33:05 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static function set_service_class($class) {
|
2013-11-08 13:12:43 +13:00
|
|
|
user_error('set_service_class is deprecated - pass as part of $options to configure_server', E_USER_WARNING);
|
|
|
|
self::configure_server(array('service' => $class));
|
2011-05-02 16:33:05 +12:00
|
|
|
}
|
|
|
|
|
2013-11-08 13:12:43 +13:00
|
|
|
/** @var SolrService | null - The instance of SolrService for core management */
|
|
|
|
static protected $service_singleton = null;
|
|
|
|
/** @var [SolrService_Core] - The instances of SolrService_Core for each core */
|
|
|
|
static protected $service_core_singletons = array();
|
2011-05-02 16:33:05 +12:00
|
|
|
|
2015-06-25 10:52:56 +12:00
|
|
|
/**
|
|
|
|
* Get a SolrService
|
|
|
|
*
|
|
|
|
* @param string $core Optional core name
|
|
|
|
* @return SolrService_Core
|
|
|
|
*/
|
2011-05-02 16:33:05 +12:00
|
|
|
static function service($core = null) {
|
2013-11-08 13:12:43 +13:00
|
|
|
$options = self::solr_options();
|
2011-05-02 16:33:05 +12:00
|
|
|
|
2013-11-08 13:12:43 +13:00
|
|
|
if (!self::$service_singleton) {
|
|
|
|
self::$service_singleton = Object::create(
|
|
|
|
$options['service'], $options['host'], $options['port'], $options['path']
|
|
|
|
);
|
2011-05-02 16:33:05 +12:00
|
|
|
}
|
|
|
|
|
2013-11-08 13:12:43 +13:00
|
|
|
if ($core) {
|
|
|
|
if (!isset(self::$service_core_singletons[$core])) {
|
2014-01-15 12:44:33 +13:00
|
|
|
self::$service_core_singletons[$core] = self::$service_singleton->serviceForCore(
|
|
|
|
singleton($core)->getIndexName()
|
|
|
|
);
|
2013-11-08 13:12:43 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
return self::$service_core_singletons[$core];
|
|
|
|
} else {
|
|
|
|
return self::$service_singleton;
|
|
|
|
}
|
2011-05-02 16:33:05 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
static function get_indexes() {
|
|
|
|
return FullTextSearch::get_indexes('SolrIndex');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-11-08 13:12:43 +13:00
|
|
|
* Include the thirdparty Solr client api library. Done this way to avoid issues where code is called in
|
|
|
|
* mysite/_config before fulltextsearch/_config has a change to update the include path.
|
2011-05-02 16:33:05 +12:00
|
|
|
*/
|
|
|
|
static function include_client_api() {
|
|
|
|
static $included = false;
|
|
|
|
|
|
|
|
if (!$included) {
|
2012-07-19 00:15:57 +02:00
|
|
|
set_include_path(get_include_path() . PATH_SEPARATOR . Director::baseFolder() . '/fulltextsearch/thirdparty/solr-php-client');
|
2011-05-02 16:33:05 +12:00
|
|
|
require_once('Apache/Solr/Service.php');
|
|
|
|
require_once('Apache/Solr/Document.php');
|
|
|
|
|
|
|
|
$included = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-29 10:26:55 +12:00
|
|
|
|
2011-05-02 16:33:05 +12:00
|
|
|
class Solr_Configure extends BuildTask {
|
|
|
|
|
|
|
|
public function run($request) {
|
2015-06-25 10:52:56 +12:00
|
|
|
// Find the IndexStore handler, which will handle uploading config files to Solr
|
|
|
|
$store = $this->getSolrConfigStore();
|
2013-02-13 10:15:14 +13:00
|
|
|
$indexes = Solr::get_indexes();
|
2015-06-25 10:52:56 +12:00
|
|
|
foreach ($indexes as $instance) {
|
2011-05-02 16:33:05 +12:00
|
|
|
|
2015-06-25 10:52:56 +12:00
|
|
|
try {
|
|
|
|
$this->updateIndex($instance, $store);
|
|
|
|
} catch(Exception $e) {
|
|
|
|
// We got an exception. Warn, but continue to next index.
|
|
|
|
$this->log("Failure: " . $e->getMessage());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update the index on the given store
|
|
|
|
*
|
|
|
|
* @param SolrIndex $instance Instance
|
|
|
|
* @param SolrConfigStore $store
|
|
|
|
*/
|
|
|
|
protected function updateIndex($instance, $store) {
|
|
|
|
$index = $instance->getIndexName();
|
|
|
|
$this->log("Configuring $index.");
|
|
|
|
$this->log("Uploading configuration ... ");
|
|
|
|
|
|
|
|
|
|
|
|
// Upload the config files for this index
|
|
|
|
$instance->uploadConfig($store);
|
|
|
|
|
|
|
|
// Then tell Solr to use those config files
|
|
|
|
$service = Solr::service();
|
|
|
|
if ($service->coreIsActive($index)) {
|
|
|
|
$this->log("Reloading core ...");
|
|
|
|
$service->coreReload($index);
|
|
|
|
} else {
|
|
|
|
$this->log("Creating core ...");
|
|
|
|
$service->coreCreate($index, $store->instanceDir($index));
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->log("Done");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get config store
|
|
|
|
*
|
|
|
|
* @return SolrConfigStore
|
|
|
|
*/
|
|
|
|
protected function getSolrConfigStore() {
|
|
|
|
$options = Solr::solr_options();
|
|
|
|
|
2013-11-08 13:12:43 +13:00
|
|
|
if (!isset($options['indexstore']) || !($indexstore = $options['indexstore'])) {
|
2011-05-02 16:33:05 +12:00
|
|
|
user_error('No index configuration for Solr provided', E_USER_ERROR);
|
|
|
|
}
|
2015-06-25 10:52:56 +12:00
|
|
|
|
2013-08-29 10:26:55 +12:00
|
|
|
// Find the IndexStore handler, which will handle uploading config files to Solr
|
|
|
|
$mode = $indexstore['mode'];
|
|
|
|
|
|
|
|
if ($mode == 'file') {
|
2015-06-25 10:52:56 +12:00
|
|
|
return new SolrConfigStore_File($indexstore);
|
2013-08-29 10:26:55 +12:00
|
|
|
} elseif ($mode == 'webdav') {
|
2015-06-25 10:52:56 +12:00
|
|
|
return new SolrConfigStore_WebDAV($indexstore);
|
2013-08-29 10:26:55 +12:00
|
|
|
} elseif (ClassInfo::exists($mode) && ClassInfo::classImplements($mode, 'SolrConfigStore')) {
|
2015-06-25 10:52:56 +12:00
|
|
|
return new $mode($indexstore);
|
2013-08-29 10:26:55 +12:00
|
|
|
} else {
|
|
|
|
user_error('Unknown Solr index mode '.$indexstore['mode'], E_USER_ERROR);
|
|
|
|
}
|
2015-06-25 10:52:56 +12:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function log($message) {
|
|
|
|
if(Director::is_cli()) {
|
|
|
|
echo $message . "\n";
|
|
|
|
} else {
|
|
|
|
echo Convert::raw2xml($message) . "<br />";
|
2012-07-19 17:16:50 +02:00
|
|
|
}
|
2015-06-25 10:52:56 +12:00
|
|
|
flush();
|
2011-05-02 16:33:05 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-29 10:26:55 +12:00
|
|
|
|
2011-05-02 16:33:05 +12:00
|
|
|
class Solr_Reindex extends BuildTask {
|
2015-05-06 17:01:41 +12:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Number of records to load and index per request
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
* @config
|
|
|
|
*/
|
|
|
|
private static $recordsPerRequest = 200;
|
2011-05-02 16:33:05 +12:00
|
|
|
|
|
|
|
public function run($request) {
|
|
|
|
increase_time_limit_to();
|
|
|
|
$self = get_class($this);
|
2012-07-19 17:16:50 +02:00
|
|
|
$verbose = isset($_GET['verbose']);
|
2011-05-02 16:33:05 +12:00
|
|
|
|
|
|
|
$originalState = SearchVariant::current_state();
|
|
|
|
|
|
|
|
if (isset($_GET['start'])) {
|
|
|
|
$this->runFrom(singleton($_GET['index']), $_GET['class'], $_GET['start'], json_decode($_GET['variantstate'], true));
|
|
|
|
}
|
|
|
|
else {
|
2012-05-14 12:29:41 +12:00
|
|
|
foreach(array('framework','sapphire') as $dirname) {
|
|
|
|
$script = sprintf("%s%s$dirname%scli-script.php", BASE_PATH, DIRECTORY_SEPARATOR, DIRECTORY_SEPARATOR);
|
|
|
|
if(file_exists($script)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-05-02 16:33:05 +12:00
|
|
|
$class = get_class($this);
|
|
|
|
|
|
|
|
foreach (Solr::get_indexes() as $index => $instance) {
|
|
|
|
echo "Rebuilding {$instance->getIndexName()}\n\n";
|
2015-05-06 17:01:41 +12:00
|
|
|
|
2012-08-22 19:36:31 +02:00
|
|
|
$classes = $instance->getClasses();
|
|
|
|
if($request->getVar('class')) {
|
|
|
|
$limitClasses = explode(',', $request->getVar('class'));
|
|
|
|
$classes = array_intersect_key($classes, array_combine($limitClasses, $limitClasses));
|
|
|
|
}
|
|
|
|
|
2015-05-07 20:44:21 +12:00
|
|
|
if($classes) {
|
|
|
|
Solr::service($index)->deleteByQuery('ClassHierarchy:(' . implode(' OR ', array_keys($classes)) . ')');
|
|
|
|
}
|
2011-05-02 16:33:05 +12:00
|
|
|
|
2012-08-22 19:36:31 +02:00
|
|
|
foreach ($classes as $class => $options) {
|
2011-05-02 16:33:05 +12:00
|
|
|
$includeSubclasses = $options['include_children'];
|
2015-05-06 17:01:41 +12:00
|
|
|
|
2011-05-02 16:33:05 +12:00
|
|
|
foreach (SearchVariant::reindex_states($class, $includeSubclasses) as $state) {
|
2012-08-28 23:21:51 +02:00
|
|
|
if ($instance->variantStateExcluded($state)) continue;
|
2015-05-06 17:01:41 +12:00
|
|
|
|
2011-05-02 16:33:05 +12:00
|
|
|
SearchVariant::activate_state($state);
|
|
|
|
|
2012-07-19 13:13:52 +12:00
|
|
|
$filter = $includeSubclasses ? "" : '"ClassName" = \''.$class."'";
|
2011-05-02 16:33:05 +12:00
|
|
|
$singleton = singleton($class);
|
2012-07-19 13:13:52 +12:00
|
|
|
$query = $singleton->get($class,$filter,null);
|
|
|
|
$dtaQuery = $query->dataQuery();
|
|
|
|
$sqlQuery = $dtaQuery->getFinalisedQuery();
|
|
|
|
$singleton->extend('augmentSQL',$sqlQuery,$dtaQuery);
|
|
|
|
$total = $query->count();
|
2011-05-02 16:33:05 +12:00
|
|
|
|
|
|
|
$statevar = json_encode($state);
|
2012-07-19 17:16:50 +02:00
|
|
|
echo "Class: $class, total: $total";
|
|
|
|
echo ($statevar) ? " in state $statevar\n" : "\n";
|
2011-05-02 16:33:05 +12:00
|
|
|
|
|
|
|
if (strpos(PHP_OS, "WIN") !== false) $statevar = '"'.str_replace('"', '\\"', $statevar).'"';
|
|
|
|
else $statevar = "'".$statevar."'";
|
|
|
|
|
|
|
|
for ($offset = 0; $offset < $total; $offset += $this->stat('recordsPerRequest')) {
|
|
|
|
echo "$offset..";
|
2013-08-29 10:26:55 +12:00
|
|
|
|
2012-07-19 16:35:26 +02:00
|
|
|
$cmd = "php $script dev/tasks/$self index=$index class=$class start=$offset variantstate=$statevar";
|
2015-05-06 17:01:41 +12:00
|
|
|
|
2014-03-19 12:14:56 +13:00
|
|
|
if($verbose) {
|
|
|
|
echo "\n Running '$cmd'\n";
|
2015-05-06 17:01:41 +12:00
|
|
|
$cmd .= " verbose=1 2>&1";
|
2014-03-19 12:14:56 +13:00
|
|
|
}
|
2015-05-06 17:01:41 +12:00
|
|
|
|
2012-07-19 17:16:50 +02:00
|
|
|
$res = $verbose ? passthru($cmd) : `$cmd`;
|
|
|
|
if($verbose) echo " ".preg_replace('/\r\n|\n/', '$0 ', $res)."\n";
|
2011-05-02 16:33:05 +12:00
|
|
|
|
|
|
|
// If we're in dev mode, commit more often for fun and profit
|
|
|
|
if (Director::isDev()) Solr::service($index)->commit();
|
2014-10-22 19:01:34 +13:00
|
|
|
|
|
|
|
// This will slow down things a tiny bit, but it is done so that we don't timeout to the database during a reindex
|
|
|
|
DB::query('SELECT 1');
|
2011-05-02 16:33:05 +12:00
|
|
|
}
|
2015-05-05 23:41:59 +12:00
|
|
|
|
|
|
|
echo "\n";
|
2011-05-02 16:33:05 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Solr::service($index)->commit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$originalState = SearchVariant::current_state();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function runFrom($index, $class, $start, $variantstate) {
|
|
|
|
$classes = $index->getClasses();
|
|
|
|
$options = $classes[$class];
|
2014-01-15 12:45:14 +13:00
|
|
|
$verbose = isset($_GET['verbose']);
|
2011-05-02 16:33:05 +12:00
|
|
|
|
|
|
|
SearchVariant::activate_state($variantstate);
|
|
|
|
|
|
|
|
$includeSubclasses = $options['include_children'];
|
|
|
|
$filter = $includeSubclasses ? "" : '"ClassName" = \''.$class."'";
|
|
|
|
|
2014-01-15 12:45:14 +13:00
|
|
|
$items = DataList::create($class)
|
|
|
|
->where($filter)
|
|
|
|
->limit($this->stat('recordsPerRequest'), $start);
|
|
|
|
|
2014-03-19 12:14:56 +13:00
|
|
|
if($verbose) echo "Adding $class";
|
2014-01-15 12:45:14 +13:00
|
|
|
foreach ($items as $item) {
|
2014-03-19 12:14:56 +13:00
|
|
|
if($verbose) echo $item->ID . ' ';
|
2014-01-15 12:45:14 +13:00
|
|
|
|
2015-05-06 17:01:41 +12:00
|
|
|
// See SearchUpdater_ObjectHandler::triggerReindex
|
|
|
|
$item->triggerReindex();
|
2014-01-15 12:45:14 +13:00
|
|
|
|
2015-05-06 17:01:41 +12:00
|
|
|
$item->destroy();
|
2014-01-15 12:45:14 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
if($verbose) echo "Done ";
|
2011-05-02 16:33:05 +12:00
|
|
|
}
|
|
|
|
|
2013-02-13 09:56:07 +13:00
|
|
|
}
|