Initial Upload
This commit is contained in:
357
lib/Entity/DayPart.php
Normal file
357
lib/Entity/DayPart.php
Normal file
@@ -0,0 +1,357 @@
|
||||
<?php
|
||||
/*
|
||||
* Copyright (C) 2023 Xibo Signage Ltd
|
||||
*
|
||||
* Xibo - Digital Signage - https://xibosignage.com
|
||||
*
|
||||
* This file is part of Xibo.
|
||||
*
|
||||
* Xibo is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU Affero General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* any later version.
|
||||
*
|
||||
* Xibo is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU Affero General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Affero General Public License
|
||||
* along with Xibo. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
namespace Xibo\Entity;
|
||||
|
||||
use Carbon\Carbon;
|
||||
use Respect\Validation\Validator as v;
|
||||
use Xibo\Event\DayPartDeleteEvent;
|
||||
use Xibo\Factory\ScheduleFactory;
|
||||
use Xibo\Service\DisplayNotifyServiceInterface;
|
||||
use Xibo\Service\LogServiceInterface;
|
||||
use Xibo\Storage\StorageServiceInterface;
|
||||
use Xibo\Support\Exception\GeneralException;
|
||||
use Xibo\Support\Exception\InvalidArgumentException;
|
||||
use Xibo\Support\Exception\NotFoundException;
|
||||
|
||||
/**
|
||||
* Class DayPart
|
||||
* @package Xibo\Entity
|
||||
*
|
||||
* @SWG\Definition()
|
||||
*/
|
||||
class DayPart implements \JsonSerializable
|
||||
{
|
||||
use EntityTrait;
|
||||
|
||||
/**
|
||||
* @SWG\Property(description="The ID of this Daypart")
|
||||
* @var int
|
||||
*/
|
||||
public $dayPartId;
|
||||
public $name;
|
||||
public $description;
|
||||
public $isRetired;
|
||||
public $userId;
|
||||
|
||||
public $startTime;
|
||||
public $endTime;
|
||||
public $exceptions;
|
||||
|
||||
/**
|
||||
* @SWG\Property(description="A readonly flag determining whether this DayPart is always")
|
||||
* @var int
|
||||
*/
|
||||
public $isAlways = 0;
|
||||
|
||||
/**
|
||||
* @SWG\Property(description="A readonly flag determining whether this DayPart is custom")
|
||||
* @var int
|
||||
*/
|
||||
public $isCustom = 0;
|
||||
|
||||
/** @var Carbon $adjustedStart Adjusted start datetime */
|
||||
public $adjustedStart;
|
||||
|
||||
/** @var Carbon Adjusted end datetime */
|
||||
public $adjustedEnd;
|
||||
|
||||
private $timeHash;
|
||||
|
||||
/** @var ScheduleFactory */
|
||||
private $scheduleFactory;
|
||||
|
||||
/** @var DisplayNotifyServiceInterface */
|
||||
private $displayNotifyService;
|
||||
|
||||
/**
|
||||
* Entity constructor.
|
||||
* @param StorageServiceInterface $store
|
||||
* @param LogServiceInterface $log
|
||||
* @param \Symfony\Component\EventDispatcher\EventDispatcherInterface $dispatcher
|
||||
*/
|
||||
public function __construct($store, $log, $dispatcher)
|
||||
{
|
||||
$this->setCommonDependencies($store, $log, $dispatcher);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param ScheduleFactory $scheduleFactory
|
||||
* @param \Xibo\Service\DisplayNotifyServiceInterface $displayNotifyService
|
||||
* @return $this
|
||||
*/
|
||||
public function setScheduleFactory($scheduleFactory, DisplayNotifyServiceInterface $displayNotifyService)
|
||||
{
|
||||
$this->scheduleFactory = $scheduleFactory;
|
||||
$this->displayNotifyService = $displayNotifyService;
|
||||
return $this;
|
||||
}
|
||||
/**
|
||||
* Calculate time hash
|
||||
* @return string
|
||||
*/
|
||||
private function calculateTimeHash()
|
||||
{
|
||||
$hash = $this->startTime . $this->endTime;
|
||||
|
||||
foreach ($this->exceptions as $exception) {
|
||||
$hash .= $exception['day'] . $exception['start'] . $exception['end'];
|
||||
}
|
||||
|
||||
return md5($hash);
|
||||
}
|
||||
|
||||
public function isSystemDayPart(): bool
|
||||
{
|
||||
return ($this->isAlways || $this->isCustom);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return int
|
||||
*/
|
||||
public function getId()
|
||||
{
|
||||
return $this->dayPartId;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return int
|
||||
*/
|
||||
public function getOwnerId()
|
||||
{
|
||||
return $this->userId;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets the Owner
|
||||
* @param int $ownerId
|
||||
*/
|
||||
public function setOwner($ownerId)
|
||||
{
|
||||
$this->userId = $ownerId;
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
public function validate()
|
||||
{
|
||||
$this->getLog()->debug('Validating daypart ' . $this->name);
|
||||
|
||||
if (!v::stringType()->notEmpty()->validate($this->name))
|
||||
throw new InvalidArgumentException(__('Name cannot be empty'), 'name');
|
||||
|
||||
// Check the start/end times are in the correct format (H:i)
|
||||
if ((strlen($this->startTime) != 8 && strlen($this->startTime) != 5) || (strlen($this->endTime) != 8 && strlen($this->endTime) != 5))
|
||||
throw new InvalidArgumentException(__('Start/End time are empty or in an incorrect format'), 'start/end time');
|
||||
|
||||
foreach ($this->exceptions as $exception) {
|
||||
if ((strlen($exception['start']) != 8 && strlen($exception['start']) != 5) || (strlen($exception['end']) != 8 && strlen($exception['end']) != 5))
|
||||
throw new InvalidArgumentException(sprintf(__('Exception Start/End time for %s are empty or in an incorrect format'), $exception['day']), 'exception start/end time');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Load
|
||||
* @return $this
|
||||
*/
|
||||
public function load()
|
||||
{
|
||||
$this->timeHash = $this->calculateTimeHash();
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param \Carbon\Carbon $date
|
||||
* @return void
|
||||
*/
|
||||
public function adjustForDate(Carbon $date)
|
||||
{
|
||||
// Matching exceptions?
|
||||
// we use a lookup because the form control uses the below date abbreviations
|
||||
$dayOfWeekLookup = ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'];
|
||||
foreach ($this->exceptions as $exception) {
|
||||
if ($exception['day'] === $dayOfWeekLookup[$date->dayOfWeekIso]) {
|
||||
$this->adjustedStart = $date->copy()->setTimeFromTimeString($exception['start']);
|
||||
$this->adjustedEnd = $date->copy()->setTimeFromTimeString($exception['end']);
|
||||
if ($this->adjustedStart >= $this->adjustedEnd) {
|
||||
$this->adjustedEnd->addDay();
|
||||
}
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// No matching exceptions.
|
||||
$this->adjustedStart = $date->copy()->setTimeFromTimeString($this->startTime);
|
||||
$this->adjustedEnd = $date->copy()->setTimeFromTimeString($this->endTime);
|
||||
if ($this->adjustedStart >= $this->adjustedEnd) {
|
||||
$this->adjustedEnd->addDay();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Save
|
||||
* @param array $options
|
||||
* @throws InvalidArgumentException
|
||||
* @throws GeneralException
|
||||
* @throws NotFoundException
|
||||
*/
|
||||
public function save($options = [])
|
||||
{
|
||||
$options = array_merge([
|
||||
'validate' => true,
|
||||
'recalculateHash' => true
|
||||
], $options);
|
||||
|
||||
if ($options['validate']) {
|
||||
$this->validate();
|
||||
}
|
||||
|
||||
if ($this->dayPartId == 0) {
|
||||
$this->add();
|
||||
} else {
|
||||
// Update
|
||||
$this->update();
|
||||
|
||||
// When we change user on reassignAllTo, we do save dayPart,
|
||||
// however it will not have required childObjectDependencies to run the below checks
|
||||
// it is also not needed to run them when we just changed the owner.
|
||||
if ($options['recalculateHash']) {
|
||||
// Compare the time hash with a new time hash to see if we need to update associated schedules
|
||||
if ($this->timeHash != $this->calculateTimeHash()) {
|
||||
$this->handleEffectedSchedules();
|
||||
} else {
|
||||
$this->getLog()->debug('Daypart hash identical, no need to update schedules. ' . $this->timeHash . ' vs ' . $this->calculateTimeHash());
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Delete
|
||||
*/
|
||||
public function delete()
|
||||
{
|
||||
if ($this->isSystemDayPart()) {
|
||||
throw new InvalidArgumentException('Cannot delete system dayParts');
|
||||
}
|
||||
|
||||
$this->getDispatcher()->dispatch(new DayPartDeleteEvent($this), DayPartDeleteEvent::$NAME);
|
||||
|
||||
// Delete all events using this daypart
|
||||
$schedules = $this->scheduleFactory->getByDayPartId($this->dayPartId);
|
||||
|
||||
foreach ($schedules as $schedule) {
|
||||
$schedule->delete();
|
||||
}
|
||||
|
||||
// Delete the daypart
|
||||
$this->getStore()->update('DELETE FROM `daypart` WHERE dayPartId = :dayPartId', ['dayPartId' => $this->dayPartId]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Add
|
||||
*/
|
||||
private function add()
|
||||
{
|
||||
$this->dayPartId = $this->getStore()->insert('
|
||||
INSERT INTO `daypart` (`name`, `description`, `isRetired`, `userId`, `startTime`, `endTime`, `exceptions`)
|
||||
VALUES (:name, :description, :isRetired, :userId, :startTime, :endTime, :exceptions)
|
||||
', [
|
||||
'name' => $this->name,
|
||||
'description' => $this->description,
|
||||
'isRetired' => $this->isRetired,
|
||||
'userId' => $this->userId,
|
||||
'startTime' => $this->startTime,
|
||||
'endTime' => $this->endTime,
|
||||
'exceptions' => json_encode(is_array($this->exceptions) ? $this->exceptions : [])
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Update
|
||||
*/
|
||||
private function update()
|
||||
{
|
||||
$this->getStore()->update('
|
||||
UPDATE `daypart`
|
||||
SET `name` = :name,
|
||||
`description` = :description,
|
||||
`isRetired` = :isRetired,
|
||||
`userId` = :userId,
|
||||
`startTime` = :startTime,
|
||||
`endTime` = :endTime,
|
||||
`exceptions` = :exceptions
|
||||
WHERE `daypart`.dayPartId = :dayPartId
|
||||
', [
|
||||
'dayPartId' => $this->dayPartId,
|
||||
'name' => $this->name,
|
||||
'description' => $this->description,
|
||||
'isRetired' => $this->isRetired,
|
||||
'userId' => $this->userId,
|
||||
'startTime' => $this->startTime,
|
||||
'endTime' => $this->endTime,
|
||||
'exceptions' => json_encode(is_array($this->exceptions) ? $this->exceptions : [])
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Handles schedules effected by an update
|
||||
* @throws NotFoundException
|
||||
* @throws GeneralException
|
||||
*/
|
||||
private function handleEffectedSchedules()
|
||||
{
|
||||
$now = Carbon::now()->format('U');
|
||||
|
||||
// Get all schedules that use this dayPart and exist after the current time.
|
||||
$schedules = $this->scheduleFactory->query(null, ['dayPartId' => $this->dayPartId, 'futureSchedulesFrom' => $now]);
|
||||
|
||||
$this->getLog()->debug('Daypart update effects ' . count($schedules) . ' schedules.');
|
||||
|
||||
foreach ($schedules as $schedule) {
|
||||
/** @var Schedule $schedule */
|
||||
$schedule
|
||||
->setDisplayNotifyService($this->displayNotifyService)
|
||||
->load();
|
||||
|
||||
// Is this schedule a recurring event?
|
||||
if ($schedule->recurrenceType != '' && $schedule->fromDt < $now) {
|
||||
$this->getLog()->debug('Schedule is for a recurring event which has already recurred');
|
||||
|
||||
// Split the scheduled event, adjusting only the recurring end date on the original event
|
||||
$newSchedule = clone $schedule;
|
||||
$schedule->recurrenceRange = $now;
|
||||
$schedule->save();
|
||||
|
||||
// Adjusting the fromdt on the new event
|
||||
$newSchedule->fromDt = Carbon::now()->addDay()->format('U');
|
||||
$newSchedule->save();
|
||||
} else {
|
||||
$this->getLog()->debug('Schedule is for a single event');
|
||||
|
||||
// Update just this single event to have the new date/time
|
||||
$schedule->save();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user