From 6b3959ba482544b57dc5d6f9e0fbddd7f6539100 Mon Sep 17 00:00:00 2001 From: Daniel Hensby Date: Thu, 8 Feb 2018 21:37:59 +0000 Subject: [PATCH] FIX Support nested transactions --- code/MSSQLDatabase.php | 33 +++++++++++++++++++++++++-------- 1 file changed, 25 insertions(+), 8 deletions(-) diff --git a/code/MSSQLDatabase.php b/code/MSSQLDatabase.php index 02f3e92..6bf0453 100644 --- a/code/MSSQLDatabase.php +++ b/code/MSSQLDatabase.php @@ -78,6 +78,11 @@ class MSSQLDatabase extends Database */ protected $fullTextEnabled = null; + /** + * @var bool + */ + protected $transactionNesting = 0; + /** * Set the default collation of the MSSQL nvarchar fields that we create. * We don't apply this to the database as a whole, so that we can use unicode collations. @@ -453,11 +458,14 @@ class MSSQLDatabase extends Database */ public function transactionStart($transactionMode = false, $sessionCharacteristics = false) { - if ($this->connector instanceof SQLServerConnector) { + if ($this->transactionNesting > 0) { + $this->transactionSavepoint('NESTEDTRANSACTION' . $this->transactionNesting); + } elseif ($this->connector instanceof SQLServerConnector) { $this->connector->transactionStart(); } else { $this->query('BEGIN TRANSACTION'); } + ++$this->transactionNesting; } public function transactionSavepoint($savepoint) @@ -469,19 +477,28 @@ class MSSQLDatabase extends Database { if ($savepoint) { $this->query("ROLLBACK TRANSACTION \"$savepoint\""); - } elseif ($this->connector instanceof SQLServerConnector) { - $this->connector->transactionRollback(); } else { - $this->query('ROLLBACK TRANSACTION'); + --$this->transactionNesting; + if ($this->transactionNesting > 0) { + $this->transactionRollback('NESTEDTRANSACTION' . $this->transactionNesting); + } elseif ($this->connector instanceof SQLServerConnector) { + $this->connector->transactionRollback(); + } else { + $this->query('ROLLBACK TRANSACTION'); + } } } public function transactionEnd($chain = false) { - if ($this->connector instanceof SQLServerConnector) { - $this->connector->transactionEnd(); - } else { - $this->query('COMMIT TRANSACTION'); + --$this->transactionNesting; + if ($this->transactionNesting <= 0) { + $this->transactionNesting = 0; + if ($this->connector instanceof SQLServerConnector) { + $this->connector->transactionEnd(); + } else { + $this->query('COMMIT TRANSACTION'); + } } }