Add repair step to set MySQL collation to utf8_bin

Set default collation of mysql connection to utf8_bin
Set utf_bin as default collation for new tables
This commit is contained in:
Robin Appelman 2014-07-07 17:37:35 +02:00 committed by Thomas Müller
parent 687cd7fe83
commit 76c709d7de
5 changed files with 152 additions and 2 deletions

View File

@ -82,6 +82,7 @@ class MDB2SchemaReader {
$name = str_replace('*dbprefix*', $this->DBTABLEPREFIX, $name); $name = str_replace('*dbprefix*', $this->DBTABLEPREFIX, $name);
$name = $this->platform->quoteIdentifier($name); $name = $this->platform->quoteIdentifier($name);
$table = $schema->createTable($name); $table = $schema->createTable($name);
$table->addOption('collate', 'utf8_bin');
break; break;
case 'create': case 'create':
case 'overwrite': case 'overwrite':

View File

@ -81,7 +81,8 @@ class Repair extends BasicEmitter {
*/ */
public static function getBeforeUpgradeRepairSteps() { public static function getBeforeUpgradeRepairSteps() {
return array( return array(
new \OC\Repair\InnoDB() new \OC\Repair\InnoDB(),
new \OC\Repair\Collation(\OC::$server->getConfig(), \OC_DB::getConnection())
); );
} }

View File

@ -61,7 +61,7 @@ class MySQL extends AbstractDatabase {
$name = $this->dbname; $name = $this->dbname;
$user = $this->dbuser; $user = $this->dbuser;
//we cant use OC_BD functions here because we need to connect as the administrative user. //we cant use OC_BD functions here because we need to connect as the administrative user.
$query = "CREATE DATABASE IF NOT EXISTS `$name`"; $query = "CREATE DATABASE IF NOT EXISTS `$name` CHARACTER SET utf8 COLLATE utf8_bin;";
$result = mysql_query($query, $connection); $result = mysql_query($query, $connection);
if(!$result) { if(!$result) {
$entry = $this->trans->t('DB Error: "%s"', array(mysql_error($connection))) . '<br />'; $entry = $this->trans->t('DB Error: "%s"', array(mysql_error($connection))) . '<br />';

75
lib/repair/collation.php Normal file
View File

@ -0,0 +1,75 @@
<?php
/**
* Copyright (c) 2014 Robin Appelman <icewind@owncloud.com>
* This file is licensed under the Affero General Public License version 3 or
* later.
* See the COPYING-README file.
*/
namespace OC\Repair;
use Doctrine\DBAL\Platforms\MySqlPlatform;
use OC\Hooks\BasicEmitter;
class Collation extends BasicEmitter implements \OC\RepairStep {
/**
* @var \OCP\IConfig
*/
protected $config;
/**
* @var \OC\DB\Connection
*/
protected $connection;
/**
* @param \OCP\IConfig $config
* @param \OC\DB\Connection $connection
*/
public function __construct($config, $connection) {
$this->connection = $connection;
$this->config = $config;
}
public function getName() {
return 'Repair MySQL collation';
}
/**
* Fix mime types
*/
public function run() {
if (!$this->connection->getDatabasePlatform() instanceof MySqlPlatform) {
$this->emit('\OC\Repair', 'info', array('Not a mysql database -> nothing to no'));
return;
}
$tables = $this->getAllNonUTF8BinTables($this->connection);
foreach ($tables as $table) {
$query = $this->connection->prepare('ALTER TABLE `' . $table . '` CONVERT TO CHARACTER SET utf8 COLLATE utf8_bin;');
$query->execute();
}
}
/**
* @param \Doctrine\DBAL\Connection $connection
* @return string[]
*/
protected function getAllNonUTF8BinTables($connection) {
$dbName = $this->config->getSystemValue("dbname");
$rows = $connection->fetchAll(
"SELECT DISTINCT(TABLE_NAME) AS `table`" .
" FROM INFORMATION_SCHEMA . COLUMNS" .
" WHERE TABLE_SCHEMA = ?" .
" AND (COLLATION_NAME <> 'utf8_bin' OR CHARACTER_SET_NAME <> 'utf8')" .
" AND TABLE_NAME LIKE \"*PREFIX*%\"",
array($dbName)
);
$result = array();
foreach ($rows as $row) {
$result[] = $row['table'];
}
return $result;
}
}

View File

@ -0,0 +1,73 @@
<?php
/**
* Copyright (c) 2014 Thomas Müller <deepdiver@owncloud.com>
* This file is licensed under the Affero General Public License version 3 or
* later.
* See the COPYING-README file.
*/
class TestCollationRepair extends \OC\Repair\Collation {
/**
* @param \Doctrine\DBAL\Connection $connection
* @return string[]
*/
public function getAllNonUTF8BinTables($connection) {
return parent::getAllNonUTF8BinTables($connection);
}
}
/**
* Tests for the converting of MySQL tables to InnoDB engine
*
* @see \OC\Repair\RepairMimeTypes
*/
class TestRepairCollation extends PHPUnit_Framework_TestCase {
/**
* @var TestCollationRepair
*/
private $repair;
/**
* @var \Doctrine\DBAL\Connection
*/
private $connection;
/**
* @var string
*/
private $tableName;
/**
* @var \OCP\IConfig
*/
private $config;
public function setUp() {
$this->connection = \OC_DB::getConnection();
$this->config = \OC::$server->getConfig();
if (!$this->connection->getDatabasePlatform() instanceof \Doctrine\DBAL\Platforms\MySqlPlatform) {
$this->markTestSkipped("Test only relevant on MySql");
}
$dbPrefix = $this->config->getSystemValue("dbtableprefix");
$this->tableName = uniqid($dbPrefix . "_collation_test");
$this->connection->exec("CREATE TABLE $this->tableName(text VARCHAR(16)) COLLATE utf8_unicode_ci");
$this->repair = new TestCollationRepair($this->config, $this->connection);
}
public function tearDown() {
$this->connection->getSchemaManager()->dropTable($this->tableName);
}
public function testCollationConvert() {
$tables = $this->repair->getAllNonUTF8BinTables($this->connection);
$this->assertGreaterThanOrEqual(1, count($tables));
$this->repair->run();
$tables = $this->repair->getAllNonUTF8BinTables($this->connection);
$this->assertCount(0, $tables);
}
}