<?php declare(strict_types=1); namespace OCA\GPodderSync\Core\EpisodeAction; use DateTimeZone; use Doctrine\DBAL\Exception\UniqueConstraintViolationException; use OCA\GPodderSync\Db\EpisodeAction\EpisodeActionEntity; use OCA\GPodderSync\Db\EpisodeAction\EpisodeActionRepository; use OCA\GPodderSync\Db\EpisodeAction\EpisodeActionWriter; use OCP\DB\Exception; class EpisodeActionSaver { private EpisodeActionRepository $episodeActionRepository; private EpisodeActionWriter $episodeActionWriter; private EpisodeActionReader $episodeActionReader; public function __construct( EpisodeActionRepository $episodeActionRepository, EpisodeActionWriter $episodeActionWriter, EpisodeActionReader $episodeActionReader ) { $this->episodeActionRepository = $episodeActionRepository; $this->episodeActionWriter = $episodeActionWriter; $this->episodeActionReader = $episodeActionReader; } /** * @param string $data * * @return EpisodeActionEntity[] */ public function saveEpisodeActions(string $data, string $userId): array { $episodeActionEntities = []; $episodeActions = $this->episodeActionReader->fromString($data); foreach ($episodeActions as $episodeAction) { $episodeActionEntity = $this->hydrateEpisodeActionEntity($episodeAction, $userId); try { $episodeActionEntities[] = $this->episodeActionWriter->save($episodeActionEntity); } catch (UniqueConstraintViolationException $uniqueConstraintViolationException) { $episodeActionEntities[] = $this->updateEpisodeAction($episodeActionEntity, $userId); } catch (Exception $exception) { if ($exception->getReason() === Exception::REASON_UNIQUE_CONSTRAINT_VIOLATION) { $episodeActionEntities[] = $this->updateEpisodeAction($episodeActionEntity, $userId); } } } return $episodeActionEntities; } /** * @param string $timestamp * * @return string */ private function convertTimestampToDbDateTimeString(string $timestamp): string { return \DateTime::createFromFormat('D F d H:i:s T Y', $timestamp) ->setTimezone(new DateTimeZone('UTC')) ->format("Y-m-d\TH:i:s"); } /** * @param EpisodeActionEntity $episodeActionEntity * * @return EpisodeActionEntity */ private function updateEpisodeAction( EpisodeActionEntity $episodeActionEntity, string $userId ): EpisodeActionEntity { $identifier = $episodeActionEntity->getGuid() ?? $episodeActionEntity->getEpisode(); $episodeActionEntityToUpdate = $this->episodeActionRepository->findByEpisodeIdentifier( $identifier, $userId ); if ($episodeActionEntityToUpdate === null && $episodeActionEntity->getGuid() !== null) { $episodeActionEntityToUpdate = $this->episodeActionRepository->findByEpisodeIdentifier( $episodeActionEntity->getEpisode(), $userId ); } $idEpisodeActionEntityToUpdate = $episodeActionEntityToUpdate->getId(); $episodeActionEntity->setId($idEpisodeActionEntityToUpdate); return $this->episodeActionWriter->update($episodeActionEntity); } /** * @param EpisodeAction $episodeAction * @param string $userId * * @return EpisodeActionEntity */ 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->setTimestamp($this->convertTimestampToDbDateTimeString($episodeAction->getTimestamp())); $episodeActionEntity->setUserId($userId); return $episodeActionEntity; } }