ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilSearchResult.php
Go to the documentation of this file.
1 <?php
2 
29 {
30  private string $permission = 'visible';
31 
32  private int $user_id;
33  private array $entries = array();
34  private array $results = array();
35  private array $observers = array();
36  private int $max_hits = 0;
37 
39  protected int $offset = 0;
40 
41  // OBJECT VARIABLES
42  protected ilAccess $ilAccess;
43  protected ilDBInterface $db;
44  protected ilTree $tree;
45  protected ilObjUser $user;
47 
48  // Stores info if MAX HITS is reached or not
49  public bool $limit_reached = false;
50 
51  protected bool $preventOverwritingMaxhits = false;
52 
53  protected ilLogger $logger;
54 
55 
56 
61  public function __construct(int $a_user_id = 0)
62  {
63  global $DIC;
64 
65  $this->logger = $DIC->logger()->src();
66  $this->ilAccess = $DIC->access();
67  $this->db = $DIC->database();
68  $this->tree = $DIC->repositoryTree();
69  $this->user = $DIC->user();
70 
71  if ($a_user_id) {
72  $this->user_id = $a_user_id;
73  } else {
74  $this->user_id = $this->user->getId();
75  }
77  $this->initUserSearchCache();
78  }
79 
83  public function setRequiredPermission(string $a_permission): void
84  {
85  $this->permission = $a_permission;
86  }
87 
88  public function getRequiredPermission(): string
89  {
90  return $this->permission;
91  }
92 
93 
94  public function setUserId(int $a_user_id): void
95  {
96  $this->user_id = $a_user_id;
97  }
98  public function getUserId(): int
99  {
100  return $this->user_id;
101  }
102 
103  public function getEntries(): array
104  {
105  return $this->entries;
106  }
107 
108  public function isLimitReached(): bool
109  {
110  return $this->limit_reached;
111  }
112 
113  public function setMaxHits(int $a_max_hits): void
114  {
115  $this->max_hits = $a_max_hits;
116  }
117  public function getMaxHits(): int
118  {
119  return $this->max_hits;
120  }
121 
125  public function isOffsetReached(int $a_counter): bool
126  {
127  return !($a_counter < $this->offset);
128  }
129 
140  public function addEntry(int $a_obj_id, string $a_type, array $found, int $a_child_id = 0): void
141  {
142  // Create new entry if it not exists
143  if (!isset($this->entries[$a_obj_id])) {
144  $this->entries[$a_obj_id]['obj_id'] = $a_obj_id;
145  $this->entries[$a_obj_id]['type'] = $a_type;
146  $this->entries[$a_obj_id]['found'] = $found;
147  $this->entries[$a_obj_id]['child'] = [];
148 
149  if ($a_child_id and $a_child_id != $a_obj_id) {
150  $this->entries[$a_obj_id]['child'][$a_child_id] = $a_child_id;
151  }
152  } else {
153  // replace or add child ('pg','st') id
154  if ($a_child_id and $a_child_id != $a_obj_id) {
155  $this->entries[$a_obj_id]['child'][$a_child_id] = $a_child_id;
156  }
157  $counter = 0;
158  foreach ($found as $position) {
159  if ($position) {
160  $this->entries[$a_obj_id]['found'][$counter] = $position;
161  }
162  $counter++;
163  }
164  }
165  }
166 
172  public function numEntries(): int
173  {
174  return count($this->getEntries());
175  }
176 
183  public function mergeEntries(ilSearchResult $result_obj): void
184  {
185  foreach ($result_obj->getEntries() as $obj_id => $entry) {
186  $this->addEntry($entry['obj_id'], $entry['type'], $entry['found']);
187  $this->__updateEntryChilds($entry['obj_id'], $entry['child']);
188  }
189  }
190 
195  public function diffEntriesFromResult(): void
196  {
197  $new_entries = $this->getEntries();
198  $this->entries = array();
199 
200  // Get all checked objects
201  foreach ($this->search_cache->getCheckedItems() as $ref_id => $obj_id) {
202  if (isset($new_entries[$obj_id])) {
203  $this->addEntry(
204  $new_entries[$obj_id]['obj_id'],
205  $new_entries[$obj_id]['type'],
206  $new_entries[$obj_id]['found']
207  );
208  $this->__updateEntryChilds(
209  $new_entries[$obj_id]['obj_id'],
210  $new_entries[$obj_id]['child']
211  );
212  }
213  }
214  }
215 
219  public function intersectEntries(ilSearchResult $result_obj): void
220  {
221  $new_entries = $this->getEntries();
222  $this->entries = [];
223  foreach ($result_obj->getEntries() as $entry) {
224  $obj_id = $entry['obj_id'];
225  if (isset($new_entries[$obj_id])) {
226  $this->addEntry(
227  $new_entries[$obj_id]['obj_id'],
228  $new_entries[$obj_id]['type'],
229  $new_entries[$obj_id]['found']
230  );
231 
232  $this->__updateEntryChilds(
233  $new_entries[$obj_id]['obj_id'],
234  $new_entries[$obj_id]['child']
235  );
236  }
237  }
238  }
239 
240  public function addResult(int $a_ref_id, int $a_obj_id, string $a_type): void
241  {
242  $this->results[$a_ref_id]['ref_id'] = $a_ref_id;
243  $this->results[$a_ref_id]['obj_id'] = $a_obj_id;
244  $this->results[$a_ref_id]['type'] = $a_type;
245  }
246 
247  public function getResults(): array
248  {
249  return $this->results;
250  }
251 
256  public function getResultIds(): array
257  {
258  $ids = [];
259  foreach ($this->getResults() as $id => $tmp) {
260  $ids[] = $id;
261  }
262  return $ids;
263  }
264 
265  public function getResultsByObjId(): array
266  {
267  $tmp_res = [];
268  foreach ($this->getResults() as $ref_id => $res_data) {
269  $tmp_res[$res_data['obj_id']][] = $ref_id;
270  }
271  return $tmp_res;
272  }
273 
274 
279  public function getUniqueResults(): array
280  {
281  $obj_ids = [];
282  $objects = [];
283  foreach ($this->results as $result) {
284  if (in_array($result['obj_id'], $obj_ids)) {
285  continue;
286  }
287  $obj_ids[] = $result['obj_id'];
288  $objects[] = $result;
289  }
290  return $objects;
291  }
292 
293  public function getResultsForPresentation(): array
294  {
295  $res = [];
296  foreach ($this->getResults() as $result) {
297  if (!is_array($result)) {
298  continue;
299  }
300 
301  $res[(int) $result['ref_id']] = (int) $result['obj_id'];
302  }
303  return $res;
304  }
305 
306  public function getSubitemIds(): array
307  {
308  $res = array();
309  foreach ($this->getResults() as $row) {
310  $res[$row['obj_id']] = $row['child'] ?? [];
311  }
312  return $res;
313  }
314 
315 
316 
322  public function filter(
323  int $a_root_node,
324  bool $check_and,
325  ?ilDate $creation_filter_date_start = null,
326  ?ilDate $creation_filter_date_end = null
327  ): bool {
328  // get ref_ids and check access
329  $counter = 0;
330  $offset_counter = 0;
331  foreach ($this->getEntries() as $entry) {
332  // boolean and failed continue
333  if ($check_and and in_array(0, $entry['found'])) {
334  continue;
335  }
336  // Types like role, rolt, user do not need rbac checks
337  $type = ilObject::_lookupType($entry['obj_id']);
338  if ($type == 'rolt' or $type == 'usr' or $type == 'role') {
339  if ($this->callListeners($entry['obj_id'], $entry)) {
340  $this->addResult($entry['obj_id'], $entry['obj_id'], $type);
341  if (is_array($entry['child'])) {
342  $counter += count($entry['child']);
343  }
344  // Stop if maximum of hits is reached
345  if (++$counter > $this->getMaxHits()) {
346  $this->limit_reached = true;
347  return true;
348  }
349  }
350  continue;
351  }
352 
353  /*
354  * (Re-)check creation date, needed for searches on other tables than obj_data (35275)
355  * Before- and after-operators also allow matching datetimes, see ilObjectSearch::performSearch.
356  */
357  if (!is_null($creation_filter_date_start) || !is_null($creation_filter_date_end)) {
358  if (
359  !ilObject::_exists($entry['obj_id']) ||
360  ($creation_date_string = ilObject::_lookupCreationDate($entry['obj_id'])) === ''
361  ) {
362  continue;
363  }
364  $creation_date = new ilDate(
365  date('Y-m-d', strtotime($creation_date_string)),
367  );
368 
369  if ($creation_filter_date_start && is_null($creation_filter_date_end)) {
370  if (!ilDate::_after($creation_date, $creation_filter_date_start)) {
371  continue;
372  }
373  } elseif ($creation_filter_date_end && is_null($creation_filter_date_start)) {
374  if (!ilDate::_before($creation_date, $creation_filter_date_end)) {
375  continue;
376  }
377  } else {
378  if (!ilDate::_within($creation_date, $creation_filter_date_start, $creation_filter_date_end)) {
379  continue;
380  }
381  }
382  }
383 
384  // Check referenced objects
385  foreach (ilObject::_getAllReferences((int) $entry['obj_id']) as $ref_id) {
386  // Failed check: if ref id check is failed by previous search
387  if ($this->search_cache->isFailed($ref_id)) {
388  continue;
389  }
390  // Offset check
391  if ($this->search_cache->isChecked($ref_id) and !$this->isOffsetReached($offset_counter)) {
392  ++$offset_counter;
393  continue;
394  }
395 
396  if (!$this->callListeners($ref_id, $entry)) {
397  continue;
398  }
399 
400 
401 
402  // RBAC check
403  $type = ilObject::_lookupType($ref_id, true);
404  if ($this->ilAccess->checkAccessOfUser(
405  $this->getUserId(),
406  $this->getRequiredPermission(),
407  '',
408  $ref_id,
409  $type,
410  $entry['obj_id']
411  )) {
412  if ($a_root_node == ROOT_FOLDER_ID or $this->tree->isGrandChild($a_root_node, $ref_id)) {
413  // Call listeners
414  #if($this->callListeners($ref_id,$entry))
415  if (1) {
416  $this->addResult($ref_id, $entry['obj_id'], $type);
417  $this->search_cache->appendToChecked($ref_id, $entry['obj_id']);
418  $this->__updateResultChilds($ref_id, $entry['child']);
419 
420  $counter++;
421  $offset_counter++;
422  // Stop if maximum of hits is reached
423 
424  if ($counter >= $this->getMaxHits()) {
425  $this->limit_reached = true;
426  $this->search_cache->setResults($this->results);
427  return true;
428  }
429  }
430  }
431  continue;
432  }
433  $this->search_cache->appendToFailed($ref_id);
434  }
435  }
436  $this->search_cache->setResults($this->results);
437  return false;
438  }
439 
445  public function filterResults(int $a_root_node): void
446  {
447  $tmp_results = $this->getResults();
448  $this->results = array();
449  foreach ($tmp_results as $result) {
450  if (isset($result['ref_id']) && $this->tree->isGrandChild($a_root_node, $result['ref_id']) && $this->tree->isInTree($result['ref_id'])) {
451  $this->addResult($result['ref_id'], $result['obj_id'], $result['type']);
452  $this->__updateResultChilds($result['ref_id'], $result['child'] ?? []);
453  }
454  }
455  }
456 
457 
463  public function save(int $a_type = ilUserSearchCache::DEFAULT_SEARCH): void
464  {
465  $this->search_cache->save();
466  }
473  public function read(int $a_type = ilUserSearchCache::DEFAULT_SEARCH): void
474  {
475  $this->results = $this->search_cache->getResults();
476  }
477 
478  // PRIVATE
486  public function __updateEntryChilds(int $a_obj_id, array $a_childs): bool
487  {
488  if ($this->entries[$a_obj_id] and is_array($a_childs)) {
489  foreach ($a_childs as $child_id) {
490  if ($child_id) {
491  $this->entries[$a_obj_id]['child'][$child_id] = $child_id;
492  }
493  }
494  return true;
495  }
496  return false;
497  }
501  public function __updateResultChilds(int $a_ref_id, array $a_childs): bool
502  {
503  if ($this->results[$a_ref_id] and is_array($a_childs)) {
504  foreach ($a_childs as $child_id) {
505  $this->results[$a_ref_id]['child'][$child_id] = $child_id;
506  }
507  return true;
508  }
509  return false;
510  }
511 
512 
513 
514  public function __initSearchSettingsObject(): void
515  {
516  $this->search_settings = new ilSearchSettings();
517  if (!$this->preventOverwritingMaxhits()) {
518  $this->setMaxHits($this->search_settings->getMaxHits());
519  }
520  }
521 
528  protected function initUserSearchCache(): void
529  {
530  $this->search_cache = ilUserSearchCache::_getInstance($this->getUserId());
531  $this->offset = $this->getMaxHits() * ($this->search_cache->getResultPageNumber() - 1) ;
532  }
533 
541  public function preventOverwritingMaxhits(?bool $a_flag = null)
542  {
543  if (null === $a_flag) {
545  }
546 
547  $this->preventOverwritingMaxhits = $a_flag;
548 
549  return $this;
550  }
551 
560  public function addObserver(object $a_class, string $a_method): bool
561  {
562  $this->observers[] = array('class' => $a_class,
563  'method' => $a_method);
564  return true;
565  }
566 
567 
568  public function callListeners(int $a_ref_id, array $a_data): bool
569  {
570  foreach ($this->observers as $observer) {
571  $class = &$observer['class'];
572  $method = $observer['method'];
573 
574  if (!$class->$method($a_ref_id, $a_data)) {
575  return false;
576  }
577  }
578  return true;
579  }
580 } // END class.Search
setMaxHits(int $a_max_hits)
$res
Definition: ltiservices.php:66
read(int $a_type=ilUserSearchCache::DEFAULT_SEARCH)
read search results
callListeners(int $a_ref_id, array $a_data)
diffEntriesFromResult()
diff entries of this instance and another result object Used for search in results ...
getResultIds()
get result ids
const ROOT_FOLDER_ID
Definition: constants.php:32
static _before(ilDateTime $start, ilDateTime $end, string $a_compare_field='', string $a_tz='')
compare two dates and check start is before end This method does not consider tz offsets.
addObserver(object $a_class, string $a_method)
The observer is used to call functions for filtering result.
static _getInstance(int $a_usr_id)
static _getAllReferences(int $id)
get all reference ids for object ID
initUserSearchCache()
Init user search cache.
ilSearchSettings $search_settings
intersectEntries(ilSearchResult $result_obj)
Build intersection of entries (all entries that are present in both result sets)
getUniqueResults()
Get unique results.
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
addResult(int $a_ref_id, int $a_obj_id, string $a_type)
static _exists(int $id, bool $reference=false, ?string $type=null)
checks if an object exists in object_data
setRequiredPermission(string $a_permission)
Set the required permission for the rbac checks in function &#39;filter()&#39;.
$ref_id
Definition: ltiauth.php:65
static _after(ilDateTime $start, ilDateTime $end, string $a_compare_field='', string $a_tz='')
compare two dates and check start is after end This method does not consider tz offsets.
addEntry(int $a_obj_id, string $a_type, array $found, int $a_child_id=0)
add search result entry Entries are stored with &#39;obj_id&#39;.
__updateResultChilds(int $a_ref_id, array $a_childs)
Update child ids for a specific result.
save(int $a_type=ilUserSearchCache::DEFAULT_SEARCH)
Save search results.
global $DIC
Definition: shib_login.php:22
__construct(int $a_user_id=0)
Constructor public.
isOffsetReached(int $a_counter)
Check if offset is reached.
ilUserSearchCache $search_cache
const IL_CAL_DATE
static _lookupCreationDate(int $obj_id)
mergeEntries(ilSearchResult $result_obj)
merge entries of this instance and another result object
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
setUserId(int $a_user_id)
Class for storing search result.
numEntries()
Check number of entries public.
filter(int $a_root_node, bool $check_and, ?ilDate $creation_filter_date_start=null, ?ilDate $creation_filter_date_end=null)
Filter search result.
__updateEntryChilds(int $a_obj_id, array $a_childs)
Update childs for a specific entry.
filterResults(int $a_root_node)
Filter search area of result set public.
static _lookupType(int $id, bool $reference=false)
preventOverwritingMaxhits(?bool $a_flag=null)
If you call this function and pass "true" the maxhits setting will not be overwritten in __initSearch...
static _within(ilDateTime $dt, ilDateTime $start, ilDateTime $end, string $a_compare_field='', string $a_tz='')
Check whether an date is within a date duration given by start and end.