partkeepr/PartKeepr

View on GitHub
src/PartKeepr/CoreBundle/DoctrineMigrations/Version20150708120022.php

Summary

Maintainability
A
1 hr
Test Coverage
<?php

namespace PartKeepr\CoreBundle\DoctrineMigrations;

use Doctrine\DBAL\Schema\Schema;

/**
 * Fixes the category trees due to the migration of doctrine2-nestedset to DoctrineExtensions.
 */
class Version20150708120022 extends BaseMigration
{
    /**
     * @param Schema $schema
     */
    public function up(Schema $schema)
    {
        $this->performDatabaseUpgrade();

        $this->fixTree('PartCategory');
        $this->fixTree('FootprintCategory');
        $this->fixTree('StorageLocationCategory');

        $this->createRootNode('partkeepr.part.category_service');
        $this->createRootNode('partkeepr.storage_location.category_service');
        $this->createRootNode('partkeepr.footprint.category_service');
    }

    protected function createRootNode($service)
    {
        $this->getContainer()->get($service)->ensureRootNodeExists();
    }

    /**
     * Fixes the tree for a given table due to the migration of doctrine2-nestedset to DoctrineExtensions.
     *
     * @param string $table The table name to fix
     *
     * @throws \Doctrine\DBAL\DBALException
     */
    public function fixTree($table)
    {
        $nodes = $this->getNodeIds($table);

        $queryBuilder = $this->connection->createQueryBuilder();
        $queryBuilder->update($table)
            ->set('parent_id', ':parent')
            ->set('root', ':root')
            ->set('lvl', ':level')
            ->where('id = :id');

        foreach ($nodes as $node) {
            $parent = $this->fetchParent($table, $node['id']);
            $level = $this->getLevel($table, $node['id']);

            if ($parent !== false) {
                $this->connection->executeUpdate(
                    $queryBuilder->getSQL(),
                    [
                        ':parent' => $parent,
                        ':id'     => $node['id'],
                        ':level'  => $level,
                        ':root'   => 1,
                    ]
                );
            } else {
                $this->connection->executeUpdate(
                    $queryBuilder->getSQL(),
                    [
                        ':parent' => null,
                        ':id'     => $node['id'],
                        ':root'   => 1,
                        ':level'  => 0,
                    ]
                );
            }
        }
    }

    /**
     * Fetches the parent node for a table and ID.
     *
     * @param $table
     * @param $id
     *
     * @return mixed
     */
    public function fetchParent($table, $id)
    {
        $queryBuilder = $this->connection->createQueryBuilder();

        $queryBuilder->select('parent.id')
            ->from($table, 'node')
            ->from($table, 'parent')
            ->where('parent.lft < node.lft')
            ->andWhere('parent.rgt > node.rgt')
            ->andWhere('node.id = :nodeid')
            ->orderBy('parent.rgt - parent.lft')
            ->setMaxResults(1);

        return $this->connection->fetchColumn($queryBuilder->getSQL(), [':nodeid' => $id], 0);
    }

    /**
     * Returns the node IDs for the table.
     *
     * @param $table
     *
     * @return array
     */
    public function getNodeIds($table)
    {
        $qb = $this->connection->createQueryBuilder();

        $qb->select('id')
            ->from($table)
            ->orderBy('id', 'ASC');

        return $this->connection->fetchAll($qb->getSQL());
    }

    /**
     * Returns the level for a given table and ID.
     *
     * @param $table
     * @param $id
     *
     * @return mixed
     */
    public function getLevel($table, $id)
    {
        $qb = $this->connection->createQueryBuilder();

        $qb->select('COUNT(*) AS level')
            ->from($table, 'node')
            ->from($table, 'parent')
            ->where('node.lft > parent.lft')
            ->andWhere('node.lft < parent.rgt')
            ->andWhere('node.id = :nodeid');

        return $this->connection->fetchAssoc($qb->getSQL(), [':nodeid' => $id])['level'];
    }

    /**
     * @param Schema $schema
     */
    public function down(Schema $schema)
    {
        // this down() migration is auto-generated, please modify it to your needs
    }
}