mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Refs #14661 -- Clarified the handling of initial data injected via custom SQL.
This is BACKWARDS INCOMPATIBLE CHANGE for anyone relying on SQL-injected initial data in a test case. git-svn-id: http://code.djangoproject.com/svn/django/trunk@15239 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -26,6 +26,10 @@ class Command(NoArgsCommand): | |||||||
|         interactive = options.get('interactive') |         interactive = options.get('interactive') | ||||||
|         show_traceback = options.get('traceback', False) |         show_traceback = options.get('traceback', False) | ||||||
|  |  | ||||||
|  |         # Stealth option -- 'load_initial_data' is used by the testing setup | ||||||
|  |         # process to disable initial fixture loading. | ||||||
|  |         load_initial_data = options.get('load_initial_data', True) | ||||||
|  |  | ||||||
|         self.style = no_style() |         self.style = no_style() | ||||||
|  |  | ||||||
|         # Import the 'management' module within each installed app, to register |         # Import the 'management' module within each installed app, to register | ||||||
| @@ -154,5 +158,7 @@ class Command(NoArgsCommand): | |||||||
|                         else: |                         else: | ||||||
|                             transaction.commit_unless_managed(using=db) |                             transaction.commit_unless_managed(using=db) | ||||||
|  |  | ||||||
|         from django.core.management import call_command |         # Load initial_data fixtures (unless that has been disabled) | ||||||
|         call_command('loaddata', 'initial_data', verbosity=verbosity, database=db) |         if load_initial_data: | ||||||
|  |             from django.core.management import call_command | ||||||
|  |             call_command('loaddata', 'initial_data', verbosity=verbosity, database=db) | ||||||
|   | |||||||
| @@ -359,7 +359,21 @@ class BaseDatabaseCreation(object): | |||||||
|         # Report syncdb messages at one level lower than that requested. |         # Report syncdb messages at one level lower than that requested. | ||||||
|         # This ensures we don't get flooded with messages during testing |         # This ensures we don't get flooded with messages during testing | ||||||
|         # (unless you really ask to be flooded) |         # (unless you really ask to be flooded) | ||||||
|         call_command('syncdb', verbosity=max(verbosity - 1, 0), interactive=False, database=self.connection.alias) |         call_command('syncdb', | ||||||
|  |             verbosity=max(verbosity - 1, 0), | ||||||
|  |             interactive=False, | ||||||
|  |             database=self.connection.alias, | ||||||
|  |             load_initial_data=False) | ||||||
|  |  | ||||||
|  |         # We need to then do a flush to ensure that any data installed by | ||||||
|  |         # custom SQL has been removed. The only test data should come from | ||||||
|  |         # test fixtures, or autogenerated from post_syncdb triggers. | ||||||
|  |         # This has the side effect of loading initial data (which was | ||||||
|  |         # intentionally skipped in the syncdb). | ||||||
|  |         call_command('flush', | ||||||
|  |             verbosity=max(verbosity - 1, 0), | ||||||
|  |             interactive=False, | ||||||
|  |             database=self.connection.alias) | ||||||
|  |  | ||||||
|         from django.core.cache import get_cache |         from django.core.cache import get_cache | ||||||
|         from django.core.cache.backends.db import BaseDatabaseCache |         from django.core.cache.backends.db import BaseDatabaseCache | ||||||
|   | |||||||
| @@ -121,6 +121,17 @@ the order in which they're executed. The only thing you can assume is | |||||||
| that, by the time your custom data files are executed, all the | that, by the time your custom data files are executed, all the | ||||||
| database tables already will have been created. | database tables already will have been created. | ||||||
|  |  | ||||||
|  | .. admonition:: Initial SQL data and testing | ||||||
|  |  | ||||||
|  |     This technique *cannot* be used to provide initial data for | ||||||
|  |     testing purposes. Django's test framework flushes the contents of | ||||||
|  |     the test database after each test; as a result, any data added | ||||||
|  |     using the custom SQL hook will be lost. | ||||||
|  |  | ||||||
|  |     If you require data for a test case, you should add it using | ||||||
|  |     either a :ref:`test fixture <topics-testing-fixtures>`, or | ||||||
|  |     programatically add it during the ``setUp()`` of your test case. | ||||||
|  |  | ||||||
| Database-backend-specific SQL data | Database-backend-specific SQL data | ||||||
| ---------------------------------- | ---------------------------------- | ||||||
|  |  | ||||||
|   | |||||||
| @@ -410,6 +410,36 @@ Bloggs'``. Although the previous behaviour was not useful for a template languag | |||||||
| designed for web designers, and was never deliberately supported, it is possible | designed for web designers, and was never deliberately supported, it is possible | ||||||
| that some templates may be broken by this change. | that some templates may be broken by this change. | ||||||
|  |  | ||||||
|  | Use of custom SQL to load initial data in tests | ||||||
|  | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|  | Django provides a custom SQL hooks as a way to inject hand-crafted SQL | ||||||
|  | into the database synchronization process. One of the possible uses | ||||||
|  | for this custom SQL is to insert data into your database. If your | ||||||
|  | custom SQL contains ``INSERT`` statements, those insertions will be | ||||||
|  | performed every time your database is synchronized. This includes the | ||||||
|  | synchronization of any test databases that are created when you run a | ||||||
|  | test suite. | ||||||
|  |  | ||||||
|  | However, in the process of testing the Django 1.3, it was discovered | ||||||
|  | that this feature has never completely worked as advertised. When | ||||||
|  | using database backends that don't support transactions, or when using | ||||||
|  | a TransactionTestCase, data that has been inserted using custom SQL | ||||||
|  | will not be visible during the testing process. | ||||||
|  |  | ||||||
|  | Unfortunately, there was no way to rectify this problem without | ||||||
|  | introducing a backwards incompatibility. Rather than leave | ||||||
|  | SQL-inserted initial data in an uncertain state, Django now enforces | ||||||
|  | the policy that data inserted by custom SQL will *not* be visible | ||||||
|  | during testing. | ||||||
|  |  | ||||||
|  | This change only affects the testing process. You can still use custom | ||||||
|  | SQL to load data into your production database as part of the syncdb | ||||||
|  | process. If you require data to exist during test conditions, you | ||||||
|  | should either insert it using :ref:`test fixtures | ||||||
|  | <topics-testing-fixtures>`, or using the ``setUp()`` method of your | ||||||
|  | test case. | ||||||
|  |  | ||||||
| .. _deprecated-features-1.3: | .. _deprecated-features-1.3: | ||||||
|  |  | ||||||
| Features deprecated in 1.3 | Features deprecated in 1.3 | ||||||
|   | |||||||
| @@ -1237,6 +1237,15 @@ documentation<dumpdata>` for more details. | |||||||
|     Fixtures with other names can always be installed manually using |     Fixtures with other names can always be installed manually using | ||||||
|     the :djadmin:`manage.py loaddata<loaddata>` command. |     the :djadmin:`manage.py loaddata<loaddata>` command. | ||||||
|  |  | ||||||
|  | .. admonition:: Initial SQL data and testing | ||||||
|  |  | ||||||
|  |     Django provides a second way to insert initial data into models -- | ||||||
|  |     the :ref:`custom SQL hook <initial-sql>`. However, this technique | ||||||
|  |     *cannot* be used to provide initial data for testing purposes. | ||||||
|  |     Django's test framework flushes the contents of the test database | ||||||
|  |     after each test; as a result, any data added using the custom SQL | ||||||
|  |     hook will be lost. | ||||||
|  |  | ||||||
| Once you've created a fixture and placed it in a ``fixtures`` directory in one | Once you've created a fixture and placed it in a ``fixtures`` directory in one | ||||||
| of your :setting:`INSTALLED_APPS`, you can use it in your unit tests by | of your :setting:`INSTALLED_APPS`, you can use it in your unit tests by | ||||||
| specifying a ``fixtures`` class attribute on your :class:`django.test.TestCase` | specifying a ``fixtures`` class attribute on your :class:`django.test.TestCase` | ||||||
|   | |||||||
| @@ -7,5 +7,3 @@ from django.db import models | |||||||
| class Simple(models.Model): | class Simple(models.Model): | ||||||
|     name = models.CharField(max_length = 50) |     name = models.CharField(max_length = 50) | ||||||
|  |  | ||||||
| # NOTE: The format of the included SQL file for this test suite is important. |  | ||||||
| # It must end with a trailing newline in order to test the fix for #2161. |  | ||||||
|   | |||||||
| @@ -5,4 +5,11 @@ from models import Simple | |||||||
|  |  | ||||||
| class InitialSQLTests(TestCase): | class InitialSQLTests(TestCase): | ||||||
|     def test_initial_sql(self): |     def test_initial_sql(self): | ||||||
|         self.assertEqual(Simple.objects.count(), 7) |         # The format of the included SQL file for this test suite is important. | ||||||
|  |         # It must end with a trailing newline in order to test the fix for #2161. | ||||||
|  |  | ||||||
|  |         # However, as pointed out by #14661, test data loaded by custom SQL | ||||||
|  |         # can't be relied upon; as a result, the test framework flushes the | ||||||
|  |         # data contents before every test. This test validates that this has | ||||||
|  |         # occurred. | ||||||
|  |         self.assertEqual(Simple.objects.count(), 0) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user