-
-
Save datacow-com/2729873 to your computer and use it in GitHub Desktop.
<?php | |
// Snippet for Symfony 2 application that uses Doctrine 2 to handle transactions | |
// It uses the names of the objects/doctrine repositories from the Beta 4 Manual of Symfony 2. | |
// Get the entity manager | |
$em = $this->getDoctrine()->getEntityManager(); | |
// suspend auto-commit | |
$em->getConnection()->beginTransaction(); | |
// Try and make the transaction | |
try { | |
// Get the account | |
$account = $this->getDoctrine()->getRepository('AcmeStoreBundle:Account')->find($id); | |
// Lower balance | |
$account->setBalance($account->getBalance() - 100); | |
// Save account | |
$em->persist($account); | |
$em->flush(); | |
// Try and commit the transaction | |
$em->getConnection()->commit(); | |
} catch (Exception $e) { | |
// Rollback the failed transaction attempt | |
$em->getConnection()->rollback(); | |
throw $e; | |
} |
Note: If you are reusing the $em
after the rollback (for example to log the exception), the $account
entity will still have the renewed balance, and calling flush will flush the new value to the database.
To work around this, call $em->clear()
to detach all entities after the rollback. See this stackOverflow question for details.
To make absolutely sure that whatever your process is trying to do is done on most recent database data, you might want to include version field into Doctrine Entity definition.
From your link:
Database transactions are fine for concurrency control during a single request. However, a database transaction should not span across requests, the so-called “user think time”. Therefore a long-running “business transaction” that spans multiple requests needs to involve several database transactions. Thus, database transactions alone can no longer control concurrency during such a long-running business transaction. Concurrency control becomes the partial responsibility of the application itself.
So, IMHO, it doesn't have anything to do with "being absolutely sure that whatever your process is trying to do is done on most recent database data", it only depends on if you need transactions on multiple requests or not, right?
In high concurrency environment, two processes running at the same time could produce a balance decreased by 100 instead of expected 200. It would be very bad luck if both threads would have executed $account->getBalance() before any of them had executed $account->setBalance(), but it happens. Beginning a transaction ensures consistent reads, but in this example one should take a write lock on $account before reading and decreasing the balance.