2 // This file is part of Moodle - http://moodle.org/
4 // Moodle is free software: you can redistribute it and/or modify
5 // it under the terms of the GNU General Public License as published by
6 // the Free Software Foundation, either version 3 of the License, or
7 // (at your option) any later version.
9 // Moodle is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU General Public License for more details.
14 // You should have received a copy of the GNU General Public License
15 // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
18 * Search base class to be extended by search areas.
20 * @package core_search
21 * @copyright 2015 David Monllao {@link http://www.davidmonllao.com}
22 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
25 namespace core_search
;
27 defined('MOODLE_INTERNAL') ||
die();
30 * Base search implementation.
32 * Components and plugins interested in filling the search engine with data should extend this class (or any extension of this
35 * @package core_search
36 * @copyright 2015 David Monllao {@link http://www.davidmonllao.com}
37 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
42 * The area name as defined in the class name.
46 protected $areaname = null;
49 * The component frankenstyle name.
53 protected $componentname = null;
56 * The component type (core or the plugin type).
60 protected $componenttype = null;
63 * The context levels the search implementation is working on.
67 protected static $levels = [CONTEXT_SYSTEM
];
72 * @throws \coding_exception
75 public final function __construct() {
77 $classname = get_class($this);
79 // Detect possible issues when defining the class.
80 if (strpos($classname, '\search') === false) {
81 throw new \
coding_exception('Search area classes should be located in \PLUGINTYPE_PLUGINNAME\search\AREANAME.');
82 } else if (strpos($classname, '_') === false) {
83 throw new \
coding_exception($classname . ' class namespace level 1 should be its component frankenstyle name');
86 $this->areaname
= substr(strrchr($classname, '\\'), 1);
87 $this->componentname
= substr($classname, 0, strpos($classname, '\\'));
88 $this->areaid
= \core_search\manager
::generate_areaid($this->componentname
, $this->areaname
);
89 $this->componenttype
= substr($this->componentname
, 0, strpos($this->componentname
, '_'));
93 * Returns context levels property.
97 public static function get_levels() {
98 return static::$levels;
102 * Returns the area id.
106 public function get_area_id() {
107 return $this->areaid
;
111 * Returns the moodle component name.
113 * It might be the plugin name (whole frankenstyle name) or the core subsystem name.
117 public function get_component_name() {
118 return $this->componentname
;
122 * Returns the component type.
124 * It might be a plugintype or 'core' for core subsystems.
128 public function get_component_type() {
129 return $this->componenttype
;
133 * Returns the area visible name.
135 * @param bool $lazyload Usually false, unless when in admin settings.
138 public function get_visible_name($lazyload = false) {
140 $component = $this->componentname
;
142 // Core subsystem strings go to lang/XX/search.php.
143 if ($this->componenttype
=== 'core') {
144 $component = 'search';
146 return get_string('search:' . $this->areaname
, $component, null, $lazyload);
150 * Returns the config var name.
152 * It depends on whether it is a moodle subsystem or a plugin as plugin-related config should remain in their own scope.
155 * @return string Config var path including the plugin (or component) and the varname
157 public function get_config_var_name() {
159 if ($this->componenttype
=== 'core') {
160 // Core subsystems config in core_search and setting name using only [a-zA-Z0-9_]+.
161 $parts = \core_search\manager
::extract_areaid_parts($this->areaid
);
162 return array('core_search', $parts[0] . '_' . $parts[1]);
165 // Plugins config in the plugin scope.
166 return array($this->componentname
, 'search_' . $this->areaname
);
170 * Returns all the search area configuration.
174 public function get_config() {
175 list($componentname, $varname) = $this->get_config_var_name();
178 $settingnames = array('_enabled', '_indexingstart', '_indexingend', '_lastindexrun', '_docsignored', '_docsprocessed', '_recordsprocessed');
179 foreach ($settingnames as $name) {
180 $config[$varname . $name] = get_config($componentname, $varname . $name);
183 // Search areas are enabled by default.
184 if ($config[$varname . '_enabled'] === false) {
185 $config[$varname . '_enabled'] = 1;
191 * Is the search component enabled by the system administrator?
195 public function is_enabled() {
196 list($componentname, $varname) = $this->get_config_var_name();
198 $value = get_config($componentname, $varname . '_enabled');
200 // Search areas are enabled by default.
201 if ($value === false) {
207 public function set_enabled($isenabled) {
208 list($componentname, $varname) = $this->get_config_var_name();
209 return set_config($varname . '_enabled', $isenabled, $componentname);
213 * Returns true if this area uses file indexing.
217 public function uses_file_indexing() {
222 * Returns a recordset ordered by modification date ASC.
224 * Each record can include any data self::get_document might need but it must:
225 * - Include an 'id' field: Unique identifier (in this area's scope) of a document to index in the search engine
226 * If the indexed content field can contain embedded files, the 'id' value should match the filearea itemid.
227 * - Only return data modified since $modifiedfrom, including $modifiedform to prevent
228 * some records from not being indexed (e.g. your-timemodified-fieldname >= $modifiedfrom)
229 * - Order the returned data by time modified in ascending order, as \core_search::manager will need to store the modified time
230 * of the last indexed document.
232 * @param int $modifiedfrom
233 * @return moodle_recordset
235 abstract public function get_recordset_by_timestamp($modifiedfrom = 0);
238 * Returns the document related with the provided record.
240 * This method receives a record with the document id and other info returned by get_recordset_by_timestamp
241 * or get_recordset_by_contexts that might be useful here. The idea is to restrict database queries to
242 * minimum as this function will be called for each document to index. As an alternative, use cached data.
244 * Internally it should use \core_search\document to standarise the documents before sending them to the search engine.
246 * Search areas should send plain text to the search engine, use the following function to convert any user
247 * input data to plain text: {@link content_to_text}
249 * Valid keys for the options array are:
250 * indexfiles => File indexing is enabled if true.
251 * lastindexedtime => The last time this area was indexed. 0 if never indexed.
253 * @param \stdClass $record A record containing, at least, the indexed document id and a modified timestamp
254 * @param array $options Options for document creation
255 * @return \core_search\document
257 abstract public function get_document($record, $options = array());
260 * Add any files to the document that should be indexed.
262 * @param document $document The current document
265 public function attach_files($document) {
270 * Can the current user see the document.
272 * @param int $id The internal search area entity id.
273 * @return bool True if the user can see it, false otherwise
275 abstract public function check_access($id);
278 * Returns a url to the document, it might match self::get_context_url().
280 * @param \core_search\document $doc
281 * @return \moodle_url
283 abstract public function get_doc_url(\core_search\document
$doc);
286 * Returns a url to the document context.
288 * @param \core_search\document $doc
289 * @return \moodle_url
291 abstract public function get_context_url(\core_search\document
$doc);