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:
parent
687cd7fe83
commit
76c709d7de
|
@ -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':
|
||||||
|
|
|
@ -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())
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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 />';
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue