ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilTreeTrashQueries.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
25 {
26  protected const QUERY_LIMIT = 10;
27 
28  private int $ref_id = 0;
29 
30  private ilTree $tree;
31 
32  private ilLogger $logger;
33 
34  private ilDBInterface $db;
35 
39  public function __construct()
40  {
41  global $DIC;
42 
43  $this->db = $DIC->database();
44  $this->logger = $DIC->logger()->tree();
45  $this->tree = $DIC->repositoryTree();
46  }
47 
52  public function isTrashedTrash(array $ref_ids): bool
53  {
54  $query = 'select tree,child,parent from tree ' .
55  'where ' . $this->db->in('child', $ref_ids, false, \ilDBConstants::T_INTEGER);
56  $res = $this->db->query($query);
57  $trashed_trash = false;
58  while ($row = $res->fetchRow(\ilDBConstants::FETCHMODE_OBJECT)) {
59  if ((int) $row->child != ((int) $row->tree * -1)) {
60  $trashed_trash = true;
61  }
62  if ($this->tree->isDeleted((int) $row->parent)) {
63  $trashed_trash = true;
64  }
65  }
66  return $trashed_trash;
67  }
68 
74  public function findRepositoryLocationForDeletedNode(int $deleted_node): int
75  {
76  $query = 'select parent from tree ' .
77  'where child = ' . $this->db->quote($deleted_node, \ilDBConstants::T_INTEGER) . ' ' .
78  'and tree = ' . $this->db->quote($deleted_node * -1, \ilDBConstants::T_INTEGER);
79  $this->logger->debug($query);
80  $res = $this->db->query($query);
81  while ($row = $res->fetchRow(\ilDBConstants::FETCHMODE_OBJECT)) {
82  return (int) $row->parent;
83  }
84  $this->logger->warning('Invalid node given for restoring to original location: deleted node id: ' . $deleted_node);
85  throw new \ilRepositoryException('Invalid node given for restoring to original location');
86  }
87 
92  public function getTrashedNodeTypesForContainer(int $ref_id): array
93  {
94  $subtreequery = $this->tree->getTrashSubTreeQuery($ref_id, ['child']);
95  $query = 'select distinct(type) obj_type from object_data obd ' .
96  'join object_reference obr on obd.obj_id = obr.obj_id ' .
97  'where ref_id in (' .
98  $subtreequery . ' ' .
99  ') ';
100 
101  $obj_types = [];
102  $res = $this->db->query($query);
103  while ($row = $res->fetchRow(\ilDBConstants::FETCHMODE_OBJECT)) {
104  $obj_types[] = (string) $row->obj_type;
105  }
106  return $obj_types;
107  }
108 
113  public function getNumberOfTrashedNodesForTrashedContainer(int $ref_id): int
114  {
115  $res = $this->db->query($this->tree->getTrashSubTreeQuery($ref_id, ['child']));
116  return $res->numRows();
117  }
118 
124  public function getTrashNodeForContainer(
125  int $ref_id,
126  array $filter,
127  int &$max_entries,
128  string $order_field,
129  string $order_direction,
130  int $limit = self::QUERY_LIMIT,
131  int $offset = 0
132  ): array {
133  $subtreequery = $this->tree->getTrashSubTreeQuery($ref_id, ['child']);
134 
135  $select = 'SELECT ref_id, obd.obj_id, type, title, description, deleted, deleted_by ';
136  $select_count = 'SELECT COUNT(ref_id) max_entries ';
137 
138  $from = 'FROM object_data obd ' .
139  'JOIN object_reference obr ON obd.obj_id = obr.obj_id ' .
140  'WHERE ref_id IN (' .
141  $subtreequery . ' ' .
142  ') ';
143 
144  $order = ' ';
145  if ($order_field) {
146  $order = 'ORDER BY ' . $order_field . ' ' . $order_direction;
147  }
148 
149  $query = $select . $from . $this->appendTrashNodeForContainerQueryFilter($filter) . $order;
150  $query_count = $select_count . $from . $this->appendTrashNodeForContainerQueryFilter($filter);
151 
155  $res_max_entries = $this->db->query($query_count);
156  while ($max_entries_row = $res_max_entries->fetchRow(\ilDBConstants::FETCHMODE_OBJECT)) {
157  $max_entries = (int) $max_entries_row->max_entries;
158  }
159 
160  $this->db->setLimit($limit, $offset);
161  $res = $this->db->query($query);
162 
163  $items = [];
164  while ($row = $res->fetchRow(\ilDBConstants::FETCHMODE_OBJECT)) {
165  $item = new \ilTreeTrashItem();
166  $item->setObjId((int) $row->obj_id);
167  $item->setRefId((int) $row->ref_id);
168  $item->setTitle((string) $row->title);
169  $item->setDescription((string) $row->description);
170  $item->setType((string) $row->type);
171  $item->setDeleted((string) $row->deleted);
172  $item->setDeletedBy((int) $row->deleted_by);
173  $items[] = $item;
174  }
175  return $items;
176  }
177 
185  public function getTrashedNodesForContainerUsingRecursion(int $ref_id): void
186  {
187  $query = 'WITH RECURSIVE trash (child,tree) AS ' .
188  '( SELECT child, tree FROM tree WHERE child = ' . $this->db->quote(
189  $ref_id,
191  ) . ' ' .
192  'UNION SELECT tc.child,tc.tree FROM tree tc JOIN tree tp ON tp.child = tc.parent ) ' .
193  'SELECT * FROM trash WHERE tree < 1 ';
194 
195  $trash_ids = [];
196  try {
197  $res = $this->db->query($query);
198  while ($row = $res->fetchRow(\ilDBConstants::FETCHMODE_OBJECT)) {
199  $trash_ids[] = (int) $row->child;
200  }
201  } catch (\ilDatabaseException $e) {
202  $this->logger->warning($query . ' is not supported');
203  throw $e;
204  }
205  }
206 
211  protected function appendTrashNodeForContainerQueryFilter(array $filter): string
212  {
213  $query = '';
214  if (isset($filter['type'])) {
215  $query .= 'and ' . $this->db->like(
216  'type',
218  $filter['type']
219  ) . ' ';
220  }
221  if (isset($filter['title'])) {
222  $query .= 'and ' . $this->db->like(
223  'title',
225  '%' . $filter['title'] . '%'
226  ) . ' ';
227  }
228 
229  // #33781 use an explizit date format to really include the 'to' date because it can be interpreted as 'xxxx-xx-xx 00:00:00'
230  if (
231  $filter['deleted']['from'] instanceof \ilDate &&
232  $filter['deleted']['to'] instanceof \ilDate) {
233  $query .= ('AND deleted BETWEEN ' .
234  $this->db->quote($filter['deleted']['from']->get(\IL_CAL_DATE), \ilDBConstants::T_TEXT) . ' AND ' .
235  $this->db->quote($filter['deleted']['to']->get(IL_CAL_DATE) . " 23:59:59", \ilDBConstants::T_TEXT) . ' ');
236  } elseif ($filter['deleted']['from'] instanceof \ilDate) {
237  $query .= 'AND deleted >= ' . $this->db->quote(
238  $filter['deleted']['from']->get(IL_CAL_DATE),
240  ) . ' ';
241  } elseif ($filter['deleted']['to'] instanceof \ilDate) {
242  $query .= 'AND deleted <= ' . $this->db->quote(
243  $filter['deleted']['to']->get(IL_CAL_DATE) . " 23:59:59",
245  ) . ' ';
246  }
247 
248  if (isset($filter['deleted_by']) && !empty($filter['deleted_by'])) {
249  $usr_id = \ilObjUser::_lookupId($filter['deleted_by']);
250  if ($usr_id > 0) {
251  $query .= 'AND deleted_by = ' . $this->db->quote($usr_id, \ilDBConstants::T_INTEGER) . ' ';
252  } else {
253  $query .= 'AND 1=2 ';
254  }
255  }
256 
257  return $query;
258  }
259 }
$res
Definition: ltiservices.php:66
getTrashNodeForContainer(int $ref_id, array $filter, int &$max_entries, string $order_field, string $order_direction, int $limit=self::QUERY_LIMIT, int $offset=0)
Get trashed nodes.
appendTrashNodeForContainerQueryFilter(array $filter)
getTrashedNodeTypesForContainer(int $ref_id)
getTrashedNodesForContainerUsingRecursion(int $ref_id)
Unfortunately not supported by mysql 5.
static _lookupId($a_user_str)
__construct()
ilTreeTrash constructor.
global $DIC
Definition: shib_login.php:22
findRepositoryLocationForDeletedNode(int $deleted_node)
getNumberOfTrashedNodesForTrashedContainer(int $ref_id)
const IL_CAL_DATE