2021-08-22 22:45:34 +02:00
|
|
|
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace OCA\GPodderSync\Core\EpisodeAction;
|
|
|
|
|
2021-10-07 14:33:16 +02:00
|
|
|
use DateTime;
|
2021-10-18 10:54:34 +02:00
|
|
|
use Doctrine\DBAL\Exception\UniqueConstraintViolationException;
|
2021-08-22 22:45:34 +02:00
|
|
|
use OCA\GPodderSync\Db\EpisodeAction\EpisodeActionEntity;
|
|
|
|
use OCA\GPodderSync\Db\EpisodeAction\EpisodeActionRepository;
|
|
|
|
use OCA\GPodderSync\Db\EpisodeAction\EpisodeActionWriter;
|
|
|
|
use OCP\DB\Exception;
|
|
|
|
|
|
|
|
class EpisodeActionSaver
|
|
|
|
{
|
|
|
|
|
2023-02-24 16:48:16 +01:00
|
|
|
private EpisodeActionRepository $episodeActionRepository;
|
|
|
|
private EpisodeActionWriter $episodeActionWriter;
|
|
|
|
private EpisodeActionReader $episodeActionReader;
|
|
|
|
|
|
|
|
private const DATETIME_FORMAT = 'Y-m-d\TH:i:s';
|
|
|
|
|
|
|
|
public function __construct(
|
|
|
|
EpisodeActionRepository $episodeActionRepository,
|
|
|
|
EpisodeActionWriter $episodeActionWriter,
|
|
|
|
EpisodeActionReader $episodeActionReader
|
|
|
|
)
|
|
|
|
{
|
|
|
|
$this->episodeActionRepository = $episodeActionRepository;
|
|
|
|
$this->episodeActionWriter = $episodeActionWriter;
|
|
|
|
$this->episodeActionReader = $episodeActionReader;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function saveEpisodeActions(array $episodeActionsArray, string $userId): array
|
|
|
|
{
|
|
|
|
$episodeActions = $this->episodeActionReader->fromArray($episodeActionsArray);
|
|
|
|
|
|
|
|
$episodeActionEntities = [];
|
2021-10-06 19:11:50 +02:00
|
|
|
|
2021-08-23 01:11:16 +02:00
|
|
|
foreach ($episodeActions as $episodeAction) {
|
2023-02-24 16:48:16 +01:00
|
|
|
$episodeActionEntity = $this->hydrateEpisodeActionEntity($episodeAction, $userId);
|
2021-08-22 22:45:34 +02:00
|
|
|
|
2023-02-24 16:48:16 +01:00
|
|
|
try {
|
2021-10-06 19:11:50 +02:00
|
|
|
$episodeActionEntities[] = $this->episodeActionWriter->save($episodeActionEntity);
|
2023-02-24 16:48:16 +01:00
|
|
|
} catch (UniqueConstraintViolationException $uniqueConstraintViolationException) {
|
|
|
|
$episodeActionEntities[] = $this->updateEpisodeAction($episodeActionEntity, $userId);
|
2021-08-23 01:11:16 +02:00
|
|
|
} catch (Exception $exception) {
|
|
|
|
if ($exception->getReason() === Exception::REASON_UNIQUE_CONSTRAINT_VIOLATION) {
|
2023-02-24 16:48:16 +01:00
|
|
|
$episodeActionEntities[] = $this->updateEpisodeAction($episodeActionEntity, $userId);
|
|
|
|
}
|
2021-08-23 01:11:16 +02:00
|
|
|
}
|
|
|
|
}
|
2023-02-24 16:48:16 +01:00
|
|
|
return $episodeActionEntities;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function convertTimestampToUnixEpoch(string $timestamp): string
|
|
|
|
{
|
|
|
|
return DateTime::createFromFormat(self::DATETIME_FORMAT, $timestamp)
|
|
|
|
->format("U");
|
|
|
|
}
|
|
|
|
|
|
|
|
private function updateEpisodeAction(
|
|
|
|
EpisodeActionEntity $episodeActionEntity,
|
|
|
|
string $userId
|
|
|
|
): EpisodeActionEntity
|
|
|
|
{
|
|
|
|
$episodeActionToUpdate = $this->findEpisodeActionToUpdate($episodeActionEntity, $userId);
|
|
|
|
|
|
|
|
$episodeActionEntity->setId($episodeActionToUpdate->getId());
|
|
|
|
|
|
|
|
$this->ensureGuidDoesNotGetNulledWithOldData($episodeActionToUpdate, $episodeActionEntity);
|
|
|
|
|
2023-02-24 20:48:53 +01:00
|
|
|
try {
|
|
|
|
return $this->episodeActionWriter->update($episodeActionEntity);
|
|
|
|
} catch (UniqueConstraintViolationException $uniqueConstraintViolationException) {
|
|
|
|
$this->deleteConflictingEpisodeAction($episodeActionEntity, $userId);
|
|
|
|
} catch (Exception $exception) {
|
|
|
|
if ($exception->getReason() === Exception::REASON_UNIQUE_CONSTRAINT_VIOLATION) {
|
|
|
|
$this->deleteConflictingEpisodeAction($episodeActionEntity, $userId);
|
|
|
|
}
|
|
|
|
}
|
2023-02-24 16:48:16 +01:00
|
|
|
return $this->episodeActionWriter->update($episodeActionEntity);
|
2023-02-24 20:48:53 +01:00
|
|
|
|
2023-02-24 16:48:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private function ensureGuidDoesNotGetNulledWithOldData(EpisodeAction $episodeActionToUpdate, EpisodeActionEntity $episodeActionEntity): void
|
|
|
|
{
|
|
|
|
$existingGuid = $episodeActionToUpdate->getGuid();
|
|
|
|
if ($existingGuid !== null && $episodeActionEntity->getGuid() === null) {
|
|
|
|
$episodeActionEntity->setGuid($existingGuid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private function hydrateEpisodeActionEntity(EpisodeAction $episodeAction, string $userId): EpisodeActionEntity
|
|
|
|
{
|
|
|
|
$episodeActionEntity = new EpisodeActionEntity();
|
|
|
|
$episodeActionEntity->setPodcast($episodeAction->getPodcast());
|
|
|
|
$episodeActionEntity->setEpisode($episodeAction->getEpisode());
|
|
|
|
$episodeActionEntity->setGuid($episodeAction->getGuid());
|
|
|
|
$episodeActionEntity->setAction($episodeAction->getAction());
|
|
|
|
$episodeActionEntity->setPosition($episodeAction->getPosition());
|
|
|
|
$episodeActionEntity->setStarted($episodeAction->getStarted());
|
|
|
|
$episodeActionEntity->setTotal($episodeAction->getTotal());
|
|
|
|
$episodeActionEntity->setTimestampEpoch($this->convertTimestampToUnixEpoch($episodeAction->getTimestamp()));
|
|
|
|
$episodeActionEntity->setUserId($userId);
|
|
|
|
|
|
|
|
return $episodeActionEntity;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function findEpisodeActionToUpdate(EpisodeActionEntity $episodeActionEntity, string $userId): ?EpisodeAction
|
|
|
|
{
|
|
|
|
$episodeAction = null;
|
|
|
|
if ($episodeActionEntity->getGuid() !== null) {
|
|
|
|
$episodeAction = $this->episodeActionRepository->findByGuid(
|
|
|
|
$episodeActionEntity->getGuid(),
|
|
|
|
$userId
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($episodeAction === null) {
|
|
|
|
$episodeAction = $this->episodeActionRepository->findByEpisodeUrl(
|
|
|
|
$episodeActionEntity->getEpisode(),
|
|
|
|
$userId
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $episodeAction;
|
|
|
|
}
|
2023-02-24 20:48:53 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param EpisodeActionEntity $episodeActionEntity
|
|
|
|
* @param string $userId
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
private function deleteConflictingEpisodeAction(EpisodeActionEntity $episodeActionEntity, string $userId): void
|
|
|
|
{
|
|
|
|
$collidingEpisodeActionId = $this->episodeActionRepository->findByEpisodeUrl($episodeActionEntity->getGuid(), $userId)->getId();
|
|
|
|
if ($collidingEpisodeActionId !== $episodeActionEntity->getId()) {
|
|
|
|
$this->episodeActionRepository->deleteEpisodeActionByEpisodeUrl($episodeActionEntity->getGuid(), $userId);
|
|
|
|
}
|
|
|
|
}
|
2021-08-22 22:45:34 +02:00
|
|
|
}
|