chamilo/chamilo-lms

View on GitHub
src/CoreBundle/Migrations/Schema/V200/Version20240323181500.php

Summary

Maintainability
A
0 mins
Test Coverage
<?php

/* For licensing terms, see /license.txt */

declare(strict_types=1);

namespace Chamilo\CoreBundle\Migrations\Schema\V200;

use Chamilo\CoreBundle\Entity\ResourceLink;
use Chamilo\CoreBundle\Entity\User;
use Chamilo\CoreBundle\Migrations\AbstractMigrationChamilo;
use Chamilo\CourseBundle\Entity\CCalendarEvent;
use DateTime;
use DateTimeZone;
use Doctrine\DBAL\Schema\Schema;

class Version20240323181500 extends AbstractMigrationChamilo
{
    public function getDescription(): string
    {
        return 'Migrate sys_calendar to c_calendar_event';
    }

    public function up(Schema $schema): void
    {
        $sysCalendars = $this->connection->fetchAllAssociative('SELECT * FROM sys_calendar');

        $utc = new DateTimeZone('UTC');
        $oldNewEventIdMap = [];

        $admin = $this->getAdmin();
        foreach ($sysCalendars as $sysCalendar) {
            $calendarEvent = $this->createCCalendarEvent(
                $sysCalendar['title'] ?: '-',
                $sysCalendar['content'],
                $sysCalendar['start_date'] ? new DateTime($sysCalendar['start_date'], $utc) : null,
                $sysCalendar['end_date'] ? new DateTime($sysCalendar['end_date'], $utc) : null,
                (bool) $sysCalendar['all_day'],
                $sysCalendar['color'] ?? '',
                $admin
            );

            $this->entityManager->persist($calendarEvent);
            $this->entityManager->flush();

            $this->addGlobalResourceLinkToNode($calendarEvent->getResourceNode());

            $oldNewEventIdMap[$sysCalendar['id']] = $calendarEvent;
        }

        $this->entityManager->flush();

        if ($schema->hasTable('agenda_reminder')) {
            $tblAgendaReminder = $schema->getTable('agenda_reminder');

            if ($tblAgendaReminder->hasColumn('type')) {
                $this->updateAgendaReminders($oldNewEventIdMap);
            }
        }
    }

    private function createCCalendarEvent(
        string $title,
        string $content,
        ?DateTime $startDate,
        ?DateTime $endDate,
        bool $allDay,
        string $color,
        User $creator
    ): CCalendarEvent {
        $calendarEvent = new CCalendarEvent();
        $calendarEvent
            ->setTitle($title)
            ->setContent($content)
            ->setStartDate($startDate)
            ->setEndDate($endDate)
            ->setAllDay($allDay)
            ->setColor($color)
            ->setCreator($creator)
            ->setResourceName($title)
            ->setParentResourceNode($creator->getResourceNode()->getId())
        ;

        return $calendarEvent;
    }

    private function addGlobalResourceLinkToNode($resourceNode): void
    {
        $globalLink = new ResourceLink();
        $globalLink->setCourse(null)
            ->setSession(null)
            ->setGroup(null)
            ->setUser(null)
        ;

        $alreadyHasGlobalLink = false;
        foreach ($resourceNode->getResourceLinks() as $existingLink) {
            if (null === $existingLink->getCourse() && null === $existingLink->getSession()
                && null === $existingLink->getGroup() && null === $existingLink->getUser()) {
                $alreadyHasGlobalLink = true;

                break;
            }
        }

        if (!$alreadyHasGlobalLink) {
            $resourceNode->addResourceLink($globalLink);
            $this->entityManager->persist($globalLink);
        }
    }

    /**
     * @param array<int, CCalendarEvent> $oldNewEventIdMap
     */
    private function updateAgendaReminders(array $oldNewEventIdMap): void
    {
        $result = $this->connection->executeQuery("SELECT * FROM agenda_reminder WHERE type = 'admin'");

        while (($reminder = $result->fetchAssociative()) !== false) {
            $oldEventId = $reminder['event_id'];
            if (\array_key_exists($oldEventId, $oldNewEventIdMap)) {
                $newEvent = $oldNewEventIdMap[$oldEventId];
                $this->addSql(
                    \sprintf(
                        'UPDATE agenda_reminder SET event_id = %d WHERE id = %d',
                        $newEvent->getIid(),
                        $reminder['id']
                    )
                );
            }
        }
    }

    public function down(Schema $schema): void
    {
        // Down migration is not defined, as data migration cannot be easily reverted
    }
}