Differences Between: [Versions 310 and 402] [Versions 311 and 402] [Versions 39 and 402] [Versions 400 and 402] [Versions 401 and 402]
1 <?php 2 // This file is part of Moodle - http://moodle.org/ 3 // 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. 8 // 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. 13 // 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/>. 16 17 /** 18 * Search base class to be extended by search areas. 19 * 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 23 */ 24 25 namespace core_search; 26 27 defined('MOODLE_INTERNAL') || die(); 28 29 /** 30 * Base search implementation. 31 * 32 * Components and plugins interested in filling the search engine with data should extend this class (or any extension of this 33 * class). 34 * 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 38 */ 39 abstract class base { 40 41 /** 42 * The area name as defined in the class name. 43 * 44 * @var string 45 */ 46 protected $areaname = null; 47 48 /** 49 * The component frankenstyle name. 50 * 51 * @var string 52 */ 53 protected $componentname = null; 54 55 /** 56 * The component type (core or the plugin type). 57 * 58 * @var string 59 */ 60 protected $componenttype = null; 61 62 /** 63 * The context levels the search implementation is working on. 64 * 65 * @var array 66 */ 67 protected static $levels = [CONTEXT_SYSTEM]; 68 69 /** 70 * An area id from the componentname and the area name. 71 * 72 * @var string 73 */ 74 public $areaid; 75 76 /** 77 * Constructor. 78 * 79 * @throws \coding_exception 80 * @return void 81 */ 82 public final function __construct() { 83 84 $classname = get_class($this); 85 86 // Detect possible issues when defining the class. 87 if (strpos($classname, '\search') === false) { 88 throw new \coding_exception('Search area classes should be located in \PLUGINTYPE_PLUGINNAME\search\AREANAME.'); 89 } else if (strpos($classname, '_') === false) { 90 throw new \coding_exception($classname . ' class namespace level 1 should be its component frankenstyle name'); 91 } 92 93 $this->areaname = substr(strrchr($classname, '\\'), 1); 94 $this->componentname = substr($classname, 0, strpos($classname, '\\')); 95 $this->areaid = \core_search\manager::generate_areaid($this->componentname, $this->areaname); 96 $this->componenttype = substr($this->componentname, 0, strpos($this->componentname, '_')); 97 } 98 99 /** 100 * Returns context levels property. 101 * 102 * @return int 103 */ 104 public static function get_levels() { 105 return static::$levels; 106 } 107 108 /** 109 * Returns the area id. 110 * 111 * @return string 112 */ 113 public function get_area_id() { 114 return $this->areaid; 115 } 116 117 /** 118 * Returns the moodle component name. 119 * 120 * It might be the plugin name (whole frankenstyle name) or the core subsystem name. 121 * 122 * @return string 123 */ 124 public function get_component_name() { 125 return $this->componentname; 126 } 127 128 /** 129 * Returns the component type. 130 * 131 * It might be a plugintype or 'core' for core subsystems. 132 * 133 * @return string 134 */ 135 public function get_component_type() { 136 return $this->componenttype; 137 } 138 139 /** 140 * Returns the area visible name. 141 * 142 * @param bool $lazyload Usually false, unless when in admin settings. 143 * @return string 144 */ 145 public function get_visible_name($lazyload = false) { 146 147 $component = $this->componentname; 148 149 // Core subsystem strings go to lang/XX/search.php. 150 if ($this->componenttype === 'core') { 151 $component = 'search'; 152 } 153 return get_string('search:' . $this->areaname, $component, null, $lazyload); 154 } 155 156 /** 157 * Returns the config var name. 158 * 159 * It depends on whether it is a moodle subsystem or a plugin as plugin-related config should remain in their own scope. 160 * 161 * @access private 162 * @return string Config var path including the plugin (or component) and the varname 163 */ 164 public function get_config_var_name() { 165 166 if ($this->componenttype === 'core') { 167 // Core subsystems config in core_search and setting name using only [a-zA-Z0-9_]+. 168 $parts = \core_search\manager::extract_areaid_parts($this->areaid); 169 return array('core_search', $parts[0] . '_' . $parts[1]); 170 } 171 172 // Plugins config in the plugin scope. 173 return array($this->componentname, 'search_' . $this->areaname); 174 } 175 176 /** 177 * Returns all the search area configuration. 178 * 179 * @return array 180 */ 181 public function get_config() { 182 list($componentname, $varname) = $this->get_config_var_name(); 183 184 $config = []; 185 $settingnames = self::get_settingnames(); 186 foreach ($settingnames as $name) { 187 $config[$varname . $name] = get_config($componentname, $varname . $name); 188 } 189 190 // Search areas are enabled by default. 191 if ($config[$varname . '_enabled'] === false) { 192 $config[$varname . '_enabled'] = 1; 193 } 194 return $config; 195 } 196 197 /** 198 * Return a list of all required setting names. 199 * 200 * @return array 201 */ 202 public static function get_settingnames() { 203 return array('_enabled', '_indexingstart', '_indexingend', '_lastindexrun', 204 '_docsignored', '_docsprocessed', '_recordsprocessed', '_partial'); 205 } 206 207 /** 208 * Is the search component enabled by the system administrator? 209 * 210 * @return bool 211 */ 212 public function is_enabled() { 213 list($componentname, $varname) = $this->get_config_var_name(); 214 215 $value = get_config($componentname, $varname . '_enabled'); 216 217 // Search areas are enabled by default. 218 if ($value === false) { 219 $value = 1; 220 } 221 return (bool)$value; 222 } 223 224 public function set_enabled($isenabled) { 225 list($componentname, $varname) = $this->get_config_var_name(); 226 return set_config($varname . '_enabled', $isenabled, $componentname); 227 } 228 229 /** 230 * Gets the length of time spent indexing this area (the last time it was indexed). 231 * 232 * @return int|bool Time in seconds spent indexing this area last time, false if never indexed 233 */ 234 public function get_last_indexing_duration() { 235 list($componentname, $varname) = $this->get_config_var_name(); 236 $start = get_config($componentname, $varname . '_indexingstart'); 237 $end = get_config($componentname, $varname . '_indexingend'); 238 if ($start && $end) { 239 return $end - $start; 240 } else { 241 return false; 242 } 243 } 244 245 /** 246 * Returns true if this area uses file indexing. 247 * 248 * @return bool 249 */ 250 public function uses_file_indexing() { 251 return false; 252 } 253 254 /** 255 * Returns a recordset ordered by modification date ASC. 256 * 257 * Each record can include any data self::get_document might need but it must: 258 * - Include an 'id' field: Unique identifier (in this area's scope) of a document to index in the search engine 259 * If the indexed content field can contain embedded files, the 'id' value should match the filearea itemid. 260 * - Only return data modified since $modifiedfrom, including $modifiedform to prevent 261 * some records from not being indexed (e.g. your-timemodified-fieldname >= $modifiedfrom) 262 * - Order the returned data by time modified in ascending order, as \core_search::manager will need to store the modified time 263 * of the last indexed document. 264 * 265 * Since Moodle 3.4, subclasses should instead implement get_document_recordset, which has 266 * an additional context parameter. This function continues to work for implementations which 267 * haven't been updated, or where the context parameter is not required. 268 * 269 * @param int $modifiedfrom 270 * @return \moodle_recordset 271 */ 272 public function get_recordset_by_timestamp($modifiedfrom = 0) { 273 $result = $this->get_document_recordset($modifiedfrom); 274 if ($result === false) { 275 throw new \coding_exception( 276 'Search area must implement get_document_recordset or get_recordset_by_timestamp'); 277 } 278 return $result; 279 } 280 281 /** 282 * Returns a recordset containing all items from this area, optionally within the given context, 283 * and including only items modifed from (>=) the specified time. The recordset must be ordered 284 * in ascending order of modified time. 285 * 286 * Each record can include any data self::get_document might need. It must include an 'id' 287 * field,a unique identifier (in this area's scope) of a document to index in the search engine. 288 * If the indexed content field can contain embedded files, the 'id' value should match the 289 * filearea itemid. 290 * 291 * The return value can be a recordset, null (if this area does not provide any results in the 292 * given context and there is no need to do a database query to find out), or false (if this 293 * facility is not currently supported by this search area). 294 * 295 * If this function returns false, then: 296 * - If indexing the entire system (no context restriction) the search indexer will try 297 * get_recordset_by_timestamp instead 298 * - If trying to index a context (e.g. when restoring a course), the search indexer will not 299 * index this area, so that restored content may not be indexed. 300 * 301 * The default implementation returns false, indicating that this facility is not supported and 302 * the older get_recordset_by_timestamp function should be used. 303 * 304 * This function must accept all possible values for the $context parameter. For example, if 305 * you are implementing this function for the forum module, it should still operate correctly 306 * if called with the context for a glossary module, or for the HTML block. (In these cases 307 * where it will not return any data, it may return null.) 308 * 309 * The $context parameter can also be null or the system context; both of these indicate that 310 * all data, without context restriction, should be returned. 311 * 312 * @param int $modifiedfrom Return only records modified after this date 313 * @param \context|null $context Context (null means no context restriction) 314 * @return \moodle_recordset|null|false Recordset / null if no results / false if not supported 315 * @since Moodle 3.4 316 */ 317 public function get_document_recordset($modifiedfrom = 0, \context $context = null) { 318 return false; 319 } 320 321 /** 322 * Checks if get_document_recordset is supported for this search area. 323 * 324 * For many uses you can simply call get_document_recordset and see if it returns false, but 325 * this function is useful when you don't want to actually call the function right away. 326 */ 327 public function supports_get_document_recordset() { 328 // Easiest way to check this is simply to see if the class has overridden the default 329 // function. 330 $method = new \ReflectionMethod($this, 'get_document_recordset'); 331 return $method->getDeclaringClass()->getName() !== self::class; 332 } 333 334 /** 335 * Returns the document related with the provided record. 336 * 337 * This method receives a record with the document id and other info returned by get_recordset_by_timestamp 338 * or get_recordset_by_contexts that might be useful here. The idea is to restrict database queries to 339 * minimum as this function will be called for each document to index. As an alternative, use cached data. 340 * 341 * Internally it should use \core_search\document to standarise the documents before sending them to the search engine. 342 * 343 * Search areas should send plain text to the search engine, use the following function to convert any user 344 * input data to plain text: {@link content_to_text} 345 * 346 * Valid keys for the options array are: 347 * indexfiles => File indexing is enabled if true. 348 * lastindexedtime => The last time this area was indexed. 0 if never indexed. 349 * 350 * The lastindexedtime value is not set if indexing a specific context rather than the whole 351 * system. 352 * 353 * @param \stdClass $record A record containing, at least, the indexed document id and a modified timestamp 354 * @param array $options Options for document creation 355 * @return \core_search\document 356 */ 357 abstract public function get_document($record, $options = array()); 358 359 /** 360 * Returns the document title to display. 361 * 362 * Allow to customize the document title string to display. 363 * 364 * @param \core_search\document $doc 365 * @return string Document title to display in the search results page 366 */ 367 public function get_document_display_title(\core_search\document $doc) { 368 369 return $doc->get('title'); 370 } 371 372 /** 373 * Return the context info required to index files for 374 * this search area. 375 * 376 * Should be onerridden by each search area. 377 * 378 * @return array 379 */ 380 public function get_search_fileareas() { 381 $fileareas = array(); 382 383 return $fileareas; 384 } 385 386 /** 387 * Files related to the current document are attached, 388 * to the document object ready for indexing by 389 * Global Search. 390 * 391 * The default implementation retrieves all files for 392 * the file areas returned by get_search_fileareas(). 393 * If you need to filter files to specific items per 394 * file area, you will need to override this method 395 * and explicitly provide the items. 396 * 397 * @param document $document The current document 398 * @return void 399 */ 400 public function attach_files($document) { 401 $fileareas = $this->get_search_fileareas(); 402 $contextid = $document->get('contextid'); 403 $component = $this->get_component_name(); 404 $itemid = $document->get('itemid'); 405 406 foreach ($fileareas as $filearea) { 407 $fs = get_file_storage(); 408 $files = $fs->get_area_files($contextid, $component, $filearea, $itemid, '', false); 409 410 foreach ($files as $file) { 411 $document->add_stored_file($file); 412 } 413 } 414 415 } 416 417 /** 418 * Can the current user see the document. 419 * 420 * @param int $id The internal search area entity id. 421 * @return int manager:ACCESS_xx constant 422 */ 423 abstract public function check_access($id); 424 425 /** 426 * Returns a url to the document, it might match self::get_context_url(). 427 * 428 * @param \core_search\document $doc 429 * @return \moodle_url 430 */ 431 abstract public function get_doc_url(\core_search\document $doc); 432 433 /** 434 * Returns a url to the document context. 435 * 436 * @param \core_search\document $doc 437 * @return \moodle_url 438 */ 439 abstract public function get_context_url(\core_search\document $doc); 440 441 /** 442 * Helper function that gets SQL useful for restricting a search query given a passed-in 443 * context, for data stored at course level. 444 * 445 * The SQL returned will be zero or more JOIN statements, surrounded by whitespace, which act 446 * as restrictions on the query based on the rows in a module table. 447 * 448 * You can pass in a null or system context, which will both return an empty string and no 449 * params. 450 * 451 * Returns an array with two nulls if there can be no results for a course within this context. 452 * 453 * If named parameters are used, these will be named gclcrs0, gclcrs1, etc. The table aliases 454 * used in SQL also all begin with gclcrs, to avoid conflicts. 455 * 456 * @param \context|null $context Context to restrict the query 457 * @param string $coursetable Name of alias for course table e.g. 'c' 458 * @param int $paramtype Type of SQL parameters to use (default question mark) 459 * @return array Array with SQL and parameters; both null if no need to query 460 * @throws \coding_exception If called with invalid params 461 */ 462 protected function get_course_level_context_restriction_sql(?\context $context, 463 $coursetable, $paramtype = SQL_PARAMS_QM) { 464 global $DB; 465 466 if (!$context) { 467 return ['', []]; 468 } 469 470 switch ($paramtype) { 471 case SQL_PARAMS_QM: 472 $param1 = '?'; 473 $param2 = '?'; 474 $key1 = 0; 475 $key2 = 1; 476 break; 477 case SQL_PARAMS_NAMED: 478 $param1 = ':gclcrs0'; 479 $param2 = ':gclcrs1'; 480 $key1 = 'gclcrs0'; 481 $key2 = 'gclcrs1'; 482 break; 483 default: 484 throw new \coding_exception('Unexpected $paramtype: ' . $paramtype); 485 } 486 487 $params = []; 488 switch ($context->contextlevel) { 489 case CONTEXT_SYSTEM: 490 $sql = ''; 491 break; 492 493 case CONTEXT_COURSECAT: 494 // Find all courses within the specified category or any sub-category. 495 $pathmatch = $DB->sql_like('gclcrscc2.path', 496 $DB->sql_concat('gclcrscc1.path', $param2)); 497 $sql = " JOIN {course_categories} gclcrscc1 ON gclcrscc1.id = $param1 498 JOIN {course_categories} gclcrscc2 ON gclcrscc2.id = $coursetable.category 499 AND (gclcrscc2.id = gclcrscc1.id OR $pathmatch) "; 500 $params[$key1] = $context->instanceid; 501 // Note: This param is a bit annoying as it obviously never changes, but sql_like 502 // throws a debug warning if you pass it anything with quotes in, so it has to be 503 // a bound parameter. 504 $params[$key2] = '/%'; 505 break; 506 507 case CONTEXT_COURSE: 508 // We just join again against the same course entry and confirm that it has the 509 // same id as the context. 510 $sql = " JOIN {course} gclcrsc ON gclcrsc.id = $coursetable.id 511 AND gclcrsc.id = $param1"; 512 $params[$key1] = $context->instanceid; 513 break; 514 515 case CONTEXT_BLOCK: 516 case CONTEXT_MODULE: 517 case CONTEXT_USER: 518 // Context cannot contain any courses. 519 return [null, null]; 520 521 default: 522 throw new \coding_exception('Unexpected contextlevel: ' . $context->contextlevel); 523 } 524 525 return [$sql, $params]; 526 } 527 528 /** 529 * Gets a list of all contexts to reindex when reindexing this search area. The list should be 530 * returned in an order that is likely to be suitable when reindexing, for example with newer 531 * contexts first. 532 * 533 * The default implementation simply returns the system context, which will result in 534 * reindexing everything in normal date order (oldest first). 535 * 536 * @return \Iterator Iterator of contexts to reindex 537 */ 538 public function get_contexts_to_reindex() { 539 return new \ArrayIterator([\context_system::instance()]); 540 } 541 542 /** 543 * Returns an icon instance for the document. 544 * 545 * @param \core_search\document $doc 546 * @return \core_search\document_icon 547 */ 548 public function get_doc_icon(document $doc) : document_icon { 549 return new document_icon('i/empty'); 550 } 551 552 /** 553 * Returns a list of category names associated with the area. 554 * 555 * @return array 556 */ 557 public function get_category_names() { 558 return [manager::SEARCH_AREA_CATEGORY_OTHER]; 559 } 560 }
title
Description
Body
title
Description
Body
title
Description
Body
title
Body