mirror of
				https://github.com/django/django.git
				synced 2025-10-31 09:41:08 +00:00 
			
		
		
		
	Fixed #24982 -- Split staticfiles tests into multiple files
This commit is contained in:
		
				
					committed by
					
						 Tim Graham
						Tim Graham
					
				
			
			
				
	
			
			
			
						parent
						
							3d7a713156
						
					
				
				
					commit
					b35b43dff8
				
			
							
								
								
									
										136
									
								
								tests/staticfiles_tests/cases.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								tests/staticfiles_tests/cases.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,136 @@ | |||||||
|  | # -*- encoding: utf-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import codecs | ||||||
|  | import os | ||||||
|  | import shutil | ||||||
|  | import tempfile | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.core.management import call_command | ||||||
|  | from django.template import Context, Template | ||||||
|  | from django.test import SimpleTestCase, override_settings | ||||||
|  | from django.utils import six | ||||||
|  | from django.utils.encoding import force_text | ||||||
|  |  | ||||||
|  | from .settings import TEST_SETTINGS | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseStaticFilesTestCase(object): | ||||||
|  |     """ | ||||||
|  |     Test case with a couple utility assertions. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def assertFileContains(self, filepath, text): | ||||||
|  |         self.assertIn( | ||||||
|  |             text, | ||||||
|  |             self._get_file(force_text(filepath)), | ||||||
|  |             "'%s' not in '%s'" % (text, filepath), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def assertFileNotFound(self, filepath): | ||||||
|  |         self.assertRaises(IOError, self._get_file, filepath) | ||||||
|  |  | ||||||
|  |     def render_template(self, template, **kwargs): | ||||||
|  |         if isinstance(template, six.string_types): | ||||||
|  |             template = Template(template) | ||||||
|  |         return template.render(Context(kwargs)).strip() | ||||||
|  |  | ||||||
|  |     def static_template_snippet(self, path, asvar=False): | ||||||
|  |         if asvar: | ||||||
|  |             return "{%% load static from staticfiles %%}{%% static '%s' as var %%}{{ var }}" % path | ||||||
|  |         return "{%% load static from staticfiles %%}{%% static '%s' %%}" % path | ||||||
|  |  | ||||||
|  |     def assertStaticRenders(self, path, result, asvar=False, **kwargs): | ||||||
|  |         template = self.static_template_snippet(path, asvar) | ||||||
|  |         self.assertEqual(self.render_template(template, **kwargs), result) | ||||||
|  |  | ||||||
|  |     def assertStaticRaises(self, exc, path, result, asvar=False, **kwargs): | ||||||
|  |         self.assertRaises(exc, self.assertStaticRenders, path, result, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override_settings(**TEST_SETTINGS) | ||||||
|  | class StaticFilesTestCase(BaseStaticFilesTestCase, SimpleTestCase): | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseCollectionTestCase(BaseStaticFilesTestCase): | ||||||
|  |     """ | ||||||
|  |     Tests shared by all file finding features (collectstatic, | ||||||
|  |     findstatic, and static serve view). | ||||||
|  |  | ||||||
|  |     This relies on the asserts defined in BaseStaticFilesTestCase, but | ||||||
|  |     is separated because some test cases need those asserts without | ||||||
|  |     all these tests. | ||||||
|  |     """ | ||||||
|  |     def setUp(self): | ||||||
|  |         super(BaseCollectionTestCase, self).setUp() | ||||||
|  |         temp_dir = tempfile.mkdtemp() | ||||||
|  |         # Override the STATIC_ROOT for all tests from setUp to tearDown | ||||||
|  |         # rather than as a context manager | ||||||
|  |         self.patched_settings = self.settings(STATIC_ROOT=temp_dir) | ||||||
|  |         self.patched_settings.enable() | ||||||
|  |         self.run_collectstatic() | ||||||
|  |         # Same comment as in runtests.teardown. | ||||||
|  |         self.addCleanup(shutil.rmtree, six.text_type(temp_dir)) | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         self.patched_settings.disable() | ||||||
|  |         super(BaseCollectionTestCase, self).tearDown() | ||||||
|  |  | ||||||
|  |     def run_collectstatic(self, **kwargs): | ||||||
|  |         call_command('collectstatic', interactive=False, verbosity=0, | ||||||
|  |                      ignore_patterns=['*.ignoreme'], **kwargs) | ||||||
|  |  | ||||||
|  |     def _get_file(self, filepath): | ||||||
|  |         assert filepath, 'filepath is empty.' | ||||||
|  |         filepath = os.path.join(settings.STATIC_ROOT, filepath) | ||||||
|  |         with codecs.open(filepath, "r", "utf-8") as f: | ||||||
|  |             return f.read() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CollectionTestCase(BaseCollectionTestCase, StaticFilesTestCase): | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestDefaults(object): | ||||||
|  |     """ | ||||||
|  |     A few standard test cases. | ||||||
|  |     """ | ||||||
|  |     def test_staticfiles_dirs(self): | ||||||
|  |         """ | ||||||
|  |         Can find a file in a STATICFILES_DIRS directory. | ||||||
|  |         """ | ||||||
|  |         self.assertFileContains('test.txt', 'Can we find') | ||||||
|  |         self.assertFileContains(os.path.join('prefix', 'test.txt'), 'Prefix') | ||||||
|  |  | ||||||
|  |     def test_staticfiles_dirs_subdir(self): | ||||||
|  |         """ | ||||||
|  |         Can find a file in a subdirectory of a STATICFILES_DIRS | ||||||
|  |         directory. | ||||||
|  |         """ | ||||||
|  |         self.assertFileContains('subdir/test.txt', 'Can we find') | ||||||
|  |  | ||||||
|  |     def test_staticfiles_dirs_priority(self): | ||||||
|  |         """ | ||||||
|  |         File in STATICFILES_DIRS has priority over file in app. | ||||||
|  |         """ | ||||||
|  |         self.assertFileContains('test/file.txt', 'STATICFILES_DIRS') | ||||||
|  |  | ||||||
|  |     def test_app_files(self): | ||||||
|  |         """ | ||||||
|  |         Can find a file in an app static/ directory. | ||||||
|  |         """ | ||||||
|  |         self.assertFileContains('test/file1.txt', 'file1 in the app dir') | ||||||
|  |  | ||||||
|  |     def test_nonascii_filenames(self): | ||||||
|  |         """ | ||||||
|  |         Can find a file with non-ASCII character in an app static/ directory. | ||||||
|  |         """ | ||||||
|  |         self.assertFileContains('test/⊗.txt', '⊗ in the app dir') | ||||||
|  |  | ||||||
|  |     def test_camelcase_filenames(self): | ||||||
|  |         """ | ||||||
|  |         Can find a file with capital letters. | ||||||
|  |         """ | ||||||
|  |         self.assertFileContains('test/camelCase.txt', 'camelCase') | ||||||
							
								
								
									
										32
									
								
								tests/staticfiles_tests/settings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								tests/staticfiles_tests/settings.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import os.path | ||||||
|  |  | ||||||
|  | from django.utils._os import upath | ||||||
|  |  | ||||||
|  | TEST_ROOT = os.path.dirname(upath(__file__)) | ||||||
|  |  | ||||||
|  | TESTFILES_PATH = os.path.join(TEST_ROOT, 'apps', 'test', 'static', 'test') | ||||||
|  |  | ||||||
|  | TEST_SETTINGS = { | ||||||
|  |     'DEBUG': True, | ||||||
|  |     'MEDIA_URL': '/media/', | ||||||
|  |     'STATIC_URL': '/static/', | ||||||
|  |     'MEDIA_ROOT': os.path.join(TEST_ROOT, 'project', 'site_media', 'media'), | ||||||
|  |     'STATIC_ROOT': os.path.join(TEST_ROOT, 'project', 'site_media', 'static'), | ||||||
|  |     'STATICFILES_DIRS': [ | ||||||
|  |         os.path.join(TEST_ROOT, 'project', 'documents'), | ||||||
|  |         ('prefix', os.path.join(TEST_ROOT, 'project', 'prefixed')), | ||||||
|  |     ], | ||||||
|  |     'STATICFILES_FINDERS': [ | ||||||
|  |         'django.contrib.staticfiles.finders.FileSystemFinder', | ||||||
|  |         'django.contrib.staticfiles.finders.AppDirectoriesFinder', | ||||||
|  |         'django.contrib.staticfiles.finders.DefaultStorageFinder', | ||||||
|  |     ], | ||||||
|  |     'INSTALLED_APPS': [ | ||||||
|  |         'django.contrib.staticfiles', | ||||||
|  |         'staticfiles_tests', | ||||||
|  |         'staticfiles_tests.apps.test', | ||||||
|  |         'staticfiles_tests.apps.no_label', | ||||||
|  |     ], | ||||||
|  | } | ||||||
							
								
								
									
										118
									
								
								tests/staticfiles_tests/test_finders.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								tests/staticfiles_tests/test_finders.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import os | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.contrib.staticfiles import finders, storage | ||||||
|  | from django.core.exceptions import ImproperlyConfigured | ||||||
|  | from django.test import SimpleTestCase, override_settings | ||||||
|  |  | ||||||
|  | from .cases import StaticFilesTestCase | ||||||
|  | from .settings import TEST_ROOT | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FinderTestCase(object): | ||||||
|  |     """ | ||||||
|  |     Base finder test mixin. | ||||||
|  |  | ||||||
|  |     On Windows, sometimes the case of the path we ask the finders for and the | ||||||
|  |     path(s) they find can differ. Compare them using os.path.normcase() to | ||||||
|  |     avoid false negatives. | ||||||
|  |     """ | ||||||
|  |     def test_find_first(self): | ||||||
|  |         src, dst = self.find_first | ||||||
|  |         found = self.finder.find(src) | ||||||
|  |         self.assertEqual(os.path.normcase(found), os.path.normcase(dst)) | ||||||
|  |  | ||||||
|  |     def test_find_all(self): | ||||||
|  |         src, dst = self.find_all | ||||||
|  |         found = self.finder.find(src, all=True) | ||||||
|  |         found = [os.path.normcase(f) for f in found] | ||||||
|  |         dst = [os.path.normcase(d) for d in dst] | ||||||
|  |         self.assertEqual(found, dst) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestFileSystemFinder(StaticFilesTestCase, FinderTestCase): | ||||||
|  |     """ | ||||||
|  |     Test FileSystemFinder. | ||||||
|  |     """ | ||||||
|  |     def setUp(self): | ||||||
|  |         super(TestFileSystemFinder, self).setUp() | ||||||
|  |         self.finder = finders.FileSystemFinder() | ||||||
|  |         test_file_path = os.path.join(TEST_ROOT, 'project', 'documents', 'test', 'file.txt') | ||||||
|  |         self.find_first = (os.path.join('test', 'file.txt'), test_file_path) | ||||||
|  |         self.find_all = (os.path.join('test', 'file.txt'), [test_file_path]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestAppDirectoriesFinder(StaticFilesTestCase, FinderTestCase): | ||||||
|  |     """ | ||||||
|  |     Test AppDirectoriesFinder. | ||||||
|  |     """ | ||||||
|  |     def setUp(self): | ||||||
|  |         super(TestAppDirectoriesFinder, self).setUp() | ||||||
|  |         self.finder = finders.AppDirectoriesFinder() | ||||||
|  |         test_file_path = os.path.join(TEST_ROOT, 'apps', 'test', 'static', 'test', 'file1.txt') | ||||||
|  |         self.find_first = (os.path.join('test', 'file1.txt'), test_file_path) | ||||||
|  |         self.find_all = (os.path.join('test', 'file1.txt'), [test_file_path]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestDefaultStorageFinder(StaticFilesTestCase, FinderTestCase): | ||||||
|  |     """ | ||||||
|  |     Test DefaultStorageFinder. | ||||||
|  |     """ | ||||||
|  |     def setUp(self): | ||||||
|  |         super(TestDefaultStorageFinder, self).setUp() | ||||||
|  |         self.finder = finders.DefaultStorageFinder( | ||||||
|  |             storage=storage.StaticFilesStorage(location=settings.MEDIA_ROOT)) | ||||||
|  |         test_file_path = os.path.join(settings.MEDIA_ROOT, 'media-file.txt') | ||||||
|  |         self.find_first = ('media-file.txt', test_file_path) | ||||||
|  |         self.find_all = ('media-file.txt', [test_file_path]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override_settings( | ||||||
|  |     STATICFILES_FINDERS=['django.contrib.staticfiles.finders.FileSystemFinder'], | ||||||
|  |     STATICFILES_DIRS=[os.path.join(TEST_ROOT, 'project', 'documents')], | ||||||
|  | ) | ||||||
|  | class TestMiscFinder(SimpleTestCase): | ||||||
|  |     """ | ||||||
|  |     A few misc finder tests. | ||||||
|  |     """ | ||||||
|  |     def test_get_finder(self): | ||||||
|  |         self.assertIsInstance(finders.get_finder( | ||||||
|  |             'django.contrib.staticfiles.finders.FileSystemFinder'), | ||||||
|  |             finders.FileSystemFinder) | ||||||
|  |  | ||||||
|  |     def test_get_finder_bad_classname(self): | ||||||
|  |         with self.assertRaises(ImportError): | ||||||
|  |             finders.get_finder('django.contrib.staticfiles.finders.FooBarFinder') | ||||||
|  |  | ||||||
|  |     def test_get_finder_bad_module(self): | ||||||
|  |         with self.assertRaises(ImportError): | ||||||
|  |             finders.get_finder('foo.bar.FooBarFinder') | ||||||
|  |  | ||||||
|  |     def test_cache(self): | ||||||
|  |         finders.get_finder.cache_clear() | ||||||
|  |         for n in range(10): | ||||||
|  |             finders.get_finder('django.contrib.staticfiles.finders.FileSystemFinder') | ||||||
|  |         cache_info = finders.get_finder.cache_info() | ||||||
|  |         self.assertEqual(cache_info.hits, 9) | ||||||
|  |         self.assertEqual(cache_info.currsize, 1) | ||||||
|  |  | ||||||
|  |     def test_searched_locations(self): | ||||||
|  |         finders.find('spam') | ||||||
|  |         self.assertEqual( | ||||||
|  |             finders.searched_locations, | ||||||
|  |             [os.path.join(TEST_ROOT, 'project', 'documents')] | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @override_settings(STATICFILES_DIRS='a string') | ||||||
|  |     def test_non_tuple_raises_exception(self): | ||||||
|  |         """ | ||||||
|  |         We can't determine if STATICFILES_DIRS is set correctly just by | ||||||
|  |         looking at the type, but we can determine if it's definitely wrong. | ||||||
|  |         """ | ||||||
|  |         self.assertRaises(ImproperlyConfigured, finders.FileSystemFinder) | ||||||
|  |  | ||||||
|  |     @override_settings(MEDIA_ROOT='') | ||||||
|  |     def test_location_empty(self): | ||||||
|  |         self.assertRaises(ImproperlyConfigured, finders.DefaultStorageFinder) | ||||||
							
								
								
									
										341
									
								
								tests/staticfiles_tests/test_management.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										341
									
								
								tests/staticfiles_tests/test_management.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,341 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import codecs | ||||||
|  | import os | ||||||
|  | import shutil | ||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.contrib.staticfiles import storage | ||||||
|  | from django.contrib.staticfiles.management.commands import collectstatic | ||||||
|  | from django.core.exceptions import ImproperlyConfigured | ||||||
|  | from django.core.management import call_command | ||||||
|  | from django.test import override_settings | ||||||
|  | from django.utils import six | ||||||
|  | from django.utils._os import symlinks_supported | ||||||
|  | from django.utils.encoding import force_text | ||||||
|  | from django.utils.functional import empty | ||||||
|  |  | ||||||
|  | from .cases import CollectionTestCase, StaticFilesTestCase, TestDefaults | ||||||
|  | from .settings import TEST_ROOT, TEST_SETTINGS | ||||||
|  | from .storage import DummyStorage | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestNoFilesCreated(object): | ||||||
|  |  | ||||||
|  |     def test_no_files_created(self): | ||||||
|  |         """ | ||||||
|  |         Make sure no files were create in the destination directory. | ||||||
|  |         """ | ||||||
|  |         self.assertEqual(os.listdir(settings.STATIC_ROOT), []) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestFindStatic(CollectionTestCase, TestDefaults): | ||||||
|  |     """ | ||||||
|  |     Test ``findstatic`` management command. | ||||||
|  |     """ | ||||||
|  |     def _get_file(self, filepath): | ||||||
|  |         out = six.StringIO() | ||||||
|  |         call_command('findstatic', filepath, all=False, verbosity=0, stdout=out) | ||||||
|  |         out.seek(0) | ||||||
|  |         lines = [l.strip() for l in out.readlines()] | ||||||
|  |         with codecs.open(force_text(lines[0].strip()), "r", "utf-8") as f: | ||||||
|  |             return f.read() | ||||||
|  |  | ||||||
|  |     def test_all_files(self): | ||||||
|  |         """ | ||||||
|  |         Test that findstatic returns all candidate files if run without --first and -v1. | ||||||
|  |         """ | ||||||
|  |         out = six.StringIO() | ||||||
|  |         call_command('findstatic', 'test/file.txt', verbosity=1, stdout=out) | ||||||
|  |         out.seek(0) | ||||||
|  |         lines = [l.strip() for l in out.readlines()] | ||||||
|  |         self.assertEqual(len(lines), 3)  # three because there is also the "Found <file> here" line | ||||||
|  |         self.assertIn('project', force_text(lines[1])) | ||||||
|  |         self.assertIn('apps', force_text(lines[2])) | ||||||
|  |  | ||||||
|  |     def test_all_files_less_verbose(self): | ||||||
|  |         """ | ||||||
|  |         Test that findstatic returns all candidate files if run without --first and -v0. | ||||||
|  |         """ | ||||||
|  |         out = six.StringIO() | ||||||
|  |         call_command('findstatic', 'test/file.txt', verbosity=0, stdout=out) | ||||||
|  |         out.seek(0) | ||||||
|  |         lines = [l.strip() for l in out.readlines()] | ||||||
|  |         self.assertEqual(len(lines), 2) | ||||||
|  |         self.assertIn('project', force_text(lines[0])) | ||||||
|  |         self.assertIn('apps', force_text(lines[1])) | ||||||
|  |  | ||||||
|  |     def test_all_files_more_verbose(self): | ||||||
|  |         """ | ||||||
|  |         Test that findstatic returns all candidate files if run without --first and -v2. | ||||||
|  |         Also, test that findstatic returns the searched locations with -v2. | ||||||
|  |         """ | ||||||
|  |         out = six.StringIO() | ||||||
|  |         call_command('findstatic', 'test/file.txt', verbosity=2, stdout=out) | ||||||
|  |         out.seek(0) | ||||||
|  |         lines = [l.strip() for l in out.readlines()] | ||||||
|  |         self.assertIn('project', force_text(lines[1])) | ||||||
|  |         self.assertIn('apps', force_text(lines[2])) | ||||||
|  |         self.assertIn("Looking in the following locations:", force_text(lines[3])) | ||||||
|  |         searched_locations = ', '.join(force_text(x) for x in lines[4:]) | ||||||
|  |         # AppDirectoriesFinder searched locations | ||||||
|  |         self.assertIn(os.path.join('staticfiles_tests', 'apps', 'test', 'static'), searched_locations) | ||||||
|  |         self.assertIn(os.path.join('staticfiles_tests', 'apps', 'no_label', 'static'), searched_locations) | ||||||
|  |         # FileSystemFinder searched locations | ||||||
|  |         self.assertIn(TEST_SETTINGS['STATICFILES_DIRS'][1][1], searched_locations) | ||||||
|  |         self.assertIn(TEST_SETTINGS['STATICFILES_DIRS'][0], searched_locations) | ||||||
|  |         # DefaultStorageFinder searched locations | ||||||
|  |         self.assertIn( | ||||||
|  |             os.path.join('staticfiles_tests', 'project', 'site_media', 'media'), | ||||||
|  |             searched_locations | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestConfiguration(StaticFilesTestCase): | ||||||
|  |     def test_location_empty(self): | ||||||
|  |         err = six.StringIO() | ||||||
|  |         for root in ['', None]: | ||||||
|  |             with override_settings(STATIC_ROOT=root): | ||||||
|  |                 with six.assertRaisesRegex( | ||||||
|  |                         self, ImproperlyConfigured, | ||||||
|  |                         'without having set the STATIC_ROOT setting to a filesystem path'): | ||||||
|  |                     call_command('collectstatic', interactive=False, verbosity=0, stderr=err) | ||||||
|  |  | ||||||
|  |     def test_local_storage_detection_helper(self): | ||||||
|  |         staticfiles_storage = storage.staticfiles_storage | ||||||
|  |         try: | ||||||
|  |             storage.staticfiles_storage._wrapped = empty | ||||||
|  |             with self.settings(STATICFILES_STORAGE='django.contrib.staticfiles.storage.StaticFilesStorage'): | ||||||
|  |                 command = collectstatic.Command() | ||||||
|  |                 self.assertTrue(command.is_local_storage()) | ||||||
|  |  | ||||||
|  |             storage.staticfiles_storage._wrapped = empty | ||||||
|  |             with self.settings(STATICFILES_STORAGE='staticfiles_tests.storage.DummyStorage'): | ||||||
|  |                 command = collectstatic.Command() | ||||||
|  |                 self.assertFalse(command.is_local_storage()) | ||||||
|  |  | ||||||
|  |             collectstatic.staticfiles_storage = storage.FileSystemStorage() | ||||||
|  |             command = collectstatic.Command() | ||||||
|  |             self.assertTrue(command.is_local_storage()) | ||||||
|  |  | ||||||
|  |             collectstatic.staticfiles_storage = DummyStorage() | ||||||
|  |             command = collectstatic.Command() | ||||||
|  |             self.assertFalse(command.is_local_storage()) | ||||||
|  |         finally: | ||||||
|  |             staticfiles_storage._wrapped = empty | ||||||
|  |             collectstatic.staticfiles_storage = staticfiles_storage | ||||||
|  |             storage.staticfiles_storage = staticfiles_storage | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestCollection(CollectionTestCase, TestDefaults): | ||||||
|  |     """ | ||||||
|  |     Test ``collectstatic`` management command. | ||||||
|  |     """ | ||||||
|  |     def test_ignore(self): | ||||||
|  |         """ | ||||||
|  |         Test that -i patterns are ignored. | ||||||
|  |         """ | ||||||
|  |         self.assertFileNotFound('test/test.ignoreme') | ||||||
|  |  | ||||||
|  |     def test_common_ignore_patterns(self): | ||||||
|  |         """ | ||||||
|  |         Common ignore patterns (*~, .*, CVS) are ignored. | ||||||
|  |         """ | ||||||
|  |         self.assertFileNotFound('test/.hidden') | ||||||
|  |         self.assertFileNotFound('test/backup~') | ||||||
|  |         self.assertFileNotFound('test/CVS') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestCollectionClear(CollectionTestCase): | ||||||
|  |     """ | ||||||
|  |     Test the ``--clear`` option of the ``collectstatic`` management command. | ||||||
|  |     """ | ||||||
|  |     def run_collectstatic(self, **kwargs): | ||||||
|  |         clear_filepath = os.path.join(settings.STATIC_ROOT, 'cleared.txt') | ||||||
|  |         with open(clear_filepath, 'w') as f: | ||||||
|  |             f.write('should be cleared') | ||||||
|  |         super(TestCollectionClear, self).run_collectstatic(clear=True) | ||||||
|  |  | ||||||
|  |     def test_cleared_not_found(self): | ||||||
|  |         self.assertFileNotFound('cleared.txt') | ||||||
|  |  | ||||||
|  |     def test_dir_not_exists(self, **kwargs): | ||||||
|  |         shutil.rmtree(six.text_type(settings.STATIC_ROOT)) | ||||||
|  |         super(TestCollectionClear, self).run_collectstatic(clear=True) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestCollectionExcludeNoDefaultIgnore(CollectionTestCase, TestDefaults): | ||||||
|  |     """ | ||||||
|  |     Test ``--exclude-dirs`` and ``--no-default-ignore`` options of the | ||||||
|  |     ``collectstatic`` management command. | ||||||
|  |     """ | ||||||
|  |     def run_collectstatic(self): | ||||||
|  |         super(TestCollectionExcludeNoDefaultIgnore, self).run_collectstatic( | ||||||
|  |             use_default_ignore_patterns=False) | ||||||
|  |  | ||||||
|  |     def test_no_common_ignore_patterns(self): | ||||||
|  |         """ | ||||||
|  |         With --no-default-ignore, common ignore patterns (*~, .*, CVS) | ||||||
|  |         are not ignored. | ||||||
|  |         """ | ||||||
|  |         self.assertFileContains('test/.hidden', 'should be ignored') | ||||||
|  |         self.assertFileContains('test/backup~', 'should be ignored') | ||||||
|  |         self.assertFileContains('test/CVS', 'should be ignored') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestCollectionDryRun(CollectionTestCase, TestNoFilesCreated): | ||||||
|  |     """ | ||||||
|  |     Test ``--dry-run`` option for ``collectstatic`` management command. | ||||||
|  |     """ | ||||||
|  |     def run_collectstatic(self): | ||||||
|  |         super(TestCollectionDryRun, self).run_collectstatic(dry_run=True) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestCollectionFilesOverride(CollectionTestCase): | ||||||
|  |     """ | ||||||
|  |     Test overriding duplicated files by ``collectstatic`` management command. | ||||||
|  |     Check for proper handling of apps order in installed apps even if file modification | ||||||
|  |     dates are in different order: | ||||||
|  |         'staticfiles_tests.apps.test', | ||||||
|  |         'staticfiles_tests.apps.no_label', | ||||||
|  |     """ | ||||||
|  |     def setUp(self): | ||||||
|  |         self.orig_path = os.path.join(TEST_ROOT, 'apps', 'no_label', 'static', 'file2.txt') | ||||||
|  |         # get modification and access times for no_label/static/file2.txt | ||||||
|  |         self.orig_mtime = os.path.getmtime(self.orig_path) | ||||||
|  |         self.orig_atime = os.path.getatime(self.orig_path) | ||||||
|  |  | ||||||
|  |         # prepare duplicate of file2.txt from no_label app | ||||||
|  |         # this file will have modification time older than no_label/static/file2.txt | ||||||
|  |         # anyway it should be taken to STATIC_ROOT because 'test' app is before | ||||||
|  |         # 'no_label' app in installed apps | ||||||
|  |         self.testfile_path = os.path.join(TEST_ROOT, 'apps', 'test', 'static', 'file2.txt') | ||||||
|  |         with open(self.testfile_path, 'w+') as f: | ||||||
|  |             f.write('duplicate of file2.txt') | ||||||
|  |         os.utime(self.testfile_path, (self.orig_atime - 1, self.orig_mtime - 1)) | ||||||
|  |         super(TestCollectionFilesOverride, self).setUp() | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         if os.path.exists(self.testfile_path): | ||||||
|  |             os.unlink(self.testfile_path) | ||||||
|  |         # set back original modification time | ||||||
|  |         os.utime(self.orig_path, (self.orig_atime, self.orig_mtime)) | ||||||
|  |         super(TestCollectionFilesOverride, self).tearDown() | ||||||
|  |  | ||||||
|  |     def test_ordering_override(self): | ||||||
|  |         """ | ||||||
|  |         Test if collectstatic takes files in proper order | ||||||
|  |         """ | ||||||
|  |         self.assertFileContains('file2.txt', 'duplicate of file2.txt') | ||||||
|  |  | ||||||
|  |         # run collectstatic again | ||||||
|  |         self.run_collectstatic() | ||||||
|  |  | ||||||
|  |         self.assertFileContains('file2.txt', 'duplicate of file2.txt') | ||||||
|  |  | ||||||
|  |         # and now change modification time of no_label/static/file2.txt | ||||||
|  |         # test app is first in installed apps so file2.txt should remain unmodified | ||||||
|  |         mtime = os.path.getmtime(self.testfile_path) | ||||||
|  |         atime = os.path.getatime(self.testfile_path) | ||||||
|  |         os.utime(self.orig_path, (mtime + 1, atime + 1)) | ||||||
|  |  | ||||||
|  |         # run collectstatic again | ||||||
|  |         self.run_collectstatic() | ||||||
|  |  | ||||||
|  |         self.assertFileContains('file2.txt', 'duplicate of file2.txt') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # The collectstatic test suite already has conflicting files since both | ||||||
|  | # project/test/file.txt and apps/test/static/test/file.txt are collected. To | ||||||
|  | # properly test for the warning not happening unless we tell it to explicitly, | ||||||
|  | # we only include static files from the default finders. | ||||||
|  | @override_settings(STATICFILES_DIRS=[]) | ||||||
|  | class TestCollectionOverwriteWarning(CollectionTestCase): | ||||||
|  |     """ | ||||||
|  |     Test warning in ``collectstatic`` output when a file is skipped because a | ||||||
|  |     previous file was already written to the same path. | ||||||
|  |     """ | ||||||
|  |     # If this string is in the collectstatic output, it means the warning we're | ||||||
|  |     # looking for was emitted. | ||||||
|  |     warning_string = 'Found another file' | ||||||
|  |  | ||||||
|  |     def _collectstatic_output(self, **kwargs): | ||||||
|  |         """ | ||||||
|  |         Run collectstatic, and capture and return the output. We want to run | ||||||
|  |         the command at highest verbosity, which is why we can't | ||||||
|  |         just call e.g. BaseCollectionTestCase.run_collectstatic() | ||||||
|  |         """ | ||||||
|  |         out = six.StringIO() | ||||||
|  |         call_command('collectstatic', interactive=False, verbosity=3, stdout=out, **kwargs) | ||||||
|  |         out.seek(0) | ||||||
|  |         return out.read() | ||||||
|  |  | ||||||
|  |     def test_no_warning(self): | ||||||
|  |         """ | ||||||
|  |         There isn't a warning if there isn't a duplicate destination. | ||||||
|  |         """ | ||||||
|  |         output = self._collectstatic_output(clear=True) | ||||||
|  |         self.assertNotIn(self.warning_string, force_text(output)) | ||||||
|  |  | ||||||
|  |     def test_warning(self): | ||||||
|  |         """ | ||||||
|  |         There is a warning when there are duplicate destinations. | ||||||
|  |         """ | ||||||
|  |         # Create new file in the no_label app that also exists in the test app. | ||||||
|  |         test_dir = os.path.join(TEST_ROOT, 'apps', 'no_label', 'static', 'test') | ||||||
|  |         if not os.path.exists(test_dir): | ||||||
|  |             os.mkdir(test_dir) | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             duplicate_path = os.path.join(test_dir, 'file.txt') | ||||||
|  |             with open(duplicate_path, 'w+') as f: | ||||||
|  |                 f.write('duplicate of file.txt') | ||||||
|  |             output = self._collectstatic_output(clear=True) | ||||||
|  |             self.assertIn(self.warning_string, force_text(output)) | ||||||
|  |         finally: | ||||||
|  |             if os.path.exists(duplicate_path): | ||||||
|  |                 os.unlink(duplicate_path) | ||||||
|  |  | ||||||
|  |         if os.path.exists(test_dir): | ||||||
|  |             os.rmdir(test_dir) | ||||||
|  |  | ||||||
|  |         # Make sure the warning went away again. | ||||||
|  |         output = self._collectstatic_output(clear=True) | ||||||
|  |         self.assertNotIn(self.warning_string, force_text(output)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override_settings(STATICFILES_STORAGE='staticfiles_tests.storage.DummyStorage') | ||||||
|  | class TestCollectionNonLocalStorage(CollectionTestCase, TestNoFilesCreated): | ||||||
|  |     """ | ||||||
|  |     Tests for #15035 | ||||||
|  |     """ | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @unittest.skipUnless(symlinks_supported(), "Must be able to symlink to run this test.") | ||||||
|  | class TestCollectionLinks(CollectionTestCase, TestDefaults): | ||||||
|  |     """ | ||||||
|  |     Test ``--link`` option for ``collectstatic`` management command. | ||||||
|  |  | ||||||
|  |     Note that by inheriting ``TestDefaults`` we repeat all | ||||||
|  |     the standard file resolving tests here, to make sure using | ||||||
|  |     ``--link`` does not change the file-selection semantics. | ||||||
|  |     """ | ||||||
|  |     def run_collectstatic(self): | ||||||
|  |         super(TestCollectionLinks, self).run_collectstatic(link=True) | ||||||
|  |  | ||||||
|  |     def test_links_created(self): | ||||||
|  |         """ | ||||||
|  |         With ``--link``, symbolic links are created. | ||||||
|  |         """ | ||||||
|  |         self.assertTrue(os.path.islink(os.path.join(settings.STATIC_ROOT, 'test.txt'))) | ||||||
|  |  | ||||||
|  |     def test_broken_symlink(self): | ||||||
|  |         """ | ||||||
|  |         Test broken symlink gets deleted. | ||||||
|  |         """ | ||||||
|  |         path = os.path.join(settings.STATIC_ROOT, 'test.txt') | ||||||
|  |         os.unlink(path) | ||||||
|  |         self.run_collectstatic() | ||||||
|  |         self.assertTrue(os.path.islink(path)) | ||||||
							
								
								
									
										416
									
								
								tests/staticfiles_tests/test_storage.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										416
									
								
								tests/staticfiles_tests/test_storage.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,416 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import os | ||||||
|  | import sys | ||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.contrib.staticfiles import finders, storage | ||||||
|  | from django.contrib.staticfiles.management.commands import collectstatic | ||||||
|  | from django.contrib.staticfiles.management.commands.collectstatic import \ | ||||||
|  |     Command as CollectstaticCommand | ||||||
|  | from django.core.cache.backends.base import BaseCache | ||||||
|  | from django.core.management import call_command | ||||||
|  | from django.test import SimpleTestCase, override_settings | ||||||
|  | from django.utils import six | ||||||
|  | from django.utils.encoding import force_text | ||||||
|  |  | ||||||
|  | from .cases import ( | ||||||
|  |     BaseCollectionTestCase, BaseStaticFilesTestCase, StaticFilesTestCase, | ||||||
|  | ) | ||||||
|  | from .settings import TEST_ROOT, TEST_SETTINGS, TESTFILES_PATH | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def hashed_file_path(test, path): | ||||||
|  |     fullpath = test.render_template(test.static_template_snippet(path)) | ||||||
|  |     return fullpath.replace(settings.STATIC_URL, '') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestHashedFiles(object): | ||||||
|  |     hashed_file_path = hashed_file_path | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         # Clear hashed files to avoid side effects among tests. | ||||||
|  |         storage.staticfiles_storage.hashed_files.clear() | ||||||
|  |  | ||||||
|  |     def test_template_tag_return(self): | ||||||
|  |         """ | ||||||
|  |         Test the CachedStaticFilesStorage backend. | ||||||
|  |         """ | ||||||
|  |         self.assertStaticRaises(ValueError, "does/not/exist.png", "/static/does/not/exist.png") | ||||||
|  |         self.assertStaticRenders("test/file.txt", "/static/test/file.dad0999e4f8f.txt") | ||||||
|  |         self.assertStaticRenders("test/file.txt", "/static/test/file.dad0999e4f8f.txt", asvar=True) | ||||||
|  |         self.assertStaticRenders("cached/styles.css", "/static/cached/styles.bb84a0240107.css") | ||||||
|  |         self.assertStaticRenders("path/", "/static/path/") | ||||||
|  |         self.assertStaticRenders("path/?query", "/static/path/?query") | ||||||
|  |  | ||||||
|  |     def test_template_tag_simple_content(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/styles.css") | ||||||
|  |         self.assertEqual(relpath, "cached/styles.bb84a0240107.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b"cached/other.css", content) | ||||||
|  |             self.assertIn(b"other.d41d8cd98f00.css", content) | ||||||
|  |  | ||||||
|  |     def test_path_ignored_completely(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/css/ignored.css") | ||||||
|  |         self.assertEqual(relpath, "cached/css/ignored.6c77f2643390.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertIn(b'#foobar', content) | ||||||
|  |             self.assertIn(b'http:foobar', content) | ||||||
|  |             self.assertIn(b'https:foobar', content) | ||||||
|  |             self.assertIn(b'data:foobar', content) | ||||||
|  |             self.assertIn(b'//foobar', content) | ||||||
|  |  | ||||||
|  |     def test_path_with_querystring(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/styles.css?spam=eggs") | ||||||
|  |         self.assertEqual(relpath, "cached/styles.bb84a0240107.css?spam=eggs") | ||||||
|  |         with storage.staticfiles_storage.open( | ||||||
|  |                 "cached/styles.bb84a0240107.css") as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b"cached/other.css", content) | ||||||
|  |             self.assertIn(b"other.d41d8cd98f00.css", content) | ||||||
|  |  | ||||||
|  |     def test_path_with_fragment(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/styles.css#eggs") | ||||||
|  |         self.assertEqual(relpath, "cached/styles.bb84a0240107.css#eggs") | ||||||
|  |         with storage.staticfiles_storage.open( | ||||||
|  |                 "cached/styles.bb84a0240107.css") as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b"cached/other.css", content) | ||||||
|  |             self.assertIn(b"other.d41d8cd98f00.css", content) | ||||||
|  |  | ||||||
|  |     def test_path_with_querystring_and_fragment(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/css/fragments.css") | ||||||
|  |         self.assertEqual(relpath, "cached/css/fragments.75433540b096.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertIn(b'fonts/font.a4b0478549d0.eot?#iefix', content) | ||||||
|  |             self.assertIn(b'fonts/font.b8d603e42714.svg#webfontIyfZbseF', content) | ||||||
|  |             self.assertIn(b'data:font/woff;charset=utf-8;base64,d09GRgABAAAAADJoAA0AAAAAR2QAAQAAAAAAAAAAAAA', content) | ||||||
|  |             self.assertIn(b'#default#VML', content) | ||||||
|  |  | ||||||
|  |     def test_template_tag_absolute(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/absolute.css") | ||||||
|  |         self.assertEqual(relpath, "cached/absolute.ae9ef2716fe3.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b"/static/cached/styles.css", content) | ||||||
|  |             self.assertIn(b"/static/cached/styles.bb84a0240107.css", content) | ||||||
|  |             self.assertIn(b'/static/cached/img/relative.acae32e4532b.png', content) | ||||||
|  |  | ||||||
|  |     def test_template_tag_denorm(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/denorm.css") | ||||||
|  |         self.assertEqual(relpath, "cached/denorm.c5bd139ad821.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b"..//cached///styles.css", content) | ||||||
|  |             self.assertIn(b"../cached/styles.bb84a0240107.css", content) | ||||||
|  |             self.assertNotIn(b"url(img/relative.png )", content) | ||||||
|  |             self.assertIn(b'url("img/relative.acae32e4532b.png', content) | ||||||
|  |  | ||||||
|  |     def test_template_tag_relative(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/relative.css") | ||||||
|  |         self.assertEqual(relpath, "cached/relative.b0375bd89156.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b"../cached/styles.css", content) | ||||||
|  |             self.assertNotIn(b'@import "styles.css"', content) | ||||||
|  |             self.assertNotIn(b'url(img/relative.png)', content) | ||||||
|  |             self.assertIn(b'url("img/relative.acae32e4532b.png")', content) | ||||||
|  |             self.assertIn(b"../cached/styles.bb84a0240107.css", content) | ||||||
|  |  | ||||||
|  |     def test_import_replacement(self): | ||||||
|  |         "See #18050" | ||||||
|  |         relpath = self.hashed_file_path("cached/import.css") | ||||||
|  |         self.assertEqual(relpath, "cached/import.2b1d40b0bbd4.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             self.assertIn(b"""import url("styles.bb84a0240107.css")""", relfile.read()) | ||||||
|  |  | ||||||
|  |     def test_template_tag_deep_relative(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/css/window.css") | ||||||
|  |         self.assertEqual(relpath, "cached/css/window.3906afbb5a17.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b'url(img/window.png)', content) | ||||||
|  |             self.assertIn(b'url("img/window.acae32e4532b.png")', content) | ||||||
|  |  | ||||||
|  |     def test_template_tag_url(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/url.css") | ||||||
|  |         self.assertEqual(relpath, "cached/url.902310b73412.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             self.assertIn(b"https://", relfile.read()) | ||||||
|  |  | ||||||
|  |     def test_post_processing(self): | ||||||
|  |         """ | ||||||
|  |         Test that post_processing behaves correctly. | ||||||
|  |  | ||||||
|  |         Files that are alterable should always be post-processed; files that | ||||||
|  |         aren't should be skipped. | ||||||
|  |  | ||||||
|  |         collectstatic has already been called once in setUp() for this testcase, | ||||||
|  |         therefore we check by verifying behavior on a second run. | ||||||
|  |         """ | ||||||
|  |         collectstatic_args = { | ||||||
|  |             'interactive': False, | ||||||
|  |             'verbosity': 0, | ||||||
|  |             'link': False, | ||||||
|  |             'clear': False, | ||||||
|  |             'dry_run': False, | ||||||
|  |             'post_process': True, | ||||||
|  |             'use_default_ignore_patterns': True, | ||||||
|  |             'ignore_patterns': ['*.ignoreme'], | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         collectstatic_cmd = CollectstaticCommand() | ||||||
|  |         collectstatic_cmd.set_options(**collectstatic_args) | ||||||
|  |         stats = collectstatic_cmd.collect() | ||||||
|  |         self.assertIn(os.path.join('cached', 'css', 'window.css'), stats['post_processed']) | ||||||
|  |         self.assertIn(os.path.join('cached', 'css', 'img', 'window.png'), stats['unmodified']) | ||||||
|  |         self.assertIn(os.path.join('test', 'nonascii.css'), stats['post_processed']) | ||||||
|  |  | ||||||
|  |     def test_css_import_case_insensitive(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/styles_insensitive.css") | ||||||
|  |         self.assertEqual(relpath, "cached/styles_insensitive.c609562b6d3c.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b"cached/other.css", content) | ||||||
|  |             self.assertIn(b"other.d41d8cd98f00.css", content) | ||||||
|  |  | ||||||
|  |     @override_settings( | ||||||
|  |         STATICFILES_DIRS=[os.path.join(TEST_ROOT, 'project', 'faulty')], | ||||||
|  |         STATICFILES_FINDERS=['django.contrib.staticfiles.finders.FileSystemFinder'], | ||||||
|  |     ) | ||||||
|  |     def test_post_processing_failure(self): | ||||||
|  |         """ | ||||||
|  |         Test that post_processing indicates the origin of the error when it | ||||||
|  |         fails. Regression test for #18986. | ||||||
|  |         """ | ||||||
|  |         finders.get_finder.cache_clear() | ||||||
|  |         err = six.StringIO() | ||||||
|  |         with self.assertRaises(Exception): | ||||||
|  |             call_command('collectstatic', interactive=False, verbosity=0, stderr=err) | ||||||
|  |         self.assertEqual("Post-processing 'faulty.css' failed!\n\n", err.getvalue()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # we set DEBUG to False here since the template tag wouldn't work otherwise | ||||||
|  | @override_settings(**dict( | ||||||
|  |     TEST_SETTINGS, | ||||||
|  |     STATICFILES_STORAGE='django.contrib.staticfiles.storage.CachedStaticFilesStorage', | ||||||
|  |     DEBUG=False, | ||||||
|  | )) | ||||||
|  | class TestCollectionCachedStorage(TestHashedFiles, BaseCollectionTestCase, | ||||||
|  |         BaseStaticFilesTestCase, SimpleTestCase): | ||||||
|  |     """ | ||||||
|  |     Tests for the Cache busting storage | ||||||
|  |     """ | ||||||
|  |     def test_cache_invalidation(self): | ||||||
|  |         name = "cached/styles.css" | ||||||
|  |         hashed_name = "cached/styles.bb84a0240107.css" | ||||||
|  |         # check if the cache is filled correctly as expected | ||||||
|  |         cache_key = storage.staticfiles_storage.hash_key(name) | ||||||
|  |         cached_name = storage.staticfiles_storage.hashed_files.get(cache_key) | ||||||
|  |         self.assertEqual(self.hashed_file_path(name), cached_name) | ||||||
|  |         # clearing the cache to make sure we re-set it correctly in the url method | ||||||
|  |         storage.staticfiles_storage.hashed_files.clear() | ||||||
|  |         cached_name = storage.staticfiles_storage.hashed_files.get(cache_key) | ||||||
|  |         self.assertEqual(cached_name, None) | ||||||
|  |         self.assertEqual(self.hashed_file_path(name), hashed_name) | ||||||
|  |         cached_name = storage.staticfiles_storage.hashed_files.get(cache_key) | ||||||
|  |         self.assertEqual(cached_name, hashed_name) | ||||||
|  |  | ||||||
|  |     def test_cache_key_memcache_validation(self): | ||||||
|  |         """ | ||||||
|  |         Handle cache key creation correctly, see #17861. | ||||||
|  |         """ | ||||||
|  |         name = ( | ||||||
|  |             "/some crazy/long filename/ with spaces Here and ?#%#$/other/stuff" | ||||||
|  |             "/some crazy/long filename/ with spaces Here and ?#%#$/other/stuff" | ||||||
|  |             "/some crazy/long filename/ with spaces Here and ?#%#$/other/stuff" | ||||||
|  |             "/some crazy/long filename/ with spaces Here and ?#%#$/other/stuff" | ||||||
|  |             "/some crazy/long filename/ with spaces Here and ?#%#$/other/stuff" | ||||||
|  |             "/some crazy/\x16\xb4" | ||||||
|  |         ) | ||||||
|  |         cache_key = storage.staticfiles_storage.hash_key(name) | ||||||
|  |         cache_validator = BaseCache({}) | ||||||
|  |         cache_validator.validate_key(cache_key) | ||||||
|  |         self.assertEqual(cache_key, 'staticfiles:821ea71ef36f95b3922a77f7364670e7') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # we set DEBUG to False here since the template tag wouldn't work otherwise | ||||||
|  | @override_settings(**dict( | ||||||
|  |     TEST_SETTINGS, | ||||||
|  |     STATICFILES_STORAGE='django.contrib.staticfiles.storage.ManifestStaticFilesStorage', | ||||||
|  |     DEBUG=False, | ||||||
|  | )) | ||||||
|  | class TestCollectionManifestStorage(TestHashedFiles, BaseCollectionTestCase, | ||||||
|  |         BaseStaticFilesTestCase, SimpleTestCase): | ||||||
|  |     """ | ||||||
|  |     Tests for the Cache busting storage | ||||||
|  |     """ | ||||||
|  |     def setUp(self): | ||||||
|  |         super(TestCollectionManifestStorage, self).setUp() | ||||||
|  |  | ||||||
|  |         self._clear_filename = os.path.join(TESTFILES_PATH, 'cleared.txt') | ||||||
|  |         with open(self._clear_filename, 'w') as f: | ||||||
|  |             f.write('to be deleted in one test') | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         super(TestCollectionManifestStorage, self).tearDown() | ||||||
|  |         if os.path.exists(self._clear_filename): | ||||||
|  |             os.unlink(self._clear_filename) | ||||||
|  |  | ||||||
|  |     def test_manifest_exists(self): | ||||||
|  |         filename = storage.staticfiles_storage.manifest_name | ||||||
|  |         path = storage.staticfiles_storage.path(filename) | ||||||
|  |         self.assertTrue(os.path.exists(path)) | ||||||
|  |  | ||||||
|  |     def test_loaded_cache(self): | ||||||
|  |         self.assertNotEqual(storage.staticfiles_storage.hashed_files, {}) | ||||||
|  |         manifest_content = storage.staticfiles_storage.read_manifest() | ||||||
|  |         self.assertIn( | ||||||
|  |             '"version": "%s"' % storage.staticfiles_storage.manifest_version, | ||||||
|  |             force_text(manifest_content) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def test_parse_cache(self): | ||||||
|  |         hashed_files = storage.staticfiles_storage.hashed_files | ||||||
|  |         manifest = storage.staticfiles_storage.load_manifest() | ||||||
|  |         self.assertEqual(hashed_files, manifest) | ||||||
|  |  | ||||||
|  |     def test_clear_empties_manifest(self): | ||||||
|  |         cleared_file_name = os.path.join('test', 'cleared.txt') | ||||||
|  |         # collect the additional file | ||||||
|  |         self.run_collectstatic() | ||||||
|  |  | ||||||
|  |         hashed_files = storage.staticfiles_storage.hashed_files | ||||||
|  |         self.assertIn(cleared_file_name, hashed_files) | ||||||
|  |  | ||||||
|  |         manifest_content = storage.staticfiles_storage.load_manifest() | ||||||
|  |         self.assertIn(cleared_file_name, manifest_content) | ||||||
|  |  | ||||||
|  |         original_path = storage.staticfiles_storage.path(cleared_file_name) | ||||||
|  |         self.assertTrue(os.path.exists(original_path)) | ||||||
|  |  | ||||||
|  |         # delete the original file form the app, collect with clear | ||||||
|  |         os.unlink(self._clear_filename) | ||||||
|  |         self.run_collectstatic(clear=True) | ||||||
|  |  | ||||||
|  |         self.assertFileNotFound(original_path) | ||||||
|  |  | ||||||
|  |         hashed_files = storage.staticfiles_storage.hashed_files | ||||||
|  |         self.assertNotIn(cleared_file_name, hashed_files) | ||||||
|  |  | ||||||
|  |         manifest_content = storage.staticfiles_storage.load_manifest() | ||||||
|  |         self.assertNotIn(cleared_file_name, manifest_content) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # we set DEBUG to False here since the template tag wouldn't work otherwise | ||||||
|  | @override_settings(**dict( | ||||||
|  |     TEST_SETTINGS, | ||||||
|  |     STATICFILES_STORAGE='staticfiles_tests.storage.SimpleCachedStaticFilesStorage', | ||||||
|  |     DEBUG=False, | ||||||
|  | )) | ||||||
|  | class TestCollectionSimpleCachedStorage(BaseCollectionTestCase, | ||||||
|  |         BaseStaticFilesTestCase, SimpleTestCase): | ||||||
|  |     """ | ||||||
|  |     Tests for the Cache busting storage | ||||||
|  |     """ | ||||||
|  |     hashed_file_path = hashed_file_path | ||||||
|  |  | ||||||
|  |     def test_template_tag_return(self): | ||||||
|  |         """ | ||||||
|  |         Test the CachedStaticFilesStorage backend. | ||||||
|  |         """ | ||||||
|  |         self.assertStaticRaises(ValueError, "does/not/exist.png", "/static/does/not/exist.png") | ||||||
|  |         self.assertStaticRenders("test/file.txt", "/static/test/file.deploy12345.txt") | ||||||
|  |         self.assertStaticRenders("cached/styles.css", "/static/cached/styles.deploy12345.css") | ||||||
|  |         self.assertStaticRenders("path/", "/static/path/") | ||||||
|  |         self.assertStaticRenders("path/?query", "/static/path/?query") | ||||||
|  |  | ||||||
|  |     def test_template_tag_simple_content(self): | ||||||
|  |         relpath = self.hashed_file_path("cached/styles.css") | ||||||
|  |         self.assertEqual(relpath, "cached/styles.deploy12345.css") | ||||||
|  |         with storage.staticfiles_storage.open(relpath) as relfile: | ||||||
|  |             content = relfile.read() | ||||||
|  |             self.assertNotIn(b"cached/other.css", content) | ||||||
|  |             self.assertIn(b"other.deploy12345.css", content) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CustomStaticFilesStorage(storage.StaticFilesStorage): | ||||||
|  |     """ | ||||||
|  |     Used in TestStaticFilePermissions | ||||||
|  |     """ | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         kwargs['file_permissions_mode'] = 0o640 | ||||||
|  |         kwargs['directory_permissions_mode'] = 0o740 | ||||||
|  |         super(CustomStaticFilesStorage, self).__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @unittest.skipIf(sys.platform.startswith('win'), "Windows only partially supports chmod.") | ||||||
|  | class TestStaticFilePermissions(BaseCollectionTestCase, StaticFilesTestCase): | ||||||
|  |  | ||||||
|  |     command_params = { | ||||||
|  |         'interactive': False, | ||||||
|  |         'post_process': True, | ||||||
|  |         'verbosity': 0, | ||||||
|  |         'ignore_patterns': ['*.ignoreme'], | ||||||
|  |         'use_default_ignore_patterns': True, | ||||||
|  |         'clear': False, | ||||||
|  |         'link': False, | ||||||
|  |         'dry_run': False, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         self.umask = 0o027 | ||||||
|  |         self.old_umask = os.umask(self.umask) | ||||||
|  |         super(TestStaticFilePermissions, self).setUp() | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         os.umask(self.old_umask) | ||||||
|  |         super(TestStaticFilePermissions, self).tearDown() | ||||||
|  |  | ||||||
|  |     # Don't run collectstatic command in this test class. | ||||||
|  |     def run_collectstatic(self, **kwargs): | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     @override_settings( | ||||||
|  |         FILE_UPLOAD_PERMISSIONS=0o655, | ||||||
|  |         FILE_UPLOAD_DIRECTORY_PERMISSIONS=0o765, | ||||||
|  |     ) | ||||||
|  |     def test_collect_static_files_permissions(self): | ||||||
|  |         collectstatic.Command().execute(**self.command_params) | ||||||
|  |         test_file = os.path.join(settings.STATIC_ROOT, "test.txt") | ||||||
|  |         test_dir = os.path.join(settings.STATIC_ROOT, "subdir") | ||||||
|  |         file_mode = os.stat(test_file)[0] & 0o777 | ||||||
|  |         dir_mode = os.stat(test_dir)[0] & 0o777 | ||||||
|  |         self.assertEqual(file_mode, 0o655) | ||||||
|  |         self.assertEqual(dir_mode, 0o765) | ||||||
|  |  | ||||||
|  |     @override_settings( | ||||||
|  |         FILE_UPLOAD_PERMISSIONS=None, | ||||||
|  |         FILE_UPLOAD_DIRECTORY_PERMISSIONS=None, | ||||||
|  |     ) | ||||||
|  |     def test_collect_static_files_default_permissions(self): | ||||||
|  |         collectstatic.Command().execute(**self.command_params) | ||||||
|  |         test_file = os.path.join(settings.STATIC_ROOT, "test.txt") | ||||||
|  |         test_dir = os.path.join(settings.STATIC_ROOT, "subdir") | ||||||
|  |         file_mode = os.stat(test_file)[0] & 0o777 | ||||||
|  |         dir_mode = os.stat(test_dir)[0] & 0o777 | ||||||
|  |         self.assertEqual(file_mode, 0o666 & ~self.umask) | ||||||
|  |         self.assertEqual(dir_mode, 0o777 & ~self.umask) | ||||||
|  |  | ||||||
|  |     @override_settings( | ||||||
|  |         FILE_UPLOAD_PERMISSIONS=0o655, | ||||||
|  |         FILE_UPLOAD_DIRECTORY_PERMISSIONS=0o765, | ||||||
|  |         STATICFILES_STORAGE='staticfiles_tests.test_storage.CustomStaticFilesStorage', | ||||||
|  |     ) | ||||||
|  |     def test_collect_static_files_subclass_of_static_storage(self): | ||||||
|  |         collectstatic.Command().execute(**self.command_params) | ||||||
|  |         test_file = os.path.join(settings.STATIC_ROOT, "test.txt") | ||||||
|  |         test_dir = os.path.join(settings.STATIC_ROOT, "subdir") | ||||||
|  |         file_mode = os.stat(test_file)[0] & 0o777 | ||||||
|  |         dir_mode = os.stat(test_dir)[0] & 0o777 | ||||||
|  |         self.assertEqual(file_mode, 0o640) | ||||||
|  |         self.assertEqual(dir_mode, 0o740) | ||||||
							
								
								
									
										10
									
								
								tests/staticfiles_tests/test_templatetags.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								tests/staticfiles_tests/test_templatetags.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | from .cases import StaticFilesTestCase | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestTemplateTag(StaticFilesTestCase): | ||||||
|  |  | ||||||
|  |     def test_template_tag(self): | ||||||
|  |         self.assertStaticRenders("does/not/exist.png", "/static/does/not/exist.png") | ||||||
|  |         self.assertStaticRenders("testfile.txt", "/static/testfile.txt") | ||||||
							
								
								
									
										46
									
								
								tests/staticfiles_tests/test_views.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								tests/staticfiles_tests/test_views.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import posixpath | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.test import override_settings | ||||||
|  |  | ||||||
|  | from .cases import StaticFilesTestCase, TestDefaults | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override_settings(ROOT_URLCONF='staticfiles_tests.urls.default') | ||||||
|  | class TestServeStatic(StaticFilesTestCase): | ||||||
|  |     """ | ||||||
|  |     Test static asset serving view. | ||||||
|  |     """ | ||||||
|  |     def _response(self, filepath): | ||||||
|  |         return self.client.get( | ||||||
|  |             posixpath.join(settings.STATIC_URL, filepath)) | ||||||
|  |  | ||||||
|  |     def assertFileContains(self, filepath, text): | ||||||
|  |         self.assertContains(self._response(filepath), text) | ||||||
|  |  | ||||||
|  |     def assertFileNotFound(self, filepath): | ||||||
|  |         self.assertEqual(self._response(filepath).status_code, 404) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override_settings(DEBUG=False) | ||||||
|  | class TestServeDisabled(TestServeStatic): | ||||||
|  |     """ | ||||||
|  |     Test serving static files disabled when DEBUG is False. | ||||||
|  |     """ | ||||||
|  |     def test_disabled_serving(self): | ||||||
|  |         self.assertFileNotFound('test.txt') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestServeStaticWithDefaultURL(TestServeStatic, TestDefaults): | ||||||
|  |     """ | ||||||
|  |     Test static asset serving view with manually configured URLconf. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @override_settings(ROOT_URLCONF='staticfiles_tests.urls.helper') | ||||||
|  | class TestServeStaticWithURLHelper(TestServeStatic, TestDefaults): | ||||||
|  |     """ | ||||||
|  |     Test static asset serving view with staticfiles_urlpatterns helper. | ||||||
|  |     """ | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user