Set default collation of mysql connection to utf8_bin Set utf_bin as default collation for new tablesremotes/origin/fix-10825
parent
687cd7fe83
commit
76c709d7de
@ -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