mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Deprecated TransactionMiddleware and TRANSACTIONS_MANAGED.
Replaced them with per-database options, for proper multi-db support. Also toned down the recommendation to tie transactions to HTTP requests. Thanks Jeremy for sharing his experience.
This commit is contained in:
		| @@ -6,10 +6,10 @@ import types | ||||
|  | ||||
| from django import http | ||||
| from django.conf import settings | ||||
| from django.core import exceptions | ||||
| from django.core import urlresolvers | ||||
| from django.core import signals | ||||
| from django.core.exceptions import MiddlewareNotUsed, PermissionDenied | ||||
| from django.db import connections, transaction | ||||
| from django.utils.encoding import force_text | ||||
| from django.utils.module_loading import import_by_path | ||||
| from django.utils import six | ||||
| @@ -65,6 +65,13 @@ class BaseHandler(object): | ||||
|         # as a flag for initialization being complete. | ||||
|         self._request_middleware = request_middleware | ||||
|  | ||||
|     def make_view_atomic(self, view): | ||||
|         if getattr(view, 'transactions_per_request', True): | ||||
|             for db in connections.all(): | ||||
|                 if db.settings_dict['ATOMIC_REQUESTS']: | ||||
|                     view = transaction.atomic(using=db.alias)(view) | ||||
|         return view | ||||
|  | ||||
|     def get_response(self, request): | ||||
|         "Returns an HttpResponse object for the given HttpRequest" | ||||
|         try: | ||||
| @@ -101,8 +108,9 @@ class BaseHandler(object): | ||||
|                             break | ||||
|  | ||||
|                 if response is None: | ||||
|                     wrapped_callback = self.make_view_atomic(callback) | ||||
|                     try: | ||||
|                         response = callback(request, *callback_args, **callback_kwargs) | ||||
|                         response = wrapped_callback(request, *callback_args, **callback_kwargs) | ||||
|                     except Exception as e: | ||||
|                         # If the view raised an exception, run it through exception | ||||
|                         # middleware, and if the exception middleware returns a | ||||
|   | ||||
| @@ -104,7 +104,7 @@ class BaseDatabaseWrapper(object): | ||||
|         conn_params = self.get_connection_params() | ||||
|         self.connection = self.get_new_connection(conn_params) | ||||
|         self.init_connection_state() | ||||
|         if not settings.TRANSACTIONS_MANAGED: | ||||
|         if self.settings_dict['AUTOCOMMIT']: | ||||
|             self.set_autocommit() | ||||
|         connection_created.send(sender=self.__class__, connection=self) | ||||
|  | ||||
| @@ -299,7 +299,7 @@ class BaseDatabaseWrapper(object): | ||||
|         if self.transaction_state: | ||||
|             managed = self.transaction_state[-1] | ||||
|         else: | ||||
|             managed = settings.TRANSACTIONS_MANAGED | ||||
|             managed = not self.settings_dict['AUTOCOMMIT'] | ||||
|  | ||||
|         if self._dirty: | ||||
|             self.rollback() | ||||
|   | ||||
| @@ -2,6 +2,7 @@ from functools import wraps | ||||
| import os | ||||
| import pkgutil | ||||
| from threading import local | ||||
| import warnings | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.core.exceptions import ImproperlyConfigured | ||||
| @@ -158,6 +159,13 @@ class ConnectionHandler(object): | ||||
|         except KeyError: | ||||
|             raise ConnectionDoesNotExist("The connection %s doesn't exist" % alias) | ||||
|  | ||||
|         conn.setdefault('ATOMIC_REQUESTS', False) | ||||
|         if settings.TRANSACTIONS_MANAGED: | ||||
|             warnings.warn( | ||||
|                 "TRANSACTIONS_MANAGED is deprecated. Use AUTOCOMMIT instead.", | ||||
|                 PendingDeprecationWarning, stacklevel=2) | ||||
|             conn.setdefault('AUTOCOMMIT', False) | ||||
|         conn.setdefault('AUTOCOMMIT', True) | ||||
|         conn.setdefault('ENGINE', 'django.db.backends.dummy') | ||||
|         if conn['ENGINE'] == 'django.db.backends.' or not conn['ENGINE']: | ||||
|             conn['ENGINE'] = 'django.db.backends.dummy' | ||||
|   | ||||
| @@ -1,4 +1,7 @@ | ||||
| from django.db import transaction | ||||
| import warnings | ||||
|  | ||||
| from django.core.exceptions import MiddlewareNotUsed | ||||
| from django.db import connection, transaction | ||||
|  | ||||
| class TransactionMiddleware(object): | ||||
|     """ | ||||
| @@ -7,6 +10,14 @@ class TransactionMiddleware(object): | ||||
|     commit, the commit is done when a successful response is created. If an | ||||
|     exception happens, the database is rolled back. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self): | ||||
|         warnings.warn( | ||||
|             "TransactionMiddleware is deprecated in favor of ATOMIC_REQUESTS.", | ||||
|             PendingDeprecationWarning, stacklevel=2) | ||||
|         if connection.settings_dict['ATOMIC_REQUESTS']: | ||||
|             raise MiddlewareNotUsed | ||||
|  | ||||
|     def process_request(self, request): | ||||
|         """Enters transaction management""" | ||||
|         transaction.enter_transaction_management() | ||||
|   | ||||
| @@ -329,9 +329,14 @@ these changes. | ||||
| 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 following transaction management APIs will be removed: | ||||
|  | ||||
|   - ``TransactionMiddleware``, | ||||
|   - the decorators and context managers ``autocommit``, ``commit_on_success``, | ||||
|     and ``commit_manually``, | ||||
|   - the ``TRANSACTIONS_MANAGED`` setting. | ||||
|  | ||||
|   Upgrade paths are described in :ref:`transactions-upgrading-from-1.5`. | ||||
|  | ||||
| * 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 | ||||
|   | ||||
| @@ -205,6 +205,10 @@ Transaction middleware | ||||
|  | ||||
| .. class:: TransactionMiddleware | ||||
|  | ||||
| .. versionchanged:: 1.6 | ||||
|     ``TransactionMiddleware`` is deprecated. The documentation of transactions | ||||
|     contains :ref:`upgrade instructions <transactions-upgrading-from-1.5>`. | ||||
|  | ||||
| Binds commit and rollback of the default database to the request/response | ||||
| phase. If a view function runs successfully, a commit is done. If it fails with | ||||
| an exception, a rollback is done. | ||||
|   | ||||
| @@ -408,6 +408,30 @@ SQLite. This can be configured using the following:: | ||||
| For other database backends, or more complex SQLite configurations, other options | ||||
| will be required. The following inner options are available. | ||||
|  | ||||
| .. setting:: DATABASE-ATOMIC_REQUESTS | ||||
|  | ||||
| ATOMIC_REQUESTS | ||||
| ~~~~~~~~~~~~~~~ | ||||
|  | ||||
| .. versionadded:: 1.6 | ||||
|  | ||||
| Default: ``False`` | ||||
|  | ||||
| Set this to ``True`` to wrap each HTTP request in a transaction on this | ||||
| database. See :ref:`tying-transactions-to-http-requests`. | ||||
|  | ||||
| .. setting:: DATABASE-AUTOCOMMIT | ||||
|  | ||||
| AUTOCOMMIT | ||||
| ~~~~~~~~~~ | ||||
|  | ||||
| .. versionadded:: 1.6 | ||||
|  | ||||
| Default: ``True`` | ||||
|  | ||||
| Set this to ``False`` if you want to :ref:`disable Django's transaction | ||||
| management <deactivate-transaction-management>` and implement your own. | ||||
|  | ||||
| .. setting:: DATABASE-ENGINE | ||||
|  | ||||
| ENGINE | ||||
| @@ -1807,6 +1831,12 @@ to ensure your processes are running in the correct environment. | ||||
| TRANSACTIONS_MANAGED | ||||
| -------------------- | ||||
|  | ||||
| .. deprecated:: 1.6 | ||||
|  | ||||
|     This setting was deprecated because its name is very misleading. Use the | ||||
|     :setting:`AUTOCOMMIT <DATABASE-AUTOCOMMIT>` key in :setting:`DATABASES` | ||||
|     entries instead. | ||||
|  | ||||
| Default: ``False`` | ||||
|  | ||||
| Set this to ``True`` if you want to :ref:`disable Django's transaction | ||||
|   | ||||
| @@ -262,9 +262,11 @@ 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` | ||||
| - ``django.middleware.transaction.TransactionMiddleware`` | ||||
| - ``django.db.transaction.autocommit`` | ||||
| - ``django.db.transaction.commit_on_success`` | ||||
| - ``django.db.transaction.commit_manually`` | ||||
| - the ``TRANSACTIONS_MANAGED`` setting | ||||
|  | ||||
| The reasons for this change and the upgrade path are described in the | ||||
| :ref:`transactions documentation <transactions-upgrading-from-1.5>`. | ||||
|   | ||||
| @@ -26,45 +26,61 @@ immediately committed to the database. :ref:`See below for details | ||||
|     Previous version of Django featured :ref:`a more complicated default | ||||
|     behavior <transactions-upgrading-from-1.5>`. | ||||
|  | ||||
| .. _tying-transactions-to-http-requests: | ||||
|  | ||||
| Tying transactions to HTTP requests | ||||
| ----------------------------------- | ||||
|  | ||||
| The recommended way to handle transactions in Web requests is to tie them to | ||||
| the request and response phases via Django's ``TransactionMiddleware``. | ||||
| A common way to handle transactions on the web is to wrap each request in a | ||||
| transaction. Set :setting:`ATOMIC_REQUESTS <DATABASE-ATOMIC_REQUESTS>` to | ||||
| ``True`` in the configuration of each database for which you want to enable | ||||
| this behavior. | ||||
|  | ||||
| It works like this. When a request starts, Django starts a transaction. If the | ||||
| response is produced without problems, Django commits any pending transactions. | ||||
| If the view function produces an exception, Django rolls back any pending | ||||
| transactions. | ||||
| response is produced without problems, Django commits the transaction. If the | ||||
| view function produces an exception, Django rolls back the transaction. | ||||
| Middleware always runs outside of this transaction. | ||||
|  | ||||
| To activate this feature, just add the ``TransactionMiddleware`` middleware to | ||||
| your :setting:`MIDDLEWARE_CLASSES` setting:: | ||||
| You may perfom partial commits and rollbacks in your view code, typically with | ||||
| the :func:`atomic` context manager. However, at the end of the view, either | ||||
| all the changes will be committed, or none of them. | ||||
|  | ||||
|     MIDDLEWARE_CLASSES = ( | ||||
|         'django.middleware.cache.UpdateCacheMiddleware', | ||||
|         'django.contrib.sessions.middleware.SessionMiddleware', | ||||
|         'django.middleware.common.CommonMiddleware', | ||||
|         'django.middleware.transaction.TransactionMiddleware', | ||||
|         'django.middleware.cache.FetchFromCacheMiddleware', | ||||
|     ) | ||||
| To disable this behavior for a specific view, you must set the | ||||
| ``transactions_per_request`` attribute of the view function itself to | ||||
| ``False``, like this:: | ||||
|  | ||||
| The order is quite important. The transaction middleware applies not only to | ||||
| view functions, but also for all middleware modules that come after it. So if | ||||
| you use the session middleware after the transaction middleware, session | ||||
| creation will be part of the transaction. | ||||
|     def my_view(request): | ||||
|         do_stuff() | ||||
|     my_view.transactions_per_request = False | ||||
|  | ||||
| The various cache middlewares are an exception: ``CacheMiddleware``, | ||||
| :class:`~django.middleware.cache.UpdateCacheMiddleware`, and | ||||
| :class:`~django.middleware.cache.FetchFromCacheMiddleware` are never affected. | ||||
| Even when using database caching, Django's cache backend uses its own database | ||||
| connection internally. | ||||
| .. warning:: | ||||
|  | ||||
| .. note:: | ||||
|     While the simplicity of this transaction model is appealing, it also makes it | ||||
|     inefficient when traffic increases. Opening a transaction for every view has | ||||
|     some overhead. The impact on performance depends on the query patterns of your | ||||
|     application and on how well your database handles locking. | ||||
|  | ||||
|     The ``TransactionMiddleware`` only affects the database aliased | ||||
|     as "default" within your :setting:`DATABASES` setting.  If you are using | ||||
|     multiple databases and want transaction control over databases other than | ||||
|     "default", you will need to write your own transaction middleware. | ||||
| .. admonition:: Per-request transactions and streaming responses | ||||
|  | ||||
|     When a view returns a :class:`~django.http.StreamingHttpResponse`, reading | ||||
|     the contents of the response will often execute code to generate the | ||||
|     content. Since the view has already returned, such code runs outside of | ||||
|     the transaction. | ||||
|  | ||||
|     Generally speaking, it isn't advisable to write to the database while | ||||
|     generating a streaming response, since there's no sensible way to handle | ||||
|     errors after starting to send the response. | ||||
|  | ||||
| In practice, this feature simply wraps every view function in the :func:`atomic` | ||||
| decorator described below. | ||||
|  | ||||
| Note that only the execution of your view in enclosed in the transactions. | ||||
| Middleware run outside of the transaction, and so does the rendering of | ||||
| template responses. | ||||
|  | ||||
| .. versionchanged:: 1.6 | ||||
|     Django used to provide this feature via ``TransactionMiddleware``, which is | ||||
|     now deprecated. | ||||
|  | ||||
| Controlling transactions explicitly | ||||
| ----------------------------------- | ||||
| @@ -283,18 +299,20 @@ if autocommit is off. Django will also refuse to turn autocommit off when an | ||||
| 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. | ||||
| You can totally disable Django's transaction management for a given database | ||||
| by setting :setting:`AUTOCOMMIT <DATABASE-AUTOCOMMIT>` to ``False`` in its | ||||
| configuration. If you do this, Django won't enable autocommit, and won't | ||||
| perform any commits. 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. | ||||
| .. versionchanged:: 1.6 | ||||
|     This used to be controlled by the ``TRANSACTIONS_MANAGED`` setting. | ||||
|  | ||||
|  | ||||
| Database-specific notes | ||||
| ======================= | ||||
| @@ -459,6 +477,35 @@ atomicity of the outer block. | ||||
| API changes | ||||
| ----------- | ||||
|  | ||||
| Transaction middleware | ||||
| ~~~~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| In Django 1.6, ``TransactionMiddleware`` is deprecated and replaced | ||||
| :setting:`ATOMIC_REQUESTS <DATABASE-ATOMIC_REQUESTS>`. While the general | ||||
| behavior is the same, there are a few differences. | ||||
|  | ||||
| With the transaction middleware, it was still possible to switch to autocommit | ||||
| or to commit explicitly in a view. Since :func:`atomic` guarantees atomicity, | ||||
| this isn't allowed any longer. | ||||
|  | ||||
| To avoid wrapping a particular view in a transaction, instead of:: | ||||
|  | ||||
|     @transaction.autocommit | ||||
|     def my_view(request): | ||||
|         do_stuff() | ||||
|  | ||||
| you must now use this pattern:: | ||||
|  | ||||
|     def my_view(request): | ||||
|         do_stuff() | ||||
|     my_view.transactions_per_request = False | ||||
|  | ||||
| The transaction middleware applied not only to view functions, but also to | ||||
| middleware modules that come after it. For instance, if you used the session | ||||
| middleware after the transaction middleware, session creation was part of the | ||||
| transaction. :setting:`ATOMIC_REQUESTS <DATABASE-ATOMIC_REQUESTS>` only | ||||
| applies to the view itself. | ||||
|  | ||||
| Managing transactions | ||||
| ~~~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| @@ -508,6 +555,13 @@ you should now use:: | ||||
|     finally: | ||||
|         transaction.set_autocommit(autocommit=False) | ||||
|  | ||||
| Disabling transaction management | ||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| Instead of setting ``TRANSACTIONS_MANAGED = True``, set the ``AUTOCOMMIT`` key | ||||
| to ``False`` in the configuration of each database, as explained in :ref | ||||
| :`deactivate-transaction-management`. | ||||
|  | ||||
| Backwards incompatibilities | ||||
| --------------------------- | ||||
|  | ||||
|   | ||||
| @@ -1,9 +1,8 @@ | ||||
| from django.core.handlers.wsgi import WSGIHandler | ||||
| from django.core.signals import request_started, request_finished | ||||
| from django.db import close_old_connections | ||||
| from django.test import RequestFactory, TestCase | ||||
| from django.db import close_old_connections, connection | ||||
| from django.test import RequestFactory, TestCase, TransactionTestCase | ||||
| from django.test.utils import override_settings | ||||
| from django.utils import six | ||||
|  | ||||
|  | ||||
| class HandlerTests(TestCase): | ||||
| @@ -37,6 +36,31 @@ class HandlerTests(TestCase): | ||||
|         self.assertEqual(response.status_code, 400) | ||||
|  | ||||
|  | ||||
| class TransactionsPerRequestTests(TransactionTestCase): | ||||
|     urls = 'handlers.urls' | ||||
|  | ||||
|     def test_no_transaction(self): | ||||
|         response = self.client.get('/in_transaction/') | ||||
|         self.assertContains(response, 'False') | ||||
|  | ||||
|     def test_auto_transaction(self): | ||||
|         old_atomic_requests = connection.settings_dict['ATOMIC_REQUESTS'] | ||||
|         try: | ||||
|             connection.settings_dict['ATOMIC_REQUESTS'] = True | ||||
|             response = self.client.get('/in_transaction/') | ||||
|         finally: | ||||
|             connection.settings_dict['ATOMIC_REQUESTS'] = old_atomic_requests | ||||
|         self.assertContains(response, 'True') | ||||
|  | ||||
|     def test_no_auto_transaction(self): | ||||
|         old_atomic_requests = connection.settings_dict['ATOMIC_REQUESTS'] | ||||
|         try: | ||||
|             connection.settings_dict['ATOMIC_REQUESTS'] = True | ||||
|             response = self.client.get('/not_in_transaction/') | ||||
|         finally: | ||||
|             connection.settings_dict['ATOMIC_REQUESTS'] = old_atomic_requests | ||||
|         self.assertContains(response, 'False') | ||||
|  | ||||
| class SignalsTests(TestCase): | ||||
|     urls = 'handlers.urls' | ||||
|  | ||||
|   | ||||
| @@ -1,9 +1,12 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.conf.urls import patterns, url | ||||
| from django.http import HttpResponse, StreamingHttpResponse | ||||
|  | ||||
| from . import views | ||||
|  | ||||
| urlpatterns = patterns('', | ||||
|     url(r'^regular/$', lambda request: HttpResponse(b"regular content")), | ||||
|     url(r'^streaming/$', lambda request: StreamingHttpResponse([b"streaming", b" ", b"content"])), | ||||
|     url(r'^regular/$', views.regular), | ||||
|     url(r'^streaming/$', views.streaming), | ||||
|     url(r'^in_transaction/$', views.in_transaction), | ||||
|     url(r'^not_in_transaction/$', views.not_in_transaction), | ||||
| ) | ||||
|   | ||||
							
								
								
									
										17
									
								
								tests/handlers/views.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								tests/handlers/views.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from django.db import connection | ||||
| from django.http import HttpResponse, StreamingHttpResponse | ||||
|  | ||||
| def regular(request): | ||||
|     return HttpResponse(b"regular content") | ||||
|  | ||||
| def streaming(request): | ||||
|     return StreamingHttpResponse([b"streaming", b" ", b"content"]) | ||||
|  | ||||
| def in_transaction(request): | ||||
|     return HttpResponse(str(connection.in_atomic_block)) | ||||
|  | ||||
| def not_in_transaction(request): | ||||
|     return HttpResponse(str(connection.in_atomic_block)) | ||||
| not_in_transaction.transactions_per_request = False | ||||
		Reference in New Issue
	
	Block a user