Fixed #14549 - Removed restriction of single FKs on intermediary tables

Thanks to Loic Bistuer for review. Minor changes to error messages
done by committer.
This commit is contained in:
Akis Kesoglou 2014-02-19 20:01:55 +02:00 committed by Anssi Kääriäinen
parent 95c74b9d69
commit c627da0ccc
8 changed files with 296 additions and 33 deletions

View File

@ -1250,9 +1250,12 @@ class OneToOneRel(ManyToOneRel):
class ManyToManyRel(object): class ManyToManyRel(object):
def __init__(self, to, related_name=None, limit_choices_to=None, def __init__(self, to, related_name=None, limit_choices_to=None,
symmetrical=True, through=None, db_constraint=True, related_query_name=None): symmetrical=True, through=None, through_fields=None,
db_constraint=True, related_query_name=None):
if through and not db_constraint: if through and not db_constraint:
raise ValueError("Can't supply a through model and db_constraint=False") raise ValueError("Can't supply a through model and db_constraint=False")
if through_fields and not through:
raise ValueError("Cannot specify through_fields without a through model")
self.to = to self.to = to
self.related_name = related_name self.related_name = related_name
self.related_query_name = related_query_name self.related_query_name = related_query_name
@ -1262,6 +1265,7 @@ class ManyToManyRel(object):
self.symmetrical = symmetrical self.symmetrical = symmetrical
self.multiple = True self.multiple = True
self.through = through self.through = through
self.through_fields = through_fields
self.db_constraint = db_constraint self.db_constraint = db_constraint
def is_hidden(self): def is_hidden(self):
@ -1849,6 +1853,7 @@ class ManyToManyField(RelatedField):
limit_choices_to=kwargs.pop('limit_choices_to', None), limit_choices_to=kwargs.pop('limit_choices_to', None),
symmetrical=kwargs.pop('symmetrical', to == RECURSIVE_RELATIONSHIP_CONSTANT), symmetrical=kwargs.pop('symmetrical', to == RECURSIVE_RELATIONSHIP_CONSTANT),
through=kwargs.pop('through', None), through=kwargs.pop('through', None),
through_fields=kwargs.pop('through_fields', None),
db_constraint=db_constraint, db_constraint=db_constraint,
) )
@ -1878,6 +1883,12 @@ class ManyToManyField(RelatedField):
return [] return []
def _check_relationship_model(self, from_model=None, **kwargs): def _check_relationship_model(self, from_model=None, **kwargs):
if hasattr(self.rel.through, '_meta'):
qualified_model_name = "%s.%s" % (
self.rel.through._meta.app_label, self.rel.through.__name__)
else:
qualified_model_name = self.rel.through
errors = [] errors = []
if self.rel.through not in apps.get_models(include_auto_created=True): if self.rel.through not in apps.get_models(include_auto_created=True):
@ -1885,13 +1896,28 @@ class ManyToManyField(RelatedField):
errors.append( errors.append(
checks.Error( checks.Error(
("Field specifies a many-to-many relation through model " ("Field specifies a many-to-many relation through model "
"'%s', which has not been installed.") % self.rel.through, "'%s', which has not been installed.") %
qualified_model_name,
hint=None, hint=None,
obj=self, obj=self,
id='fields.E331', id='fields.E331',
) )
) )
elif self.rel.through_fields is not None:
if not len(self.rel.through_fields) >= 2 or not (self.rel.through_fields[0] and self.rel.through_fields[1]):
errors.append(
checks.Error(
("The field is given an iterable for through_fields, "
"which does not provide the names for both link fields "
"that Django should use for the relation through model "
"'%s'.") % qualified_model_name,
hint=None,
obj=self,
id='fields.E337',
)
)
elif not isinstance(self.rel.through, six.string_types): elif not isinstance(self.rel.through, six.string_types):
assert from_model is not None, \ assert from_model is not None, \
@ -1926,13 +1952,16 @@ class ManyToManyField(RelatedField):
seen_self = sum(from_model == getattr(field.rel, 'to', None) seen_self = sum(from_model == getattr(field.rel, 'to', None)
for field in self.rel.through._meta.fields) for field in self.rel.through._meta.fields)
if seen_self > 2: if seen_self > 2 and not self.rel.through_fields:
errors.append( errors.append(
checks.Error( checks.Error(
("The model is used as an intermediate model by " ("The model is used as an intermediate model by "
"'%s', but it has more than two foreign keys " "'%s', but it has more than two foreign keys "
"to '%s', which is ambiguous.") % (self, from_model_name), "to '%s', which is ambiguous. You must specify "
hint=None, "which two foreign keys Django should use via the "
"through_fields keyword argument.") % (self, from_model_name),
hint=("Use through_fields to specify which two "
"foreign keys Django should use."),
obj=self.rel.through, obj=self.rel.through,
id='fields.E333', id='fields.E333',
) )
@ -1945,12 +1974,14 @@ class ManyToManyField(RelatedField):
seen_to = sum(to_model == getattr(field.rel, 'to', None) seen_to = sum(to_model == getattr(field.rel, 'to', None)
for field in self.rel.through._meta.fields) for field in self.rel.through._meta.fields)
if seen_from > 1: if seen_from > 1 and not self.rel.through_fields:
errors.append( errors.append(
checks.Error( checks.Error(
("The model is used as an intermediate model by " ("The model is used as an intermediate model by "
"'%s', but it has more than one foreign key " "'%s', but it has more than one foreign key "
"from '%s', which is ambiguous.") % (self, from_model_name), "from '%s', which is ambiguous. You must specify "
"which foreign key Django should use via the "
"through_fields keyword argument.") % (self, from_model_name),
hint=('If you want to create a recursive relationship, ' hint=('If you want to create a recursive relationship, '
'use ForeignKey("self", symmetrical=False, ' 'use ForeignKey("self", symmetrical=False, '
'through="%s").') % relationship_model_name, 'through="%s").') % relationship_model_name,
@ -1959,12 +1990,14 @@ class ManyToManyField(RelatedField):
) )
) )
if seen_to > 1: if seen_to > 1 and not self.rel.through_fields:
errors.append( errors.append(
checks.Error( checks.Error(
("The model is used as an intermediate model by " ("The model is used as an intermediate model by "
"'%s', but it has more than one foreign key " "'%s', but it has more than one foreign key "
"to '%s', which is ambiguous.") % (self, to_model_name), "to '%s', which is ambiguous. You must specify "
"which foreign key Django should use via the "
"through_fields keyword argument.") % (self, to_model_name),
hint=('If you want to create a recursive ' hint=('If you want to create a recursive '
'relationship, use ForeignKey("self", ' 'relationship, use ForeignKey("self", '
'symmetrical=False, through="%s").') % relationship_model_name, 'symmetrical=False, through="%s").') % relationship_model_name,
@ -2057,10 +2090,18 @@ class ManyToManyField(RelatedField):
cache_attr = '_m2m_%s_cache' % attr cache_attr = '_m2m_%s_cache' % attr
if hasattr(self, cache_attr): if hasattr(self, cache_attr):
return getattr(self, cache_attr) return getattr(self, cache_attr)
if self.rel.through_fields is not None:
link_field_name = self.rel.through_fields[0]
else:
link_field_name = None
for f in self.rel.through._meta.fields: for f in self.rel.through._meta.fields:
if hasattr(f, 'rel') and f.rel and f.rel.to == related.model: if hasattr(f, 'rel') and f.rel and f.rel.to == related.model and \
(link_field_name is None or link_field_name == f.name):
setattr(self, cache_attr, getattr(f, attr)) setattr(self, cache_attr, getattr(f, attr))
return getattr(self, cache_attr) return getattr(self, cache_attr)
# We only reach here if we're given an invalid field name via the
# `through_fields` argument
raise FieldDoesNotExist(link_field_name)
def _get_m2m_reverse_attr(self, related, attr): def _get_m2m_reverse_attr(self, related, attr):
"Function that can be curried to provide the related accessor or DB column name for the m2m table" "Function that can be curried to provide the related accessor or DB column name for the m2m table"
@ -2068,9 +2109,13 @@ class ManyToManyField(RelatedField):
if hasattr(self, cache_attr): if hasattr(self, cache_attr):
return getattr(self, cache_attr) return getattr(self, cache_attr)
found = False found = False
if self.rel.through_fields is not None:
link_field_name = self.rel.through_fields[1]
else:
link_field_name = None
for f in self.rel.through._meta.fields: for f in self.rel.through._meta.fields:
if hasattr(f, 'rel') and f.rel and f.rel.to == related.parent_model: if hasattr(f, 'rel') and f.rel and f.rel.to == related.parent_model:
if related.model == related.parent_model: if link_field_name is None and related.model == related.parent_model:
# If this is an m2m-intermediate to self, # If this is an m2m-intermediate to self,
# the first foreign key you find will be # the first foreign key you find will be
# the source column. Keep searching for # the source column. Keep searching for
@ -2080,10 +2125,15 @@ class ManyToManyField(RelatedField):
break break
else: else:
found = True found = True
else: elif link_field_name is None or link_field_name == f.name:
setattr(self, cache_attr, getattr(f, attr)) setattr(self, cache_attr, getattr(f, attr))
break break
return getattr(self, cache_attr) try:
return getattr(self, cache_attr)
except AttributeError:
# We only reach here if we're given an invalid reverse field
# name via the `through_fields` argument
raise FieldDoesNotExist(link_field_name)
def value_to_string(self, obj): def value_to_string(self, obj):
data = '' data = ''

View File

@ -90,10 +90,11 @@ Related Fields
* **fields.E330**: ManyToManyFields cannot be unique. * **fields.E330**: ManyToManyFields cannot be unique.
* **fields.E331**: Field specifies a many-to-many relation through model ``%s``, which has not been installed. * **fields.E331**: Field specifies a many-to-many relation through model ``%s``, which has not been installed.
* **fields.E332**: Many-to-many fields with intermediate tables must not be symmetrical. * **fields.E332**: Many-to-many fields with intermediate tables must not be symmetrical.
* **fields.E333**: The model is used as an intermediate model by ``<model>``, but it has more than two foreign keys to ``<model>``, which is ambiguous. * **fields.E333**: The model is used as an intermediate model by ``<model>``, but it has more than two foreign keys to ``<model>``, which is ambiguous. You must specify which two foreign keys Django should use via the through_fields keyword argument.
* **fields.E334**: The model is used as an intermediate model by ``<model>``, but it has more than one foreign key from ``<model>``, which is ambiguous. * **fields.E334**: The model is used as an intermediate model by ``<model>``, but it has more than one foreign key from ``<model>``, which is ambiguous. You must specify which foreign key Django should use via the through_fields keyword argument.
* **fields.E335**: The model is used as an intermediate model by ``<model>``, but it has more than one foreign key to ``<model>``, which is ambiguous. * **fields.E335**: The model is used as an intermediate model by ``<model>``, but it has more than one foreign key to ``<model>``, which is ambiguous. You must specify which foreign key Django should use via the through_fields keyword argument.
* **fields.E336**: The model is used as an intermediary model by ``<model>``, but it does not have foreign key to ``<model>`` or ``<model>``." * **fields.E336**: The model is used as an intermediary model by ``<model>``, but it does not have foreign key to ``<model>`` or ``<model>``.
* **fields.E337**: The field is given an iterable for through_fields, which does not provide the names for both link fields that Django should use for the relation through <model>.
Signals Signals
~~~~~~~ ~~~~~~~

View File

@ -1337,6 +1337,55 @@ that control how the relationship functions.
:ref:`extra data with a many-to-many relationship :ref:`extra data with a many-to-many relationship
<intermediary-manytomany>`. <intermediary-manytomany>`.
.. attribute:: ManyToManyField.through_fields
.. versionadded:: 1.7
Only used when a custom intermediary model is specified. Django will
normally determine which fields of the intermediary model to use in order
to establish a many-to-many relationship automatically. However,
consider the following models::
from django.db import models
class Person(models.Model):
name = models.CharField(max_length=50)
class Group(models.Model):
name = models.CharField(max_length=128)
members = models.ManyToManyField(Person, through='Membership', through_fields=('person', 'group'))
class Membership(models.Model):
person = models.ForeignKey(Person)
group = models.ForeignKey(Group)
inviter = models.ForeignKey(Person, related_name="membership_invites")
invite_reason = models.CharField(max_length=64)
``Membership`` has *two* foreign keys to ``Person`` (``person`` and
``inviter``), which makes the relationship ambiguous and Django can't know
which one to use. In this case, you must explicitly specify which
foreign keys Django should use using ``through_fields``, as in the example
above.
``through_fields`` accepts a 2-tuple ``('field1', 'field2')``, where
``field1`` is the name of the foreign key to the target model (``person``
in this case), and ``field2`` the name of the foreign key to the model the
:class:`ManyToManyField` is defined on (``group`` in this case).
When you have more than one foreign key on an intermediary model to any
(or even both) of the models participating in a many-to-many relationship,
you *must* specify ``through_fields``. This also applies to
:ref:`recursive relationships <recursive-relationships>`
when an intermediary model is used and there are more than two
foreign keys to the model, or you want to explicitly specify which two
Django should use.
Recursive relationships using an intermediary model are always defined as
non-symmetrical -- that is, with :attr:`symmetrical=False <ManyToManyField.symmetrical>`
-- therefore, there is the concept of a "source" and a "target". In that
case ``'field1'`` will be treated as the "source" of the relationship and
``'field2'`` as the "target".
.. attribute:: ManyToManyField.db_table .. attribute:: ManyToManyField.db_table
The name of the table to create for storing the many-to-many data. If this The name of the table to create for storing the many-to-many data. If this

View File

@ -658,6 +658,12 @@ Models
* You can use a single list for :attr:`~django.db.models.Options.index_together` * You can use a single list for :attr:`~django.db.models.Options.index_together`
(rather than a list of lists) when specifying a single set of fields. (rather than a list of lists) when specifying a single set of fields.
* Custom intermediate models having more than one foreign key to any of the
models participating in a many-to-many relationship are now permitted,
provided you explicitly specify which foreign keys should be used by setting
the new :attr:`ManyToManyField.through_fields <django.db.models.ManyToManyField.through_fields>`
argument.
Signals Signals
^^^^^^^ ^^^^^^^

View File

@ -434,30 +434,38 @@ something like this::
invite_reason = models.CharField(max_length=64) invite_reason = models.CharField(max_length=64)
When you set up the intermediary model, you explicitly specify foreign When you set up the intermediary model, you explicitly specify foreign
keys to the models that are involved in the ManyToMany relation. This keys to the models that are involved in the many-to-many relationship. This
explicit declaration defines how the two models are related. explicit declaration defines how the two models are related.
There are a few restrictions on the intermediate model: There are a few restrictions on the intermediate model:
* Your intermediate model must contain one - and *only* one - foreign key * Your intermediate model must contain one - and *only* one - foreign key
to the target model (this would be ``Person`` in our example). If you to the target model (this would be ``Person`` in our example), or you must
have more than one foreign key, a validation error will be raised. explicitly specify the foreign keys Django should use for the relationship
using :attr:`ManyToManyField.through_fields <ManyToManyField.through_fields>`.
If you have more than one foreign key and ``through_fields`` is not
specified, a validation error will be raised. A similar restriction applies
to the foreign key to the source model (this would be ``Group`` in our
example).
* Your intermediate model must contain one - and *only* one - foreign key * For a model which has a many-to-many relationship to itself through an
to the source model (this would be ``Group`` in our example). If you intermediary model, two foreign keys to the same model are permitted, but
have more than one foreign key, a validation error will be raised. they will be treated as the two (different) sides of the many-to-many
relationship. If there are *more* than two foreign keys though, you
* The only exception to this is a model which has a many-to-many must also specify ``through_fields`` as above, or a validation error
relationship to itself, through an intermediary model. In this will be raised.
case, two foreign keys to the same model are permitted, but they
will be treated as the two (different) sides of the many-to-many
relation.
* When defining a many-to-many relationship from a model to * When defining a many-to-many relationship from a model to
itself, using an intermediary model, you *must* use itself, using an intermediary model, you *must* use
:attr:`symmetrical=False <ManyToManyField.symmetrical>` (see :attr:`symmetrical=False <ManyToManyField.symmetrical>` (see
:ref:`the model field reference <manytomany-arguments>`). :ref:`the model field reference <manytomany-arguments>`).
.. versionchanged:: 1.7
In Django 1.6 and earlier, intermediate models containing more than one
foreign key to any of the models involved in the many-to-many relationship
used to be prohibited.
Now that you have set up your :class:`~django.db.models.ManyToManyField` to use Now that you have set up your :class:`~django.db.models.ManyToManyField` to use
your intermediary model (``Membership``, in this case), you're ready to start your intermediary model (``Membership``, in this case), you're ready to start
creating some many-to-many relationships. You do this by creating instances of creating some many-to-many relationships. You do this by creating instances of

View File

@ -3,6 +3,7 @@ from __future__ import unicode_literals
from django.core.checks import Error from django.core.checks import Error
from django.db import models from django.db import models
from django.db.models.fields import FieldDoesNotExist
from django.test.utils import override_settings from django.test.utils import override_settings
from django.test.testcases import skipIfDBFeature from django.test.testcases import skipIfDBFeature
@ -81,7 +82,9 @@ class RelativeFieldTests(IsolatedModelsTestCase):
Error( Error(
("The model is used as an intermediate model by " ("The model is used as an intermediate model by "
"'invalid_models_tests.Group.field', but it has more than one " "'invalid_models_tests.Group.field', but it has more than one "
"foreign key to 'Person', which is ambiguous."), "foreign key to 'Person', which is ambiguous. You must specify "
"which foreign key Django should use via the through_fields "
"keyword argument."),
hint=('If you want to create a recursive relationship, use ' hint=('If you want to create a recursive relationship, use '
'ForeignKey("self", symmetrical=False, ' 'ForeignKey("self", symmetrical=False, '
'through="AmbiguousRelationship").'), 'through="AmbiguousRelationship").'),
@ -205,8 +208,10 @@ class RelativeFieldTests(IsolatedModelsTestCase):
Error( Error(
("The model is used as an intermediate model by " ("The model is used as an intermediate model by "
"'invalid_models_tests.Person.friends', but it has more than two " "'invalid_models_tests.Person.friends', but it has more than two "
"foreign keys to 'Person', which is ambiguous."), "foreign keys to 'Person', which is ambiguous. You must specify "
hint=None, "which two foreign keys Django should use via the through_fields "
"keyword argument."),
hint='Use through_fields to specify which two foreign keys Django should use.',
obj=InvalidRelationship, obj=InvalidRelationship,
id='fields.E333', id='fields.E333',
), ),
@ -1051,3 +1056,84 @@ class ComplexClashTests(IsolatedModelsTestCase):
), ),
] ]
self.assertEqual(errors, expected) self.assertEqual(errors, expected)
class M2mThroughFieldsTests(IsolatedModelsTestCase):
def test_m2m_field_argument_validation(self):
"""
Tests that ManyToManyField accepts the ``through_fields`` kwarg
only if an intermediary table is specified.
"""
class Fan(models.Model):
pass
self.assertRaisesMessage(
ValueError, 'Cannot specify through_fields without a through model',
models.ManyToManyField, Fan, through_fields=('f1', 'f2'))
def test_invalid_m2m_field(self):
"""
Tests that providing invalid source field name to ManyToManyField.through_fields
raises FieldDoesNotExist.
"""
class Fan(models.Model):
pass
class Event(models.Model):
invitees = models.ManyToManyField(Fan, through='Invitation', through_fields=('invalid_field', 'invitee'))
class Invitation(models.Model):
event = models.ForeignKey(Event)
invitee = models.ForeignKey(Fan)
inviter = models.ForeignKey(Fan, related_name='+')
with self.assertRaisesMessage(FieldDoesNotExist, 'invalid_field'):
Event().invitees.all()
def test_invalid_m2m_reverse_field(self):
"""
Tests that providing invalid reverse field name to ManyToManyField.through_fields
raises FieldDoesNotExist.
"""
class Fan(models.Model):
pass
class Event(models.Model):
invitees = models.ManyToManyField(Fan, through='Invitation', through_fields=('event', 'invalid_field'))
class Invitation(models.Model):
event = models.ForeignKey(Event)
invitee = models.ForeignKey(Fan)
inviter = models.ForeignKey(Fan, related_name='+')
with self.assertRaisesMessage(FieldDoesNotExist, 'invalid_field'):
Event().invitees.all()
def test_explicit_field_names(self):
"""
Tests that if ``through_fields`` kwarg is given, it must specify both
link fields of the intermediary table.
"""
class Fan(models.Model):
pass
class Event(models.Model):
invitees = models.ManyToManyField(Fan, through='Invitation', through_fields=(None, 'invitee'))
class Invitation(models.Model):
event = models.ForeignKey(Event)
invitee = models.ForeignKey(Fan)
inviter = models.ForeignKey(Fan, related_name='+')
field = Event._meta.get_field('invitees')
errors = field.check(from_model=Event)
expected = [
Error(
("The field is given an iterable for through_fields, "
"which does not provide the names for both link fields "
"that Django should use for the relation through model "
"'invalid_models_tests.Invitation'."),
hint=None,
obj=field,
id='fields.E337')]
self.assertEqual(expected, errors)

View File

@ -77,3 +77,39 @@ class Friendship(models.Model):
first = models.ForeignKey(PersonSelfRefM2M, related_name="rel_from_set") first = models.ForeignKey(PersonSelfRefM2M, related_name="rel_from_set")
second = models.ForeignKey(PersonSelfRefM2M, related_name="rel_to_set") second = models.ForeignKey(PersonSelfRefM2M, related_name="rel_to_set")
date_friended = models.DateTimeField() date_friended = models.DateTimeField()
# Custom through link fields
@python_2_unicode_compatible
class Event(models.Model):
title = models.CharField(max_length=50)
invitees = models.ManyToManyField(Person, through='Invitation', through_fields=('event', 'invitee'), related_name='events_invited')
def __str__(self):
return self.title
class Invitation(models.Model):
event = models.ForeignKey(Event, related_name='invitations')
# field order is deliberately inverted. the target field is "invitee".
inviter = models.ForeignKey(Person, related_name='invitations_sent')
invitee = models.ForeignKey(Person, related_name='invitations')
@python_2_unicode_compatible
class Employee(models.Model):
name = models.CharField(max_length=5)
subordinates = models.ManyToManyField('self', through="Relationship", through_fields=('source', 'target'), symmetrical=False)
class Meta:
ordering = ('pk',)
def __str__(self):
return self.name
class Relationship(models.Model):
# field order is deliberately inverted.
another = models.ForeignKey(Employee, related_name="rel_another_set", null=True)
target = models.ForeignKey(Employee, related_name="rel_target_set")
source = models.ForeignKey(Employee, related_name="rel_source_set")

View File

@ -6,7 +6,7 @@ from operator import attrgetter
from django.test import TestCase from django.test import TestCase
from .models import (Person, Group, Membership, CustomMembership, from .models import (Person, Group, Membership, CustomMembership,
PersonSelfRefM2M, Friendship) PersonSelfRefM2M, Friendship, Event, Invitation, Employee, Relationship)
class M2mThroughTests(TestCase): class M2mThroughTests(TestCase):
@ -276,6 +276,33 @@ class M2mThroughTests(TestCase):
attrgetter("name") attrgetter("name")
) )
def test_through_fields(self):
"""
Tests that relations with intermediary tables with multiple FKs
to the M2M's ``to`` model are possible.
"""
event = Event.objects.create(title='Rockwhale 2014')
Invitation.objects.create(event=event, inviter=self.bob, invitee=self.jim)
Invitation.objects.create(event=event, inviter=self.bob, invitee=self.jane)
self.assertQuerysetEqual(event.invitees.all(), [
'Jane',
'Jim',
], attrgetter('name'))
def test_through_fields_self_referential(self):
john = Employee.objects.create(name='john')
peter = Employee.objects.create(name='peter')
mary = Employee.objects.create(name='mary')
harry = Employee.objects.create(name='harry')
Relationship.objects.create(source=john, target=peter, another=None)
Relationship.objects.create(source=john, target=mary, another=None)
Relationship.objects.create(source=john, target=harry, another=peter)
self.assertQuerysetEqual(john.subordinates.all(), [
'peter',
'mary',
'harry',
], attrgetter('name'))
def test_query_tests(self): def test_query_tests(self):
Membership.objects.create(person=self.jim, group=self.rock) Membership.objects.create(person=self.jim, group=self.rock)
m2 = Membership.objects.create(person=self.jane, group=self.rock) m2 = Membership.objects.create(person=self.jane, group=self.rock)