Opened 9 years ago
Closed 9 years ago
#26934 closed Bug (duplicate)
Admin inline save fails when inline contains readonly primary key
| Reported by: | Kevin Lee | Owned by: | nobody |
|---|---|---|---|
| Component: | contrib.admin | Version: | 1.9 |
| Severity: | Normal | Keywords: | |
| Cc: | Triage Stage: | Unreviewed | |
| Has patch: | no | Needs documentation: | no |
| Needs tests: | no | Patch needs improvement: | no |
| Easy pickings: | no | UI/UX: | no |
Description
Example:
For an AbstractBaseCode model (password reset), the "code" field is as follows (note the primary_key argument):
code = models.CharField(_('code'), max_length=40, primary_key=True)
And the inline admin class is:
class PasswordResetCodeInline(admin.TabularInline):
model = PasswordResetCode
fieldsets = (
(None, {
'fields': ('code', 'created_at')
}),
)
readonly_fields = ('code', 'created_at')
def has_add_permission(self, request):
return False
When saving a user in Django admin, which contains the PasswordResetCodeInline, the following error is issued:
MultiValueDictKeyError at /admin/accounts/user/30/change/ "'passwordresetcode_set-0-code'" Exception location: /.../python3.5/site-packages/django/utils/datastructures.py in __getitem__, line 85
This occurs because line 587 in ModelForm._construct_form in forms/models.py looks for the primary key "code" within the editable fields. Since it is readonly, there is no input or hidden field passed along that includes the primary key.
Here is a snippet where the error occurs (on the last line):
def _construct_form(self, i, **kwargs):
if self.is_bound and i < self.initial_form_count():
pk_key = "%s-%s" % (self.add_prefix(i), self.model._meta.pk.name)
pk = self.data[pk_key]
Traceback:
Traceback (most recent call last):
File "/.../lib/python3.5/site-packages/django/core/handlers/base.py", line 149, in get_response
response = self.process_exception_by_middleware(e, request)
File "/.../lib/python3.5/site-packages/django/core/handlers/base.py", line 147, in get_response
response = wrapped_callback(request, *callback_args, **callback_kwargs)
File "/.../lib/python3.5/site-packages/django/contrib/admin/options.py", line 541, in wrapper
return self.admin_site.admin_view(view)(*args, **kwargs)
File "/.../lib/python3.5/site-packages/django/utils/decorators.py", line 149, in _wrapped_view
response = view_func(request, *args, **kwargs)
File "/.../lib/python3.5/site-packages/django/views/decorators/cache.py", line 57, in _wrapped_view_func
response = view_func(request, *args, **kwargs)
File "/.../lib/python3.5/site-packages/django/contrib/admin/sites.py", line 244, in inner
return view(request, *args, **kwargs)
File "/.../lib/python3.5/site-packages/django/contrib/admin/options.py", line 1440, in change_view
return self.changeform_view(request, object_id, form_url, extra_context)
File "/.../lib/python3.5/site-packages/django/utils/decorators.py", line 67, in _wrapper
return bound_func(*args, **kwargs)
File "/.../lib/python3.5/site-packages/django/utils/decorators.py", line 149, in _wrapped_view
response = view_func(request, *args, **kwargs)
File "/.../lib/python3.5/site-packages/django/utils/decorators.py", line 63, in bound_func
return func.__get__(self, type(self))(*args2, **kwargs2)
File "/usr/local/Cellar/python3/3.5.1/Frameworks/Python.framework/Versions/3.5/lib/python3.5/contextlib.py", line 30, in inner
return func(*args, **kwds)
File "/.../lib/python3.5/site-packages/django/contrib/admin/options.py", line 1377, in changeform_view
if all_valid(formsets) and form_validated:
File "/.../lib/python3.5/site-packages/django/forms/formsets.py", line 451, in all_valid
if not formset.is_valid():
File "/.../lib/python3.5/site-packages/django/forms/formsets.py", line 316, in is_valid
self.errors
File "/.../lib/python3.5/site-packages/django/forms/formsets.py", line 290, in errors
self.full_clean()
File "/.../lib/python3.5/site-packages/django/forms/formsets.py", line 338, in full_clean
form = self.forms[i]
File "/.../lib/python3.5/site-packages/django/utils/functional.py", line 33, in __get__
res = instance.__dict__[self.name] = self.func(instance)
File "/.../lib/python3.5/site-packages/django/forms/formsets.py", line 144, in forms
for i in range(self.total_form_count())]
File "/.../lib/python3.5/site-packages/django/forms/formsets.py", line 144, in <listcomp>
for i in range(self.total_form_count())]
File "/.../lib/python3.5/site-packages/django/forms/models.py", line 881, in _construct_form
form = super(BaseInlineFormSet, self)._construct_form(i, **kwargs)
File "/.../lib/python3.5/site-packages/django/forms/models.py", line 587, in _construct_form
pk = self.data[pk_key]
File "/.../lib/python3.5/site-packages/django/utils/datastructures.py", line 85, in __getitem__
raise MultiValueDictKeyError(repr(key))
Looks like a duplicate of #15665.