ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
MailBoxQuery.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
21 namespace ILIAS\Mail\Message;
22 
23 use ilDBInterface;
24 use ilDBConstants;
26 use DateTimeZone;
29 
34 {
35  private const DEFAULT_ORDER_COLUMN = MailBoxOrderColumn::SEND_TIME;
37 
38  private ilDBInterface $db;
39  private ?int $folder_id = null;
40  private ?string $sender = null;
41  private ?string $recipients = null;
42  private ?string $subject = null;
43  private ?string $body = null;
44  private ?bool $is_unread = null;
45  private ?bool $is_system = null;
46  private ?bool $has_attachment = null;
50  private ?array $filtered_ids = null;
51  private int $limit = 999999;
52  private int $offset = 0;
53  private MailBoxOrderColumn $order_column = self::DEFAULT_ORDER_COLUMN;
54  private string $order_direction = self::DEFAULT_ORDER_DIRECTION;
55 
56  public function __construct(
57  private readonly int $user_id,
58  ) {
59  global $DIC;
60  $this->db = $DIC->database();
61  }
62 
63  public function withFolderId(?int $folder_id): MailBoxQuery
64  {
65  $clone = clone $this;
66  $clone->folder_id = $folder_id;
67 
68  return $clone;
69  }
70 
71  public function withSender(?string $sender): MailBoxQuery
72  {
73  $clone = clone $this;
74  $clone->sender = $sender;
75 
76  return $clone;
77  }
78 
79  public function withRecipients(?string $recipients): MailBoxQuery
80  {
81  $clone = clone $this;
82  $clone->recipients = $recipients;
83 
84  return $clone;
85  }
86 
87  public function withSubject(?string $subject): MailBoxQuery
88  {
89  $clone = clone $this;
90  $clone->subject = $subject;
91 
92  return $clone;
93  }
94 
95  public function withBody(?string $body): MailBoxQuery
96  {
97  $clone = clone $this;
98  $clone->body = $body;
99 
100  return $clone;
101  }
102 
103  public function withIsUnread(?bool $is_unread): MailBoxQuery
104  {
105  $clone = clone $this;
106  $clone->is_unread = $is_unread;
107 
108  return $clone;
109  }
110 
111  public function withIsSystem(?bool $is_system): MailBoxQuery
112  {
113  $clone = clone $this;
114  $clone->is_system = $is_system;
115 
116  return $clone;
117  }
118 
119  public function withHasAttachment(?bool $has_attachment): MailBoxQuery
120  {
121  $clone = clone $this;
122  $clone->has_attachment = $has_attachment;
123 
124  return $clone;
125  }
126 
127  public function withPeriodStart(?DateTimeImmutable $period_start): MailBoxQuery
128  {
129  $clone = clone $this;
130  $clone->period_start = $period_start;
131 
132  return $clone;
133  }
134 
135  public function withPeriodEnd(?DateTimeImmutable $period_end): MailBoxQuery
136  {
137  $clone = clone $this;
138  $clone->period_end = $period_end;
139 
140  return $clone;
141  }
142 
146  public function withFilteredIds(?array $filtered_ids): MailBoxQuery
147  {
148  $clone = clone $this;
149  $clone->filtered_ids = $filtered_ids;
150 
151  return $clone;
152  }
153 
154  public function withLimit(int $limit): MailBoxQuery
155  {
156  $clone = clone $this;
157  $clone->limit = $limit;
158 
159  return $clone;
160  }
161 
162  public function withOffset(int $offset): MailBoxQuery
163  {
164  $clone = clone $this;
165  $clone->offset = $offset;
166 
167  return $clone;
168  }
169 
170  public function withOrderColumn(?MailBoxOrderColumn $order_column): MailBoxQuery
171  {
172  $clone = clone $this;
173  if ($order_column !== null) {
174  $clone->order_column = $order_column;
175  } else {
176  $clone->order_column = self::DEFAULT_ORDER_COLUMN;
177  }
178 
179  return $clone;
180  }
181 
182  public function withOrderDirection(?string $order_direction): MailBoxQuery
183  {
184  $clone = clone $this;
185  if (in_array($order_direction, [Order::ASC, Order::DESC])) {
186  $clone->order_direction = $order_direction;
187  } else {
188  $clone->order_direction = self::DEFAULT_ORDER_DIRECTION;
189  }
190 
191  return $clone;
192  }
193 
197  public function countUnread(): int
198  {
199  return $this->withIsUnread(true)->count();
200  }
201 
205  public function count(): int
206  {
207  if ($this->filtered_ids === []) {
208  return 0;
209  }
210 
211  $query = 'SELECT COUNT(m.mail_id) cnt '
212  . $this->getFrom()
213  . $this->getWhere();
214 
215  $res = $this->db->query($query);
216  if ($row = $this->db->fetchAssoc($res)) {
217  return (int) $row['cnt'];
218  }
219 
220  return 0;
221  }
222 
227  public function queryMailIds(): array
228  {
229  if ($this->filtered_ids === []) {
230  return [];
231  }
232 
233  $query = 'SELECT m.mail_id '
234  . $this->getFrom()
235  . $this->getWhere();
236 
237  $ids = [];
238  $res = $this->db->query($query);
239  while ($row = $this->db->fetchAssoc($res)) {
240  $ids[] = (int) $row['mail_id'];
241  }
242 
243  return $ids;
244  }
245 
251  public function query($short): array
252  {
253  if ($this->filtered_ids === []) {
254  return [];
255  }
256 
257  if ($short) {
258  $query = 'SELECT m.mail_id, m.user_id, m.folder_id, m.sender_id, m.send_time, '
259  . 'm.m_status, m.m_subject, m.import_name, m.rcp_to, m.attachments'
260  . $this->getFrom()
261  . $this->getWhere();
262  } else {
263  $query = 'SELECT m.*'
264  . $this->getFrom()
265  . $this->getWhere();
266  }
267 
268  if ($this->order_column === MailBoxOrderColumn::FROM) {
269  $query .= ' ORDER BY '
270  . ' u.firstname ' . $this->order_direction . ', '
271  . ' u.lastname ' . $this->order_direction . ', '
272  . ' u.login ' . $this->order_direction . ', '
273  . ' m.import_name ' . $this->order_direction;
274  } else {
275  $query .= ' ORDER BY ' . $this->order_column->value . ' ' . $this->order_direction;
276  }
277 
278  $this->db->setLimit($this->limit, $this->offset);
279  $res = $this->db->query($query);
280 
281  $set = [];
282  while ($row = $this->db->fetchAssoc($res)) {
283  $set[] = new MailRecordData(
284  isset($row['mail_id']) ? (int) $row['mail_id'] : 0,
285  isset($row['user_id']) ? (int) $row['user_id'] : 0,
286  isset($row['folder_id']) ? (int) $row['folder_id'] : 0,
287  isset($row['sender_id']) ? (int) $row['sender_id'] : null,
288  isset($row['send_time']) ? new DateTimeImmutable($row['send_time']) : null,
289  isset($row['m_status']) ? (string) $row['m_status'] : null,
290  isset($row['m_subject']) ? (string) $row['m_subject'] : null,
291  isset($row['import_name']) ? (string) $row['import_name'] : null,
292  isset($row['use_placeholders']) ? (bool) $row['use_placeholders'] : false,
293  isset($row['m_message']) ? (string) $row['m_message'] : null,
294  isset($row['rcp_to']) ? (string) $row['rcp_to'] : null,
295  isset($row['rcp_cc']) ? (string) $row['rcp_cc'] : null,
296  isset($row['rcp_bcc']) ? (string) $row['rcp_bcc'] : null,
297  isset($row['attachments']) ? (array) unserialize(
298  stripslashes($row['attachments']),
299  ['allowed_classes' => false]
300  ) : [],
301  isset($row['tpl_ctx_id']) ? (string) $row['tpl_ctx_id'] : null,
302  isset($row['tpl_ctx_params']) ? (string) $row['tpl_ctx_params'] : null
303  );
304  }
305 
306  return $set;
307  }
308 
309  private function getFrom()
310  {
311  return " FROM mail m
312  LEFT JOIN usr_data u ON u.usr_id = m.sender_id
313  LEFT JOIN usr_pref p ON p.usr_id = m.sender_id AND p.keyword = 'public_profile'";
314  }
315 
316  private function getWhere(): string
317  {
318  $parts = [];
319 
320  // minimum condition: only mailbox of the given user
321  $parts[] = 'm.user_id = ' . $this->db->quote($this->user_id, ilDBConstants::T_INTEGER);
322 
323  // sender conditions have to respect searchability and visibility of profile fields
324  $sender_conditions = [];
325  if (($this->sender ?? '') !== '') {
326  $sender_conditions[] = $this->db->like('u.login', ilDBConstants::T_TEXT, '%%' . $this->sender . '%%');
327 
328  if (ilUserSearchOptions::_isEnabled('firstname')) {
329  $sender_conditions[] = '(' .
330  $this->db->like('u.firstname', ilDBConstants::T_TEXT, '%%' . $this->sender . '%%')
331  . " AND p.value = 'y')";
332  }
333 
334  if (ilUserSearchOptions::_isEnabled('lastname')) {
335  $sender_conditions[] = '(' .
336  $this->db->like('u.lastname', ilDBConstants::T_TEXT, '%%' . $this->sender . '%%')
337  . " AND p.value = 'y')";
338  }
339  }
340  if (!empty($sender_conditions)) {
341  $parts[] = '(' . implode(' OR ', $sender_conditions) . ')';
342  }
343 
344  // other text conditions
345  $text_conditions = [
346  [$this->recipients, 'CONCAT(CONCAT(m.rcp_to, m.rcp_cc), m.rcp_bcc)'],
347  [$this->subject, 'm.m_subject'],
348  [$this->body, 'm.m_message'],
349  ];
350 
351  foreach ($text_conditions as $cond) {
352  if (($cond[0] ?? '') !== '') {
353  $parts[] = $this->db->like(
354  $cond[1],
356  '%%' . $cond[0] . '%%',
357  false
358  );
359  }
360  }
361 
362  if ($this->folder_id !== null) {
363  $parts[] = 'm.folder_id = ' . $this->db->quote($this->folder_id, ilDBConstants::T_INTEGER);
364  }
365 
366  if ($this->is_unread === true) {
367  $parts[] = 'm.m_status = ' . $this->db->quote('unread', 'text');
368  } elseif ($this->is_unread === false) {
369  $parts[] = 'm.m_status != ' . $this->db->quote('unread', 'text');
370  }
371 
372  if ($this->is_system === true) {
373  $parts[] = 'm.sender_id = ' . $this->db->quote(ANONYMOUS_USER_ID, ilDBConstants::T_INTEGER);
374  } elseif ($this->is_system === false) {
375  $parts[] = 'm.sender_id != ' . $this->db->quote(ANONYMOUS_USER_ID, ilDBConstants::T_INTEGER);
376  }
377 
378  if ($this->has_attachment === true) {
379  $parts[] = '(m.attachments != ' . $this->db->quote(serialize(null), ilDBConstants::T_TEXT)
380  . ' AND m.attachments != ' . $this->db->quote(serialize([]), ilDBConstants::T_TEXT) . ')';
381  } elseif ($this->has_attachment === false) {
382  $parts[] = '(m.attachments = ' . $this->db->quote(serialize(null), ilDBConstants::T_TEXT)
383  . ' OR m.attachments = ' . $this->db->quote(serialize([]), ilDBConstants::T_TEXT) . ')';
384  }
385 
386  if ($this->period_start !== null) {
387  $parts[] = 'm.send_time >= ' . $this->db->quote(
388  // convert to server time zone (set by ilias initialisation)
389  $this->period_start
390  ->setTimezone(new DateTimeZone(date_default_timezone_get()))
391  ->format('Y-m-d H:i:s'),
393  );
394  }
395 
396  if ($this->period_end !== null) {
397  $parts[] = 'm.send_time <= ' . $this->db->quote(
398  // convert to server time zone (set by ilias initialisation)
399  $this->period_end
400  ->setTimezone(new DateTimeZone(date_default_timezone_get()))
401  ->format('Y-m-d H:i:s'),
403  );
404  }
405 
406  if (!empty($this->filtered_ids)) {
407  $parts[] = $this->db->in(
408  'm.mail_id',
409  $this->filtered_ids,
410  false,
412  ) . ' ';
413  }
414 
415  if ($parts !== []) {
416  return ' WHERE ' . implode(' AND ', $parts);
417  }
418 
419  return '';
420  }
421 }
$res
Definition: ltiservices.php:66
Database query for mails of a user.
const ANONYMOUS_USER_ID
Definition: constants.php:27
withHasAttachment(?bool $has_attachment)
if($clientAssertionType !='urn:ietf:params:oauth:client-assertion-type:jwt-bearer'|| $grantType !='client_credentials') $parts
Definition: ltitoken.php:61
withFilteredIds(?array $filtered_ids)
MailBoxOrderColumn $order_column
countUnread()
Count the number of unread mails with applied filter.
queryMailIds()
Get a list of mail ids.
DateTimeImmutable $period_start
withOrderColumn(?MailBoxOrderColumn $order_column)
__construct(private readonly int $user_id,)
withOrderDirection(?string $order_direction)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
global $DIC
Definition: shib_login.php:22
withPeriodEnd(?DateTimeImmutable $period_end)
count()
Count the number of all mails with applied filter.
withRecipients(?string $recipients)
withPeriodStart(?DateTimeImmutable $period_start)
const DESC
Definition: Order.php:31
query($short)
Query for mail data with applied filter.