Skip to content

Commit fc62e17

Browse files
marianaasilvagradyyshangxiaoMatthew Newton
authored andcommitted
Fixed django#12241 -- Preserved query strings when using "Save and continue/add another" in admin.
Co-authored-by: Grady Yu <[email protected]> Co-authored-by: David Sanders <[email protected]> Co-authored-by: Matthew Newton <[email protected]>
1 parent 6e369f3 commit fc62e17

File tree

3 files changed

+95
-4
lines changed

3 files changed

+95
-4
lines changed

django/contrib/admin/options.py

Lines changed: 31 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,9 @@
33
import json
44
import re
55
from functools import partial, update_wrapper
6+
from urllib.parse import parse_qsl
67
from urllib.parse import quote as urlquote
8+
from urllib.parse import urlparse
79

810
from django import forms
911
from django.conf import settings
@@ -1346,12 +1348,17 @@ def render_change_form(
13461348
context,
13471349
)
13481350

1351+
def _get_preserved_qsl(self, request, preserved_filters):
1352+
query_string = urlparse(request.build_absolute_uri()).query
1353+
return parse_qsl(query_string.replace(preserved_filters, ""))
1354+
13491355
def response_add(self, request, obj, post_url_continue=None):
13501356
"""
13511357
Determine the HttpResponse for the add_view stage.
13521358
"""
13531359
opts = obj._meta
13541360
preserved_filters = self.get_preserved_filters(request)
1361+
preserved_qsl = self._get_preserved_qsl(request, preserved_filters)
13551362
obj_url = reverse(
13561363
"admin:%s_%s_change" % (opts.app_label, opts.model_name),
13571364
args=(quote(obj.pk),),
@@ -1409,7 +1416,11 @@ def response_add(self, request, obj, post_url_continue=None):
14091416
if post_url_continue is None:
14101417
post_url_continue = obj_url
14111418
post_url_continue = add_preserved_filters(
1412-
{"preserved_filters": preserved_filters, "opts": opts},
1419+
{
1420+
"preserved_filters": preserved_filters,
1421+
"preserved_qsl": preserved_qsl,
1422+
"opts": opts,
1423+
},
14131424
post_url_continue,
14141425
)
14151426
return HttpResponseRedirect(post_url_continue)
@@ -1425,7 +1436,12 @@ def response_add(self, request, obj, post_url_continue=None):
14251436
self.message_user(request, msg, messages.SUCCESS)
14261437
redirect_url = request.path
14271438
redirect_url = add_preserved_filters(
1428-
{"preserved_filters": preserved_filters, "opts": opts}, redirect_url
1439+
{
1440+
"preserved_filters": preserved_filters,
1441+
"preserved_qsl": preserved_qsl,
1442+
"opts": opts,
1443+
},
1444+
redirect_url,
14291445
)
14301446
return HttpResponseRedirect(redirect_url)
14311447

@@ -1471,6 +1487,7 @@ def response_change(self, request, obj):
14711487

14721488
opts = self.opts
14731489
preserved_filters = self.get_preserved_filters(request)
1490+
preserved_qsl = self._get_preserved_qsl(request, preserved_filters)
14741491

14751492
msg_dict = {
14761493
"name": opts.verbose_name,
@@ -1487,7 +1504,12 @@ def response_change(self, request, obj):
14871504
self.message_user(request, msg, messages.SUCCESS)
14881505
redirect_url = request.path
14891506
redirect_url = add_preserved_filters(
1490-
{"preserved_filters": preserved_filters, "opts": opts}, redirect_url
1507+
{
1508+
"preserved_filters": preserved_filters,
1509+
"preserved_qsl": preserved_qsl,
1510+
"opts": opts,
1511+
},
1512+
redirect_url,
14911513
)
14921514
return HttpResponseRedirect(redirect_url)
14931515

@@ -1524,7 +1546,12 @@ def response_change(self, request, obj):
15241546
current_app=self.admin_site.name,
15251547
)
15261548
redirect_url = add_preserved_filters(
1527-
{"preserved_filters": preserved_filters, "opts": opts}, redirect_url
1549+
{
1550+
"preserved_filters": preserved_filters,
1551+
"preserved_qsl": preserved_qsl,
1552+
"opts": opts,
1553+
},
1554+
redirect_url,
15281555
)
15291556
return HttpResponseRedirect(redirect_url)
15301557

django/contrib/admin/templatetags/admin_urls.py

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -22,11 +22,15 @@ def admin_urlquote(value):
2222
def add_preserved_filters(context, url, popup=False, to_field=None):
2323
opts = context.get("opts")
2424
preserved_filters = context.get("preserved_filters")
25+
preserved_qsl = context.get("preserved_qsl")
2526

2627
parsed_url = list(urlparse(url))
2728
parsed_qs = dict(parse_qsl(parsed_url[4]))
2829
merged_qs = {}
2930

31+
if preserved_qsl:
32+
merged_qs.update(preserved_qsl)
33+
3034
if opts and preserved_filters:
3135
preserved_filters = dict(parse_qsl(preserved_filters))
3236

tests/admin_views/tests.py

Lines changed: 60 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -328,6 +328,66 @@ def test_add_with_GET_args(self):
328328
msg_prefix="Couldn't find an input with the right value in the response",
329329
)
330330

331+
def test_add_query_string_persists(self):
332+
save_options = [
333+
{"_addanother": "1"}, # "Save and add another".
334+
{"_continue": "1"}, # "Save and continue editing".
335+
{"_saveasnew": "1"}, # "Save as new".
336+
]
337+
other_options = [
338+
"",
339+
"_changelist_filters=is_staff__exact%3D0",
340+
f"{IS_POPUP_VAR}=1",
341+
f"{TO_FIELD_VAR}=id",
342+
]
343+
url = reverse("admin:auth_user_add")
344+
for i, save_option in enumerate(save_options):
345+
for j, other_option in enumerate(other_options):
346+
with self.subTest(save_option=save_option, other_option=other_option):
347+
qsl = "username=newuser"
348+
if other_option:
349+
qsl = f"{qsl}&{other_option}"
350+
response = self.client.post(
351+
f"{url}?{qsl}",
352+
{
353+
"username": f"newuser{i}{j}",
354+
"password1": "newpassword",
355+
"password2": "newpassword",
356+
**save_option,
357+
},
358+
)
359+
parsed_url = urlparse(response.url)
360+
self.assertEqual(parsed_url.query, qsl)
361+
362+
def test_change_query_string_persists(self):
363+
save_options = [
364+
{"_addanother": "1"}, # "Save and add another".
365+
{"_continue": "1"}, # "Save and continue editing".
366+
]
367+
other_options = [
368+
"",
369+
"_changelist_filters=warm%3D1",
370+
f"{IS_POPUP_VAR}=1",
371+
f"{TO_FIELD_VAR}=id",
372+
]
373+
url = reverse("admin:admin_views_color_change", args=(self.color1.pk,))
374+
for save_option in save_options:
375+
for other_option in other_options:
376+
with self.subTest(save_option=save_option, other_option=other_option):
377+
qsl = "value=blue"
378+
if other_option:
379+
qsl = f"{qsl}&{other_option}"
380+
response = self.client.post(
381+
f"{url}?{qsl}",
382+
{
383+
"value": "gold",
384+
"warm": True,
385+
**save_option,
386+
},
387+
)
388+
parsed_url = urlparse(response.url)
389+
self.assertEqual(parsed_url.query, qsl)
390+
331391
def test_basic_edit_GET(self):
332392
"""
333393
A smoke test to ensure GET on the change_view works.

0 commit comments

Comments
 (0)