#13794 closed Bug (fixed)
Django does not respect to_field's model on an inline model admin
Reported by: | Owned by: | gautier | |
---|---|---|---|
Component: | Forms | Version: | dev |
Severity: | Normal | Keywords: | admin inline to_field formset |
Cc: | ghayoun@… | Triage Stage: | Accepted |
Has patch: | yes | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description (last modified by )
The problem occurs in the function __unicode__
of ModelB
When editing a ModelA
instance in admin site, all ModelB
instances have a modela_id
set to the pk
of the ModelA
instance, instead of the value of the to_field
's field. And, when accessing to the field modela of a ModelB
instance, a DoesNotExist
exception is raised.
This problem does not occur in the shell :
In [3]: ModelA.objects.all()[0].modelb_set.all()[0].modela_id Out[3]: u'TRUC' In [6]: ModelB.objects.all()[0].modela_id Out[6]: u'TRUC'
See below to reproduce
# models.py class ModelA(models.Model): code = models.CharField(max_length=20, unique=True) class ModelB(models.Model): modela = models.ForeignKey(ModelA, to_field="code") position = models.IntegerField() def __unicode__(self): return u"%s" % self.modela # admin.py class ModelBInlineAdmin(admin.TabularInline): model = ModelB class ModelAAdmin(admin.ModelAdmin): model = ModelA inlines = [ModelBInlineAdmin] admin.site.register(ModelA, ModelAAdmin)
Attachments (6)
Change History (39)
comment:1 by , 15 years ago
Cc: | added |
---|
comment:2 by , 15 years ago
Has patch: | set |
---|---|
Keywords: | formset added |
Owner: | changed from | to
Status: | new → assigned |
Triage Stage: | Unreviewed → Accepted |
comment:3 by , 14 years ago
Component: | django.contrib.admin → Forms |
---|
comment:4 by , 14 years ago
Severity: | → Normal |
---|---|
Type: | → Bug |
comment:5 by , 14 years ago
Easy pickings: | unset |
---|---|
Patch needs improvement: | set |
fix_inline_model_with_to_field.diff fails to apply cleanly on to trunk
by , 12 years ago
Attachment: | to_field_formsets.patch added |
---|
comment:7 by , 12 years ago
Patch needs improvement: | unset |
---|---|
Version: | 1.2 → 1.5-beta-1 |
I updated the patch so that is does not change behavior at all for no to_field formsets. Any feedback or suggestions welcome.
This bug has been present for a while and would be nice to get fixed.
comment:8 by , 12 years ago
Status: | assigned → new |
---|
comment:10 by , 12 years ago
I asked Florian (who reported #11043) to have a look; he didn't have much time but here are his initial thoughts:
that's confusing code; when you look at django/forms/models.py line 707 -- do you have any idea where self.fk should come from?
FormSet = modelformset_factory(model, **kwargs) FormSet.fk = fk
both patches seem to fix the problem (or at least the attached testcase), which makes me think that the testcase isn't good enough, might have to trigger validation
I can't figure out why get_queryset is affected by _construct_form
ah constructing a formset executes the query and then changes the instances, now on to figure out if there could be a problem if that data is deleted :/
https://code.djangoproject.com/attachment/ticket/13794/fix_inline_model_with_to_field.diff this patch looks better, the updated one seems wrong or useless
comment:11 by , 12 years ago
The patch 'to_field_formsets.patch' was an attempt to fix the problem in the safest way possible. All tests pass with line 719 removed, but the comment above suggests that line should not be removed. If you are convinced that line isn't needed then fix_inline_model_with_to_field.diff is fine. I just wanted to provide an option that is guaranteed not to break any working code but the fix the problem.
If you think this needs more investigation please let me know and I'll try to find out if that line is just a relic of the old way or if it is actually used.
comment:12 by , 12 years ago
After further investigation it seems that main functional difference between the two is that not setting the fk value on form construction (fix_inline_model_with_to_field.diff) results in the relationship field not being available on a new instance until after full_clean, specifically construct_instance, is called on the formset. Setting the fk value on construction (to_field_formsets.patch), means that the fk value and subsequently the relationship is available when dealing with a new item.
Adding this line to the test in to_field_formsets.path will make the test pass for that patch but fail for fix_inline_model_with_to_field.diff:
self.assertEqual(formset[1].instance.user_id, "guido")
So there is a slight chance of fix_inline_model_with_to_field.diff breaking currently working code while to_field_formsets.patch will not. But as far as I can tell this undocumented so it's really your call either way.
The problem is that right now, this does have potential to create invalid relationships, leading to data corruption. If the value set in _construct_form happens to be the same as a valid pk value of that same table, then the save happens silently resetting all your relationships to point to the wrong record.
comment:13 by , 12 years ago
Severity: | Normal → Release blocker |
---|
comment:14 by , 12 years ago
Severity: | Release blocker → Normal |
---|
Why would this be a release blocker?
comment:16 by , 12 years ago
This bug was filed three years ago and Django has had several major releases since then.
If you think it should block the next release please explain why on the django-developers mailing list, with a little bit more than 5 words. Thank you!
comment:17 by , 12 years ago
Actually I did but I'm having a hard time getting any response. I did the 5 for one, but I think that part of the problem is that the title of the bug makes it seem like a minor defect. Granted, like most of the to_field related bugs, it is a bit of an edge case and probably not too many people have run into it, however for those that do it can silently and irreparably replace your relationships. Leading to very hard to track down bugs and real data loss. My understanding is that these sorts of bugs are the ones that are release blockers.
There are two possible patches here and I think I outlined the differences and potential problems with each above. If more clarification is needed I am happy to help or improve the patches in any way I can.
by , 11 years ago
Attachment: | to_field_formsets.2.patch added |
---|
comment:18 by , 11 years ago
Version: | 1.5-beta-1 → master |
---|
Updated patch to apply against master. Would be really nice to get this fixed. Again this bug can have the nasty side effect of reseting your relationships to point to the wrong record.
comment:19 by , 11 years ago
Patch needs improvement: | set |
---|
The latest version of of the patch doesn't include the test from the earlier version. Please uncheck "Patch needs improvement" if someone can update it. Thanks.
by , 11 years ago
Attachment: | to_field_formsets.3.diff added |
---|
comment:21 by , 11 years ago
No tests fail if I remove the line fk_value = getattr(fk_value, 'pk', fk_value)
. Could you clarify if that's needed and add a test if so?
by , 11 years ago
Attachment: | to_field_formsets.4.diff added |
---|
comment:22 by , 11 years ago
It's an edge case of an edge case. It was done to match what happens in save_new. Added a test.
comment:23 by , 11 years ago
Resolution: | → fixed |
---|---|
Status: | new → closed |
comment:24 by , 11 years ago
Thanks for the commit, any chance of this making it into 1.7? It is a potentially destructive bug so it would be nice to see it in a stable version sooner rather than later.
comment:27 by , 11 years ago
IMO, as the fixed issue can cause data loss in formsets (and not only in contrib.admin inlines like its summary mentions) it qualifies for getting backported to all the currently supported branches.
I've pushed the backport commits to:
- https://github.com/ramiro/django/commit/bd7baed03f023e22933579283c0e9ef222dda212
- https://github.com/ramiro/django/commit/4a3c5904c1d69737bfeb475405294e3614ea2c25
- https://github.com/ramiro/django/commit/c89684cf2f8ac378846c226bee1c858172c1edc1
Opinions?
comment:28 by , 11 years ago
1.4 and 1.5 are in security-fix only mode and I don't think 1.5 will get another release unless there are security issues before 1.7 is released. I'd be fine with backporting to 1.6, although it's funny to consider it a critical bug when it's been open for 4 years.
Same cause as #10243, #11043, fixed by [10756].
But the merging of soc2009/model-validation in [12098] broke the fix, by forcing the value of the related primary key into the foreign key field.
My patch removes this behaviour, and add a regression test. All other unit tests still pass.