ILIAS  trunk Revision v12.0_alpha-377-g3641b37b9db
class.ilBookingReservation.php
Go to the documentation of this file.
1<?php
2
24{
25 public const STATUS_IN_USE = 2;
26 public const STATUS_CANCELLED = 5;
27
28 protected ilDBInterface $db;
29 protected int $id = 0;
30 protected int $object_id = 0;
31 protected int $user_id = 0;
32 protected int $from = 0;
33 protected int $to = 0;
34 protected int $status = 0;
35 protected int $group_id = 0;
36 protected int $assigner_id = 0;
37 protected int $context_obj_id = 0;
38 protected string $message = "";
39 protected \ILIAS\BookingManager\Reservations\ReservationDBRepository $repo;
40
41 public function __construct(
42 ?int $a_id = null
43 ) {
44 global $DIC;
45
46 $this->db = $DIC->database();
47 $this->id = (int) $a_id;
48
49 $this->repo = $DIC->bookingManager()
50 ->internal()
51 ->repo()
52 ->reservation();
53 $this->read();
54 }
55
56 public function getId(): int
57 {
58 return $this->id;
59 }
60
61 public function setObjectId(int $a_object_id): void
62 {
63 $this->object_id = $a_object_id;
64 }
65
66 public function getObjectId(): int
67 {
68 return $this->object_id;
69 }
70
71 public function setUserId(int $a_user_id): void
72 {
73 $this->user_id = $a_user_id;
74 }
75
76 public function getUserId(): int
77 {
78 return $this->user_id;
79 }
80
81 public function setMessage(string $message): void
82 {
83 $this->message = $message;
84 }
85
86 public function getMessage(): string
87 {
88 return $this->message;
89 }
90
94 public function setAssignerId(int $a_assigner_id): void
95 {
96 $this->assigner_id = $a_assigner_id;
97 }
98
99 public function getAssignerId(): int
100 {
101 return $this->assigner_id;
102 }
103
107 public function setFrom(int $a_from): void
108 {
109 $this->from = $a_from;
110 }
111
112 public function getFrom(): int
113 {
114 return $this->from;
115 }
116
120 public function setTo(int $a_to): void
121 {
122 $this->to = $a_to;
123 }
124
125 public function getTo(): int
126 {
127 return $this->to;
128 }
129
133 public function setStatus(?int $a_status): void
134 {
135 if ($a_status === null) {
136 $this->status = 0;
137 }
138 if (self::isValidStatus((int) $a_status)) {
139 $this->status = (int) $a_status;
140 }
141 }
142
146 public function getStatus(): ?int
147 {
148 return $this->status;
149 }
150
151 public static function isValidStatus(int $a_status): bool
152 {
153 return in_array($a_status, array(self::STATUS_IN_USE, self::STATUS_CANCELLED));
154 }
155
156 public function setGroupId(int $a_group_id): void
157 {
158 $this->group_id = $a_group_id;
159 }
160
161 public function getGroupId(): int
162 {
163 return $this->group_id;
164 }
165
169 public function setContextObjId(int $a_val): void
170 {
171 $this->context_obj_id = $a_val;
172 }
173
177 public function getContextObjId(): int
178 {
180 }
181
182 protected function read(): void
183 {
184 if ($this->id && $row = $this->repo->getForId($this->id)) {
185 $this->setUserId($row['user_id']);
186 $this->setAssignerId((int) $row['assigner_id']);
187 $this->setObjectId($row['object_id']);
188 $this->setFrom($row['date_from']);
189 $this->setTo($row['date_to']);
190 $this->setStatus($row['status']);
191 $this->setGroupId((int) $row['group_id']);
192 $this->setContextObjId((int) $row['context_obj_id']);
193 $this->setMessage($row['message']);
194 }
195 }
196
197 public function save(): bool
198 {
199 if ($this->id) {
200 return false;
201 }
202
203 $this->id = $this->repo->create(
204 $this->getUserId(),
205 $this->getAssignerId(),
206 $this->getObjectId(),
207 $this->getContextObjId(),
208 $this->getFrom(),
209 $this->getTo(),
210 $this->getStatus(),
211 $this->getGroupId(),
212 $this->getMessage()
213 );
214 return ($this->id > 0);
215 }
216
217 public function update(): bool
218 {
219 if (!$this->id) {
220 return false;
221 }
222
223 $this->repo->update(
224 $this->id,
225 $this->getUserId(),
226 $this->getAssignerId(),
227 $this->getObjectId(),
228 $this->getContextObjId(),
229 $this->getFrom(),
230 $this->getTo(),
231 $this->getStatus(),
232 $this->getGroupId(),
233 $this->getMessage()
234 );
235 return true;
236 }
237
238 public function delete(): void
239 {
240 $this->repo->delete($this->id);
241 }
242
243
247 public static function getAvailableObject(
248 array $a_ids,
249 int $a_from,
250 int $a_to,
251 bool $a_return_single = true,
252 bool $a_return_counter = false
253 ): array {
254 global $DIC;
255
257
258 $repo = $DIC->bookingManager()->internal()->repo()->reservation();
259
260 $blocked = $counter = array();
261 foreach ($repo->getNumberOfReservations($a_ids, $a_from, $a_to) as $row) {
262 if ($row['cnt'] >= $nr_map[$row['object_id']]) {
263 $blocked[] = $row['object_id'];
264 } elseif ($a_return_counter) {
265 $counter[$row['object_id']] = $nr_map[$row['object_id']] - (int) $row['cnt'];
266 }
267 }
268
269 // #17868 - validate against schedule availability
270 foreach ($a_ids as $obj_id) {
271 $bobj = new ilBookingObject($obj_id);
272 if ($bobj->getScheduleId()) {
273 $schedule = new ilBookingSchedule($bobj->getScheduleId());
274
275 $av_from = ($schedule->getAvailabilityFrom() && !$schedule->getAvailabilityFrom()->isNull())
276 ? $schedule->getAvailabilityFrom()->get(IL_CAL_UNIX)
277 : null;
278 $av_to = ($schedule->getAvailabilityTo() && !$schedule->getAvailabilityTo()->isNull())
279 ? $schedule->getAvailabilityTo()->get(IL_CAL_UNIX)
280 : null;
281
282 if (($av_from && $a_from < $av_from) ||
283 ($av_to && $a_to > $av_to)) {
284 $blocked[] = $obj_id;
285 unset($counter[$obj_id]);
286 }
287 }
288 }
289
290 $available = array_diff($a_ids, $blocked);
291 if (count($available)) {
292 if ($a_return_counter) {
293 foreach ($a_ids as $id) {
294 if (!isset($counter[$id])) {
295 $counter[$id] = $nr_map[$id];
296 }
297 }
298 return $counter;
299 }
300 if ($a_return_single) {
301 return array_shift($available);
302 }
303 return $available;
304 }
305 return [];
306 }
307
308 public static function isObjectAvailableInPeriod(
309 int $a_obj_id,
310 ilBookingSchedule $a_schedule,
311 ?int $a_from,
312 ?int $a_to
313 ): bool {
314 global $DIC;
315
316 if (!$a_from) {
317 $a_from = time();
318 }
319 if (!$a_to) {
320 $a_to = strtotime("+1year", $a_from);
321 }
322
323 if ($a_from > $a_to) {
324 return false;
325 }
326
327 // all nr of reservations in period that are not over yet (to >= now)
328 $repo = $DIC->bookingManager()->internal()->repo()->reservation();
329 $res = $repo->getNumberOfReservations([$a_obj_id], $a_from, $a_to, true);
330 $booked_in_period = (int) ($res[$a_obj_id]["cnt"] ?? 0);
331
332 $per_slot = ilBookingObject::getNrOfItemsForObjects(array($a_obj_id));
333 $per_slot = $per_slot[$a_obj_id];
334
335 // max available nr of items per (week)day
336 $schedule_slots = array();
337 $definition = $a_schedule->getDefinition();
338 $map = array_flip(array("su", "mo", "tu", "we", "th", "fr", "sa"));
339 foreach ($definition as $day => $day_slots) {
340 $schedule_slots[$map[$day]] = $day_slots;
341 }
342
343 $av_from = ($a_schedule->getAvailabilityFrom() && !$a_schedule->getAvailabilityFrom()->isNull())
344 ? $a_schedule->getAvailabilityFrom()->get(IL_CAL_UNIX)
345 : null;
346 $av_to = ($a_schedule->getAvailabilityTo() && !$a_schedule->getAvailabilityTo()->isNull())
347 ? strtotime($a_schedule->getAvailabilityTo()->get(IL_CAL_DATE) . " 23:59:59")
348 : null;
349
350 // sum up max available (to >= now) items in period per (week)day
351 $available_in_period = 0;
352 $loop = 0;
353 while ($a_from < $a_to &&
354 ++$loop < 1000) {
355 // any slots for current weekday?
356 $day_slots = $schedule_slots[date("w", $a_from)] ?? false;
357 if ($day_slots) {
358 foreach ($day_slots as $slot) {
359 // convert slot to current datetime
360 $slot = explode("-", $slot);
361 $slot_from = strtotime(date("Y-m-d", $a_from) . " " . $slot[0]);
362 $slot_to = strtotime(date("Y-m-d", $a_from) . " " . $slot[1]);
363 // slot has to be in the future and part of schedule availability
364 if ($slot_from >= $av_from &&
365 ($slot_to <= $av_to || is_null($av_to)) &&
366 $slot_to > time()) {
367 $available_in_period += $per_slot;
368 }
369 }
370 }
371
372 $a_from += (60 * 60 * 24);
373 }
374 return $available_in_period - $booked_in_period > 0;
375 }
376
377 //check if the user reached the limit of bookings in this booking pool.
378 public static function isBookingPoolLimitReachedByUser(
379 int $a_user_id,
380 int $a_pool_id
381 ): int {
382 global $DIC;
383 $ilDB = $DIC->database();
384 $object_manager = $DIC->bookingManager()->internal()->domain()
385 ->objects($a_pool_id);
386
387 $booking_pool_objects = $object_manager->getObjectIds();
388
389 $query = "SELECT count(user_id) total" .
390 " FROM booking_reservation" .
391 " WHERE " . $ilDB->in('object_id', $booking_pool_objects, false, 'integer') .
392 " AND user_id = " . $a_user_id .
393 " AND (status IS NULL OR status <> " . self::STATUS_CANCELLED . ')';
394 $res = $ilDB->query($query);
395 $row = $res->fetchRow(ilDBConstants::FETCHMODE_ASSOC);
396
397 return (int) $row['total'];
398 }
399
403 public static function getMembersWithoutReservation(
404 int $a_object_id
405 ): array {
406 global $DIC;
407 $ilDB = $DIC->database();
408
409 $pool_id = ilBookingObject::lookupPoolId($a_object_id);
410
411 $res = array();
412 $query = 'SELECT DISTINCT bm.user_id user_id' .
413 ' FROM booking_member bm' .
414 ' WHERE bm.booking_pool_id = ' . $ilDB->quote($pool_id, 'integer') .
415 ' AND bm.user_id NOT IN (' .
416 'SELECT user_id' .
417 ' FROM booking_reservation' .
418 ' WHERE object_id = ' . $ilDB->quote($a_object_id, 'integer') .
419 ' AND (status IS NULL OR status <> ' . self::STATUS_CANCELLED . '))';
420
421 $set = $ilDB->query($query);
422
423 while ($row = $ilDB->fetchAssoc($set)) {
424 $res[] = (int) $row['user_id'];
425 }
426
427 return $res;
428 }
429
430 public static function isObjectAvailableNoSchedule(int $a_obj_id): bool
431 {
432 $available = self::getNumAvailablesNoSchedule($a_obj_id);
433 return (bool) $available; // #11864
434 }
435
436 public static function numAvailableFromObjectNoSchedule(int $a_obj_id): int
437 {
438 return self::getNumAvailablesNoSchedule($a_obj_id);
439 }
440
441 public static function getNumAvailablesNoSchedule(int $a_obj_id): int
442 {
443 global $DIC;
444
445 $ilDB = $DIC->database();
446
447 $all = ilBookingObject::getNrOfItemsForObjects(array($a_obj_id));
448 $all = $all[$a_obj_id];
449
450 $set = $ilDB->query('SELECT COUNT(*) cnt' .
451 ' FROM booking_reservation r' .
452 ' JOIN booking_object o ON (o.booking_object_id = r.object_id)' .
453 ' WHERE (status IS NULL OR status <> ' . $ilDB->quote(self::STATUS_CANCELLED, 'integer') . ')' .
454 ' AND r.object_id = ' . $ilDB->quote($a_obj_id, 'integer'));
455 $cnt = $ilDB->fetchAssoc($set);
456 $cnt = (int) $cnt['cnt'];
457
458 return $all - $cnt; // #11864
459 }
460
464 public static function getCurrentOrUpcomingReservation(
465 int $a_object_id
466 ): array {
467 global $DIC;
468
469 $ilDB = $DIC->database();
470
471 $now = $ilDB->quote(time(), 'integer');
472
473 $ilDB->setLimit(1);
474 $set = $ilDB->query('SELECT user_id, status, date_from, date_to' .
475 ' FROM booking_reservation' .
476 ' WHERE ((date_from <= ' . $now . ' AND date_to >= ' . $now . ')' .
477 ' OR date_from > ' . $now . ')' .
478 ' AND (status <> ' . $ilDB->quote(self::STATUS_CANCELLED, 'integer') .
479 ' OR STATUS IS NULL) AND object_id = ' . $ilDB->quote($a_object_id, 'integer') .
480 ' ORDER BY date_from');
481 return $ilDB->fetchAssoc($set);
482 }
483
487 public static function getObjectReservationForUser(
488 int $a_object_id,
489 int $a_user_id
490 ): ?array {
491 global $DIC;
492
493 $ilDB = $DIC->database();
494
495 $set = $ilDB->query('SELECT booking_reservation_id FROM booking_reservation' .
496 ' WHERE user_id = ' . $ilDB->quote($a_user_id, 'integer') .
497 ' AND object_id = ' . $ilDB->quote($a_object_id, 'integer') .
498 ' AND (status <> ' . $ilDB->quote(self::STATUS_CANCELLED, 'integer') .
499 ' OR STATUS IS NULL)');
500 $res = array();
501 while ($row = $ilDB->fetchAssoc($set)) {
502 $res[] = (int) $row['booking_reservation_id'];
503 }
504 return $res;
505 }
506
511 public static function getList(
512 array $a_object_ids,
513 int $a_limit = 10,
514 int $a_offset = 0,
515 array $filter = []
516 ): array {
517 global $DIC;
518
519 $ilDB = $DIC->database();
520
521 $sql = 'SELECT r.*,o.title' .
522 ' FROM booking_reservation r' .
523 ' JOIN booking_object o ON (o.booking_object_id = r.object_id)';
524
525 $count_sql = 'SELECT COUNT(*) AS counter' .
526 ' FROM booking_reservation r' .
527 ' JOIN booking_object o ON (o.booking_object_id = r.object_id)';
528
529 $where = array($ilDB->in('r.object_id', $a_object_ids, '', 'integer'));
530 if (isset($filter['status'])) {
531 if ($filter['status'] > 0) {
532 $where[] = 'status = ' . $ilDB->quote($filter['status'], 'integer');
533 } else {
534 $where[] = '(status != ' . $ilDB->quote(-$filter['status'], 'integer') .
535 ' OR status IS NULL)';
536 }
537 }
538 if (isset($filter['from'])) {
539 $where[] = 'date_from >= ' . $ilDB->quote($filter['from'], 'integer');
540 }
541 if (isset($filter['to'])) {
542 $where[] = 'date_to <= ' . $ilDB->quote($filter['to'], 'integer');
543 }
544 if (count($where)) {
545 $sql .= ' WHERE ' . implode(' AND ', $where);
546 $count_sql .= ' WHERE ' . implode(' AND ', $where);
547 }
548
549 $set = $ilDB->query($count_sql);
550 $row = $ilDB->fetchAssoc($set);
551 $counter = $row['counter'];
552
553 $sql .= ' ORDER BY date_from DESC, booking_reservation_id DESC';
554
555 $ilDB->setLimit($a_limit, $a_offset);
556 $set = $ilDB->query($sql);
557 $res = array();
558 while ($row = $ilDB->fetchAssoc($set)) {
559 $res[] = $row;
560 }
561
562 return array('data' => $res, 'counter' => $counter);
563 }
564
565
572 public static function getUserFilter(
573 array $a_object_ids
574 ): array {
575 global $DIC;
576
577 $ilDB = $DIC->database();
578
579 $res = array();
580
581 $sql = "SELECT ud.usr_id,ud.lastname,ud.firstname,ud.login" .
582 " FROM usr_data ud " .
583 " LEFT JOIN booking_reservation r ON (r.user_id = ud.usr_id)" .
584 " WHERE ud.usr_id <> " . $ilDB->quote(ANONYMOUS_USER_ID, "integer") .
585 " AND " . $ilDB->in("r.object_id", $a_object_ids, "", "integer") .
586 " ORDER BY ud.lastname,ud.firstname";
587 $set = $ilDB->query($sql);
588 while ($row = $ilDB->fetchAssoc($set)) {
589 $res[$row["usr_id"]] = $row["lastname"] . ", " . $row["firstname"] .
590 " (" . $row["login"] . ")";
591 }
592
593 return $res;
594 }
595
596
603 public static function changeStatus(
604 array $a_ids,
605 int $a_status
606 ): int {
607 global $DIC;
608
609 $ilDB = $DIC->database();
610
611 if (self::isValidStatus($a_status)) {
612 return $ilDB->manipulate('UPDATE booking_reservation' .
613 ' SET status = ' . $ilDB->quote($a_status, 'integer') .
614 ' WHERE ' . $ilDB->in('booking_reservation_id', $a_ids, '', 'integer'));
615 }
616 return 0;
617 }
618
619 // get calendar id of reservation
620 public function getCalendarEntry(): int
621 {
622 $ilDB = $this->db;
623
624 $set = $ilDB->query("SELECT ce.cal_id FROM cal_entries ce" .
625 " JOIN cal_cat_assignments cca ON ce.cal_id = cca.cal_id" .
626 " JOIN cal_categories cc ON cca.cat_id = cc.cat_id" .
627 " JOIN booking_reservation br ON ce.context_id = br.booking_reservation_id" .
628 " WHERE cc.obj_id = " . $ilDB->quote($this->getUserId(), 'integer') .
629 " AND br.user_id = " . $ilDB->quote($this->getUserId(), 'integer') .
630 " AND cc.type = " . $ilDB->quote(ilCalendarCategory::TYPE_BOOK, 'integer') .
631 " AND ce.context_id = " . $ilDB->quote($this->getId(), 'integer'));
632 $row = $ilDB->fetchAssoc($set);
633 return (int) ($row["cal_id"] ?? 0);
634 }
635
640 public static function getCancelDetails(
641 int $a_obj_id,
642 int $a_user_id,
643 int $a_from,
644 int $a_to
645 ): array {
646 global $DIC;
647
648 $ilDB = $DIC->database();
649
650 $res = array();
651
652 $sql = "SELECT booking_reservation_id" .
653 " FROM booking_reservation" .
654 " WHERE object_id = " . $ilDB->quote($a_obj_id, "integer") .
655 " AND user_id = " . $ilDB->quote($a_user_id, "integer") .
656 " AND date_from = " . $ilDB->quote($a_from, "integer") .
657 " AND date_to = " . $ilDB->quote($a_to, "integer") .
658 " AND (status IS NULL" .
659 " OR status <> " . $ilDB->quote(self::STATUS_CANCELLED, "integer") . ")";
660 $set = $ilDB->query($sql);
661 while ($row = $ilDB->fetchAssoc($set)) {
662 $res[] = (int) $row["booking_reservation_id"];
663 }
664
665 return $res;
666 }
667}
const IL_CAL_DATE
const IL_CAL_UNIX
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static getNrOfItemsForObjects(array $a_obj_ids)
Get nr of available items for a set of object ids.
static lookupPoolId(int $object_id)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static getObjectReservationForUser(int $a_object_id, int $a_user_id)
static getCurrentOrUpcomingReservation(int $a_object_id)
Get details about object reservation.
static getCancelDetails(int $a_obj_id, int $a_user_id, int $a_from, int $a_to)
Get reservation ids from aggregated id for cancellation.
static getNumAvailablesNoSchedule(int $a_obj_id)
static isValidStatus(int $a_status)
ILIAS BookingManager Reservations ReservationDBRepository $repo
setAssignerId(int $a_assigner_id)
Set assigner user id.
static isObjectAvailableNoSchedule(int $a_obj_id)
static changeStatus(array $a_ids, int $a_status)
Batch update reservation status.
static isObjectAvailableInPeriod(int $a_obj_id, ilBookingSchedule $a_schedule, ?int $a_from, ?int $a_to)
static isBookingPoolLimitReachedByUser(int $a_user_id, int $a_pool_id)
static getMembersWithoutReservation(int $a_object_id)
static getList(array $a_object_ids, int $a_limit=10, int $a_offset=0, array $filter=[])
List all reservations.
static numAvailableFromObjectNoSchedule(int $a_obj_id)
static getAvailableObject(array $a_ids, int $a_from, int $a_to, bool $a_return_single=true, bool $a_return_counter=false)
Check if any of given objects are bookable.
static getUserFilter(array $a_object_ids)
Get all users who have reservations for object(s)
setTo(int $a_to)
Set booking to date.
setStatus(?int $a_status)
Set booking status.
setFrom(int $a_from)
Set booking from date.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
const ANONYMOUS_USER_ID
Definition: constants.php:27
Interface ilDBInterface.
$res
Definition: ltiservices.php:69
to(\GdImage $image, ?int $quality=null)
Currently this is the only way to make a FileStream from a GD image resource.
global $DIC
Definition: shib_login.php:26
$counter