MOON
Server: Apache
System: Linux res.emeff.ca 3.10.0-962.3.2.lve1.5.24.10.el7.x86_64 #1 SMP Wed Mar 20 07:36:02 EDT 2019 x86_64
User: accemeff (1004)
PHP: 7.0.33
Disabled: NONE
Upload Files
File: /home/accemeff/vendor/craftcms/cms/src/migrations/m160920_231045_usergroup_handle_title_unique.php
<?php

namespace craft\migrations;

use craft\db\Migration;
use craft\db\Query;
use craft\db\Table;
use craft\helpers\MigrationHelper;

/**
 * m160920_231045_usergroup_handle_title_unique migration.
 */
class m160920_231045_usergroup_handle_title_unique extends Migration
{
    /**
     * @inheritdoc
     */
    public function safeUp()
    {
        $this->_handleDupes('handle');
        $this->_handleDupes('name');

        if (!MigrationHelper::doesIndexExist(Table::USERGROUPS, 'handle', true, $this->db)) {
            $this->createIndex(null, Table::USERGROUPS, ['handle'], true);
        }

        if (!MigrationHelper::doesIndexExist(Table::USERGROUPS, 'name', true, $this->db)) {
            $this->createIndex(null, Table::USERGROUPS, ['name'], true);
        }
    }

    /**
     * @param string $type Either 'handle' or 'name'
     */
    private function _handleDupes(string $type)
    {
        echo '    > looking for duplicate user group ' . $type . 's ...';
        // Get any duplicates.
        $duplicates = (new Query())
            ->select($type)
            ->from([Table::USERGROUPS])
            ->groupBy([$type])
            ->having('count(' . $this->db->quoteValue($type) . ') > ' . $this->db->quoteValue('1'))
            ->all($this->db);

        if (!empty($duplicates)) {
            echo ' found ' . count($duplicates) . "\n";

            foreach ($duplicates as $duplicate) {
                echo '    > fixing duplicate "' . $duplicate[$type] . '" user group ' . $type . "s\n";

                $rows = (new Query())
                    ->from([Table::USERGROUPS])
                    ->where([$type => $duplicate[$type]])
                    ->orderBy(['dateCreated' => SORT_ASC])
                    ->all($this->db);

                // Find more than one?
                if (count($rows) > 1) {
                    // Skip the first (the earliest created), since presumably it's the good one.
                    unset($rows[0]);

                    foreach ($rows as $row) {
                        $newString = null;

                        // Let's give this 100 tries.
                        for ($counter = 1; $counter <= 100; $counter++) {
                            if ($type === 'handle') {
                                $newString = $duplicate[$type] . $counter;
                            } else {
                                $newString = $duplicate[$type] . ' ' . $counter;
                            }

                            $exists = (new Query())
                                ->from([Table::USERGROUPS])
                                ->where([$type => $newString])
                                ->exists($this->db);

                            // Found a free one.
                            if (!$exists) {
                                break;
                            }
                        }

                        // Let's update with a unique one.
                        $this->update(Table::USERGROUPS, [$type => $newString], ['id' => $row['id']]);
                    }
                }
            }
        } else {
            echo " none found\n";
        }
    }

    /**
     * @inheritdoc
     */
    public function safeDown()
    {
        echo "m160920_231045_usergroup_handle_title_unique cannot be reverted.\n";

        return false;
    }
}