123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677 |
- import json
- import os
- import shutil
- import sys
- import tempfile
- import unittest
- from io import StringIO
- from pathlib import Path
- from unittest import mock
- from django.conf import settings
- from django.contrib.staticfiles import finders, storage
- from django.contrib.staticfiles.management.commands.collectstatic import (
- Command as CollectstaticCommand,
- )
- from django.core.management import call_command
- from django.test import SimpleTestCase, override_settings
- from .cases import CollectionTestCase
- from .settings import TEST_ROOT
- def hashed_file_path(test, path):
- fullpath = test.render_template(test.static_template_snippet(path))
- return fullpath.replace(settings.STATIC_URL, '')
- class TestHashedFiles:
- 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 assertPostCondition(self):
- """
- Assert post conditions for a test are met. Must be manually called at
- the end of each test.
- """
- pass
- def test_template_tag_return(self):
- 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.5e0040571e1a.css")
- self.assertStaticRenders("path/", "/static/path/")
- self.assertStaticRenders("path/?query", "/static/path/?query")
- self.assertPostCondition()
- def test_template_tag_simple_content(self):
- relpath = self.hashed_file_path("cached/styles.css")
- self.assertEqual(relpath, "cached/styles.5e0040571e1a.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)
- self.assertPostCondition()
- def test_path_ignored_completely(self):
- relpath = self.hashed_file_path("cached/css/ignored.css")
- self.assertEqual(relpath, "cached/css/ignored.554da52152af.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'chrome:foobar', content)
- self.assertIn(b'//foobar', content)
- self.assertPostCondition()
- def test_path_with_querystring(self):
- relpath = self.hashed_file_path("cached/styles.css?spam=eggs")
- self.assertEqual(relpath, "cached/styles.5e0040571e1a.css?spam=eggs")
- with storage.staticfiles_storage.open("cached/styles.5e0040571e1a.css") as relfile:
- content = relfile.read()
- self.assertNotIn(b"cached/other.css", content)
- self.assertIn(b"other.d41d8cd98f00.css", content)
- self.assertPostCondition()
- def test_path_with_fragment(self):
- relpath = self.hashed_file_path("cached/styles.css#eggs")
- self.assertEqual(relpath, "cached/styles.5e0040571e1a.css#eggs")
- with storage.staticfiles_storage.open("cached/styles.5e0040571e1a.css") as relfile:
- content = relfile.read()
- self.assertNotIn(b"cached/other.css", content)
- self.assertIn(b"other.d41d8cd98f00.css", content)
- self.assertPostCondition()
- def test_path_with_querystring_and_fragment(self):
- relpath = self.hashed_file_path("cached/css/fragments.css")
- self.assertEqual(relpath, "cached/css/fragments.a60c0e74834f.css")
- with storage.staticfiles_storage.open(relpath) as relfile:
- content = relfile.read()
- self.assertIn(b'fonts/font.b9b105392eb8.eot?#iefix', content)
- self.assertIn(b'fonts/font.b8d603e42714.svg#webfontIyfZbseF', content)
- self.assertIn(b'fonts/font.b8d603e42714.svg#path/to/../../fonts/font.svg', content)
- self.assertIn(b'data:font/woff;charset=utf-8;base64,d09GRgABAAAAADJoAA0AAAAAR2QAAQAAAAAAAAAAAAA', content)
- self.assertIn(b'#default#VML', content)
- self.assertPostCondition()
- def test_template_tag_absolute(self):
- relpath = self.hashed_file_path("cached/absolute.css")
- self.assertEqual(relpath, "cached/absolute.eb04def9f9a4.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.5e0040571e1a.css", content)
- self.assertNotIn(b"/static/styles_root.css", content)
- self.assertIn(b"/static/styles_root.401f2509a628.css", content)
- self.assertIn(b'/static/cached/img/relative.acae32e4532b.png', content)
- self.assertPostCondition()
- def test_template_tag_absolute_root(self):
- """
- Like test_template_tag_absolute, but for a file in STATIC_ROOT (#26249).
- """
- relpath = self.hashed_file_path("absolute_root.css")
- self.assertEqual(relpath, "absolute_root.f821df1b64f7.css")
- with storage.staticfiles_storage.open(relpath) as relfile:
- content = relfile.read()
- self.assertNotIn(b"/static/styles_root.css", content)
- self.assertIn(b"/static/styles_root.401f2509a628.css", content)
- self.assertPostCondition()
- def test_template_tag_relative(self):
- relpath = self.hashed_file_path("cached/relative.css")
- self.assertEqual(relpath, "cached/relative.c3e9e1ea6f2e.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.5e0040571e1a.css", content)
- self.assertPostCondition()
- def test_import_replacement(self):
- "See #18050"
- relpath = self.hashed_file_path("cached/import.css")
- self.assertEqual(relpath, "cached/import.f53576679e5a.css")
- with storage.staticfiles_storage.open(relpath) as relfile:
- self.assertIn(b"""import url("styles.5e0040571e1a.css")""", relfile.read())
- self.assertPostCondition()
- def test_template_tag_deep_relative(self):
- relpath = self.hashed_file_path("cached/css/window.css")
- self.assertEqual(relpath, "cached/css/window.5d5c10836967.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)
- self.assertPostCondition()
- 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())
- self.assertPostCondition()
- @override_settings(
- STATICFILES_DIRS=[os.path.join(TEST_ROOT, 'project', 'loop')],
- STATICFILES_FINDERS=['django.contrib.staticfiles.finders.FileSystemFinder'],
- )
- def test_import_loop(self):
- finders.get_finder.cache_clear()
- err = StringIO()
- with self.assertRaisesMessage(RuntimeError, 'Max post-process passes exceeded'):
- call_command('collectstatic', interactive=False, verbosity=0, stderr=err)
- self.assertEqual("Post-processing 'All' failed!\n\n", err.getvalue())
- self.assertPostCondition()
- def test_post_processing(self):
- """
- 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'])
- # No file should be yielded twice.
- self.assertCountEqual(stats['post_processed'], set(stats['post_processed']))
- self.assertPostCondition()
- def test_css_import_case_insensitive(self):
- relpath = self.hashed_file_path("cached/styles_insensitive.css")
- self.assertEqual(relpath, "cached/styles_insensitive.3fa427592a53.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)
- self.assertPostCondition()
- def test_css_source_map(self):
- relpath = self.hashed_file_path('cached/source_map.css')
- self.assertEqual(relpath, 'cached/source_map.b2fceaf426aa.css')
- with storage.staticfiles_storage.open(relpath) as relfile:
- content = relfile.read()
- self.assertNotIn(b'/*# sourceMappingURL=source_map.css.map */', content)
- self.assertIn(
- b'/*# sourceMappingURL=source_map.css.99914b932bd3.map */',
- content,
- )
- self.assertPostCondition()
- def test_css_source_map_sensitive(self):
- relpath = self.hashed_file_path('cached/source_map_sensitive.css')
- self.assertEqual(relpath, 'cached/source_map_sensitive.456683f2106f.css')
- with storage.staticfiles_storage.open(relpath) as relfile:
- content = relfile.read()
- self.assertIn(b'/*# sOuRcEMaPpInGURL=source_map.css.map */', content)
- self.assertNotIn(
- b'/*# sourceMappingURL=source_map.css.99914b932bd3.map */',
- content,
- )
- self.assertPostCondition()
- def test_js_source_map(self):
- relpath = self.hashed_file_path('cached/source_map.js')
- self.assertEqual(relpath, 'cached/source_map.cd45b8534a87.js')
- with storage.staticfiles_storage.open(relpath) as relfile:
- content = relfile.read()
- self.assertNotIn(b'//# sourceMappingURL=source_map.js.map', content)
- self.assertIn(
- b'//# sourceMappingURL=source_map.js.99914b932bd3.map',
- content,
- )
- self.assertPostCondition()
- def test_js_source_map_sensitive(self):
- relpath = self.hashed_file_path('cached/source_map_sensitive.js')
- self.assertEqual(relpath, 'cached/source_map_sensitive.5da96fdd3cb3.js')
- with storage.staticfiles_storage.open(relpath) as relfile:
- content = relfile.read()
- self.assertIn(b'//# sOuRcEMaPpInGURL=source_map.js.map', content)
- self.assertNotIn(
- b'//# sourceMappingURL=source_map.js.99914b932bd3.map',
- content,
- )
- self.assertPostCondition()
- @override_settings(
- STATICFILES_DIRS=[os.path.join(TEST_ROOT, 'project', 'faulty')],
- STATICFILES_FINDERS=['django.contrib.staticfiles.finders.FileSystemFinder'],
- )
- def test_post_processing_failure(self):
- """
- post_processing indicates the origin of the error when it fails.
- """
- finders.get_finder.cache_clear()
- err = 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())
- self.assertPostCondition()
- @override_settings(STATICFILES_STORAGE='staticfiles_tests.storage.ExtraPatternsStorage')
- class TestExtraPatternsStorage(CollectionTestCase):
- def setUp(self):
- storage.staticfiles_storage.hashed_files.clear() # avoid cache interference
- super().setUp()
- def cached_file_path(self, path):
- fullpath = self.render_template(self.static_template_snippet(path))
- return fullpath.replace(settings.STATIC_URL, '')
- def test_multi_extension_patterns(self):
- """
- With storage classes having several file extension patterns, only the
- files matching a specific file pattern should be affected by the
- substitution (#19670).
- """
- # CSS files shouldn't be touched by JS patterns.
- relpath = self.cached_file_path("cached/import.css")
- self.assertEqual(relpath, "cached/import.f53576679e5a.css")
- with storage.staticfiles_storage.open(relpath) as relfile:
- self.assertIn(b'import url("styles.5e0040571e1a.css")', relfile.read())
- # Confirm JS patterns have been applied to JS files.
- relpath = self.cached_file_path("cached/test.js")
- self.assertEqual(relpath, "cached/test.388d7a790d46.js")
- with storage.staticfiles_storage.open(relpath) as relfile:
- self.assertIn(b'JS_URL("import.f53576679e5a.css")', relfile.read())
- @override_settings(
- STATICFILES_STORAGE='django.contrib.staticfiles.storage.ManifestStaticFilesStorage',
- )
- class TestCollectionManifestStorage(TestHashedFiles, CollectionTestCase):
- """
- Tests for the Cache busting storage
- """
- def setUp(self):
- super().setUp()
- temp_dir = tempfile.mkdtemp()
- os.makedirs(os.path.join(temp_dir, 'test'))
- self._clear_filename = os.path.join(temp_dir, 'test', 'cleared.txt')
- with open(self._clear_filename, 'w') as f:
- f.write('to be deleted in one test')
- self.patched_settings = self.settings(
- STATICFILES_DIRS=settings.STATICFILES_DIRS + [temp_dir],
- )
- self.patched_settings.enable()
- self.addCleanup(shutil.rmtree, temp_dir)
- self._manifest_strict = storage.staticfiles_storage.manifest_strict
- def tearDown(self):
- self.patched_settings.disable()
- if os.path.exists(self._clear_filename):
- os.unlink(self._clear_filename)
- storage.staticfiles_storage.manifest_strict = self._manifest_strict
- super().tearDown()
- def assertPostCondition(self):
- hashed_files = storage.staticfiles_storage.hashed_files
- # The in-memory version of the manifest matches the one on disk
- # since a properly created manifest should cover all filenames.
- if hashed_files:
- manifest = storage.staticfiles_storage.load_manifest()
- self.assertEqual(hashed_files, manifest)
- 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_manifest_does_not_exist(self):
- storage.staticfiles_storage.manifest_name = 'does.not.exist.json'
- self.assertIsNone(storage.staticfiles_storage.read_manifest())
- def test_manifest_does_not_ignore_permission_error(self):
- with mock.patch('builtins.open', side_effect=PermissionError):
- with self.assertRaises(PermissionError):
- storage.staticfiles_storage.read_manifest()
- 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,
- 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 = storage.staticfiles_storage.clean_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)
- def test_missing_entry(self):
- missing_file_name = 'cached/missing.css'
- configured_storage = storage.staticfiles_storage
- self.assertNotIn(missing_file_name, configured_storage.hashed_files)
- # File name not found in manifest
- with self.assertRaisesMessage(ValueError, "Missing staticfiles manifest entry for '%s'" % missing_file_name):
- self.hashed_file_path(missing_file_name)
- configured_storage.manifest_strict = False
- # File doesn't exist on disk
- err_msg = "The file '%s' could not be found with %r." % (missing_file_name, configured_storage._wrapped)
- with self.assertRaisesMessage(ValueError, err_msg):
- self.hashed_file_path(missing_file_name)
- content = StringIO()
- content.write('Found')
- configured_storage.save(missing_file_name, content)
- # File exists on disk
- self.hashed_file_path(missing_file_name)
- def test_intermediate_files(self):
- cached_files = os.listdir(os.path.join(settings.STATIC_ROOT, 'cached'))
- # Intermediate files shouldn't be created for reference.
- self.assertEqual(
- len([
- cached_file
- for cached_file in cached_files
- if cached_file.startswith('relative.')
- ]),
- 2,
- )
- @override_settings(STATICFILES_STORAGE='staticfiles_tests.storage.NoneHashStorage')
- class TestCollectionNoneHashStorage(CollectionTestCase):
- hashed_file_path = hashed_file_path
- def test_hashed_name(self):
- relpath = self.hashed_file_path('cached/styles.css')
- self.assertEqual(relpath, 'cached/styles.css')
- @override_settings(
- STATICFILES_STORAGE='staticfiles_tests.storage.NoPostProcessReplacedPathStorage'
- )
- class TestCollectionNoPostProcessReplacedPaths(CollectionTestCase):
- run_collectstatic_in_setUp = False
- def test_collectstatistic_no_post_process_replaced_paths(self):
- stdout = StringIO()
- self.run_collectstatic(verbosity=1, stdout=stdout)
- self.assertIn('post-processed', stdout.getvalue())
- @override_settings(STATICFILES_STORAGE='staticfiles_tests.storage.SimpleStorage')
- class TestCollectionSimpleStorage(CollectionTestCase):
- hashed_file_path = hashed_file_path
- def setUp(self):
- storage.staticfiles_storage.hashed_files.clear() # avoid cache interference
- super().setUp()
- def test_template_tag_return(self):
- 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 CustomManifestStorage(storage.ManifestStaticFilesStorage):
- def __init__(self, *args, manifest_storage=None, **kwargs):
- manifest_storage = storage.StaticFilesStorage(
- location=kwargs.pop('manifest_location'),
- )
- super().__init__(*args, manifest_storage=manifest_storage, **kwargs)
- class TestCustomManifestStorage(SimpleTestCase):
- def setUp(self):
- self.manifest_path = Path(tempfile.mkdtemp())
- self.addCleanup(shutil.rmtree, self.manifest_path)
- self.staticfiles_storage = CustomManifestStorage(
- manifest_location=self.manifest_path,
- )
- self.manifest_file = self.manifest_path / self.staticfiles_storage.manifest_name
- # Manifest without paths.
- self.manifest = {'version': self.staticfiles_storage.manifest_version}
- with self.manifest_file.open('w') as manifest_file:
- json.dump(self.manifest, manifest_file)
- def test_read_manifest(self):
- self.assertEqual(
- self.staticfiles_storage.read_manifest(),
- json.dumps(self.manifest),
- )
- def test_read_manifest_nonexistent(self):
- os.remove(self.manifest_file)
- self.assertIsNone(self.staticfiles_storage.read_manifest())
- def test_save_manifest_override(self):
- self.assertIs(self.manifest_file.exists(), True)
- self.staticfiles_storage.save_manifest()
- self.assertIs(self.manifest_file.exists(), True)
- new_manifest = json.loads(self.staticfiles_storage.read_manifest())
- self.assertIn('paths', new_manifest)
- self.assertNotEqual(new_manifest, self.manifest)
- def test_save_manifest_create(self):
- os.remove(self.manifest_file)
- self.staticfiles_storage.save_manifest()
- self.assertIs(self.manifest_file.exists(), True)
- new_manifest = json.loads(self.staticfiles_storage.read_manifest())
- self.assertIn('paths', new_manifest)
- self.assertNotEqual(new_manifest, self.manifest)
- class CustomStaticFilesStorage(storage.StaticFilesStorage):
- """
- Used in TestStaticFilePermissions
- """
- def __init__(self, *args, **kwargs):
- kwargs['file_permissions_mode'] = 0o640
- kwargs['directory_permissions_mode'] = 0o740
- super().__init__(*args, **kwargs)
- @unittest.skipIf(sys.platform == 'win32', "Windows only partially supports chmod.")
- class TestStaticFilePermissions(CollectionTestCase):
- command_params = {
- 'interactive': False,
- 'verbosity': 0,
- 'ignore_patterns': ['*.ignoreme'],
- }
- def setUp(self):
- self.umask = 0o027
- self.old_umask = os.umask(self.umask)
- super().setUp()
- def tearDown(self):
- os.umask(self.old_umask)
- super().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):
- call_command('collectstatic', **self.command_params)
- static_root = Path(settings.STATIC_ROOT)
- test_file = static_root / 'test.txt'
- file_mode = test_file.stat().st_mode & 0o777
- self.assertEqual(file_mode, 0o655)
- tests = [
- static_root / 'subdir',
- static_root / 'nested',
- static_root / 'nested' / 'css',
- ]
- for directory in tests:
- with self.subTest(directory=directory):
- dir_mode = directory.stat().st_mode & 0o777
- self.assertEqual(dir_mode, 0o765)
- @override_settings(
- FILE_UPLOAD_PERMISSIONS=None,
- FILE_UPLOAD_DIRECTORY_PERMISSIONS=None,
- )
- def test_collect_static_files_default_permissions(self):
- call_command('collectstatic', **self.command_params)
- static_root = Path(settings.STATIC_ROOT)
- test_file = static_root / 'test.txt'
- file_mode = test_file.stat().st_mode & 0o777
- self.assertEqual(file_mode, 0o666 & ~self.umask)
- tests = [
- static_root / 'subdir',
- static_root / 'nested',
- static_root / 'nested' / 'css',
- ]
- for directory in tests:
- with self.subTest(directory=directory):
- dir_mode = directory.stat().st_mode & 0o777
- 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):
- call_command('collectstatic', **self.command_params)
- static_root = Path(settings.STATIC_ROOT)
- test_file = static_root / 'test.txt'
- file_mode = test_file.stat().st_mode & 0o777
- self.assertEqual(file_mode, 0o640)
- tests = [
- static_root / 'subdir',
- static_root / 'nested',
- static_root / 'nested' / 'css',
- ]
- for directory in tests:
- with self.subTest(directory=directory):
- dir_mode = directory.stat().st_mode & 0o777
- self.assertEqual(dir_mode, 0o740)
- @override_settings(
- STATICFILES_STORAGE='django.contrib.staticfiles.storage.ManifestStaticFilesStorage',
- )
- class TestCollectionHashedFilesCache(CollectionTestCase):
- """
- Files referenced from CSS use the correct final hashed name regardless of
- the order in which the files are post-processed.
- """
- hashed_file_path = hashed_file_path
- def setUp(self):
- super().setUp()
- self._temp_dir = temp_dir = tempfile.mkdtemp()
- os.makedirs(os.path.join(temp_dir, 'test'))
- self.addCleanup(shutil.rmtree, temp_dir)
- def _get_filename_path(self, filename):
- return os.path.join(self._temp_dir, 'test', filename)
- def test_file_change_after_collectstatic(self):
- # Create initial static files.
- file_contents = (
- ('foo.png', 'foo'),
- ('bar.css', 'url("foo.png")\nurl("xyz.png")'),
- ('xyz.png', 'xyz'),
- )
- for filename, content in file_contents:
- with open(self._get_filename_path(filename), 'w') as f:
- f.write(content)
- with self.modify_settings(STATICFILES_DIRS={'append': self._temp_dir}):
- finders.get_finder.cache_clear()
- err = StringIO()
- # First collectstatic run.
- call_command('collectstatic', interactive=False, verbosity=0, stderr=err)
- relpath = self.hashed_file_path('test/bar.css')
- with storage.staticfiles_storage.open(relpath) as relfile:
- content = relfile.read()
- self.assertIn(b'foo.acbd18db4cc2.png', content)
- self.assertIn(b'xyz.d16fb36f0911.png', content)
- # Change the contents of the png files.
- for filename in ('foo.png', 'xyz.png'):
- with open(self._get_filename_path(filename), 'w+b') as f:
- f.write(b"new content of file to change its hash")
- # The hashes of the png files in the CSS file are updated after
- # a second collectstatic.
- call_command('collectstatic', interactive=False, verbosity=0, stderr=err)
- relpath = self.hashed_file_path('test/bar.css')
- with storage.staticfiles_storage.open(relpath) as relfile:
- content = relfile.read()
- self.assertIn(b'foo.57a5cb9ba68d.png', content)
- self.assertIn(b'xyz.57a5cb9ba68d.png', content)
|