Spade
Mini Shell
| Directory:~$ /home/lmsyaran/public_html/joomla4/ |
| [Home] [System Details] [Kill Me] |
PK"�[|I�33
access.xmlnu�[���<?xml version="1.0"
encoding="utf-8" ?>
<access component="com_finder">
<section name="component">
<action name="core.admin" title="JACTION_ADMIN"
description="JACTION_ADMIN_COMPONENT_DESC" />
<action name="core.options"
title="JACTION_OPTIONS"
description="JACTION_OPTIONS_COMPONENT_DESC" />
<action name="core.manage" title="JACTION_MANAGE"
description="JACTION_MANAGE_COMPONENT_DESC" />
<action name="core.create" title="JACTION_CREATE"
description="JACTION_CREATE_COMPONENT_DESC" />
<action name="core.delete" title="JACTION_DELETE"
description="JACTION_DELETE_COMPONENT_DESC" />
<action name="core.edit" title="JACTION_EDIT"
description="JACTION_EDIT_COMPONENT_DESC" />
<action name="core.edit.state"
title="JACTION_EDITSTATE"
description="JACTION_EDITSTATE_COMPONENT_DESC" />
</section>
</access>
PK"�[T�:�d"d"
config.xmlnu�[���<?xml version="1.0"
encoding="utf-8"?>
<config>
<fieldset
name="search"
label="COM_FINDER_FIELDSET_SEARCH_OPTIONS_LABEL"
description="COM_FINDER_FIELDSET_SEARCH_OPTIONS_DESCRIPTION"
>
<field
name="enabled"
type="radio"
label="COM_FINDER_CONFIG_GATHER_SEARCH_STATISTICS_LABEL"
description="COM_FINDER_CONFIG_GATHER_SEARCH_STATISTICS_DESCRIPTION"
class="btn-group btn-group-yesno"
default="0"
>
<option value="1">JYES</option>
<option value="0">JNO</option>
</field>
<field
name="show_description"
type="radio"
label="COM_FINDER_CONFIG_SHOW_DESCRIPTION_LABEL"
description="COM_FINDER_CONFIG_SHOW_DESCRIPTION_DESCRIPTION"
class="btn-group btn-group-yesno"
default="1"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="description_length"
type="number"
label="COM_FINDER_CONFIG_DESCRIPTION_LENGTH_LABEL"
description="COM_FINDER_CONFIG_DESCRIPTION_LENGTH_DESCRIPTION"
size="5"
default="255"
filter="integer"
showon="show_description:1"
/>
<field
name="allow_empty_query"
type="radio"
label="COM_FINDER_CONFIG_ALLOW_EMPTY_QUERY_LABEL"
description="COM_FINDER_CONFIG_ALLOW_EMPTY_QUERY_DESCRIPTION"
class="btn-group btn-group-yesno"
default="0"
>
<option value="1">JYES</option>
<option value="0">JNO</option>
</field>
<field
name="show_url"
type="radio"
label="COM_FINDER_CONFIG_SHOW_URL_LABEL"
description="COM_FINDER_CONFIG_SHOW_URL_DESCRIPTION"
class="btn-group btn-group-yesno"
default="1"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="show_autosuggest"
type="radio"
label="COM_FINDER_CONFIG_SHOW_AUTOSUGGEST_LABEL"
description="COM_FINDER_CONFIG_SHOW_AUTOSUGGEST_DESCRIPTION"
class="btn-group btn-group-yesno"
default="1"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="show_suggested_query"
type="radio"
label="COM_FINDER_CONFIG_SHOW_SUGGESTED_QUERY_LABEL"
description="COM_FINDER_CONFIG_SHOW_SUGGESTED_QUERY_DESC"
class="btn-group btn-group-yesno"
default="1"
validate="options"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="show_explained_query"
type="radio"
label="COM_FINDER_CONFIG_SHOW_EXPLAINED_QUERY_LABEL"
description="COM_FINDER_CONFIG_SHOW_EXPLAINED_QUERY_DESC"
class="btn-group btn-group-yesno"
default="1"
validate="options"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="show_advanced"
type="radio"
label="COM_FINDER_CONFIG_SHOW_ADVANCED_LABEL"
description="COM_FINDER_CONFIG_SHOW_ADVANCED_DESCRIPTION"
class="btn-group btn-group-yesno"
default="1"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="show_advanced_tips"
type="radio"
label="COM_FINDER_CONFIG_SHOW_ADVANCED_TIPS_LABEL"
description="COM_FINDER_CONFIG_SHOW_ADVANCED_TIPS_DESCRIPTION"
class="btn-group btn-group-yesno"
default="1"
showon="show_advanced:1"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="expand_advanced"
type="radio"
label="COM_FINDER_CONFIG_EXPAND_ADVANCED_LABEL"
description="COM_FINDER_CONFIG_EXPAND_ADVANCED_DESCRIPTION"
class="btn-group btn-group-yesno"
default="0"
showon="show_advanced:1"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="show_date_filters"
type="radio"
label="COM_FINDER_CONFIG_SHOW_DATE_FILTERS_LABEL"
description="COM_FINDER_CONFIG_SHOW_DATE_FILTERS_DESCRIPTION"
class="btn-group btn-group-yesno"
default="0"
showon="show_advanced:1"
>
<option value="1">JSHOW</option>
<option value="0">JHIDE</option>
</field>
<field
name="sort_order"
type="list"
label="COM_FINDER_CONFIG_SORT_ORDER_LABEL"
description="COM_FINDER_CONFIG_SORT_ORDER_DESC"
default="relevance"
validate="options"
>
<option
value="relevance">COM_FINDER_CONFIG_SORT_OPTION_RELEVANCE</option>
<option
value="date">COM_FINDER_CONFIG_SORT_OPTION_START_DATE</option>
<option
value="price">COM_FINDER_CONFIG_SORT_OPTION_LIST_PRICE</option>
</field>
<field
name="sort_direction"
type="list"
label="COM_FINDER_CONFIG_SORT_DIRECTION_LABEL"
description="COM_FINDER_CONFIG_SORT_DIRECTION_DESC"
default="desc"
validate="options"
>
<option
value="desc">COM_FINDER_CONFIG_SORT_OPTION_DESCENDING</option>
<option
value="asc">COM_FINDER_CONFIG_SORT_OPTION_ASCENDING</option>
</field>
<field
name="highlight_terms"
type="radio"
label="COM_FINDER_CONFIG_HILIGHT_CONTENT_SEARCH_TERMS_LABEL"
description="COM_FINDER_CONFIG_HILIGHT_CONTENT_SEARCH_TERMS_DESCRIPTION"
class="btn-group btn-group-yesno"
default="1"
>
<option value="1">JYES</option>
<option value="0">JNO</option>
</field>
<field
name="opensearch_name"
type="text"
label="COM_FINDER_CONFIG_FIELD_OPENSEARCH_NAME_LABEL"
description="COM_FINDER_CONFIG_FIELD_OPENSEARCH_NAME_DESCRIPTION"
default=""
/>
<field
name="opensearch_description"
type="textarea"
label="COM_FINDER_CONFIG_FIELD_OPENSEARCH_DESCRIPTON_LABEL"
description="COM_FINDER_CONFIG_FIELD_OPENSEARCH_DESCRIPTON_DESCRIPTION"
default=""
cols="30"
rows="2"
/>
</fieldset>
<fieldset
name="index"
label="COM_FINDER_FIELDSET_INDEX_OPTIONS_LABEL"
description="COM_FINDER_FIELDSET_INDEX_OPTIONS_DESCRIPTION"
>
<field
name="batch_size"
type="list"
label="COM_FINDER_CONFIG_BATCH_SIZE_LABEL"
description="COM_FINDER_CONFIG_BATCH_SIZE_DESCRIPTION"
default="50"
validate="options"
>
<option value="5">J5</option>
<option value="10">J10</option>
<option value="25">J25</option>
<option value="50">J50</option>
<option value="75">J75</option>
<option value="100">J100</option>
<option value="150">J150</option>
<option value="200">J200</option>
<option value="250">J250</option>
<option value="300">J300</option>
</field>
<field
name="memory_table_limit"
type="number"
label="COM_FINDER_CONFIG_MEMORY_TABLE_LIMIT_LABEL"
description="COM_FINDER_CONFIG_MEMORY_TABLE_LIMIT_DESCRIPTION"
size="10"
default="30000"
filter="integer"
/>
<field
name="title_multiplier"
type="number"
label="COM_FINDER_CONFIG_TITLE_MULTIPLIER_LABEL"
description="COM_FINDER_CONFIG_TITLE_MULTIPLIER_DESCRIPTION"
size="5"
default="1.7"
/>
<field
name="text_multiplier"
type="number"
label="COM_FINDER_CONFIG_TEXT_MULTIPLIER_LABEL"
description="COM_FINDER_CONFIG_TEXT_MULTIPLIER_DESCRIPTION"
size="5"
default="0.7"
/>
<field
name="meta_multiplier"
type="number"
label="COM_FINDER_CONFIG_META_MULTIPLIER_LABEL"
description="COM_FINDER_CONFIG_META_MULTIPLIER_DESCRIPTION"
size="5"
default="1.2"
/>
<field
name="path_multiplier"
type="number"
label="COM_FINDER_CONFIG_PATH_MULTIPLIER_LABEL"
description="COM_FINDER_CONFIG_PATH_MULTIPLIER_DESCRIPTION"
size="5"
default="2.0"
/>
<field
name="misc_multiplier"
type="number"
label="COM_FINDER_CONFIG_MISC_MULTIPLIER_LABEL"
description="COM_FINDER_CONFIG_MISC_MULTIPLIER_DESCRIPTION"
size="5"
default="0.3"
/>
<field
name="stem"
type="radio"
label="COM_FINDER_CONFIG_STEMMER_ENABLE_LABEL"
description="COM_FINDER_CONFIG_STEMMER_ENABLE_DESCRIPTION"
class="btn-group btn-group-yesno"
default="1"
>
<option value="1">JYES</option>
<option value="0">JNO</option>
</field>
<field
name="stemmer"
type="list"
label="COM_FINDER_CONFIG_STEMMER_LABEL"
description="COM_FINDER_CONFIG_STEMMER_DESCRIPTION"
default="snowball"
showon="stem:1"
>
<option
value="porter_en">COM_FINDER_CONFIG_STEMMER_PORTER_EN</option>
<option
value="fr">COM_FINDER_CONFIG_STEMMER_FR</option>
<option
value="snowball">COM_FINDER_CONFIG_STEMMER_SNOWBALL</option>
</field>
<field
name="enable_logging"
type="radio"
label="COM_FINDER_CONFIG_ENABLE_LOGGING_LABEL"
description="COM_FINDER_CONFIG_ENABLE_LOGGING_DESCRIPTION"
class="btn-group btn-group-yesno"
default="0"
>
<option value="1">JYES</option>
<option value="0">JNO</option>
</field>
</fieldset>
<fieldset
name="permissions"
label="JCONFIG_PERMISSIONS_LABEL"
description="JCONFIG_PERMISSIONS_DESC"
>
<field
name="rules"
type="rules"
label="JCONFIG_PERMISSIONS_LABEL"
filter="rules"
validate="rules"
component="com_finder"
section="component"
/>
</fieldset>
</config>
PK"�[�.����controller.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Base controller class for Finder.
*
* @since 2.5
*/
class FinderController extends JControllerLegacy
{
/**
* The default view.
*
* @var string
* @since 2.5
*/
protected $default_view = 'index';
/**
* Method to display a view.
*
* @param boolean $cachable If true, the view output will be cached
* @param array $urlparams An array of safe URL parameters and their
variable types, for valid values see {@link JFilterInput::clean()}.
*
* @return FinderController A JControllerLegacy object to support
chaining.
*
* @since 2.5
*/
public function display($cachable = false, $urlparams = array())
{
JLoader::register('FinderHelper', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/finder.php');
$view = $this->input->get('view', 'index',
'word');
$layout = $this->input->get('layout', 'index',
'word');
$filterId = $this->input->get('filter_id', null,
'int');
// Check for edit form.
if ($view === 'filter' && $layout === 'edit'
&& !$this->checkEditId('com_finder.edit.filter',
$filterId))
{
// Somehow the person just went to the form - we don't allow that.
$this->setError(JText::sprintf('JLIB_APPLICATION_ERROR_UNHELD_ID',
$filterId));
$this->setMessage($this->getError(), 'error');
$this->setRedirect(JRoute::_('index.php?option=com_finder&view=filters',
false));
return false;
}
return parent::display();
}
}
PK"�[Eӓ�controllers/filter.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\Utilities\ArrayHelper;
/**
* Indexer controller class for Finder.
*
* @since 2.5
*/
class FinderControllerFilter extends JControllerForm
{
/**
* Method to save a record.
*
* @param string $key The name of the primary key of the URL
variable.
* @param string $urlVar The name of the URL variable if different
from the primary key (sometimes required to avoid router collisions).
*
* @return boolean True if successful, false otherwise.
*
* @since 2.5
*/
public function save($key = null, $urlVar = null)
{
// Check for request forgeries.
$this->checkToken();
$app = JFactory::getApplication();
$input = $app->input;
$model = $this->getModel();
$table = $model->getTable();
$data = $input->post->get('jform', array(),
'array');
$checkin = property_exists($table, 'checked_out');
$context = "$this->option.edit.$this->context";
$task = $this->getTask();
// Determine the name of the primary key for the data.
if (empty($key))
{
$key = $table->getKeyName();
}
// To avoid data collisions the urlVar may be different from the primary
key.
if (empty($urlVar))
{
$urlVar = $key;
}
$recordId = $input->get($urlVar, '', 'int');
if (!$this->checkEditId($context, $recordId))
{
// Somehow the person just went to the form and tried to save it. We
don't allow that.
$this->setError(JText::sprintf('JLIB_APPLICATION_ERROR_UNHELD_ID',
$recordId));
$this->setMessage($this->getError(), 'error');
$this->setRedirect(JRoute::_('index.php?option=' .
$this->option . '&view=' . $this->view_list .
$this->getRedirectToListAppend(), false));
return false;
}
// Populate the row id from the session.
$data[$key] = $recordId;
// The save2copy task needs to be handled slightly differently.
if ($task === 'save2copy')
{
// Check-in the original row.
if ($checkin && $model->checkin($data[$key]) === false)
{
// Check-in failed. Go back to the item and display a notice.
$this->setError(JText::sprintf('JLIB_APPLICATION_ERROR_CHECKIN_FAILED',
$model->getError()));
$this->setMessage($this->getError(), 'error');
$this->setRedirect('index.php?option=' . $this->option
. '&view=' . $this->view_item .
$this->getRedirectToItemAppend($recordId, $urlVar));
return false;
}
// Reset the ID and then treat the request as for Apply.
$data[$key] = 0;
$task = 'apply';
}
// Access check.
if (!$this->allowSave($data, $key))
{
$this->setError(JText::_('JLIB_APPLICATION_ERROR_SAVE_NOT_PERMITTED'));
$this->setMessage($this->getError(), 'error');
$this->setRedirect(JRoute::_('index.php?option=' .
$this->option . '&view=' . $this->view_list .
$this->getRedirectToListAppend(), false));
return false;
}
// Validate the posted data.
// Sometimes the form needs some posted data, such as for plugins and
modules.
$form = $model->getForm($data, false);
if (!$form)
{
$app->enqueueMessage($model->getError(), 'error');
return false;
}
// Test whether the data is valid.
$validData = $model->validate($form, $data);
// Check for validation errors.
if ($validData === false)
{
// Get the validation messages.
$errors = $model->getErrors();
// Push up to three validation messages out to the user.
for ($i = 0, $n = count($errors); $i < $n && $i < 3; $i++)
{
if ($errors[$i] instanceof Exception)
{
$app->enqueueMessage($errors[$i]->getMessage(),
'warning');
}
else
{
$app->enqueueMessage($errors[$i], 'warning');
}
}
// Save the data in the session.
$app->setUserState($context . '.data', $data);
// Redirect back to the edit screen.
$this->setRedirect(
JRoute::_('index.php?option=' . $this->option .
'&view=' . $this->view_item .
$this->getRedirectToItemAppend($recordId, $key), false)
);
return false;
}
// Get and sanitize the filter data.
$validData['data'] = $input->post->get('t',
array(), 'array');
$validData['data'] =
array_unique($validData['data']);
$validData['data'] =
ArrayHelper::toInteger($validData['data']);
// Remove any values of zero.
if (array_search(0, $validData['data'], true))
{
unset($validData['data'][array_search(0,
$validData['data'], true)]);
}
// Attempt to save the data.
if (!$model->save($validData))
{
// Save the data in the session.
$app->setUserState($context . '.data', $validData);
// Redirect back to the edit screen.
$this->setError(JText::sprintf('JLIB_APPLICATION_ERROR_SAVE_FAILED',
$model->getError()));
$this->setMessage($this->getError(), 'error');
$this->setRedirect(
JRoute::_('index.php?option=' . $this->option .
'&view=' . $this->view_item .
$this->getRedirectToItemAppend($recordId, $key), false)
);
return false;
}
// Save succeeded, so check-in the record.
if ($checkin && $model->checkin($validData[$key]) === false)
{
// Save the data in the session.
$app->setUserState($context . '.data', $validData);
// Check-in failed, so go back to the record and display a notice.
$this->setError(JText::sprintf('JLIB_APPLICATION_ERROR_CHECKIN_FAILED',
$model->getError()));
$this->setMessage($this->getError(), 'error');
$this->setRedirect('index.php?option=' . $this->option .
'&view=' . $this->view_item .
$this->getRedirectToItemAppend($recordId, $key));
return false;
}
$this->setMessage(
JText::_(
(JFactory::getLanguage()->hasKey($this->text_prefix . ($recordId
=== 0 && $app->isClient('site') ? '_SUBMIT'
: '') . '_SAVE_SUCCESS')
? $this->text_prefix : 'JLIB_APPLICATION') . ($recordId
=== 0 && $app->isClient('site') ? '_SUBMIT'
: '') . '_SAVE_SUCCESS'
)
);
// Redirect the user and adjust session state based on the chosen task.
switch ($task)
{
case 'apply':
// Set the record data in the session.
$recordId = $model->getState($this->context . '.id');
$this->holdEditId($context, $recordId);
$app->setUserState($context . '.data', null);
$model->checkout($recordId);
// Redirect back to the edit screen.
$this->setRedirect(
JRoute::_('index.php?option=' . $this->option .
'&view=' . $this->view_item .
$this->getRedirectToItemAppend($recordId, $key), false)
);
break;
case 'save2new':
// Clear the record id and data from the session.
$this->releaseEditId($context, $recordId);
$app->setUserState($context . '.data', null);
// Redirect back to the edit screen.
$this->setRedirect(
JRoute::_('index.php?option=' . $this->option .
'&view=' . $this->view_item .
$this->getRedirectToItemAppend(null, $key), false)
);
break;
default:
// Clear the record id and data from the session.
$this->releaseEditId($context, $recordId);
$app->setUserState($context . '.data', null);
// Redirect to the list screen.
$this->setRedirect(
JRoute::_('index.php?option=' . $this->option .
'&view=' . $this->view_list .
$this->getRedirectToListAppend(), false)
);
break;
}
// Invoke the postSave method to allow for the child class to access the
model.
$this->postSaveHook($model, $validData);
return true;
}
}
PK"�[.M||controllers/filters.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Filters controller class for Finder.
*
* @since 2.5
*/
class FinderControllerFilters extends JControllerAdmin
{
/**
* Method to get a model object, loading it if required.
*
* @param string $name The model name. Optional.
* @param string $prefix The class prefix. Optional.
* @param array $config Configuration array for model. Optional.
*
* @return JModelLegacy The model.
*
* @since 2.5
*/
public function getModel($name = 'Filter', $prefix =
'FinderModel', $config = array('ignore_request' =>
true))
{
return parent::getModel($name, $prefix, $config);
}
}
PK"�[c�G'GGcontrollers/index.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Index controller class for Finder.
*
* @since 2.5
*/
class FinderControllerIndex extends JControllerAdmin
{
/**
* Method to get a model object, loading it if required.
*
* @param string $name The model name. Optional.
* @param string $prefix The class prefix. Optional.
* @param array $config Configuration array for model. Optional.
*
* @return JModelLegacy The model.
*
* @since 2.5
*/
public function getModel($name = 'Index', $prefix =
'FinderModel', $config = array('ignore_request' =>
true))
{
return parent::getModel($name, $prefix, $config);
}
/**
* Method to purge all indexed links from the database.
*
* @return boolean True on success.
*
* @since 2.5
*/
public function purge()
{
$this->checkToken();
// Remove the script time limit.
@set_time_limit(0);
$model = $this->getModel('Index', 'FinderModel');
// Attempt to purge the index.
$return = $model->purge();
if (!$return)
{
$message = JText::_('COM_FINDER_INDEX_PURGE_FAILED',
$model->getError());
$this->setRedirect('index.php?option=com_finder&view=index',
$message);
return false;
}
else
{
$message = JText::_('COM_FINDER_INDEX_PURGE_SUCCESS');
$this->setRedirect('index.php?option=com_finder&view=index',
$message);
return true;
}
}
}
PK"�[��$��)�)controllers/indexer.json.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
// Register dependent classes.
JLoader::register('FinderIndexer', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/indexer/indexer.php');
/**
* Indexer controller class for Finder.
*
* @since 2.5
*/
class FinderControllerIndexer extends JControllerLegacy
{
/**
* Method to start the indexer.
*
* @return void
*
* @since 2.5
*/
public function start()
{
$params = JComponentHelper::getParams('com_finder');
if ($params->get('enable_logging', '0'))
{
$options['format'] =
'{DATE}\t{TIME}\t{LEVEL}\t{CODE}\t{MESSAGE}';
$options['text_file'] = 'indexer.php';
JLog::addLogger($options);
}
// Log the start
try
{
JLog::add('Starting the indexer', JLog::INFO);
}
catch (RuntimeException $exception)
{
// Informational log only
}
// We don't want this form to be cached.
$app = JFactory::getApplication();
$app->setHeader('Expires', 'Mon, 1 Jan 2001 00:00:00
GMT', true);
$app->setHeader('Last-Modified', gmdate('D, d M Y
H:i:s') . ' GMT', true);
$app->setHeader('Cache-Control', 'no-store, no-cache,
must-revalidate, post-check=0, pre-check=0', false);
$app->setHeader('Pragma', 'no-cache');
// Check for a valid token. If invalid, send a 403 with the error
message.
JSession::checkToken('request') or static::sendResponse(new
Exception(JText::_('JINVALID_TOKEN_NOTICE'), 403));
// Put in a buffer to silence noise.
ob_start();
// Reset the indexer state.
FinderIndexer::resetState();
// Import the finder plugins.
JPluginHelper::importPlugin('finder');
// Add the indexer language to JS
JText::script('COM_FINDER_AN_ERROR_HAS_OCCURRED');
JText::script('COM_FINDER_NO_ERROR_RETURNED');
// Start the indexer.
try
{
// Trigger the onStartIndex event.
JEventDispatcher::getInstance()->trigger('onStartIndex');
// Get the indexer state.
$state = FinderIndexer::getState();
$state->start = 1;
// Send the response.
static::sendResponse($state);
}
// Catch an exception and return the response.
catch (Exception $e)
{
static::sendResponse($e);
}
}
/**
* Method to run the next batch of content through the indexer.
*
* @return void
*
* @since 2.5
*/
public function batch()
{
$params = JComponentHelper::getParams('com_finder');
if ($params->get('enable_logging', '0'))
{
$options['format'] =
'{DATE}\t{TIME}\t{LEVEL}\t{CODE}\t{MESSAGE}';
$options['text_file'] = 'indexer.php';
JLog::addLogger($options);
}
// Log the start
try
{
JLog::add('Starting the indexer batch process', JLog::INFO);
}
catch (RuntimeException $exception)
{
// Informational log only
}
// We don't want this form to be cached.
$app = JFactory::getApplication();
$app->setHeader('Expires', 'Mon, 1 Jan 2001 00:00:00
GMT', true);
$app->setHeader('Last-Modified', gmdate('D, d M Y
H:i:s') . ' GMT', true);
$app->setHeader('Cache-Control', 'no-store, no-cache,
must-revalidate, post-check=0, pre-check=0', false);
$app->setHeader('Pragma', 'no-cache');
// Check for a valid token. If invalid, send a 403 with the error
message.
JSession::checkToken('request') or static::sendResponse(new
Exception(JText::_('JINVALID_TOKEN_NOTICE'), 403));
// Put in a buffer to silence noise.
ob_start();
// Remove the script time limit.
@set_time_limit(0);
// Get the indexer state.
$state = FinderIndexer::getState();
// Reset the batch offset.
$state->batchOffset = 0;
// Update the indexer state.
FinderIndexer::setState($state);
// Import the finder plugins.
JPluginHelper::importPlugin('finder');
/*
* We are going to swap out the raw document object with an HTML document
* in order to work around some plugins that don't do proper
environment
* checks before trying to use HTML document functions.
*/
$raw = clone JFactory::getDocument();
$lang = JFactory::getLanguage();
// Get the document properties.
$attributes = array (
'charset' => 'utf-8',
'lineend' => 'unix',
'tab' => ' ',
'language' => $lang->getTag(),
'direction' => $lang->isRtl() ? 'rtl' :
'ltr'
);
// Get the HTML document.
$html = JDocument::getInstance('html', $attributes);
// Todo: Why is this document fetched and immediately overwritten?
$doc = JFactory::getDocument();
// Swap the documents.
$doc = $html;
// Get the admin application.
$admin = clone JFactory::getApplication();
// Get the site app.
$site = JApplicationCms::getInstance('site');
// Swap the app.
$app = JFactory::getApplication();
// Todo: Why is the app fetched and immediately overwritten?
$app = $site;
// Start the indexer.
try
{
// Trigger the onBeforeIndex event.
JEventDispatcher::getInstance()->trigger('onBeforeIndex');
// Trigger the onBuildIndex event.
JEventDispatcher::getInstance()->trigger('onBuildIndex');
// Get the indexer state.
$state = FinderIndexer::getState();
$state->start = 0;
$state->complete = 0;
// Swap the documents back.
$doc = $raw;
// Swap the applications back.
$app = $admin;
// Log batch completion and memory high-water mark.
try
{
JLog::add('Batch completed, peak memory usage: ' .
number_format(memory_get_peak_usage(true)) . ' bytes',
JLog::INFO);
}
catch (RuntimeException $exception)
{
// Informational log only
}
// Send the response.
static::sendResponse($state);
}
// Catch an exception and return the response.
catch (Exception $e)
{
// Swap the documents back.
$doc = $raw;
// Send the response.
static::sendResponse($e);
}
}
/**
* Method to optimize the index and perform any necessary cleanup.
*
* @return void
*
* @since 2.5
*/
public function optimize()
{
// We don't want this form to be cached.
$app = JFactory::getApplication();
$app->setHeader('Expires', 'Mon, 1 Jan 2001 00:00:00
GMT', true);
$app->setHeader('Last-Modified', gmdate('D, d M Y
H:i:s') . ' GMT', true);
$app->setHeader('Cache-Control', 'no-store, no-cache,
must-revalidate, post-check=0, pre-check=0', false);
$app->setHeader('Pragma', 'no-cache');
// Check for a valid token. If invalid, send a 403 with the error
message.
JSession::checkToken('request') or static::sendResponse(new
Exception(JText::_('JINVALID_TOKEN_NOTICE'), 403));
// Put in a buffer to silence noise.
ob_start();
// Import the finder plugins.
JPluginHelper::importPlugin('finder');
try
{
// Optimize the index
FinderIndexer::getInstance()->optimize();
// Get the indexer state.
$state = FinderIndexer::getState();
$state->start = 0;
$state->complete = 1;
// Send the response.
static::sendResponse($state);
}
// Catch an exception and return the response.
catch (Exception $e)
{
static::sendResponse($e);
}
}
/**
* Method to handle a send a JSON response. The body parameter
* can be an Exception object for when an error has occurred or
* a JObject for a good response.
*
* @param mixed $data JObject on success, Exception on error.
[optional]
*
* @return void
*
* @since 2.5
*/
public static function sendResponse($data = null)
{
// This method always sends a JSON response
$app = JFactory::getApplication();
$app->mimeType = 'application/json';
$params = JComponentHelper::getParams('com_finder');
if ($params->get('enable_logging', '0'))
{
$options['format'] =
'{DATE}\t{TIME}\t{LEVEL}\t{CODE}\t{MESSAGE}';
$options['text_file'] = 'indexer.php';
JLog::addLogger($options);
}
// Send the assigned error code if we are catching an exception.
if ($data instanceof Exception)
{
try
{
JLog::add($data->getMessage(), JLog::ERROR);
}
catch (RuntimeException $exception)
{
// Informational log only
}
$app->setHeader('status', $data->getCode());
}
// Create the response object.
$response = new FinderIndexerResponse($data);
// Add the buffer.
$response->buffer = JDEBUG ? ob_get_contents() : ob_end_clean();
// Send the JSON response.
$app->setHeader('Content-Type', $app->mimeType . ';
charset=' . $app->charSet);
$app->sendHeaders();
echo json_encode($response);
// Close the application.
$app->close();
}
}
/**
* Finder Indexer JSON Response Class
*
* @since 2.5
*/
class FinderIndexerResponse
{
/**
* Class Constructor
*
* @param mixed $state The processing state for the indexer
*
* @since 2.5
*/
public function __construct($state)
{
$params = JComponentHelper::getParams('com_finder');
if ($params->get('enable_logging', '0'))
{
$options['format'] =
'{DATE}\t{TIME}\t{LEVEL}\t{CODE}\t{MESSAGE}';
$options['text_file'] = 'indexer.php';
JLog::addLogger($options);
}
// The old token is invalid so send a new one.
$this->token = JFactory::getSession()->getFormToken();
// Check if we are dealing with an error.
if ($state instanceof Exception)
{
// Log the error
try
{
JLog::add($state->getMessage(), JLog::ERROR);
}
catch (RuntimeException $exception)
{
// Informational log only
}
// Prepare the error response.
$this->error = true;
$this->header =
JText::_('COM_FINDER_INDEXER_HEADER_ERROR');
$this->message = $state->getMessage();
}
else
{
// Prepare the response data.
$this->batchSize = (int) $state->batchSize;
$this->batchOffset = (int) $state->batchOffset;
$this->totalItems = (int) $state->totalItems;
$this->startTime = $state->startTime;
$this->endTime = JFactory::getDate()->toSql();
$this->start = !empty($state->start) ? (int) $state->start : 0;
$this->complete = !empty($state->complete) ? (int)
$state->complete : 0;
// Set the appropriate messages.
if ($this->totalItems <= 0 && $this->complete)
{
$this->header =
JText::_('COM_FINDER_INDEXER_HEADER_COMPLETE');
$this->message =
JText::_('COM_FINDER_INDEXER_MESSAGE_COMPLETE');
}
elseif ($this->totalItems <= 0)
{
$this->header =
JText::_('COM_FINDER_INDEXER_HEADER_OPTIMIZE');
$this->message =
JText::_('COM_FINDER_INDEXER_MESSAGE_OPTIMIZE');
}
else
{
$this->header =
JText::_('COM_FINDER_INDEXER_HEADER_RUNNING');
$this->message =
JText::_('COM_FINDER_INDEXER_MESSAGE_RUNNING');
}
}
}
}
// Register the error handler.
JError::setErrorHandling(E_ALL, 'callback',
array('FinderControllerIndexer', 'sendResponse'));
PK"�[0ɭ�ttcontrollers/maps.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Maps controller class for Finder.
*
* @since 2.5
*/
class FinderControllerMaps extends JControllerAdmin
{
/**
* Method to get a model object, loading it if required.
*
* @param string $name The model name. Optional.
* @param string $prefix The class prefix. Optional.
* @param array $config Configuration array for model. Optional.
*
* @return JModelLegacy The model.
*
* @since 1.6
*/
public function getModel($name = 'Maps', $prefix =
'FinderModel', $config = array('ignore_request' =>
true))
{
return parent::getModel($name, $prefix, $config);
}
}
PK"�[����AA
finder.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
if (!JFactory::getUser()->authorise('core.manage',
'com_finder'))
{
throw new
JAccessExceptionNotallowed(JText::_('JERROR_ALERTNOAUTHOR'),
403);
}
$controller = JControllerLegacy::getInstance('Finder');
$controller->execute(JFactory::getApplication()->input->get('task'));
$controller->redirect();
PK"�[ŜA
finder.xmlnu�[���<?xml version="1.0"
encoding="utf-8"?>
<extension type="component" version="3.1"
method="upgrade">
<name>com_finder</name>
<author>Joomla! Project</author>
<copyright>(C) 2005 - 2020 Open Source Matters. All rights
reserved.</copyright>
<creationDate>August 2011</creationDate>
<license>GNU General Public License version 2 or later; see
LICENSE.txt</license>
<authorEmail>admin@joomla.org</authorEmail>
<authorUrl>www.joomla.org</authorUrl>
<version>3.0.0</version>
<description>COM_FINDER_XML_DESCRIPTION</description>
<menu link="option=com_finder">COM_FINDER</menu>
<files folder="site">
<filename>controller.php</filename>
<filename>finder.php</filename>
<filename>router.php</filename>
<folder>controllers</folder>
<folder>helpers</folder>
<folder>models</folder>
<folder>views</folder>
</files>
<media destination="com_finder" folder="media">
<folder>js</folder>
<folder>css</folder>
</media>
<install>
<sql>
<file charset="utf8"
driver="mysql">sql/install.mysql.sql</file>
<file charset="utf8"
driver="postgresql">sql/install.postgresql.sql</file>
</sql>
</install>
<uninstall>
<sql>
<file charset="utf8"
driver="mysql">sql/uninstall.mysql.sql</file>
<file charset="utf8"
driver="postgresql">sql/uninstall.postgresql.sql</file>
</sql>
</uninstall>
<languages folder="site">
<language
tag="en-GB">language/en-GB.com_finder.ini</language>
</languages>
<administration>
<files folder="admin">
<filename>access.xml</filename>
<filename>config.xml</filename>
<filename>controller.php</filename>
<filename>finder.php</filename>
<folder>controllers</folder>
<folder>helpers</folder>
<folder>models</folder>
<folder>sql</folder>
<folder>tables</folder>
<folder>views</folder>
</files>
<languages folder="admin">
<language
tag="en-GB">language/en-GB.com_finder.ini</language>
<language
tag="en-GB">language/en-GB.com_finder.sys.ini</language>
</languages>
<menu img="class:finder"
link="option=com_finder">COM_FINDER</menu>
</administration>
</extension>
PK"�[f���P P helpers/finder.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Helper class for Finder.
*
* @since 2.5
*/
class FinderHelper
{
/**
* The extension name.
*
* @var string
* @since 2.5
*/
public static $extension = 'com_finder';
/**
* Configure the Linkbar.
*
* @param string $vName The name of the active view.
*
* @return void
*
* @since 2.5
*/
public static function addSubmenu($vName)
{
JHtmlSidebar::addEntry(
JText::_('COM_FINDER_SUBMENU_INDEX'),
'index.php?option=com_finder&view=index',
$vName === 'index'
);
JHtmlSidebar::addEntry(
JText::_('COM_FINDER_SUBMENU_MAPS'),
'index.php?option=com_finder&view=maps',
$vName === 'maps'
);
JHtmlSidebar::addEntry(
JText::_('COM_FINDER_SUBMENU_FILTERS'),
'index.php?option=com_finder&view=filters',
$vName === 'filters'
);
}
/**
* Gets the finder system plugin extension id.
*
* @return integer The finder system plugin extension id.
*
* @since 3.6.0
*/
public static function getFinderPluginId()
{
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select($db->quoteName('extension_id'))
->from($db->quoteName('#__extensions'))
->where($db->quoteName('folder') . ' = ' .
$db->quote('content'))
->where($db->quoteName('element') . ' = ' .
$db->quote('finder'));
$db->setQuery($query);
try
{
$result = (int) $db->loadResult();
}
catch (RuntimeException $e)
{
JError::raiseWarning(500, $e->getMessage());
}
return $result;
}
/**
* Gets a list of the actions that can be performed.
*
* @return JObject A JObject containing the allowed actions.
*
* @since 2.5
* @deprecated 3.2 Use JHelperContent::getActions() instead
*/
public static function getActions()
{
// Log usage of deprecated function
try
{
JLog::add(
sprintf('%s() is deprecated. Use JHelperContent::getActions() with
new arguments order instead.', __METHOD__),
JLog::WARNING,
'deprecated'
);
}
catch (RuntimeException $exception)
{
// Informational log only
}
// Get list of actions
return JHelperContent::getActions('com_finder');
}
}
PK"�[L&Sߢ�helpers/html/finder.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderHelperLanguage', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/language.php');
use Joomla\Utilities\ArrayHelper;
/**
* HTML behavior class for Finder.
*
* @since 2.5
*/
abstract class JHtmlFinder
{
/**
* Creates a list of types to filter on.
*
* @return array An array containing the types that can be selected.
*
* @since 2.5
*/
public static function typeslist()
{
// Load the finder types.
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select('DISTINCT t.title AS text, t.id AS value')
->from($db->quoteName('#__finder_types') . ' AS
t')
->join('LEFT',
$db->quoteName('#__finder_links') . ' AS l ON l.type_id =
t.id')
->order('t.title ASC');
$db->setQuery($query);
try
{
$rows = $db->loadObjectList();
}
catch (RuntimeException $e)
{
return array();
}
// Compile the options.
$options = array();
$lang = JFactory::getLanguage();
foreach ($rows as $row)
{
$key =
$lang->hasKey(FinderHelperLanguage::branchPlural($row->text)) ?
FinderHelperLanguage::branchPlural($row->text) : $row->text;
$options[] = JHtml::_('select.option', $row->value,
JText::sprintf('COM_FINDER_ITEM_X_ONLY', JText::_($key)));
}
return $options;
}
/**
* Creates a list of maps.
*
* @return array An array containing the maps that can be selected.
*
* @since 2.5
*/
public static function mapslist()
{
// Load the finder types.
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select($db->quoteName('title', 'text'))
->select($db->quoteName('id', 'value'))
->from($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('parent_id') . ' = 1');
$db->setQuery($query);
try
{
$branches = $db->loadObjectList();
}
catch (RuntimeException $e)
{
JError::raiseWarning(500, $db->getMessage());
}
// Translate.
$lang = JFactory::getLanguage();
foreach ($branches as $branch)
{
$key = FinderHelperLanguage::branchPlural($branch->text);
$branch->translatedText = $lang->hasKey($key) ? JText::_($key) :
$branch->text;
}
// Order by title.
$branches = ArrayHelper::sortObjects($branches,
'translatedText', 1, true, true);
// Compile the options.
$options = array();
$options[] = JHtml::_('select.option', '',
JText::_('COM_FINDER_MAPS_SELECT_BRANCH'));
// Convert the values to options.
foreach ($branches as $branch)
{
$options[] = JHtml::_('select.option', $branch->value,
$branch->translatedText);
}
return $options;
}
/**
* Creates a list of published states.
*
* @return array An array containing the states that can be selected.
*
* @since 2.5
*/
public static function statelist()
{
return array(
JHtml::_('select.option', '1',
JText::sprintf('COM_FINDER_ITEM_X_ONLY',
JText::_('JPUBLISHED'))),
JHtml::_('select.option', '0',
JText::sprintf('COM_FINDER_ITEM_X_ONLY',
JText::_('JUNPUBLISHED')))
);
}
}
PK"�[X��!�U�Uhelpers/indexer/adapter.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\Utilities\ArrayHelper;
JLoader::register('FinderIndexer', __DIR__ .
'/indexer.php');
JLoader::register('FinderIndexerHelper', __DIR__ .
'/helper.php');
JLoader::register('FinderIndexerResult', __DIR__ .
'/result.php');
JLoader::register('FinderIndexerTaxonomy', __DIR__ .
'/taxonomy.php');
/**
* Prototype adapter class for the Finder indexer package.
*
* @since 2.5
*/
abstract class FinderIndexerAdapter extends JPlugin
{
/**
* The context is somewhat arbitrary but it must be unique or there will
be
* conflicts when managing plugin/indexer state. A good best practice is
to
* use the plugin name suffix as the context. For example, if the plugin
is
* named 'plgFinderContent', the context could be
'Content'.
*
* @var string
* @since 2.5
*/
protected $context;
/**
* The extension name.
*
* @var string
* @since 2.5
*/
protected $extension;
/**
* The sublayout to use when rendering the results.
*
* @var string
* @since 2.5
*/
protected $layout;
/**
* The mime type of the content the adapter indexes.
*
* @var string
* @since 2.5
*/
protected $mime;
/**
* The access level of an item before save.
*
* @var integer
* @since 2.5
*/
protected $old_access;
/**
* The access level of a category before save.
*
* @var integer
* @since 2.5
*/
protected $old_cataccess;
/**
* The type of content the adapter indexes.
*
* @var string
* @since 2.5
*/
protected $type_title;
/**
* The type id of the content.
*
* @var integer
* @since 2.5
*/
protected $type_id;
/**
* The database object.
*
* @var object
* @since 2.5
*/
protected $db;
/**
* The table name.
*
* @var string
* @since 2.5
*/
protected $table;
/**
* The indexer object.
*
* @var FinderIndexer
* @since 3.0
*/
protected $indexer;
/**
* The field the published state is stored in.
*
* @var string
* @since 2.5
*/
protected $state_field = 'state';
/**
* Method to instantiate the indexer adapter.
*
* @param object $subject The object to observe.
* @param array $config An array that holds the plugin
configuration.
*
* @since 2.5
*/
public function __construct(&$subject, $config)
{
// Get the database object.
$this->db = JFactory::getDbo();
// Call the parent constructor.
parent::__construct($subject, $config);
// Get the type id.
$this->type_id = $this->getTypeId();
// Add the content type if it doesn't exist and is set.
if (empty($this->type_id) && !empty($this->type_title))
{
$this->type_id =
FinderIndexerHelper::addContentType($this->type_title, $this->mime);
}
// Check for a layout override.
if ($this->params->get('layout'))
{
$this->layout = $this->params->get('layout');
}
// Get the indexer object
$this->indexer = FinderIndexer::getInstance();
}
/**
* Method to get the adapter state and push it into the indexer.
*
* @return void
*
* @since 2.5
* @throws Exception on error.
*/
public function onStartIndex()
{
// Get the indexer state.
$iState = FinderIndexer::getState();
// Get the number of content items.
$total = (int) $this->getContentCount();
// Add the content count to the total number of items.
$iState->totalItems += $total;
// Populate the indexer state information for the adapter.
$iState->pluginState[$this->context]['total'] = $total;
$iState->pluginState[$this->context]['offset'] = 0;
// Set the indexer state.
FinderIndexer::setState($iState);
}
/**
* Method to prepare for the indexer to be run. This method will often
* be used to include dependencies and things of that nature.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on error.
*/
public function onBeforeIndex()
{
// Get the indexer and adapter state.
$iState = FinderIndexer::getState();
$aState = $iState->pluginState[$this->context];
// Check the progress of the indexer and the adapter.
if ($iState->batchOffset == $iState->batchSize ||
$aState['offset'] == $aState['total'])
{
return true;
}
// Run the setup method.
return $this->setup();
}
/**
* Method to index a batch of content items. This method can be called by
* the indexer many times throughout the indexing process depending on how
* much content is available for indexing. It is important to track the
* progress correctly so we can display it to the user.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on error.
*/
public function onBuildIndex()
{
// Get the indexer and adapter state.
$iState = FinderIndexer::getState();
$aState = $iState->pluginState[$this->context];
// Check the progress of the indexer and the adapter.
if ($iState->batchOffset == $iState->batchSize ||
$aState['offset'] == $aState['total'])
{
return true;
}
// Get the batch offset and size.
$offset = (int) $aState['offset'];
$limit = (int) ($iState->batchSize - $iState->batchOffset);
// Get the content items to index.
$items = $this->getItems($offset, $limit);
// Iterate through the items and index them.
for ($i = 0, $n = count($items); $i < $n; $i++)
{
// Index the item.
$this->index($items[$i]);
// Adjust the offsets.
$offset++;
$iState->batchOffset++;
$iState->totalItems--;
}
// Update the indexer state.
$aState['offset'] = $offset;
$iState->pluginState[$this->context] = $aState;
FinderIndexer::setState($iState);
return true;
}
/**
* Method to change the value of a content item's property in the
links
* table. This is used to synchronize published and access states that
* are changed when not editing an item directly.
*
* @param string $id The ID of the item to change.
* @param string $property The property that is being changed.
* @param integer $value The new value of that property.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function change($id, $property, $value)
{
// Check for a property we know how to handle.
if ($property !== 'state' && $property !==
'access')
{
return true;
}
// Get the URL for the content id.
$item = $this->db->quote($this->getUrl($id, $this->extension,
$this->layout));
// Update the content items.
$query = $this->db->getQuery(true)
->update($this->db->quoteName('#__finder_links'))
->set($this->db->quoteName($property) . ' = ' . (int)
$value)
->where($this->db->quoteName('url') . ' = '
. $item);
$this->db->setQuery($query);
$this->db->execute();
return true;
}
/**
* Method to index an item.
*
* @param FinderIndexerResult $item The item to index as a
FinderIndexerResult object.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
abstract protected function index(FinderIndexerResult $item);
/**
* Method to reindex an item.
*
* @param integer $id The ID of the item to reindex.
*
* @return void
*
* @since 2.5
* @throws Exception on database error.
*/
protected function reindex($id)
{
// Run the setup method.
$this->setup();
// Remove the old item.
$this->remove($id);
// Get the item.
$item = $this->getItem($id);
// Index the item.
$this->index($item);
}
/**
* Method to remove an item from the index.
*
* @param string $id The ID of the item to remove.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function remove($id)
{
// Get the item's URL
$url = $this->db->quote($this->getUrl($id, $this->extension,
$this->layout));
// Get the link ids for the content items.
$query = $this->db->getQuery(true)
->select($this->db->quoteName('link_id'))
->from($this->db->quoteName('#__finder_links'))
->where($this->db->quoteName('url') . ' = '
. $url);
$this->db->setQuery($query);
$items = $this->db->loadColumn();
// Check the items.
if (empty($items))
{
return true;
}
// Remove the items.
foreach ($items as $item)
{
$this->indexer->remove($item);
}
return true;
}
/**
* Method to setup the adapter before indexing.
*
* @return boolean True on success, false on failure.
*
* @since 2.5
* @throws Exception on database error.
*/
abstract protected function setup();
/**
* Method to update index data on category access level changes
*
* @param JTable $row A JTable object
*
* @return void
*
* @since 2.5
*/
protected function categoryAccessChange($row)
{
$query = clone $this->getStateQuery();
$query->where('c.id = ' . (int) $row->id);
// Get the access level.
$this->db->setQuery($query);
$items = $this->db->loadObjectList();
// Adjust the access level for each item within the category.
foreach ($items as $item)
{
// Set the access level.
$temp = max($item->access, $row->access);
// Update the item.
$this->change((int) $item->id, 'access', $temp);
// Reindex the item
$this->reindex($row->id);
}
}
/**
* Method to update index data on category access level changes
*
* @param array $pks A list of primary key ids of the content that
has changed state.
* @param integer $value The value of the state that the content has
been changed to.
*
* @return void
*
* @since 2.5
*/
protected function categoryStateChange($pks, $value)
{
/*
* The item's published state is tied to the category
* published state so we need to look up all published states
* before we change anything.
*/
foreach ($pks as $pk)
{
$query = clone $this->getStateQuery();
$query->where('c.id = ' . (int) $pk);
// Get the published states.
$this->db->setQuery($query);
$items = $this->db->loadObjectList();
// Adjust the state for each item within the category.
foreach ($items as $item)
{
// Translate the state.
$temp = $this->translateState($item->state, $value);
// Update the item.
$this->change($item->id, 'state', $temp);
// Reindex the item
$this->reindex($item->id);
}
}
}
/**
* Method to check the existing access level for categories
*
* @param JTable $row A JTable object
*
* @return void
*
* @since 2.5
*/
protected function checkCategoryAccess($row)
{
$query = $this->db->getQuery(true)
->select($this->db->quoteName('access'))
->from($this->db->quoteName('#__categories'))
->where($this->db->quoteName('id') . ' = '
. (int) $row->id);
$this->db->setQuery($query);
// Store the access level to determine if it changes
$this->old_cataccess = $this->db->loadResult();
}
/**
* Method to check the existing access level for items
*
* @param JTable $row A JTable object
*
* @return void
*
* @since 2.5
*/
protected function checkItemAccess($row)
{
$query = $this->db->getQuery(true)
->select($this->db->quoteName('access'))
->from($this->db->quoteName($this->table))
->where($this->db->quoteName('id') . ' = '
. (int) $row->id);
$this->db->setQuery($query);
// Store the access level to determine if it changes
$this->old_access = $this->db->loadResult();
}
/**
* Method to get the number of content items available to index.
*
* @return integer The number of content items available to index.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function getContentCount()
{
$return = 0;
// Get the list query.
$query = $this->getListQuery();
// Check if the query is valid.
if (empty($query))
{
return $return;
}
// Tweak the SQL query to make the total lookup faster.
if ($query instanceof JDatabaseQuery)
{
$query = clone $query;
$query->clear('select')
->select('COUNT(*)')
->clear('order');
}
// Get the total number of content items to index.
$this->db->setQuery($query);
return (int) $this->db->loadResult();
}
/**
* Method to get a content item to index.
*
* @param integer $id The id of the content item.
*
* @return FinderIndexerResult A FinderIndexerResult object.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function getItem($id)
{
// Get the list query and add the extra WHERE clause.
$query = $this->getListQuery();
$query->where('a.id = ' . (int) $id);
// Get the item to index.
$this->db->setQuery($query);
$row = $this->db->loadAssoc();
// Convert the item to a result object.
$item = ArrayHelper::toObject((array) $row,
'FinderIndexerResult');
// Set the item type.
$item->type_id = $this->type_id;
// Set the item layout.
$item->layout = $this->layout;
return $item;
}
/**
* Method to get a list of content items to index.
*
* @param integer $offset The list offset.
* @param integer $limit The list limit.
* @param JDatabaseQuery $query A JDatabaseQuery object. [optional]
*
* @return array An array of FinderIndexerResult objects.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function getItems($offset, $limit, $query = null)
{
$items = array();
// Get the content items to index.
$this->db->setQuery($this->getListQuery($query), $offset,
$limit);
$rows = $this->db->loadAssocList();
// Convert the items to result objects.
foreach ($rows as $row)
{
// Convert the item to a result object.
$item = ArrayHelper::toObject((array) $row,
'FinderIndexerResult');
// Set the item type.
$item->type_id = $this->type_id;
// Set the mime type.
$item->mime = $this->mime;
// Set the item layout.
$item->layout = $this->layout;
// Set the extension if present
if (isset($row->extension))
{
$item->extension = $row->extension;
}
// Add the item to the stack.
$items[] = $item;
}
return $items;
}
/**
* Method to get the SQL query used to retrieve the list of content items.
*
* @param mixed $query A JDatabaseQuery object. [optional]
*
* @return JDatabaseQuery A database object.
*
* @since 2.5
*/
protected function getListQuery($query = null)
{
// Check if we can use the supplied SQL query.
return $query instanceof JDatabaseQuery ? $query :
$this->db->getQuery(true);
}
/**
* Method to get the plugin type
*
* @param integer $id The plugin ID
*
* @return string The plugin type
*
* @since 2.5
*/
protected function getPluginType($id)
{
// Prepare the query
$query = $this->db->getQuery(true)
->select($this->db->quoteName('element'))
->from($this->db->quoteName('#__extensions'))
->where($this->db->quoteName('extension_id') . '
= ' . (int) $id);
$this->db->setQuery($query);
return $this->db->loadResult();
}
/**
* Method to get a SQL query to load the published and access states for
* an article and category.
*
* @return JDatabaseQuery A database object.
*
* @since 2.5
*/
protected function getStateQuery()
{
$query = $this->db->getQuery(true);
// Item ID
$query->select('a.id');
// Item and category published state
$query->select('a.' . $this->state_field . ' AS
state, c.published AS cat_state');
// Item and category access levels
$query->select('a.access, c.access AS cat_access')
->from($this->table . ' AS a')
->join('LEFT', '#__categories AS c ON c.id =
a.catid');
return $query;
}
/**
* Method to get the query clause for getting items to update by time.
*
* @param string $time The modified timestamp.
*
* @return JDatabaseQuery A database object.
*
* @since 2.5
*/
protected function getUpdateQueryByTime($time)
{
// Build an SQL query based on the modified time.
$query = $this->db->getQuery(true)
->where('a.modified >= ' .
$this->db->quote($time));
return $query;
}
/**
* Method to get the query clause for getting items to update by id.
*
* @param array $ids The ids to load.
*
* @return JDatabaseQuery A database object.
*
* @since 2.5
*/
protected function getUpdateQueryByIds($ids)
{
// Build an SQL query based on the item ids.
$query = $this->db->getQuery(true)
->where('a.id IN(' . implode(',', $ids) .
')');
return $query;
}
/**
* Method to get the type id for the adapter content.
*
* @return integer The numeric type id for the content.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function getTypeId()
{
// Get the type id from the database.
$query = $this->db->getQuery(true)
->select($this->db->quoteName('id'))
->from($this->db->quoteName('#__finder_types'))
->where($this->db->quoteName('title') . ' =
' . $this->db->quote($this->type_title));
$this->db->setQuery($query);
return (int) $this->db->loadResult();
}
/**
* Method to get the URL for the item. The URL is how we look up the link
* in the Finder index.
*
* @param integer $id The id of the item.
* @param string $extension The extension the category is in.
* @param string $view The view for the URL.
*
* @return string The URL of the item.
*
* @since 2.5
*/
protected function getUrl($id, $extension, $view)
{
return 'index.php?option=' . $extension .
'&view=' . $view . '&id=' . $id;
}
/**
* Method to get the page title of any menu item that is linked to the
* content item, if it exists and is set.
*
* @param string $url The URL of the item.
*
* @return mixed The title on success, null if not found.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function getItemMenuTitle($url)
{
$return = null;
// Set variables
$user = JFactory::getUser();
$groups = implode(',', $user->getAuthorisedViewLevels());
// Build a query to get the menu params.
$query = $this->db->getQuery(true)
->select($this->db->quoteName('params'))
->from($this->db->quoteName('#__menu'))
->where($this->db->quoteName('link') . ' =
' . $this->db->quote($url))
->where($this->db->quoteName('published') . ' =
1')
->where($this->db->quoteName('access') . ' IN
(' . $groups . ')');
// Get the menu params from the database.
$this->db->setQuery($query);
$params = $this->db->loadResult();
// Check the results.
if (empty($params))
{
return $return;
}
// Instantiate the params.
$params = json_decode($params);
// Get the page title if it is set.
if (isset($params->page_title) && $params->page_title)
{
$return = $params->page_title;
}
return $return;
}
/**
* Method to update index data on access level changes
*
* @param JTable $row A JTable object
*
* @return void
*
* @since 2.5
*/
protected function itemAccessChange($row)
{
$query = clone $this->getStateQuery();
$query->where('a.id = ' . (int) $row->id);
// Get the access level.
$this->db->setQuery($query);
$item = $this->db->loadObject();
// Set the access level.
$temp = max($row->access, $item->cat_access);
// Update the item.
$this->change((int) $row->id, 'access', $temp);
}
/**
* Method to update index data on published state changes
*
* @param array $pks A list of primary key ids of the content that
has changed state.
* @param integer $value The value of the state that the content has
been changed to.
*
* @return void
*
* @since 2.5
*/
protected function itemStateChange($pks, $value)
{
/*
* The item's published state is tied to the category
* published state so we need to look up all published states
* before we change anything.
*/
foreach ($pks as $pk)
{
$query = clone $this->getStateQuery();
$query->where('a.id = ' . (int) $pk);
// Get the published states.
$this->db->setQuery($query);
$item = $this->db->loadObject();
// Translate the state.
$temp = $this->translateState($value, $item->cat_state);
// Update the item.
$this->change($pk, 'state', $temp);
// Reindex the item
$this->reindex($pk);
}
}
/**
* Method to update index data when a plugin is disabled
*
* @param array $pks A list of primary key ids of the content that has
changed state.
*
* @return void
*
* @since 2.5
*/
protected function pluginDisable($pks)
{
// Since multiple plugins may be disabled at a time, we need to check
first
// that we're handling the appropriate one for the context
foreach ($pks as $pk)
{
if ($this->getPluginType($pk) == strtolower($this->context))
{
// Get all of the items to unindex them
$query = clone $this->getStateQuery();
$this->db->setQuery($query);
$items = $this->db->loadColumn();
// Remove each item
foreach ($items as $item)
{
$this->remove($item);
}
}
}
}
/**
* Method to translate the native content states into states that the
* indexer can use.
*
* @param integer $item The item state.
* @param integer $category The category state. [optional]
*
* @return integer The translated indexer state.
*
* @since 2.5
*/
protected function translateState($item, $category = null)
{
// If category is present, factor in its states as well
if ($category !== null && $category == 0)
{
$item = 0;
}
// Translate the state
switch ($item)
{
// Published and archived items only should return a published state
case 1;
case 2:
return 1;
// All other states should return an unpublished state
default:
case 0:
return 0;
}
}
}
PK"�[�o���K�K
helpers/indexer/driver/mysql.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
jimport('joomla.filesystem.file');
/**
* Indexer class supporting MySQL(i) for the Finder indexer package.
*
* The indexer class provides the core functionality of the Finder
* search engine. It is responsible for adding and updating the
* content links table; extracting and scoring tokens; and maintaining
* all referential information for the content.
*
* Note: All exceptions thrown from within this class should be caught
* by the controller.
*
* @since 3.0
*/
class FinderIndexerDriverMysql extends FinderIndexer
{
/**
* Method to index a content item.
*
* @param FinderIndexerResult $item The content item to index.
* @param string $format The format of the content.
[optional]
*
* @return integer The ID of the record in the links table.
*
* @since 3.0
* @throws Exception on database error.
*/
public function index($item, $format = 'html')
{
// Mark beforeIndexing in the profiler.
static::$profiler ?
static::$profiler->mark('beforeIndexing') : null;
$db = $this->db;
$nd = $db->getNullDate();
// Check if the item is in the database.
$query = $db->getQuery(true)
->select($db->quoteName('link_id') . ', ' .
$db->quoteName('md5sum'))
->from($db->quoteName('#__finder_links'))
->where($db->quoteName('url') . ' = ' .
$db->quote($item->url));
// Load the item from the database.
$db->setQuery($query);
$link = $db->loadObject();
// Get the indexer state.
$state = static::getState();
// Get the signatures of the item.
$curSig = static::getSignature($item);
$oldSig = isset($link->md5sum) ? $link->md5sum : null;
// Get the other item information.
$linkId = empty($link->link_id) ? null : $link->link_id;
$isNew = empty($link->link_id) ? true : false;
// Check the signatures. If they match, the item is up to date.
if (!$isNew && $curSig == $oldSig)
{
return $linkId;
}
/*
* If the link already exists, flush all the term maps for the item.
* Maps are stored in 16 tables so we need to iterate through and flush
* each table one at a time.
*/
if (!$isNew)
{
for ($i = 0; $i <= 15; $i++)
{
// Flush the maps for the link.
$query->clear()
->delete($db->quoteName('#__finder_links_terms' .
dechex($i)))
->where($db->quoteName('link_id') . ' = ' .
(int) $linkId);
$db->setQuery($query);
$db->execute();
}
// Remove the taxonomy maps.
FinderIndexerTaxonomy::removeMaps($linkId);
}
// Mark afterUnmapping in the profiler.
static::$profiler ?
static::$profiler->mark('afterUnmapping') : null;
// Perform cleanup on the item data.
$item->publish_start_date = (int) $item->publish_start_date != 0 ?
$item->publish_start_date : $nd;
$item->publish_end_date = (int) $item->publish_end_date != 0 ?
$item->publish_end_date : $nd;
$item->start_date = (int) $item->start_date != 0 ?
$item->start_date : $nd;
$item->end_date = (int) $item->end_date != 0 ? $item->end_date :
$nd;
// Prepare the item description.
$item->description = FinderIndexerHelper::parse($item->summary);
/*
* Now, we need to enter the item into the links table. If the item
* already exists in the database, we need to use an UPDATE query.
* Otherwise, we need to use an INSERT to get the link id back.
*/
if ($isNew)
{
$columnsArray = array(
$db->quoteName('url'),
$db->quoteName('route'), $db->quoteName('title'),
$db->quoteName('description'),
$db->quoteName('indexdate'),
$db->quoteName('published'),
$db->quoteName('state'),
$db->quoteName('access'),
$db->quoteName('language'),
$db->quoteName('type_id'),
$db->quoteName('object'),
$db->quoteName('publish_start_date'),
$db->quoteName('publish_end_date'),
$db->quoteName('start_date'),
$db->quoteName('end_date'),
$db->quoteName('list_price'),
$db->quoteName('sale_price')
);
// Insert the link.
$query->clear()
->insert($db->quoteName('#__finder_links'))
->columns($columnsArray)
->values(
$db->quote($item->url) . ', '
. $db->quote($item->route) . ', '
. $db->quote($item->title) . ', '
. $db->quote($item->description) . ', '
. $query->currentTimestamp() . ', '
. '1, '
. (int) $item->state . ', '
. (int) $item->access . ', '
. $db->quote($item->language) . ', '
. (int) $item->type_id . ', '
. $db->quote(serialize($item)) . ', '
. $db->quote($item->publish_start_date) . ', '
. $db->quote($item->publish_end_date) . ', '
. $db->quote($item->start_date) . ', '
. $db->quote($item->end_date) . ', '
. (double) ($item->list_price ?: 0) . ', '
. (double) ($item->sale_price ?: 0)
);
$db->setQuery($query);
$db->execute();
// Get the link id.
$linkId = (int) $db->insertid();
}
else
{
// Update the link.
$query->clear()
->update($db->quoteName('#__finder_links'))
->set($db->quoteName('route') . ' = ' .
$db->quote($item->route))
->set($db->quoteName('title') . ' = ' .
$db->quote($item->title))
->set($db->quoteName('description') . ' = ' .
$db->quote($item->description))
->set($db->quoteName('indexdate') . ' = ' .
$query->currentTimestamp())
->set($db->quoteName('state') . ' = ' . (int)
$item->state)
->set($db->quoteName('access') . ' = ' .
(int) $item->access)
->set($db->quoteName('language') . ' = ' .
$db->quote($item->language))
->set($db->quoteName('type_id') . ' = ' .
(int) $item->type_id)
->set($db->quoteName('object') . ' = ' .
$db->quote(serialize($item)))
->set($db->quoteName('publish_start_date') . ' =
' . $db->quote($item->publish_start_date))
->set($db->quoteName('publish_end_date') . ' =
' . $db->quote($item->publish_end_date))
->set($db->quoteName('start_date') . ' = ' .
$db->quote($item->start_date))
->set($db->quoteName('end_date') . ' = ' .
$db->quote($item->end_date))
->set($db->quoteName('list_price') . ' = ' .
(double) ($item->list_price ?: 0))
->set($db->quoteName('sale_price') . ' = ' .
(double) ($item->sale_price ?: 0))
->where('link_id = ' . (int) $linkId);
$db->setQuery($query);
$db->execute();
}
// Set up the variables we will need during processing.
$count = 0;
// Mark afterLinking in the profiler.
static::$profiler ? static::$profiler->mark('afterLinking')
: null;
// Truncate the tokens tables.
$db->truncateTable('#__finder_tokens');
// Truncate the tokens aggregate table.
$db->truncateTable('#__finder_tokens_aggregate');
/*
* Process the item's content. The items can customize their
* processing instructions to define extra properties to process
* or rearrange how properties are weighted.
*/
foreach ($item->getInstructions() as $group => $properties)
{
// Iterate through the properties of the group.
foreach ($properties as $property)
{
// Check if the property exists in the item.
if (empty($item->$property))
{
continue;
}
// Tokenize the property.
if (is_array($item->$property))
{
// Tokenize an array of content and add it to the database.
foreach ($item->$property as $ip)
{
/*
* If the group is path, we need to a few extra processing
* steps to strip the extension and convert slashes and dashes
* to spaces.
*/
if ($group === static::PATH_CONTEXT)
{
$ip = JFile::stripExt($ip);
$ip = str_replace(array('/', '-'), '
', $ip);
}
// Tokenize a string of content and add it to the database.
$count += $this->tokenizeToDb($ip, $group, $item->language,
$format);
// Check if we're approaching the memory limit of the token
table.
if ($count >
static::$state->options->get('memory_table_limit', 30000))
{
$this->toggleTables(false);
}
}
}
else
{
/*
* If the group is path, we need to a few extra processing
* steps to strip the extension and convert slashes and dashes
* to spaces.
*/
if ($group === static::PATH_CONTEXT)
{
$item->$property = JFile::stripExt($item->$property);
$item->$property = str_replace('/', ' ',
$item->$property);
$item->$property = str_replace('-', ' ',
$item->$property);
}
// Tokenize a string of content and add it to the database.
$count += $this->tokenizeToDb($item->$property, $group,
$item->language, $format);
// Check if we're approaching the memory limit of the token
table.
if ($count >
static::$state->options->get('memory_table_limit', 30000))
{
$this->toggleTables(false);
}
}
}
}
/*
* Process the item's taxonomy. The items can customize their
* taxonomy mappings to define extra properties to map.
*/
foreach ($item->getTaxonomy() as $branch => $nodes)
{
// Iterate through the nodes and map them to the branch.
foreach ($nodes as $node)
{
// Add the node to the tree.
$nodeId = FinderIndexerTaxonomy::addNode($branch, $node->title,
$node->state, $node->access);
// Add the link => node map.
FinderIndexerTaxonomy::addMap($linkId, $nodeId);
}
}
// Mark afterProcessing in the profiler.
static::$profiler ?
static::$profiler->mark('afterProcessing') : null;
/*
* At this point, all of the item's content has been parsed,
tokenized
* and inserted into the #__finder_tokens table. Now, we need to
* aggregate all the data into that table into a more usable form. The
* aggregated data will be inserted into #__finder_tokens_aggregate
* table.
*/
$query = 'INSERT INTO ' .
$db->quoteName('#__finder_tokens_aggregate') .
' (' . $db->quoteName('term_id') .
', ' . $db->quoteName('map_suffix') .
', ' . $db->quoteName('term') .
', ' . $db->quoteName('stem') .
', ' . $db->quoteName('common') .
', ' . $db->quoteName('phrase') .
', ' . $db->quoteName('term_weight') .
', ' . $db->quoteName('context') .
', ' . $db->quoteName('context_weight') .
', ' . $db->quoteName('total_weight') .
', ' . $db->quoteName('language') .
')' .
' SELECT' .
' COALESCE(t.term_id, 0), \'\', t1.term, t1.stem,
t1.common, t1.phrase, t1.weight, t1.context,' .
' ROUND( t1.weight * COUNT( t2.term ) * %F, 8 ) AS context_weight,
0, t1.language' .
' FROM (' .
' SELECT DISTINCT t1.term, t1.stem, t1.common, t1.phrase,
t1.weight, t1.context, t1.language' .
' FROM ' . $db->quoteName('#__finder_tokens') .
' AS t1' .
' WHERE t1.context = %d' .
' ) AS t1' .
' JOIN ' . $db->quoteName('#__finder_tokens') .
' AS t2 ON t2.term = t1.term' .
' LEFT JOIN ' . $db->quoteName('#__finder_terms')
. ' AS t ON t.term = t1.term' .
' WHERE t2.context = %d' .
' GROUP BY t1.term, t.term_id, t1.term, t1.stem, t1.common,
t1.phrase, t1.weight, t1.context, t1.language' .
' ORDER BY t1.term DESC';
// Iterate through the contexts and aggregate the tokens per context.
foreach ($state->weights as $context => $multiplier)
{
// Run the query to aggregate the tokens for this context..
$db->setQuery(sprintf($query, $multiplier, $context, $context));
$db->execute();
}
// Mark afterAggregating in the profiler.
static::$profiler ?
static::$profiler->mark('afterAggregating') : null;
/*
* When we pulled down all of the aggregate data, we did a LEFT JOIN
* over the terms table to try to find all the term ids that
* already exist for our tokens. If any of the rows in the aggregate
* table have a term of 0, then no term record exists for that
* term so we need to add it to the terms table.
*/
$db->setQuery(
'INSERT IGNORE INTO ' .
$db->quoteName('#__finder_terms') .
' (' . $db->quoteName('term') .
', ' . $db->quoteName('stem') .
', ' . $db->quoteName('common') .
', ' . $db->quoteName('phrase') .
', ' . $db->quoteName('weight') .
', ' . $db->quoteName('soundex') .
', ' . $db->quoteName('language') . ')'
.
' SELECT ta.term, ta.stem, ta.common, ta.phrase, ta.term_weight,
SOUNDEX(ta.term), ta.language' .
' FROM ' .
$db->quoteName('#__finder_tokens_aggregate') . ' AS
ta' .
' WHERE ta.term_id = 0' .
' GROUP BY ta.term, ta.stem, ta.common, ta.phrase, ta.term_weight,
SOUNDEX(ta.term), ta.language'
);
$db->execute();
/*
* Now, we just inserted a bunch of new records into the terms table
* so we need to go back and update the aggregate table with all the
* new term ids.
*/
$query = $db->getQuery(true)
->update($db->quoteName('#__finder_tokens_aggregate') .
' AS ta')
->join('INNER',
$db->quoteName('#__finder_terms') . ' AS t ON t.term =
ta.term')
->set('ta.term_id = t.term_id')
->where('ta.term_id = 0');
$db->setQuery($query);
$db->execute();
// Mark afterTerms in the profiler.
static::$profiler ? static::$profiler->mark('afterTerms') :
null;
/*
* After we've made sure that all of the terms are in the terms
table
* and the aggregate table has the correct term ids, we need to update
* the links counter for each term by one.
*/
$query->clear()
->update($db->quoteName('#__finder_terms') . ' AS
t')
->join('INNER',
$db->quoteName('#__finder_tokens_aggregate') . ' AS ta ON
ta.term_id = t.term_id')
->set('t.' . $db->quoteName('links') . '
= t.links + 1');
$db->setQuery($query);
$db->execute();
// Mark afterTerms in the profiler.
static::$profiler ? static::$profiler->mark('afterTerms') :
null;
/*
* Before we can insert all of the mapping rows, we have to figure out
* which mapping table the rows need to be inserted into. The mapping
* table for each term is based on the first character of the md5 of
* the first character of the term. In php, it would be expressed as
* substr(md5(substr($token, 0, 1)), 0, 1)
*/
$query->clear()
->update($db->quoteName('#__finder_tokens_aggregate'))
->set($db->quoteName('map_suffix') . ' =
SUBSTR(MD5(SUBSTR(' . $db->quoteName('term') . ', 1,
1)), 1, 1)');
$db->setQuery($query);
$db->execute();
/*
* At this point, the aggregate table contains a record for each
* term in each context. So, we're going to pull down all of that
* data while grouping the records by term and add all of the
* sub-totals together to arrive at the final total for each token for
* this link. Then, we insert all of that data into the appropriate
* mapping table.
*/
for ($i = 0; $i <= 15; $i++)
{
// Get the mapping table suffix.
$suffix = dechex($i);
/*
* We have to run this query 16 times, one for each link => term
* mapping table.
*/
$db->setQuery(
'INSERT INTO ' .
$db->quoteName('#__finder_links_terms' . $suffix) .
' (' . $db->quoteName('link_id') .
', ' . $db->quoteName('term_id') .
', ' . $db->quoteName('weight') . ')'
.
' SELECT ' . (int) $linkId . ', ' .
$db->quoteName('term_id') . ',' .
' ROUND(SUM(' . $db->quoteName('context_weight')
. '), 8)' .
' FROM ' .
$db->quoteName('#__finder_tokens_aggregate') .
' WHERE ' . $db->quoteName('map_suffix') .
' = ' . $db->quote($suffix) .
' GROUP BY ' . $db->quoteName('term') . ',
' . $db->quoteName('term_id') .
' ORDER BY ' . $db->quoteName('term') . '
DESC'
);
$db->execute();
}
// Mark afterMapping in the profiler.
static::$profiler ? static::$profiler->mark('afterMapping')
: null;
// Update the signature.
$query->clear()
->update($db->quoteName('#__finder_links'))
->set($db->quoteName('md5sum') . ' = ' .
$db->quote($curSig))
->where($db->quoteName('link_id') . ' = ' .
$db->quote($linkId));
$db->setQuery($query);
$db->execute();
// Mark afterSigning in the profiler.
static::$profiler ? static::$profiler->mark('afterSigning')
: null;
// Truncate the tokens tables.
$db->truncateTable('#__finder_tokens');
// Truncate the tokens aggregate table.
$db->truncateTable('#__finder_tokens_aggregate');
// Toggle the token tables back to memory tables.
$this->toggleTables(true);
// Mark afterTruncating in the profiler.
static::$profiler ?
static::$profiler->mark('afterTruncating') : null;
return $linkId;
}
/**
* Method to optimize the index. We use this method to remove unused terms
* and any other optimizations that might be necessary.
*
* @return boolean True on success.
*
* @since 3.0
* @throws Exception on database error.
*/
public function optimize()
{
// Get the database object.
$db = $this->db;
$query = $db->getQuery(true);
// Delete all orphaned terms.
$query->delete($db->quoteName('#__finder_terms'))
->where($db->quoteName('links') . ' <= 0');
$db->setQuery($query);
$db->execute();
// Optimize the links table.
$db->setQuery('OPTIMIZE TABLE ' .
$db->quoteName('#__finder_links'));
$db->execute();
for ($i = 0; $i <= 15; $i++)
{
// Optimize the terms mapping table.
$db->setQuery('OPTIMIZE TABLE ' .
$db->quoteName('#__finder_links_terms' . dechex($i)));
$db->execute();
}
// Optimize the filters table.
$db->setQuery('OPTIMIZE TABLE ' .
$db->quoteName('#__finder_filters'));
$db->execute();
// Optimize the terms common table.
$db->setQuery('OPTIMIZE TABLE ' .
$db->quoteName('#__finder_terms_common'));
$db->execute();
// Optimize the types table.
$db->setQuery('OPTIMIZE TABLE ' .
$db->quoteName('#__finder_types'));
$db->execute();
// Remove the orphaned taxonomy nodes.
FinderIndexerTaxonomy::removeOrphanNodes();
// Optimize the taxonomy mapping table.
$db->setQuery('OPTIMIZE TABLE ' .
$db->quoteName('#__finder_taxonomy_map'));
$db->execute();
// Optimize the taxonomy table.
$db->setQuery('OPTIMIZE TABLE ' .
$db->quoteName('#__finder_taxonomy'));
$db->execute();
return true;
}
/**
* Method to switch the token tables from Memory tables to MyISAM tables
* when they are close to running out of memory.
*
* @param boolean $memory Flag to control how they should be toggled.
*
* @return boolean True on success.
*
* @since 3.0
* @throws Exception on database error.
*/
protected function toggleTables($memory)
{
static $state;
// Get the database adapter.
$db = $this->db;
// Check if we are setting the tables to the Memory engine.
if ($memory === true && $state !== true)
{
// Set the tokens table to Memory.
$db->setQuery('ALTER TABLE ' .
$db->quoteName('#__finder_tokens') . ' ENGINE =
MEMORY');
$db->execute();
// Set the tokens aggregate table to Memory.
$db->setQuery('ALTER TABLE ' .
$db->quoteName('#__finder_tokens_aggregate') . ' ENGINE =
MEMORY');
$db->execute();
// Set the internal state.
$state = $memory;
}
// We must be setting the tables to the MyISAM engine.
elseif ($memory === false && $state !== false)
{
// Set the tokens table to MyISAM.
$db->setQuery('ALTER TABLE ' .
$db->quoteName('#__finder_tokens') . ' ENGINE =
MYISAM');
$db->execute();
// Set the tokens aggregate table to MyISAM.
$db->setQuery('ALTER TABLE ' .
$db->quoteName('#__finder_tokens_aggregate') . ' ENGINE =
MYISAM');
$db->execute();
// Set the internal state.
$state = $memory;
}
return true;
}
}
PK"�[�RC��?�?%helpers/indexer/driver/postgresql.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\CMS\Factory;
use Joomla\String\StringHelper;
jimport('joomla.filesystem.file');
/**
* Indexer class supporting PostgreSQL for the Finder indexer package.
*
* @since 3.0
*/
class FinderIndexerDriverPostgresql extends FinderIndexer
{
/**
* Method to index a content item.
*
* @param FinderIndexerResult $item The content item to index.
* @param string $format The format of the content.
[optional]
*
* @return integer The ID of the record in the links table.
*
* @since 3.0
* @throws Exception on database error.
*/
public function index($item, $format = 'html')
{
// Mark beforeIndexing in the profiler.
static::$profiler ?
static::$profiler->mark('beforeIndexing') : null;
$db = $this->db;
$nd = $db->getNullDate();
// Check if the item is in the database.
$query = $db->getQuery(true)
->select($db->quoteName('link_id') . ', ' .
$db->quoteName('md5sum'))
->from($db->quoteName('#__finder_links'))
->where($db->quoteName('url') . ' = ' .
$db->quote($item->url));
// Load the item from the database.
$db->setQuery($query);
$link = $db->loadObject();
// Get the indexer state.
$state = static::getState();
// Get the signatures of the item.
$curSig = static::getSignature($item);
$oldSig = isset($link->md5sum) ? $link->md5sum : null;
// Get the other item information.
$linkId = empty($link->link_id) ? null : $link->link_id;
$isNew = empty($link->link_id) ? true : false;
// Check the signatures. If they match, the item is up to date.
if (!$isNew && $curSig === $oldSig)
{
return $linkId;
}
/*
* If the link already exists, flush all the term maps for the item.
* Maps are stored in 16 tables so we need to iterate through and flush
* each table one at a time.
*/
if (!$isNew)
{
for ($i = 0; $i <= 15; $i++)
{
// Flush the maps for the link.
$query->clear()
->delete($db->quoteName('#__finder_links_terms' .
dechex($i)))
->where($db->quoteName('link_id') . ' = ' .
(int) $linkId);
$db->setQuery($query);
$db->execute();
}
// Remove the taxonomy maps.
FinderIndexerTaxonomy::removeMaps($linkId);
}
// Mark afterUnmapping in the profiler.
static::$profiler ?
static::$profiler->mark('afterUnmapping') : null;
// Perform cleanup on the item data.
$item->publish_start_date = (int) $item->publish_start_date != 0 ?
$item->publish_start_date : $nd;
$item->publish_end_date = (int) $item->publish_end_date != 0 ?
$item->publish_end_date : $nd;
$item->start_date = (int) $item->start_date != 0 ?
$item->start_date : $nd;
$item->end_date = (int) $item->end_date != 0 ? $item->end_date :
$nd;
// Prepare the item description.
$item->description = FinderIndexerHelper::parse($item->summary);
/*
* Now, we need to enter the item into the links table. If the item
* already exists in the database, we need to use an UPDATE query.
* Otherwise, we need to use an INSERT to get the link id back.
*/
$entry = new stdClass;
$entry->url = $item->url;
$entry->route = $item->route;
$entry->title = $item->title;
// We are shortening the description in order to not run into length
issues with this field
$entry->description = StringHelper::substr($item->description, 0,
32000);
$entry->indexdate = Factory::getDate()->toSql();
$entry->state = (int) $item->state;
$entry->access = (int) $item->access;
$entry->language = $item->language;
$entry->type_id = (int) $item->type_id;
$entry->object = '';
$entry->publish_start_date = $item->publish_start_date;
$entry->publish_end_date = $item->publish_end_date;
$entry->start_date = $item->start_date;
$entry->end_date = $item->end_date;
$entry->list_price = (double) ($item->list_price ?: 0);
$entry->sale_price = (double) ($item->sale_price ?: 0);
if ($isNew)
{
// Insert the link and get its id.
$db->insertObject('#__finder_links', $entry);
$linkId = (int) $db->insertid();
}
else
{
// Update the link.
$entry->link_id = $linkId;
$db->updateObject('#__finder_links', $entry,
'link_id');
}
// Set up the variables we will need during processing.
$count = 0;
// Mark afterLinking in the profiler.
static::$profiler ? static::$profiler->mark('afterLinking')
: null;
// Truncate the tokens tables.
$db->truncateTable('#__finder_tokens');
// Truncate the tokens aggregate table.
$db->truncateTable('#__finder_tokens_aggregate');
/*
* Process the item's content. The items can customize their
* processing instructions to define extra properties to process
* or rearrange how properties are weighted.
*/
foreach ($item->getInstructions() as $group => $properties)
{
// Iterate through the properties of the group.
foreach ($properties as $property)
{
// Check if the property exists in the item.
if (empty($item->$property))
{
continue;
}
// Tokenize the property.
if (is_array($item->$property))
{
// Tokenize an array of content and add it to the database.
foreach ($item->$property as $ip)
{
/*
* If the group is path, we need to a few extra processing
* steps to strip the extension and convert slashes and dashes
* to spaces.
*/
if ($group === static::PATH_CONTEXT)
{
$ip = JFile::stripExt($ip);
$ip = str_replace(array('/', '-'), '
', $ip);
}
// Tokenize a string of content and add it to the database.
$count += $this->tokenizeToDb($ip, $group, $item->language,
$format);
// Check if we're approaching the memory limit of the token
table.
if ($count >
static::$state->options->get('memory_table_limit', 30000))
{
$this->toggleTables(false);
}
}
}
else
{
/*
* If the group is path, we need to a few extra processing
* steps to strip the extension and convert slashes and dashes
* to spaces.
*/
if ($group === static::PATH_CONTEXT)
{
$item->$property = JFile::stripExt($item->$property);
$item->$property = str_replace('/', ' ',
$item->$property);
$item->$property = str_replace('-', ' ',
$item->$property);
}
// Tokenize a string of content and add it to the database.
$count += $this->tokenizeToDb($item->$property, $group,
$item->language, $format);
// Check if we're approaching the memory limit of the token
table.
if ($count >
static::$state->options->get('memory_table_limit', 30000))
{
$this->toggleTables(false);
}
}
}
}
/*
* Process the item's taxonomy. The items can customize their
* taxonomy mappings to define extra properties to map.
*/
foreach ($item->getTaxonomy() as $branch => $nodes)
{
// Iterate through the nodes and map them to the branch.
foreach ($nodes as $node)
{
// Add the node to the tree.
$nodeId = FinderIndexerTaxonomy::addNode($branch, $node->title,
$node->state, $node->access);
// Add the link => node map.
FinderIndexerTaxonomy::addMap($linkId, $nodeId);
}
}
// Mark afterProcessing in the profiler.
static::$profiler ?
static::$profiler->mark('afterProcessing') : null;
/*
* At this point, all of the item's content has been parsed,
tokenized
* and inserted into the #__finder_tokens table. Now, we need to
* aggregate all the data into that table into a more usable form. The
* aggregated data will be inserted into #__finder_tokens_aggregate
* table.
*/
$query = 'INSERT INTO ' .
$db->quoteName('#__finder_tokens_aggregate') .
' (' . $db->quoteName('term_id') .
', ' . $db->quoteName('term') .
', ' . $db->quoteName('stem') .
', ' . $db->quoteName('common') .
', ' . $db->quoteName('phrase') .
', ' . $db->quoteName('term_weight') .
', ' . $db->quoteName('context') .
', ' . $db->quoteName('context_weight') .
', ' . $db->quoteName('language') .
')' .
' SELECT' .
' t.term_id, t1.term, t1.stem, t1.common, t1.phrase, t1.weight,
t1.context,' .
' ROUND( t1.weight * COUNT( t2.term ) * %F, 8 ) AS context_weight,
t1.language' .
' FROM (' .
' SELECT DISTINCT t1.term, t1.stem, t1.common, t1.phrase,
t1.weight, t1.context, t1.language' .
' FROM ' . $db->quoteName('#__finder_tokens')
. ' AS t1' .
' WHERE t1.context = %d' .
' ) AS t1' .
' JOIN ' . $db->quoteName('#__finder_tokens') .
' AS t2 ON t2.term = t1.term' .
' LEFT JOIN ' .
$db->quoteName('#__finder_terms') . ' AS t ON t.term =
t1.term' .
' WHERE t2.context = %d AND t.term_id IS NOT NULL' .
' GROUP BY t1.term, t.term_id, t1.term, t1.stem, t1.common,
t1.phrase, t1.weight, t1.context, t1.language' .
' ORDER BY t1.term DESC';
// Iterate through the contexts and aggregate the tokens per context.
foreach ($state->weights as $context => $multiplier)
{
// Run the query to aggregate the tokens for this context..
$db->setQuery(sprintf($query, $multiplier, $context, $context));
$db->execute();
}
// Mark afterAggregating in the profiler.
static::$profiler ?
static::$profiler->mark('afterAggregating') : null;
/*
* When we pulled down all of the aggregate data, we did a LEFT JOIN
* over the terms table to try to find all the term ids that
* already exist for our tokens. If any of the rows in the aggregate
* table have a term of 0, then no term record exists for that
* term so we need to add it to the terms table.
*/
// Emulation of IGNORE INTO behaviour
$db->setQuery(
' SELECT ta.term' .
' FROM ' .
$db->quoteName('#__finder_tokens_aggregate') . ' AS
ta' .
' WHERE ta.term_id = 0'
);
if ($db->loadRow() === null)
{
$db->setQuery(
'INSERT INTO ' .
$db->quoteName('#__finder_terms') .
' (' . $db->quoteName('term') .
', ' . $db->quoteName('stem') .
', ' . $db->quoteName('common') .
', ' . $db->quoteName('phrase') .
', ' . $db->quoteName('weight') .
', ' . $db->quoteName('soundex') .
', ' . $db->quoteName('language') .
')' .
' SELECT ta.term, ta.stem, ta.common, ta.phrase, ta.term_weight,
SOUNDEX(ta.term), ta.language' .
' FROM ' .
$db->quoteName('#__finder_tokens_aggregate') . ' AS
ta' .
' WHERE ta.term_id = 0' .
' GROUP BY ta.term, ta.stem, ta.common, ta.phrase, ta.term_weight,
SOUNDEX(ta.term), ta.language'
);
$db->execute();
}
/*
* Now, we just inserted a bunch of new records into the terms table
* so we need to go back and update the aggregate table with all the
* new term ids.
*/
$query = $db->getQuery(true)
->update($db->quoteName('#__finder_tokens_aggregate') .
' AS ta')
->join('INNER',
$db->quoteName('#__finder_terms') . ' AS t ON t.term =
ta.term')
->set('term_id = t.term_id')
->where('ta.term_id = 0');
$db->setQuery($query);
$db->execute();
// Mark afterTerms in the profiler.
static::$profiler ? static::$profiler->mark('afterTerms') :
null;
/*
* After we've made sure that all of the terms are in the terms
table
* and the aggregate table has the correct term ids, we need to update
* the links counter for each term by one.
*/
$query->clear()
->update($db->quoteName('#__finder_terms') . ' AS
t')
->join('INNER',
$db->quoteName('#__finder_tokens_aggregate') . ' AS ta ON
ta.term_id = t.term_id')
->set($db->quoteName('links') . ' = t.links +
1');
$db->setQuery($query);
$db->execute();
// Mark afterTerms in the profiler.
static::$profiler ? static::$profiler->mark('afterTerms') :
null;
/*
* Before we can insert all of the mapping rows, we have to figure out
* which mapping table the rows need to be inserted into. The mapping
* table for each term is based on the first character of the md5 of
* the first character of the term. In php, it would be expressed as
* substr(md5(substr($token, 0, 1)), 0, 1)
*/
$query->clear()
->update($db->quoteName('#__finder_tokens_aggregate'))
->set($db->quoteName('map_suffix') . ' =
SUBSTR(MD5(SUBSTR(' . $db->quoteName('term') . ', 1,
1)), 1, 1)');
$db->setQuery($query);
$db->execute();
/*
* At this point, the aggregate table contains a record for each
* term in each context. So, we're going to pull down all of that
* data while grouping the records by term and add all of the
* sub-totals together to arrive at the final total for each token for
* this link. Then, we insert all of that data into the appropriate
* mapping table.
*/
for ($i = 0; $i <= 15; $i++)
{
// Get the mapping table suffix.
$suffix = dechex($i);
/*
* We have to run this query 16 times, one for each link => term
* mapping table.
*/
$db->setQuery(
'INSERT INTO ' .
$db->quoteName('#__finder_links_terms' . $suffix) .
' (' . $db->quoteName('link_id') .
', ' . $db->quoteName('term_id') .
', ' . $db->quoteName('weight') . ')'
.
' SELECT ' . (int) $linkId . ', ' .
$db->quoteName('term_id') . ',' .
' ROUND(SUM(' . $db->quoteName('context_weight')
. '), 8)' .
' FROM ' .
$db->quoteName('#__finder_tokens_aggregate') .
' WHERE ' . $db->quoteName('map_suffix') .
' = ' . $db->quote($suffix) .
' GROUP BY ' . $db->quoteName('term') . ',
' . $db->quoteName('term_id') .
' ORDER BY ' . $db->quoteName('term') . '
DESC'
);
$db->execute();
}
// Mark afterMapping in the profiler.
static::$profiler ? static::$profiler->mark('afterMapping')
: null;
// Update the signature.
$query->clear()
->update($db->quoteName('#__finder_links'))
->set($db->quoteName('md5sum') . ' = ' .
$db->quote($curSig))
->where($db->quoteName('link_id') . ' = ' .
$db->quote($linkId));
$db->setQuery($query);
$db->execute();
// Mark afterSigning in the profiler.
static::$profiler ? static::$profiler->mark('afterSigning')
: null;
// Truncate the tokens tables.
$db->truncateTable('#__finder_tokens');
// Truncate the tokens aggregate table.
$db->truncateTable('#__finder_tokens_aggregate');
// Toggle the token tables back to memory tables.
$this->toggleTables(true);
// Mark afterTruncating in the profiler.
static::$profiler ?
static::$profiler->mark('afterTruncating') : null;
return $linkId;
}
/**
* Method to optimize the index. We use this method to remove unused terms
* and any other optimizations that might be necessary.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
public function optimize()
{
// Get the database object.
$db = $this->db;
$query = $db->getQuery(true);
// Delete all orphaned terms.
$query->delete($db->quoteName('#__finder_terms'))
->where($db->quoteName('links') . ' <= 0');
$db->setQuery($query);
$db->execute();
// Optimize the links table.
$db->setQuery('VACUUM ' .
$db->quoteName('#__finder_links'));
$db->execute();
$db->setQuery('REINDEX TABLE ' .
$db->quoteName('#__finder_links'));
$db->execute();
for ($i = 0; $i <= 15; $i++)
{
// Optimize the terms mapping table.
$db->setQuery('VACUUM ' .
$db->quoteName('#__finder_links_terms' . dechex($i)));
$db->execute();
$db->setQuery('REINDEX TABLE ' .
$db->quoteName('#__finder_links_terms' . dechex($i)));
$db->execute();
}
// Optimize the filters table.
$db->setQuery('REINDEX TABLE ' .
$db->quoteName('#__finder_filters'));
$db->execute();
// Optimize the terms common table.
$db->setQuery('REINDEX TABLE ' .
$db->quoteName('#__finder_terms_common'));
$db->execute();
// Optimize the types table.
$db->setQuery('REINDEX TABLE ' .
$db->quoteName('#__finder_types'));
$db->execute();
// Remove the orphaned taxonomy nodes.
FinderIndexerTaxonomy::removeOrphanNodes();
// Optimize the taxonomy mapping table.
$db->setQuery('REINDEX TABLE ' .
$db->quoteName('#__finder_taxonomy_map'));
$db->execute();
// Optimize the taxonomy table.
$db->setQuery('REINDEX TABLE ' .
$db->quoteName('#__finder_taxonomy'));
$db->execute();
return true;
}
}
PK"�[f�X�GGGG!helpers/indexer/driver/sqlsrv.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
jimport('joomla.filesystem.file');
/**
* Indexer class supporting SQL Server for the Finder indexer package.
*
* The indexer class provides the core functionality of the Finder
* search engine. It is responsible for adding and updating the
* content links table; extracting and scoring tokens; and maintaining
* all referential information for the content.
*
* Note: All exceptions thrown from within this class should be caught
* by the controller.
*
* @since 3.1
*/
class FinderIndexerDriverSqlsrv extends FinderIndexer
{
/**
* Method to index a content item.
*
* @param FinderIndexerResult $item The content item to index.
* @param string $format The format of the content.
[optional]
*
* @return integer The ID of the record in the links table.
*
* @since 3.1
* @throws Exception on database error.
*/
public function index($item, $format = 'html')
{
// Mark beforeIndexing in the profiler.
static::$profiler ?
static::$profiler->mark('beforeIndexing') : null;
$db = $this->db;
$nd = $db->getNullDate();
// Check if the item is in the database.
$query = $db->getQuery(true)
->select($db->quoteName('link_id') . ', ' .
$db->quoteName('md5sum'))
->from($db->quoteName('#__finder_links'))
->where($db->quoteName('url') . ' = ' .
$db->quote($item->url));
// Load the item from the database.
$db->setQuery($query);
$link = $db->loadObject();
// Get the indexer state.
$state = static::getState();
// Get the signatures of the item.
$curSig = static::getSignature($item);
$oldSig = isset($link->md5sum) ? $link->md5sum : null;
// Get the other item information.
$linkId = empty($link->link_id) ? null : $link->link_id;
$isNew = empty($link->link_id) ? true : false;
// Check the signatures. If they match, the item is up to date.
if (!$isNew && $curSig === $oldSig)
{
return $linkId;
}
/*
* If the link already exists, flush all the term maps for the item.
* Maps are stored in 16 tables so we need to iterate through and flush
* each table one at a time.
*/
if (!$isNew)
{
for ($i = 0; $i <= 15; $i++)
{
// Flush the maps for the link.
$query->clear()
->delete($db->quoteName('#__finder_links_terms' .
dechex($i)))
->where($db->quoteName('link_id') . ' = ' .
(int) $linkId);
$db->setQuery($query);
$db->execute();
}
// Remove the taxonomy maps.
FinderIndexerTaxonomy::removeMaps($linkId);
}
// Mark afterUnmapping in the profiler.
static::$profiler ?
static::$profiler->mark('afterUnmapping') : null;
// Perform cleanup on the item data.
$item->publish_start_date = (int) $item->publish_start_date != 0 ?
$item->publish_start_date : $nd;
$item->publish_end_date = (int) $item->publish_end_date != 0 ?
$item->publish_end_date : $nd;
$item->start_date = (int) $item->start_date != 0 ?
$item->start_date : $nd;
$item->end_date = (int) $item->end_date != 0 ? $item->end_date :
$nd;
// Prepare the item description.
$item->description = FinderIndexerHelper::parse($item->summary);
/*
* Now, we need to enter the item into the links table. If the item
* already exists in the database, we need to use an UPDATE query.
* Otherwise, we need to use an INSERT to get the link id back.
*/
if ($isNew)
{
$columnsArray = array(
$db->quoteName('url'),
$db->quoteName('route'), $db->quoteName('title'),
$db->quoteName('description'),
$db->quoteName('indexdate'),
$db->quoteName('published'),
$db->quoteName('state'),
$db->quoteName('access'),
$db->quoteName('language'),
$db->quoteName('type_id'),
$db->quoteName('object'),
$db->quoteName('publish_start_date'),
$db->quoteName('publish_end_date'),
$db->quoteName('start_date'),
$db->quoteName('end_date'),
$db->quoteName('list_price'),
$db->quoteName('sale_price')
);
// Insert the link.
$query->clear()
->insert($db->quoteName('#__finder_links'))
->columns($columnsArray)
->values(
$db->quote($item->url) . ', '
. $db->quote($item->route) . ', '
. $db->quote($item->title) . ', '
. $db->quote($item->description) . ', '
. $query->currentTimestamp() . ', '
. '1, '
. (int) $item->state . ', '
. (int) $item->access . ', '
. $db->quote($item->language) . ', '
. (int) $item->type_id . ', '
. $db->quote(serialize($item)) . ', '
. $db->quote($item->publish_start_date) . ', '
. $db->quote($item->publish_end_date) . ', '
. $db->quote($item->start_date) . ', '
. $db->quote($item->end_date) . ', '
. (double) ($item->list_price ?: 0) . ', '
. (double) ($item->sale_price ?: 0)
);
$db->setQuery($query);
$db->execute();
// Get the link id.
$linkId = (int) $db->insertid();
}
else
{
// Update the link.
$query->clear()
->update($db->quoteName('#__finder_links'))
->set($db->quoteName('route') . ' = ' .
$db->quote($item->route))
->set($db->quoteName('title') . ' = ' .
$db->quote($item->title))
->set($db->quoteName('description') . ' = ' .
$db->quote($item->description))
->set($db->quoteName('indexdate') . ' = ' .
$query->currentTimestamp())
->set($db->quoteName('state') . ' = ' . (int)
$item->state)
->set($db->quoteName('access') . ' = ' .
(int) $item->access)
->set($db->quoteName('language') . ' = ' .
$db->quote($item->language))
->set($db->quoteName('type_id') . ' = ' .
(int) $item->type_id)
->set($db->quoteName('object') . ' = ' .
$db->quote(serialize($item)))
->set($db->quoteName('publish_start_date') . ' =
' . $db->quote($item->publish_start_date))
->set($db->quoteName('publish_end_date') . ' =
' . $db->quote($item->publish_end_date))
->set($db->quoteName('start_date') . ' = ' .
$db->quote($item->start_date))
->set($db->quoteName('end_date') . ' = ' .
$db->quote($item->end_date))
->set($db->quoteName('list_price') . ' = ' .
(double) ($item->list_price ?: 0))
->set($db->quoteName('sale_price') . ' = ' .
(double) ($item->sale_price ?: 0))
->where('link_id = ' . (int) $linkId);
$db->setQuery($query);
$db->execute();
}
// Set up the variables we will need during processing.
$count = 0;
// Mark afterLinking in the profiler.
static::$profiler ? static::$profiler->mark('afterLinking')
: null;
// Truncate the tokens tables.
$db->truncateTable('#__finder_tokens');
// Truncate the tokens aggregate table.
$db->truncateTable('#__finder_tokens_aggregate');
/*
* Process the item's content. The items can customize their
* processing instructions to define extra properties to process
* or rearrange how properties are weighted.
*/
foreach ($item->getInstructions() as $group => $properties)
{
// Iterate through the properties of the group.
foreach ($properties as $property)
{
// Check if the property exists in the item.
if (empty($item->$property))
{
continue;
}
// Tokenize the property.
if (is_array($item->$property))
{
// Tokenize an array of content and add it to the database.
foreach ($item->$property as $ip)
{
/*
* If the group is path, we need to a few extra processing
* steps to strip the extension and convert slashes and dashes
* to spaces.
*/
if ($group === static::PATH_CONTEXT)
{
$ip = JFile::stripExt($ip);
$ip = str_replace(array('/', '-'), '
', $ip);
}
// Tokenize a string of content and add it to the database.
$count += $this->tokenizeToDb($ip, $group, $item->language,
$format);
// Check if we're approaching the memory limit of the token
table.
if ($count >
static::$state->options->get('memory_table_limit', 30000))
{
$this->toggleTables(false);
}
}
}
else
{
/*
* If the group is path, we need to a few extra processing
* steps to strip the extension and convert slashes and dashes
* to spaces.
*/
if ($group === static::PATH_CONTEXT)
{
$item->$property = JFile::stripExt($item->$property);
$item->$property = str_replace('/', ' ',
$item->$property);
$item->$property = str_replace('-', ' ',
$item->$property);
}
// Tokenize a string of content and add it to the database.
$count += $this->tokenizeToDb($item->$property, $group,
$item->language, $format);
// Check if we're approaching the memory limit of the token
table.
if ($count >
static::$state->options->get('memory_table_limit', 30000))
{
$this->toggleTables(false);
}
}
}
}
/*
* Process the item's taxonomy. The items can customize their
* taxonomy mappings to define extra properties to map.
*/
foreach ($item->getTaxonomy() as $branch => $nodes)
{
// Iterate through the nodes and map them to the branch.
foreach ($nodes as $node)
{
// Add the node to the tree.
$nodeId = FinderIndexerTaxonomy::addNode($branch, $node->title,
$node->state, $node->access);
// Add the link => node map.
FinderIndexerTaxonomy::addMap($linkId, $nodeId);
}
}
// Mark afterProcessing in the profiler.
static::$profiler ?
static::$profiler->mark('afterProcessing') : null;
/*
* At this point, all of the item's content has been parsed,
tokenized
* and inserted into the #__finder_tokens table. Now, we need to
* aggregate all the data into that table into a more usable form. The
* aggregated data will be inserted into #__finder_tokens_aggregate
* table.
*/
$query = 'INSERT INTO ' .
$db->quoteName('#__finder_tokens_aggregate') .
' (' . $db->quoteName('term_id') .
', ' . $db->quoteName('term') .
', ' . $db->quoteName('stem') .
', ' . $db->quoteName('common') .
', ' . $db->quoteName('phrase') .
', ' . $db->quoteName('term_weight') .
', ' . $db->quoteName('context') .
', ' . $db->quoteName('context_weight') .
', ' . $db->quoteName('language') .
')' .
' SELECT' .
' t.term_id, t1.term, t1.stem, t1.common, t1.phrase, t1.weight,
t1.context,' .
' ROUND( t1.weight * COUNT( t2.term ) * %F, 8 ) AS context_weight,
t1.language' .
' FROM (' .
' SELECT DISTINCT t1.term, t1.stem, t1.common, t1.phrase,
t1.weight, t1.context, t1.language' .
' FROM ' . $db->quoteName('#__finder_tokens')
. ' AS t1' .
' WHERE t1.context = %d' .
' ) AS t1' .
' JOIN ' . $db->quoteName('#__finder_tokens') .
' AS t2 ON t2.term = t1.term' .
' LEFT JOIN ' .
$db->quoteName('#__finder_terms') . ' AS t ON t.term =
t1.term' .
' WHERE t2.context = %d' .
' GROUP BY t1.term, t.term_id, t1.term, t1.stem, t1.common,
t1.phrase, t1.weight, t1.context, t1.language' .
' ORDER BY t1.term DESC';
// Iterate through the contexts and aggregate the tokens per context.
foreach ($state->weights as $context => $multiplier)
{
// Run the query to aggregate the tokens for this context..
$db->setQuery(sprintf($query, $multiplier, $context, $context));
$db->execute();
}
// Mark afterAggregating in the profiler.
static::$profiler ?
static::$profiler->mark('afterAggregating') : null;
/*
* When we pulled down all of the aggregate data, we did a LEFT JOIN
* over the terms table to try to find all the term ids that
* already exist for our tokens. If any of the rows in the aggregate
* table have a term of 0, then no term record exists for that
* term so we need to add it to the terms table.
*/
$db->setQuery(
'INSERT INTO ' .
$db->quoteName('#__finder_terms') .
' (' . $db->quoteName('term') .
', ' . $db->quoteName('stem') .
', ' . $db->quoteName('common') .
', ' . $db->quoteName('phrase') .
', ' . $db->quoteName('weight') .
', ' . $db->quoteName('soundex') . ')'
.
' SELECT ta.term, ta.stem, ta.common, ta.phrase, ta.term_weight,
SOUNDEX(ta.term)' .
' FROM ' .
$db->quoteName('#__finder_tokens_aggregate') . ' AS
ta' .
' WHERE ta.term_id IS NULL' .
' GROUP BY ta.term, ta.stem, ta.common, ta.phrase,
ta.term_weight'
);
$db->execute();
/*
* Now, we just inserted a bunch of new records into the terms table
* so we need to go back and update the aggregate table with all the
* new term ids.
*/
$query = $db->getQuery(true)
->update('ta')
->set('ta.term_id = t.term_id from #__finder_tokens_aggregate AS
ta INNER JOIN #__finder_terms AS t ON t.term = ta.term')
->where('ta.term_id IS NULL');
$db->setQuery($query);
$db->execute();
// Mark afterTerms in the profiler.
static::$profiler ? static::$profiler->mark('afterTerms') :
null;
/*
* After we've made sure that all of the terms are in the terms
table
* and the aggregate table has the correct term ids, we need to update
* the links counter for each term by one.
*/
$query->clear()
->update('t')
->set('t.links = t.links + 1 FROM #__finder_terms AS t INNER
JOIN #__finder_tokens_aggregate AS ta ON ta.term_id = t.term_id');
$db->setQuery($query);
$db->execute();
// Mark afterTerms in the profiler.
static::$profiler ? static::$profiler->mark('afterTerms') :
null;
/*
* Before we can insert all of the mapping rows, we have to figure out
* which mapping table the rows need to be inserted into. The mapping
* table for each term is based on the first character of the md5 of
* the first character of the term. In php, it would be expressed as
* substr(md5(substr($token, 0, 1)), 0, 1)
*/
$query->clear()
->update($db->quoteName('#__finder_tokens_aggregate'))
->set($db->quoteName('map_suffix') . " =
SUBSTRING(HASHBYTES('MD5', SUBSTRING(" .
$db->quoteName('term') . ', 1, 1)), 1, 1)');
$db->setQuery($query);
$db->execute();
/*
* At this point, the aggregate table contains a record for each
* term in each context. So, we're going to pull down all of that
* data while grouping the records by term and add all of the
* sub-totals together to arrive at the final total for each token for
* this link. Then, we insert all of that data into the appropriate
* mapping table.
*/
for ($i = 0; $i <= 15; $i++)
{
// Get the mapping table suffix.
$suffix = dechex($i);
/*
* We have to run this query 16 times, one for each link => term
* mapping table.
*/
$db->setQuery(
'INSERT INTO ' .
$db->quoteName('#__finder_links_terms' . $suffix) .
' (' . $db->quoteName('link_id') .
', ' . $db->quoteName('term_id') .
', ' . $db->quoteName('weight') . ')'
.
' SELECT ' . (int) $linkId . ', ' .
$db->quoteName('term_id') . ',' .
' ROUND(SUM(' . $db->quoteName('context_weight')
. '), 8)' .
' FROM ' .
$db->quoteName('#__finder_tokens_aggregate') .
' WHERE ' . $db->quoteName('map_suffix') .
' = ' . $db->quote($suffix) .
' GROUP BY term, term_id' .
' ORDER BY ' . $db->quoteName('term') . '
DESC'
);
$db->execute();
}
// Mark afterMapping in the profiler.
static::$profiler ? static::$profiler->mark('afterMapping')
: null;
// Update the signature.
$query->clear()
->update($db->quoteName('#__finder_links'))
->set($db->quoteName('md5sum') . ' = ' .
$db->quote($curSig))
->where($db->quoteName('link_id') . ' = ' .
$db->quote($linkId));
$db->setQuery($query);
$db->execute();
// Mark afterSigning in the profiler.
static::$profiler ? static::$profiler->mark('afterSigning')
: null;
// Truncate the tokens tables.
$db->truncateTable('#__finder_tokens');
// Truncate the tokens aggregate table.
$db->truncateTable('#__finder_tokens_aggregate');
// Toggle the token tables back to memory tables.
$this->toggleTables(true);
// Mark afterTruncating in the profiler.
static::$profiler ?
static::$profiler->mark('afterTruncating') : null;
return $linkId;
}
/**
* Method to remove a link from the index.
*
* @param integer $linkId The id of the link.
*
* @return boolean True on success.
*
* @since 3.1
* @throws Exception on database error.
*/
public function remove($linkId)
{
$db = $this->db;
$query = $db->getQuery(true);
// Update the link counts and remove the mapping records.
for ($i = 0; $i <= 15; $i++)
{
// Update the link counts for the terms.
$query->update('t')
->set('t.links = t.links - 1 from #__finder_terms AS t INNER
JOIN #__finder_links_terms' . dechex($i) . ' AS m ON m.term_id =
t.term_id')
->where('m.link_id = ' . $db->quote((int) $linkId));
$db->setQuery($query);
$db->execute();
// Remove all records from the mapping tables.
$query->clear()
->delete($db->quoteName('#__finder_links_terms' .
dechex($i)))
->where($db->quoteName('link_id') . ' = ' .
(int) $linkId);
$db->setQuery($query);
$db->execute();
}
// Delete all orphaned terms.
$query->clear()
->delete($db->quoteName('#__finder_terms'))
->where($db->quoteName('links') . ' <= 0');
$db->setQuery($query);
$db->execute();
// Delete the link from the index.
$query->clear()
->delete($db->quoteName('#__finder_links'))
->where($db->quoteName('link_id') . ' = ' .
$db->quote((int) $linkId));
$db->setQuery($query);
$db->execute();
// Remove the taxonomy maps.
FinderIndexerTaxonomy::removeMaps($linkId);
// Remove the orphaned taxonomy nodes.
FinderIndexerTaxonomy::removeOrphanNodes();
return true;
}
/**
* Method to optimize the index. We use this method to remove unused terms
* and any other optimizations that might be necessary.
*
* @return boolean True on success.
*
* @since 3.1
* @throws Exception on database error.
*/
public function optimize()
{
// Get the database object.
$db = $this->db;
$query = $db->getQuery(true);
// Delete all orphaned terms.
$query->delete($db->quoteName('#__finder_terms'))
->where($db->quoteName('links') . ' <= 0');
$db->setQuery($query);
$db->execute();
// Remove the orphaned taxonomy nodes.
FinderIndexerTaxonomy::removeOrphanNodes();
return true;
}
}
PK"�[:��̴8�8helpers/indexer/helper.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\Registry\Registry;
use Joomla\String\StringHelper;
JLoader::register('FinderIndexerParser', __DIR__ .
'/parser.php');
JLoader::register('FinderIndexerStemmer', __DIR__ .
'/stemmer.php');
JLoader::register('FinderIndexerToken', __DIR__ .
'/token.php');
/**
* Helper class for the Finder indexer package.
*
* @since 2.5
*/
class FinderIndexerHelper
{
/**
* The token stemmer object. The stemmer is set by whatever class
* wishes to use it but it must be an instance of FinderIndexerStemmer.
*
* @var FinderIndexerStemmer
* @since 2.5
*/
public static $stemmer;
/**
* A state flag, in order to not constantly check if the stemmer is an
instance of FinderIndexerStemmer
*
* @var boolean
* @since 3.7.0
*/
protected static $stemmerOK;
/**
* Method to parse input into plain text.
*
* @param string $input The raw input.
* @param string $format The format of the input. [optional]
*
* @return string The parsed input.
*
* @since 2.5
* @throws Exception on invalid parser.
*/
public static function parse($input, $format = 'html')
{
// Get a parser for the specified format and parse the input.
return FinderIndexerParser::getInstance($format)->parse($input);
}
/**
* Method to tokenize a text string.
*
* @param string $input The input to tokenize.
* @param string $lang The language of the input.
* @param boolean $phrase Flag to indicate whether input could be a
phrase. [optional]
*
* @return array|FinderIndexerToken An array of FinderIndexerToken
objects or a single FinderIndexerToken object.
*
* @since 2.5
*/
public static function tokenize($input, $lang, $phrase = false)
{
static $cache;
$store = StringHelper::strlen($input) < 128 ? md5($input .
'::' . $lang . '::' . $phrase) : null;
// Check if the string has been tokenized already.
if ($store && isset($cache[$store]))
{
return $cache[$store];
}
$tokens = array();
$quotes = html_entity_decode('‘’'',
ENT_QUOTES, 'UTF-8');
// Get the simple language key.
$lang = static::getPrimaryLanguage($lang);
/*
* Parsing the string input into terms is a multi-step process.
*
* Regexes:
* 1. Remove everything except letters, numbers, quotes, apostrophe,
plus, dash, period, and comma.
* 2. Remove plus, dash, period, and comma characters located before
letter characters.
* 3. Remove plus, dash, period, and comma characters located after
other characters.
* 4. Remove plus, period, and comma characters enclosed in alphabetical
characters. Ungreedy.
* 5. Remove orphaned apostrophe, plus, dash, period, and comma
characters.
* 6. Remove orphaned quote characters.
* 7. Replace the assorted single quotation marks with the ASCII
standard single quotation.
* 8. Remove multiple space characters and replaces with a single space.
*/
$input = StringHelper::strtolower($input);
$input =
preg_replace('#[^\pL\pM\pN\p{Pi}\p{Pf}\'+-.,]+#mui', '
', $input);
$input = preg_replace('#(^|\s)[+-.,]+([\pL\pM]+)#mui', '
$1', $input);
$input = preg_replace('#([\pL\pM\pN]+)[+-.,]+(\s|$)#mui',
'$1 ', $input);
$input = preg_replace('#([\pL\pM]+)[+.,]+([\pL\pM]+)#muiU',
'$1 $2', $input);
$input = preg_replace('#(^|\s)[\'+-.,]+(\s|$)#mui', '
', $input);
$input = preg_replace('#(^|\s)[\p{Pi}\p{Pf}]+(\s|$)#mui',
' ', $input);
$input = preg_replace('#[' . $quotes . ']+#mui',
'\'', $input);
$input = preg_replace('#\s+#mui', ' ', $input);
$input = trim($input);
// Explode the normalized string to get the terms.
$terms = explode(' ', $input);
/*
* If we have Unicode support and are dealing with Chinese text, Chinese
* has to be handled specially because there are not necessarily any
spaces
* between the "words". So, we have to test if the words belong
to the Chinese
* character set and if so, explode them into single glyphs or
"words".
*/
if ($lang === 'zh')
{
// Iterate through the terms and test if they contain Chinese.
for ($i = 0, $n = count($terms); $i < $n; $i++)
{
$charMatches = array();
$charCount = preg_match_all('#[\p{Han}]#mui', $terms[$i],
$charMatches);
// Split apart any groups of Chinese characters.
for ($j = 0; $j < $charCount; $j++)
{
$tSplit = StringHelper::str_ireplace($charMatches[0][$j],
'', $terms[$i], false);
if ((bool) $tSplit)
{
$terms[$i] = $tSplit;
}
else
{
unset($terms[$i]);
}
$terms[] = $charMatches[0][$j];
}
}
// Reset array keys.
$terms = array_values($terms);
}
/*
* If we have to handle the input as a phrase, that means we don't
* tokenize the individual terms and we do not create the two and three
* term combinations. The phrase must contain more than one word!
*/
if ($phrase === true && count($terms) > 1)
{
// Create tokens from the phrase.
$tokens[] = new FinderIndexerToken($terms, $lang);
}
else
{
// Create tokens from the terms.
for ($i = 0, $n = count($terms); $i < $n; $i++)
{
$tokens[] = new FinderIndexerToken($terms[$i], $lang);
}
// Create two and three word phrase tokens from the individual words.
for ($i = 0, $n = count($tokens); $i < $n; $i++)
{
// Setup the phrase positions.
$i2 = $i + 1;
$i3 = $i + 2;
// Create the two word phrase.
if ($i2 < $n && isset($tokens[$i2]))
{
// Tokenize the two word phrase.
$token = new FinderIndexerToken(
array(
$tokens[$i]->term,
$tokens[$i2]->term
), $lang, $lang === 'zh' ? '' : ' '
);
$token->derived = true;
// Add the token to the stack.
$tokens[] = $token;
}
// Create the three word phrase.
if ($i3 < $n && isset($tokens[$i3]))
{
// Tokenize the three word phrase.
$token = new FinderIndexerToken(
array(
$tokens[$i]->term,
$tokens[$i2]->term,
$tokens[$i3]->term
), $lang, $lang === 'zh' ? '' : ' '
);
$token->derived = true;
// Add the token to the stack.
$tokens[] = $token;
}
}
}
if ($store)
{
$cache[$store] = count($tokens) > 1 ? $tokens : array_shift($tokens);
return $cache[$store];
}
else
{
return count($tokens) > 1 ? $tokens : array_shift($tokens);
}
}
/**
* Method to get the base word of a token. This method uses the public
* {@link FinderIndexerHelper::$stemmer} object if it is set. If no
stemmer is set,
* the original token is returned.
*
* @param string $token The token to stem.
* @param string $lang The language of the token.
*
* @return string The root token.
*
* @since 2.5
*/
public static function stem($token, $lang)
{
// Trim apostrophes at either end of the token.
$token = trim($token, '\'');
// Trim everything after any apostrophe in the token.
if ($res = explode('\'', $token))
{
$token = $res[0];
}
if (static::$stemmerOK === true)
{
return static::$stemmer->stem($token, $lang);
}
else
{
// Stem the token if we have a valid stemmer to use.
if (static::$stemmer instanceof FinderIndexerStemmer)
{
static::$stemmerOK = true;
return static::$stemmer->stem($token, $lang);
}
}
return $token;
}
/**
* Method to add a content type to the database.
*
* @param string $title The type of content. For example: PDF
* @param string $mime The mime type of the content. For example: PDF
[optional]
*
* @return integer The id of the content type.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function addContentType($title, $mime = null)
{
static $types;
$db = JFactory::getDbo();
$query = $db->getQuery(true);
// Check if the types are loaded.
if (empty($types))
{
// Build the query to get the types.
$query->select('*')
->from($db->quoteName('#__finder_types'));
// Get the types.
$db->setQuery($query);
$types = $db->loadObjectList('title');
}
// Check if the type already exists.
if (isset($types[$title]))
{
return (int) $types[$title]->id;
}
// Add the type.
$query->clear()
->insert($db->quoteName('#__finder_types'))
->columns(array($db->quoteName('title'),
$db->quoteName('mime')))
->values($db->quote($title) . ', ' .
$db->quote($mime));
$db->setQuery($query);
$db->execute();
// Return the new id.
return (int) $db->insertid();
}
/**
* Method to check if a token is common in a language.
*
* @param string $token The token to test.
* @param string $lang The language to reference.
*
* @return boolean True if common, false otherwise.
*
* @since 2.5
*/
public static function isCommon($token, $lang)
{
static $data;
static $default;
$langCode = $lang;
// If language requested is wildcard, use the default language.
if ($default === null && $lang === '*')
{
$default = strstr(self::getDefaultLanguage(), '-', true);
$langCode = $default;
}
// Load the common tokens for the language if necessary.
if (!isset($data[$langCode]))
{
$data[$langCode] = self::getCommonWords($langCode);
}
// Check if the token is in the common array.
return in_array($token, $data[$langCode], true);
}
/**
* Method to get an array of common terms for a language.
*
* @param string $lang The language to use.
*
* @return array Array of common terms.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function getCommonWords($lang)
{
$db = JFactory::getDbo();
// Create the query to load all the common terms for the language.
$query = $db->getQuery(true)
->select($db->quoteName('term'))
->from($db->quoteName('#__finder_terms_common'))
->where($db->quoteName('language') . ' = ' .
$db->quote($lang));
// Load all of the common terms for the language.
$db->setQuery($query);
return $db->loadColumn();
}
/**
* Method to get the default language for the site.
*
* @return string The default language string.
*
* @since 2.5
*/
public static function getDefaultLanguage()
{
static $lang;
// We need to go to com_languages to get the site default language,
it's the best we can guess.
if (empty($lang))
{
$lang =
JComponentHelper::getParams('com_languages')->get('site',
'en-GB');
}
return $lang;
}
/**
* Method to parse a language/locale key and return a simple language
string.
*
* @param string $lang The language/locale key. For example: en-GB
*
* @return string The simple language string. For example: en
*
* @since 2.5
*/
public static function getPrimaryLanguage($lang)
{
static $data;
// Only parse the identifier if necessary.
if (!isset($data[$lang]))
{
if (is_callable(array('Locale',
'getPrimaryLanguage')))
{
// Get the language key using the Locale package.
$data[$lang] = Locale::getPrimaryLanguage($lang);
}
else
{
// Get the language key using string position.
$data[$lang] = StringHelper::substr($lang, 0,
StringHelper::strpos($lang, '-'));
}
}
return $data[$lang];
}
/**
* Method to get the path (SEF route) for a content item.
*
* @param string $url The non-SEF route to the content item.
*
* @return string The path for the content item.
*
* @since 2.5
* @deprecated 4.0
*/
public static function getContentPath($url)
{
static $router;
// Only get the router once.
if (!($router instanceof JRouter))
{
// Get and configure the site router.
$config = JFactory::getConfig();
$router = JRouter::getInstance('site');
$router->setMode($config->get('sef', 1));
}
// Build the relative route.
$uri = $router->build($url);
$route = $uri->toString(array('path', 'query',
'fragment'));
$route = str_replace(JUri::base(true) . '/', '',
$route);
return $route;
}
/**
* Method to get extra data for a content before being indexed. This is
how
* we add Comments, Tags, Labels, etc. that should be available to Finder.
*
* @param FinderIndexerResult $item The item to index as a
FinderIndexerResult object.
*
* @return boolean True on success, false on failure.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function getContentExtras(FinderIndexerResult $item)
{
// Get the event dispatcher.
$dispatcher = JEventDispatcher::getInstance();
// Load the finder plugin group.
JPluginHelper::importPlugin('finder');
// Trigger the event.
$results = $dispatcher->trigger('onPrepareFinderContent',
array(&$item));
// Check the returned results. This is for plugins that don't throw
// exceptions when they encounter serious errors.
if (in_array(false, $results))
{
throw new Exception($dispatcher->getError(), 500);
}
return true;
}
/**
* Method to process content text using the onContentPrepare event
trigger.
*
* @param string $text The content to process.
* @param Registry $params The parameters object.
[optional]
* @param FinderIndexerResult $item The item which get prepared.
[optional]
*
* @return string The processed content.
*
* @since 2.5
*/
public static function prepareContent($text, $params = null,
FinderIndexerResult $item = null)
{
static $loaded;
// Get the dispatcher.
$dispatcher = JEventDispatcher::getInstance();
// Load the content plugins if necessary.
if (empty($loaded))
{
JPluginHelper::importPlugin('content');
$loaded = true;
}
// Instantiate the parameter object if necessary.
if (!($params instanceof Registry))
{
$registry = new Registry($params);
$params = $registry;
}
// Create a mock content object.
$content = JTable::getInstance('Content');
$content->text = $text;
if ($item)
{
$content->bind((array) $item);
$content->bind($item->getElements());
}
if ($item && !empty($item->context))
{
$content->context = $item->context;
}
// Fire the onContentPrepare event.
$dispatcher->trigger('onContentPrepare',
array('com_finder.indexer', &$content, &$params, 0));
return $content->text;
}
}
PK"�[.!���:�:helpers/indexer/indexer.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\String\StringHelper;
JLoader::register('FinderIndexerHelper', __DIR__ .
'/helper.php');
JLoader::register('FinderIndexerParser', __DIR__ .
'/parser.php');
JLoader::register('FinderIndexerStemmer', __DIR__ .
'/stemmer.php');
JLoader::register('FinderIndexerTaxonomy', __DIR__ .
'/taxonomy.php');
JLoader::register('FinderIndexerToken', __DIR__ .
'/token.php');
jimport('joomla.filesystem.file');
/**
* Main indexer class for the Finder indexer package.
*
* The indexer class provides the core functionality of the Finder
* search engine. It is responsible for adding and updating the
* content links table; extracting and scoring tokens; and maintaining
* all referential information for the content.
*
* Note: All exceptions thrown from within this class should be caught
* by the controller.
*
* @since 2.5
*/
abstract class FinderIndexer
{
/**
* The title context identifier.
*
* @var integer
* @since 2.5
*/
const TITLE_CONTEXT = 1;
/**
* The text context identifier.
*
* @var integer
* @since 2.5
*/
const TEXT_CONTEXT = 2;
/**
* The meta context identifier.
*
* @var integer
* @since 2.5
*/
const META_CONTEXT = 3;
/**
* The path context identifier.
*
* @var integer
* @since 2.5
*/
const PATH_CONTEXT = 4;
/**
* The misc context identifier.
*
* @var integer
* @since 2.5
*/
const MISC_CONTEXT = 5;
/**
* The indexer state object.
*
* @var JObject
* @since 2.5
*/
public static $state;
/**
* The indexer profiler object.
*
* @var JProfiler
* @since 2.5
*/
public static $profiler;
/**
* Database driver cache.
*
* @var JDatabaseDriver
* @since 3.8.0
*/
protected $db;
/**
* Reusable Query Template. To be used with clone.
*
* @var JDatabaseQuery
* @since 3.8.0
*/
protected $addTokensToDbQueryTemplate;
/**
* FinderIndexer constructor.
*
* @since 3.8.0
*/
public function __construct()
{
$this->db = JFactory::getDbo();
$db = $this->db;
/**
* Set up query template for addTokensToDb, we will be cloning this
template when needed.
* This is about twice as fast as calling the clear function or setting
up a new object.
*/
$this->addTokensToDbQueryTemplate =
$db->getQuery(true)->insert($db->quoteName('#__finder_tokens'))
->columns(
array(
$db->quoteName('term'),
$db->quoteName('stem'),
$db->quoteName('common'),
$db->quoteName('phrase'),
$db->quoteName('weight'),
$db->quoteName('context'),
$db->quoteName('language')
)
);
}
/**
* Returns a reference to the FinderIndexer object.
*
* @return FinderIndexer instance based on the database driver
*
* @since 3.0
* @throws RuntimeException if driver class for indexer not present.
*/
public static function getInstance()
{
// Setup the adapter for the indexer.
$serverType = JFactory::getDbo()->getServerType();
// For `mssql` server types, convert the type to `sqlsrv`
if ($serverType === 'mssql')
{
$serverType = 'sqlsrv';
}
$path = __DIR__ . '/driver/' . $serverType . '.php';
$class = 'FinderIndexerDriver' . ucfirst($serverType);
// Check if a parser exists for the format.
if (file_exists($path))
{
// Instantiate the parser.
JLoader::register($class, $path);
return new $class;
}
// Throw invalid format exception.
throw new
RuntimeException(JText::sprintf('COM_FINDER_INDEXER_INVALID_DRIVER',
$serverType));
}
/**
* Method to get the indexer state.
*
* @return object The indexer state object.
*
* @since 2.5
*/
public static function getState()
{
// First, try to load from the internal state.
if ((bool) static::$state)
{
return static::$state;
}
// If we couldn't load from the internal state, try the session.
$session = JFactory::getSession();
$data = $session->get('_finder.state', null);
// If the state is empty, load the values for the first time.
if (empty($data))
{
$data = new JObject;
// Load the default configuration options.
$data->options = JComponentHelper::getParams('com_finder');
// Setup the weight lookup information.
$data->weights = array(
self::TITLE_CONTEXT =>
round($data->options->get('title_multiplier', 1.7), 2),
self::TEXT_CONTEXT =>
round($data->options->get('text_multiplier', 0.7), 2),
self::META_CONTEXT =>
round($data->options->get('meta_multiplier', 1.2), 2),
self::PATH_CONTEXT =>
round($data->options->get('path_multiplier', 2.0), 2),
self::MISC_CONTEXT =>
round($data->options->get('misc_multiplier', 0.3), 2)
);
// Set the current time as the start time.
$data->startTime = JFactory::getDate()->toSql();
// Set the remaining default values.
$data->batchSize = (int)
$data->options->get('batch_size', 50);
$data->batchOffset = 0;
$data->totalItems = 0;
$data->pluginState = array();
}
// Setup the profiler if debugging is enabled.
if (JFactory::getApplication()->get('debug'))
{
static::$profiler = JProfiler::getInstance('FinderIndexer');
}
// Setup the stemmer.
if ($data->options->get('stem', 1) &&
$data->options->get('stemmer', 'porter_en'))
{
FinderIndexerHelper::$stemmer =
FinderIndexerStemmer::getInstance($data->options->get('stemmer',
'porter_en'));
}
// Set the state.
static::$state = $data;
return static::$state;
}
/**
* Method to set the indexer state.
*
* @param object $data A new indexer state object.
*
* @return boolean True on success, false on failure.
*
* @since 2.5
*/
public static function setState($data)
{
// Check the state object.
if (empty($data) || !$data instanceof JObject)
{
return false;
}
// Set the new internal state.
static::$state = $data;
// Set the new session state.
JFactory::getSession()->set('_finder.state', $data);
return true;
}
/**
* Method to reset the indexer state.
*
* @return void
*
* @since 2.5
*/
public static function resetState()
{
// Reset the internal state to null.
self::$state = null;
// Reset the session state to null.
JFactory::getSession()->set('_finder.state', null);
}
/**
* Method to index a content item.
*
* @param FinderIndexerResult $item The content item to index.
* @param string $format The format of the content.
[optional]
*
* @return integer The ID of the record in the links table.
*
* @since 2.5
* @throws Exception on database error.
*/
abstract public function index($item, $format = 'html');
/**
* Method to remove a link from the index.
*
* @param integer $linkId The id of the link.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
public function remove($linkId)
{
$db = $this->db;
$query = $db->getQuery(true);
// Update the link counts and remove the mapping records.
for ($i = 0; $i <= 15; $i++)
{
// Update the link counts for the terms.
$query->clear()
->update($db->quoteName('#__finder_terms',
't'))
->join('INNER',
$db->quoteName('#__finder_links_terms' . dechex($i),
'm') .
' ON ' . $db->quoteName('m.term_id') . ' =
' . $db->quoteName('t.term_id')
)
->set($db->quoteName('links') . ' = ' .
$db->quoteName('links') . ' - 1')
->where($db->quoteName('m.link_id') . ' = ' .
(int) $linkId);
$db->setQuery($query)->execute();
// Remove all records from the mapping tables.
$query->clear()
->delete($db->quoteName('#__finder_links_terms' .
dechex($i)))
->where($db->quoteName('link_id') . ' = ' .
(int) $linkId);
$db->setQuery($query)->execute();
}
// Delete all orphaned terms.
$query->clear()
->delete($db->quoteName('#__finder_terms'))
->where($db->quoteName('links') . ' <= 0');
$db->setQuery($query)->execute();
// Delete the link from the index.
$query->clear()
->delete($db->quoteName('#__finder_links'))
->where($db->quoteName('link_id') . ' = ' .
(int) $linkId);
$db->setQuery($query)->execute();
// Remove the taxonomy maps.
FinderIndexerTaxonomy::removeMaps($linkId);
// Remove the orphaned taxonomy nodes.
FinderIndexerTaxonomy::removeOrphanNodes();
return true;
}
/**
* Method to optimize the index. We use this method to remove unused terms
* and any other optimizations that might be necessary.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
abstract public function optimize();
/**
* Method to get a content item's signature.
*
* @param object $item The content item to index.
*
* @return string The content item's signature.
*
* @since 2.5
*/
protected static function getSignature($item)
{
// Get the indexer state.
$state = static::getState();
// Get the relevant configuration variables.
$config = array(
$state->weights,
$state->options->get('stem', 1),
$state->options->get('stemmer', 'porter_en')
);
return md5(serialize(array($item, $config)));
}
/**
* Method to parse input, tokenize it, and then add it to the database.
*
* @param mixed $input String or resource to use as input. A
resource input will automatically be chunked to conserve
* memory. Strings will be chunked if longer
than 2K in size.
* @param integer $context The context of the input. See context
constants.
* @param string $lang The language of the input.
* @param string $format The format of the input.
*
* @return integer The number of tokens extracted from the input.
*
* @since 2.5
*/
protected function tokenizeToDb($input, $context, $lang, $format)
{
$count = 0;
$buffer = null;
if (empty($input))
{
return $count;
}
// If the input is a resource, batch the process out.
if (is_resource($input))
{
// Batch the process out to avoid memory limits.
while (!feof($input))
{
// Read into the buffer.
$buffer .= fread($input, 2048);
/*
* If we haven't reached the end of the file, seek to the last
* space character and drop whatever is after that to make sure
* we didn't truncate a term while reading the input.
*/
if (!feof($input))
{
// Find the last space character.
$ls = strrpos($buffer, ' ');
// Adjust string based on the last space character.
if ($ls)
{
// Truncate the string to the last space character.
$string = substr($buffer, 0, $ls);
// Adjust the buffer based on the last space for the next iteration
and trim.
$buffer = StringHelper::trim(substr($buffer, $ls));
}
// No space character was found.
else
{
$string = $buffer;
}
}
// We've reached the end of the file, so parse whatever remains.
else
{
$string = $buffer;
}
// Parse, tokenise and add tokens to the database.
$count = $this->tokenizeToDbShort($string, $context, $lang, $format,
$count);
unset($string, $tokens);
}
return $count;
}
// Parse, tokenise and add tokens to the database.
$count = $this->tokenizeToDbShort($input, $context, $lang, $format,
$count);
return $count;
}
/**
* Method to parse input, tokenise it, then add the tokens to the
database.
*
* @param string $input String to parse, tokenise and add to
database.
* @param integer $context The context of the input. See context
constants.
* @param string $lang The language of the input.
* @param string $format The format of the input.
* @param integer $count The number of tokens processed so far.
*
* @return integer Cumulative number of tokens extracted from the input
so far.
*
* @since 3.7.0
*/
private function tokenizeToDbShort($input, $context, $lang, $format,
$count)
{
// Parse the input.
$input = FinderIndexerHelper::parse($input, $format);
// Check the input.
if (empty($input))
{
return $count;
}
// Tokenize the input.
$tokens = FinderIndexerHelper::tokenize($input, $lang);
// Add the tokens to the database.
$count += $this->addTokensToDb($tokens, $context);
// Check if we're approaching the memory limit of the token table.
if ($count >
static::$state->options->get('memory_table_limit', 30000))
{
$this->toggleTables(false);
}
return $count;
}
/**
* Method to add a set of tokens to the database.
*
* @param mixed $tokens An array or single FinderIndexerToken object.
* @param mixed $context The context of the tokens. See context
constants. [optional]
*
* @return integer The number of tokens inserted into the database.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function addTokensToDb($tokens, $context = '')
{
// Get the database object.
$db = $this->db;
// Count the number of token values.
$values = 0;
if (($tokens instanceof FinderIndexerToken) === false)
{
// Break into chunks of no more than 1000 items
$chunks = count($tokens) > 1000
? array_chunk($tokens, 1000)
: array($tokens);
foreach ($chunks as $chunkTokens)
{
// Cloning a new query template is twice as fast as calling the clear
function
$query = clone $this->addTokensToDbQueryTemplate;
// Iterate through the tokens to create SQL value sets.
foreach ($chunkTokens as $token)
{
$query->values(
$db->quote($token->term) . ', '
. $db->quote($token->stem) . ', '
. (int) $token->common . ', '
. (int) $token->phrase . ', '
. $db->escape((float) $token->weight) . ', '
. (int) $context . ', '
. $db->quote($token->language)
);
++$values;
}
$db->setQuery($query)->execute();
}
}
else
{
$query = clone $this->addTokensToDbQueryTemplate;
$query->values(
$db->quote($tokens->term) . ', '
. $db->quote($tokens->stem) . ', '
. (int) $tokens->common . ', '
. (int) $tokens->phrase . ', '
. $db->escape((float) $tokens->weight) . ', '
. (int) $context . ', '
. $db->quote($tokens->language)
);
++$values;
$db->setQuery($query)->execute();
}
return $values;
}
/**
* Method to switch the token tables from Memory tables to Disk tables
* when they are close to running out of memory.
* Since this is not supported/implemented in all DB-drivers, the default
is a stub method, which simply returns true.
*
* @param boolean $memory Flag to control how they should be toggled.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function toggleTables($memory)
{
return true;
}
}
PK"�[�:�{{helpers/indexer/parser/html.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderIndexerParser', dirname(__DIR__) .
'/parser.php');
/**
* HTML Parser class for the Finder indexer package.
*
* @since 2.5
*/
class FinderIndexerParserHtml extends FinderIndexerParser
{
/**
* Method to parse input and extract the plain text. Because this method
is
* called from both inside and outside the indexer, it needs to be able to
* batch out its parsing functionality to deal with the inefficiencies of
* regular expressions. We will parse recursively in 2KB chunks.
*
* @param string $input The input to parse.
*
* @return string The plain text input.
*
* @since 2.5
*/
public function parse($input)
{
// Strip invalid UTF-8 characters.
$input = iconv('utf-8', 'utf-8//IGNORE', $input);
// Remove anything between <head> and </head> tags. Do this
first
// because there might be <script> or <style> tags nested
inside.
$input = $this->removeBlocks($input, '<head>',
'</head>');
// Convert <style> and <noscript> tags to <script> tags
// so we can remove them efficiently.
$search = array(
'<style', '</style',
'<noscript', '</noscript',
);
$replace = array(
'<script', '</script',
'<script', '</script',
);
$input = str_replace($search, $replace, $input);
// Strip all script blocks.
$input = $this->removeBlocks($input, '<script',
'</script>');
// Decode HTML entities.
$input = html_entity_decode($input, ENT_QUOTES, 'UTF-8');
// Convert entities equivalent to spaces to actual spaces.
$input = str_replace(array(' ',
' '), ' ', $input);
// Add a space before both the OPEN and CLOSE tags of BLOCK and LINE
BREAKING elements,
// e.g. 'all<h1><em>m</em>obile
List</h1>' will become 'all mobile List'
$input = preg_replace('/(<|<\/)(' .
'address|article|aside|blockquote|br|canvas|dd|div|dl|dt|' .
'fieldset|figcaption|figure|footer|form|h1|h2|h3|h4|h5|h6|header|hgroup|hr|li|'
.
'main|nav|noscript|ol|output|p|pre|section|table|tfoot|ul|video'
.
')\b/i', ' $1$2', $input
);
// Strip HTML tags.
$input = strip_tags($input);
return parent::parse($input);
}
/**
* Method to process HTML input and extract the plain text.
*
* @param string $input The input to process.
*
* @return string The plain text input.
*
* @since 2.5
*/
protected function process($input)
{
// Replace any amount of white space with a single space.
return preg_replace('#\s+#u', ' ', $input);
}
/**
* Method to remove blocks of text between a start and an end tag.
* Each block removed is effectively replaced by a single space.
*
* Note: The start tag and the end tag must be different.
* Note: Blocks must not be nested.
* Note: This method will function correctly with multi-byte strings.
*
* @param string $input String to be processed.
* @param string $startTag String representing the start tag.
* @param string $endTag String representing the end tag.
*
* @return string with blocks removed.
*
* @since 3.4
*/
private function removeBlocks($input, $startTag, $endTag)
{
$return = '';
$offset = 0;
$startTagLength = strlen($startTag);
$endTagLength = strlen($endTag);
// Find the first start tag.
$start = stripos($input, $startTag);
// If no start tags were found, return the string unchanged.
if ($start === false)
{
return $input;
}
// Look for all blocks defined by the start and end tags.
while ($start !== false)
{
// Accumulate the substring up to the start tag.
$return .= substr($input, $offset, $start - $offset) . ' ';
// Look for an end tag corresponding to the start tag.
$end = stripos($input, $endTag, $start + $startTagLength);
// If no corresponding end tag, leave the string alone.
if ($end === false)
{
// Fix the offset so part of the string is not duplicated.
$offset = $start;
break;
}
// Advance the start position.
$offset = $end + $endTagLength;
// Look for the next start tag and loop.
$start = stripos($input, $startTag, $offset);
}
// Add in the final substring after the last end tag.
$return .= substr($input, $offset);
return $return;
}
}
PK"�[�>LO--helpers/indexer/parser/rtf.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderIndexerParser', dirname(__DIR__) .
'/parser.php');
/**
* RTF Parser class for the Finder indexer package.
*
* @since 2.5
*/
class FinderIndexerParserRtf extends FinderIndexerParser
{
/**
* Method to process RTF input and extract the plain text.
*
* @param string $input The input to process.
*
* @return string The plain text input.
*
* @since 2.5
*/
protected function process($input)
{
// Remove embedded pictures.
$input = preg_replace('#{\\\pict[^}]*}#mi', '',
$input);
// Remove control characters.
$input = str_replace(array('{', '}',
"\\\n"), array(' ', ' ', "\n"),
$input);
$input = preg_replace('#\\\([^;]+?);#m', ' ',
$input);
$input = preg_replace('#\\\[\'a-zA-Z0-9]+#mi', '
', $input);
return $input;
}
}
PK"�[�/AB��helpers/indexer/parser/txt.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderIndexerParser', dirname(__DIR__) .
'/parser.php');
/**
* Text Parser class for the Finder indexer package.
*
* @since 2.5
*/
class FinderIndexerParserTxt extends FinderIndexerParser
{
/**
* Method to process Text input and extract the plain text.
*
* @param string $input The input to process.
*
* @return string The plain text input.
*
* @since 2.5
*/
protected function process($input)
{
return $input;
}
}
PK"�[1
�G��helpers/indexer/parser.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Parser base class for the Finder indexer package.
*
* @since 2.5
*/
abstract class FinderIndexerParser
{
/**
* Method to get a parser, creating it if necessary.
*
* @param string $format The type of parser to load.
*
* @return FinderIndexerParser A FinderIndexerParser instance.
*
* @since 2.5
* @throws Exception on invalid parser.
*/
public static function getInstance($format)
{
static $instances;
// Only create one parser for each format.
if (isset($instances[$format]))
{
return $instances[$format];
}
// Create an array of instances if necessary.
if (!is_array($instances))
{
$instances = array();
}
// Setup the adapter for the parser.
$format = JFilterInput::getInstance()->clean($format,
'cmd');
$path = __DIR__ . '/parser/' . $format . '.php';
$class = 'FinderIndexerParser' . ucfirst($format);
// Check if a parser exists for the format.
if (!file_exists($path))
{
// Throw invalid format exception.
throw new
Exception(JText::sprintf('COM_FINDER_INDEXER_INVALID_PARSER',
$format));
}
// Instantiate the parser.
JLoader::register($class, $path);
$instances[$format] = new $class;
return $instances[$format];
}
/**
* Method to parse input and extract the plain text. Because this method
is
* called from both inside and outside the indexer, it needs to be able to
* batch out its parsing functionality to deal with the inefficiencies of
* regular expressions. We will parse recursively in 2KB chunks.
*
* @param string $input The input to parse.
*
* @return string The plain text input.
*
* @since 2.5
*/
public function parse($input)
{
// If the input is less than 2KB we can parse it in one go.
if (strlen($input) <= 2048)
{
return $this->process($input);
}
// Input is longer than 2Kb so parse it in chunks of 2Kb or less.
$start = 0;
$end = strlen($input);
$chunk = 2048;
$return = null;
while ($start < $end)
{
// Setup the string.
$string = substr($input, $start, $chunk);
// Find the last space character if we aren't at the end.
$ls = (($start + $chunk) < $end ? strrpos($string, ' ') :
false);
// Truncate to the last space character.
if ($ls !== false)
{
$string = substr($string, 0, $ls);
}
// Adjust the start position for the next iteration.
$start += ($ls !== false ? ($ls + 1 - $chunk) + $chunk : $chunk);
// Parse the chunk.
$return .= $this->process($string);
}
return $return;
}
/**
* Method to process input and extract the plain text.
*
* @param string $input The input to process.
*
* @return string The plain text input.
*
* @since 2.5
*/
abstract protected function process($input);
}
PK"�[��,̋̋helpers/indexer/query.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\Registry\Registry;
use Joomla\String\StringHelper;
use Joomla\Utilities\ArrayHelper;
JLoader::register('FinderIndexerHelper', __DIR__ .
'/helper.php');
JLoader::register('FinderIndexerTaxonomy', __DIR__ .
'/taxonomy.php');
JLoader::register('FinderHelperRoute', JPATH_SITE .
'/components/com_finder/helpers/route.php');
JLoader::register('FinderHelperLanguage', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/language.php');
/**
* Query class for the Finder indexer package.
*
* @since 2.5
*/
class FinderIndexerQuery
{
/**
* Flag to show whether the query can return results.
*
* @var boolean
* @since 2.5
*/
public $search;
/**
* The query input string.
*
* @var string
* @since 2.5
*/
public $input;
/**
* The language of the query.
*
* @var string
* @since 2.5
*/
public $language;
/**
* The query string matching mode.
*
* @var string
* @since 2.5
*/
public $mode;
/**
* The included tokens.
*
* @var array
* @since 2.5
*/
public $included = array();
/**
* The excluded tokens.
*
* @var array
* @since 2.5
*/
public $excluded = array();
/**
* The tokens to ignore because no matches exist.
*
* @var array
* @since 2.5
*/
public $ignored = array();
/**
* The operators used in the query input string.
*
* @var array
* @since 2.5
*/
public $operators = array();
/**
* The terms to highlight as matches.
*
* @var array
* @since 2.5
*/
public $highlight = array();
/**
* The number of matching terms for the query input.
*
* @var integer
* @since 2.5
*/
public $terms;
/**
* The static filter id.
*
* @var string
* @since 2.5
*/
public $filter;
/**
* The taxonomy filters. This is a multi-dimensional array of taxonomy
* branches as the first level and then the taxonomy nodes as the values.
*
* For example:
* $filters = array(
* 'Type' = array(10, 32, 29, 11, ...);
* 'Label' = array(20, 314, 349, 91, 82, ...);
* ...
* );
*
* @var array
* @since 2.5
*/
public $filters = array();
/**
* The start date filter.
*
* @var string
* @since 2.5
*/
public $date1;
/**
* The end date filter.
*
* @var string
* @since 2.5
*/
public $date2;
/**
* The start date filter modifier.
*
* @var string
* @since 2.5
*/
public $when1;
/**
* The end date filter modifier.
*
* @var string
* @since 2.5
*/
public $when2;
/**
* Method to instantiate the query object.
*
* @param array $options An array of query options.
*
* @since 2.5
* @throws Exception on database error.
*/
public function __construct($options)
{
// Get the input string.
$this->input = isset($options['input']) ?
$options['input'] : null;
// Get the empty query setting.
$this->empty = isset($options['empty']) ? (bool)
$options['empty'] : false;
// Get the input language.
$this->language = !empty($options['language']) ?
$options['language'] : FinderIndexerHelper::getDefaultLanguage();
$this->language =
FinderIndexerHelper::getPrimaryLanguage($this->language);
// Get the matching mode.
$this->mode = 'AND';
// Initialize the temporary date storage.
$this->dates = new Registry;
// Populate the temporary date storage.
if (!empty($options['date1']))
{
$this->dates->set('date1', $options['date1']);
}
if (!empty($options['date2']))
{
$this->dates->set('date2', $options['date2']);
}
if (!empty($options['when1']))
{
$this->dates->set('when1', $options['when1']);
}
if (!empty($options['when2']))
{
$this->dates->set('when2', $options['when2']);
}
// Process the static taxonomy filters.
if (!empty($options['filter']))
{
$this->processStaticTaxonomy($options['filter']);
}
// Process the dynamic taxonomy filters.
if (!empty($options['filters']))
{
$this->processDynamicTaxonomy($options['filters']);
}
// Get the date filters.
$d1 = $this->dates->get('date1');
$d2 = $this->dates->get('date2');
$w1 = $this->dates->get('when1');
$w2 = $this->dates->get('when2');
// Process the date filters.
if (!empty($d1) || !empty($d2))
{
$this->processDates($d1, $d2, $w1, $w2);
}
// Process the input string.
$this->processString($this->input, $this->language,
$this->mode);
// Get the number of matching terms.
foreach ($this->included as $token)
{
$this->terms += count($token->matches);
}
// Remove the temporary date storage.
unset($this->dates);
// Lastly, determine whether this query can return a result set.
// Check if we have a query string.
if (!empty($this->input))
{
$this->search = true;
}
// Check if we can search without a query string.
elseif ($this->empty && (!empty($this->filter) ||
!empty($this->filters) || !empty($this->date1) ||
!empty($this->date2)))
{
$this->search = true;
}
// We do not have a valid search query.
else
{
$this->search = false;
}
}
/**
* Method to convert the query object into a URI string.
*
* @param string $base The base URI. [optional]
*
* @return string The complete query URI.
*
* @since 2.5
*/
public function toUri($base = '')
{
// Set the base if not specified.
if ($base === '')
{
$base = 'index.php?option=com_finder&view=search';
}
// Get the base URI.
$uri = JUri::getInstance($base);
// Add the static taxonomy filter if present.
if ((bool) $this->filter)
{
$uri->setVar('f', $this->filter);
}
// Get the filters in the request.
$t =
JFactory::getApplication()->input->request->get('t',
array(), 'array');
// Add the dynamic taxonomy filters if present.
if ((bool) $this->filters)
{
foreach ($this->filters as $nodes)
{
foreach ($nodes as $node)
{
if (!in_array($node, $t))
{
continue;
}
$uri->setVar('t[]', $node);
}
}
}
// Add the input string if present.
if (!empty($this->input))
{
$uri->setVar('q', $this->input);
}
// Add the start date if present.
if (!empty($this->date1))
{
$uri->setVar('d1', $this->date1);
}
// Add the end date if present.
if (!empty($this->date2))
{
$uri->setVar('d2', $this->date2);
}
// Add the start date modifier if present.
if (!empty($this->when1))
{
$uri->setVar('w1', $this->when1);
}
// Add the end date modifier if present.
if (!empty($this->when2))
{
$uri->setVar('w2', $this->when2);
}
// Add a menu item id if one is not present.
if (!$uri->getVar('Itemid'))
{
// Get the menu item id.
$query = array(
'view' => $uri->getVar('view'),
'f' => $uri->getVar('f'),
'q' => $uri->getVar('q'),
);
$item = FinderHelperRoute::getItemid($query);
// Add the menu item id if present.
if ($item !== null)
{
$uri->setVar('Itemid', $item);
}
}
return $uri->toString(array('path', 'query'));
}
/**
* Method to get a list of excluded search term ids.
*
* @return array An array of excluded term ids.
*
* @since 2.5
*/
public function getExcludedTermIds()
{
$results = array();
// Iterate through the excluded tokens and compile the matching terms.
for ($i = 0, $c = count($this->excluded); $i < $c; $i++)
{
$results = array_merge($results, $this->excluded[$i]->matches);
}
// Sanitize the terms.
$results = array_unique($results);
return ArrayHelper::toInteger($results);
}
/**
* Method to get a list of included search term ids.
*
* @return array An array of included term ids.
*
* @since 2.5
*/
public function getIncludedTermIds()
{
$results = array();
// Iterate through the included tokens and compile the matching terms.
for ($i = 0, $c = count($this->included); $i < $c; $i++)
{
// Check if we have any terms.
if (empty($this->included[$i]->matches))
{
continue;
}
// Get the term.
$term = $this->included[$i]->term;
// Prepare the container for the term if necessary.
if (!array_key_exists($term, $results))
{
$results[$term] = array();
}
// Add the matches to the stack.
$results[$term] = array_merge($results[$term],
$this->included[$i]->matches);
}
// Sanitize the terms.
foreach ($results as $key => $value)
{
$results[$key] = array_unique($results[$key]);
$results[$key] = ArrayHelper::toInteger($results[$key]);
}
return $results;
}
/**
* Method to get a list of required search term ids.
*
* @return array An array of required term ids.
*
* @since 2.5
*/
public function getRequiredTermIds()
{
$results = array();
// Iterate through the included tokens and compile the matching terms.
for ($i = 0, $c = count($this->included); $i < $c; $i++)
{
// Check if the token is required.
if ($this->included[$i]->required)
{
// Get the term.
$term = $this->included[$i]->term;
// Prepare the container for the term if necessary.
if (!array_key_exists($term, $results))
{
$results[$term] = array();
}
// Add the matches to the stack.
$results[$term] = array_merge($results[$term],
$this->included[$i]->matches);
}
}
// Sanitize the terms.
foreach ($results as $key => $value)
{
$results[$key] = array_unique($results[$key]);
$results[$key] = ArrayHelper::toInteger($results[$key]);
}
return $results;
}
/**
* Method to process the static taxonomy input. The static taxonomy input
* comes in the form of a pre-defined search filter that is assigned to
the
* search form.
*
* @param integer $filterId The id of static filter.
*
* @return boolean True on success, false on failure.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function processStaticTaxonomy($filterId)
{
// Get the database object.
$db = JFactory::getDbo();
// Initialize user variables
$groups = implode(',',
JFactory::getUser()->getAuthorisedViewLevels());
// Load the predefined filter.
$query = $db->getQuery(true)
->select('f.data, f.params')
->from($db->quoteName('#__finder_filters') . ' AS
f')
->where('f.filter_id = ' . (int) $filterId);
$db->setQuery($query);
$return = $db->loadObject();
// Check the returned filter.
if (empty($return))
{
return false;
}
// Set the filter.
$this->filter = (int) $filterId;
// Get a parameter object for the filter date options.
$registry = new Registry($return->params);
$params = $registry;
// Set the dates if not already set.
$this->dates->def('d1', $params->get('d1'));
$this->dates->def('d2', $params->get('d2'));
$this->dates->def('w1', $params->get('w1'));
$this->dates->def('w2', $params->get('w2'));
// Remove duplicates and sanitize.
$filters = explode(',', $return->data);
$filters = array_unique($filters);
$filters = ArrayHelper::toInteger($filters);
// Remove any values of zero.
if (in_array(0, $filters, true) !== false)
{
unset($filters[array_search(0, $filters, true)]);
}
// Check if we have any real input.
if (empty($filters))
{
return true;
}
/*
* Create the query to get filters from the database. We do this for
* two reasons: one, it allows us to ensure that the filters being used
* are real; two, we need to sort the filters by taxonomy branch.
*/
$query->clear()
->select('t1.id, t1.title, t2.title AS branch')
->from($db->quoteName('#__finder_taxonomy') . ' AS
t1')
->join('INNER',
$db->quoteName('#__finder_taxonomy') . ' AS t2 ON t2.id =
t1.parent_id')
->where('t1.state = 1')
->where('t1.access IN (' . $groups . ')')
->where('t1.id IN (' . implode(',', $filters) .
')')
->where('t2.state = 1')
->where('t2.access IN (' . $groups . ')');
// Load the filters.
$db->setQuery($query);
$results = $db->loadObjectList();
// Sort the filter ids by branch.
foreach ($results as $result)
{
$this->filters[$result->branch][$result->title] = (int)
$result->id;
}
return true;
}
/**
* Method to process the dynamic taxonomy input. The dynamic taxonomy
input
* comes in the form of select fields that the user chooses from. The
* dynamic taxonomy input is processed AFTER the static taxonomy input
* because the dynamic options can be used to further narrow a static
* taxonomy filter.
*
* @param array $filters An array of taxonomy node ids.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function processDynamicTaxonomy($filters)
{
// Initialize user variables
$groups = implode(',',
JFactory::getUser()->getAuthorisedViewLevels());
// Remove duplicates and sanitize.
$filters = array_unique($filters);
$filters = ArrayHelper::toInteger($filters);
// Remove any values of zero.
if (in_array(0, $filters, true) !== false)
{
unset($filters[array_search(0, $filters, true)]);
}
// Check if we have any real input.
if (empty($filters))
{
return true;
}
// Get the database object.
$db = JFactory::getDbo();
$query = $db->getQuery(true);
/*
* Create the query to get filters from the database. We do this for
* two reasons: one, it allows us to ensure that the filters being used
* are real; two, we need to sort the filters by taxonomy branch.
*/
$query->select('t1.id, t1.title, t2.title AS branch')
->from($db->quoteName('#__finder_taxonomy') . ' AS
t1')
->join('INNER',
$db->quoteName('#__finder_taxonomy') . ' AS t2 ON t2.id =
t1.parent_id')
->where('t1.state = 1')
->where('t1.access IN (' . $groups . ')')
->where('t1.id IN (' . implode(',', $filters) .
')')
->where('t2.state = 1')
->where('t2.access IN (' . $groups . ')');
// Load the filters.
$db->setQuery($query);
$results = $db->loadObjectList();
// Cleared filter branches.
$cleared = array();
/*
* Sort the filter ids by branch. Because these filters are designed to
* override and further narrow the items selected in the static filter,
* we will clear the values from the static filter on a branch by
* branch basis before adding the dynamic filters. So, if the static
* filter defines a type filter of "articles" and three
"category"
* filters but the user only limits the category further, the category
* filters will be flushed but the type filters will not.
*/
foreach ($results as $result)
{
// Check if the branch has been cleared.
if (!in_array($result->branch, $cleared, true))
{
// Clear the branch.
$this->filters[$result->branch] = array();
// Add the branch to the cleared list.
$cleared[] = $result->branch;
}
// Add the filter to the list.
$this->filters[$result->branch][$result->title] = (int)
$result->id;
}
return true;
}
/**
* Method to process the query date filters to determine start and end
* date limitations.
*
* @param string $date1 The first date filter.
* @param string $date2 The second date filter.
* @param string $when1 The first date modifier.
* @param string $when2 The second date modifier.
*
* @return boolean True on success.
*
* @since 2.5
*/
protected function processDates($date1, $date2, $when1, $when2)
{
// Clean up the inputs.
$date1 = trim(StringHelper::strtolower($date1));
$date2 = trim(StringHelper::strtolower($date2));
$when1 = trim(StringHelper::strtolower($when1));
$when2 = trim(StringHelper::strtolower($when2));
// Get the time offset.
$offset = JFactory::getApplication()->get('offset');
// Array of allowed when values.
$whens = array('before', 'after', 'exact');
// The value of 'today' is a special case that we need to
handle.
if ($date1 ===
StringHelper::strtolower(JText::_('COM_FINDER_QUERY_FILTER_TODAY')))
{
$date1 = JFactory::getDate('now',
$offset)->format('%Y-%m-%d');
}
// Try to parse the date string.
$date = JFactory::getDate($date1, $offset);
// Check if the date was parsed successfully.
if ($date->toUnix() !== null)
{
// Set the date filter.
$this->date1 = $date->toSql();
$this->when1 = in_array($when1, $whens, true) ? $when1 :
'before';
}
// The value of 'today' is a special case that we need to
handle.
if ($date2 ===
StringHelper::strtolower(JText::_('COM_FINDER_QUERY_FILTER_TODAY')))
{
$date2 = JFactory::getDate('now',
$offset)->format('%Y-%m-%d');
}
// Try to parse the date string.
$date = JFactory::getDate($date2, $offset);
// Check if the date was parsed successfully.
if ($date->toUnix() !== null)
{
// Set the date filter.
$this->date2 = $date->toSql();
$this->when2 = in_array($when2, $whens, true) ? $when2 :
'before';
}
return true;
}
/**
* Method to process the query input string and extract required,
optional,
* and excluded tokens; taxonomy filters; and date filters.
*
* @param string $input The query input string.
* @param string $lang The query input language.
* @param string $mode The query matching mode.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function processString($input, $lang, $mode)
{
// Clean up the input string.
$input = html_entity_decode($input, ENT_QUOTES, 'UTF-8');
$input = StringHelper::strtolower($input);
$input = preg_replace('#\s+#mi', ' ', $input);
$input = trim($input);
$debug = JFactory::getConfig()->get('debug_lang');
/*
* First, we need to handle string based modifiers. String based
* modifiers could potentially include things like
"category:blah" or
* "before:2009-10-21" or "type:article", etc.
*/
$patterns = array(
'before' =>
JText::_('COM_FINDER_FILTER_WHEN_BEFORE'),
'after' =>
JText::_('COM_FINDER_FILTER_WHEN_AFTER'),
);
// Add the taxonomy branch titles to the possible patterns.
foreach (FinderIndexerTaxonomy::getBranchTitles() as $branch)
{
// Add the pattern.
$patterns[$branch] =
StringHelper::strtolower(JText::_(FinderHelperLanguage::branchSingular($branch)));
}
// Container for search terms and phrases.
$terms = array();
$phrases = array();
// Cleared filter branches.
$cleared = array();
/*
* Compile the suffix pattern. This is used to match the values of the
* filter input string. Single words can be input directly, multi-word
* values have to be wrapped in double quotes.
*/
$quotes = html_entity_decode('‘’'',
ENT_QUOTES, 'UTF-8');
$suffix = '(([\w\d' . $quotes .
'-]+)|\"([\w\d\s' . $quotes . '-]+)\")';
/*
* Iterate through the possible filter patterns and search for matches.
* We need to match the key, colon, and a value pattern for the match
* to be valid.
*/
foreach ($patterns as $modifier => $pattern)
{
$matches = array();
if ($debug)
{
$pattern = substr($pattern, 2, -2);
}
// Check if the filter pattern is in the input string.
if (preg_match('#' . $pattern . '\s*:\s*' . $suffix
. '#mi', $input, $matches))
{
// Get the value given to the modifier.
$value = isset($matches[3]) ? $matches[3] : $matches[1];
// Now we have to handle the filter string.
switch ($modifier)
{
// Handle a before and after date filters.
case 'before':
case 'after':
{
// Get the time offset.
$offset = JFactory::getApplication()->get('offset');
// Array of allowed when values.
$whens = array('before', 'after',
'exact');
// The value of 'today' is a special case that we need to
handle.
if ($value ===
StringHelper::strtolower(JText::_('COM_FINDER_QUERY_FILTER_TODAY')))
{
$value = JFactory::getDate('now',
$offset)->format('%Y-%m-%d');
}
// Try to parse the date string.
$date = JFactory::getDate($value, $offset);
// Check if the date was parsed successfully.
if ($date->toUnix() !== null)
{
// Set the date filter.
$this->date1 = $date->toSql();
$this->when1 = in_array($modifier, $whens, true) ? $modifier :
'before';
}
break;
}
// Handle a taxonomy branch filter.
default:
{
// Try to find the node id.
$return = FinderIndexerTaxonomy::getNodeByTitle($modifier, $value);
// Check if the node id was found.
if ($return)
{
// Check if the branch has been cleared.
if (!in_array($modifier, $cleared, true))
{
// Clear the branch.
$this->filters[$modifier] = array();
// Add the branch to the cleared list.
$cleared[] = $modifier;
}
// Add the filter to the list.
$this->filters[$modifier][$return->title] = (int)
$return->id;
}
break;
}
}
// Clean up the input string again.
$input = str_replace($matches[0], '', $input);
$input = preg_replace('#\s+#mi', ' ', $input);
$input = trim($input);
}
}
/*
* Extract the tokens enclosed in double quotes so that we can handle
* them as phrases.
*/
if (StringHelper::strpos($input, '"') !== false)
{
$matches = array();
// Extract the tokens enclosed in double quotes.
if (preg_match_all('#\"([^"]+)\"#m', $input,
$matches))
{
/*
* One or more phrases were found so we need to iterate through
* them, tokenize them as phrases, and remove them from the raw
* input string before we move on to the next processing step.
*/
foreach ($matches[1] as $key => $match)
{
// Find the complete phrase in the input string.
$pos = StringHelper::strpos($input, $matches[0][$key]);
$len = StringHelper::strlen($matches[0][$key]);
// Add any terms that are before this phrase to the stack.
if (trim(StringHelper::substr($input, 0, $pos)))
{
$terms = array_merge($terms, explode(' ',
trim(StringHelper::substr($input, 0, $pos))));
}
// Strip out everything up to and including the phrase.
$input = StringHelper::substr($input, $pos + $len);
// Clean up the input string again.
$input = preg_replace('#\s+#mi', ' ', $input);
$input = trim($input);
// Get the number of words in the phrase.
$parts = explode(' ', $match);
// Check if the phrase is longer than three words.
if (count($parts) > 3)
{
/*
* If the phrase is longer than three words, we need to
* break it down into smaller chunks of phrases that
* are less than or equal to three words. We overlap
* the chunks so that we can ensure that a match is
* found for the complete phrase and not just portions
* of it.
*/
for ($i = 0, $c = count($parts); $i < $c; $i += 2)
{
// Set up the chunk.
$chunk = array();
// The chunk has to be assembled based on how many
// pieces are available to use.
switch ($c - $i)
{
/*
* If only one word is left, we can break from
* the switch and loop because the last word
* was already used at the end of the last
* chunk.
*/
case 1:
break 2;
// If there words are left, we use them both as
// the last chunk of the phrase and we're done.
case 2:
$chunk[] = $parts[$i];
$chunk[] = $parts[$i + 1];
break;
// If there are three or more words left, we
// build a three word chunk and continue on.
default:
$chunk[] = $parts[$i];
$chunk[] = $parts[$i + 1];
$chunk[] = $parts[$i + 2];
break;
}
// If the chunk is not empty, add it as a phrase.
if (count($chunk))
{
$phrases[] = implode(' ', $chunk);
$terms[] = implode(' ', $chunk);
}
}
}
else
{
// The phrase is <= 3 words so we can use it as is.
$phrases[] = $match;
$terms[] = $match;
}
}
}
}
// Add the remaining terms if present.
if ((bool) $input)
{
$terms = array_merge($terms, explode(' ', $input));
}
// An array of our boolean operators. $operator => $translation
$operators = array(
'AND' =>
StringHelper::strtolower(JText::_('COM_FINDER_QUERY_OPERATOR_AND')),
'OR' =>
StringHelper::strtolower(JText::_('COM_FINDER_QUERY_OPERATOR_OR')),
'NOT' =>
StringHelper::strtolower(JText::_('COM_FINDER_QUERY_OPERATOR_NOT')),
);
// If language debugging is enabled you need to ignore the debug strings
in matching.
if (JDEBUG)
{
$debugStrings = array('**', '??');
$operators = str_replace($debugStrings, '', $operators);
}
/*
* Iterate through the terms and perform any sorting that needs to be
* done based on boolean search operators. Terms that are before an
* and/or/not modifier have to be handled in relation to their operator.
*/
for ($i = 0, $c = count($terms); $i < $c; $i++)
{
// Check if the term is followed by an operator that we understand.
if (isset($terms[$i + 1]) && in_array($terms[$i + 1],
$operators, true))
{
// Get the operator mode.
$op = array_search($terms[$i + 1], $operators, true);
// Handle the AND operator.
if ($op === 'AND' && isset($terms[$i + 2]))
{
// Tokenize the current term.
$token = FinderIndexerHelper::tokenize($terms[$i], $lang, true);
// Todo: The previous function call may return an array, which seems
not to be handled by the next one, which expects an object
$token = $this->getTokenData($token);
// Set the required flag.
$token->required = true;
// Add the current token to the stack.
$this->included[] = $token;
$this->highlight = array_merge($this->highlight,
array_keys($token->matches));
// Skip the next token (the mode operator).
$this->operators[] = $terms[$i + 1];
// Tokenize the term after the next term (current plus two).
$other = FinderIndexerHelper::tokenize($terms[$i + 2], $lang, true);
$other = $this->getTokenData($other);
// Set the required flag.
$other->required = true;
// Add the token after the next token to the stack.
$this->included[] = $other;
$this->highlight = array_merge($this->highlight,
array_keys($other->matches));
// Remove the processed phrases if possible.
if (($pk = array_search($terms[$i], $phrases, true)) !== false)
{
unset($phrases[$pk]);
}
if (($pk = array_search($terms[$i + 2], $phrases, true)) !== false)
{
unset($phrases[$pk]);
}
// Remove the processed terms.
unset($terms[$i], $terms[$i + 1], $terms[$i + 2]);
// Adjust the loop.
$i += 2;
continue;
}
// Handle the OR operator.
elseif ($op === 'OR' && isset($terms[$i + 2]))
{
// Tokenize the current term.
$token = FinderIndexerHelper::tokenize($terms[$i], $lang, true);
$token = $this->getTokenData($token);
// Set the required flag.
$token->required = false;
// Add the current token to the stack.
if ((bool) $token->matches)
{
$this->included[] = $token;
$this->highlight = array_merge($this->highlight,
array_keys($token->matches));
}
else
{
$this->ignored[] = $token;
}
// Skip the next token (the mode operator).
$this->operators[] = $terms[$i + 1];
// Tokenize the term after the next term (current plus two).
$other = FinderIndexerHelper::tokenize($terms[$i + 2], $lang, true);
$other = $this->getTokenData($other);
// Set the required flag.
$other->required = false;
// Add the token after the next token to the stack.
if ((bool) $other->matches)
{
$this->included[] = $other;
$this->highlight = array_merge($this->highlight,
array_keys($other->matches));
}
else
{
$this->ignored[] = $other;
}
// Remove the processed phrases if possible.
if (($pk = array_search($terms[$i], $phrases, true)) !== false)
{
unset($phrases[$pk]);
}
if (($pk = array_search($terms[$i + 2], $phrases, true)) !== false)
{
unset($phrases[$pk]);
}
// Remove the processed terms.
unset($terms[$i], $terms[$i + 1], $terms[$i + 2]);
// Adjust the loop.
$i += 2;
continue;
}
}
// Handle an orphaned OR operator.
elseif (isset($terms[$i + 1]) && array_search($terms[$i],
$operators, true) === 'OR')
{
// Skip the next token (the mode operator).
$this->operators[] = $terms[$i];
// Tokenize the next term (current plus one).
$other = FinderIndexerHelper::tokenize($terms[$i + 1], $lang, true);
$other = $this->getTokenData($other);
// Set the required flag.
$other->required = false;
// Add the token after the next token to the stack.
if ((bool) $other->matches)
{
$this->included[] = $other;
$this->highlight = array_merge($this->highlight,
array_keys($other->matches));
}
else
{
$this->ignored[] = $other;
}
// Remove the processed phrase if possible.
if (($pk = array_search($terms[$i + 1], $phrases, true)) !== false)
{
unset($phrases[$pk]);
}
// Remove the processed terms.
unset($terms[$i], $terms[$i + 1]);
// Adjust the loop.
$i++;
continue;
}
// Handle the NOT operator.
elseif (isset($terms[$i + 1]) && array_search($terms[$i],
$operators, true) === 'NOT')
{
// Skip the next token (the mode operator).
$this->operators[] = $terms[$i];
// Tokenize the next term (current plus one).
$other = FinderIndexerHelper::tokenize($terms[$i + 1], $lang, true);
$other = $this->getTokenData($other);
// Set the required flag.
$other->required = false;
// Add the next token to the stack.
if ((bool) $other->matches)
{
$this->excluded[] = $other;
}
else
{
$this->ignored[] = $other;
}
// Remove the processed phrase if possible.
if (($pk = array_search($terms[$i + 1], $phrases, true)) !== false)
{
unset($phrases[$pk]);
}
// Remove the processed terms.
unset($terms[$i], $terms[$i + 1]);
// Adjust the loop.
$i++;
continue;
}
}
/*
* Iterate through any search phrases and tokenize them. We handle
* phrases as autonomous units and do not break them down into two and
* three word combinations.
*/
for ($i = 0, $c = count($phrases); $i < $c; $i++)
{
// Tokenize the phrase.
$token = FinderIndexerHelper::tokenize($phrases[$i], $lang, true);
$token = $this->getTokenData($token);
// Set the required flag.
$token->required = true;
// Add the current token to the stack.
$this->included[] = $token;
$this->highlight = array_merge($this->highlight,
array_keys($token->matches));
// Remove the processed term if possible.
if (($pk = array_search($phrases[$i], $terms, true)) !== false)
{
unset($terms[$pk]);
}
// Remove the processed phrase.
unset($phrases[$i]);
}
/*
* Handle any remaining tokens using the standard processing mechanism.
*/
if ((bool) $terms)
{
// Tokenize the terms.
$terms = implode(' ', $terms);
$tokens = FinderIndexerHelper::tokenize($terms, $lang, false);
// Make sure we are working with an array.
$tokens = is_array($tokens) ? $tokens : array($tokens);
// Get the token data and required state for all the tokens.
foreach ($tokens as $token)
{
// Get the token data.
$token = $this->getTokenData($token);
// Set the required flag for the token.
$token->required = $mode === 'AND' ? (!$token->phrase)
: false;
// Add the token to the appropriate stack.
if ($token->required || (bool) $token->matches)
{
$this->included[] = $token;
$this->highlight = array_merge($this->highlight,
array_keys($token->matches));
}
else
{
$this->ignored[] = $token;
}
}
}
return true;
}
/**
* Method to get the base and similar term ids and, if necessary,
suggested
* term data from the database. The terms ids are identified based on a
* 'like' match in MySQL and/or a common stem. If no term ids
could be
* found, then we know that we will not be able to return any results for
* that term and we should try to find a similar term to use that we can
* match so that we can suggest the alternative search query to the user.
*
* @param FinderIndexerToken $token A FinderIndexerToken object.
*
* @return FinderIndexerToken A FinderIndexerToken object.
*
* @since 2.5
* @throws Exception on database error.
*/
protected function getTokenData($token)
{
// Get the database object.
$db = JFactory::getDbo();
// Create a database query to build match the token.
$query = $db->getQuery(true)
->select('t.term, t.term_id')
->from('#__finder_terms AS t');
/*
* If the token is a phrase, the lookup process is fairly simple. If
* the token is a word, it is a little more complicated. We have to
* create two queries to lookup the term and the stem respectively,
* then union the result sets together. This is MUCH faster than using
* an or condition in the database query.
*/
if ($token->phrase)
{
// Add the phrase to the query.
$query->where('t.term = ' . $db->quote($token->term))
->where('t.phrase = 1');
}
else
{
// Add the term to the query.
$query->where('t.term = ' . $db->quote($token->term))
->where('t.phrase = 0');
// Clone the query, replace the WHERE clause.
$sub = clone $query;
$sub->clear('where');
$sub->where('t.stem = ' . $db->quote($token->stem));
$sub->where('t.phrase = 0');
// Union the two queries.
$query->union($sub);
}
// Get the terms.
$db->setQuery($query);
$matches = $db->loadObjectList();
// Check the matching terms.
if ((bool) $matches)
{
// Add the matches to the token.
for ($i = 0, $c = count($matches); $i < $c; $i++)
{
$token->matches[$matches[$i]->term] = (int)
$matches[$i]->term_id;
}
}
// If no matches were found, try to find a similar but better token.
if (empty($token->matches))
{
// Create a database query to get the similar terms.
// TODO: PostgreSQL doesn't support SOUNDEX out of the box
$query->clear()
->select('DISTINCT t.term_id AS id, t.term AS term')
->from('#__finder_terms AS t')
// ->where('t.soundex = ' .
soundex($db->quote($token->term)))
->where('t.soundex = SOUNDEX(' .
$db->quote($token->term) . ')')
->where('t.phrase = ' . (int) $token->phrase);
// Get the terms.
$db->setQuery($query);
$results = $db->loadObjectList();
// Check if any similar terms were found.
if (empty($results))
{
return $token;
}
// Stack for sorting the similar terms.
$suggestions = array();
// Get the levnshtein distance for all suggested terms.
foreach ($results as $sk => $st)
{
// Get the levenshtein distance between terms.
$distance = levenshtein($st->term, $token->term);
// Make sure the levenshtein distance isn't over 50.
if ($distance < 50)
{
$suggestions[$sk] = $distance;
}
}
// Sort the suggestions.
asort($suggestions, SORT_NUMERIC);
// Get the closest match.
$keys = array_keys($suggestions);
$key = $keys[0];
// Add the suggested term.
$token->suggestion = $results[$key]->term;
}
return $token;
}
}
PK"�[��FL""helpers/indexer/result.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderIndexer', __DIR__ .
'/indexer.php');
/**
* Result class for the Finder indexer package.
*
* This class uses magic __get() and __set() methods to prevent properties
* being added that might confuse the system. All properties not explicitly
* declared will be pushed into the elements array and can be accessed
* explicitly using the getElement() method.
*
* @since 2.5
*/
class FinderIndexerResult
{
/**
* An array of extra result properties.
*
* @var array
* @since 2.5
*/
protected $elements = array();
/**
* This array tells the indexer which properties should be indexed and
what
* weights to use for those properties.
*
* @var array
* @since 2.5
*/
protected $instructions = array(
FinderIndexer::TITLE_CONTEXT => array('title',
'subtitle', 'id'),
FinderIndexer::TEXT_CONTEXT => array('summary',
'body'),
FinderIndexer::META_CONTEXT => array('meta',
'list_price', 'sale_price'),
FinderIndexer::PATH_CONTEXT => array('path',
'alias'),
FinderIndexer::MISC_CONTEXT => array('comments'),
);
/**
* The indexer will use this data to create taxonomy mapping entries for
* the item so that it can be filtered by type, label, category,
* or whatever.
*
* @var array
* @since 2.5
*/
protected $taxonomy = array();
/**
* The content URL.
*
* @var string
* @since 2.5
*/
public $url;
/**
* The content route.
*
* @var string
* @since 2.5
*/
public $route;
/**
* The content title.
*
* @var string
* @since 2.5
*/
public $title;
/**
* The content description.
*
* @var string
* @since 2.5
*/
public $description;
/**
* The published state of the result.
*
* @var integer
* @since 2.5
*/
public $published;
/**
* The content published state.
*
* @var integer
* @since 2.5
*/
public $state;
/**
* The content access level.
*
* @var integer
* @since 2.5
*/
public $access;
/**
* The content language.
*
* @var string
* @since 2.5
*/
public $language = '*';
/**
* The publishing start date.
*
* @var string
* @since 2.5
*/
public $publish_start_date;
/**
* The publishing end date.
*
* @var string
* @since 2.5
*/
public $publish_end_date;
/**
* The generic start date.
*
* @var string
* @since 2.5
*/
public $start_date;
/**
* The generic end date.
*
* @var string
* @since 2.5
*/
public $end_date;
/**
* The item list price.
*
* @var mixed
* @since 2.5
*/
public $list_price;
/**
* The item sale price.
*
* @var mixed
* @since 2.5
*/
public $sale_price;
/**
* The content type id. This is set by the adapter.
*
* @var integer
* @since 2.5
*/
public $type_id;
/**
* The default language for content.
*
* @var string
* @since 3.0.2
*/
public $defaultLanguage;
/**
* Constructor
*
* @since 3.0.3
*/
public function __construct()
{
$this->defaultLanguage =
JComponentHelper::getParams('com_languages')->get('site',
'en-GB');
}
/**
* The magic set method is used to push additional values into the
elements
* array in order to preserve the cleanliness of the object.
*
* @param string $name The name of the element.
* @param mixed $value The value of the element.
*
* @return void
*
* @since 2.5
*/
public function __set($name, $value)
{
$this->setElement($name, $value);
}
/**
* The magic get method is used to retrieve additional element values from
the elements array.
*
* @param string $name The name of the element.
*
* @return mixed The value of the element if set, null otherwise.
*
* @since 2.5
*/
public function __get($name)
{
return $this->getElement($name);
}
/**
* The magic isset method is used to check the state of additional element
values in the elements array.
*
* @param string $name The name of the element.
*
* @return boolean True if set, false otherwise.
*
* @since 2.5
*/
public function __isset($name)
{
return isset($this->elements[$name]);
}
/**
* The magic unset method is used to unset additional element values in
the elements array.
*
* @param string $name The name of the element.
*
* @return void
*
* @since 2.5
*/
public function __unset($name)
{
unset($this->elements[$name]);
}
/**
* Method to retrieve additional element values from the elements array.
*
* @param string $name The name of the element.
*
* @return mixed The value of the element if set, null otherwise.
*
* @since 2.5
*/
public function getElement($name)
{
// Get the element value if set.
if (array_key_exists($name, $this->elements))
{
return $this->elements[$name];
}
return null;
}
/**
* Method to retrieve all elements.
*
* @return array The elements
*
* @since 3.8.3
*/
public function getElements()
{
return $this->elements;
}
/**
* Method to set additional element values in the elements array.
*
* @param string $name The name of the element.
* @param mixed $value The value of the element.
*
* @return void
*
* @since 2.5
*/
public function setElement($name, $value)
{
$this->elements[$name] = $value;
}
/**
* Method to get all processing instructions.
*
* @return array An array of processing instructions.
*
* @since 2.5
*/
public function getInstructions()
{
return $this->instructions;
}
/**
* Method to add a processing instruction for an item property.
*
* @param string $group The group to associate the property with.
* @param string $property The property to process.
*
* @return void
*
* @since 2.5
*/
public function addInstruction($group, $property)
{
// Check if the group exists. We can't add instructions for unknown
groups.
// Check if the property exists in the group.
if (array_key_exists($group, $this->instructions) &&
!in_array($property, $this->instructions[$group], true))
{
// Add the property to the group.
$this->instructions[$group][] = $property;
}
}
/**
* Method to remove a processing instruction for an item property.
*
* @param string $group The group to associate the property with.
* @param string $property The property to process.
*
* @return void
*
* @since 2.5
*/
public function removeInstruction($group, $property)
{
// Check if the group exists. We can't remove instructions for
unknown groups.
if (array_key_exists($group, $this->instructions))
{
// Search for the property in the group.
$key = array_search($property, $this->instructions[$group]);
// If the property was found, remove it.
if ($key !== false)
{
unset($this->instructions[$group][$key]);
}
}
}
/**
* Method to get the taxonomy maps for an item.
*
* @param string $branch The taxonomy branch to get. [optional]
*
* @return array An array of taxonomy maps.
*
* @since 2.5
*/
public function getTaxonomy($branch = null)
{
// Get the taxonomy branch if available.
if ($branch !== null && isset($this->taxonomy[$branch]))
{
// Filter the input.
$branch =
preg_replace('#[^\pL\pM\pN\p{Pi}\p{Pf}\'+-.,_]+#mui', '
', $branch);
return $this->taxonomy[$branch];
}
return $this->taxonomy;
}
/**
* Method to add a taxonomy map for an item.
*
* @param string $branch The title of the taxonomy branch to add the
node to.
* @param string $title The title of the taxonomy node.
* @param integer $state The published state of the taxonomy node.
[optional]
* @param integer $access The access level of the taxonomy node.
[optional]
*
* @return void
*
* @since 2.5
*/
public function addTaxonomy($branch, $title, $state = 1, $access = 1)
{
// Filter the input.
$branch =
preg_replace('#[^\pL\pM\pN\p{Pi}\p{Pf}\'+-.,_]+#mui', '
', $branch);
// Create the taxonomy node.
$node = new JObject;
$node->title = $title;
$node->state = (int) $state;
$node->access = (int) $access;
// Add the node to the taxonomy branch.
$this->taxonomy[$branch][$node->title] = $node;
}
/**
* Method to set the item language
*
* @return void
*
* @since 3.0
*/
public function setLanguage()
{
if ($this->language == '')
{
$this->language = $this->defaultLanguage;
}
}
}
PK"�[��h(h(helpers/indexer/stemmer/fr.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderIndexerStemmer', dirname(__DIR__) .
'/stemmer.php');
/**
* French stemmer class for Smart Search indexer.
*
* First contributed by Eric Sanou (bobotche@hotmail.fr)
* This class is inspired in Alexis Ulrich's French stemmer code
(http://alx2002.free.fr)
*
* @since 3.0
*/
class FinderIndexerStemmerFr extends FinderIndexerStemmer
{
/**
* Stemming rules.
*
* @var array
* @since 3.0
*/
private static $stemRules;
/**
* Method to stem a token and return the root.
*
* @param string $token The token to stem.
* @param string $lang The language of the token.
*
* @return string The root token.
*
* @since 3.0
*/
public function stem($token, $lang)
{
// Check if the token is long enough to merit stemming.
if (strlen($token) <= 2)
{
return $token;
}
// Check if the language is French or All.
if ($lang !== 'fr' && $lang !== '*')
{
return $token;
}
// Stem the token if it is not in the cache.
if (!isset($this->cache[$lang][$token]))
{
// Stem the token.
$result = self::getStem($token);
// Add the token to the cache.
$this->cache[$lang][$token] = $result;
}
return $this->cache[$lang][$token];
}
/**
* French stemmer rules variables.
*
* @return array The rules
*
* @since 3.0
*/
protected static function getStemRules()
{
if (self::$stemRules)
{
return self::$stemRules;
}
$vars = array();
// French accented letters in ISO-8859-1 encoding
$vars['accents'] = chr(224) . chr(226) . chr(232) . chr(233) .
chr(234) . chr(235) . chr(238) . chr(239)
. chr(244) . chr(251) . chr(249) . chr(231);
// The rule patterns include all accented words for french language
$vars['rule_pattern'] = '/^([a-z' .
$vars['accents'] . ']*)(\*){0,1}(\d)([a-z' .
$vars['accents'] . ']*)([.|>])/';
// French vowels (including y) in ISO-8859-1 encoding
$vars['vowels'] = chr(97) . chr(224) . chr(226) . chr(101) .
chr(232) . chr(233) . chr(234) . chr(235)
. chr(105) . chr(238) . chr(239) . chr(111) . chr(244) . chr(117) .
chr(251) . chr(249) . chr(121);
// The French rules in ISO-8859-1 encoding
$vars['rules'] = array(
'esre1>', 'esio1>', 'siol1.',
'siof0.', 'sioe0.', 'sio3>',
'st1>', 'sf1>', 'sle1>',
'slo1>', 's' . chr(233) . '1>',
chr(233) . 'tuae5.',
chr(233) . 'tuae2.', 'tnia0.', 'tniv1.',
'tni3>', 'suor1.', 'suo0.',
'sdrail5.', 'sdrai4.', 'er' . chr(232) .
'i1>', 'sesue3x>',
'esuey5i.', 'esue2x>', 'se1>',
'er' . chr(232) . 'g3.', 'eca1>',
'esiah0.', 'esi1>', 'siss2.',
'sir2>', 'sit2>', 'egan' . chr(233) .
'1.',
'egalli6>', 'egass1.', 'egas0.',
'egat3.', 'ega3>', 'ette4>',
'ett2>', 'etio1.', 'tio' . chr(231) .
'4c.', 'tio0.', 'et1>',
'eb1>',
'snia1>', 'eniatnau8>', 'eniatn4.',
'enia1>', 'niatnio3.', 'niatg3.',
'e' . chr(233) . '1>', chr(233) .
'hcat1.', chr(233) . 'hca4.',
chr(233) . 'tila5>', chr(233) . 'tici5.',
chr(233) . 'tir1.', chr(233) . 'ti3>', chr(233) .
'gan1.', chr(233) . 'ga3>',
chr(233) . 'tehc1.', chr(233) . 'te3>', chr(233)
. 'it0.', chr(233) . '1>', 'eire4.',
'eirue5.', 'eio1.', 'eia1.',
'ei1>', 'eng1.',
'xuaessi7.', 'xuae1>', 'uaes0.',
'uae3.', 'xuave2l.', 'xuav2li>',
'xua3la>', 'ela1>', 'lart2.',
'lani2>', 'la' . chr(233) . '2>',
'siay4i.', 'siassia7.', 'siarv1*.',
'sia1>', 'tneiayo6i.', 'tneiay6i.',
'tneiassia9.', 'tneiareio7.', 'tneia5>',
'tneia4>', 'tiario4.',
'tiarim3.', 'tiaria3.', 'tiaris3.',
'tiari5.', 'tiarve6>', 'tiare5>',
'iare4>', 'are3>', 'tiay4i.',
'tia3>', 'tnay4i.',
'em' . chr(232) . 'iu5>', 'em' .
chr(232) . 'i4>', 'tnaun3.', 'tnauqo3.',
'tnau4>', 'tnaf0.', 'tnat' . chr(233) .
'2>', 'tna3>', 'tno3>',
'zeiy4i.', 'zey3i.', 'zeire5>',
'zeird4.', 'zeirio4.', 'ze2>',
'ssiab0.', 'ssia4.', 'ssi3.',
'tnemma6>', 'tnemesuey9i.',
'tnemesue8>',
'tnemevi7.', 'tnemessia5.', 'tnemessi8.',
'tneme5>', 'tnemia4.', 'tnem' . chr(233) .
'5>', 'el2l>', 'lle3le>',
'let' . chr(244) . '0.',
'lepp0.', 'le2>', 'srei1>',
'reit3.', 'reila2.', 'rei3>',
'ert' . chr(226) . 'e5.', 'ert' . chr(226) .
chr(233) . '1.',
'ert' . chr(226) . '4.', 'drai4.',
'erdro0.', 'erute5.', 'ruta0.',
'eruta1.', 'erutiov1.', 'erub3.',
'eruh3.', 'erul3.', 'er2r>',
'nn1>',
'r' . chr(232) . 'i3.', 'srev0.',
'sr1>', 'rid2>', 're2>',
'xuei4.', 'esuei5.', 'lbati3.',
'lba3>', 'rueis0.', 'ruehcn4.',
'ecirta6.',
'ruetai6.', 'rueta5.', 'rueir0.',
'rue3>', 'esseti6.', 'essere6>',
'esserd1.', 'esse4>', 'essiab1.',
'essia5.', 'essio1.', 'essi4.',
'essal4.', 'essa1>', 'ssab1.',
'essurp1.', 'essu4.', 'essi1.',
'ssor1.', 'essor2.', 'esso1>',
'ess2>', 'tio3.', 'r' . chr(232) .
's2re.',
'r' . chr(232) . '0e.', 'esn1.',
'eu1>', 'sua0.', 'su1>',
'utt1>', 'tu' . chr(231) . '3c.',
'u' . chr(231) . '2c.', 'ur1.',
'ehcn2>',
'ehcu1>', 'snorr3.', 'snoru3.',
'snorua3.', 'snorv3.', 'snorio4.',
'snori5.', 'snore5>', 'snortt4>',
'snort' . chr(238) . 'a7.', 'snort3.',
'snor4.', 'snossi6.', 'snoire6.',
'snoird5.', 'snoitai7.', 'snoita6.',
'snoits1>', 'noits0.', 'snoi4>',
'noitaci7>', 'noitai6.', 'noita5.',
'noitu4.', 'noi3>', 'snoya0.',
'snoy4i.', 'sno' . chr(231) . 'a1.',
'sno' . chr(231) . 'r1.', 'snoe4.',
'snosiar1>', 'snola1.', 'sno3>',
'sno1>', 'noll2.', 'tnennei4.',
'ennei2>', 'snei1>', 'sne' . chr(233) .
'1>', 'enne' . chr(233) . '5e.',
'ne' . chr(233) . '3e.', 'neic0.',
'neiv0.', 'nei3.', 'sc1.',
'sd1.', 'sg1.', 'sni1.', 'tiu0.',
'ti2.', 'sp1>', 'sna1>',
'sue1.', 'enn2>', 'nong2.',
'noss2.', 'rioe4.',
'riot0.', 'riorc1.', 'riovec5.',
'rio3.', 'ric2.', 'ril2.',
'tnerim3.', 'tneris3>', 'tneri5.',
't' . chr(238) . 'a3.', 'riss2.',
't' . chr(238) . '2.', 't' . chr(226) .
'2>', 'ario2.', 'arim1.',
'ara1.', 'aris1.', 'ari3.',
'art1>', 'ardn2.', 'arr1.',
'arua1.',
'aro1.', 'arv1.', 'aru1.',
'ar2.', 'rd1.', 'ud1.', 'ul1.',
'ini1.', 'rin2.', 'tnessiab3.',
'tnessia7.', 'tnessi6.', 'tnessni4.',
'sini2.',
'sl1.', 'iard3.', 'iario3.',
'ia2>', 'io0.', 'iule2.',
'i1>', 'sid2.', 'sic2.',
'esoi4.', 'ed1.', 'ai2>',
'a1>', 'adr1.',
'tner' . chr(232) . '5>', 'evir1.',
'evio4>', 'evi3.', 'fita4.',
'fi2>', 'enie1.', 'sare4>',
'sari4>', 'sard3.', 'sart2>',
'sa2.',
'tnessa6>', 'tnessu6>', 'tnegna3.',
'tnegi3.', 'tneg0.', 'tneru5>',
'tnemg0.', 'tnerni4.', 'tneiv1.',
'tne3>', 'une1.', 'en1>',
'nitn2.',
'ecnay5i.', 'ecnal1.', 'ecna4.',
'ec1>', 'nn1.', 'rit2>',
'rut2>', 'rud2.', 'ugn1>',
'eg1>', 'tuo0.', 'tul2>', 't'
. chr(251) . '2>',
'ev1>', 'v' . chr(232) . '2ve>',
'rtt1>', 'emissi6.', 'em1.',
'ehc1.', 'c' . chr(233) . 'i2c' . chr(232) .
'.', 'libi2l.', 'llie1.',
'liei4i.', 'xuev1.', 'xuey4i.',
'xueni5>', 'xuell4.', 'xuere5.',
'xue3>', 'rb' . chr(233) . '3rb' .
chr(232) . '.', 'tur2.',
'rir' . chr(233) . '4re.', 'rir2.',
'c' . chr(226) . '2ca.', 'snu1.',
'rt' . chr(238) . 'a4.', 'long2.',
'vec2.', chr(231) . '1c>',
'ssilp3.', 'silp2.', 't' . chr(232) .
'hc2te.', 'n' . chr(232) . 'm2ne.',
'llepp1.', 'tan2.', 'rv' . chr(232) .
'3rve.',
'rv' . chr(233) . '3rve.', 'r' . chr(232)
. '2re.', 'r' . chr(233) . '2re.',
't' . chr(232) . '2te.', 't' . chr(233) .
'2te.', 'epp1.',
'eya2i.', 'ya1i.', 'yo1i.',
'esu1.', 'ugi1.', 'tt1.', 'end0.'
);
self::$stemRules = $vars;
return self::$stemRules;
}
/**
* Returns the number of the first rule from the rule number
* that can be applied to the given reversed input.
* returns -1 if no rule can be applied, ie the stem has been found
*
* @param string $reversedInput The input to check in reversed order
* @param integer $ruleNumber The rule number to check
*
* @return integer Number of the first rule
*
* @since 3.0
*/
private static function getFirstRule($reversedInput, $ruleNumber)
{
$vars = static::getStemRules();
$nb_rules = count($vars['rules']);
for ($i = $ruleNumber; $i < $nb_rules; $i++)
{
// Gets the letters from the current rule
$rule = $vars['rules'][$i];
$rule = preg_replace($vars['rule_pattern'], "\\1",
$rule);
if (strncasecmp(utf8_decode($rule), $reversedInput,
strlen(utf8_decode($rule))) == 0)
{
return $i;
}
}
return -1;
}
/**
* Check the acceptability of a stem for French language
*
* @param string $reversedStem The stem to check in reverse form
*
* @return boolean True if stem is acceptable
*
* @since 3.0
*/
private static function check($reversedStem)
{
$vars = static::getStemRules();
if (preg_match('/[' . $vars['vowels'] .
']$/', utf8_encode($reversedStem)))
{
// If the form starts with a vowel then at least two letters must remain
after stemming (e.g.: "etaient" --> "et")
return (strlen($reversedStem) > 2);
}
else
{
// If the reversed stem starts with a consonant then at least two
letters must remain after stemming
if (strlen($reversedStem) <= 2)
{
return false;
}
// And at least one of these must be a vowel or "y"
return preg_match('/[' . $vars['vowels'] .
']/', utf8_encode($reversedStem));
}
}
/**
* Paice/Husk stemmer which returns a stem for the given $input
*
* @param string $input The word for which we want the stem in UTF-8
*
* @return string The stem
*
* @since 3.0
*/
private static function getStem($input)
{
$vars = static::getStemRules();
$reversed_input = strrev(utf8_decode($input));
$rule_number = 0;
// This loop goes through the rules' array until it finds an ending
one (ending by '.') or the last one ('end0.')
while (true)
{
$rule_number = self::getFirstRule($reversed_input, $rule_number);
if ($rule_number === -1)
{
// No other rule can be applied => the stem has been found
break;
}
$rule = $vars['rules'][$rule_number];
preg_match($vars['rule_pattern'], $rule, $matches);
$reversed_stem = utf8_decode($matches[4]) . substr($reversed_input,
$matches[3]);
if (self::check($reversed_stem))
{
$reversed_input = $reversed_stem;
if ($matches[5] === '.')
{
break;
}
}
else
{
// Go to another rule
$rule_number++;
}
}
return utf8_encode(strrev($reversed_input));
}
}
PK"�[�(''%helpers/indexer/stemmer/porter_en.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderIndexerStemmer', dirname(__DIR__) .
'/stemmer.php');
/**
* Porter English stemmer class for the Finder indexer package.
*
* This class was adapted from one written by Richard Heyes.
* See copyright and link information above.
*
* @since 2.5
*/
class FinderIndexerStemmerPorter_En extends FinderIndexerStemmer
{
/**
* Regex for matching a consonant.
*
* @var string
* @since 2.5
*/
private static $regex_consonant =
'(?:[bcdfghjklmnpqrstvwxz]|(?<=[aeiou])y|^y)';
/**
* Regex for matching a vowel
*
* @var string
* @since 2.5
*/
private static $regex_vowel = '(?:[aeiou]|(?<![aeiou])y)';
/**
* Method to stem a token and return the root.
*
* @param string $token The token to stem.
* @param string $lang The language of the token.
*
* @return string The root token.
*
* @since 2.5
*/
public function stem($token, $lang)
{
// Check if the token is long enough to merit stemming.
if (strlen($token) <= 2)
{
return $token;
}
// Check if the language is English or All.
if ($lang !== 'en' && $lang !== '*')
{
return $token;
}
// Stem the token if it is not in the cache.
if (!isset($this->cache[$lang][$token]))
{
// Stem the token.
$result = $token;
$result = self::step1ab($result);
$result = self::step1c($result);
$result = self::step2($result);
$result = self::step3($result);
$result = self::step4($result);
$result = self::step5($result);
// Add the token to the cache.
$this->cache[$lang][$token] = $result;
}
return $this->cache[$lang][$token];
}
/**
* Step 1
*
* @param string $word The token to stem.
*
* @return string
*
* @since 2.5
*/
private static function step1ab($word)
{
// Part a
if (substr($word, -1) === 's')
{
self::replace($word, 'sses', 'ss')
|| self::replace($word, 'ies', 'i')
|| self::replace($word, 'ss', 'ss')
|| self::replace($word, 's', '');
}
// Part b
if (substr($word, -2, 1) !== 'e' || !self::replace($word,
'eed', 'ee', 0))
{
// First rule
$v = self::$regex_vowel;
// Words ending with ing and ed
// Note use of && and OR, for precedence reasons
if (preg_match("#$v+#", substr($word, 0, -3)) &&
self::replace($word, 'ing', '')
|| preg_match("#$v+#", substr($word, 0, -2)) &&
self::replace($word, 'ed', ''))
{
// If one of above two test successful
if (!self::replace($word, 'at', 'ate') &&
!self::replace($word, 'bl', 'ble') &&
!self::replace($word, 'iz', 'ize'))
{
// Double consonant ending
$wordSubStr = substr($word, -2);
if ($wordSubStr !== 'll' && $wordSubStr !==
'ss' && $wordSubStr !== 'zz' &&
self::doubleConsonant($word))
{
$word = substr($word, 0, -1);
}
elseif (self::m($word) === 1 && self::cvc($word))
{
$word .= 'e';
}
}
}
}
return $word;
}
/**
* Step 1c
*
* @param string $word The token to stem.
*
* @return string
*
* @since 2.5
*/
private static function step1c($word)
{
$v = self::$regex_vowel;
if (substr($word, -1) === 'y' &&
preg_match("#$v+#", substr($word, 0, -1)))
{
self::replace($word, 'y', 'i');
}
return $word;
}
/**
* Step 2
*
* @param string $word The token to stem.
*
* @return string
*
* @since 2.5
*/
private static function step2($word)
{
switch (substr($word, -2, 1))
{
case 'a':
self::replace($word, 'ational', 'ate', 0)
|| self::replace($word, 'tional', 'tion', 0);
break;
case 'c':
self::replace($word, 'enci', 'ence', 0)
|| self::replace($word, 'anci', 'ance', 0);
break;
case 'e':
self::replace($word, 'izer', 'ize', 0);
break;
case 'g':
self::replace($word, 'logi', 'log', 0);
break;
case 'l':
self::replace($word, 'entli', 'ent', 0)
|| self::replace($word, 'ousli', 'ous', 0)
|| self::replace($word, 'alli', 'al', 0)
|| self::replace($word, 'bli', 'ble', 0)
|| self::replace($word, 'eli', 'e', 0);
break;
case 'o':
self::replace($word, 'ization', 'ize', 0)
|| self::replace($word, 'ation', 'ate', 0)
|| self::replace($word, 'ator', 'ate', 0);
break;
case 's':
self::replace($word, 'iveness', 'ive', 0)
|| self::replace($word, 'fulness', 'ful', 0)
|| self::replace($word, 'ousness', 'ous', 0)
|| self::replace($word, 'alism', 'al', 0);
break;
case 't':
self::replace($word, 'biliti', 'ble', 0)
|| self::replace($word, 'aliti', 'al', 0)
|| self::replace($word, 'iviti', 'ive', 0);
break;
}
return $word;
}
/**
* Step 3
*
* @param string $word The token to stem.
*
* @return string
*
* @since 2.5
*/
private static function step3($word)
{
switch (substr($word, -2, 1))
{
case 'a':
self::replace($word, 'ical', 'ic', 0);
break;
case 's':
self::replace($word, 'ness', '', 0);
break;
case 't':
self::replace($word, 'icate', 'ic', 0)
|| self::replace($word, 'iciti', 'ic', 0);
break;
case 'u':
self::replace($word, 'ful', '', 0);
break;
case 'v':
self::replace($word, 'ative', '', 0);
break;
case 'z':
self::replace($word, 'alize', 'al', 0);
break;
}
return $word;
}
/**
* Step 4
*
* @param string $word The token to stem.
*
* @return string
*
* @since 2.5
*/
private static function step4($word)
{
switch (substr($word, -2, 1))
{
case 'a':
self::replace($word, 'al', '', 1);
break;
case 'c':
self::replace($word, 'ance', '', 1)
|| self::replace($word, 'ence', '', 1);
break;
case 'e':
self::replace($word, 'er', '', 1);
break;
case 'i':
self::replace($word, 'ic', '', 1);
break;
case 'l':
self::replace($word, 'able', '', 1)
|| self::replace($word, 'ible', '', 1);
break;
case 'n':
self::replace($word, 'ant', '', 1)
|| self::replace($word, 'ement', '', 1)
|| self::replace($word, 'ment', '', 1)
|| self::replace($word, 'ent', '', 1);
break;
case 'o':
$wordSubStr = substr($word, -4);
if ($wordSubStr === 'tion' || $wordSubStr ===
'sion')
{
self::replace($word, 'ion', '', 1);
}
else
{
self::replace($word, 'ou', '', 1);
}
break;
case 's':
self::replace($word, 'ism', '', 1);
break;
case 't':
self::replace($word, 'ate', '', 1)
|| self::replace($word, 'iti', '', 1);
break;
case 'u':
self::replace($word, 'ous', '', 1);
break;
case 'v':
self::replace($word, 'ive', '', 1);
break;
case 'z':
self::replace($word, 'ize', '', 1);
break;
}
return $word;
}
/**
* Step 5
*
* @param string $word The token to stem.
*
* @return string
*
* @since 2.5
*/
private static function step5($word)
{
// Part a
if (substr($word, -1) === 'e')
{
if (self::m(substr($word, 0, -1)) > 1)
{
self::replace($word, 'e', '');
}
elseif (self::m(substr($word, 0, -1)) === 1)
{
if (!self::cvc(substr($word, 0, -1)))
{
self::replace($word, 'e', '');
}
}
}
// Part b
if (self::m($word) > 1 && self::doubleConsonant($word)
&& substr($word, -1) === 'l')
{
$word = substr($word, 0, -1);
}
return $word;
}
/**
* Replaces the first string with the second, at the end of the string. If
third
* arg is given, then the preceding string must match that m count at
least.
*
* @param string $str String to check
* @param string $check Ending to check for
* @param string $repl Replacement string
* @param integer $m Optional minimum number of m() to meet
*
* @return boolean Whether the $check string was at the end
* of the $str string. True does not necessarily mean
* that it was replaced.
*
* @since 2.5
*/
private static function replace(&$str, $check, $repl, $m = null)
{
$len = 0 - strlen($check);
if (substr($str, $len) === $check)
{
$substr = substr($str, 0, $len);
if ($m === null || self::m($substr) > $m)
{
$str = $substr . $repl;
}
return true;
}
return false;
}
/**
* m() measures the number of consonant sequences in $str. if c is
* a consonant sequence and v a vowel sequence, and <..> indicates
arbitrary
* presence,
*
* <c><v> gives 0
* <c>vc<v> gives 1
* <c>vcvc<v> gives 2
* <c>vcvcvc<v> gives 3
*
* @param string $str The string to return the m count for
*
* @return integer The m count
*
* @since 2.5
*/
private static function m($str)
{
$c = self::$regex_consonant;
$v = self::$regex_vowel;
$str = preg_replace("#^$c+#", '', $str);
$str = preg_replace("#$v+$#", '', $str);
preg_match_all("#($v+$c+)#", $str, $matches);
return count($matches[1]);
}
/**
* Returns true/false as to whether the given string contains two
* of the same consonant next to each other at the end of the string.
*
* @param string $str String to check
*
* @return boolean Result
*
* @since 2.5
*/
private static function doubleConsonant($str)
{
$c = self::$regex_consonant;
return preg_match("#$c{2}$#", $str, $matches) &&
$matches[0][0] === $matches[0][1];
}
/**
* Checks for ending CVC sequence where second C is not W, X or Y
*
* @param string $str String to check
*
* @return boolean Result
*
* @since 2.5
*/
private static function cvc($str)
{
$c = self::$regex_consonant;
$v = self::$regex_vowel;
return preg_match("#($c$v$c)$#", $str, $matches) &&
strlen($matches[1]) === 3 && $matches[1][2] !== 'w'
&& $matches[1][2] !== 'x'
&& $matches[1][2] !== 'y';
}
}
PK"�[�Ӷ��
�
$helpers/indexer/stemmer/snowball.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderIndexerStemmer', dirname(__DIR__) .
'/stemmer.php');
/**
* Snowball stemmer class for the Finder indexer package.
*
* @since 2.5
*/
class FinderIndexerStemmerSnowball extends FinderIndexerStemmer
{
/**
* Method to stem a token and return the root.
*
* @param string $token The token to stem.
* @param string $lang The language of the token.
*
* @return string The root token.
*
* @since 2.5
*/
public function stem($token, $lang)
{
// Language to use if All is specified.
static $defaultLang = '';
// If language is All then try to get site default language.
if ($lang === '*' && $defaultLang === '')
{
$languages = JLanguageHelper::getLanguages();
$defaultLang = isset($languages[0]->sef) ? $languages[0]->sef :
'*';
$lang = $defaultLang;
}
// Stem the token if it is not in the cache.
if (!isset($this->cache[$lang][$token]))
{
// Get the stem function from the language string.
switch ($lang)
{
// Danish stemmer.
case 'da':
$function = 'stem_danish';
break;
// German stemmer.
case 'de':
$function = 'stem_german';
break;
// English stemmer.
default:
case 'en':
$function = 'stem_english';
break;
// Spanish stemmer.
case 'es':
$function = 'stem_spanish';
break;
// Finnish stemmer.
case 'fi':
$function = 'stem_finnish';
break;
// French stemmer.
case 'fr':
$function = 'stem_french';
break;
// Hungarian stemmer.
case 'hu':
$function = 'stem_hungarian';
break;
// Italian stemmer.
case 'it':
$function = 'stem_italian';
break;
// Norwegian stemmer.
case 'nb':
$function = 'stem_norwegian';
break;
// Dutch stemmer.
case 'nl':
$function = 'stem_dutch';
break;
// Portuguese stemmer.
case 'pt':
$function = 'stem_portuguese';
break;
// Romanian stemmer.
case 'ro':
$function = 'stem_romanian';
break;
// Russian stemmer.
case 'ru':
$function = 'stem_russian_unicode';
break;
// Swedish stemmer.
case 'sv':
$function = 'stem_swedish';
break;
// Turkish stemmer.
case 'tr':
$function = 'stem_turkish_unicode';
break;
}
// Stem the word if the stemmer method exists.
$this->cache[$lang][$token] = function_exists($function) ?
$function($token) : $token;
}
return $this->cache[$lang][$token];
}
}
PK"�[x���helpers/indexer/stemmer.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Stemmer base class for the Finder indexer package.
*
* @since 2.5
*/
abstract class FinderIndexerStemmer
{
/**
* An internal cache of stemmed tokens.
*
* @var array
* @since 2.5
*/
public $cache = array();
/**
* Method to get a stemmer, creating it if necessary.
*
* @param string $adapter The type of stemmer to load.
*
* @return FinderIndexerStemmer A FinderIndexerStemmer instance.
*
* @since 2.5
* @throws Exception on invalid stemmer.
*/
public static function getInstance($adapter)
{
static $instances;
// Only create one stemmer for each adapter.
if (isset($instances[$adapter]))
{
return $instances[$adapter];
}
// Create an array of instances if necessary.
if (!is_array($instances))
{
$instances = array();
}
// Setup the adapter for the stemmer.
$adapter = JFilterInput::getInstance()->clean($adapter,
'cmd');
$path = __DIR__ . '/stemmer/' . $adapter . '.php';
$class = 'FinderIndexerStemmer' . ucfirst($adapter);
// Check if a stemmer exists for the adapter.
if (!file_exists($path))
{
// Throw invalid adapter exception.
throw new
Exception(JText::sprintf('COM_FINDER_INDEXER_INVALID_STEMMER',
$adapter));
}
// Instantiate the stemmer.
JLoader::register($class, $path);
$instances[$adapter] = new $class;
return $instances[$adapter];
}
/**
* Method to stem a token and return the root.
*
* @param string $token The token to stem.
* @param string $lang The language of the token.
*
* @return string The root token.
*
* @since 2.5
*/
abstract public function stem($token, $lang);
}
PK"�[̝Vqj'j'helpers/indexer/taxonomy.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Stemmer base class for the Finder indexer package.
*
* @since 2.5
*/
class FinderIndexerTaxonomy
{
/**
* An internal cache of taxonomy branch data.
*
* @var array
* @since 2.5
*/
public static $branches = array();
/**
* An internal cache of taxonomy node data.
*
* @var array
* @since 2.5
*/
public static $nodes = array();
/**
* Method to add a branch to the taxonomy tree.
*
* @param string $title The title of the branch.
* @param integer $state The published state of the branch.
[optional]
* @param integer $access The access state of the branch. [optional]
*
* @return integer The id of the branch.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function addBranch($title, $state = 1, $access = 1)
{
// Check to see if the branch is in the cache.
if (isset(static::$branches[$title]))
{
return static::$branches[$title]->id;
}
// Check to see if the branch is in the table.
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select('*')
->from($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('parent_id') . ' = 1')
->where($db->quoteName('title') . ' = ' .
$db->quote($title));
$db->setQuery($query);
// Get the result.
$result = $db->loadObject();
// Check if the database matches the input data.
if ((bool) $result && $result->state == $state &&
$result->access == $access)
{
// The data matches, add the item to the cache.
static::$branches[$title] = $result;
return static::$branches[$title]->id;
}
/*
* The database did not match the input. This could be because the
* state has changed or because the branch does not exist. Let's
figure
* out which case is true and deal with it.
*/
$branch = new JObject;
if (empty($result))
{
// Prepare the branch object.
$branch->parent_id = 1;
$branch->title = $title;
$branch->state = (int) $state;
$branch->access = (int) $access;
}
else
{
// Prepare the branch object.
$branch->id = (int) $result->id;
$branch->parent_id = (int) $result->parent_id;
$branch->title = $result->title;
$branch->state = (int) $result->title;
$branch->access = (int) $result->access;
$branch->ordering = (int) $result->ordering;
}
// Store the branch.
static::storeNode($branch);
// Add the branch to the cache.
static::$branches[$title] = $branch;
return static::$branches[$title]->id;
}
/**
* Method to add a node to the taxonomy tree.
*
* @param string $branch The title of the branch to store the node
in.
* @param string $title The title of the node.
* @param integer $state The published state of the node. [optional]
* @param integer $access The access state of the node. [optional]
*
* @return integer The id of the node.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function addNode($branch, $title, $state = 1, $access = 1)
{
// Check to see if the node is in the cache.
if (isset(static::$nodes[$branch][$title]))
{
return static::$nodes[$branch][$title]->id;
}
// Get the branch id, insert it if it does not exist.
$branchId = static::addBranch($branch);
// Check to see if the node is in the table.
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select('*')
->from($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('parent_id') . ' = ' .
$db->quote($branchId))
->where($db->quoteName('title') . ' = ' .
$db->quote($title));
$db->setQuery($query);
// Get the result.
$result = $db->loadObject();
// Check if the database matches the input data.
if ((bool) $result && $result->state == $state &&
$result->access == $access)
{
// The data matches, add the item to the cache.
static::$nodes[$branch][$title] = $result;
return static::$nodes[$branch][$title]->id;
}
/*
* The database did not match the input. This could be because the
* state has changed or because the node does not exist. Let's
figure
* out which case is true and deal with it.
*/
$node = new JObject;
if (empty($result))
{
// Prepare the node object.
$node->parent_id = (int) $branchId;
$node->title = $title;
$node->state = (int) $state;
$node->access = (int) $access;
}
else
{
// Prepare the node object.
$node->id = (int) $result->id;
$node->parent_id = (int) $result->parent_id;
$node->title = $result->title;
$node->state = (int) $result->title;
$node->access = (int) $result->access;
$node->ordering = (int) $result->ordering;
}
// Store the node.
static::storeNode($node);
// Add the node to the cache.
static::$nodes[$branch][$title] = $node;
return static::$nodes[$branch][$title]->id;
}
/**
* Method to add a map entry between a link and a taxonomy node.
*
* @param integer $linkId The link to map to.
* @param integer $nodeId The node to map to.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function addMap($linkId, $nodeId)
{
// Insert the map.
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select($db->quoteName('link_id'))
->from($db->quoteName('#__finder_taxonomy_map'))
->where($db->quoteName('link_id') . ' = ' .
(int) $linkId)
->where($db->quoteName('node_id') . ' = ' .
(int) $nodeId);
$db->setQuery($query);
$db->execute();
$id = (int) $db->loadResult();
$map = new JObject;
$map->link_id = (int) $linkId;
$map->node_id = (int) $nodeId;
if ($id)
{
$db->updateObject('#__finder_taxonomy_map', $map,
array('link_id', 'node_id'));
}
else
{
$db->insertObject('#__finder_taxonomy_map', $map);
}
return true;
}
/**
* Method to get the title of all taxonomy branches.
*
* @return array An array of branch titles.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function getBranchTitles()
{
$db = JFactory::getDbo();
// Set user variables
$groups = implode(',',
JFactory::getUser()->getAuthorisedViewLevels());
// Create a query to get the taxonomy branch titles.
$query = $db->getQuery(true)
->select($db->quoteName('title'))
->from($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('parent_id') . ' = 1')
->where($db->quoteName('state') . ' = 1')
->where($db->quoteName('access') . ' IN (' .
$groups . ')');
// Get the branch titles.
$db->setQuery($query);
return $db->loadColumn();
}
/**
* Method to find a taxonomy node in a branch.
*
* @param string $branch The branch to search.
* @param string $title The title of the node.
*
* @return mixed Integer id on success, null on no match.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function getNodeByTitle($branch, $title)
{
$db = JFactory::getDbo();
// Set user variables
$groups = implode(',',
JFactory::getUser()->getAuthorisedViewLevels());
// Create a query to get the node.
$query = $db->getQuery(true)
->select('t1.*')
->from($db->quoteName('#__finder_taxonomy') . ' AS
t1')
->join('INNER',
$db->quoteName('#__finder_taxonomy') . ' AS t2 ON t2.id =
t1.parent_id')
->where('t1.access IN (' . $groups . ')')
->where('t1.state = 1')
->where('t1.title LIKE ' .
$db->quote($db->escape($title) . '%'))
->where('t2.access IN (' . $groups . ')')
->where('t2.state = 1')
->where('t2.title = ' . $db->quote($branch));
// Get the node.
$db->setQuery($query, 0, 1);
return $db->loadObject();
}
/**
* Method to remove map entries for a link.
*
* @param integer $linkId The link to remove.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function removeMaps($linkId)
{
// Delete the maps.
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->delete($db->quoteName('#__finder_taxonomy_map'))
->where($db->quoteName('link_id') . ' = ' .
(int) $linkId);
$db->setQuery($query);
$db->execute();
return true;
}
/**
* Method to remove orphaned taxonomy nodes and branches.
*
* @return integer The number of deleted rows.
*
* @since 2.5
* @throws Exception on database error.
*/
public static function removeOrphanNodes()
{
// Delete all orphaned nodes.
$db = JFactory::getDbo();
$query = $db->getQuery(true);
$subquery = $db->getQuery(true);
$subquery1 = $db->getQuery(true);
$subquery1->select($db->quoteName('t.id'))
->from($db->quoteName('#__finder_taxonomy',
't'))
->join('LEFT',
$db->quoteName('#__finder_taxonomy_map', 'm') .
' ON ' . $db->quoteName('m.node_id') . '='
. $db->quoteName('t.id'))
->where($db->quoteName('t.parent_id') . ' > 1
')
->where($db->quoteName('m.link_id') . ' IS
NULL');
$subquery->select($db->quoteName('id'))
->from('(' . $subquery1 . ') temp');
$query->delete($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('id') . ' IN (' .
$subquery . ')');
$db->setQuery($query);
$db->execute();
return $db->getAffectedRows();
}
/**
* Method to store a node to the database. This method will accept either
a branch or a node.
*
* @param object $item The item to store.
*
* @return boolean True on success.
*
* @since 2.5
* @throws Exception on database error.
*/
protected static function storeNode($item)
{
$db = JFactory::getDbo();
// Check if we are updating or inserting the item.
if (empty($item->id))
{
// Insert the item.
$db->insertObject('#__finder_taxonomy', $item,
'id');
}
else
{
// Update the item.
$db->updateObject('#__finder_taxonomy', $item,
'id');
}
return true;
}
}
PK"�[�j���helpers/indexer/token.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\String\StringHelper;
/**
* Token class for the Finder indexer package.
*
* @since 2.5
*/
class FinderIndexerToken
{
/**
* This is the term that will be referenced in the terms table and the
* mapping tables.
*
* @var string
* @since 2.5
*/
public $term;
/**
* The stem is used to match the root term and produce more potential
* matches when searching the index.
*
* @var string
* @since 2.5
*/
public $stem;
/**
* If the token is numeric, it is likely to be short and uncommon so the
* weight is adjusted to compensate for that situation.
*
* @var boolean
* @since 2.5
*/
public $numeric;
/**
* If the token is a common term, the weight is adjusted to compensate for
* the higher frequency of the term in relation to other terms.
*
* @var boolean
* @since 2.5
*/
public $common;
/**
* Flag for phrase tokens.
*
* @var boolean
* @since 2.5
*/
public $phrase;
/**
* The length is used to calculate the weight of the token.
*
* @var integer
* @since 2.5
*/
public $length;
/**
* The weight is calculated based on token size and whether the token is
* considered a common term.
*
* @var integer
* @since 2.5
*/
public $weight;
/**
* The simple language identifier for the token.
*
* @var string
* @since 2.5
*/
public $language;
/**
* The container for matches.
*
* @var array
* @since 3.8.12
*/
public $matches = array();
/**
* Is derived token (from individual words)
*
* @var boolean
* @since 3.8.12
*/
public $derived;
/**
* The suggested term
*
* @var string
* @since 3.8.12
*/
public $suggestion;
/**
* Method to construct the token object.
*
* @param mixed $term The term as a string for words or an array
for phrases.
* @param string $lang The simple language identifier.
* @param string $spacer The space separator for phrases. [optional]
*
* @since 2.5
*/
public function __construct($term, $lang, $spacer = ' ')
{
$this->language = $lang;
// Tokens can be a single word or an array of words representing a
phrase.
if (is_array($term))
{
// Populate the token instance.
$this->term = implode($spacer, $term);
$this->stem = implode($spacer,
array_map(array('FinderIndexerHelper', 'stem'), $term,
array($lang)));
$this->numeric = false;
$this->common = false;
$this->phrase = true;
$this->length = StringHelper::strlen($this->term);
/*
* Calculate the weight of the token.
*
* 1. Length of the token up to 30 and divide by 30, add 1.
* 2. Round weight to 4 decimal points.
*/
$this->weight = (($this->length >= 30 ? 30 : $this->length)
/ 30) + 1;
$this->weight = round($this->weight, 4);
}
else
{
// Populate the token instance.
$this->term = $term;
$this->stem = FinderIndexerHelper::stem($this->term, $lang);
$this->numeric = (is_numeric($this->term) || (bool)
preg_match('#^[0-9,.\-\+]+$#', $this->term));
$this->common = $this->numeric ? false :
FinderIndexerHelper::isCommon($this->term, $lang);
$this->phrase = false;
$this->length = StringHelper::strlen($this->term);
/*
* Calculate the weight of the token.
*
* 1. Length of the token up to 15 and divide by 15.
* 2. If common term, divide weight by 8.
* 3. If numeric, multiply weight by 1.5.
* 4. Round weight to 4 decimal points.
*/
$this->weight = ($this->length >= 15 ? 15 : $this->length) /
15;
$this->weight = $this->common === true ? $this->weight / 8 :
$this->weight;
$this->weight = $this->numeric === true ? $this->weight * 1.5 :
$this->weight;
$this->weight = round($this->weight, 4);
}
}
}
PK"�[(��~##helpers/language.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Finder language helper class.
*
* @since 2.5
*/
class FinderHelperLanguage
{
/**
* Method to return a plural language code for a taxonomy branch.
*
* @param string $branchName Branch title.
*
* @return string Language key code.
*
* @since 2.5
*/
public static function branchPlural($branchName)
{
$return = preg_replace('/[^a-zA-Z0-9]+/', '_',
strtoupper($branchName));
if ($return !== '_')
{
return 'PLG_FINDER_QUERY_FILTER_BRANCH_P_' . $return;
}
return $branchName;
}
/**
* Method to return a singular language code for a taxonomy branch.
*
* @param string $branchName Branch name.
*
* @return string Language key code.
*
* @since 2.5
*/
public static function branchSingular($branchName)
{
$return = preg_replace('/[^a-zA-Z0-9]+/', '_',
strtoupper($branchName));
return 'PLG_FINDER_QUERY_FILTER_BRANCH_S_' . $return;
}
/**
* Method to return the language name for a language taxonomy branch.
*
* @param string $branchName Language branch name.
*
* @return string The language title.
*
* @since 3.6.0
*/
public static function branchLanguageTitle($branchName)
{
$title = $branchName;
if ($branchName === '*')
{
$title = JText::_('JALL_LANGUAGE');
}
else
{
$languages = JLanguageHelper::getLanguages('lang_code');
if (isset($languages[$branchName]))
{
$title = $languages[$branchName]->title;
}
}
return $title;
}
/**
* Method to load Smart Search component language file.
*
* @return void
*
* @since 2.5
*/
public static function loadComponentLanguage()
{
JFactory::getLanguage()->load('com_finder', JPATH_SITE);
}
/**
* Method to load Smart Search plugin language files.
*
* @return void
*
* @since 2.5
*/
public static function loadPluginLanguage()
{
static $loaded = false;
// If already loaded, don't load again.
if ($loaded)
{
return;
}
$loaded = true;
// Get array of all the enabled Smart Search plugin names.
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select(array($db->qn('name'),
$db->qn('element')))
->from($db->quoteName('#__extensions'))
->where($db->quoteName('type') . ' = ' .
$db->quote('plugin'))
->where($db->quoteName('folder') . ' = ' .
$db->quote('finder'))
->where($db->quoteName('enabled') . ' = 1');
$db->setQuery($query);
$plugins = $db->loadObjectList();
if (empty($plugins))
{
return;
}
// Load generic language strings.
$lang = JFactory::getLanguage();
$lang->load('plg_content_finder', JPATH_ADMINISTRATOR);
// Load language file for each plugin.
foreach ($plugins as $plugin)
{
$lang->load($plugin->name, JPATH_ADMINISTRATOR)
|| $lang->load($plugin->name, JPATH_PLUGINS .
'/finder/' . $plugin->element);
}
}
}
PK"�[�x���models/fields/branches.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('JPATH_BASE') or die();
JFormHelper::loadFieldClass('list');
/**
* Search Branches field for the Finder package.
*
* @since 3.5
*/
class JFormFieldBranches extends JFormFieldList
{
/**
* The form field type.
*
* @var string
* @since 3.5
*/
protected $type = 'Branches';
/**
* Method to get the field options.
*
* @return array The field option objects.
*
* @since 3.5
*/
public function getOptions()
{
return JHtml::_('finder.mapslist');
}
}
PK"�[F�=�zzmodels/fields/contentmap.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('JPATH_PLATFORM') or die;
JFormHelper::loadFieldClass('groupedlist');
JLoader::register('FinderHelperLanguage', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/language.php');
/**
* Supports a select grouped list of finder content map.
*
* @since 3.6.0
*/
class JFormFieldContentMap extends JFormFieldGroupedList
{
/**
* The form field type.
*
* @var string
* @since 3.6.0
*/
public $type = 'ContentMap';
/**
* Method to get the list of content map options grouped by first level.
*
* @return array The field option objects as a nested array in groups.
*
* @since 3.6.0
*/
protected function getGroups()
{
$groups = array();
// Get the database object and a new query object.
$db = JFactory::getDbo();
// Levels subquery.
$levelQuery = $db->getQuery(true);
$levelQuery->select('title AS branch_title, 1 as level')
->select($db->quoteName('id'))
->from($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('parent_id') . ' = 1');
$levelQuery2 = $db->getQuery(true);
$levelQuery2->select('b.title AS branch_title, 2 as level')
->select($db->quoteName('a.id'))
->from($db->quoteName('#__finder_taxonomy',
'a'))
->join('LEFT',
$db->quoteName('#__finder_taxonomy', 'b') . '
ON ' . $db->qn('a.parent_id') . ' = ' .
$db->qn('b.id'))
->where($db->quoteName('a.parent_id') . ' NOT IN
(0, 1)');
$levelQuery->union($levelQuery2);
// Main query.
$query = $db->getQuery(true)
->select($db->quoteName('a.title', 'text'))
->select($db->quoteName('a.id', 'value'))
->select($db->quoteName('d.level'))
->from($db->quoteName('#__finder_taxonomy',
'a'))
->join('LEFT', '(' . $levelQuery . ') AS d
ON ' . $db->qn('d.id') . ' = ' .
$db->qn('a.id'))
->where($db->quoteName('a.parent_id') . ' <>
0')
->order('d.branch_title ASC, d.level ASC, a.title ASC');
$db->setQuery($query);
try
{
$contentMap = $db->loadObjectList();
}
catch (RuntimeException $e)
{
return;
}
// Build the grouped list array.
if ($contentMap)
{
$lang = JFactory::getLanguage();
foreach ($contentMap as $branch)
{
if ((int) $branch->level === 1)
{
$name = $branch->text;
}
else
{
$levelPrefix = str_repeat('- ', max(0, $branch->level -
1));
if (trim($name, '**') === 'Language')
{
$text = FinderHelperLanguage::branchLanguageTitle($branch->text);
}
else
{
$key = FinderHelperLanguage::branchSingular($branch->text);
$text = $lang->hasKey($key) ? JText::_($key) : $branch->text;
}
// Initialize the group if necessary.
if (!isset($groups[$name]))
{
$groups[$name] = array();
}
$groups[$name][] = JHtml::_('select.option',
$branch->value, $levelPrefix . $text);
}
}
}
// Merge any additional groups in the XML definition.
$groups = array_merge(parent::getGroups(), $groups);
return $groups;
}
}
PK"�[KY!��models/fields/contenttypes.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('JPATH_BASE') or die();
use Joomla\Utilities\ArrayHelper;
JLoader::register('FinderHelperLanguage', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/language.php');
JFormHelper::loadFieldClass('list');
/**
* Content Types Filter field for the Finder package.
*
* @since 3.6.0
*/
class JFormFieldContentTypes extends JFormFieldList
{
/**
* The form field type.
*
* @var string
* @since 3.6.0
*/
protected $type = 'ContentTypes';
/**
* Method to get the field options.
*
* @return array The field option objects.
*
* @since 3.6.0
*/
public function getOptions()
{
$lang = JFactory::getLanguage();
$options = array();
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select($db->quoteName('id', 'value'))
->select($db->quoteName('title', 'text'))
->from($db->quoteName('#__finder_types'));
// Get the options.
$db->setQuery($query);
try
{
$contentTypes = $db->loadObjectList();
}
catch (RuntimeException $e)
{
JError::raiseWarning(500, $db->getMessage());
}
// Translate.
foreach ($contentTypes as $contentType)
{
$key = FinderHelperLanguage::branchSingular($contentType->text);
$contentType->translatedText = $lang->hasKey($key) ?
JText::_($key) : $contentType->text;
}
// Order by title.
$contentTypes = ArrayHelper::sortObjects($contentTypes,
'translatedText', 1, true, true);
// Convert the values to options.
foreach ($contentTypes as $contentType)
{
$options[] = JHtml::_('select.option', $contentType->value,
$contentType->translatedText);
}
// Merge any additional options in the XML definition.
$options = array_merge(parent::getOptions(), $options);
return $options;
}
}
PK"�[�����models/fields/directories.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
// Load the base adapter.
JLoader::register('FinderIndexerAdapter', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/indexer/adapter.php');
JFormHelper::loadFieldClass('list');
/**
* Renders a list of directories.
*
* @since 2.5
* @deprecated 4.0 Use JFormFieldFolderlist
*/
class JFormFieldDirectories extends JFormFieldList
{
/**
* The form field type.
*
* @var string
* @since 2.5
*/
protected $type = 'Directories';
/**
* Method to get the field options.
*
* @return array The field option objects.
*
* @since 2.5
*/
public function getOptions()
{
$values = array();
$options = array();
$exclude = array(
JPATH_ADMINISTRATOR,
JPATH_INSTALLATION,
JPATH_LIBRARIES,
JPATH_PLUGINS,
JPATH_SITE . '/cache',
JPATH_SITE . '/components',
JPATH_SITE . '/includes',
JPATH_SITE . '/language',
JPATH_SITE . '/modules',
JPATH_THEMES,
JFactory::getApplication()->get('log_path'),
JFactory::getApplication()->get('tmp_path')
);
// Get the base directories.
jimport('joomla.filesystem.folder');
$dirs = JFolder::folders(JPATH_SITE, '.', false, true);
// Iterate through the base directories and find the subdirectories.
foreach ($dirs as $dir)
{
// Check if the directory should be excluded.
if (in_array($dir, $exclude))
{
continue;
}
// Get the child directories.
$return = JFolder::folders($dir, '.', true, true);
// Merge the directories.
if (is_array($return))
{
$values[] = $dir;
$values = array_merge($values, $return);
}
}
// Convert the values to options.
foreach ($values as $value)
{
$options[] = JHtml::_('select.option', str_replace(JPATH_SITE
. '/', '', $value), str_replace(JPATH_SITE .
'/', '', $values));
}
// Add a null option.
array_unshift($options, JHtml::_('select.option', '',
'- ' . JText::_('JNONE') . ' -'));
return $options;
}
}
PK"�[�K
��models/fields/searchfilter.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('JPATH_BASE') or die();
JFormHelper::loadFieldClass('list');
/**
* Search Filter field for the Finder package.
*
* @since 2.5
*/
class JFormFieldSearchFilter extends JFormFieldList
{
/**
* The form field type.
*
* @var string
* @since 2.5
*/
protected $type = 'SearchFilter';
/**
* Method to get the field options.
*
* @return array The field option objects.
*
* @since 2.5
*/
public function getOptions()
{
// Build the query.
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select('f.title AS text, f.filter_id AS value')
->from($db->quoteName('#__finder_filters') . ' AS
f')
->where('f.state = 1')
->order('f.title ASC');
$db->setQuery($query);
$options = $db->loadObjectList();
array_unshift($options, JHtml::_('select.option', '',
JText::_('COM_FINDER_SELECT_SEARCH_FILTER'), 'value',
'text'));
return $options;
}
}
PK"�[^o D}}models/filter.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Filter model class for Finder.
*
* @since 2.5
*/
class FinderModelFilter extends JModelAdmin
{
/**
* The prefix to use with controller messages.
*
* @var string
* @since 2.5
*/
protected $text_prefix = 'COM_FINDER';
/**
* Model context string.
*
* @var string
* @since 2.5
*/
protected $context = 'com_finder.filter';
/**
* Custom clean cache method.
*
* @param string $group The component name. [optional]
* @param integer $clientId The client ID. [optional]
*
* @return void
*
* @since 2.5
*/
protected function cleanCache($group = 'com_finder', $clientId =
1)
{
parent::cleanCache($group, $clientId);
}
/**
* Method to get the filter data.
*
* @return FinderTableFilter|boolean The filter data or false on a
failure.
*
* @since 2.5
*/
public function getFilter()
{
$filter_id = (int) $this->getState('filter.id');
// Get a FinderTableFilter instance.
$filter = $this->getTable();
// Attempt to load the row.
$return = $filter->load($filter_id);
// Check for a database error.
if ($return === false && $filter->getError())
{
$this->setError($filter->getError());
return false;
}
// Process the filter data.
if (!empty($filter->data))
{
$filter->data = explode(',', $filter->data);
}
elseif (empty($filter->data))
{
$filter->data = array();
}
// Check for a database error.
if ($this->_db->getErrorNum())
{
$this->setError($this->_db->getErrorMsg());
return false;
}
return $filter;
}
/**
* Method to get the record form.
*
* @param array $data Data for the form. [optional]
* @param boolean $loadData True if the form is to load its own data
(default case), false if not. [optional]
*
* @return JForm|boolean A JForm object on success, false on failure
*
* @since 2.5
*/
public function getForm($data = array(), $loadData = true)
{
// Get the form.
$form = $this->loadForm('com_finder.filter',
'filter', array('control' => 'jform',
'load_data' => $loadData));
if (empty($form))
{
return false;
}
return $form;
}
/**
* Returns a JTable object, always creating it.
*
* @param string $type The table type to instantiate. [optional]
* @param string $prefix A prefix for the table class name. [optional]
* @param array $config Configuration array for model. [optional]
*
* @return JTable A database object
*
* @since 2.5
*/
public function getTable($type = 'Filter', $prefix =
'FinderTable', $config = array())
{
return JTable::getInstance($type, $prefix, $config);
}
/**
* Method to get the data that should be injected in the form.
*
* @return mixed The data for the form.
*
* @since 2.5
*/
protected function loadFormData()
{
// Check the session for previously entered form data.
$data =
JFactory::getApplication()->getUserState('com_finder.edit.filter.data',
array());
if (empty($data))
{
$data = $this->getItem();
}
$this->preprocessData('com_finder.filter', $data);
return $data;
}
/**
* Method to get the total indexed items
*
* @return number the number of indexed items
*
* @since 3.5
*/
public function getTotal()
{
$db = JFactory::getDbo();
$query = $db->getQuery(true)
->select('MAX(link_id)')
->from('#__finder_links');
return $db->setQuery($query)->loadResult();
}
}
PK"�[[�l.models/filters.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Filters model class for Finder.
*
* @since 2.5
*/
class FinderModelFilters extends JModelList
{
/**
* Constructor.
*
* @param array $config An associative array of configuration
settings. [optional]
*
* @since 2.5
* @see JControllerLegacy
*/
public function __construct($config = array())
{
if (empty($config['filter_fields']))
{
$config['filter_fields'] = array(
'filter_id', 'a.filter_id',
'title', 'a.title',
'state', 'a.state',
'created_by_alias', 'a.created_by_alias',
'created', 'a.created',
'map_count', 'a.map_count'
);
}
parent::__construct($config);
}
/**
* Build an SQL query to load the list data.
*
* @return JDatabaseQuery A JDatabaseQuery object
*
* @since 2.5
*/
protected function getListQuery()
{
$db = $this->getDbo();
$query = $db->getQuery(true);
// Select all fields from the table.
$query->select('a.*')
->from($db->quoteName('#__finder_filters',
'a'));
// Join over the users for the checked out user.
$query->select($db->quoteName('uc.name',
'editor'))
->join('LEFT', $db->quoteName('#__users',
'uc') . ' ON ' . $db->quoteName('uc.id') .
' = ' . $db->quoteName('a.checked_out'));
// Join over the users for the author.
$query->select($db->quoteName('ua.name',
'user_name'))
->join('LEFT', $db->quoteName('#__users',
'ua') . ' ON ' . $db->quoteName('ua.id') .
' = ' . $db->quoteName('a.created_by'));
// Check for a search filter.
if ($search = $this->getState('filter.search'))
{
$search = $db->quote('%' . str_replace(' ',
'%', $db->escape(trim($search), true) . '%'));
$query->where($db->quoteName('a.title') . ' LIKE
' . $search);
}
// If the model is set to check item state, add to the query.
$state = $this->getState('filter.state');
if (is_numeric($state))
{
$query->where($db->quoteName('a.state') . ' =
' . (int) $state);
}
// Add the list ordering clause.
$query->order($db->escape($this->getState('list.ordering',
'a.title') . ' ' .
$db->escape($this->getState('list.direction',
'ASC'))));
return $query;
}
/**
* Method to get a store id based on model configuration state.
*
* This is necessary because the model is used by the component and
* different modules that might need different sets of data or different
* ordering requirements.
*
* @param string $id A prefix for the store id. [optional]
*
* @return string A store id.
*
* @since 2.5
*/
protected function getStoreId($id = '')
{
// Compile the store id.
$id .= ':' . $this->getState('filter.search');
$id .= ':' . $this->getState('filter.state');
return parent::getStoreId($id);
}
/**
* Method to auto-populate the model state. Calling getState in this
method will result in recursion.
*
* @param string $ordering An optional ordering field. [optional]
* @param string $direction An optional direction. [optional]
*
* @return void
*
* @since 2.5
*/
protected function populateState($ordering = 'a.title',
$direction = 'asc')
{
// Load the filter state.
$this->setState('filter.search',
$this->getUserStateFromRequest($this->context .
'.filter.search', 'filter_search', '',
'string'));
$this->setState('filter.state',
$this->getUserStateFromRequest($this->context .
'.filter.state', 'filter_state', '',
'cmd'));
// Load the parameters.
$params = JComponentHelper::getParams('com_finder');
$this->setState('params', $params);
// List state information.
parent::populateState($ordering, $direction);
}
}
PK"�[?�=�jjmodels/forms/filter.xmlnu�[���<?xml
version="1.0" encoding="utf-8"?>
<form>
<fieldset>
<field
name="filter_id"
type="text"
label="JGLOBAL_FIELD_ID_LABEL"
description="JGLOBAL_FIELD_ID_DESC"
class="readonly"
size="10"
default="0"
readonly="true"
/>
<field
name="title"
type="text"
label="JGLOBAL_TITLE"
description="COM_FINDER_FILTER_TITLE_DESCRIPTION"
class="input-xxlarge input-large-text"
size="40"
id="title"
required="true"
/>
<field
name="alias"
type="text"
label="JFIELD_ALIAS_LABEL"
description="JFIELD_ALIAS_DESC"
hint="JFIELD_ALIAS_PLACEHOLDER"
size="45"
/>
<field
name="created"
type="calendar"
label="JGLOBAL_FIELD_CREATED_LABEL"
description="JGLOBAL_FIELD_CREATED_DESC"
translateformat="true"
showtime="true"
size="22"
filter="user_utc"
/>
<field
name="modified"
type="calendar"
label="JGLOBAL_FIELD_MODIFIED_LABEL"
description="COM_FINDER_FIELD_MODIFIED_DESCRIPTION"
class="readonly"
translateformat="true"
showtime="true"
size="22"
readonly="true"
filter="user_utc"
/>
<field
name="created_by"
type="user"
label="COM_FINDER_FIELD_CREATED_BY_LABEL"
description="COM_FINDER_FIELD_CREATED_BY_DESC"
/>
<field
name="created_by_alias"
type="text"
label="COM_FINDER_FIELD_CREATED_BY_ALIAS_LABEL"
description="COM_FINDER_FIELD_CREATED_BY_ALIAS_DESC"
size="20"
/>
<field
name="modified_by"
type="user"
label="JGLOBAL_FIELD_MODIFIED_BY_LABEL"
class="readonly"
readonly="true"
filter="unset"
/>
<field
name="checked_out"
type="hidden"
filter="unset"
/>
<field
name="checked_out_time"
type="hidden"
filter="unset"
/>
<field
name="state"
type="list"
label="JSTATUS"
description="JFIELD_PUBLISHED_DESC"
class="chzn-color-state"
filter="intval"
size="1"
default="1"
>
<option value="1">JPUBLISHED</option>
<option value="0">JUNPUBLISHED</option>
</field>
<field
name="map_count"
type="text"
label="COM_FINDER_FILTER_MAP_COUNT"
description="COM_FINDER_FILTER_MAP_COUNT_DESCRIPTION"
class="readonly"
size="10"
default="0"
readonly="true"
/>
</fieldset>
<fields name="params">
<fieldset name="jbasic"
label="COM_FINDER_FILTER_FIELDSET_PARAMS">
<field
name="w1"
type="list"
label="COM_FINDER_FILTER_WHEN_START_DATE_LABEL"
description="COM_FINDER_FILTER_WHEN_START_DATE_DESCRIPTION"
default=""
filter="string"
>
<option value="">JNONE</option>
<option
value="-1">COM_FINDER_FILTER_WHEN_BEFORE</option>
<option
value="0">COM_FINDER_FILTER_WHEN_EXACTLY</option>
<option
value="1">COM_FINDER_FILTER_WHEN_AFTER</option>
</field>
<field
name="d1"
type="calendar"
label="COM_FINDER_FILTER_START_DATE_LABEL"
description="COM_FINDER_FILTER_START_DATE_DESCRIPTION"
translateformat="true"
size="22"
filter="user_utc"
/>
<field
name="w2"
type="list"
label="COM_FINDER_FILTER_WHEN_END_DATE_LABEL"
description="COM_FINDER_FILTER_WHEN_END_DATE_DESCRIPTION"
default=""
filter="string"
>
<option value="">JNONE</option>
<option
value="-1">COM_FINDER_FILTER_WHEN_BEFORE</option>
<option
value="0">COM_FINDER_FILTER_WHEN_EXACTLY</option>
<option
value="1">COM_FINDER_FILTER_WHEN_AFTER</option>
</field>
<field
name="d2"
type="calendar"
label="COM_FINDER_FILTER_END_DATE_LABEL"
description="COM_FINDER_FILTER_END_DATE_DESCRIPTION"
translateformat="true"
size="22"
filter="user_utc"
/>
</fieldset>
</fields>
</form>
PK"�[�@*$��models/forms/filter_filters.xmlnu�[���<?xml
version="1.0" encoding="utf-8"?>
<form>
<fields name="filter">
<field
name="search"
type="text"
inputmode="search"
label="COM_FINDER_SEARCH_FILTER_SEARCH_LABEL"
description="COM_FINDER_SEARCH_FILTER_SEARCH_DESC"
hint="JSEARCH_FILTER"
/>
<field
name="state"
type="status"
label="COM_FINDER_FILTER_PUBLISHED"
description="COM_FINDER_FILTER_PUBLISHED_DESC"
filter="0,1"
onchange="this.form.submit();"
>
<option value="">JOPTION_SELECT_PUBLISHED</option>
</field>
</fields>
<fields name="list">
<field
name="fullordering"
type="list"
onchange="this.form.submit();"
default="a.title ASC"
validate="options"
>
<option value="">JGLOBAL_SORT_BY</option>
<option value="a.state ASC">JSTATUS_ASC</option>
<option value="a.state DESC">JSTATUS_DESC</option>
<option value="a.title
ASC">JGLOBAL_TITLE_ASC</option>
<option value="a.title
DESC">JGLOBAL_TITLE_DESC</option>
<option value="a.created_by_alias
ASC">COM_FINDER_HEADING_CREATED_BY_ASC</option>
<option value="a.created_by_alias
DESC">COM_FINDER_HEADING_CREATED_BY_DESC</option>
<option value="a.created
ASC">COM_FINDER_HEADING_CREATED_ON_ASC</option>
<option value="a.created
DESC">COM_FINDER_HEADING_CREATED_ON_DESC</option>
<option value="a.map_count
ASC">COM_FINDER_HEADING_MAP_COUNT_ASC</option>
<option value="a.map_count
DESC">COM_FINDER_HEADING_MAP_COUNT_DESC</option>
<option value="a.filter_id
ASC">JGRID_HEADING_ID_ASC</option>
<option value="a.filter_id
DESC">JGRID_HEADING_ID_DESC</option>
</field>
<field
name="limit"
type="limitbox"
class="input-mini"
default="25"
onchange="this.form.submit();"
/>
</fields>
</form>
PK"�[}SB((models/forms/filter_index.xmlnu�[���<?xml
version="1.0" encoding="utf-8"?>
<form>
<fields name="filter">
<field
name="search"
type="text"
inputmode="search"
label="COM_FINDER_INDEX_SEARCH_LABEL"
description="COM_FINDER_INDEX_SEARCH_DESC"
hint="JSEARCH_FILTER"
/>
<field
name="state"
type="status"
label="COM_FINDER_FILTER_PUBLISHED"
description="COM_FINDER_FILTER_PUBLISHED_DESC"
filter="0,1"
onchange="this.form.submit();"
>
<option value="">JOPTION_SELECT_PUBLISHED</option>
</field>
<field
name="type"
type="ContentTypes"
label="JOPTION_FILTER_CATEGORY"
description="JOPTION_FILTER_CATEGORY_DESC"
onchange="this.form.submit();"
>
<option
value="">COM_FINDER_MAPS_SELECT_TYPE</option>
</field>
<field
name="content_map"
type="ContentMap"
label="COM_FINDER_FILTER_CONTENT_MAP_LABEL"
description="COM_FINDER_FILTER_CONTENT_MAP_DESC"
onchange="this.form.submit();"
>
<option
value="">COM_FINDER_FILTER_SELECT_CONTENT_MAP</option>
</field>
</fields>
<fields name="list">
<field
name="fullordering"
type="list"
onchange="this.form.submit();"
default="l.title ASC"
validate="options"
>
<option value="">JGLOBAL_SORT_BY</option>
<option value="l.published
ASC">JSTATUS_ASC</option>
<option value="l.published
DESC">JSTATUS_DESC</option>
<option value="l.title
ASC">JGLOBAL_TITLE_ASC</option>
<option value="l.title
DESC">JGLOBAL_TITLE_DESC</option>
<option value="t.title
ASC">COM_FINDER_INDEX_HEADING_INDEX_TYPE_ASC</option>
<option value="t.title
DESC">COM_FINDER_INDEX_HEADING_INDEX_TYPE_DESC</option>
<option value="l.indexdate
ASC">COM_FINDER_INDEX_HEADING_INDEX_DATE_ASC</option>
<option value="l.indexdate
DESC">COM_FINDER_INDEX_HEADING_INDEX_DATE_DESC</option>
<option value="l.url
ASC">COM_FINDER_INDEX_HEADING_LINK_URL_ASC</option>
<option value="l.url
DESC">COM_FINDER_INDEX_HEADING_LINK_URL_DESC</option>
</field>
<field
name="limit"
type="limitbox"
class="input-mini"
default="25"
onchange="this.form.submit();"
/>
</fields>
</form>
PK"�[��o���models/forms/filter_maps.xmlnu�[���<?xml
version="1.0" encoding="utf-8"?>
<form>
<fields name="filter">
<field
name="search"
type="text"
inputmode="search"
label="COM_FINDER_SEARCH_SEARCH_QUERY_LABEL"
description="COM_FINDER_SEARCH_SEARCH_QUERY_DESC"
hint="JSEARCH_FILTER"
/>
<field
name="state"
type="status"
label="COM_FINDER_FILTER_PUBLISHED"
description="COM_FINDER_FILTER_PUBLISHED_DESC"
filter="0,1"
onchange="this.form.submit();"
>
<option value="">JOPTION_SELECT_PUBLISHED</option>
</field>
<field
name="branch"
type="branches"
default="0"
onchange="this.form.submit();"
/>
<field
name="level"
type="integer"
label="JOPTION_FILTER_LEVEL"
description="JOPTION_FILTER_LEVEL_DESC"
first="1"
last="2"
step="1"
languages="*"
onchange="this.form.submit();"
>
<option
value="">JOPTION_SELECT_MAX_LEVELS</option>
</field>
</fields>
<fields name="list">
<field
name="fullordering"
type="list"
onchange="this.form.submit();"
default="d.branch_title ASC"
validate="options"
>
<option value="">JGLOBAL_SORT_BY</option>
<option value="d.branch_title
ASC">JGLOBAL_TITLE_ASC</option>
<option value="d.branch_title
DESC">JGLOBAL_TITLE_DESC</option>
<option value="a.state ASC">JSTATUS_ASC</option>
<option value="a.state DESC">JSTATUS_DESC</option>
</field>
<field
name="limit"
type="limitbox"
class="input-mini"
default="25"
onchange="this.form.submit();"
/>
</fields>
</form>
PK"�[��-�-models/index.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Index model class for Finder.
*
* @since 2.5
*/
class FinderModelIndex extends JModelList
{
/**
* The event to trigger after deleting the data.
*
* @var string
* @since 2.5
*/
protected $event_after_delete = 'onContentAfterDelete';
/**
* The event to trigger before deleting the data.
*
* @var string
* @since 2.5
*/
protected $event_before_delete = 'onContentBeforeDelete';
/**
* Constructor.
*
* @param array $config An associative array of configuration
settings. [optional]
*
* @since 2.5
* @see JControllerLegacy
*/
public function __construct($config = array())
{
if (empty($config['filter_fields']))
{
$config['filter_fields'] = array(
'state', 'published', 'l.published',
'title', 'l.title',
'type', 'type_id', 'l.type_id',
't.title', 't_title',
'url', 'l.url',
'indexdate', 'l.indexdate',
'content_map',
);
}
parent::__construct($config);
}
/**
* Method to test whether a record can be deleted.
*
* @param object $record A record object.
*
* @return boolean True if allowed to delete the record. Defaults to the
permission for the component.
*
* @since 2.5
*/
protected function canDelete($record)
{
return JFactory::getUser()->authorise('core.delete',
$this->option);
}
/**
* Method to test whether a record can have its state changed.
*
* @param object $record A record object.
*
* @return boolean True if allowed to change the state of the record.
Defaults to the permission for the component.
*
* @since 2.5
*/
protected function canEditState($record)
{
return JFactory::getUser()->authorise('core.edit.state',
$this->option);
}
/**
* Method to delete one or more records.
*
* @param array $pks An array of record primary keys.
*
* @return boolean True if successful, false if an error occurs.
*
* @since 2.5
*/
public function delete(&$pks)
{
$dispatcher = JEventDispatcher::getInstance();
$pks = (array) $pks;
$table = $this->getTable();
// Include the content plugins for the on delete events.
JPluginHelper::importPlugin('content');
// Iterate the items to delete each one.
foreach ($pks as $i => $pk)
{
if ($table->load($pk))
{
if ($this->canDelete($table))
{
$context = $this->option . '.' . $this->name;
// Trigger the onContentBeforeDelete event.
$result = $dispatcher->trigger($this->event_before_delete,
array($context, $table));
if (in_array(false, $result, true))
{
$this->setError($table->getError());
return false;
}
if (!$table->delete($pk))
{
$this->setError($table->getError());
return false;
}
// Trigger the onContentAfterDelete event.
$dispatcher->trigger($this->event_after_delete, array($context,
$table));
}
else
{
// Prune items that you can't change.
unset($pks[$i]);
$error = $this->getError();
if ($error)
{
$this->setError($error);
}
else
{
$this->setError(JText::_('JLIB_APPLICATION_ERROR_DELETE_NOT_PERMITTED'));
}
}
}
else
{
$this->setError($table->getError());
return false;
}
}
// Clear the component's cache
$this->cleanCache();
return true;
}
/**
* Build an SQL query to load the list data.
*
* @return JDatabaseQuery A JDatabaseQuery object
*
* @since 2.5
*/
protected function getListQuery()
{
$db = $this->getDbo();
$query = $db->getQuery(true)
->select('l.*')
->select($db->quoteName('t.title', 't_title'))
->from($db->quoteName('#__finder_links', 'l'))
->join('INNER',
$db->quoteName('#__finder_types', 't') . ' ON
' . $db->quoteName('t.id') . ' = ' .
$db->quoteName('l.type_id'));
// Check the type filter.
$type = $this->getState('filter.type');
if (is_numeric($type))
{
$query->where($db->quoteName('l.type_id') . ' =
' . (int) $type);
}
// Check the map filter.
$contentMapId = $this->getState('filter.content_map');
if (is_numeric($contentMapId))
{
$query->join('INNER',
$db->quoteName('#__finder_taxonomy_map', 'm') .
' ON ' . $db->quoteName('m.link_id') . ' =
' . $db->quoteName('l.link_id'))
->where($db->quoteName('m.node_id') . ' = ' .
(int) $contentMapId);
}
// Check for state filter.
$state = $this->getState('filter.state');
if (is_numeric($state))
{
$query->where($db->quoteName('l.published') . ' =
' . (int) $state);
}
// Check the search phrase.
$search = $this->getState('filter.search');
if (!empty($search))
{
$search = $db->quote('%' . str_replace(' ',
'%', $db->escape(trim($search), true) . '%'));
$orSearchSql = $db->quoteName('l.title') . ' LIKE
' . $search . ' OR ' . $db->quoteName('l.url')
. ' LIKE ' . $search;
// Filter by indexdate only if $search doesn't contains non-ascii
characters
if (!preg_match('/[^\x00-\x7F]/', $search))
{
$orSearchSql .= ' OR ' .
$query->castAsChar($db->quoteName('l.indexdate')) . '
LIKE ' . $search;
}
$query->where('(' . $orSearchSql . ')');
}
// Handle the list ordering.
$listOrder = $this->getState('list.ordering',
'l.title');
$listDir = $this->getState('list.direction',
'ASC');
if ($listOrder === 't.title')
{
$ordering = $db->quoteName('t.title') . ' ' .
$db->escape($listDir) . ', ' .
$db->quoteName('l.title') . ' ' .
$db->escape($listDir);
}
else
{
$ordering = $db->escape($listOrder) . ' ' .
$db->escape($listDir);
}
$query->order($ordering);
return $query;
}
/**
* Method to get the state of the Smart Search Plugins.
*
* @return array Array of relevant plugins and whether they are enabled
or not.
*
* @since 2.5
*/
public function getPluginState()
{
$db = $this->getDbo();
$query = $db->getQuery(true)
->select('name, enabled')
->from($db->quoteName('#__extensions'))
->where($db->quoteName('type') . ' = ' .
$db->quote('plugin'))
->where($db->quoteName('folder') . ' IN (' .
$db->quote('system') . ',' .
$db->quote('content') . ')')
->where($db->quoteName('element') . ' = ' .
$db->quote('finder'));
$db->setQuery($query);
return $db->loadObjectList('name');
}
/**
* Method to get a store id based on model configuration state.
*
* This is necessary because the model is used by the component and
* different modules that might need different sets of data or different
* ordering requirements.
*
* @param string $id A prefix for the store id. [optional]
*
* @return string A store id.
*
* @since 2.5
*/
protected function getStoreId($id = '')
{
// Compile the store id.
$id .= ':' . $this->getState('filter.search');
$id .= ':' . $this->getState('filter.state');
$id .= ':' . $this->getState('filter.type');
$id .= ':' .
$this->getState('filter.content_map');
return parent::getStoreId($id);
}
/**
* Gets the total of indexed items.
*
* @return integer The total of indexed items.
*
* @since 3.6.0
*/
public function getTotalIndexed()
{
$db = $this->getDbo();
$query = $db->getQuery(true)
->select('COUNT(link_id)')
->from($db->quoteName('#__finder_links'));
$db->setQuery($query);
$db->execute();
return (int) $db->loadResult();
}
/**
* Returns a JTable object, always creating it.
*
* @param string $type The table type to instantiate. [optional]
* @param string $prefix A prefix for the table class name. [optional]
* @param array $config Configuration array for model. [optional]
*
* @return JTable A database object
*
* @since 2.5
*/
public function getTable($type = 'Link', $prefix =
'FinderTable', $config = array())
{
return JTable::getInstance($type, $prefix, $config);
}
/**
* Method to purge the index, deleting all links.
*
* @return boolean True on success, false on failure.
*
* @since 2.5
* @throws Exception on database error
*/
public function purge()
{
$db = $this->getDbo();
// Truncate the links table.
$db->truncateTable('#__finder_links');
// Truncate the links terms tables.
for ($i = 0; $i <= 15; $i++)
{
// Get the mapping table suffix.
$suffix = dechex($i);
$db->truncateTable('#__finder_links_terms' . $suffix);
}
// Truncate the terms table.
$db->truncateTable('#__finder_terms');
// Truncate the taxonomy map table.
$db->truncateTable('#__finder_taxonomy_map');
// Delete all the taxonomy nodes except the root.
$query = $db->getQuery(true)
->delete($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('id') . ' > 1');
$db->setQuery($query);
$db->execute();
// Truncate the tokens tables.
$db->truncateTable('#__finder_tokens');
// Truncate the tokens aggregate table.
$db->truncateTable('#__finder_tokens_aggregate');
return true;
}
/**
* Method to auto-populate the model state. Calling getState in this
method will result in recursion.
*
* @param string $ordering An optional ordering field. [optional]
* @param string $direction An optional direction. [optional]
*
* @return void
*
* @since 2.5
*/
protected function populateState($ordering = 'l.title',
$direction = 'asc')
{
// Load the filter state.
$this->setState('filter.search',
$this->getUserStateFromRequest($this->context .
'.filter.search', 'filter_search', '',
'string'));
$this->setState('filter.state',
$this->getUserStateFromRequest($this->context .
'.filter.state', 'filter_state', '',
'cmd'));
$this->setState('filter.type',
$this->getUserStateFromRequest($this->context .
'.filter.type', 'filter_type', '',
'cmd'));
$this->setState('filter.content_map',
$this->getUserStateFromRequest($this->context .
'.filter.content_map', 'filter_content_map',
'', 'cmd'));
// Load the parameters.
$params = JComponentHelper::getParams('com_finder');
$this->setState('params', $params);
// List state information.
parent::populateState($ordering, $direction);
}
/**
* Method to change the published state of one or more records.
*
* @param array $pks A list of the primary keys to change.
* @param integer $value The value of the published state. [optional]
*
* @return boolean True on success.
*
* @since 2.5
*/
public function publish(&$pks, $value = 1)
{
$dispatcher = JEventDispatcher::getInstance();
$user = JFactory::getUser();
$table = $this->getTable();
$pks = (array) $pks;
// Include the content plugins for the change of state event.
JPluginHelper::importPlugin('content');
// Access checks.
foreach ($pks as $i => $pk)
{
$table->reset();
if ($table->load($pk) && !$this->canEditState($table))
{
// Prune items that you can't change.
unset($pks[$i]);
$this->setError(JText::_('JLIB_APPLICATION_ERROR_EDITSTATE_NOT_PERMITTED'));
return false;
}
}
// Attempt to change the state of the records.
if (!$table->publish($pks, $value, $user->get('id')))
{
$this->setError($table->getError());
return false;
}
$context = $this->option . '.' . $this->name;
// Trigger the onContentChangeState event.
$result = $dispatcher->trigger('onContentChangeState',
array($context, $pks, $value));
if (in_array(false, $result, true))
{
$this->setError($table->getError());
return false;
}
// Clear the component's cache
$this->cleanCache();
return true;
}
}
PK"�[�r���models/indexer.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Indexer model class for Finder.
*
* @since 2.5
*/
class FinderModelIndexer extends JModelLegacy
{
}
PK"�[ᅑ�**models/maps.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die();
/**
* Maps model for the Finder package.
*
* @since 2.5
*/
class FinderModelMaps extends JModelList
{
/**
* Constructor.
*
* @param array $config An associative array of configuration
settings. [optional]
*
* @since 2.5
* @see JControllerLegacy
*/
public function __construct($config = array())
{
if (empty($config['filter_fields']))
{
$config['filter_fields'] = array(
'state', 'a.state',
'title', 'a.title',
'branch',
'branch_title', 'd.branch_title',
'level', 'd.level',
);
}
parent::__construct($config);
}
/**
* Method to test whether a record can be deleted.
*
* @param object $record A record object.
*
* @return boolean True if allowed to delete the record. Defaults to the
permission for the component.
*
* @since 2.5
*/
protected function canDelete($record)
{
return JFactory::getUser()->authorise('core.delete',
$this->option);
}
/**
* Method to test whether a record can have its state changed.
*
* @param object $record A record object.
*
* @return boolean True if allowed to change the state of the record.
Defaults to the permission for the component.
*
* @since 2.5
*/
protected function canEditState($record)
{
return JFactory::getUser()->authorise('core.edit.state',
$this->option);
}
/**
* Method to delete one or more records.
*
* @param array $pks An array of record primary keys.
*
* @return boolean True if successful, false if an error occurs.
*
* @since 2.5
*/
public function delete(&$pks)
{
$dispatcher = JEventDispatcher::getInstance();
$pks = (array) $pks;
$table = $this->getTable();
// Include the content plugins for the on delete events.
JPluginHelper::importPlugin('content');
// Iterate the items to delete each one.
foreach ($pks as $i => $pk)
{
if ($table->load($pk))
{
if ($this->canDelete($table))
{
$context = $this->option . '.' . $this->name;
// Trigger the onContentBeforeDelete event.
$result = $dispatcher->trigger('onContentBeforeDelete',
array($context, $table));
if (in_array(false, $result, true))
{
$this->setError($table->getError());
return false;
}
if (!$table->delete($pk))
{
$this->setError($table->getError());
return false;
}
// Trigger the onContentAfterDelete event.
$dispatcher->trigger('onContentAfterDelete',
array($context, $table));
}
else
{
// Prune items that you can't change.
unset($pks[$i]);
$error = $this->getError();
if ($error)
{
$this->setError($error);
}
else
{
$this->setError(JText::_('JLIB_APPLICATION_ERROR_DELETE_NOT_PERMITTED'));
}
}
}
else
{
$this->setError($table->getError());
return false;
}
}
// Clear the component's cache
$this->cleanCache();
return true;
}
/**
* Build an SQL query to load the list data.
*
* @return JDatabaseQuery A JDatabaseQuery object
*
* @since 2.5
*/
protected function getListQuery()
{
$db = $this->getDbo();
// Select all fields from the table.
$query = $db->getQuery(true)
->select('a.id, a.parent_id, a.title, a.state, a.access,
a.ordering')
->select('CASE WHEN a.parent_id = 1 THEN 1 ELSE 2 END AS
level')
->select('p.title AS parent_title')
->from($db->quoteName('#__finder_taxonomy',
'a'))
->leftJoin($db->quoteName('#__finder_taxonomy',
'p') . ' ON p.id = a.parent_id')
->where('a.parent_id != 0');
$childQuery = $db->getQuery(true)
->select('parent_id')
->select('COUNT(*) AS num_children')
->from($db->quoteName('#__finder_taxonomy'))
->where('parent_id != 0')
->group('parent_id');
// Join to get children.
$query->select('b.num_children');
$query->select('CASE WHEN a.parent_id = 1 THEN a.title ELSE
p.title END AS branch_title');
$query->leftJoin('(' . $childQuery . ') AS b ON
b.parent_id = a.id');
// Join to get the map links.
$stateQuery = $db->getQuery(true)
->select('m.node_id')
->select('COUNT(NULLIF(l.published, 0)) AS
count_published')
->select('COUNT(NULLIF(l.published, 1)) AS
count_unpublished')
->from($db->quoteName('#__finder_taxonomy_map',
'm'))
->leftJoin($db->quoteName('#__finder_links',
'l') . ' ON l.link_id = m.link_id')
->group('m.node_id');
$query->select('COALESCE(s.count_published, 0) AS
count_published');
$query->select('COALESCE(s.count_unpublished, 0) AS
count_unpublished');
$query->leftJoin('(' . $stateQuery . ') AS s ON
s.node_id = a.id');
// If the model is set to check item state, add to the query.
$state = $this->getState('filter.state');
if (is_numeric($state))
{
$query->where('a.state = ' . (int) $state);
}
// Filter over level.
$level = $this->getState('filter.level');
if (is_numeric($level) && (int) $level === 1)
{
$query->where('a.parent_id = 1');
}
// Filter the maps over the branch if set.
$branchId = $this->getState('filter.branch');
if (is_numeric($branchId))
{
$query->where('a.parent_id = ' . (int) $branchId);
}
// Filter the maps over the search string if set.
if ($search = $this->getState('filter.search'))
{
$search = $db->quote('%' . str_replace(' ',
'%', $db->escape(trim($search), true) . '%'));
$query->where('a.title LIKE ' . $search);
}
// Handle the list ordering.
$listOrdering = $this->getState('list.ordering',
'd.branch_title');
$listDirn = $this->getState('list.direction',
'ASC');
if ($listOrdering === 'd.branch_title')
{
$query->order("branch_title $listDirn, level ASC, a.title
$listDirn");
}
elseif ($listOrdering === 'a.state')
{
$query->order("a.state $listDirn, branch_title $listDirn, level
ASC");
}
return $query;
}
/**
* Returns a record count for the query.
*
* @param JDatabaseQuery|string $query The query.
*
* @return integer Number of rows for query.
*
* @since 3.0
*/
protected function _getListCount($query)
{
$query = clone $query;
$query->clear('select')->clear('join')->clear('order')->clear('limit')->clear('offset')->select('COUNT(*)');
return (int) $this->getDbo()->setQuery($query)->loadResult();
}
/**
* Method to get a store id based on model configuration state.
*
* This is necessary because the model is used by the component and
* different modules that might need different sets of data or different
* ordering requirements.
*
* @param string $id A prefix for the store id. [optional]
*
* @return string A store id.
*
* @since 2.5
*/
protected function getStoreId($id = '')
{
// Compile the store id.
$id .= ':' . $this->getState('filter.search');
$id .= ':' . $this->getState('filter.state');
$id .= ':' . $this->getState('filter.branch');
$id .= ':' . $this->getState('filter.level');
return parent::getStoreId($id);
}
/**
* Returns a JTable object, always creating it.
*
* @param string $type The table type to instantiate. [optional]
* @param string $prefix A prefix for the table class name. [optional]
* @param array $config Configuration array for model. [optional]
*
* @return JTable A database object
*
* @since 2.5
*/
public function getTable($type = 'Map', $prefix =
'FinderTable', $config = array())
{
return JTable::getInstance($type, $prefix, $config);
}
/**
* Method to auto-populate the model state. Calling getState in this
method will result in recursion.
*
* @param string $ordering An optional ordering field. [optional]
* @param string $direction An optional direction. [optional]
*
* @return void
*
* @since 2.5
*/
protected function populateState($ordering = 'd.branch_title',
$direction = 'ASC')
{
// Load the filter state.
$this->setState('filter.search',
$this->getUserStateFromRequest($this->context .
'.filter.search', 'filter_search', '',
'string'));
$this->setState('filter.state',
$this->getUserStateFromRequest($this->context .
'.filter.state', 'filter_state', '',
'cmd'));
$this->setState('filter.branch',
$this->getUserStateFromRequest($this->context .
'.filter.branch', 'filter_branch', '',
'cmd'));
$this->setState('filter.level',
$this->getUserStateFromRequest($this->context .
'.filter.level', 'filter_level', '',
'cmd'));
// Load the parameters.
$params = JComponentHelper::getParams('com_finder');
$this->setState('params', $params);
// List state information.
parent::populateState($ordering, $direction);
}
/**
* Method to change the published state of one or more records.
*
* @param array $pks A list of the primary keys to change.
* @param integer $value The value of the published state. [optional]
*
* @return boolean True on success.
*
* @since 2.5
*/
public function publish(&$pks, $value = 1)
{
$dispatcher = JEventDispatcher::getInstance();
$user = JFactory::getUser();
$table = $this->getTable();
$pks = (array) $pks;
// Include the content plugins for the change of state event.
JPluginHelper::importPlugin('content');
// Access checks.
foreach ($pks as $i => $pk)
{
$table->reset();
if ($table->load($pk) && !$this->canEditState($table))
{
// Prune items that you can't change.
unset($pks[$i]);
$this->setError(JText::_('JLIB_APPLICATION_ERROR_EDITSTATE_NOT_PERMITTED'));
return false;
}
}
// Attempt to change the state of the records.
if (!$table->publish($pks, $value, $user->get('id')))
{
$this->setError($table->getError());
return false;
}
$context = $this->option . '.' . $this->name;
// Trigger the onContentChangeState event.
$result = $dispatcher->trigger('onContentChangeState',
array($context, $pks, $value));
if (in_array(false, $result, true))
{
$this->setError($table->getError());
return false;
}
// Clear the component's cache
$this->cleanCache();
return true;
}
/**
* Method to purge all maps from the taxonomy.
*
* @return boolean Returns true on success, false on failure.
*
* @since 2.5
*/
public function purge()
{
$db = $this->getDbo();
$query = $db->getQuery(true)
->delete($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('parent_id') . ' >
1');
$db->setQuery($query);
$db->execute();
$query->clear()
->delete($db->quoteName('#__finder_taxonomy_map'));
$db->setQuery($query);
$db->execute();
return true;
}
}
PK"�[���VVmodels/statistics.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Statistics model class for Finder.
*
* @since 2.5
*/
class FinderModelStatistics extends JModelLegacy
{
/**
* Method to get the component statistics
*
* @return JObject The component statistics
*
* @since 2.5
*/
public function getData()
{
// Initialise
$db = $this->getDbo();
$query = $db->getQuery(true);
$data = new JObject;
$query->select('COUNT(term_id)')
->from($db->quoteName('#__finder_terms'));
$db->setQuery($query);
$data->term_count = $db->loadResult();
$query->clear()
->select('COUNT(link_id)')
->from($db->quoteName('#__finder_links'));
$db->setQuery($query);
$data->link_count = $db->loadResult();
$query->clear()
->select('COUNT(id)')
->from($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('parent_id') . ' = 1');
$db->setQuery($query);
$data->taxonomy_branch_count = $db->loadResult();
$query->clear()
->select('COUNT(id)')
->from($db->quoteName('#__finder_taxonomy'))
->where($db->quoteName('parent_id') . ' >
1');
$db->setQuery($query);
$data->taxonomy_node_count = $db->loadResult();
$query->clear()
->select('t.title AS type_title, COUNT(a.link_id) AS
link_count')
->from($db->quoteName('#__finder_links') . ' AS
a')
->join('INNER',
$db->quoteName('#__finder_types') . ' AS t ON t.id =
a.type_id')
->group('a.type_id, t.title')
->order($db->quoteName('type_title') . '
ASC');
$db->setQuery($query);
$data->type_list = $db->loadObjectList();
$lang = JFactory::getLanguage();
$plugins = JPluginHelper::getPlugin('finder');
foreach ($plugins as $plugin)
{
$lang->load('plg_finder_' . $plugin->name .
'.sys', JPATH_ADMINISTRATOR, null, false, true)
|| $lang->load('plg_finder_' . $plugin->name .
'.sys', JPATH_PLUGINS . '/finder/' . $plugin->name,
null, false, true);
}
return $data;
}
}
PK"�[��aa;a;sql/install.mysql.sqlnu�[���--
-- Table structure for table `#__finder_filters`
--
CREATE TABLE IF NOT EXISTS `#__finder_filters` (
`filter_id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`title` varchar(255) NOT NULL,
`alias` varchar(255) NOT NULL,
`state` tinyint(1) NOT NULL DEFAULT 1,
`created` datetime NOT NULL DEFAULT '0000-00-00 00:00:00',
`created_by` int(10) unsigned NOT NULL,
`created_by_alias` varchar(255) NOT NULL,
`modified` datetime NOT NULL DEFAULT '0000-00-00 00:00:00',
`modified_by` int(10) unsigned NOT NULL DEFAULT 0,
`checked_out` int(10) unsigned NOT NULL DEFAULT 0,
`checked_out_time` datetime NOT NULL DEFAULT '0000-00-00
00:00:00',
`map_count` int(10) unsigned NOT NULL DEFAULT 0,
`data` text NOT NULL,
`params` mediumtext,
PRIMARY KEY (`filter_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links`
--
CREATE TABLE IF NOT EXISTS `#__finder_links` (
`link_id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`url` varchar(255) NOT NULL,
`route` varchar(255) NOT NULL,
`title` varchar(400) DEFAULT NULL,
`description` varchar(255) DEFAULT NULL,
`indexdate` datetime NOT NULL DEFAULT '0000-00-00 00:00:00',
`md5sum` varchar(32) DEFAULT NULL,
`published` tinyint(1) NOT NULL DEFAULT 1,
`state` int(5) DEFAULT 1,
`access` int(5) DEFAULT 0,
`language` varchar(8) NOT NULL,
`publish_start_date` datetime NOT NULL DEFAULT '0000-00-00
00:00:00',
`publish_end_date` datetime NOT NULL DEFAULT '0000-00-00
00:00:00',
`start_date` datetime NOT NULL DEFAULT '0000-00-00 00:00:00',
`end_date` datetime NOT NULL DEFAULT '0000-00-00 00:00:00',
`list_price` double unsigned NOT NULL DEFAULT 0,
`sale_price` double unsigned NOT NULL DEFAULT 0,
`type_id` int(11) NOT NULL,
`object` mediumblob NOT NULL,
PRIMARY KEY (`link_id`),
KEY `idx_type` (`type_id`),
KEY `idx_title` (`title`(100)),
KEY `idx_md5` (`md5sum`),
KEY `idx_url` (`url`(75)),
KEY `idx_published_list`
(`published`,`state`,`access`,`publish_start_date`,`publish_end_date`,`list_price`),
KEY `idx_published_sale`
(`published`,`state`,`access`,`publish_start_date`,`publish_end_date`,`sale_price`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms0`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms0` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms1`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms1` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms2`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms2` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms3`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms3` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms4`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms4` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms5`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms5` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms6`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms6` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms7`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms7` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms8`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms8` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_terms9`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_terms9` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_termsa`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_termsa` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_termsb`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_termsb` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_termsc`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_termsc` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_termsd`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_termsd` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_termse`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_termse` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_links_termsf`
--
CREATE TABLE IF NOT EXISTS `#__finder_links_termsf` (
`link_id` int(10) unsigned NOT NULL,
`term_id` int(10) unsigned NOT NULL,
`weight` float unsigned NOT NULL,
PRIMARY KEY (`link_id`,`term_id`),
KEY `idx_term_weight` (`term_id`,`weight`),
KEY `idx_link_term_weight` (`link_id`,`term_id`,`weight`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_taxonomy`
--
CREATE TABLE IF NOT EXISTS `#__finder_taxonomy` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`parent_id` int(10) unsigned NOT NULL DEFAULT 0,
`title` varchar(255) NOT NULL,
`state` tinyint(1) unsigned NOT NULL DEFAULT 1,
`access` tinyint(1) unsigned NOT NULL DEFAULT 0,
`ordering` tinyint(1) unsigned NOT NULL DEFAULT 0,
PRIMARY KEY (`id`),
KEY `parent_id` (`parent_id`),
KEY `state` (`state`),
KEY `ordering` (`ordering`),
KEY `access` (`access`),
KEY `idx_parent_published` (`parent_id`,`state`,`access`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Dumping data for table `#__finder_taxonomy`
--
REPLACE INTO `#__finder_taxonomy` (`id`, `parent_id`, `title`, `state`,
`access`, `ordering`) VALUES
(1, 0, 'ROOT', 0, 0, 0);
--
-- Table structure for table `#__finder_taxonomy_map`
--
CREATE TABLE IF NOT EXISTS `#__finder_taxonomy_map` (
`link_id` int(10) unsigned NOT NULL,
`node_id` int(10) unsigned NOT NULL,
PRIMARY KEY (`link_id`,`node_id`),
KEY `link_id` (`link_id`),
KEY `node_id` (`node_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_terms`
--
CREATE TABLE IF NOT EXISTS `#__finder_terms` (
`term_id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`term` varchar(75) NOT NULL,
`stem` varchar(75) NOT NULL,
`common` tinyint(1) unsigned NOT NULL DEFAULT 0,
`phrase` tinyint(1) unsigned NOT NULL DEFAULT 0,
`weight` float unsigned NOT NULL DEFAULT 0,
`soundex` varchar(75) NOT NULL,
`links` int(10) NOT NULL DEFAULT 0,
`language` char(3) NOT NULL DEFAULT '',
PRIMARY KEY (`term_id`),
UNIQUE KEY `idx_term` (`term`),
KEY `idx_term_phrase` (`term`,`phrase`),
KEY `idx_stem_phrase` (`stem`,`phrase`),
KEY `idx_soundex_phrase` (`soundex`,`phrase`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_terms_common`
--
CREATE TABLE IF NOT EXISTS `#__finder_terms_common` (
`term` varchar(75) NOT NULL,
`language` varchar(3) NOT NULL,
KEY `idx_word_lang` (`term`,`language`),
KEY `idx_lang` (`language`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Dumping data for table `#__finder_terms_common`
--
REPLACE INTO `#__finder_terms_common` (`term`, `language`) VALUES
('a', 'en'),
('about', 'en'),
('after', 'en'),
('ago', 'en'),
('all', 'en'),
('am', 'en'),
('an', 'en'),
('and', 'en'),
('any', 'en'),
('are', 'en'),
('aren''t', 'en'),
('as', 'en'),
('at', 'en'),
('be', 'en'),
('but', 'en'),
('by', 'en'),
('for', 'en'),
('from', 'en'),
('get', 'en'),
('go', 'en'),
('how', 'en'),
('if', 'en'),
('in', 'en'),
('into', 'en'),
('is', 'en'),
('isn''t', 'en'),
('it', 'en'),
('its', 'en'),
('me', 'en'),
('more', 'en'),
('most', 'en'),
('must', 'en'),
('my', 'en'),
('new', 'en'),
('no', 'en'),
('none', 'en'),
('not', 'en'),
('nothing', 'en'),
('of', 'en'),
('off', 'en'),
('often', 'en'),
('old', 'en'),
('on', 'en'),
('onc', 'en'),
('once', 'en'),
('only', 'en'),
('or', 'en'),
('other', 'en'),
('our', 'en'),
('ours', 'en'),
('out', 'en'),
('over', 'en'),
('page', 'en'),
('she', 'en'),
('should', 'en'),
('small', 'en'),
('so', 'en'),
('some', 'en'),
('than', 'en'),
('thank', 'en'),
('that', 'en'),
('the', 'en'),
('their', 'en'),
('theirs', 'en'),
('them', 'en'),
('then', 'en'),
('there', 'en'),
('these', 'en'),
('they', 'en'),
('this', 'en'),
('those', 'en'),
('thus', 'en'),
('time', 'en'),
('times', 'en'),
('to', 'en'),
('too', 'en'),
('true', 'en'),
('under', 'en'),
('until', 'en'),
('up', 'en'),
('upon', 'en'),
('use', 'en'),
('user', 'en'),
('users', 'en'),
('version', 'en'),
('very', 'en'),
('via', 'en'),
('want', 'en'),
('was', 'en'),
('way', 'en'),
('were', 'en'),
('what', 'en'),
('when', 'en'),
('where', 'en'),
('which', 'en'),
('who', 'en'),
('whom', 'en'),
('whose', 'en'),
('why', 'en'),
('wide', 'en'),
('will', 'en'),
('with', 'en'),
('within', 'en'),
('without', 'en'),
('would', 'en'),
('yes', 'en'),
('yet', 'en'),
('you', 'en'),
('your', 'en'),
('yours', 'en');
--
-- Table structure for table `#__finder_tokens`
--
CREATE TABLE IF NOT EXISTS `#__finder_tokens` (
`term` varchar(75) NOT NULL,
`stem` varchar(75) NOT NULL,
`common` tinyint(1) unsigned NOT NULL DEFAULT 0,
`phrase` tinyint(1) unsigned NOT NULL DEFAULT 0,
`weight` float unsigned NOT NULL DEFAULT 1,
`context` tinyint(1) unsigned NOT NULL DEFAULT 2,
`language` char(3) NOT NULL DEFAULT '',
KEY `idx_word` (`term`),
KEY `idx_context` (`context`)
) ENGINE=MEMORY DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_tokens_aggregate`
--
CREATE TABLE IF NOT EXISTS `#__finder_tokens_aggregate` (
`term_id` int(10) unsigned NOT NULL,
`map_suffix` char(1) NOT NULL,
`term` varchar(75) NOT NULL,
`stem` varchar(75) NOT NULL,
`common` tinyint(1) unsigned NOT NULL DEFAULT 0,
`phrase` tinyint(1) unsigned NOT NULL DEFAULT 0,
`term_weight` float unsigned NOT NULL,
`context` tinyint(1) unsigned NOT NULL DEFAULT 2,
`context_weight` float unsigned NOT NULL,
`total_weight` float unsigned NOT NULL,
`language` char(3) NOT NULL DEFAULT '',
KEY `token` (`term`),
KEY `keyword_id` (`term_id`)
) ENGINE=MEMORY DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
--
-- Table structure for table `#__finder_types`
--
CREATE TABLE IF NOT EXISTS `#__finder_types` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`title` varchar(100) NOT NULL,
`mime` varchar(100) NOT NULL,
PRIMARY KEY (`id`),
UNIQUE KEY `title` (`title`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 DEFAULT COLLATE=utf8mb4_general_ci;
PK"�[�; ��sql/install.postgresql.sqlnu�[���--
-- Table: #__finder_filters
--
CREATE TABLE "#__finder_filters" (
"filter_id" serial NOT NULL,
"title" character varying(255) NOT NULL,
"alias" character varying(255) NOT NULL,
"state" smallint DEFAULT 1 NOT NULL,
"created" timestamp without time zone DEFAULT '1970-01-01
00:00:00' NOT NULL,
"created_by" integer NOT NULL,
"created_by_alias" character varying(255) NOT NULL,
"modified" timestamp without time zone DEFAULT '1970-01-01
00:00:00' NOT NULL,
"modified_by" integer DEFAULT 0 NOT NULL,
"checked_out" integer DEFAULT 0 NOT NULL,
"checked_out_time" timestamp without time zone DEFAULT
'1970-01-01 00:00:00' NOT NULL,
"map_count" integer DEFAULT 0 NOT NULL,
"data" text NOT NULL,
"params" text,
PRIMARY KEY ("filter_id")
);
--
-- Table: #__finder_links
--
CREATE TABLE "#__finder_links" (
"link_id" serial NOT NULL,
"url" character varying(255) NOT NULL,
"route" character varying(255) NOT NULL,
"title" character varying(255) DEFAULT NULL,
"description" character varying(255) DEFAULT NULL,
"indexdate" timestamp without time zone DEFAULT
'1970-01-01 00:00:00' NOT NULL,
"md5sum" character varying(32) DEFAULT NULL,
"published" smallint DEFAULT 1 NOT NULL,
"state" integer DEFAULT 1,
"access" integer DEFAULT 0,
"language" character varying(8) NOT NULL,
"publish_start_date" timestamp without time zone DEFAULT
'1970-01-01 00:00:00' NOT NULL,
"publish_end_date" timestamp without time zone DEFAULT
'1970-01-01 00:00:00' NOT NULL,
"start_date" timestamp without time zone DEFAULT
'1970-01-01 00:00:00' NOT NULL,
"end_date" timestamp without time zone DEFAULT '1970-01-01
00:00:00' NOT NULL,
"list_price" numeric(8,2) DEFAULT 0 NOT NULL,
"sale_price" numeric(8,2) DEFAULT 0 NOT NULL,
"type_id" bigint NOT NULL,
"object" bytea NOT NULL,
PRIMARY KEY ("link_id")
);
CREATE INDEX "#__finder_links_idx_type" on
"#__finder_links" ("type_id");
CREATE INDEX "#__finder_links_idx_title" on
"#__finder_links" ("title");
CREATE INDEX "#__finder_links_idx_md5" on
"#__finder_links" ("md5sum");
CREATE INDEX "#__finder_links_idx_url" on
"#__finder_links" (url(75));
CREATE INDEX "#__finder_links_idx_published_list" on
"#__finder_links" ("published", "state",
"access", "publish_start_date",
"publish_end_date", "list_price");
CREATE INDEX "#__finder_links_idx_published_sale" on
"#__finder_links" ("published", "state",
"access", "publish_start_date",
"publish_end_date", "sale_price");
--
-- Table: #__finder_links_terms0
--
CREATE TABLE "#__finder_links_terms0" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms0_idx_term_weight" on
"#__finder_links_terms0" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms0_idx_link_term_weight" on
"#__finder_links_terms0" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms1
--
CREATE TABLE "#__finder_links_terms1" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms1_idx_term_weight" on
"#__finder_links_terms1" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms1_idx_link_term_weight" on
"#__finder_links_terms1" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms2
--
CREATE TABLE "#__finder_links_terms2" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms2_idx_term_weight" on
"#__finder_links_terms2" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms2_idx_link_term_weight" on
"#__finder_links_terms2" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms3
--
CREATE TABLE "#__finder_links_terms3" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms3_idx_term_weight" on
"#__finder_links_terms3" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms3_idx_link_term_weight" on
"#__finder_links_terms3" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms4
--
CREATE TABLE "#__finder_links_terms4" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms4_idx_term_weight" on
"#__finder_links_terms4" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms4_idx_link_term_weight" on
"#__finder_links_terms4" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms5
--
CREATE TABLE "#__finder_links_terms5" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms5_idx_term_weight" on
"#__finder_links_terms5" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms5_idx_link_term_weight" on
"#__finder_links_terms5" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms6
--
CREATE TABLE "#__finder_links_terms6" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms6_idx_term_weight" on
"#__finder_links_terms6" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms6_idx_link_term_weight" on
"#__finder_links_terms6" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms7
--
CREATE TABLE "#__finder_links_terms7" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms7_idx_term_weight" on
"#__finder_links_terms7" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms7_idx_link_term_weight" on
"#__finder_links_terms7" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms8
--
CREATE TABLE "#__finder_links_terms8" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms8_idx_term_weight" on
"#__finder_links_terms8" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms8_idx_link_term_weight" on
"#__finder_links_terms8" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_terms9
--
CREATE TABLE "#__finder_links_terms9" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_terms9_idx_term_weight" on
"#__finder_links_terms9" ("term_id",
"weight");
CREATE INDEX "#__finder_links_terms9_idx_link_term_weight" on
"#__finder_links_terms9" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_termsa
--
CREATE TABLE "#__finder_links_termsa" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_termsa_idx_term_weight" on
"#__finder_links_termsa" ("term_id",
"weight");
CREATE INDEX "#__finder_links_termsa_idx_link_term_weight" on
"#__finder_links_termsa" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_termsb
--
CREATE TABLE "#__finder_links_termsb" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_termsb_idx_term_weight" on
"#__finder_links_termsb" ("term_id",
"weight");
CREATE INDEX "#__finder_links_termsb_idx_link_term_weight" on
"#__finder_links_termsb" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_termsc
--
CREATE TABLE "#__finder_links_termsc" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_termsc_idx_term_weight" on
"#__finder_links_termsc" ("term_id",
"weight");
CREATE INDEX "#__finder_links_termsc_idx_link_term_weight" on
"#__finder_links_termsc" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_termsd
--
CREATE TABLE "#__finder_links_termsd" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_termsd_idx_term_weight" on
"#__finder_links_termsd" ("term_id",
"weight");
CREATE INDEX "#__finder_links_termsd_idx_link_term_weight" on
"#__finder_links_termsd" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_termse
--
CREATE TABLE "#__finder_links_termse" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_termse_idx_term_weight" on
"#__finder_links_termse" ("term_id",
"weight");
CREATE INDEX "#__finder_links_termse_idx_link_term_weight" on
"#__finder_links_termse" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_links_termsf
--
CREATE TABLE "#__finder_links_termsf" (
"link_id" integer NOT NULL,
"term_id" integer NOT NULL,
"weight" numeric(8,2) NOT NULL,
PRIMARY KEY ("link_id", "term_id")
);
CREATE INDEX "#__finder_links_termsf_idx_term_weight" on
"#__finder_links_termsf" ("term_id",
"weight");
CREATE INDEX "#__finder_links_termsf_idx_link_term_weight" on
"#__finder_links_termsf" ("link_id",
"term_id", "weight");
--
-- Table: #__finder_taxonomy
--
CREATE TABLE "#__finder_taxonomy" (
"id" serial NOT NULL,
"parent_id" integer DEFAULT 0 NOT NULL,
"title" character varying(255) NOT NULL,
"state" smallint DEFAULT 1 NOT NULL,
"access" smallint DEFAULT 0 NOT NULL,
"ordering" smallint DEFAULT 0 NOT NULL,
PRIMARY KEY ("id")
);
CREATE INDEX "#__finder_taxonomy_parent_id" on
"#__finder_taxonomy" ("parent_id");
CREATE INDEX "#__finder_taxonomy_state" on
"#__finder_taxonomy" ("state");
CREATE INDEX "#__finder_taxonomy_ordering" on
"#__finder_taxonomy" ("ordering");
CREATE INDEX "#__finder_taxonomy_access" on
"#__finder_taxonomy" ("access");
CREATE INDEX "#__finder_taxonomy_idx_parent_published" on
"#__finder_taxonomy" ("parent_id", "state",
"access");
--
-- Dumping data for table #__finder_taxonomy
--
UPDATE "#__finder_taxonomy" SET ("id",
"parent_id", "title", "state",
"access", "ordering") = (1, 0, 'ROOT', 0, 0,
0)
WHERE "id"=1;
INSERT INTO "#__finder_taxonomy" ("id",
"parent_id", "title", "state",
"access", "ordering")
SELECT 1, 0, 'ROOT', 0, 0, 0 WHERE 1 NOT IN
(SELECT 1 FROM "#__finder_taxonomy" WHERE "id"=1);
--
-- Table: #__finder_taxonomy_map
--
CREATE TABLE "#__finder_taxonomy_map" (
"link_id" integer NOT NULL,
"node_id" integer NOT NULL,
PRIMARY KEY ("link_id", "node_id")
);
CREATE INDEX "#__finder_taxonomy_map_link_id" on
"#__finder_taxonomy_map" ("link_id");
CREATE INDEX "#__finder_taxonomy_map_node_id" on
"#__finder_taxonomy_map" ("node_id");
--
-- Table: #__finder_terms
--
CREATE TABLE "#__finder_terms" (
"term_id" serial NOT NULL,
"term" character varying(75) NOT NULL,
"stem" character varying(75) NOT NULL,
"common" smallint DEFAULT 0 NOT NULL,
"phrase" smallint DEFAULT 0 NOT NULL,
"weight" numeric(8,2) DEFAULT 0 NOT NULL,
"soundex" character varying(75) NOT NULL,
"links" integer DEFAULT 0 NOT NULL,
PRIMARY KEY ("term_id"),
CONSTRAINT "#__finder_terms_idx_term" UNIQUE ("term")
);
CREATE INDEX "#__finder_terms_idx_term_phrase" on
"#__finder_terms" ("term", "phrase");
CREATE INDEX "#__finder_terms_idx_stem_phrase" on
"#__finder_terms" ("stem", "phrase");
CREATE INDEX "#__finder_terms_idx_soundex_phrase" on
"#__finder_terms" ("soundex", "phrase");
--
-- Table: #__finder_terms_common
--
CREATE TABLE "#__finder_terms_common" (
"term" character varying(75) NOT NULL,
"language" character varying(3) NOT NULL
);
CREATE INDEX "#__finder_terms_common_idx_word_lang" on
"#__finder_terms_common" ("term",
"language");
CREATE INDEX "#__finder_terms_common_idx_lang" on
"#__finder_terms_common" ("language");
--
-- Dumping data for table `#__finder_terms_common`
--
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('a', 'en') WHERE
"term"='a';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'a', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='a');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('about', 'en') WHERE
"term"='about';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'about', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='about');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('after', 'en') WHERE
"term"='after';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'after', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='after');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('ago', 'en') WHERE
"term"='ago';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'ago', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='ago');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('all', 'en') WHERE
"term"='all';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'all', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='all');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('am', 'en') WHERE
"term"='am';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'am', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='am');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('an', 'en') WHERE
"term"='an';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'an', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='an');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('and', 'en') WHERE
"term"='and';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'and', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='and');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('any', 'en') WHERE
"term"='any';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'any', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='any');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('are', 'en') WHERE
"term"='are';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'are', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='are');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('aren''t', 'en')
WHERE "term"='aren''t';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'aren''t', 'en' WHERE 1 NOT IN (SELECT
1 FROM "#__finder_terms_common" WHERE
"term"='aren''t');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('as', 'en') WHERE
"term"='as';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'as', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='as');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('at', 'en') WHERE
"term"='at';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'at', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='at');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('be', 'en') WHERE
"term"='be';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'be', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='be');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('but', 'en') WHERE
"term"='but';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'but', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='but');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('by', 'en') WHERE
"term"='by';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'by', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='by');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('for', 'en') WHERE
"term"='for';
INSERT INTO "#__finder_terms_common" ("term",
"language") SELECT 'for', 'en' WHERE 1 NOT IN
(SELECT 1 FROM "#__finder_terms_common" WHERE
"term"='for');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('from', 'en') WHERE
"term"='from';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'from', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='from');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('get', 'en') WHERE
"term"='get';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'get', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='get');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('go', 'en') WHERE
"term"='go';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'go', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='go');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('how', 'en') WHERE
"term"='how';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'how', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='how');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('if', 'en') WHERE
"term"='if';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'if', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='if');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('in', 'en') WHERE
"term"='in';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'in', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='in');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('into', 'en') WHERE
"term"='into';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'into', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='into');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('is', 'en') WHERE
"term"='is';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'is', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='is');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('isn''t', 'en')
WHERE "term"='isn''t';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'isn''t', 'en' WHERE 1 NOT IN (SELECT
1 FROM "#__finder_terms_common" WHERE
"term"='isn''t');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('it', 'en') WHERE
"term"='it';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'it', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='it');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('its', 'en') WHERE
"term"='its';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'its', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='its');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('me', 'en') WHERE
"term"='me';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'me', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='me');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('more', 'en') WHERE
"term"='more';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'more', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='more');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('most', 'en') WHERE
"term"='most';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'most', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='most');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('must', 'en') WHERE
"term"='must';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'must', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='must');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('my', 'en') WHERE
"term"='my';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'my', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='my');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('new', 'en') WHERE
"term"='new';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'new', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='new');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('no', 'en') WHERE
"term"='no';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'no', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='no');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('none', 'en') WHERE
"term"='none';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'none', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='none');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('not', 'en') WHERE
"term"='not';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'not', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='not');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('nothing', 'en') WHERE
"term"='nothing';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'nothing', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='nothing');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('of', 'en') WHERE
"term"='of';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'of', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='of');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('off', 'en') WHERE
"term"='off';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'off', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='off');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('often', 'en') WHERE
"term"='often';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'often', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='often');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('old', 'en') WHERE
"term"='old';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'old', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='old');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('on', 'en') WHERE
"term"='on';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'on', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='on');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('onc', 'en') WHERE
"term"='onc';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'onc', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='onc');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('once', 'en') WHERE
"term"='once';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'once', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='once');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('only', 'en') WHERE
"term"='only';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'only', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='only');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('or', 'en') WHERE
"term"='or';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'or', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='or');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('other', 'en') WHERE
"term"='other';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'other', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='other');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('our', 'en') WHERE
"term"='our';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'our', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='our');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('ours', 'en') WHERE
"term"='ours';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'ours', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='ours');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('out', 'en') WHERE
"term"='out';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'out', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='out');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('over', 'en') WHERE
"term"='over';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'over', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='over');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('page', 'en') WHERE
"term"='page';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'page', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='page');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('she', 'en') WHERE
"term"='she';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'she', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='she');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('should', 'en') WHERE
"term"='should';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'should', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='should');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('small', 'en') WHERE
"term"='small';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'small', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='small');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('so', 'en') WHERE
"term"='so';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'so', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='so');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('some', 'en') WHERE
"term"='some';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'some', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='some');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('than', 'en') WHERE
"term"='than';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'than', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='than');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('thank', 'en') WHERE
"term"='thank';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'thank', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='thank');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('that', 'en') WHERE
"term"='that';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'that', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='that');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('the', 'en') WHERE
"term"='the';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'the', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='the');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('their', 'en') WHERE
"term"='their';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'their', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='their');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('theirs', 'en') WHERE
"term"='theirs';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'theirs', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='theirs');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('them', 'en') WHERE
"term"='them';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'them', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='them');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('then', 'en') WHERE
"term"='then';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'then', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='then');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('there', 'en') WHERE
"term"='there';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'there', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='there');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('these', 'en') WHERE
"term"='these';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'these', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='these');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('they', 'en') WHERE
"term"='they';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'they', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='they');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('this', 'en') WHERE
"term"='this';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'this', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='this');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('those', 'en') WHERE
"term"='those';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'those', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='those');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('thus', 'en') WHERE
"term"='thus';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'thus', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='thus');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('time', 'en') WHERE
"term"='time';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'time', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='time');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('times', 'en') WHERE
"term"='times';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'times', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='times');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('to', 'en') WHERE
"term"='to';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'to', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='to');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('too', 'en') WHERE
"term"='too';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'too', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='too');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('true', 'en') WHERE
"term"='true';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'true', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='true');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('under', 'en')WHERE
"term"='under';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'under', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='under');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('until', 'en') WHERE
"term"='until';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'until', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='until');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('up', 'en') WHERE
"term"='up';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'up', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='up');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('upon', 'en') WHERE
"term"='upon';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'upon', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='upon');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('use', 'en') WHERE
"term"='use';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'use', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='use');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('user', 'en') WHERE
"term"='user';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'user', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='user');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('users', 'en') WHERE
"term"='users';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'users', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='users');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('version', 'en') WHERE
"term"='version';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'version', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='version');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('very', 'en') WHERE
"term"='very';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'very', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='very');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('via', 'en') WHERE
"term"='via';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'via', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='via');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('want', 'en') WHERE
"term"='want';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'want', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='want');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('was', 'en') WHERE
"term"='was';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'was', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='was');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('way', 'en') WHERE
"term"='way';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'way', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='way');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('were', 'en') WHERE
"term"='were';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'were', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='were');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('what', 'en') WHERE
"term"='what';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'what', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='what');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('when', 'en') WHERE
"term"='when';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'when', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='when');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('where', 'en') WHERE
"term"='where';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'where', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='where');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('which', 'en') WHERE
"term"='which';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'which', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='which');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('who', 'en') WHERE
"term"='who';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'who', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='who');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('whom', 'en') WHERE
"term"='whom';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'whom', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='whom');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('whose', 'en') WHERE
"term"='whose';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'whose', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='whose');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('why', 'en') WHERE
"term"='why';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'why', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='why');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('wide', 'en') WHERE
"term"='wide';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'wide', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='wide');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('will', 'en') WHERE
"term"='will';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'will', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='will');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('with', 'en') WHERE
"term"='with';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'with', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='with');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('within', 'en') WHERE
"term"='within';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'within', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='within');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('without', 'en') WHERE
"term"='without';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'without', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='without');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('would', 'en') WHERE
"term"='would';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'would', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='would');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('yes', 'en') WHERE
"term"='yes';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'yes', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='yes');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('yet', 'en') WHERE
"term"='yet';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'yet', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='yet');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('you', 'en') WHERE
"term"='you';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'you', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE "term"='you');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('your', 'en') WHERE
"term"='your';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'your', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='your');
--
UPDATE "#__finder_terms_common" SET ("term",
"language") = ('yours', 'en') WHERE
"term"='yours';
INSERT INTO "#__finder_terms_common" ("term",
"language")
SELECT 'yours', 'en' WHERE 1 NOT IN (SELECT 1 FROM
"#__finder_terms_common" WHERE
"term"='yours');
--
-- Table: #__finder_tokens
--
CREATE TABLE "#__finder_tokens" (
"term" character varying(75) NOT NULL,
"stem" character varying(75) NOT NULL,
"common" smallint DEFAULT 0 NOT NULL,
"phrase" smallint DEFAULT 0 NOT NULL,
"weight" numeric(8,2) DEFAULT 1 NOT NULL,
"context" smallint DEFAULT 2 NOT NULL
);
CREATE INDEX "#__finder_tokens_idx_word" on
"#__finder_tokens" ("term");
CREATE INDEX "#__finder_tokens_idx_context" on
"#__finder_tokens" ("context");
--
-- Table: #__finder_tokens_aggregate
--
CREATE TABLE "#__finder_tokens_aggregate" (
"term_id" integer NOT NULL,
"map_suffix" character(1) NOT NULL,
"term" character varying(75) NOT NULL,
"stem" character varying(75) NOT NULL,
"common" smallint DEFAULT 0 NOT NULL,
"phrase" smallint DEFAULT 0 NOT NULL,
"term_weight" numeric(8,2) NOT NULL,
"context" smallint DEFAULT 2 NOT NULL,
"context_weight" numeric(8,2) NOT NULL,
"total_weight" numeric(8,2) NOT NULL
);
CREATE INDEX "#__finder_tokens_aggregate_token" on
"#__finder_tokens_aggregate" ("term");
CREATE INDEX "_#__finder_tokens_aggregate_keyword_id" on
"#__finder_tokens_aggregate" ("term_id");
--
-- Table: #__finder_types
--
CREATE TABLE "#__finder_types" (
"id" serial NOT NULL,
"title" character varying(100) NOT NULL,
"mime" character varying(100) NOT NULL,
PRIMARY KEY ("id"),
CONSTRAINT "#__finder_types_title" UNIQUE ("title")
);
PK"�[�1�iwwsql/uninstall.mysql.sqlnu�[���DROP
TABLE IF EXISTS `#__finder_filters`;
DROP TABLE IF EXISTS `#__finder_links`;
DROP TABLE IF EXISTS `#__finder_links_terms0`;
DROP TABLE IF EXISTS `#__finder_links_terms1`;
DROP TABLE IF EXISTS `#__finder_links_terms2`;
DROP TABLE IF EXISTS `#__finder_links_terms3`;
DROP TABLE IF EXISTS `#__finder_links_terms4`;
DROP TABLE IF EXISTS `#__finder_links_terms5`;
DROP TABLE IF EXISTS `#__finder_links_terms6`;
DROP TABLE IF EXISTS `#__finder_links_terms7`;
DROP TABLE IF EXISTS `#__finder_links_terms8`;
DROP TABLE IF EXISTS `#__finder_links_terms9`;
DROP TABLE IF EXISTS `#__finder_links_termsa`;
DROP TABLE IF EXISTS `#__finder_links_termsb`;
DROP TABLE IF EXISTS `#__finder_links_termsc`;
DROP TABLE IF EXISTS `#__finder_links_termsd`;
DROP TABLE IF EXISTS `#__finder_links_termse`;
DROP TABLE IF EXISTS `#__finder_links_termsf`;
DROP TABLE IF EXISTS `#__finder_taxonomy`;
DROP TABLE IF EXISTS `#__finder_taxonomy_map`;
DROP TABLE IF EXISTS `#__finder_terms`;
DROP TABLE IF EXISTS `#__finder_terms_common`;
DROP TABLE IF EXISTS `#__finder_tokens`;
DROP TABLE IF EXISTS `#__finder_tokens_aggregate`;
DROP TABLE IF EXISTS `#__finder_types`;
PK"�[�|��wwsql/uninstall.postgresql.sqlnu�[���DROP
TABLE IF EXISTS "#__finder_filters";
DROP TABLE IF EXISTS "#__finder_links";
DROP TABLE IF EXISTS "#__finder_links_terms0";
DROP TABLE IF EXISTS "#__finder_links_terms1";
DROP TABLE IF EXISTS "#__finder_links_terms2";
DROP TABLE IF EXISTS "#__finder_links_terms3";
DROP TABLE IF EXISTS "#__finder_links_terms4";
DROP TABLE IF EXISTS "#__finder_links_terms5";
DROP TABLE IF EXISTS "#__finder_links_terms6";
DROP TABLE IF EXISTS "#__finder_links_terms7";
DROP TABLE IF EXISTS "#__finder_links_terms8";
DROP TABLE IF EXISTS "#__finder_links_terms9";
DROP TABLE IF EXISTS "#__finder_links_termsa";
DROP TABLE IF EXISTS "#__finder_links_termsb";
DROP TABLE IF EXISTS "#__finder_links_termsc";
DROP TABLE IF EXISTS "#__finder_links_termsd";
DROP TABLE IF EXISTS "#__finder_links_termse";
DROP TABLE IF EXISTS "#__finder_links_termsf";
DROP TABLE IF EXISTS "#__finder_taxonomy";
DROP TABLE IF EXISTS "#__finder_taxonomy_map";
DROP TABLE IF EXISTS "#__finder_terms";
DROP TABLE IF EXISTS "#__finder_terms_common";
DROP TABLE IF EXISTS "#__finder_tokens";
DROP TABLE IF EXISTS "#__finder_tokens_aggregate";
DROP TABLE IF EXISTS "#__finder_types";
PK"�[h�܅��tables/filter.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\Registry\Registry;
use Joomla\Utilities\ArrayHelper;
/**
* Filter table class for the Finder package.
*
* @since 2.5
*/
class FinderTableFilter extends JTable
{
/**
* Constructor
*
* @param JDatabaseDriver $db JDatabaseDriver connector object.
*
* @since 2.5
*/
public function __construct(&$db)
{
parent::__construct('#__finder_filters', 'filter_id',
$db);
$this->setColumnAlias('published', 'state');
}
/**
* Method to bind an associative array or object to the JTable instance.
This
* method only binds properties that are publicly accessible and
optionally
* takes an array of properties to ignore when binding.
*
* @param array $array Named array
* @param mixed $ignore An optional array or space separated list of
properties
* to ignore while binding. [optional]
*
* @return mixed Null if operation was satisfactory, otherwise returns
an error string
*
* @since 2.5
*/
public function bind($array, $ignore = '')
{
if (isset($array['params']) &&
is_array($array['params']))
{
$registry = new Registry($array['params']);
$array['params'] = (string) $registry;
}
return parent::bind($array, $ignore);
}
/**
* Method to perform sanity checks on the JTable instance properties to
ensure
* they are safe to store in the database. Child classes should override
this
* method to make sure the data they are storing in the database is safe
and
* as expected before storage.
*
* @return boolean True if the instance is sane and able to be stored in
the database.
*
* @since 2.5
*/
public function check()
{
if (trim($this->alias) === '')
{
$this->alias = $this->title;
}
$this->alias = JApplicationHelper::stringURLSafe($this->alias);
if (trim(str_replace('-', '', $this->alias)) ===
'')
{
$this->alias =
JFactory::getDate()->format('Y-m-d-H-i-s');
}
$params = new Registry($this->params);
$nullDate = $this->_db->getNullDate();
$d1 = $params->get('d1', $nullDate);
$d2 = $params->get('d2', $nullDate);
// Check the end date is not earlier than the start date.
if ($d2 > $nullDate && $d2 < $d1)
{
// Swap the dates.
$params->set('d1', $d2);
$params->set('d2', $d1);
$this->params = (string) $params;
}
return true;
}
/**
* Method to set the publishing state for a row or list of rows in the
database
* table. The method respects checked out rows by other users and will
attempt
* to checkin rows that it can after adjustments are made.
*
* @param mixed $pks An array of primary key values to update.
If not
* set the instance property value is used.
[optional]
* @param integer $state The publishing state. eg. [0 = unpublished,
1 = published] [optional]
* @param integer $userId The user id of the user performing the
operation. [optional]
*
* @return boolean True on success.
*
* @since 2.5
*/
public function publish($pks = null, $state = 1, $userId = 0)
{
$k = $this->_tbl_key;
// Sanitize input.
$pks = ArrayHelper::toInteger($pks);
$userId = (int) $userId;
$state = (int) $state;
// If there are no primary keys set check to see if the instance key is
set.
if (empty($pks))
{
if ($this->$k)
{
$pks = array($this->$k);
}
// Nothing to set publishing state on, return false.
else
{
$this->setError(JText::_('JLIB_DATABASE_ERROR_NO_ROWS_SELECTED'));
return false;
}
}
// Build the WHERE clause for the primary keys.
$where = $k . '=' . implode(' OR ' . $k .
'=', $pks);
// Determine if there is checkin support for the table.
if (!property_exists($this, 'checked_out') ||
!property_exists($this, 'checked_out_time'))
{
$checkin = '';
}
else
{
$checkin = ' AND (checked_out = 0 OR checked_out = ' . (int)
$userId . ')';
}
// Update the publishing state for rows with the given primary keys.
$query = $this->_db->getQuery(true)
->update($this->_db->quoteName($this->_tbl))
->set($this->_db->quoteName('state') . ' =
' . (int) $state)
->where($where);
$this->_db->setQuery($query . $checkin);
try
{
$this->_db->execute();
}
catch (RuntimeException $e)
{
$this->setError($e->getMessage());
return false;
}
// If checkin is supported and all rows were adjusted, check them in.
if ($checkin && count($pks) ===
$this->_db->getAffectedRows())
{
// Checkin the rows.
foreach ($pks as $pk)
{
$this->checkIn($pk);
}
}
// If the JTable instance value is in the list of primary keys that were
set, set the instance.
if (in_array($this->$k, $pks))
{
$this->state = $state;
}
$this->setError('');
return true;
}
/**
* Method to store a row in the database from the JTable instance
properties.
* If a primary key value is set the row with that primary key value will
be
* updated with the instance property values. If no primary key value is
set
* a new row will be inserted into the database with the properties from
the
* JTable instance.
*
* @param boolean $updateNulls True to update fields even if they are
null. [optional]
*
* @return boolean True on success.
*
* @since 2.5
*/
public function store($updateNulls = false)
{
$date = JFactory::getDate()->toSql();
$userId = JFactory::getUser()->id;
$this->modified = $date;
if ($this->filter_id)
{
// Existing item
$this->modified_by = $userId;
}
else
{
// New item. A filter's created field can be set by the user,
// so we don't touch it if it is set.
if (!(int) $this->created)
{
$this->created = $date;
}
if (empty($this->created_by))
{
$this->created_by = $userId;
}
}
if (is_array($this->data))
{
$this->map_count = count($this->data);
$this->data = implode(',', $this->data);
}
else
{
$this->map_count = 0;
$this->data = implode(',', array());
}
// Verify that the alias is unique
$table = JTable::getInstance('Filter', 'FinderTable',
array('dbo' => $this->_db));
if ($table->load(array('alias' => $this->alias))
&& ($table->filter_id != $this->filter_id ||
$this->filter_id == 0))
{
$this->setError(JText::_('JLIB_DATABASE_ERROR_ARTICLE_UNIQUE_ALIAS'));
return false;
}
return parent::store($updateNulls);
}
}
PK"�[>��u__tables/link.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Link table class for the Finder package.
*
* @since 2.5
*/
class FinderTableLink extends JTable
{
/**
* Constructor
*
* @param JDatabaseDriver $db JDatabaseDriver connector object.
*
* @since 2.5
*/
public function __construct(&$db)
{
parent::__construct('#__finder_links', 'link_id',
$db);
}
}
PK"�[�&�� � tables/map.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
use Joomla\Utilities\ArrayHelper;
/**
* Map table class for the Finder package.
*
* @since 2.5
*/
class FinderTableMap extends JTable
{
/**
* Constructor
*
* @param JDatabaseDriver $db JDatabaseDriver connector object.
*
* @since 2.5
*/
public function __construct(&$db)
{
parent::__construct('#__finder_taxonomy', 'id', $db);
}
/**
* Method to set the publishing state for a row or list of rows in the
database
* table. The method respects checked out rows by other users and will
attempt
* to checkin rows that it can after adjustments are made.
*
* @param mixed $pks An array of primary key values to update.
If not
* set the instance property value is used.
[optional]
* @param integer $state The publishing state. eg. [0 = unpublished,
1 = published] [optional]
* @param integer $userId The user id of the user performing the
operation. [optional]
*
* @return boolean True on success.
*
* @since 2.5
*/
public function publish($pks = null, $state = 1, $userId = 0)
{
$k = $this->_tbl_key;
// Sanitize input.
$pks = ArrayHelper::toInteger($pks);
$state = (int) $state;
// If there are no primary keys set check to see if the instance key is
set.
if (empty($pks))
{
if ($this->$k)
{
$pks = array($this->$k);
}
// Nothing to set publishing state on, return false.
else
{
$this->setError(JText::_('JLIB_DATABASE_ERROR_NO_ROWS_SELECTED'));
return false;
}
}
// Build the WHERE clause for the primary keys.
$where = $k . '=' . implode(' OR ' . $k .
'=', $pks);
// Update the publishing state for rows with the given primary keys.
$query = $this->_db->getQuery(true)
->update($this->_db->quoteName($this->_tbl))
->set($this->_db->quoteName('state') . ' =
' . (int) $state)
->where($where);
$this->_db->setQuery($query);
try
{
$this->_db->execute();
}
catch (RuntimeException $e)
{
$this->setError($e->getMessage());
return false;
}
// If the JTable instance value is in the list of primary keys that were
set, set the instance.
if (in_array($this->$k, $pks))
{
$this->state = $state;
}
$this->setError('');
return true;
}
}
PK"�[^�M�
�
views/filter/tmpl/edit.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JHtml::_('behavior.formvalidator');
JHtml::_('behavior.keepalive');
JHtml::_('formbehavior.chosen', 'select');
JHtml::_('behavior.tabstate');
JFactory::getDocument()->addScriptDeclaration('
Joomla.submitbutton = function(task)
{
if (task == "filter.cancel" ||
document.formvalidator.isValid(document.getElementById("adminForm")))
{
Joomla.submitform(task, document.getElementById("adminForm"));
}
};
jQuery(document).ready(function($) {
$("#rightbtn").on("click", function() {
if($(this).text() == "' .
JText::_('COM_FINDER_FILTER_SHOW_ALL') . '") {
$(".collapse:not(.in)").each(function (index) {
$(this).collapse("toggle");
});
$(this).text("' .
JText::_('COM_FINDER_FILTER_HIDE_ALL') . '");
} else {
$(this).text("' .
JText::_('COM_FINDER_FILTER_SHOW_ALL') . '");
$(".collapse.in").each(function (index) {
$(this).collapse("toggle");
});
}
return false;
});
$(".filter-node").change(function() {
$(\'input[id="jform_map_count"]\').val(document.querySelectorAll(\'input[type="checkbox"]:checked\').length);
});
});
');
JFactory::getDocument()->addStyleDeclaration('
.accordion-inner .control-group .controls {
margin-left: 10px;
}
.accordion-inner > .control-group {
margin-bottom: 0;
}
');
?>
<form action="<?php echo
JRoute::_('index.php?option=com_finder&view=filter&layout=edit&filter_id='
. (int) $this->item->filter_id); ?>" method="post"
name="adminForm" id="adminForm"
class="form-validate">
<?php echo JLayoutHelper::render('joomla.edit.title_alias',
$this); ?>
<div class="form-horizontal">
<?php echo JHtml::_('bootstrap.startTabSet',
'myTab', array('active' => 'details'));
?>
<?php echo JHtml::_('bootstrap.addTab', 'myTab',
'details', JText::_('COM_FINDER_EDIT_FILTER')); ?>
<div class="row-fluid">
<div class="span9">
<?php if ($this->total > 0) : ?>
<div class="well">
<?php echo $this->form->renderField('map_count');
?>
</div>
<button class="btn jform-rightbtn"
type="button"
onclick="jQuery('.filter-node').each(function () {
this.click(); });">
<span class="icon-checkbox-partial"
aria-hidden="true"></span> <?php echo
JText::_('JGLOBAL_SELECTION_INVERT'); ?></button>
<button class="btn pull-right" type="button"
id="rightbtn" ><?php echo
JText::_('COM_FINDER_FILTER_SHOW_ALL'); ?></button>
<hr>
<?php endif; ?>
<?php echo JHtml::_('filter.slider',
array('selected_nodes' => $this->filter->data)); ?>
</div>
<div class="span3">
<?php echo JLayoutHelper::render('joomla.edit.global',
$this); ?>
</div>
</div>
<?php echo JHtml::_('bootstrap.endTab'); ?>
<?php echo JHtml::_('bootstrap.addTab', 'myTab',
'publishing', JText::_('JGLOBAL_FIELDSET_PUBLISHING'));
?>
<div class="row-fluid form-horizontal-desktop">
<?php echo
JLayoutHelper::render('joomla.edit.publishingdata', $this); ?>
</div>
<?php echo JHtml::_('bootstrap.endTab'); ?>
<?php echo JLayoutHelper::render('joomla.edit.params',
$this); ?>
<?php echo JHtml::_('bootstrap.endTabSet'); ?>
</div>
<input type="hidden" name="task" value=""
/>
<input type="hidden" name="return"
value="<?php echo
JFactory::getApplication()->input->get('return',
'', 'BASE64'); ?>" />
<?php echo JHtml::_('form.token'); ?>
</form>
PK"�[h��h
h
views/filter/view.html.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Filter view class for Finder.
*
* @since 2.5
*/
class FinderViewFilter extends JViewLegacy
{
/**
* The filter object
*
* @var FinderTableFilter
*
* @since 3.6.2
*/
protected $filter;
/**
* The JForm object
*
* @var JForm
*
* @since 3.6.2
*/
protected $form;
/**
* The active item
*
* @var JObject|boolean
*
* @since 3.6.2
*/
protected $item;
/**
* The model state
*
* @var mixed
*
* @since 3.6.2
*/
protected $state;
/**
* The total indexed items
*
* @var integer
*
* @since 3.8.0
*/
protected $total;
/**
* Method to display the view.
*
* @param string $tpl A template file to load. [optional]
*
* @return mixed A string if successful, otherwise a JError object.
*
* @since 2.5
*/
public function display($tpl = null)
{
// Load the view data.
$this->filter = $this->get('Filter');
$this->item = $this->get('Item');
$this->form = $this->get('Form');
$this->state = $this->get('State');
$this->total = $this->get('Total');
// Check for errors.
if (count($errors = $this->get('Errors')))
{
throw new Exception(implode("\n", $errors), 500);
}
JHtml::addIncludePath(JPATH_COMPONENT . '/helpers/html');
JHtml::addIncludePath(JPATH_SITE .
'/components/com_finder/helpers/html');
// Configure the toolbar.
$this->addToolbar();
return parent::display($tpl);
}
/**
* Method to configure the toolbar for this view.
*
* @return void
*
* @since 2.5
*/
protected function addToolbar()
{
JFactory::getApplication()->input->set('hidemainmenu',
true);
$isNew = ($this->item->filter_id == 0);
$checkedOut = !($this->item->checked_out == 0 ||
$this->item->checked_out == JFactory::getUser()->id);
$canDo = JHelperContent::getActions('com_finder');
// Configure the toolbar.
JToolbarHelper::title(
$isNew ? JText::_('COM_FINDER_FILTER_NEW_TOOLBAR_TITLE') :
JText::_('COM_FINDER_FILTER_EDIT_TOOLBAR_TITLE'),
'zoom-in finder'
);
// Set the actions for new and existing records.
if ($isNew)
{
// For new records, check the create permission.
if ($canDo->get('core.create'))
{
JToolbarHelper::apply('filter.apply');
JToolbarHelper::save('filter.save');
JToolbarHelper::save2new('filter.save2new');
}
JToolbarHelper::cancel('filter.cancel');
}
else
{
// Can't save the record if it's checked out.
// Since it's an existing record, check the edit permission.
if (!$checkedOut && $canDo->get('core.edit'))
{
JToolbarHelper::apply('filter.apply');
JToolbarHelper::save('filter.save');
// We can save this record, but check the create permission to see if
we can return to make a new one.
if ($canDo->get('core.create'))
{
JToolbarHelper::save2new('filter.save2new');
}
}
// If an existing item, can save as a copy
if ($canDo->get('core.create'))
{
JToolbarHelper::save2copy('filter.save2copy');
}
JToolbarHelper::cancel('filter.cancel',
'JTOOLBAR_CLOSE');
}
JToolbarHelper::divider();
JToolbarHelper::help('JHELP_COMPONENTS_FINDER_MANAGE_SEARCH_FILTERS_EDIT');
}
}
PK"�[aۇD��views/filters/tmpl/default.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JHtml::_('formbehavior.chosen', 'select');
JHtml::_('bootstrap.tooltip');
$user = JFactory::getUser();
$userId = $user->get('id');
$listOrder =
$this->escape($this->state->get('list.ordering'));
$listDirn =
$this->escape($this->state->get('list.direction'));
JText::script('COM_FINDER_INDEX_CONFIRM_DELETE_PROMPT');
JFactory::getDocument()->addScriptDeclaration('
Joomla.submitbutton = function(pressbutton)
{
if (pressbutton == "filters.delete")
{
if
(confirm(Joomla.JText._("COM_FINDER_INDEX_CONFIRM_DELETE_PROMPT")))
{
Joomla.submitform(pressbutton);
}
else
{
return false;
}
}
Joomla.submitform(pressbutton);
};
');
?>
<form action="<?php echo
JRoute::_('index.php?option=com_finder&view=filters');
?>" method="post" name="adminForm"
id="adminForm">
<?php if (!empty( $this->sidebar)) : ?>
<div id="j-sidebar-container" class="span2">
<?php echo $this->sidebar; ?>
</div>
<div id="j-main-container" class="span10">
<?php else : ?>
<div id="j-main-container">
<?php endif; ?>
<?php echo
JLayoutHelper::render('joomla.searchtools.default',
array('view' => $this)); ?>
<div class="clearfix"> </div>
<?php if (empty($this->items)) : ?>
<div class="alert alert-no-items">
<?php echo JText::_('COM_FINDER_NO_RESULTS_OR_FILTERS');
?>
</div>
<?php else : ?>
<table class="table table-striped">
<thead>
<tr>
<th width="1%" class="nowrap center">
<?php echo JHtml::_('grid.checkall'); ?>
</th>
<th width="1%" class="nowrap">
<?php echo JHtml::_('searchtools.sort',
'JSTATUS', 'a.state', $listDirn, $listOrder); ?>
</th>
<th class="nowrap">
<?php echo JHtml::_('searchtools.sort',
'JGLOBAL_TITLE', 'a.title', $listDirn, $listOrder);
?>
</th>
<th width="10%" class="nowrap hidden-phone">
<?php echo JHtml::_('searchtools.sort',
'COM_FINDER_HEADING_CREATED_BY', 'a.created_by_alias',
$listDirn, $listOrder); ?>
</th>
<th width="10%" class="nowrap hidden-phone">
<?php echo JHtml::_('searchtools.sort',
'COM_FINDER_HEADING_CREATED_ON', 'a.created',
$listDirn, $listOrder); ?>
</th>
<th width="5%" class="nowrap hidden-phone">
<?php echo JHtml::_('searchtools.sort',
'COM_FINDER_HEADING_MAP_COUNT', 'a.map_count',
$listDirn, $listOrder); ?>
</th>
<th width="1%" class="nowrap hidden-phone">
<?php echo JHtml::_('searchtools.sort',
'JGRID_HEADING_ID', 'a.filter_id', $listDirn,
$listOrder); ?>
</th>
</tr>
</thead>
<tfoot>
<tr>
<td colspan="7">
<?php echo $this->pagination->getListFooter(); ?>
</td>
</tr>
</tfoot>
<tbody>
<?php
$canCreate =
$user->authorise('core.create', 'com_finder');
$canEdit =
$user->authorise('core.edit', 'com_finder');
$userAuthoriseCoreManage =
$user->authorise('core.manage', 'com_checkin');
$userAuthoriseCoreEditState =
$user->authorise('core.edit.state', 'com_finder');
$userId = $user->get('id');
foreach ($this->items as $i => $item) :
$canCheckIn = $userAuthoriseCoreManage || $item->checked_out ==
$userId || $item->checked_out == 0;
$canChange = $userAuthoriseCoreEditState && $canCheckIn;
$escapedTitle = $this->escape($item->title);
?>
<tr class="row<?php echo $i % 2; ?>">
<td class="center">
<?php echo JHtml::_('grid.id', $i,
$item->filter_id); ?>
</td>
<td class="center nowrap">
<?php echo JHtml::_('jgrid.published', $item->state,
$i, 'filters.', $canChange); ?>
</td>
<td>
<?php if ($item->checked_out) : ?>
<?php echo JHtml::_('jgrid.checkedout', $i,
$item->editor, $item->checked_out_time, 'filters.',
$canCheckIn); ?>
<?php endif; ?>
<?php if ($canEdit) : ?>
<a href="<?php echo
JRoute::_('index.php?option=com_finder&task=filter.edit&filter_id='
. (int) $item->filter_id); ?>">
<?php echo $escapedTitle; ?></a>
<?php else : ?>
<?php echo $escapedTitle; ?>
<?php endif; ?>
</td>
<td class="nowrap hidden-phone">
<?php echo $item->created_by_alias ?: $item->user_name;
?>
</td>
<td class="nowrap hidden-phone">
<?php echo JHtml::_('date', $item->created,
JText::_('DATE_FORMAT_LC4')); ?>
</td>
<td class="nowrap hidden-phone">
<?php echo $item->map_count; ?>
</td>
<td class="hidden-phone">
<?php echo (int) $item->filter_id; ?>
</td>
</tr>
<?php endforeach; ?>
</tbody>
</table>
<?php endif; ?>
<input type="hidden" name="task"
value="" />
<input type="hidden" name="boxchecked"
value="0" />
<?php echo JHtml::_('form.token'); ?>
</div>
</form>
PK"�[����views/filters/view.html.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Filters view class for Finder.
*
* @since 2.5
*/
class FinderViewFilters extends JViewLegacy
{
/**
* An array of items
*
* @var array
*
* @since 3.6.1
*/
protected $items;
/**
* The pagination object
*
* @var JPagination
*
* @since 3.6.1
*/
protected $pagination;
/**
* The HTML markup for the sidebar
*
* @var string
*
* @since 3.6.1
*/
protected $sidebar;
/**
* The model state
*
* @var mixed
*
* @since 3.6.1
*/
protected $state;
/**
* The total number of items
*
* @var integer
*
* @since 3.6.1
*/
protected $total;
/**
* Method to display the view.
*
* @param string $tpl A template file to load. [optional]
*
* @return mixed A string if successful, otherwise a JError object.
*
* @since 2.5
*/
public function display($tpl = null)
{
// Load the view data.
$this->items = $this->get('Items');
$this->pagination = $this->get('Pagination');
$this->total = $this->get('Total');
$this->state = $this->get('State');
$this->filterForm = $this->get('FilterForm');
$this->activeFilters = $this->get('ActiveFilters');
FinderHelper::addSubmenu('filters');
// Check for errors.
if (count($errors = $this->get('Errors')))
{
throw new Exception(implode("\n", $errors), 500);
}
JHtml::addIncludePath(JPATH_COMPONENT . '/helpers/html');
// Configure the toolbar.
$this->addToolbar();
$this->sidebar = JHtmlSidebar::render();
return parent::display($tpl);
}
/**
* Method to configure the toolbar for this view.
*
* @return void
*
* @since 2.5
*/
protected function addToolbar()
{
$canDo = JHelperContent::getActions('com_finder');
JToolbarHelper::title(JText::_('COM_FINDER_FILTERS_TOOLBAR_TITLE'),
'zoom-in finder');
$toolbar = JToolbar::getInstance('toolbar');
if ($canDo->get('core.create'))
{
JToolbarHelper::addNew('filter.add');
JToolbarHelper::editList('filter.edit');
JToolbarHelper::divider();
}
if ($canDo->get('core.edit.state'))
{
JToolbarHelper::publishList('filters.publish');
JToolbarHelper::unpublishList('filters.unpublish');
JToolbarHelper::checkin('filters.checkin');
JToolbarHelper::divider();
}
if ($canDo->get('core.admin') ||
$canDo->get('core.options'))
{
JToolbarHelper::preferences('com_finder');
}
JToolbarHelper::divider();
$toolbar->appendButton('Popup', 'bars',
'COM_FINDER_STATISTICS',
'index.php?option=com_finder&view=statistics&tmpl=component',
550, 350);
JToolbarHelper::divider();
JToolbarHelper::help('JHELP_COMPONENTS_FINDER_MANAGE_SEARCH_FILTERS');
if ($canDo->get('core.delete'))
{
JToolbarHelper::deleteList('', 'filters.delete');
JToolbarHelper::divider();
}
}
}
PK"�['�e���views/index/tmpl/default.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JHtml::_('bootstrap.tooltip');
JHtml::_('formbehavior.chosen', 'select');
JHtml::_('bootstrap.popover');
$listOrder =
$this->escape($this->state->get('list.ordering'));
$listDirn =
$this->escape($this->state->get('list.direction'));
$lang = JFactory::getLanguage();
JText::script('COM_FINDER_INDEX_CONFIRM_PURGE_PROMPT');
JText::script('COM_FINDER_INDEX_CONFIRM_DELETE_PROMPT');
JFactory::getDocument()->addScriptDeclaration('
Joomla.submitbutton = function(pressbutton)
{
if (pressbutton == "index.purge")
{
if
(confirm(Joomla.JText._("COM_FINDER_INDEX_CONFIRM_PURGE_PROMPT")))
{
Joomla.submitform(pressbutton);
}
else
{
return false;
}
}
if (pressbutton == "index.delete")
{
if
(confirm(Joomla.JText._("COM_FINDER_INDEX_CONFIRM_DELETE_PROMPT")))
{
Joomla.submitform(pressbutton);
}
else
{
return false;
}
}
Joomla.submitform(pressbutton);
};
');
?>
<form action="<?php echo
JRoute::_('index.php?option=com_finder&view=index');
?>" method="post" name="adminForm"
id="adminForm">
<?php if (!empty( $this->sidebar)) : ?>
<div id="j-sidebar-container" class="span2">
<?php echo $this->sidebar; ?>
</div>
<div id="j-main-container" class="span10">
<?php else : ?>
<div id="j-main-container">
<?php endif; ?>
<?php echo
JLayoutHelper::render('joomla.searchtools.default',
array('view' => $this)); ?>
<table class="table table-striped">
<thead>
<tr>
<th width="1%" class="nowrap center">
<?php echo JHtml::_('grid.checkall'); ?>
</th>
<th width="1%" class="nowrap center">
<?php echo JHtml::_('searchtools.sort',
'JSTATUS', 'l.published', $listDirn, $listOrder); ?>
</th>
<th class="nowrap">
<?php echo JHtml::_('searchtools.sort',
'JGLOBAL_TITLE', 'l.title', $listDirn, $listOrder);
?>
</th>
<th width="5%" class="nowrap hidden-phone">
<?php echo JHtml::_('searchtools.sort',
'COM_FINDER_INDEX_HEADING_INDEX_TYPE', 't.title',
$listDirn, $listOrder); ?>
</th>
<th width="1%" class="nowrap hidden-phone">
<?php echo JHtml::_('searchtools.sort',
'COM_FINDER_INDEX_HEADING_INDEX_DATE', 'l.indexdate',
$listDirn, $listOrder); ?>
</th>
<th width="1%" class="nowrap center hidden-phone
hidden-tablet">
<?php echo JText::_('COM_FINDER_INDEX_HEADING_DETAILS');
?>
</th>
<th width="35%" class="nowrap hidden-phone
hidden-tablet">
<?php echo JHtml::_('searchtools.sort',
'COM_FINDER_INDEX_HEADING_LINK_URL', 'l.url',
$listDirn, $listOrder); ?>
</th>
</tr>
</thead>
<tfoot>
<tr>
<td colspan="7">
<?php echo $this->pagination->getListFooter(); ?>
</td>
</tr>
</tfoot>
<tbody>
<?php $canChange =
JFactory::getUser()->authorise('core.manage',
'com_finder'); ?>
<?php foreach ($this->items as $i => $item) : ?>
<tr class="row<?php echo $i % 2; ?>">
<td class="center">
<?php echo JHtml::_('grid.id', $i, $item->link_id);
?>
</td>
<td class="center">
<?php echo JHtml::_('jgrid.published',
$item->published, $i, 'index.', $canChange, 'cb');
?>
</td>
<td>
<label for="cb<?php echo $i ?>">
<?php echo $this->escape($item->title); ?>
</label>
</td>
<td class="small nowrap hidden-phone">
<?php
$key = FinderHelperLanguage::branchSingular($item->t_title);
echo $lang->hasKey($key) ? JText::_($key) : $item->t_title;
?>
</td>
<td class="small nowrap hidden-phone">
<?php echo JHtml::_('date', $item->indexdate,
JText::_('DATE_FORMAT_LC4')); ?>
</td>
<td class="center hidden-phone hidden-tablet">
<?php if ((int) $item->publish_start_date || (int)
$item->publish_end_date || (int) $item->start_date || (int)
$item->end_date) : ?>
<span class="icon-calendar pop hasPopover"
aria-hidden="true" data-placement="left"
title="<?php echo
JText::_('COM_FINDER_INDEX_DATE_INFO_TITLE'); ?>"
data-content="<?php echo
JText::sprintf('COM_FINDER_INDEX_DATE_INFO',
$item->publish_start_date, $item->publish_end_date,
$item->start_date, $item->end_date); ?>"></span>
<span class="element-invisible"><?php echo
JText::sprintf('COM_FINDER_INDEX_DATE_INFO',
$item->publish_start_date, $item->publish_end_date,
$item->start_date, $item->end_date); ?></span>
<?php endif; ?>
</td>
<td class="small break-word hidden-phone
hidden-tablet">
<?php echo (strlen($item->url) > 80) ? substr($item->url,
0, 70) . '...' : $item->url; ?>
</td>
</tr>
<?php endforeach; ?>
</tbody>
</table>
<input type="hidden" name="task"
value="display" />
<input type="hidden" name="boxchecked"
value="0" />
<?php echo JHtml::_('form.token'); ?>
</div>
</form>
PK"�[۾}__views/index/view.html.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderHelperLanguage', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/language.php');
/**
* Index view class for Finder.
*
* @since 2.5
*/
class FinderViewIndex extends JViewLegacy
{
/**
* An array of items
*
* @var array
*
* @since 3.6.1
*/
protected $items;
/**
* The pagination object
*
* @var JPagination
*
* @since 3.6.1
*/
protected $pagination;
/**
* The state of core Smart Search plugins
*
* @var array
*
* @since 3.6.1
*/
protected $pluginState;
/**
* The HTML markup for the sidebar
*
* @var string
*
* @since 3.6.1
*/
protected $sidebar;
/**
* The model state
*
* @var mixed
*
* @since 3.6.1
*/
protected $state;
/**
* The total number of items
*
* @var integer
*
* @since 3.6.1
*/
protected $total;
/**
* Method to display the view.
*
* @param string $tpl A template file to load. [optional]
*
* @return mixed A string if successful, otherwise a JError object.
*
* @since 2.5
*/
public function display($tpl = null)
{
// Load plugin language files.
FinderHelperLanguage::loadPluginLanguage();
$this->items = $this->get('Items');
$this->total = $this->get('Total');
$this->pagination = $this->get('Pagination');
$this->state = $this->get('State');
$this->pluginState = $this->get('pluginState');
$this->filterForm = $this->get('FilterForm');
$this->activeFilters = $this->get('ActiveFilters');
FinderHelper::addSubmenu('index');
// Check for errors.
if (count($errors = $this->get('Errors')))
{
JError::raiseError(500, implode("\n", $errors));
return false;
}
if (!$this->pluginState['plg_content_finder']->enabled)
{
$link =
JRoute::_('index.php?option=com_plugins&task=plugin.edit&extension_id='
. FinderHelper::getFinderPluginId());
JFactory::getApplication()->enqueueMessage(JText::sprintf('COM_FINDER_INDEX_PLUGIN_CONTENT_NOT_ENABLED',
$link), 'warning');
}
elseif ($this->get('TotalIndexed') === 0)
{
JFactory::getApplication()->enqueueMessage(JText::_('COM_FINDER_INDEX_NO_DATA')
. ' ' . JText::_('COM_FINDER_INDEX_TIP'),
'notice');
}
JHtml::addIncludePath(JPATH_COMPONENT . '/helpers/html');
// Configure the toolbar.
$this->addToolbar();
$this->sidebar = JHtmlSidebar::render();
return parent::display($tpl);
}
/**
* Method to configure the toolbar for this view.
*
* @return void
*
* @since 2.5
*/
protected function addToolbar()
{
$canDo = JHelperContent::getActions('com_finder');
JToolbarHelper::title(JText::_('COM_FINDER_INDEX_TOOLBAR_TITLE'),
'zoom-in finder');
$toolbar = JToolbar::getInstance('toolbar');
$toolbar->appendButton(
'Popup', 'archive', 'COM_FINDER_INDEX',
'index.php?option=com_finder&view=indexer&tmpl=component',
500, 210, 0, 0,
'window.parent.location.reload()',
'COM_FINDER_HEADING_INDEXER'
);
if ($canDo->get('core.edit.state'))
{
JToolbarHelper::publishList('index.publish');
JToolbarHelper::unpublishList('index.unpublish');
}
if ($canDo->get('core.admin') ||
$canDo->get('core.options'))
{
JToolbarHelper::preferences('com_finder');
}
$toolbar->appendButton('Popup', 'bars',
'COM_FINDER_STATISTICS',
'index.php?option=com_finder&view=statistics&tmpl=component',
550, 350);
if ($canDo->get('core.delete'))
{
JToolbarHelper::deleteList('', 'index.delete');
}
if ($canDo->get('core.edit.state'))
{
JToolbarHelper::trash('index.purge',
'COM_FINDER_INDEX_TOOLBAR_PURGE', false);
}
JToolbarHelper::help('JHELP_COMPONENTS_FINDER_MANAGE_INDEXED_CONTENT');
}
}
PK"�[�
@�rrviews/indexer/tmpl/default.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JHtml::_('behavior.keepalive');
JHtml::_('behavior.core');
JHtml::_('jquery.framework');
JHtml::_('script', 'com_finder/indexer.js',
array('version' => 'auto', 'relative'
=> true));
JFactory::getDocument()->addScriptDeclaration('var msg =
"' . JText::_('COM_FINDER_INDEXER_MESSAGE_COMPLETE') .
'";');
?>
<div id="finder-indexer-container">
<br /><br />
<h1 id="finder-progress-header"><?php echo
JText::_('COM_FINDER_INDEXER_HEADER_INIT'); ?></h1>
<p id="finder-progress-message"><?php echo
JText::_('COM_FINDER_INDEXER_MESSAGE_INIT'); ?></p>
<div id="progress" class="progress progress-striped
active">
<div id="progress-bar" class="bar bar-success"
aria-valuenow="0" aria-valuemin="0"
aria-valuemax="100"></div>
</div>
<input id="finder-indexer-token" type="hidden"
name="<?php echo JFactory::getSession()->getFormToken();
?>" value="1" />
</div>
PK"�[�����views/indexer/view.html.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Indexer view class for Finder.
*
* @since 2.5
*/
class FinderViewIndexer extends JViewLegacy
{
}
PK"�[{�s��views/maps/tmpl/default.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JHtml::_('formbehavior.chosen', 'select');
JHtml::_('bootstrap.tooltip');
$listOrder =
$this->escape($this->state->get('list.ordering'));
$listDirn =
$this->escape($this->state->get('list.direction'));
$lang = JFactory::getLanguage();
$branchFilter =
$this->escape($this->state->get('filter.branch'));
$colSpan = $branchFilter ? 5 : 6;
JText::script('COM_FINDER_MAPS_CONFIRM_DELETE_PROMPT');
JFactory::getDocument()->addScriptDeclaration('
Joomla.submitbutton = function(pressbutton)
{
if (pressbutton == "map.delete")
{
if
(confirm(Joomla.JText._("COM_FINDER_MAPS_CONFIRM_DELETE_PROMPT")))
{
Joomla.submitform(pressbutton);
}
else
{
return false;
}
}
Joomla.submitform(pressbutton);
};
');
?>
<form action="<?php echo
JRoute::_('index.php?option=com_finder&view=maps');
?>" method="post" name="adminForm"
id="adminForm">
<?php if (!empty( $this->sidebar)) : ?>
<div id="j-sidebar-container" class="span2">
<?php echo $this->sidebar; ?>
</div>
<div id="j-main-container" class="span10">
<?php else : ?>
<div id="j-main-container">
<?php endif; ?>
<?php echo
JLayoutHelper::render('joomla.searchtools.default',
array('view' => $this)); ?>
<div class="clearfix"> </div>
<?php if (empty($this->items)) : ?>
<div class="alert alert-no-items">
<?php echo JText::_('COM_FINDER_MAPS_NO_CONTENT'); ?>
</div>
<?php else : ?>
<table class="table table-striped">
<thead>
<tr>
<th width="1%" class="center nowrap">
<?php echo JHtml::_('grid.checkall'); ?>
</th>
<th width="1%" class="center nowrap">
<?php echo JHtml::_('searchtools.sort',
'JSTATUS', 'a.state', $listDirn, $listOrder); ?>
</th>
<th class="nowrap">
<?php echo JHtml::_('searchtools.sort',
'JGLOBAL_TITLE', 'd.branch_title', $listDirn,
$listOrder); ?>
</th>
<?php if (!$branchFilter) : ?>
<th width="1%" class="nowrap center">
<?php echo JText::_('COM_FINDER_HEADING_CHILDREN');
?>
</th>
<?php endif; ?>
<th width="1%" class="nowrap center">
<span class="icon-publish"
aria-hidden="true"></span>
<span class="hidden-phone"><?php echo
JText::_('COM_FINDER_MAPS_COUNT_PUBLISHED_ITEMS');
?></span>
</th>
<th width="1%" class="nowrap center">
<span class="icon-unpublish"
aria-hidden="true"></span>
<span class="hidden-phone"><?php echo
JText::_('COM_FINDER_MAPS_COUNT_UNPUBLISHED_ITEMS');
?></span>
</th>
</tr>
</thead>
<tfoot>
<tr>
<td colspan="<?php echo $colSpan; ?>">
<?php echo $this->pagination->getListFooter(); ?>
</td>
</tr>
</tfoot>
<tbody>
<?php $canChange =
JFactory::getUser()->authorise('core.manage',
'com_finder'); ?>
<?php foreach ($this->items as $i => $item) : ?>
<tr class="row<?php echo $i % 2; ?>">
<td class="center">
<?php echo JHtml::_('grid.id', $i, $item->id); ?>
</td>
<td class="center nowrap">
<?php echo JHtml::_('jgrid.published', $item->state,
$i, 'maps.', $canChange, 'cb'); ?>
</td>
<td>
<?php
if (trim($item->parent_title, '**') ===
'Language')
{
$title = FinderHelperLanguage::branchLanguageTitle($item->title);
}
else
{
$key = FinderHelperLanguage::branchSingular($item->title);
$title = $lang->hasKey($key) ? JText::_($key) : $item->title;
}
?>
<?php if ((int) $item->num_children === 0) : ?>
<span class="gi">—</span>
<?php endif; ?>
<label for="cb<?php echo $i; ?>"
style="display:inline-block;">
<?php echo $this->escape($title); ?>
</label>
<?php if ($this->escape(trim($title, '**')) ===
'Language' && JLanguageMultilang::isEnabled()) : ?>
<strong><?php echo
JText::_('COM_FINDER_MAPS_MULTILANG'); ?></strong>
<?php endif; ?>
</td>
<?php if (!$branchFilter) : ?>
<td class="center btns">
<?php if ((int) $item->num_children !== 0) : ?>
<a href="<?php echo
JRoute::_('index.php?option=com_finder&view=maps&filter[branch]='
. $item->id); ?>">
<span class="badge <?php if ($item->num_children >
0) echo 'badge-info'; ?>"><?php echo
$item->num_children; ?></span></a>
<?php else : ?>
-
<?php endif; ?>
</td>
<?php endif; ?>
<td class="center btns">
<?php if ((int) $item->num_children === 0) : ?>
<a class="badge <?php if ((int) $item->count_published
> 0) echo 'badge-success'; ?>" title="<?php
echo JText::_('COM_FINDER_MAPS_COUNT_PUBLISHED_ITEMS');
?>" href="<?php echo
JRoute::_('index.php?option=com_finder&view=index&filter[state]=1&filter[content_map]='
. $item->id); ?>">
<?php echo (int) $item->count_published; ?></a>
<?php else : ?>
-
<?php endif; ?>
</td>
<td class="center btns">
<?php if ((int) $item->num_children === 0) : ?>
<a class="badge <?php if ((int)
$item->count_unpublished > 0) echo 'badge-important';
?>" title="<?php echo
JText::_('COM_FINDER_MAPS_COUNT_UNPUBLISHED_ITEMS'); ?>"
href="<?php echo
JRoute::_('index.php?option=com_finder&view=index&filter[state]=0&filter[content_map]='
. $item->id); ?>">
<?php echo (int) $item->count_unpublished; ?></a>
<?php else : ?>
-
<?php endif; ?>
</td>
</tr>
<?php endforeach; ?>
</tbody>
</table>
<?php endif; ?>
</div>
<input type="hidden" name="task"
value="display" />
<input type="hidden" name="boxchecked"
value="0" />
<?php echo JHtml::_('form.token'); ?>
</form>
PK"�[
�u��views/maps/view.html.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
JLoader::register('FinderHelperLanguage', JPATH_ADMINISTRATOR .
'/components/com_finder/helpers/language.php');
/**
* Groups view class for Finder.
*
* @since 2.5
*/
class FinderViewMaps extends JViewLegacy
{
/**
* An array of items
*
* @var array
*
* @since 3.6.1
*/
protected $items;
/**
* The pagination object
*
* @var JPagination
*
* @since 3.6.1
*/
protected $pagination;
/**
* The HTML markup for the sidebar
*
* @var string
*
* @since 3.6.1
*/
protected $sidebar;
/**
* The model state
*
* @var object
*
* @since 3.6.1
*/
protected $state;
/**
* The total number of items
*
* @var object
*
* @since 3.6.1
*/
protected $total;
/**
* Method to display the view.
*
* @param string $tpl A template file to load. [optional]
*
* @return mixed A string if successful, otherwise a JError object.
*
* @since 2.5
*/
public function display($tpl = null)
{
// Load plugin language files.
FinderHelperLanguage::loadPluginLanguage();
// Load the view data.
$this->items = $this->get('Items');
$this->total = $this->get('Total');
$this->pagination = $this->get('Pagination');
$this->state = $this->get('State');
$this->filterForm = $this->get('FilterForm');
$this->activeFilters = $this->get('ActiveFilters');
FinderHelper::addSubmenu('maps');
// Check for errors.
if (count($errors = $this->get('Errors')))
{
throw new Exception(implode("\n", $errors), 500);
}
JHtml::addIncludePath(JPATH_COMPONENT . '/helpers/html');
// Prepare the view.
$this->addToolbar();
$this->sidebar = JHtmlSidebar::render();
return parent::display($tpl);
}
/**
* Method to configure the toolbar for this view.
*
* @return void
*
* @since 2.5
*/
protected function addToolbar()
{
$canDo = JHelperContent::getActions('com_finder');
JToolbarHelper::title(JText::_('COM_FINDER_MAPS_TOOLBAR_TITLE'),
'zoom-in finder');
if ($canDo->get('core.edit.state'))
{
JToolbarHelper::publishList('maps.publish');
JToolbarHelper::unpublishList('maps.unpublish');
JToolbarHelper::divider();
}
if ($canDo->get('core.admin') ||
$canDo->get('core.options'))
{
JToolbarHelper::preferences('com_finder');
}
JToolbarHelper::divider();
JToolbar::getInstance('toolbar')->appendButton(
'Popup',
'bars',
'COM_FINDER_STATISTICS',
'index.php?option=com_finder&view=statistics&tmpl=component',
550,
350
);
JToolbarHelper::divider();
JToolbarHelper::help('JHELP_COMPONENTS_FINDER_MANAGE_CONTENT_MAPS');
if ($canDo->get('core.delete'))
{
JToolbarHelper::deleteList('', 'maps.delete');
JToolbarHelper::divider();
}
}
}
PK"�[^��;OO!views/statistics/tmpl/default.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
?>
<h3>
<?php echo JText::_('COM_FINDER_STATISTICS_TITLE'); ?>
</h3>
<div class="row-fluid">
<div class="span12">
<p class="tab-description"><?php echo
JText::sprintf('COM_FINDER_STATISTICS_STATS_DESCRIPTION',
number_format($this->data->term_count, 0,
JText::_('DECIMALS_SEPARATOR'),
JText::_('THOUSANDS_SEPARATOR')),
number_format($this->data->link_count, 0,
JText::_('DECIMALS_SEPARATOR'),
JText::_('THOUSANDS_SEPARATOR')),
number_format($this->data->taxonomy_node_count, 0,
JText::_('DECIMALS_SEPARATOR'),
JText::_('THOUSANDS_SEPARATOR')),
number_format($this->data->taxonomy_branch_count, 0,
JText::_('DECIMALS_SEPARATOR'),
JText::_('THOUSANDS_SEPARATOR'))); ?></p>
<table class="table table-striped table-condensed">
<thead>
<tr>
<th>
<?php echo
JText::_('COM_FINDER_STATISTICS_LINK_TYPE_HEADING'); ?>
</th>
<th>
<?php echo
JText::_('COM_FINDER_STATISTICS_LINK_TYPE_COUNT'); ?>
</th>
</tr>
</thead>
<tbody>
<?php foreach ($this->data->type_list as $type) : ?>
<tr>
<td>
<?php
$lang_key = 'PLG_FINDER_STATISTICS_' .
str_replace(' ', '_', $type->type_title);
$lang_string = JText::_($lang_key);
echo $lang_string === $lang_key ? $type->type_title :
$lang_string;
?>
</td>
<td>
<span class="badge badge-info"><?php echo
number_format($type->link_count, 0,
JText::_('DECIMALS_SEPARATOR'),
JText::_('THOUSANDS_SEPARATOR')); ?></span>
</td>
</tr>
<?php endforeach; ?>
<tr>
<td>
<strong><?php echo
JText::_('COM_FINDER_STATISTICS_LINK_TYPE_TOTAL');
?></strong>
</td>
<td>
<span class="badge badge-info"><?php echo
number_format($this->data->link_count, 0,
JText::_('DECIMALS_SEPARATOR'),
JText::_('THOUSANDS_SEPARATOR')); ?></span>
</td>
</tr>
</tbody>
</table>
</div>
</div>
PK"�[�W���views/statistics/view.html.phpnu�[���<?php
/**
* @package Joomla.Administrator
* @subpackage com_finder
*
* @copyright Copyright (C) 2005 - 2020 Open Source Matters, Inc. All
rights reserved.
* @license GNU General Public License version 2 or later; see
LICENSE.txt
*/
defined('_JEXEC') or die;
/**
* Statistics view class for Finder.
*
* @since 2.5
*/
class FinderViewStatistics extends JViewLegacy
{
/**
* The index statistics
*
* @var JObject
*
* @since 3.6.1
*/
protected $data;
/**
* Method to display the view.
*
* @param string $tpl A template file to load. [optional]
*
* @return mixed A string if successful, otherwise a JError object.
*
* @since 2.5
*/
public function display($tpl = null)
{
// Load the view data.
$this->data = $this->get('Data');
// Check for errors.
if (count($errors = $this->get('Errors')))
{
throw new Exception(implode("\n", $errors), 500);
}
return parent::display($tpl);
}
}
PK��[J�bW��
css/dates.cssnu�[���#finder-filter-window
{
margin: 10px 0 10px;
overflow: auto;
padding: 0;
width: 100%;
}
ul#finder-filter-select-dates {
list-style: none;
margin: 0;
padding: 0;
}
ul#finder-filter-select-dates li.filter-date {
background: none;
float: left;
list-style: none;
margin: 0;
padding: 5px 0;
text-align: left;
width: 49%;
}
ul#finder-filter-select-dates li.filter-date select.filter-date-operator {
margin-right: 10px;
}
PK��[2y4|� � css/finder.cssnu�[���#advanced-search
{
text-align:left;
width:100%;
padding:5px 0 15px;
}
#advanced-search-toggle {
cursor:pointer;
}
#search-query-explained {
padding:10px 0;
}
#search-query-explained span.term,
#search-query-explained span.date,
#search-query-explained span.when,
#search-query-explained span.branch,
#search-query-explained span.node,
#search-query-explained span.op {
font-weight:bold;
}
#search-query-explained span.op {
text-transform:uppercase;
}
#search-results li.search-result .mime-pdf {
padding-left:20px;
background:url(../../system/images/pdf_button.png) no-repeat;
}
#search-results .search-pagination,
#search-results .pagination,
#search-results .search-pages-counter {
clear:both;
margin:0 auto;
}
#highlighter-start, #highlighter-end {
display:none;
height:0;
opacity:0;
}
span.highlight {
background-color:#FFFFCC;
font-weight:bold;
padding:1px 0;
}
ul.autocompleter-choices {
position:absolute;
margin:0;
padding:0;
list-style:none;
border:1px solid #EEEEEE;
background-color:white;
border-right-color:#DDDDDD;
border-bottom-color:#DDDDDD;
text-align:left;
font-family:Verdana, Geneva, Arial, Helvetica, sans-serif;
z-index:50;
}
ul.autocompleter-choices li {
background:none;
position:relative;
padding:0.1em 1.5em 0.1em 1em;
cursor:pointer;
font-weight:normal;
font-size:1em;
}
ul.autocompleter-choices li.autocompleter-selected {
background-color:#444;
color:#fff;
}
ul.autocompleter-choices span.autocompleter-queried {
font-weight:bold;
}
ul.autocompleter-choices li.autocompleter-selected
span.autocompleter-queried {
color:#9FCFFF;
}
.autocomplete-suggestions {
border: 1px solid #999;
background: #FFF; cursor: default;
overflow: auto;
-webkit-box-shadow: 1px 4px 3px rgba(50, 50, 50, 0.64);
-moz-box-shadow: 1px 4px 3px rgba(50, 50, 50, 0.64);
box-shadow: 1px 4px 3px rgba(50, 50, 50, 0.64);
}
.autocomplete-suggestion {
padding: 2px 5px;
white-space: nowrap;
overflow: hidden;
}
.autocomplete-no-suggestion {
padding: 2px 5px;
}
.autocomplete-selected {
background: #F0F0F0;
}
.autocomplete-suggestions strong {
font-weight: bold; color: #000;
}
.autocomplete-group {
padding: 2px 5px;
}
.autocomplete-group strong {
font-weight: bold;
font-size: 16px;
color: #000;
display: block;
border-bottom: 1px solid #000;
}
ul#finder-filter-select-list {
top: 4em !important;
}
.finder-selects .inputbox {
width: 180px;
}
div#finder-filter-window .filter-branch .finder-selects {
float: left;
}
PK��[�����css/indexer.cssnu�[���#finder-indexer-container
{
text-align: center;
}
#finder-progress-container {
width: 350px;
margin: 0 auto;
}
h1.finder-error {
color: #FF0000;
}
p.finder-error {
color: #FF0000;
font-weight:bold;
}
PK��[��.j�?�?js/autocompleter.jsnu�[���//@deprecated
3.4
var Observer = new Class({
Implements: [Options, Events],
options: {
periodical: false,
delay: 1000
},
initialize: function (el, onFired, options) {
this.element = document.id(el) || $document.id(el);
this.addEvent('onFired', onFired);
this.setOptions(options);
this.bound = this.changed.bind(this);
this.resume();
},
changed: function () {
var value = this.element.get('value');
if ($equals(this.value, value)) return;
this.clear();
this.value = value;
this.timeout = this.onFired.delay(this.options.delay, this);
},
setValue: function (value) {
this.value = value;
this.element.set('value', value);
return this.clear();
},
onFired: function () {
this.fireEvent('onFired', [this.value, this.element]);
},
clear: function () {
clearTimeout(this.timeout || null);
return this;
},
pause: function () {
if (this.timer) clearTimeout(this.timer);
else this.element.removeEvent('keyup', this.bound);
return this.clear();
},
resume: function () {
this.value = this.element.get('value');
if (this.options.periodical) this.timer =
this.changed.periodical(this.options.periodical, this);
else this.element.addEvent('keyup', this.bound);
return this;
}
});
var $equals = function (obj1, obj2) {
return (obj1 == obj2 || JSON.encode(obj1) == JSON.encode(obj2));
};
var Autocompleter = new Class({
Implements: [Options, Events],
options: {
minLength: 1,
markQuery: true,
width: 'inherit',
maxChoices: 10,
injectChoice: null,
customChoices: null,
emptyChoices: null,
visibleChoices: true,
className: 'autocompleter-choices',
zIndex: 1000,
delay: 400,
observerOptions: {},
fxOptions: {},
autoSubmit: false,
overflow: false,
overflowMargin: 25,
selectFirst: false,
filter: null,
filterCase: false,
filterSubset: false,
forceSelect: false,
selectMode: true,
choicesMatch: null,
multiple: false,
separator: ', ',
separatorSplit: /\s*[,;]\s*/,
autoTrim: false,
allowDupes: false,
cache: true,
relative: false
},
initialize: function (element, options) {
this.element = document.id(element);
this.setOptions(options);
this.build();
this.observer = new Observer(this.element, this.prefetch.bind(this),
Object.merge({}, {
'delay': this.options.delay
}, this.options.observerOptions));
this.queryValue = null;
if (this.options.filter) this.filter = this.options.filter.bind(this);
var mode = this.options.selectMode;
this.typeAhead = (mode == 'type-ahead');
this.selectMode = (mode === true) ? 'selection' : mode;
this.cached = [];
},
build: function () {
if (document.id(this.options.customChoices)) {
this.choices = this.options.customChoices;
} else {
this.choices = new Element('ul', {
'class': this.options.className,
'styles': {
'zIndex': this.options.zIndex
}
}).inject(document.body);
this.relative = false;
if (this.options.relative) {
this.choices.inject(this.element, 'after');
this.relative = this.element.getOffsetParent();
}
this.fix = new OverlayFix(this.choices);
}
if (!this.options.separator.test(this.options.separatorSplit)) {
this.options.separatorSplit = this.options.separator;
}
this.fx = (!this.options.fxOptions) ? null : new Fx.Tween(this.choices,
Object.merge({}, {
'property': 'opacity',
'link': 'cancel',
'duration': 200
}, this.options.fxOptions)).addEvent('onStart',
Chain.prototype.clearChain).set(0);
this.element.setProperty('autocomplete',
'off').addEvent((Browser.ie || Browser.safari || Browser.chrome)
? 'keydown' : 'keypress',
this.onCommand.bind(this)).addEvent('click',
this.onCommand.bind(this, [false])).addEvent('focus',
this.toggleFocus.pass({
bind: this,
arguments: true,
delay: 100
})).addEvent('blur', this.toggleFocus.pass({
bind: this,
arguments: false,
delay: 100
}));
},
destroy: function () {
if (this.fix) this.fix.destroy();
this.choices = this.selected = this.choices.destroy();
},
toggleFocus: function (state) {
this.focussed = state;
if (!state) this.hideChoices(true);
this.fireEvent((state) ? 'onFocus' : 'onBlur',
[this.element]);
},
onCommand: function (e) {
if (!e && this.focussed) return this.prefetch();
if (e && e.key && !e.shift) {
switch (e.key) {
case 'enter':
if (this.element.value != this.opted) return true;
if (this.selected && this.visible) {
this.choiceSelect(this.selected);
return !!(this.options.autoSubmit);
}
break;
case 'up':
case 'down':
if (!this.prefetch() && this.queryValue !== null) {
var up = (e.key == 'up');
this.choiceOver((this.selected || this.choices)[(this.selected) ?
((up) ? 'getPrevious' : 'getNext') : ((up) ?
'getLast' : 'getFirst')](this.options.choicesMatch),
true);
}
return false;
case 'esc':
case 'tab':
this.hideChoices(true);
break;
}
}
return true;
},
setSelection: function (finish) {
var input = this.selected.inputValue,
value = input;
var start = this.queryValue.length,
end = input.length;
if (input.substr(0, start).toLowerCase() !=
this.queryValue.toLowerCase()) start = 0;
if (this.options.multiple) {
var split = this.options.separatorSplit;
value = this.element.value;
start += this.queryIndex;
end += this.queryIndex;
var old = value.substr(this.queryIndex).split(split, 1)[0];
value = value.substr(0, this.queryIndex) + input +
value.substr(this.queryIndex + old.length);
if (finish) {
var tokens = value.split(this.options.separatorSplit).filter(function
(entry) {
return this.test(entry);
}, /[^\s,]+/);
if (!this.options.allowDupes) tokens = [].combine(tokens);
var sep = this.options.separator;
value = tokens.join(sep) + sep;
end = value.length;
}
}
this.observer.setValue(value);
this.opted = value;
if (finish || this.selectMode == 'pick') start = end;
this.element.selectRange(start, end);
this.fireEvent('onSelection', [this.element, this.selected,
value, input]);
},
showChoices: function () {
var match = this.options.choicesMatch,
first = this.choices.getFirst(match);
this.selected = this.selectedValue = null;
if (this.fix) {
var pos = this.element.getCoordinates(this.relative),
width = this.options.width || 'auto';
this.choices.setStyles({
'left': pos.left,
'top': pos.bottom,
'width': (width === true || width == 'inherit') ?
pos.width : width
});
}
if (!first) return;
if (!this.visible) {
this.visible = true;
this.choices.setStyle('display', '');
if (this.fx) this.fx.start(1);
this.fireEvent('onShow', [this.element, this.choices]);
}
if (this.options.selectFirst || this.typeAhead || first.inputValue ==
this.queryValue) this.choiceOver(first, this.typeAhead);
var items = this.choices.getChildren(match),
max = this.options.maxChoices;
var styles = {
'overflowY': 'hidden',
'height': ''
};
this.overflown = false;
if (items.length > max) {
var item = items[max - 1];
styles.overflowY = 'scroll';
styles.height = item.getCoordinates(this.choices).bottom;
this.overflown = true;
};
this.choices.setStyles(styles);
this.fix.show();
if (this.options.visibleChoices) {
var scroll = document.getScroll(),
size = document.getSize(),
coords = this.choices.getCoordinates();
if (coords.right > scroll.x + size.x) scroll.x = coords.right -
size.x;
if (coords.bottom > scroll.y + size.y) scroll.y = coords.bottom -
size.y;
window.scrollTo(Math.min(scroll.x, coords.left), Math.min(scroll.y,
coords.top));
}
},
// TODO: No $arguments in MT 1.3
hideChoices: function (clear) {
if (clear) {
var value = this.element.value;
if (this.options.forceSelect) value = this.opted;
if (this.options.autoTrim) {
value =
value.split(this.options.separatorSplit).filter($arguments(0)).join(this.options.separator);
}
this.observer.setValue(value);
}
if (!this.visible) return;
this.visible = false;
if (this.selected)
this.selected.removeClass('autocompleter-selected');
this.observer.clear();
var hide = function () {
this.choices.setStyle('display', 'none');
this.fix.hide();
}.bind(this);
if (this.fx) this.fx.start(0).chain(hide);
else hide();
this.fireEvent('onHide', [this.element, this.choices]);
},
prefetch: function () {
var value = this.element.value,
query = value;
if (this.options.multiple) {
var split = this.options.separatorSplit;
var values = value.split(split);
var index = this.element.getSelectedRange().start;
var toIndex = value.substr(0, index).split(split);
var last = toIndex.length - 1;
index -= toIndex[last].length;
query = values[last];
}
if (query.length < this.options.minLength) {
this.hideChoices();
} else {
if (query === this.queryValue || (this.visible && query ==
this.selectedValue)) {
if (this.visible) return false;
this.showChoices();
} else {
this.queryValue = query;
this.queryIndex = index;
if (!this.fetchCached()) this.query();
}
}
return true;
},
fetchCached: function () {
return false;
if (!this.options.cache || !this.cached || !this.cached.length ||
this.cached.length >= this.options.maxChoices || this.queryValue) return
false;
this.update(this.filter(this.cached));
return true;
},
update: function (tokens) {
this.choices.empty();
this.cached = tokens;
var type = tokens && typeOf(tokens);
if (!type || (type == 'array' && !tokens.length) ||
(type == 'hash' && !tokens.getLength())) {
(this.options.emptyChoices || this.hideChoices).call(this);
} else {
if (this.options.maxChoices < tokens.length &&
!this.options.overflow) tokens.length = this.options.maxChoices;
tokens.each(this.options.injectChoice ||
function (token) {
var choice = new Element('li', {
'html': this.markQueryValue(token)
});
choice.inputValue = token;
this.addChoiceEvents(choice).inject(this.choices);
}, this);
this.showChoices();
}
},
choiceOver: function (choice, selection) {
if (!choice || choice == this.selected) return;
if (this.selected)
this.selected.removeClass('autocompleter-selected');
this.selected = choice.addClass('autocompleter-selected');
this.fireEvent('onSelect', [this.element, this.selected,
selection]);
if (!this.selectMode) this.opted = this.element.value;
if (!selection) return;
this.selectedValue = this.selected.inputValue;
if (this.overflown) {
var coords = this.selected.getCoordinates(this.choices),
margin = this.options.overflowMargin,
top = this.choices.scrollTop,
height = this.choices.offsetHeight,
bottom = top + height;
if (coords.top - margin < top && top) this.choices.scrollTop
= Math.max(coords.top - margin, 0);
else if (coords.bottom + margin > bottom) this.choices.scrollTop =
Math.min(coords.bottom - height + margin, bottom);
}
if (this.selectMode) this.setSelection();
},
choiceSelect: function (choice) {
if (choice) this.choiceOver(choice);
this.setSelection(true);
this.queryValue = false;
this.hideChoices();
},
filter: function (tokens) {
return (tokens || this.tokens).filter(function (token) {
return this.test(token);
}, new RegExp(((this.options.filterSubset) ? '' :
'^') + this.queryValue.escapeRegExp(), (this.options.filterCase)
? '' : 'i'));
},
markQueryValue: function (str) {
return (!this.options.markQuery || !this.queryValue) ? str :
str.replace(new RegExp('(' + ((this.options.filterSubset) ?
'' : '^') + this.queryValue.escapeRegExp() +
')', (this.options.filterCase) ? '' : 'i'),
'<span
class="autocompleter-queried">$1</span>');
},
addChoiceEvents: function (el) {
return el.addEvents({
'mouseover': this.choiceOver.bind(this, el),
'click': this.choiceSelect.bind(this, el)
});
}
});
var OverlayFix = new Class({
initialize: function (el) {
if (Browser.ie) {
this.element = document.id(el);
this.relative = this.element.getOffsetParent();
this.fix = new Element('iframe', {
'frameborder': '0',
'scrolling': 'no',
'src': 'javascript:false;',
'styles': {
'position': 'absolute',
'border': 'none',
'display': 'none',
'filter':
'progid:DXImageTransform.Microsoft.Alpha(opacity=0)'
}
}).inject(this.element, 'after');
}
},
show: function () {
if (this.fix) {
var coords = this.element.getCoordinates(this.relative);
delete coords.right;
delete coords.bottom;
this.fix.setStyles(Object.append(coords, {
'display': '',
'zIndex': (this.element.getStyle('zIndex') || 1) -
1
}));
}
return this;
},
hide: function () {
if (this.fix) this.fix.setStyle('display', 'none');
return this;
},
destroy: function () {
if (this.fix) this.fix = this.fix.destroy();
}
});
Element.implement({
getSelectedRange: function () {
if (!Browser.ie) return {
start: this.selectionStart,
end: this.selectionEnd
};
var pos = {
start: 0,
end: 0
};
var range = this.getDocument().selection.createRange();
if (!range || range.parentElement() != this) return pos;
var dup = range.duplicate();
if (this.type == 'text') {
pos.start = 0 - dup.moveStart('character', -100000);
pos.end = pos.start + range.text.length;
} else {
var value = this.value;
var offset = value.length - value.match(/[\n\r]*$/)[0].length;
dup.moveToElementText(this);
dup.setEndPoint('StartToEnd', range);
pos.end = offset - dup.text.length;
dup.setEndPoint('StartToStart', range);
pos.start = offset - dup.text.length;
}
return pos;
},
selectRange: function (start, end) {
if (Browser.ie) {
var diff = this.value.substr(start, end - start).replace(/\r/g,
'').length;
start = this.value.substr(0, start).replace(/\r/g, '').length;
var range = this.createTextRange();
range.collapse(true);
range.moveEnd('character', start + diff);
range.moveStart('character', start);
range.select();
} else {
this.focus();
this.setSelectionRange(start, end);
}
return this;
}
});
Autocompleter.Base = Autocompleter;
Autocompleter.Request = new Class({
Extends: Autocompleter,
options: {
postData: {},
ajaxOptions: {},
postVar: 'value'
},
query: function () {
var data = this.options.postData.unlink || {};
data[this.options.postVar] = this.queryValue;
var indicator = document.id(this.options.indicator);
if (indicator) indicator.setStyle('display', '');
var cls = this.options.indicatorClass;
if (cls) this.element.addClass(cls);
this.fireEvent('onRequest', [this.element, this.request, data,
this.queryValue]);
this.request.send({
'data': data
});
},
queryResponse: function () {
var indicator = document.id(this.options.indicator);
if (indicator) indicator.setStyle('display', 'none');
var cls = this.options.indicatorClass;
if (cls) this.element.removeClass(cls);
return this.fireEvent('onComplete', [this.element,
this.request]);
}
});
Autocompleter.Request.JSON = new Class({
Extends: Autocompleter.Request,
initialize: function (el, url, options) {
this.parent(el, options);
this.request = new Request.JSON(Object.merge({}, {
'url': url,
'link': 'cancel'
}, this.options.ajaxOptions)).addEvent('onComplete',
this.queryResponse.bind(this));
},
queryResponse: function (response) {
this.parent();
this.update(response);
}
});
Autocompleter.Request.HTML = new Class({
Extends: Autocompleter.Request,
initialize: function (el, url, options) {
this.parent(el, options);
this.request = new Request.HTML(Object.merge({}, {
'url': url,
'link': 'cancel',
'update': this.choices
}, this.options.ajaxOptions)).addEvent('onComplete',
this.queryResponse.bind(this));
},
queryResponse: function (tree, elements) {
this.parent();
if (!elements || !elements.length) {
this.hideChoices();
} else {
this.choices.getChildren(this.options.choicesMatch).each(this.options.injectChoice
||
function (choice) {
var value = choice.innerHTML;
choice.inputValue = value;
this.addChoiceEvents(choice.set('html',
this.markQueryValue(value)));
}, this);
this.showChoices();
}
}
});
Autocompleter.Ajax = {
Base: Autocompleter.Request,
Json: Autocompleter.Request.JSON,
Xhtml: Autocompleter.Request.HTML
};
PK��[k��`qq
js/indexer.jsnu�[���var FinderIndexer
= function(){
var totalItems= null;
var batchSize= null;
var offset= null;
var progress= null;
var optimized= false;
var path =
'index.php?option=com_finder&tmpl=component&format=json';
var initialize = function () {
offset = 0;
progress = 0;
path = path + '&' +
jQuery('#finder-indexer-token').attr('name') +
'=1';
getRequest('indexer.start');
};
var getRequest= function (task) {
jQuery.ajax({
type : "GET",
url : path,
data : 'task=' + task,
dataType : 'json',
success : handleResponse,
error : handleFailure
});
};
var handleResponse = function (json, resp) {
try {
if (json === null) {
throw resp;
}
if (json.error) {
throw json;
}
if (json.start) {
totalItems = json.totalItems;
}
offset += json.batchOffset;
updateProgress(json.header, json.message);
if (offset < totalItems) {
getRequest('indexer.batch');
} else if (!optimized) {
optimized = true;
getRequest('indexer.optimize');
}
} catch (error) {
jQuery('#progress').remove();
try {
if (json.error) {
jQuery('#finder-progress-header').text(json.header).addClass('finder-error');
jQuery('#finder-progress-message').html(json.message).addClass('finder-error');
}
} catch (ignore) {
if (error === '') {
error = Joomla.JText._('COM_FINDER_NO_ERROR_RETURNED');
}
jQuery('#finder-progress-header').text(Joomla.JText._('COM_FINDER_AN_ERROR_HAS_OCCURRED')).addClass('finder-error');
jQuery('#finder-progress-message').html(error).addClass('finder-error');
}
}
return true;
};
var handleFailure= function (xhr) {
json = (typeof xhr == 'object' && xhr.responseText) ?
xhr.responseText : null;
json = json ? jQuery.parseJSON(json) : null;
jQuery('#progress').remove();
if (json) {
json = json.responseText != null ? Json.evaluate(json.responseText,
true) : json;
}
var header = json ? json.header :
Joomla.JText._('COM_FINDER_AN_ERROR_HAS_OCCURRED');
var message = json ? json.message :
Joomla.JText._('COM_FINDER_MESSAGE_RETURNED') + ' <br
/>' + json;
jQuery('#finder-progress-header').text(header).addClass('finder-error');
jQuery('#finder-progress-message').html(message).addClass('finder-error');
};
var updateProgress = function (header, message) {
progress = (offset / totalItems) * 100;
jQuery('#finder-progress-header').text(header);
jQuery('#finder-progress-message').html(message);
if (progress < 100) {
jQuery('#progress-bar').css('width', progress +
'%').attr('aria-valuenow', progress);
}
else {
jQuery('#progress-bar').removeClass('bar-success').addClass('bar-warning').attr('aria-valuemin',
100).attr('aria-valuemax', 200);
jQuery('#progress-bar').css('width', progress +
'%').attr('aria-valuenow', progress);
}
if (message == msg) {
jQuery('#progress').remove();
window.parent.jQuery('#modal-archive',
parent.document).modal('hide');
}
};
initialize();
};
jQuery(function () {
Indexer = new FinderIndexer();
if (typeof window.parent.SqueezeBox == 'object') {
jQuery(window.parent.SqueezeBox).on('close', function () {
window.parent.location.reload(true);
});
}
});
PK"�[|I�33
access.xmlnu�[���PK"�[T�:�d"d"
mconfig.xmlnu�[���PK"�[�.����&controller.phpnu�[���PK"�[Eӓ��,controllers/filter.phpnu�[���PK"�[.M||IJcontrollers/filters.phpnu�[���PK"�[c�G'GGNcontrollers/index.phpnu�[���PK"�[��$��)�)�Tcontrollers/indexer.json.phpnu�[���PK"�[0ɭ�tt�~controllers/maps.phpnu�[���PK"�[����AA
h�finder.phpnu�[���PK"�[ŜA
�finder.xmlnu�[���PK"�[f���P P +�helpers/finder.phpnu�[���PK"�[L&Sߢ���helpers/html/finder.phpnu�[���PK"�[X��!�U�U��helpers/indexer/adapter.phpnu�[���PK"�[�o���K�K
��helpers/indexer/driver/mysql.phpnu�[���PK"�[�RC��?�?%�Ehelpers/indexer/driver/postgresql.phpnu�[���PK"�[f�X�GGGG!��helpers/indexer/driver/sqlsrv.phpnu�[���PK"�[:��̴8�8X�helpers/indexer/helper.phpnu�[���PK"�[.!���:�:Vhelpers/indexer/indexer.phpnu�[���PK"�[�:�{{MAhelpers/indexer/parser/html.phpnu�[���PK"�[�>LO--Shelpers/indexer/parser/rtf.phpnu�[���PK"�[�/AB���Whelpers/indexer/parser/txt.phpnu�[���PK"�[1
�G���Zhelpers/indexer/parser.phpnu�[���PK"�[��,̋̋�fhelpers/indexer/query.phpnu�[���PK"�[��FL""�helpers/indexer/result.phpnu�[���PK"�[��h(h(nhelpers/indexer/stemmer/fr.phpnu�[���PK"�[�(''%$>helpers/indexer/stemmer/porter_en.phpnu�[���PK"�[�Ӷ��
�
$�ehelpers/indexer/stemmer/snowball.phpnu�[���PK"�[x����phelpers/indexer/stemmer.phpnu�[���PK"�[̝Vqj'j'�xhelpers/indexer/taxonomy.phpnu�[���PK"�[�j���F�helpers/indexer/token.phpnu�[���PK"�[(��~##K�helpers/language.phpnu�[���PK"�[�x�����models/fields/branches.phpnu�[���PK"�[F�=�zz�models/fields/contentmap.phpnu�[���PK"�[KY!����models/fields/contenttypes.phpnu�[���PK"�[�������models/fields/directories.phpnu�[���PK"�[�K
����models/fields/searchfilter.phpnu�[���PK"�[^o D}}v�models/filter.phpnu�[���PK"�[[�l.4�models/filters.phpnu�[���PK"�[?�=�jjwmodels/forms/filter.xmlnu�[���PK"�[�@*$��(models/forms/filter_filters.xmlnu�[���PK"�[}SB((/models/forms/filter_index.xmlnu�[���PK"�[��o����models/forms/filter_maps.xmlnu�[���PK"�[��-�-�$models/index.phpnu�[���PK"�[�r����Rmodels/indexer.phpnu�[���PK"�[ᅑ�**�Tmodels/maps.phpnu�[���PK"�[���VV�~models/statistics.phpnu�[���PK"�[��aa;a;v�sql/install.mysql.sqlnu�[���PK"�[�;
���sql/install.postgresql.sqlnu�[���PK"�[�1�iwwTesql/uninstall.mysql.sqlnu�[���PK"�[�|��wwjsql/uninstall.postgresql.sqlnu�[���PK"�[h�܅���ntables/filter.phpnu�[���PK"�[>��u__��tables/link.phpnu�[���PK"�[�&�� � ��tables/map.phpnu�[���PK"�[^�M�
�
��views/filter/tmpl/edit.phpnu�[���PK"�[h��h
h
�views/filter/view.html.phpnu�[���PK"�[aۇD����views/filters/tmpl/default.phpnu�[���PK"�[������views/filters/view.html.phpnu�[���PK"�['�e�����views/index/tmpl/default.phpnu�[���PK"�[۾}__��views/index/view.html.phpnu�[���PK"�[�
@�rrM�views/indexer/tmpl/default.phpnu�[���PK"�[�����
�views/indexer/view.html.phpnu�[���PK"�[{�s���views/maps/tmpl/default.phpnu�[���PK"�[
�u���views/maps/view.html.phpnu�[���PK"�[^��;OO!�views/statistics/tmpl/default.phpnu�[���PK"�[�W���r'views/statistics/view.html.phpnu�[���PK��[J�bW��
�+css/dates.cssnu�[���PK��[2y4|� � p-css/finder.cssnu�[���PK��[�����n7css/indexer.cssnu�[���PK��[��.j�?�?~8js/autocompleter.jsnu�[���PK��[k��`qq
Pxjs/indexer.jsnu�[���PKFF��