summaryrefslogtreecommitdiff
path: root/tests/staticfiles_tests/test_management.py
diff options
context:
space:
mode:
authorMoritz Sichert <moritz.sichert@googlemail.com>2015-07-01 13:43:25 +0200
committerTim Graham <timograham@gmail.com>2015-07-01 09:41:27 -0400
commitb35b43dff81d46e930ffa1e05eb50968b8557102 (patch)
tree7b4bd08113c07e74a6834247a09fd775828a3639 /tests/staticfiles_tests/test_management.py
parent3d7a713156c39ee197ee447130b3e6e078acbadc (diff)
downloaddjango-b35b43dff81d46e930ffa1e05eb50968b8557102.tar.gz
Fixed #24982 -- Split staticfiles tests into multiple files
Diffstat (limited to 'tests/staticfiles_tests/test_management.py')
-rw-r--r--tests/staticfiles_tests/test_management.py341
1 files changed, 341 insertions, 0 deletions
diff --git a/tests/staticfiles_tests/test_management.py b/tests/staticfiles_tests/test_management.py
new file mode 100644
index 0000000000..784e5b74fd
--- /dev/null
+++ b/tests/staticfiles_tests/test_management.py
@@ -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))