1
0
mirror of https://github.com/django/django.git synced 2025-10-25 14:46:09 +00:00

[2.2.x] Fixed #30179 -- Fixed form Media merging when pairwise merging is insufficient.

Thanks gasman for the tests, and codingjoe and timgraham for the review.

Backport of 231b513926 from master.
This commit is contained in:
Matthias Kestenholz
2019-02-23 16:41:56 +01:00
committed by Tim Graham
parent 77e53da127
commit 459f7c804d
4 changed files with 108 additions and 95 deletions

View File

@@ -6,16 +6,21 @@ import copy
import datetime import datetime
import re import re
import warnings import warnings
from collections import defaultdict
from itertools import chain from itertools import chain
from django.conf import settings from django.conf import settings
from django.forms.utils import to_current_timezone from django.forms.utils import to_current_timezone
from django.templatetags.static import static from django.templatetags.static import static
from django.utils import datetime_safe, formats from django.utils import datetime_safe, formats
from django.utils.datastructures import OrderedSet
from django.utils.dates import MONTHS from django.utils.dates import MONTHS
from django.utils.formats import get_format from django.utils.formats import get_format
from django.utils.html import format_html, html_safe from django.utils.html import format_html, html_safe
from django.utils.safestring import mark_safe from django.utils.safestring import mark_safe
from django.utils.topological_sort import (
CyclicDependencyError, stable_topological_sort,
)
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from .renderers import get_default_renderer from .renderers import get_default_renderer
@@ -59,22 +64,15 @@ class Media:
@property @property
def _css(self): def _css(self):
css = self._css_lists[0] css = defaultdict(list)
# filter(None, ...) avoids calling merge with empty dicts. for css_list in self._css_lists:
for obj in filter(None, self._css_lists[1:]): for medium, sublist in css_list.items():
css = { css[medium].append(sublist)
medium: self.merge(css.get(medium, []), obj.get(medium, [])) return {medium: self.merge(*lists) for medium, lists in css.items()}
for medium in css.keys() | obj.keys()
}
return css
@property @property
def _js(self): def _js(self):
js = self._js_lists[0] return self.merge(*self._js_lists)
# filter(None, ...) avoids calling merge() with empty lists.
for obj in filter(None, self._js_lists[1:]):
js = self.merge(js, obj)
return js
def render(self): def render(self):
return mark_safe('\n'.join(chain.from_iterable(getattr(self, 'render_' + name)() for name in MEDIA_TYPES))) return mark_safe('\n'.join(chain.from_iterable(getattr(self, 'render_' + name)() for name in MEDIA_TYPES)))
@@ -115,39 +113,37 @@ class Media:
raise KeyError('Unknown media type "%s"' % name) raise KeyError('Unknown media type "%s"' % name)
@staticmethod @staticmethod
def merge(list_1, list_2): def merge(*lists):
""" """
Merge two lists while trying to keep the relative order of the elements. Merge lists while trying to keep the relative order of the elements.
Warn if the lists have the same two elements in a different relative Warn if the lists have the same elements in a different relative order.
order.
For static assets it can be important to have them included in the DOM For static assets it can be important to have them included in the DOM
in a certain order. In JavaScript you may not be able to reference a in a certain order. In JavaScript you may not be able to reference a
global or in CSS you might want to override a style. global or in CSS you might want to override a style.
""" """
# Start with a copy of list_1. dependency_graph = defaultdict(set)
combined_list = list(list_1) all_items = OrderedSet()
last_insert_index = len(list_1) for list_ in filter(None, lists):
# Walk list_2 in reverse, inserting each element into combined_list if head = list_[0]
# it doesn't already exist. # The first items depend on nothing but have to be part of the
for path in reversed(list_2): # dependency graph to be included in the result.
dependency_graph.setdefault(head, set())
for item in list_:
all_items.add(item)
# No self dependencies
if head != item:
dependency_graph[item].add(head)
head = item
try: try:
# Does path already exist in the list? return stable_topological_sort(all_items, dependency_graph)
index = combined_list.index(path) except CyclicDependencyError:
except ValueError:
# Add path to combined_list since it doesn't exist.
combined_list.insert(last_insert_index, path)
else:
if index > last_insert_index:
warnings.warn( warnings.warn(
'Detected duplicate Media files in an opposite order:\n' 'Detected duplicate Media files in an opposite order: {}'.format(
'%s\n%s' % (combined_list[last_insert_index], combined_list[index]), ', '.join(repr(l) for l in lists)
MediaOrderConflictWarning, ), MediaOrderConflictWarning,
) )
# path already exists in the list. Update last_insert_index so return list(all_items)
# that the following elements are inserted in front of this one.
last_insert_index = index
return combined_list
def __add__(self, other): def __add__(self, other):
combined = Media() combined = Media()

