FIX Add Nested DB transaction support (#7848)

* TEST Prove nested transactions break

* Add nested transaction support
This commit is contained in:
Daniel Hensby 2018-02-08 21:28:32 +00:00 committed by Damian Mooyman
parent 09f88c76cf
commit d3278d5470
2 changed files with 95 additions and 45 deletions

View File

@ -57,6 +57,11 @@ class MySQLDatabase extends Database
*/ */
private static $collation = 'utf8_general_ci'; private static $collation = 'utf8_general_ci';
/**
* @var bool
*/
protected $transactionNesting = 0;
public function connect($parameters) public function connect($parameters)
{ {
// Ensure that driver is available (required by PDO) // Ensure that driver is available (required by PDO)
@ -300,16 +305,21 @@ class MySQLDatabase extends Database
public function transactionStart($transactionMode = false, $sessionCharacteristics = false) public function transactionStart($transactionMode = false, $sessionCharacteristics = false)
{ {
// This sets the isolation level for the NEXT transaction, not the current one. if ($this->transactionNesting > 0) {
if ($transactionMode) { $this->transactionSavepoint('NESTEDTRANSACTION' . $this->transactionNesting);
$this->query('SET TRANSACTION ' . $transactionMode); } else {
} // This sets the isolation level for the NEXT transaction, not the current one.
if ($transactionMode) {
$this->query('SET TRANSACTION ' . $transactionMode);
}
$this->query('START TRANSACTION'); $this->query('START TRANSACTION');
if ($sessionCharacteristics) { if ($sessionCharacteristics) {
$this->query('SET SESSION TRANSACTION ' . $sessionCharacteristics); $this->query('SET SESSION TRANSACTION ' . $sessionCharacteristics);
}
} }
++$this->transactionNesting;
} }
public function transactionSavepoint($savepoint) public function transactionSavepoint($savepoint)
@ -322,13 +332,22 @@ class MySQLDatabase extends Database
if ($savepoint) { if ($savepoint) {
$this->query('ROLLBACK TO ' . $savepoint); $this->query('ROLLBACK TO ' . $savepoint);
} else { } else {
$this->query('ROLLBACK'); --$this->transactionNesting;
if ($this->transactionNesting > 0) {
$this->transactionRollback('NESTEDTRANSACTION' . $this->transactionNesting);
} else {
$this->query('ROLLBACK');
}
} }
} }
public function transactionEnd($chain = false) public function transactionEnd($chain = false)
{ {
$this->query('COMMIT AND ' . ($chain ? '' : 'NO ') . 'CHAIN'); --$this->transactionNesting;
if ($this->transactionNesting <= 0) {
$this->transactionNesting = 0;
$this->query('COMMIT AND ' . ($chain ? '' : 'NO ') . 'CHAIN');
}
} }
public function comparisonClause( public function comparisonClause(

View File

@ -5,57 +5,88 @@ namespace SilverStripe\ORM\Tests;
use SilverStripe\ORM\DB; use SilverStripe\ORM\DB;
use SilverStripe\ORM\DataObject; use SilverStripe\ORM\DataObject;
use SilverStripe\Dev\SapphireTest; use SilverStripe\Dev\SapphireTest;
use SilverStripe\ORM\Tests\TransactionTest\TestObject;
class TransactionTest extends SapphireTest class TransactionTest extends SapphireTest
{ {
protected $usesDatabase = true;
protected static $extra_dataobjects = array( protected static $extra_dataobjects = [
TransactionTest\TestObject::class TransactionTest\TestObject::class,
); ];
public static function setUpBeforeClass()
{
parent::setUpBeforeClass();
if (!DB::get_conn()->supportsTransactions()) {
static::markTestSkipped('Current database does not support transactions');
}
}
public function testNestedTransaction()
{
$this->assertCount(0, TestObject::get());
try {
DB::get_conn()->withTransaction(function () {
$obj = TransactionTest\TestObject::create();
$obj->Title = 'Test';
$obj->write();
$this->assertCount(1, TestObject::get());
DB::get_conn()->withTransaction(function () {
$obj = TransactionTest\TestObject::create();
$obj->Title = 'Test2';
$obj->write();
$this->assertCount(2, TestObject::get());
});
throw new \Exception('roll back transaction');
});
} catch (\Exception $e) {
$this->assertEquals('roll back transaction', $e->getMessage());
}
$this->assertCount(0, TestObject::get());
}
public function testCreateWithTransaction() public function testCreateWithTransaction()
{ {
DB::get_conn()->transactionStart();
$obj = new TransactionTest\TestObject();
$obj->Title = 'First page';
$obj->write();
if (DB::get_conn()->supportsTransactions()==true) { $obj = new TransactionTest\TestObject();
DB::get_conn()->transactionStart(); $obj->Title = 'Second page';
$obj=new TransactionTest\TestObject(); $obj->write();
$obj->Title='First page';
$obj->write();
$obj=new TransactionTest\TestObject(); //Create a savepoint here:
$obj->Title='Second page'; DB::get_conn()->transactionSavepoint('rollback');
$obj->write();
//Create a savepoint here: $obj = new TransactionTest\TestObject();
DB::get_conn()->transactionSavepoint('rollback'); $obj->Title = 'Third page';
$obj->write();
$obj=new TransactionTest\TestObject(); $obj = new TransactionTest\TestObject();
$obj->Title='Third page'; $obj->Title = 'Fourth page';
$obj->write(); $obj->write();
$obj=new TransactionTest\TestObject(); //Revert to a savepoint:
$obj->Title='Fourth page'; DB::get_conn()->transactionRollback('rollback');
$obj->write();
//Revert to a savepoint: DB::get_conn()->transactionEnd();
DB::get_conn()->transactionRollback('rollback');
DB::get_conn()->transactionEnd(); $first = DataObject::get(TransactionTest\TestObject::class, "\"Title\"='First page'");
$second = DataObject::get(TransactionTest\TestObject::class, "\"Title\"='Second page'");
$third = DataObject::get(TransactionTest\TestObject::class, "\"Title\"='Third page'");
$fourth = DataObject::get(TransactionTest\TestObject::class, "\"Title\"='Fourth page'");
$first=DataObject::get(TransactionTest\TestObject::class, "\"Title\"='First page'"); //These pages should be in the system
$second=DataObject::get(TransactionTest\TestObject::class, "\"Title\"='Second page'"); $this->assertTrue(is_object($first) && $first->exists());
$third=DataObject::get(TransactionTest\TestObject::class, "\"Title\"='Third page'"); $this->assertTrue(is_object($second) && $second->exists());
$fourth=DataObject::get(TransactionTest\TestObject::class, "\"Title\"='Fourth page'");
//These pages should be in the system //These pages should NOT exist, we reverted to a savepoint:
$this->assertTrue(is_object($first) && $first->exists()); $this->assertFalse(is_object($third) && $third->exists());
$this->assertTrue(is_object($second) && $second->exists()); $this->assertFalse(is_object($fourth) && $fourth->exists());
//These pages should NOT exist, we reverted to a savepoint:
$this->assertFalse(is_object($third) && $third->exists());
$this->assertFalse(is_object($fourth) && $fourth->exists());
} else {
$this->markTestSkipped('Current database does not support transactions');
}
} }
} }