mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #9919 -- Added note on the need to mark transactions as dirty when using raw SQL.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@11022 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -616,6 +616,8 @@ call, since they are conflicting options. | ||||
| Both the ``depth`` argument and the ability to specify field names in the call | ||||
| to ``select_related()`` are new in Django version 1.0. | ||||
|  | ||||
| .. _extra: | ||||
|  | ||||
| ``extra(select=None, where=None, params=None, tables=None, order_by=None, select_params=None)`` | ||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
|   | ||||
| @@ -29,6 +29,45 @@ is required. For example:: | ||||
|  | ||||
|         return row | ||||
|  | ||||
| .. _transactions-and-raw-sql: | ||||
|  | ||||
| Transactions and raw SQL | ||||
| ------------------------ | ||||
| If you are using transaction decorators (such as ``commit_on_success``) to | ||||
| wrap your views and provide transaction control, you don't have to make a | ||||
| manual call to ``transaction.commit_unless_managed()`` -- you can manually | ||||
| commit if you want to, but you aren't required to, since the decorator will | ||||
| commit for you. However, if you don't manually commit your changes, you will | ||||
| need to manually mark the transaction as dirty, using | ||||
| ``transaction.set_dirty()``:: | ||||
|  | ||||
|     @commit_on_success | ||||
|     def my_custom_sql_view(request, value): | ||||
|         from django.db import connection, transaction | ||||
|         cursor = connection.cursor() | ||||
|  | ||||
|         # Data modifying operation | ||||
|         cursor.execute("UPDATE bar SET foo = 1 WHERE baz = %s", [value]) | ||||
|  | ||||
|         # Since we modified data, mark the transaction as dirty | ||||
|         transaction.set_dirty() | ||||
|  | ||||
|         # Data retrieval operation. This doesn't dirty the transaction, | ||||
|         # so no call to set_dirty() is required. | ||||
|         cursor.execute("SELECT foo FROM bar WHERE baz = %s", [value]) | ||||
|         row = cursor.fetchone() | ||||
|  | ||||
|         return render_to_response('template.html', {'row': row}) | ||||
|  | ||||
| The call to ``set_dirty()`` is made automatically when you use the Django ORM | ||||
| to make data modifying database calls. However, when you use raw SQL, Django | ||||
| has no way of knowing if your SQL modifies data or not. The manual call to | ||||
| ``set_dirty()`` ensures that Django knows that there are modifications that | ||||
| must be committed. | ||||
|  | ||||
| Connections and cursors | ||||
| ----------------------- | ||||
|  | ||||
| ``connection`` and ``cursor`` mostly implement the standard `Python DB-API`_ | ||||
| (except when it comes to :ref:`transaction handling <topics-db-transactions>`). | ||||
| If you're not familiar with the Python DB-API, note that the SQL statement in | ||||
| @@ -39,9 +78,12 @@ necessary. (Also note that Django expects the ``"%s"`` placeholder, *not* the | ||||
| ``"?"`` placeholder, which is used by the SQLite Python bindings. This is for | ||||
| the sake of consistency and sanity.) | ||||
|  | ||||
| An easier option? | ||||
| ----------------- | ||||
|  | ||||
| A final note: If all you want to do is a custom ``WHERE`` clause, you can just | ||||
| use the ``where``, ``tables`` and ``params`` arguments to the standard lookup | ||||
| API. | ||||
| use the ``where``, ``tables`` and ``params`` arguments to the | ||||
| :ref:`extra clause <extra>` in the standard queryset API. | ||||
|  | ||||
| .. _Python DB-API: http://www.python.org/peps/pep-0249.html | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user