mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Added some assertions to enforce the atomicity of atomic.
This commit is contained in:
		| @@ -70,6 +70,7 @@ signals.request_started.connect(reset_queries) | |||||||
| # their lifetime. NB: abort() doesn't do anything outside of a transaction. | # their lifetime. NB: abort() doesn't do anything outside of a transaction. | ||||||
| def close_old_connections(**kwargs): | def close_old_connections(**kwargs): | ||||||
|     for conn in connections.all(): |     for conn in connections.all(): | ||||||
|  |         # Remove this when the legacy transaction management goes away. | ||||||
|         try: |         try: | ||||||
|             conn.abort() |             conn.abort() | ||||||
|         except DatabaseError: |         except DatabaseError: | ||||||
|   | |||||||
| @@ -157,6 +157,7 @@ class BaseDatabaseWrapper(object): | |||||||
|         Commits a transaction and resets the dirty flag. |         Commits a transaction and resets the dirty flag. | ||||||
|         """ |         """ | ||||||
|         self.validate_thread_sharing() |         self.validate_thread_sharing() | ||||||
|  |         self.validate_no_atomic_block() | ||||||
|         self._commit() |         self._commit() | ||||||
|         self.set_clean() |         self.set_clean() | ||||||
|  |  | ||||||
| @@ -165,6 +166,7 @@ class BaseDatabaseWrapper(object): | |||||||
|         Rolls back a transaction and resets the dirty flag. |         Rolls back a transaction and resets the dirty flag. | ||||||
|         """ |         """ | ||||||
|         self.validate_thread_sharing() |         self.validate_thread_sharing() | ||||||
|  |         self.validate_no_atomic_block() | ||||||
|         self._rollback() |         self._rollback() | ||||||
|         self.set_clean() |         self.set_clean() | ||||||
|  |  | ||||||
| @@ -265,6 +267,8 @@ class BaseDatabaseWrapper(object): | |||||||
|         If you switch off transaction management and there is a pending |         If you switch off transaction management and there is a pending | ||||||
|         commit/rollback, the data will be commited, unless "forced" is True. |         commit/rollback, the data will be commited, unless "forced" is True. | ||||||
|         """ |         """ | ||||||
|  |         self.validate_no_atomic_block() | ||||||
|  |  | ||||||
|         self.transaction_state.append(managed) |         self.transaction_state.append(managed) | ||||||
|  |  | ||||||
|         if not managed and self.is_dirty() and not forced: |         if not managed and self.is_dirty() and not forced: | ||||||
| @@ -280,6 +284,8 @@ class BaseDatabaseWrapper(object): | |||||||
|         over to the surrounding block, as a commit will commit all changes, even |         over to the surrounding block, as a commit will commit all changes, even | ||||||
|         those from outside. (Commits are on connection level.) |         those from outside. (Commits are on connection level.) | ||||||
|         """ |         """ | ||||||
|  |         self.validate_no_atomic_block() | ||||||
|  |  | ||||||
|         if self.transaction_state: |         if self.transaction_state: | ||||||
|             del self.transaction_state[-1] |             del self.transaction_state[-1] | ||||||
|         else: |         else: | ||||||
| @@ -305,10 +311,19 @@ class BaseDatabaseWrapper(object): | |||||||
|         """ |         """ | ||||||
|         Enable or disable autocommit. |         Enable or disable autocommit. | ||||||
|         """ |         """ | ||||||
|  |         self.validate_no_atomic_block() | ||||||
|         self.ensure_connection() |         self.ensure_connection() | ||||||
|         self._set_autocommit(autocommit) |         self._set_autocommit(autocommit) | ||||||
|         self.autocommit = autocommit |         self.autocommit = autocommit | ||||||
|  |  | ||||||
|  |     def validate_no_atomic_block(self): | ||||||
|  |         """ | ||||||
|  |         Raise an error if an atomic block is active. | ||||||
|  |         """ | ||||||
|  |         if self.in_atomic_block: | ||||||
|  |             raise TransactionManagementError( | ||||||
|  |                 "This is forbidden when an 'atomic' block is active.") | ||||||
|  |  | ||||||
|     def abort(self): |     def abort(self): | ||||||
|         """ |         """ | ||||||
|         Roll back any ongoing transaction and clean the transaction state |         Roll back any ongoing transaction and clean the transaction state | ||||||
|   | |||||||
| @@ -367,6 +367,9 @@ def autocommit(using=None): | |||||||
|     this decorator is useful if you globally activated transaction management in |     this decorator is useful if you globally activated transaction management in | ||||||
|     your settings file and want the default behavior in some view functions. |     your settings file and want the default behavior in some view functions. | ||||||
|     """ |     """ | ||||||
|  |     warnings.warn("autocommit is deprecated in favor of set_autocommit.", | ||||||
|  |         PendingDeprecationWarning, stacklevel=2) | ||||||
|  |  | ||||||
|     def entering(using): |     def entering(using): | ||||||
|         enter_transaction_management(managed=False, using=using) |         enter_transaction_management(managed=False, using=using) | ||||||
|  |  | ||||||
| @@ -382,6 +385,9 @@ def commit_on_success(using=None): | |||||||
|     a rollback is made. This is one of the most common ways to do transaction |     a rollback is made. This is one of the most common ways to do transaction | ||||||
|     control in Web apps. |     control in Web apps. | ||||||
|     """ |     """ | ||||||
|  |     warnings.warn("commit_on_success is deprecated in favor of atomic.", | ||||||
|  |         PendingDeprecationWarning, stacklevel=2) | ||||||
|  |  | ||||||
|     def entering(using): |     def entering(using): | ||||||
|         enter_transaction_management(using=using) |         enter_transaction_management(using=using) | ||||||
|  |  | ||||||
| @@ -409,6 +415,9 @@ def commit_manually(using=None): | |||||||
|     own -- it's up to the user to call the commit and rollback functions |     own -- it's up to the user to call the commit and rollback functions | ||||||
|     themselves. |     themselves. | ||||||
|     """ |     """ | ||||||
|  |     warnings.warn("commit_manually is deprecated in favor of set_autocommit.", | ||||||
|  |         PendingDeprecationWarning, stacklevel=2) | ||||||
|  |  | ||||||
|     def entering(using): |     def entering(using): | ||||||
|         enter_transaction_management(using=using) |         enter_transaction_management(using=using) | ||||||
|  |  | ||||||
| @@ -420,10 +429,15 @@ def commit_manually(using=None): | |||||||
| def commit_on_success_unless_managed(using=None): | def commit_on_success_unless_managed(using=None): | ||||||
|     """ |     """ | ||||||
|     Transitory API to preserve backwards-compatibility while refactoring. |     Transitory API to preserve backwards-compatibility while refactoring. | ||||||
|  |  | ||||||
|  |     Once the legacy transaction management is fully deprecated, this should | ||||||
|  |     simply be replaced by atomic. Until then, it's necessary to avoid making a | ||||||
|  |     commit where Django didn't use to, since entering atomic in managed mode | ||||||
|  |     triggers a commmit. | ||||||
|     """ |     """ | ||||||
|     connection = get_connection(using) |     connection = get_connection(using) | ||||||
|     if connection.autocommit and not connection.in_atomic_block: |     if connection.autocommit or connection.in_atomic_block: | ||||||
|         return commit_on_success(using) |         return atomic(using) | ||||||
|     else: |     else: | ||||||
|         def entering(using): |         def entering(using): | ||||||
|             pass |             pass | ||||||
|   | |||||||
| @@ -329,6 +329,10 @@ these changes. | |||||||
| 1.8 | 1.8 | ||||||
| --- | --- | ||||||
|  |  | ||||||
|  | * The decorators and context managers ``django.db.transaction.autocommit``, | ||||||
|  |   ``commit_on_success`` and ``commit_manually`` will be removed. See | ||||||
|  |   :ref:`transactions-upgrading-from-1.5`. | ||||||
|  |  | ||||||
| * The :ttag:`cycle` and :ttag:`firstof` template tags will auto-escape their | * The :ttag:`cycle` and :ttag:`firstof` template tags will auto-escape their | ||||||
|   arguments. In 1.6 and 1.7, this behavior is provided by the version of these |   arguments. In 1.6 and 1.7, this behavior is provided by the version of these | ||||||
|   tags in the ``future`` template tag library. |   tags in the ``future`` template tag library. | ||||||
|   | |||||||
| @@ -105,16 +105,14 @@ you just won't get any of the nice new unittest2 features. | |||||||
| Transaction context managers | Transaction context managers | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| Users of Python 2.5 and above may now use :ref:`transaction management functions | Users of Python 2.5 and above may now use transaction management functions as | ||||||
| <transaction-management-functions>` as `context managers`_. For example:: | `context managers`_. For example:: | ||||||
|  |  | ||||||
|     with transaction.autocommit(): |     with transaction.autocommit(): | ||||||
|         # ... |         # ... | ||||||
|  |  | ||||||
| .. _context managers: http://docs.python.org/glossary.html#term-context-manager | .. _context managers: http://docs.python.org/glossary.html#term-context-manager | ||||||
|  |  | ||||||
| For more information, see :ref:`transaction-management-functions`. |  | ||||||
|  |  | ||||||
| Configurable delete-cascade | Configurable delete-cascade | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -148,16 +148,14 @@ you just won't get any of the nice new unittest2 features. | |||||||
| Transaction context managers | Transaction context managers | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| Users of Python 2.5 and above may now use :ref:`transaction management functions | Users of Python 2.5 and above may now use transaction management functions as | ||||||
| <transaction-management-functions>` as `context managers`_. For example:: | `context managers`_. For example:: | ||||||
|  |  | ||||||
|     with transaction.autocommit(): |     with transaction.autocommit(): | ||||||
|         # ... |         # ... | ||||||
|  |  | ||||||
| .. _context managers: http://docs.python.org/glossary.html#term-context-manager | .. _context managers: http://docs.python.org/glossary.html#term-context-manager | ||||||
|  |  | ||||||
| For more information, see :ref:`transaction-management-functions`. |  | ||||||
|  |  | ||||||
| Configurable delete-cascade | Configurable delete-cascade | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ should improve performance. The existing APIs were deprecated, and new APIs | |||||||
| were introduced, as described in :doc:`/topics/db/transactions`. | were introduced, as described in :doc:`/topics/db/transactions`. | ||||||
|  |  | ||||||
| Please review carefully the list of :ref:`known backwards-incompatibilities | Please review carefully the list of :ref:`known backwards-incompatibilities | ||||||
| <transactions-changes-from-1.5>` to determine if you need to make changes in | <transactions-upgrading-from-1.5>` to determine if you need to make changes in | ||||||
| your code. | your code. | ||||||
|  |  | ||||||
| Persistent database connections | Persistent database connections | ||||||
| @@ -163,7 +163,7 @@ Backwards incompatible changes in 1.6 | |||||||
| * Database-level autocommit is enabled by default in Django 1.6. While this | * Database-level autocommit is enabled by default in Django 1.6. While this | ||||||
|   doesn't change the general spirit of Django's transaction management, there |   doesn't change the general spirit of Django's transaction management, there | ||||||
|   are a few known backwards-incompatibities, described in the :ref:`transaction |   are a few known backwards-incompatibities, described in the :ref:`transaction | ||||||
|   management docs <transactions-changes-from-1.5>`. You should review your code |   management docs <transactions-upgrading-from-1.5>`. You should review your code | ||||||
|   to determine if you're affected. |   to determine if you're affected. | ||||||
|  |  | ||||||
| * In previous versions, database-level autocommit was only an option for | * In previous versions, database-level autocommit was only an option for | ||||||
| @@ -256,6 +256,19 @@ Backwards incompatible changes in 1.6 | |||||||
| Features deprecated in 1.6 | Features deprecated in 1.6 | ||||||
| ========================== | ========================== | ||||||
|  |  | ||||||
|  | Transaction management APIs | ||||||
|  | ~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|  | Transaction management was completely overhauled in Django 1.6, and the | ||||||
|  | current APIs are deprecated: | ||||||
|  |  | ||||||
|  | - :func:`django.db.transaction.autocommit` | ||||||
|  | - :func:`django.db.transaction.commit_on_success` | ||||||
|  | - :func:`django.db.transaction.commit_manually` | ||||||
|  |  | ||||||
|  | The reasons for this change and the upgrade path are described in the | ||||||
|  | :ref:`transactions documentation <transactions-upgrading-from-1.5>`. | ||||||
|  |  | ||||||
| Changes to :ttag:`cycle` and :ttag:`firstof` | Changes to :ttag:`cycle` and :ttag:`firstof` | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ immediately committed to the database. :ref:`See below for details | |||||||
|  |  | ||||||
| .. versionchanged:: 1.6 | .. versionchanged:: 1.6 | ||||||
|     Previous version of Django featured :ref:`a more complicated default |     Previous version of Django featured :ref:`a more complicated default | ||||||
|     behavior <transactions-changes-from-1.5>`. |     behavior <transactions-upgrading-from-1.5>`. | ||||||
|  |  | ||||||
| Tying transactions to HTTP requests | Tying transactions to HTTP requests | ||||||
| ----------------------------------- | ----------------------------------- | ||||||
| @@ -89,7 +89,7 @@ Django provides a single API to control database transactions. | |||||||
|     database. If this argument isn't provided, Django uses the ``"default"`` |     database. If this argument isn't provided, Django uses the ``"default"`` | ||||||
|     database. |     database. | ||||||
|  |  | ||||||
|     ``atomic`` is usable both as a decorator:: |     ``atomic`` is usable both as a `decorator`_:: | ||||||
|  |  | ||||||
|         from django.db import transaction |         from django.db import transaction | ||||||
|  |  | ||||||
| @@ -98,7 +98,7 @@ Django provides a single API to control database transactions. | |||||||
|             # This code executes inside a transaction. |             # This code executes inside a transaction. | ||||||
|             do_stuff() |             do_stuff() | ||||||
|  |  | ||||||
|     and as a context manager:: |     and as a `context manager`_:: | ||||||
|  |  | ||||||
|         from django.db import transaction |         from django.db import transaction | ||||||
|  |  | ||||||
| @@ -110,6 +110,9 @@ Django provides a single API to control database transactions. | |||||||
|                 # This code executes inside a transaction. |                 # This code executes inside a transaction. | ||||||
|                 do_more_stuff() |                 do_more_stuff() | ||||||
|  |  | ||||||
|  |     .. _decorator: http://docs.python.org/glossary.html#term-decorator | ||||||
|  |     .. _context manager: http://docs.python.org/glossary.html#term-context-manager | ||||||
|  |  | ||||||
|     Wrapping ``atomic`` in a try/except block allows for natural handling of |     Wrapping ``atomic`` in a try/except block allows for natural handling of | ||||||
|     integrity errors:: |     integrity errors:: | ||||||
|  |  | ||||||
| @@ -145,189 +148,6 @@ Django provides a single API to control database transactions. | |||||||
|     - releases or rolls back to the savepoint when exiting an inner block; |     - releases or rolls back to the savepoint when exiting an inner block; | ||||||
|     - commits or rolls back the transaction when exiting the outermost block. |     - commits or rolls back the transaction when exiting the outermost block. | ||||||
|  |  | ||||||
| .. _transaction-management-functions: |  | ||||||
|  |  | ||||||
| Controlling transaction management in views |  | ||||||
| =========================================== |  | ||||||
|  |  | ||||||
| For most people, implicit request-based transactions work wonderfully. However, |  | ||||||
| if you need more fine-grained control over how transactions are managed, you can |  | ||||||
| use a set of functions in ``django.db.transaction`` to control transactions on a |  | ||||||
| per-function or per-code-block basis. |  | ||||||
|  |  | ||||||
| These functions, described in detail below, can be used in two different ways: |  | ||||||
|  |  | ||||||
| * As a decorator_ on a particular function. For example:: |  | ||||||
|  |  | ||||||
|     from django.db import transaction |  | ||||||
|  |  | ||||||
|     @transaction.commit_on_success |  | ||||||
|     def viewfunc(request): |  | ||||||
|         # ... |  | ||||||
|         # this code executes inside a transaction |  | ||||||
|         # ... |  | ||||||
|  |  | ||||||
| * As a `context manager`_ around a particular block of code:: |  | ||||||
|  |  | ||||||
|     from django.db import transaction |  | ||||||
|  |  | ||||||
|     def viewfunc(request): |  | ||||||
|         # ... |  | ||||||
|         # this code executes using default transaction management |  | ||||||
|         # ... |  | ||||||
|  |  | ||||||
|         with transaction.commit_on_success(): |  | ||||||
|             # ... |  | ||||||
|             # this code executes inside a transaction |  | ||||||
|             # ... |  | ||||||
|  |  | ||||||
| Both techniques work with all supported version of Python. |  | ||||||
|  |  | ||||||
| .. _decorator: http://docs.python.org/glossary.html#term-decorator |  | ||||||
| .. _context manager: http://docs.python.org/glossary.html#term-context-manager |  | ||||||
|  |  | ||||||
| For maximum compatibility, all of the examples below show transactions using the |  | ||||||
| decorator syntax, but all of the follow functions may be used as context |  | ||||||
| managers, too. |  | ||||||
|  |  | ||||||
| .. note:: |  | ||||||
|  |  | ||||||
|     Although the examples below use view functions as examples, these |  | ||||||
|     decorators and context managers can be used anywhere in your code |  | ||||||
|     that you need to deal with transactions. |  | ||||||
|  |  | ||||||
| .. _topics-db-transactions-autocommit: |  | ||||||
|  |  | ||||||
| .. function:: autocommit |  | ||||||
|  |  | ||||||
|     Use the ``autocommit`` decorator to switch a view function to Django's |  | ||||||
|     default commit behavior. |  | ||||||
|  |  | ||||||
|     Example:: |  | ||||||
|  |  | ||||||
|         from django.db import transaction |  | ||||||
|  |  | ||||||
|         @transaction.autocommit |  | ||||||
|         def viewfunc(request): |  | ||||||
|             .... |  | ||||||
|  |  | ||||||
|         @transaction.autocommit(using="my_other_database") |  | ||||||
|         def viewfunc2(request): |  | ||||||
|             .... |  | ||||||
|  |  | ||||||
|     Within ``viewfunc()``, transactions will be committed as soon as you call |  | ||||||
|     ``model.save()``, ``model.delete()``, or any other function that writes to |  | ||||||
|     the database.  ``viewfunc2()`` will have this same behavior, but for the |  | ||||||
|     ``"my_other_database"`` connection. |  | ||||||
|  |  | ||||||
| .. function:: commit_on_success |  | ||||||
|  |  | ||||||
|     Use the ``commit_on_success`` decorator to use a single transaction for all |  | ||||||
|     the work done in a function:: |  | ||||||
|  |  | ||||||
|         from django.db import transaction |  | ||||||
|  |  | ||||||
|         @transaction.commit_on_success |  | ||||||
|         def viewfunc(request): |  | ||||||
|             .... |  | ||||||
|  |  | ||||||
|         @transaction.commit_on_success(using="my_other_database") |  | ||||||
|         def viewfunc2(request): |  | ||||||
|             .... |  | ||||||
|  |  | ||||||
|     If the function returns successfully, then Django will commit all work done |  | ||||||
|     within the function at that point. If the function raises an exception, |  | ||||||
|     though, Django will roll back the transaction. |  | ||||||
|  |  | ||||||
| .. function:: commit_manually |  | ||||||
|  |  | ||||||
|     Use the ``commit_manually`` decorator if you need full control over |  | ||||||
|     transactions. It tells Django you'll be managing the transaction on your |  | ||||||
|     own. |  | ||||||
|  |  | ||||||
|     Whether you are writing or simply reading from the database, you must |  | ||||||
|     ``commit()`` or ``rollback()`` explicitly or Django will raise a |  | ||||||
|     :exc:`TransactionManagementError` exception. This is required when reading |  | ||||||
|     from the database because ``SELECT`` statements may call functions which |  | ||||||
|     modify tables, and thus it is impossible to know if any data has been |  | ||||||
|     modified. |  | ||||||
|  |  | ||||||
|     Manual transaction management looks like this:: |  | ||||||
|  |  | ||||||
|         from django.db import transaction |  | ||||||
|  |  | ||||||
|         @transaction.commit_manually |  | ||||||
|         def viewfunc(request): |  | ||||||
|             ... |  | ||||||
|             # You can commit/rollback however and whenever you want |  | ||||||
|             transaction.commit() |  | ||||||
|             ... |  | ||||||
|  |  | ||||||
|             # But you've got to remember to do it yourself! |  | ||||||
|             try: |  | ||||||
|                 ... |  | ||||||
|             except: |  | ||||||
|                 transaction.rollback() |  | ||||||
|             else: |  | ||||||
|                 transaction.commit() |  | ||||||
|  |  | ||||||
|         @transaction.commit_manually(using="my_other_database") |  | ||||||
|         def viewfunc2(request): |  | ||||||
|             .... |  | ||||||
|  |  | ||||||
| .. _topics-db-transactions-requirements: |  | ||||||
|  |  | ||||||
| Requirements for transaction handling |  | ||||||
| ===================================== |  | ||||||
|  |  | ||||||
| Django requires that every transaction that is opened is closed before the |  | ||||||
| completion of a request. |  | ||||||
|  |  | ||||||
| If you are using :func:`autocommit` (the default commit mode) or |  | ||||||
| :func:`commit_on_success`, this will be done for you automatically. However, |  | ||||||
| if you are manually managing transactions (using the :func:`commit_manually` |  | ||||||
| decorator), you must ensure that the transaction is either committed or rolled |  | ||||||
| back before a request is completed. |  | ||||||
|  |  | ||||||
| This applies to all database operations, not just write operations. Even |  | ||||||
| if your transaction only reads from the database, the transaction must |  | ||||||
| be committed or rolled back before you complete a request. |  | ||||||
|  |  | ||||||
| .. _managing-autocommit: |  | ||||||
|  |  | ||||||
| Managing autocommit |  | ||||||
| =================== |  | ||||||
|  |  | ||||||
| .. versionadded:: 1.6 |  | ||||||
|  |  | ||||||
| Django provides a straightforward API to manage the autocommit state of each |  | ||||||
| database connection, if you need to. |  | ||||||
|  |  | ||||||
| .. function:: get_autocommit(using=None) |  | ||||||
|  |  | ||||||
| .. function:: set_autocommit(using=None, autocommit=True) |  | ||||||
|  |  | ||||||
| These functions take a ``using`` argument which should be the name of a |  | ||||||
| database. If it isn't provided, Django uses the ``"default"`` database. |  | ||||||
|  |  | ||||||
| .. _deactivate-transaction-management: |  | ||||||
|  |  | ||||||
| How to globally deactivate transaction management |  | ||||||
| ================================================= |  | ||||||
|  |  | ||||||
| Control freaks can totally disable all transaction management by setting |  | ||||||
| :setting:`TRANSACTIONS_MANAGED` to ``True`` in the Django settings file. If |  | ||||||
| you do this, Django won't enable autocommit. You'll get the regular behavior |  | ||||||
| of the underlying database library. |  | ||||||
|  |  | ||||||
| This requires you to commit explicitly every transaction, even those started |  | ||||||
| by Django or by third-party libraries. Thus, this is best used in situations |  | ||||||
| where you want to run your own transaction-controlling middleware or do |  | ||||||
| something really strange. |  | ||||||
|  |  | ||||||
| In almost all situations, you'll be better off using the default behavior, or |  | ||||||
| the transaction middleware, and only modify selected functions as needed. |  | ||||||
|  |  | ||||||
| .. _topics-db-transactions-savepoints: | .. _topics-db-transactions-savepoints: | ||||||
|  |  | ||||||
| Savepoints | Savepoints | ||||||
| @@ -339,13 +159,19 @@ available with the SQLite (≥ 3.6.8), PostgreSQL, Oracle and MySQL (when using | |||||||
| the InnoDB storage engine) backends. Other backends provide the savepoint | the InnoDB storage engine) backends. Other backends provide the savepoint | ||||||
| functions, but they're empty operations -- they don't actually do anything. | functions, but they're empty operations -- they don't actually do anything. | ||||||
|  |  | ||||||
| Savepoints aren't especially useful if you are using the default | Savepoints aren't especially useful if you are using autocommit, the default | ||||||
| ``autocommit`` behavior of Django. However, if you are using | behavior of Django. However, once you open a transaction with :func:`atomic`, | ||||||
| ``commit_on_success`` or ``commit_manually``, each open transaction will build | you build up a series of database operations awaiting a commit or rollback. If | ||||||
| up a series of database operations, awaiting a commit or rollback. If you | you issue a rollback, the entire transaction is rolled back. Savepoints | ||||||
| issue a rollback, the entire transaction is rolled back. Savepoints provide | provide the ability to perform a fine-grained rollback, rather than the full | ||||||
| the ability to perform a fine-grained rollback, rather than the full rollback | rollback that would be performed by ``transaction.rollback()``. | ||||||
| that would be performed by ``transaction.rollback()``. |  | ||||||
|  | .. versionchanged:: 1.6 | ||||||
|  |  | ||||||
|  | When the :func:`atomic` decorator is nested, it creates a savepoint to allow | ||||||
|  | partial commit or rollback. You're strongly encouraged to use :func:`atomic` | ||||||
|  | rather than the functions described below, but they're still part of the | ||||||
|  | public API, and there's no plan to deprecate them. | ||||||
|  |  | ||||||
| Each of these functions takes a ``using`` argument which should be the name of | Each of these functions takes a ``using`` argument which should be the name of | ||||||
| a database for which the behavior applies.  If no ``using`` argument is | a database for which the behavior applies.  If no ``using`` argument is | ||||||
| @@ -374,15 +200,17 @@ The following example demonstrates the use of savepoints:: | |||||||
|  |  | ||||||
|     from django.db import transaction |     from django.db import transaction | ||||||
|  |  | ||||||
|     @transaction.commit_manually |     # open a transaction | ||||||
|  |     @transaction.atomic | ||||||
|     def viewfunc(request): |     def viewfunc(request): | ||||||
|  |  | ||||||
|       a.save() |       a.save() | ||||||
|       # open transaction now contains a.save() |       # transaction now contains a.save() | ||||||
|  |  | ||||||
|       sid = transaction.savepoint() |       sid = transaction.savepoint() | ||||||
|  |  | ||||||
|       b.save() |       b.save() | ||||||
|       # open transaction now contains a.save() and b.save() |       # transaction now contains a.save() and b.save() | ||||||
|  |  | ||||||
|       if want_to_keep_b: |       if want_to_keep_b: | ||||||
|           transaction.savepoint_commit(sid) |           transaction.savepoint_commit(sid) | ||||||
| @@ -391,7 +219,82 @@ The following example demonstrates the use of savepoints:: | |||||||
|           transaction.savepoint_rollback(sid) |           transaction.savepoint_rollback(sid) | ||||||
|           # open transaction now contains only a.save() |           # open transaction now contains only a.save() | ||||||
|  |  | ||||||
|       transaction.commit() | Autocommit | ||||||
|  | ========== | ||||||
|  |  | ||||||
|  | .. _autocommit-details: | ||||||
|  |  | ||||||
|  | Why Django uses autocommit | ||||||
|  | -------------------------- | ||||||
|  |  | ||||||
|  | In the SQL standards, each SQL query starts a transaction, unless one is | ||||||
|  | already in progress. Such transactions must then be committed or rolled back. | ||||||
|  |  | ||||||
|  | This isn't always convenient for application developers. To alleviate this | ||||||
|  | problem, most databases provide an autocommit mode. When autocommit is turned | ||||||
|  | on, each SQL query is wrapped in its own transaction. In other words, the | ||||||
|  | transaction is not only automatically started, but also automatically | ||||||
|  | committed. | ||||||
|  |  | ||||||
|  | :pep:`249`, the Python Database API Specification v2.0, requires autocommit to | ||||||
|  | be initially turned off. Django overrides this default and turns autocommit | ||||||
|  | on. | ||||||
|  |  | ||||||
|  | To avoid this, you can :ref:`deactivate the transaction management | ||||||
|  | <deactivate-transaction-management>`, but it isn't recommended. | ||||||
|  |  | ||||||
|  | .. versionchanged:: 1.6 | ||||||
|  |     Before Django 1.6, autocommit was turned off, and it was emulated by | ||||||
|  |     forcing a commit after write operations in the ORM. | ||||||
|  |  | ||||||
|  | .. warning:: | ||||||
|  |  | ||||||
|  |     If you're using the database API directly — for instance, you're running | ||||||
|  |     SQL queries with ``cursor.execute()`` — be aware that autocommit is on, | ||||||
|  |     and consider wrapping your operations in a transaction, with | ||||||
|  |     :func:`atomic`, to ensure consistency. | ||||||
|  |  | ||||||
|  | .. _managing-autocommit: | ||||||
|  |  | ||||||
|  | Managing autocommit | ||||||
|  | ------------------- | ||||||
|  |  | ||||||
|  | .. versionadded:: 1.6 | ||||||
|  |  | ||||||
|  | Django provides a straightforward API to manage the autocommit state of each | ||||||
|  | database connection, if you need to. | ||||||
|  |  | ||||||
|  | .. function:: get_autocommit(using=None) | ||||||
|  |  | ||||||
|  | .. function:: set_autocommit(using=None, autocommit=True) | ||||||
|  |  | ||||||
|  | These functions take a ``using`` argument which should be the name of a | ||||||
|  | database. If it isn't provided, Django uses the ``"default"`` database. | ||||||
|  |  | ||||||
|  | Autocommit is initially turned on. If you turn it off, it's your | ||||||
|  | responsibility to restore it. | ||||||
|  |  | ||||||
|  | :func:`atomic` requires autocommit to be turned on; it will raise an exception | ||||||
|  | if autocommit is off. Django will also refuse to turn autocommit off when an | ||||||
|  | :func:`atomic` block is active, because that would break atomicity. | ||||||
|  |  | ||||||
|  | .. _deactivate-transaction-management: | ||||||
|  |  | ||||||
|  | Deactivating transaction management | ||||||
|  | ----------------------------------- | ||||||
|  |  | ||||||
|  | Control freaks can totally disable all transaction management by setting | ||||||
|  | :setting:`TRANSACTIONS_MANAGED` to ``True`` in the Django settings file. If | ||||||
|  | you do this, Django won't enable autocommit. You'll get the regular behavior | ||||||
|  | of the underlying database library. | ||||||
|  |  | ||||||
|  | This requires you to commit explicitly every transaction, even those started | ||||||
|  | by Django or by third-party libraries. Thus, this is best used in situations | ||||||
|  | where you want to run your own transaction-controlling middleware or do | ||||||
|  | something really strange. | ||||||
|  |  | ||||||
|  | In almost all situations, you'll be better off using the default behavior, or | ||||||
|  | the transaction middleware, and only modify selected functions as needed. | ||||||
|  |  | ||||||
| Database-specific notes | Database-specific notes | ||||||
| ======================= | ======================= | ||||||
| @@ -477,45 +380,57 @@ transaction. For example:: | |||||||
| In this example, ``a.save()`` will not be undone in the case where | In this example, ``a.save()`` will not be undone in the case where | ||||||
| ``b.save()`` raises an exception. | ``b.save()`` raises an exception. | ||||||
|  |  | ||||||
| Under the hood | .. _transactions-upgrading-from-1.5: | ||||||
| ============== |  | ||||||
|  |  | ||||||
| .. _autocommit-details: | Changes from Django 1.5 and earlier | ||||||
|  | =================================== | ||||||
|  |  | ||||||
| Details on autocommit | The features described below were deprecated in Django 1.6 and will be removed | ||||||
| --------------------- | in Django 1.8. They're documented in order to ease the migration to the new | ||||||
|  | transaction management APIs. | ||||||
|  |  | ||||||
| In the SQL standards, each SQL query starts a transaction, unless one is | Legacy APIs | ||||||
| already in progress. Such transactions must then be committed or rolled back. | ----------- | ||||||
|  |  | ||||||
| This isn't always convenient for application developers. To alleviate this | The following functions, defined in ``django.db.transaction``, provided a way | ||||||
| problem, most databases provide an autocommit mode. When autocommit is turned | to control transactions on a per-function or per-code-block basis. They could | ||||||
| on, each SQL query is wrapped in its own transaction. In other words, the | be used as decorators or as context managers, and they accepted a ``using`` | ||||||
| transaction is not only automatically started, but also automatically | argument, exactly like :func:`atomic`. | ||||||
| committed. |  | ||||||
|  |  | ||||||
| :pep:`249`, the Python Database API Specification v2.0, requires autocommit to | .. function:: autocommit | ||||||
| be initially turned off. Django overrides this default and turns autocommit |  | ||||||
| on. |  | ||||||
|  |  | ||||||
| To avoid this, you can :ref:`deactivate the transaction management |     Enable Django's default autocommit behavior. | ||||||
| <deactivate-transaction-management>`, but it isn't recommended. |  | ||||||
|  |  | ||||||
| .. versionchanged:: 1.6 |     Transactions will be committed as soon as you call ``model.save()``, | ||||||
|     Before Django 1.6, autocommit was turned off, and it was emulated by |     ``model.delete()``, or any other function that writes to the database. | ||||||
|     forcing a commit after write operations in the ORM. |  | ||||||
|  |  | ||||||
| .. warning:: | .. function:: commit_on_success | ||||||
|  |  | ||||||
|     If you're using the database API directly — for instance, you're running |     Use a single transaction for all the work done in a function. | ||||||
|     SQL queries with ``cursor.execute()`` — be aware that autocommit is on, |  | ||||||
|     and consider wrapping your operations in a transaction to ensure |     If the function returns successfully, then Django will commit all work done | ||||||
|     consistency. |     within the function at that point. If the function raises an exception, | ||||||
|  |     though, Django will roll back the transaction. | ||||||
|  |  | ||||||
|  | .. function:: commit_manually | ||||||
|  |  | ||||||
|  |     Tells Django you'll be managing the transaction on your own. | ||||||
|  |  | ||||||
|  |     Whether you are writing or simply reading from the database, you must | ||||||
|  |     ``commit()`` or ``rollback()`` explicitly or Django will raise a | ||||||
|  |     :exc:`TransactionManagementError` exception. This is required when reading | ||||||
|  |     from the database because ``SELECT`` statements may call functions which | ||||||
|  |     modify tables, and thus it is impossible to know if any data has been | ||||||
|  |     modified. | ||||||
|  |  | ||||||
| .. _transaction-states: | .. _transaction-states: | ||||||
|  |  | ||||||
| Transaction management states | Transaction states | ||||||
| ----------------------------- | ------------------ | ||||||
|  |  | ||||||
|  | The three functions described above relied on a concept called "transaction | ||||||
|  | states". This mechanisme was deprecated in Django 1.6, but it's still | ||||||
|  | available until Django 1.8.. | ||||||
|  |  | ||||||
| At any time, each database connection is in one of these two states: | At any time, each database connection is in one of these two states: | ||||||
|  |  | ||||||
| @@ -529,35 +444,80 @@ Django starts in auto mode. ``TransactionMiddleware``, | |||||||
| Internally, Django keeps a stack of states. Activations and deactivations must | Internally, Django keeps a stack of states. Activations and deactivations must | ||||||
| be balanced. | be balanced. | ||||||
|  |  | ||||||
| For example, at the beginning of each HTTP request, ``TransactionMiddleware`` | For example, ``commit_on_success`` switches to managed mode when entering the | ||||||
| switches to managed mode; at the end of the request, it commits or rollbacks, | block of code it controls; when exiting the block, it commits or rollbacks, | ||||||
| and switches back to auto mode. | and switches back to auto mode. | ||||||
|  |  | ||||||
| .. admonition:: Nesting decorators / context managers | So :func:`commit_on_success` really has two effects: it changes the | ||||||
|  | transaction state and it defines an transaction block. Nesting will give the | ||||||
|  | expected results in terms of transaction state, but not in terms of | ||||||
|  | transaction semantics. Most often, the inner block will commit, breaking the | ||||||
|  | atomicity of the outer block. | ||||||
|  |  | ||||||
|     :func:`commit_on_success` has two effects: it changes the transaction | :func:`autocommit` and :func:`commit_manually` have similar limitations. | ||||||
|     state, and defines an atomic transaction block. |  | ||||||
|  |  | ||||||
|     Nesting with :func:`autocommit` and :func:`commit_manually` will give the | API changes | ||||||
|     expected results in terms of transaction state, but not in terms of | ----------- | ||||||
|     transaction semantics. Most often, the inner block will commit, breaking |  | ||||||
|     the atomicity of the outer block. |  | ||||||
|  |  | ||||||
| Django currently doesn't provide any APIs to create transactions in auto mode. | Managing transactions | ||||||
|  | ~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| .. _transactions-changes-from-1.5: | Starting with Django 1.6, :func:`atomic` is the only supported API for | ||||||
|  | defining a transaction. Unlike the deprecated APIs, it's nestable and always | ||||||
|  | guarantees atomicity. | ||||||
|  |  | ||||||
| Changes from Django 1.5 and earlier | In most cases, it will be a drop-in replacement for :func:`commit_on_success`. | ||||||
| =================================== |  | ||||||
|  | During the deprecation period, it's possible to use :func:`atomic` within | ||||||
|  | :func:`autocommit`, :func:`commit_on_success` or :func:`commit_manually`. | ||||||
|  | However, the reverse is forbidden, because nesting the old decorators / | ||||||
|  | context managers breaks atomicity. | ||||||
|  |  | ||||||
|  | If you enter :func:`atomic` while you're in managed mode, it will trigger a | ||||||
|  | commit to start from a clean slate. | ||||||
|  |  | ||||||
|  | Managing autocommit | ||||||
|  | ~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|  | Django 1.6 introduces an explicit :ref:`API for mananging autocommit | ||||||
|  | <managing-autocommit>`. | ||||||
|  |  | ||||||
|  | To disable autocommit temporarily, instead of:: | ||||||
|  |  | ||||||
|  |     with transaction.commit_manually(): | ||||||
|  |         # do stuff | ||||||
|  |  | ||||||
|  | you should now use:: | ||||||
|  |  | ||||||
|  |     transaction.set_autocommit(autocommit=False) | ||||||
|  |     try: | ||||||
|  |         # do stuff | ||||||
|  |     finally: | ||||||
|  |         transaction.set_autocommit(autocommit=True) | ||||||
|  |  | ||||||
|  | To enable autocommit temporarily, instead of:: | ||||||
|  |  | ||||||
|  |     with transaction.autocommit(): | ||||||
|  |         # do stuff | ||||||
|  |  | ||||||
|  | you should now use:: | ||||||
|  |  | ||||||
|  |     transaction.set_autocommit(autocommit=True) | ||||||
|  |     try: | ||||||
|  |         # do stuff | ||||||
|  |     finally: | ||||||
|  |         transaction.set_autocommit(autocommit=False) | ||||||
|  |  | ||||||
|  | Backwards incompatibilities | ||||||
|  | --------------------------- | ||||||
|  |  | ||||||
| Since version 1.6, Django uses database-level autocommit in auto mode. | Since version 1.6, Django uses database-level autocommit in auto mode. | ||||||
|  |  | ||||||
| Previously, it implemented application-level autocommit by triggering a commit | Previously, it implemented application-level autocommit by triggering a commit | ||||||
| after each ORM write. | after each ORM write. | ||||||
|  |  | ||||||
| As a consequence, each database query (for instance, an | As a consequence, each database query (for instance, an ORM read) started a | ||||||
| ORM read) started a transaction that lasted until the next ORM write. Such | transaction that lasted until the next ORM write. Such "automatic | ||||||
| "automatic transactions" no longer exist in Django 1.6. | transactions" no longer exist in Django 1.6. | ||||||
|  |  | ||||||
| There are four known scenarios where this is backwards-incompatible. | There are four known scenarios where this is backwards-incompatible. | ||||||
|  |  | ||||||
| @@ -565,7 +525,7 @@ Note that managed mode isn't affected at all. This section assumes auto mode. | |||||||
| See the :ref:`description of modes <transaction-states>` above. | See the :ref:`description of modes <transaction-states>` above. | ||||||
|  |  | ||||||
| Sequences of custom SQL queries | Sequences of custom SQL queries | ||||||
| ------------------------------- | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| If you're executing several :ref:`custom SQL queries <executing-custom-sql>` | If you're executing several :ref:`custom SQL queries <executing-custom-sql>` | ||||||
| in a row, each one now runs in its own transaction, instead of sharing the | in a row, each one now runs in its own transaction, instead of sharing the | ||||||
| @@ -577,20 +537,20 @@ usually followed by a call to ``transaction.commit_unless_managed``, which | |||||||
| isn't necessary any more and should be removed. | isn't necessary any more and should be removed. | ||||||
|  |  | ||||||
| Select for update | Select for update | ||||||
| ----------------- | ~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| If you were relying on "automatic transactions" to provide locking between | If you were relying on "automatic transactions" to provide locking between | ||||||
| :meth:`~django.db.models.query.QuerySet.select_for_update` and a subsequent | :meth:`~django.db.models.query.QuerySet.select_for_update` and a subsequent | ||||||
| write operation — an extremely fragile design, but nonetheless possible — you | write operation — an extremely fragile design, but nonetheless possible — you | ||||||
| must wrap the relevant code in :func:`commit_on_success`. | must wrap the relevant code in :func:`atomic`. | ||||||
|  |  | ||||||
| Using a high isolation level | Using a high isolation level | ||||||
| ---------------------------- | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| If you were using the "repeatable read" isolation level or higher, and if you | If you were using the "repeatable read" isolation level or higher, and if you | ||||||
| relied on "automatic transactions" to guarantee consistency between successive | relied on "automatic transactions" to guarantee consistency between successive | ||||||
| reads, the new behavior is backwards-incompatible. To maintain consistency, | reads, the new behavior might be backwards-incompatible. To enforce | ||||||
| you must wrap such sequences in :func:`commit_on_success`. | consistency, you must wrap such sequences in :func:`atomic`. | ||||||
|  |  | ||||||
| MySQL defaults to "repeatable read" and SQLite to "serializable"; they may be | MySQL defaults to "repeatable read" and SQLite to "serializable"; they may be | ||||||
| affected by this problem. | affected by this problem. | ||||||
| @@ -602,10 +562,9 @@ PostgreSQL and Oracle default to "read committed" and aren't affected, unless | |||||||
| you changed the isolation level. | you changed the isolation level. | ||||||
|  |  | ||||||
| Using unsupported database features | Using unsupported database features | ||||||
| ----------------------------------- | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| With triggers, views, or functions, it's possible to make ORM reads result in | With triggers, views, or functions, it's possible to make ORM reads result in | ||||||
| database modifications. Django 1.5 and earlier doesn't deal with this case and | database modifications. Django 1.5 and earlier doesn't deal with this case and | ||||||
| it's theoretically possible to observe a different behavior after upgrading to | it's theoretically possible to observe a different behavior after upgrading to | ||||||
| Django 1.6 or later. In doubt, use :func:`commit_on_success` to enforce | Django 1.6 or later. In doubt, use :func:`atomic` to enforce integrity. | ||||||
| integrity. |  | ||||||
|   | |||||||
| @@ -522,7 +522,8 @@ class FkConstraintsTests(TransactionTestCase): | |||||||
|         """ |         """ | ||||||
|         When constraint checks are disabled, should be able to write bad data without IntegrityErrors. |         When constraint checks are disabled, should be able to write bad data without IntegrityErrors. | ||||||
|         """ |         """ | ||||||
|         with transaction.commit_manually(): |         transaction.set_autocommit(autocommit=False) | ||||||
|  |         try: | ||||||
|             # Create an Article. |             # Create an Article. | ||||||
|             models.Article.objects.create(headline="Test article", pub_date=datetime.datetime(2010, 9, 4), reporter=self.r) |             models.Article.objects.create(headline="Test article", pub_date=datetime.datetime(2010, 9, 4), reporter=self.r) | ||||||
|             # Retrive it from the DB |             # Retrive it from the DB | ||||||
| @@ -536,12 +537,15 @@ class FkConstraintsTests(TransactionTestCase): | |||||||
|                 self.fail("IntegrityError should not have occurred.") |                 self.fail("IntegrityError should not have occurred.") | ||||||
|             finally: |             finally: | ||||||
|                 transaction.rollback() |                 transaction.rollback() | ||||||
|  |         finally: | ||||||
|  |             transaction.set_autocommit(autocommit=True) | ||||||
|  |  | ||||||
|     def test_disable_constraint_checks_context_manager(self): |     def test_disable_constraint_checks_context_manager(self): | ||||||
|         """ |         """ | ||||||
|         When constraint checks are disabled (using context manager), should be able to write bad data without IntegrityErrors. |         When constraint checks are disabled (using context manager), should be able to write bad data without IntegrityErrors. | ||||||
|         """ |         """ | ||||||
|         with transaction.commit_manually(): |         transaction.set_autocommit(autocommit=False) | ||||||
|  |         try: | ||||||
|             # Create an Article. |             # Create an Article. | ||||||
|             models.Article.objects.create(headline="Test article", pub_date=datetime.datetime(2010, 9, 4), reporter=self.r) |             models.Article.objects.create(headline="Test article", pub_date=datetime.datetime(2010, 9, 4), reporter=self.r) | ||||||
|             # Retrive it from the DB |             # Retrive it from the DB | ||||||
| @@ -554,12 +558,15 @@ class FkConstraintsTests(TransactionTestCase): | |||||||
|                 self.fail("IntegrityError should not have occurred.") |                 self.fail("IntegrityError should not have occurred.") | ||||||
|             finally: |             finally: | ||||||
|                 transaction.rollback() |                 transaction.rollback() | ||||||
|  |         finally: | ||||||
|  |             transaction.set_autocommit(autocommit=True) | ||||||
|  |  | ||||||
|     def test_check_constraints(self): |     def test_check_constraints(self): | ||||||
|         """ |         """ | ||||||
|         Constraint checks should raise an IntegrityError when bad data is in the DB. |         Constraint checks should raise an IntegrityError when bad data is in the DB. | ||||||
|         """ |         """ | ||||||
|         with transaction.commit_manually(): |         try: | ||||||
|  |             transaction.set_autocommit(autocommit=False) | ||||||
|             # Create an Article. |             # Create an Article. | ||||||
|             models.Article.objects.create(headline="Test article", pub_date=datetime.datetime(2010, 9, 4), reporter=self.r) |             models.Article.objects.create(headline="Test article", pub_date=datetime.datetime(2010, 9, 4), reporter=self.r) | ||||||
|             # Retrive it from the DB |             # Retrive it from the DB | ||||||
| @@ -572,6 +579,8 @@ class FkConstraintsTests(TransactionTestCase): | |||||||
|                         connection.check_constraints() |                         connection.check_constraints() | ||||||
|             finally: |             finally: | ||||||
|                 transaction.rollback() |                 transaction.rollback() | ||||||
|  |         finally: | ||||||
|  |             transaction.set_autocommit(autocommit=True) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ThreadTests(TestCase): | class ThreadTests(TestCase): | ||||||
|   | |||||||
| @@ -25,7 +25,8 @@ class SampleTestCase(TestCase): | |||||||
|  |  | ||||||
| class TestNoInitialDataLoading(TransactionTestCase): | class TestNoInitialDataLoading(TransactionTestCase): | ||||||
|     def test_syncdb(self): |     def test_syncdb(self): | ||||||
|         with transaction.commit_manually(): |         transaction.set_autocommit(autocommit=False) | ||||||
|  |         try: | ||||||
|             Book.objects.all().delete() |             Book.objects.all().delete() | ||||||
|  |  | ||||||
|             management.call_command( |             management.call_command( | ||||||
| @@ -35,6 +36,9 @@ class TestNoInitialDataLoading(TransactionTestCase): | |||||||
|             ) |             ) | ||||||
|             self.assertQuerysetEqual(Book.objects.all(), []) |             self.assertQuerysetEqual(Book.objects.all(), []) | ||||||
|             transaction.rollback() |             transaction.rollback() | ||||||
|  |         finally: | ||||||
|  |             transaction.set_autocommit(autocommit=True) | ||||||
|  |  | ||||||
|  |  | ||||||
|     def test_flush(self): |     def test_flush(self): | ||||||
|         # Test presence of fixture (flush called by TransactionTestCase) |         # Test presence of fixture (flush called by TransactionTestCase) | ||||||
| @@ -45,7 +49,8 @@ class TestNoInitialDataLoading(TransactionTestCase): | |||||||
|             lambda a: a.name |             lambda a: a.name | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         with transaction.commit_manually(): |         transaction.set_autocommit(autocommit=False) | ||||||
|  |         try: | ||||||
|             management.call_command( |             management.call_command( | ||||||
|                 'flush', |                 'flush', | ||||||
|                 verbosity=0, |                 verbosity=0, | ||||||
| @@ -55,6 +60,8 @@ class TestNoInitialDataLoading(TransactionTestCase): | |||||||
|             ) |             ) | ||||||
|             self.assertQuerysetEqual(Book.objects.all(), []) |             self.assertQuerysetEqual(Book.objects.all(), []) | ||||||
|             transaction.rollback() |             transaction.rollback() | ||||||
|  |         finally: | ||||||
|  |             transaction.set_autocommit(autocommit=True) | ||||||
|  |  | ||||||
|  |  | ||||||
| class FixtureTestCase(TestCase): | class FixtureTestCase(TestCase): | ||||||
|   | |||||||
| @@ -684,5 +684,8 @@ class TestTicket11101(TransactionTestCase): | |||||||
|     @skipUnlessDBFeature('supports_transactions') |     @skipUnlessDBFeature('supports_transactions') | ||||||
|     def test_ticket_11101(self): |     def test_ticket_11101(self): | ||||||
|         """Test that fixtures can be rolled back (ticket #11101).""" |         """Test that fixtures can be rolled back (ticket #11101).""" | ||||||
|         ticket_11101 = transaction.commit_manually(self.ticket_11101) |         transaction.set_autocommit(autocommit=False) | ||||||
|         ticket_11101() |         try: | ||||||
|  |             self.ticket_11101() | ||||||
|  |         finally: | ||||||
|  |             transaction.set_autocommit(autocommit=True) | ||||||
|   | |||||||
| @@ -24,6 +24,8 @@ from django.utils.encoding import force_str | |||||||
| from django.utils.six.moves import xrange | from django.utils.six.moves import xrange | ||||||
| from django.utils.unittest import expectedFailure | from django.utils.unittest import expectedFailure | ||||||
|  |  | ||||||
|  | from transactions.tests import IgnorePendingDeprecationWarningsMixin | ||||||
|  |  | ||||||
| from .models import Band | from .models import Band | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -670,11 +672,12 @@ class ETagGZipMiddlewareTest(TestCase): | |||||||
|  |  | ||||||
|         self.assertNotEqual(gzip_etag, nogzip_etag) |         self.assertNotEqual(gzip_etag, nogzip_etag) | ||||||
|  |  | ||||||
| class TransactionMiddlewareTest(TransactionTestCase): | class TransactionMiddlewareTest(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|     """ |     """ | ||||||
|     Test the transaction middleware. |     Test the transaction middleware. | ||||||
|     """ |     """ | ||||||
|     def setUp(self): |     def setUp(self): | ||||||
|  |         super(TransactionMiddlewareTest, self).setUp() | ||||||
|         self.request = HttpRequest() |         self.request = HttpRequest() | ||||||
|         self.request.META = { |         self.request.META = { | ||||||
|             'SERVER_NAME': 'testserver', |             'SERVER_NAME': 'testserver', | ||||||
| @@ -686,6 +689,7 @@ class TransactionMiddlewareTest(TransactionTestCase): | |||||||
|  |  | ||||||
|     def tearDown(self): |     def tearDown(self): | ||||||
|         transaction.abort() |         transaction.abort() | ||||||
|  |         super(TransactionMiddlewareTest, self).tearDown() | ||||||
|  |  | ||||||
|     def test_request(self): |     def test_request(self): | ||||||
|         TransactionMiddleware().process_request(self.request) |         TransactionMiddleware().process_request(self.request) | ||||||
|   | |||||||
| @@ -1,9 +1,10 @@ | |||||||
| from __future__ import absolute_import | from __future__ import absolute_import | ||||||
|  |  | ||||||
| import sys | import sys | ||||||
|  | import warnings | ||||||
|  |  | ||||||
| from django.db import connection, transaction, IntegrityError | from django.db import connection, transaction, IntegrityError | ||||||
| from django.test import TestCase, TransactionTestCase, skipUnlessDBFeature | from django.test import TransactionTestCase, skipUnlessDBFeature | ||||||
| from django.utils import six | from django.utils import six | ||||||
| from django.utils.unittest import skipUnless | from django.utils.unittest import skipUnless | ||||||
|  |  | ||||||
| @@ -158,7 +159,69 @@ class AtomicInsideTransactionTests(AtomicTests): | |||||||
|         self.atomic.__exit__(*sys.exc_info()) |         self.atomic.__exit__(*sys.exc_info()) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransactionTests(TransactionTestCase): | class AtomicInsideLegacyTransactionManagementTests(AtomicTests): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         transaction.enter_transaction_management() | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         # The tests access the database after exercising 'atomic', making the | ||||||
|  |         # connection dirty; a rollback is required to make it clean. | ||||||
|  |         transaction.rollback() | ||||||
|  |         transaction.leave_transaction_management() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @skipUnless(connection.features.uses_savepoints, | ||||||
|  |         "'atomic' requires transactions and savepoints.") | ||||||
|  | class AtomicErrorsTests(TransactionTestCase): | ||||||
|  |  | ||||||
|  |     def test_atomic_requires_autocommit(self): | ||||||
|  |         transaction.set_autocommit(autocommit=False) | ||||||
|  |         try: | ||||||
|  |             with self.assertRaises(transaction.TransactionManagementError): | ||||||
|  |                 with transaction.atomic(): | ||||||
|  |                     pass | ||||||
|  |         finally: | ||||||
|  |             transaction.set_autocommit(autocommit=True) | ||||||
|  |  | ||||||
|  |     def test_atomic_prevents_disabling_autocommit(self): | ||||||
|  |         autocommit = transaction.get_autocommit() | ||||||
|  |         with transaction.atomic(): | ||||||
|  |             with self.assertRaises(transaction.TransactionManagementError): | ||||||
|  |                 transaction.set_autocommit(autocommit=not autocommit) | ||||||
|  |         # Make sure autocommit wasn't changed. | ||||||
|  |         self.assertEqual(connection.autocommit, autocommit) | ||||||
|  |  | ||||||
|  |     def test_atomic_prevents_calling_transaction_methods(self): | ||||||
|  |         with transaction.atomic(): | ||||||
|  |             with self.assertRaises(transaction.TransactionManagementError): | ||||||
|  |                 transaction.commit() | ||||||
|  |             with self.assertRaises(transaction.TransactionManagementError): | ||||||
|  |                 transaction.rollback() | ||||||
|  |  | ||||||
|  |     def test_atomic_prevents_calling_transaction_management_methods(self): | ||||||
|  |         with transaction.atomic(): | ||||||
|  |             with self.assertRaises(transaction.TransactionManagementError): | ||||||
|  |                 transaction.enter_transaction_management() | ||||||
|  |             with self.assertRaises(transaction.TransactionManagementError): | ||||||
|  |                 transaction.leave_transaction_management() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class IgnorePendingDeprecationWarningsMixin(object): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         super(IgnorePendingDeprecationWarningsMixin, self).setUp() | ||||||
|  |         self.catch_warnings = warnings.catch_warnings() | ||||||
|  |         self.catch_warnings.__enter__() | ||||||
|  |         warnings.filterwarnings("ignore", category=PendingDeprecationWarning) | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         self.catch_warnings.__exit__(*sys.exc_info()) | ||||||
|  |         super(IgnorePendingDeprecationWarningsMixin, self).tearDown() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TransactionTests(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|  |  | ||||||
|     def create_a_reporter_then_fail(self, first, last): |     def create_a_reporter_then_fail(self, first, last): | ||||||
|         a = Reporter(first_name=first, last_name=last) |         a = Reporter(first_name=first, last_name=last) | ||||||
|         a.save() |         a.save() | ||||||
| @@ -313,7 +376,7 @@ class TransactionTests(TransactionTestCase): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransactionRollbackTests(TransactionTestCase): | class TransactionRollbackTests(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|     def execute_bad_sql(self): |     def execute_bad_sql(self): | ||||||
|         cursor = connection.cursor() |         cursor = connection.cursor() | ||||||
|         cursor.execute("INSERT INTO transactions_reporter (first_name, last_name) VALUES ('Douglas', 'Adams');") |         cursor.execute("INSERT INTO transactions_reporter (first_name, last_name) VALUES ('Douglas', 'Adams');") | ||||||
| @@ -330,7 +393,7 @@ class TransactionRollbackTests(TransactionTestCase): | |||||||
|         self.assertRaises(IntegrityError, execute_bad_sql) |         self.assertRaises(IntegrityError, execute_bad_sql) | ||||||
|         transaction.rollback() |         transaction.rollback() | ||||||
|  |  | ||||||
| class TransactionContextManagerTests(TransactionTestCase): | class TransactionContextManagerTests(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|     def create_reporter_and_fail(self): |     def create_reporter_and_fail(self): | ||||||
|         Reporter.objects.create(first_name="Bob", last_name="Holtzman") |         Reporter.objects.create(first_name="Bob", last_name="Holtzman") | ||||||
|         raise Exception |         raise Exception | ||||||
|   | |||||||
| @@ -6,10 +6,12 @@ from django.test import TransactionTestCase, skipUnlessDBFeature | |||||||
| from django.test.utils import override_settings | from django.test.utils import override_settings | ||||||
| from django.utils.unittest import skipIf, skipUnless, expectedFailure | from django.utils.unittest import skipIf, skipUnless, expectedFailure | ||||||
|  |  | ||||||
|  | from transactions.tests import IgnorePendingDeprecationWarningsMixin | ||||||
|  |  | ||||||
| from .models import Mod, M2mA, M2mB | from .models import Mod, M2mA, M2mB | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestTransactionClosing(TransactionTestCase): | class TestTransactionClosing(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|     """ |     """ | ||||||
|     Tests to make sure that transactions are properly closed |     Tests to make sure that transactions are properly closed | ||||||
|     when they should be, and aren't left pending after operations |     when they should be, and aren't left pending after operations | ||||||
| @@ -166,7 +168,7 @@ class TestTransactionClosing(TransactionTestCase): | |||||||
|         (connection.settings_dict['NAME'] == ':memory:' or |         (connection.settings_dict['NAME'] == ':memory:' or | ||||||
|          not connection.settings_dict['NAME']), |          not connection.settings_dict['NAME']), | ||||||
|         'Test uses multiple connections, but in-memory sqlite does not support this') |         'Test uses multiple connections, but in-memory sqlite does not support this') | ||||||
| class TestNewConnection(TransactionTestCase): | class TestNewConnection(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|     """ |     """ | ||||||
|     Check that new connections don't have special behaviour. |     Check that new connections don't have special behaviour. | ||||||
|     """ |     """ | ||||||
| @@ -211,7 +213,7 @@ class TestNewConnection(TransactionTestCase): | |||||||
|  |  | ||||||
| @skipUnless(connection.vendor == 'postgresql', | @skipUnless(connection.vendor == 'postgresql', | ||||||
|             "This test only valid for PostgreSQL") |             "This test only valid for PostgreSQL") | ||||||
| class TestPostgresAutocommitAndIsolation(TransactionTestCase): | class TestPostgresAutocommitAndIsolation(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|     """ |     """ | ||||||
|     Tests to make sure psycopg2's autocommit mode and isolation level |     Tests to make sure psycopg2's autocommit mode and isolation level | ||||||
|     is restored after entering and leaving transaction management. |     is restored after entering and leaving transaction management. | ||||||
| @@ -292,7 +294,7 @@ class TestPostgresAutocommitAndIsolation(TransactionTestCase): | |||||||
|         self.assertTrue(connection.autocommit) |         self.assertTrue(connection.autocommit) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestManyToManyAddTransaction(TransactionTestCase): | class TestManyToManyAddTransaction(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|     def test_manyrelated_add_commit(self): |     def test_manyrelated_add_commit(self): | ||||||
|         "Test for https://code.djangoproject.com/ticket/16818" |         "Test for https://code.djangoproject.com/ticket/16818" | ||||||
|         a = M2mA.objects.create() |         a = M2mA.objects.create() | ||||||
| @@ -307,7 +309,7 @@ class TestManyToManyAddTransaction(TransactionTestCase): | |||||||
|         self.assertEqual(a.others.count(), 1) |         self.assertEqual(a.others.count(), 1) | ||||||
|  |  | ||||||
|  |  | ||||||
| class SavepointTest(TransactionTestCase): | class SavepointTest(IgnorePendingDeprecationWarningsMixin, TransactionTestCase): | ||||||
|  |  | ||||||
|     @skipIf(connection.vendor == 'sqlite', |     @skipIf(connection.vendor == 'sqlite', | ||||||
|             "SQLite doesn't support savepoints in managed mode") |             "SQLite doesn't support savepoints in managed mode") | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user