View File

@@ -497,10 +497,10 @@ class TestInlineMedia(TestDataMixin, TestCase):
response.context['inline_admin_formsets'][0].media._js, response.context['inline_admin_formsets'][0].media._js,
[ [
'admin/js/vendor/jquery/jquery.min.js', 'admin/js/vendor/jquery/jquery.min.js',
'admin/js/jquery.init.js',
'admin/js/inlines.min.js',
'my_awesome_inline_scripts.js', 'my_awesome_inline_scripts.js',
'custom_number.js', 'custom_number.js',
'admin/js/jquery.init.js',
'admin/js/inlines.min.js',
] ]
) )
self.assertContains(response, 'my_awesome_inline_scripts.js') self.assertContains(response, 'my_awesome_inline_scripts.js')

View File

@@ -139,4 +139,4 @@ class AutocompleteMixinTests(TestCase):
else: else:
expected_files = base_files expected_files = base_files
with translation.override(lang): with translation.override(lang):
self.assertEqual(AutocompleteSelect(rel, admin.site).media._js, expected_files) self.assertEqual(AutocompleteSelect(rel, admin.site).media._js, list(expected_files))

View File

@@ -25,8 +25,8 @@ class FormsMediaTestCase(SimpleTestCase):
) )
self.assertEqual( self.assertEqual(
repr(m), repr(m),
"Media(css={'all': ('path/to/css1', '/path/to/css2')}, " "Media(css={'all': ['path/to/css1', '/path/to/css2']}, "
"js=('/path/to/js1', 'http://media.other.com/path/to/js2', 'https://secure.other.com/path/to/js3'))" "js=['/path/to/js1', 'http://media.other.com/path/to/js2', 'https://secure.other.com/path/to/js3'])"
) )
class Foo: class Foo:
@@ -125,8 +125,8 @@ class FormsMediaTestCase(SimpleTestCase):
<link href="/path/to/css3" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css3" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script> <script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script> <script type="text/javascript" src="/path/to/js4"></script>
<script type="text/javascript" src="/path/to/js4"></script>""" <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
) )
# media addition hasn't affected the original objects # media addition hasn't affected the original objects
@@ -151,6 +151,17 @@ class FormsMediaTestCase(SimpleTestCase):
self.assertEqual(str(w4.media), """<link href="/path/to/css1" type="text/css" media="all" rel="stylesheet"> self.assertEqual(str(w4.media), """<link href="/path/to/css1" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script>""") <script type="text/javascript" src="/path/to/js1"></script>""")
def test_media_deduplication(self):
# A deduplication test applied directly to a Media object, to confirm
# that the deduplication doesn't only happen at the point of merging
# two or more media objects.
media = Media(
css={'all': ('/path/to/css1', '/path/to/css1')},
js=('/path/to/js1', '/path/to/js1'),
)
self.assertEqual(str(media), """<link href="/path/to/css1" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script>""")
def test_media_property(self): def test_media_property(self):
############################################################### ###############################################################
# Property-based media definitions # Property-based media definitions
@@ -197,12 +208,12 @@ class FormsMediaTestCase(SimpleTestCase):
self.assertEqual( self.assertEqual(
str(w6.media), str(w6.media),
"""<link href="http://media.example.com/static/path/to/css1" type="text/css" media="all" rel="stylesheet"> """<link href="http://media.example.com/static/path/to/css1" type="text/css" media="all" rel="stylesheet">
<link href="/path/to/css2" type="text/css" media="all" rel="stylesheet">
<link href="/other/path" type="text/css" media="all" rel="stylesheet"> <link href="/other/path" type="text/css" media="all" rel="stylesheet">
<link href="/path/to/css2" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script> <script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="/other/js"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script> <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
<script type="text/javascript" src="/other/js"></script>"""
) )
def test_media_inheritance(self): def test_media_inheritance(self):
@@ -247,8 +258,8 @@ class FormsMediaTestCase(SimpleTestCase):
<link href="/path/to/css2" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css2" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script> <script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script> <script type="text/javascript" src="/path/to/js4"></script>
<script type="text/javascript" src="/path/to/js4"></script>""" <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
) )
def test_media_inheritance_from_property(self): def test_media_inheritance_from_property(self):
@@ -322,8 +333,8 @@ class FormsMediaTestCase(SimpleTestCase):
<link href="/path/to/css2" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css2" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script> <script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script> <script type="text/javascript" src="/path/to/js4"></script>
<script type="text/javascript" src="/path/to/js4"></script>""" <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
) )
def test_media_inheritance_single_type(self): def test_media_inheritance_single_type(self):
@@ -420,8 +431,8 @@ class FormsMediaTestCase(SimpleTestCase):
<link href="/path/to/css3" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css3" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script> <script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script> <script type="text/javascript" src="/path/to/js4"></script>
<script type="text/javascript" src="/path/to/js4"></script>""" <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
) )
def test_form_media(self): def test_form_media(self):
@@ -462,8 +473,8 @@ class FormsMediaTestCase(SimpleTestCase):
<link href="/path/to/css3" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css3" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script> <script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script> <script type="text/javascript" src="/path/to/js4"></script>
<script type="text/javascript" src="/path/to/js4"></script>""" <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
) )
# Form media can be combined to produce a single media definition. # Form media can be combined to produce a single media definition.
@@ -477,8 +488,8 @@ class FormsMediaTestCase(SimpleTestCase):
<link href="/path/to/css3" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css3" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script> <script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script> <script type="text/javascript" src="/path/to/js4"></script>
<script type="text/javascript" src="/path/to/js4"></script>""" <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
) )
# Forms can also define media, following the same rules as widgets. # Forms can also define media, following the same rules as widgets.
@@ -495,28 +506,28 @@ class FormsMediaTestCase(SimpleTestCase):
self.assertEqual( self.assertEqual(
str(f3.media), str(f3.media),
"""<link href="http://media.example.com/static/path/to/css1" type="text/css" media="all" rel="stylesheet"> """<link href="http://media.example.com/static/path/to/css1" type="text/css" media="all" rel="stylesheet">
<link href="/some/form/css" type="text/css" media="all" rel="stylesheet">
<link href="/path/to/css2" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css2" type="text/css" media="all" rel="stylesheet">
<link href="/path/to/css3" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css3" type="text/css" media="all" rel="stylesheet">
<link href="/some/form/css" type="text/css" media="all" rel="stylesheet">
<script type="text/javascript" src="/path/to/js1"></script> <script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="/some/form/javascript"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>
<script type="text/javascript" src="/path/to/js4"></script> <script type="text/javascript" src="/path/to/js4"></script>
<script type="text/javascript" src="/some/form/javascript"></script>""" <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
) )
# Media works in templates # Media works in templates
self.assertEqual( self.assertEqual(
Template("{{ form.media.js }}{{ form.media.css }}").render(Context({'form': f3})), Template("{{ form.media.js }}{{ form.media.css }}").render(Context({'form': f3})),
"""<script type="text/javascript" src="/path/to/js1"></script> """<script type="text/javascript" src="/path/to/js1"></script>
<script type="text/javascript" src="/some/form/javascript"></script>
<script type="text/javascript" src="http://media.other.com/path/to/js2"></script> <script type="text/javascript" src="http://media.other.com/path/to/js2"></script>
<script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>
<script type="text/javascript" src="/path/to/js4"></script> <script type="text/javascript" src="/path/to/js4"></script>
<script type="text/javascript" src="/some/form/javascript"></script>""" <script type="text/javascript" src="https://secure.other.com/path/to/js3"></script>"""
"""<link href="http://media.example.com/static/path/to/css1" type="text/css" media="all" rel="stylesheet"> """<link href="http://media.example.com/static/path/to/css1" type="text/css" media="all" rel="stylesheet">
<link href="/some/form/css" type="text/css" media="all" rel="stylesheet">
<link href="/path/to/css2" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css2" type="text/css" media="all" rel="stylesheet">
<link href="/path/to/css3" type="text/css" media="all" rel="stylesheet"> <link href="/path/to/css3" type="text/css" media="all" rel="stylesheet">"""
<link href="/some/form/css" type="text/css" media="all" rel="stylesheet">"""
) )
def test_html_safe(self): def test_html_safe(self):
@@ -526,19 +537,23 @@ class FormsMediaTestCase(SimpleTestCase):
def test_merge(self): def test_merge(self):
test_values = ( test_values = (
(([1, 2], [3, 4]), [1, 2, 3, 4]), (([1, 2], [3, 4]), [1, 3, 2, 4]),
(([1, 2], [2, 3]), [1, 2, 3]), (([1, 2], [2, 3]), [1, 2, 3]),
(([2, 3], [1, 2]), [1, 2, 3]), (([2, 3], [1, 2]), [1, 2, 3]),
(([1, 3], [2, 3]), [1, 2, 3]), (([1, 3], [2, 3]), [1, 2, 3]),
(([1, 2], [1, 3]), [1, 2, 3]), (([1, 2], [1, 3]), [1, 2, 3]),
(([1, 2], [3, 2]), [1, 3, 2]), (([1, 2], [3, 2]), [1, 3, 2]),
(([1, 2], [1, 2]), [1, 2]),
([[1, 2], [1, 3], [2, 3], [5, 7], [5, 6], [6, 7, 9], [8, 9]], [1, 5, 8, 2, 6, 3, 7, 9]),
((), []),
(([1, 2],), [1, 2]),
) )
for (list1, list2), expected in test_values: for lists, expected in test_values:
with self.subTest(list1=list1, list2=list2): with self.subTest(lists=lists):
self.assertEqual(Media.merge(list1, list2), expected) self.assertEqual(Media.merge(*lists), expected)
def test_merge_warning(self): def test_merge_warning(self):
msg = 'Detected duplicate Media files in an opposite order:\n1\n2' msg = 'Detected duplicate Media files in an opposite order: [1, 2], [2, 1]'
with self.assertWarnsMessage(RuntimeWarning, msg): with self.assertWarnsMessage(RuntimeWarning, msg):
self.assertEqual(Media.merge([1, 2], [2, 1]), [1, 2]) self.assertEqual(Media.merge([1, 2], [2, 1]), [1, 2])
@@ -546,28 +561,30 @@ class FormsMediaTestCase(SimpleTestCase):
""" """
The relative order of scripts is preserved in a three-way merge. The relative order of scripts is preserved in a three-way merge.
""" """
# custom_widget.js doesn't depend on jquery.js. widget1 = Media(js=['color-picker.js'])
widget1 = Media(js=['custom_widget.js']) widget2 = Media(js=['text-editor.js'])
widget2 = Media(js=['jquery.js', 'uses_jquery.js']) widget3 = Media(js=['text-editor.js', 'text-editor-extras.js', 'color-picker.js'])
form_media = widget1 + widget2 merged = widget1 + widget2 + widget3
# The relative ordering of custom_widget.js and jquery.js has been self.assertEqual(merged._js, ['text-editor.js', 'text-editor-extras.js', 'color-picker.js'])
# established (but without a real need to).
self.assertEqual(form_media._js, ['custom_widget.js', 'jquery.js', 'uses_jquery.js']) def test_merge_js_three_way2(self):
# The inline also uses custom_widget.js. This time, it's at the end. # The merge prefers to place 'c' before 'b' and 'g' before 'h' to
inline_media = Media(js=['jquery.js', 'also_jquery.js']) + Media(js=['custom_widget.js']) # preserve the original order. The preference 'c'->'b' is overridden by
merged = form_media + inline_media # widget3's media, but 'g'->'h' survives in the final ordering.
self.assertEqual(merged._js, ['custom_widget.js', 'jquery.js', 'uses_jquery.js', 'also_jquery.js']) widget1 = Media(js=['a', 'c', 'f', 'g', 'k'])
widget2 = Media(js=['a', 'b', 'f', 'h', 'k'])
widget3 = Media(js=['b', 'c', 'f', 'k'])
merged = widget1 + widget2 + widget3
self.assertEqual(merged._js, ['a', 'b', 'c', 'f', 'g', 'h', 'k'])
def test_merge_css_three_way(self): def test_merge_css_three_way(self):
widget1 = Media(css={'screen': ['a.css']}) widget1 = Media(css={'screen': ['c.css'], 'all': ['d.css', 'e.css']})
widget2 = Media(css={'screen': ['b.css']}) widget2 = Media(css={'screen': ['a.css']})
widget3 = Media(css={'all': ['c.css']}) widget3 = Media(css={'screen': ['a.css', 'b.css', 'c.css'], 'all': ['e.css']})
form1 = widget1 + widget2 merged = widget1 + widget2
form2 = widget2 + widget1 # c.css comes before a.css because widget1 + widget2 establishes this
# form1 and form2 have a.css and b.css in different order... # order.
self.assertEqual(form1._css, {'screen': ['a.css', 'b.css']}) self.assertEqual(merged._css, {'screen': ['c.css', 'a.css'], 'all': ['d.css', 'e.css']})
self.assertEqual(form2._css, {'screen': ['b.css', 'a.css']}) merged = merged + widget3
# ...but merging succeeds as the relative ordering of a.css and b.css # widget3 contains an explicit ordering of c.css and a.css.
# was never specified. self.assertEqual(merged._css, {'screen': ['a.css', 'b.css', 'c.css'], 'all': ['d.css', 'e.css']})
merged = widget3 + form1 + form2
self.assertEqual(merged._css, {'screen': ['a.css', 'b.css'], 'all': ['c.css']})