2019-08-02 15:53:28 +12:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace SilverStripe\ORM\Tests;
|
|
|
|
|
2023-07-07 15:56:31 +12:00
|
|
|
use mysqli_driver;
|
2019-08-02 15:53:28 +12:00
|
|
|
use SilverStripe\Dev\SapphireTest;
|
|
|
|
use SilverStripe\Dev\TestOnly;
|
2023-07-07 15:56:31 +12:00
|
|
|
use SilverStripe\ORM\Connect\DatabaseException;
|
2019-08-02 15:53:28 +12:00
|
|
|
use SilverStripe\ORM\DB;
|
2023-07-07 15:56:31 +12:00
|
|
|
use SilverStripe\ORM\Tests\MySQLiConnectorTest\MySQLiConnector;
|
2022-05-26 12:06:51 +12:00
|
|
|
use SilverStripe\Tests\ORM\Utf8\Utf8TestHelper;
|
2024-09-18 13:53:44 +12:00
|
|
|
use PHPUnit\Framework\Attributes\DataProvider;
|
|
|
|
use PHPUnit\Framework\Attributes\RequiresPhpExtension;
|
2019-08-02 15:53:28 +12:00
|
|
|
|
2024-09-18 13:53:44 +12:00
|
|
|
#[RequiresPhpExtension('mysqli')]
|
2019-08-02 15:53:28 +12:00
|
|
|
class MySQLiConnectorTest extends SapphireTest implements TestOnly
|
|
|
|
{
|
2023-07-07 15:56:31 +12:00
|
|
|
/** @var array project database settings configuration */
|
|
|
|
private $config = [];
|
|
|
|
|
|
|
|
private function getConnector(?string $charset = null, ?string $collation = null, bool $selectDb = false)
|
2019-08-02 15:53:28 +12:00
|
|
|
{
|
2023-07-07 15:56:31 +12:00
|
|
|
$config = $this->config;
|
|
|
|
|
|
|
|
if ($charset) {
|
|
|
|
$config['charset'] = $charset;
|
|
|
|
}
|
|
|
|
if ($collation) {
|
|
|
|
$config['collation'] = $collation;
|
|
|
|
}
|
|
|
|
|
2019-08-02 15:53:28 +12:00
|
|
|
$config['database'] = 'information_schema';
|
|
|
|
|
2023-07-07 15:56:31 +12:00
|
|
|
$connector = new MySQLiConnector();
|
|
|
|
$connector->connect($config, $selectDb);
|
|
|
|
|
|
|
|
return $connector;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setUp(): void
|
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
2024-10-09 12:18:03 +13:00
|
|
|
$config = DB::getConfig(DB::CONN_PRIMARY);
|
2023-07-07 15:56:31 +12:00
|
|
|
|
2022-04-14 13:12:59 +12:00
|
|
|
if (strtolower(substr($config['type'] ?? '', 0, 5)) !== 'mysql') {
|
2023-07-07 15:56:31 +12:00
|
|
|
$this->markTestSkipped("The test only relevant for MySQL - but $config[type] is in use");
|
2019-08-02 15:53:28 +12:00
|
|
|
}
|
|
|
|
|
2023-07-07 15:56:31 +12:00
|
|
|
$this->config = $config;
|
|
|
|
}
|
|
|
|
|
2024-09-18 13:53:44 +12:00
|
|
|
#[DataProvider('charsetProvider')]
|
2023-07-07 15:56:31 +12:00
|
|
|
public function testConnectionCharsetControl($charset, $defaultCollation)
|
|
|
|
{
|
|
|
|
$connector = $this->getConnector($charset);
|
2019-08-02 15:53:28 +12:00
|
|
|
$connection = $connector->getMysqliConnection();
|
|
|
|
|
|
|
|
$cset = $connection->get_charset();
|
|
|
|
|
2022-05-26 12:06:51 +12:00
|
|
|
// Note: we do not need to update the utf charset here because mysqli with newer
|
|
|
|
// version of mysql/mariadb still self-reports as 'utf8' rather than 'utf8mb3'
|
2024-06-18 09:37:39 +12:00
|
|
|
// This is unlike MySQLiConnectorTest::testConnectionCollationControl()
|
2019-08-02 15:53:28 +12:00
|
|
|
$this->assertEquals($charset, $cset->charset);
|
|
|
|
$this->assertEquals($defaultCollation, $cset->collation);
|
|
|
|
|
|
|
|
unset($cset, $connection, $connector, $config);
|
|
|
|
}
|
|
|
|
|
2024-09-18 13:53:44 +12:00
|
|
|
#[DataProvider('charsetProvider')]
|
2019-08-02 15:53:28 +12:00
|
|
|
public function testConnectionCollationControl($charset, $defaultCollation, $customCollation)
|
|
|
|
{
|
2023-07-07 15:56:31 +12:00
|
|
|
$connector = $this->getConnector($charset, $customCollation);
|
2019-08-02 15:53:28 +12:00
|
|
|
$connection = $connector->getMysqliConnection();
|
|
|
|
|
|
|
|
$cset = $connection->get_charset();
|
|
|
|
|
|
|
|
$this->assertEquals($charset, $cset->charset);
|
|
|
|
|
|
|
|
/* Warning! This is a MySQLi limitation.
|
|
|
|
* If it changes in the future versions, this test may break.
|
|
|
|
* We are still testing for it as a limitation and a
|
|
|
|
* reminder that it exists.
|
|
|
|
*
|
|
|
|
* To make sure that we actually have correct collation see
|
|
|
|
* - testUtf8mb4GeneralCollation
|
|
|
|
* - testUtf8mb4UnicodeCollation
|
|
|
|
*/
|
|
|
|
$this->assertEquals(
|
|
|
|
$defaultCollation,
|
|
|
|
$cset->collation,
|
|
|
|
'This is an issue with mysqli. It always returns "default" collation, even if another is active'
|
|
|
|
);
|
|
|
|
|
|
|
|
$cset = $connection->query('show variables like "character_set_connection"')->fetch_array()[1];
|
|
|
|
$collation = $connection->query('show variables like "collation_connection"')->fetch_array()[1];
|
|
|
|
|
2022-05-26 12:06:51 +12:00
|
|
|
$helper = new Utf8TestHelper();
|
|
|
|
$this->assertEquals($helper->getUpdatedUtfCharsetForCurrentDB($charset), $cset);
|
|
|
|
$this->assertEquals($helper->getUpdatedUtfCollationForCurrentDB($customCollation), $collation);
|
2019-08-02 15:53:28 +12:00
|
|
|
|
|
|
|
$connection->close();
|
|
|
|
unset($cset, $connection, $connector, $config);
|
|
|
|
}
|
|
|
|
|
2024-09-18 13:53:44 +12:00
|
|
|
public static function charsetProvider()
|
2019-08-02 15:53:28 +12:00
|
|
|
{
|
|
|
|
return [
|
|
|
|
['ascii', 'ascii_general_ci', 'ascii_bin'],
|
|
|
|
['utf8', 'utf8_general_ci', 'utf8_unicode_520_ci'],
|
|
|
|
['utf8mb4', 'utf8mb4_general_ci', 'utf8mb4_unicode_520_ci']
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testUtf8mb4GeneralCollation()
|
|
|
|
{
|
2023-07-07 15:56:31 +12:00
|
|
|
$connector = $this->getConnector('utf8mb4', 'utf8mb4_general_ci', true);
|
2019-08-02 15:53:28 +12:00
|
|
|
$connection = $connector->getMysqliConnection();
|
|
|
|
|
|
|
|
$result = $connection->query(
|
|
|
|
"select `a`.`value` from (select 'rst' `value` union select 'rßt' `value`) `a` order by `value`"
|
|
|
|
)->fetch_all();
|
|
|
|
|
|
|
|
$this->assertCount(1, $result, '`utf8mb4_general_ci` handles both values as equal to "rst"');
|
|
|
|
$this->assertEquals('rst', $result[0][0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testUtf8mb4UnicodeCollation()
|
|
|
|
{
|
2023-07-07 15:56:31 +12:00
|
|
|
$connector = $this->getConnector('utf8mb4', 'utf8mb4_unicode_ci', true);
|
2019-08-02 15:53:28 +12:00
|
|
|
$connection = $connector->getMysqliConnection();
|
|
|
|
|
|
|
|
$result = $connection->query(
|
|
|
|
"select `a`.`value` from (select 'rst' `value` union select 'rßt' `value`) `a` order by `value`"
|
|
|
|
)->fetch_all();
|
|
|
|
|
|
|
|
$this->assertCount(2, $result, '`utf8mb4_unicode_ci` must recognise "rst" and "rßt" as different values');
|
|
|
|
$this->assertEquals('rßt', $result[0][0]);
|
|
|
|
$this->assertEquals('rst', $result[1][0]);
|
|
|
|
}
|
2023-07-07 15:56:31 +12:00
|
|
|
|
|
|
|
public function testQueryThrowsDatabaseErrorOnMySQLiError()
|
|
|
|
{
|
|
|
|
$connector = $this->getConnector();
|
|
|
|
$driver = new mysqli_driver();
|
|
|
|
// The default with PHP >= 8.0
|
|
|
|
$driver->report_mode = MYSQLI_REPORT_OFF;
|
|
|
|
$this->expectException(DatabaseException::class);
|
|
|
|
$connector = $this->getConnector(null, null, true);
|
|
|
|
$connector->query('force an error with invalid SQL');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testQueryThrowsDatabaseErrorOnMySQLiException()
|
|
|
|
{
|
|
|
|
$driver = new mysqli_driver();
|
|
|
|
// The default since PHP 8.1 - https://www.php.net/manual/en/mysqli-driver.report-mode.php
|
|
|
|
$driver->report_mode = MYSQLI_REPORT_ERROR | MYSQLI_REPORT_STRICT;
|
|
|
|
$this->expectException(DatabaseException::class);
|
|
|
|
$connector = $this->getConnector(null, null, true);
|
|
|
|
$connector->query('force an error with invalid SQL');
|
|
|
|
}
|
2019-08-02 15:53:28 +12:00
|
|
|
}
|