Unverified Commit e4978c22 authored by Jonathan H. Wage's avatar Jonathan H. Wage Committed by GitHub

Merge pull request #3530 from doctrine/fix-readonly-sqlite-exception-test

Improve ExceptionTest::testConnectionExceptionSqLite
parents 945b2710 b244b9e6
...@@ -9,10 +9,13 @@ use Doctrine\DBAL\Schema\Schema; ...@@ -9,10 +9,13 @@ use Doctrine\DBAL\Schema\Schema;
use Doctrine\DBAL\Schema\Table; use Doctrine\DBAL\Schema\Table;
use Doctrine\Tests\DbalFunctionalTestCase; use Doctrine\Tests\DbalFunctionalTestCase;
use Throwable; use Throwable;
use const PHP_OS;
use function array_merge; use function array_merge;
use function chmod; use function chmod;
use function defined; use function exec;
use function file_exists; use function file_exists;
use function posix_geteuid;
use function posix_getpwuid;
use function sprintf; use function sprintf;
use function sys_get_temp_dir; use function sys_get_temp_dir;
use function touch; use function touch;
...@@ -284,25 +287,28 @@ class ExceptionTest extends DbalFunctionalTestCase ...@@ -284,25 +287,28 @@ class ExceptionTest extends DbalFunctionalTestCase
$this->connection->executeQuery($sql); $this->connection->executeQuery($sql);
} }
/** public function testConnectionExceptionSqLite()
* @dataProvider getSqLiteOpenConnection
*/
public function testConnectionExceptionSqLite($mode, $exceptionClass)
{ {
if ($this->connection->getDatabasePlatform()->getName() !== 'sqlite') { if ($this->connection->getDatabasePlatform()->getName() !== 'sqlite') {
$this->markTestSkipped('Only fails this way on sqlite'); $this->markTestSkipped('Only fails this way on sqlite');
} }
// mode 0 is considered read-only on Windows
$mode = PHP_OS === 'Linux' ? 0444 : 0000;
$filename = sprintf('%s/%s', sys_get_temp_dir(), 'doctrine_failed_connection_' . $mode . '.db'); $filename = sprintf('%s/%s', sys_get_temp_dir(), 'doctrine_failed_connection_' . $mode . '.db');
if (file_exists($filename)) { if (file_exists($filename)) {
chmod($filename, 0200); // make the file writable again, so it can be removed on Windows $this->cleanupReadOnlyFile($filename);
unlink($filename);
} }
touch($filename); touch($filename);
chmod($filename, $mode); chmod($filename, $mode);
if ($this->isLinuxRoot()) {
exec(sprintf('chattr +i %s', $filename));
}
$params = [ $params = [
'driver' => 'pdo_sqlite', 'driver' => 'pdo_sqlite',
'path' => $filename, 'path' => $filename,
...@@ -313,19 +319,21 @@ class ExceptionTest extends DbalFunctionalTestCase ...@@ -313,19 +319,21 @@ class ExceptionTest extends DbalFunctionalTestCase
$table = $schema->createTable('no_connection'); $table = $schema->createTable('no_connection');
$table->addColumn('id', 'integer'); $table->addColumn('id', 'integer');
$this->expectException($exceptionClass); $this->expectException(Exception\ReadOnlyException::class);
foreach ($schema->toSql($conn->getDatabasePlatform()) as $sql) { $this->expectExceptionMessage(<<<EOT
$conn->exec($sql); An exception occurred while executing 'CREATE TABLE no_connection (id INTEGER NOT NULL)':
}
}
public function getSqLiteOpenConnection() SQLSTATE[HY000]: General error: 8 attempt to write a readonly database
{ EOT
return [ );
// mode 0 is considered read-only on Windows
[0000, defined('PHP_WINDOWS_VERSION_BUILD') ? Exception\ReadOnlyException::class : Exception\ConnectionException::class], try {
[0444, Exception\ReadOnlyException::class], foreach ($schema->toSql($conn->getDatabasePlatform()) as $sql) {
]; $conn->exec($sql);
}
} finally {
$this->cleanupReadOnlyFile($filename);
}
} }
/** /**
...@@ -395,4 +403,19 @@ class ExceptionTest extends DbalFunctionalTestCase ...@@ -395,4 +403,19 @@ class ExceptionTest extends DbalFunctionalTestCase
$schemaManager->dropTable('owning_table'); $schemaManager->dropTable('owning_table');
$schemaManager->dropTable('constraint_error_table'); $schemaManager->dropTable('constraint_error_table');
} }
private function isLinuxRoot() : bool
{
return PHP_OS === 'Linux' && posix_getpwuid(posix_geteuid())['name'] === 'root';
}
private function cleanupReadOnlyFile(string $filename) : void
{
if ($this->isLinuxRoot()) {
exec(sprintf('chattr -i %s', $filename));
}
chmod($filename, 0200); // make the file writable again, so it can be removed on Windows
unlink($filename);
}
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment