magic-removal: Fixes #1346 -- Added ability for m2m relations to self to be optionally non-symmetrical. Added unit tests for non-symmetrical behaviour.

git-svn-id: http://code.djangoproject.com/svn/django/branches/magic-removal@2383 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
Russell Keith-Magee 2006-02-24 10:36:04 +00:00
parent 4f6d2a1d08
commit 4708aee311
2 changed files with 117 additions and 10 deletions

View File

@ -275,6 +275,7 @@ class ReverseManyRelatedObjectsDescriptor(object):
qn = backend.quote_name qn = backend.quote_name
this_opts = instance.__class__._meta this_opts = instance.__class__._meta
symmetrical = self.field.rel.symmetrical
rel_model = self.field.rel.to rel_model = self.field.rel.to
rel_opts = rel_model._meta rel_opts = rel_model._meta
join_table = qn(self.field.m2m_db_table()) join_table = qn(self.field.m2m_db_table())
@ -301,8 +302,8 @@ class ReverseManyRelatedObjectsDescriptor(object):
_add_m2m_items(self, superclass, rel_model, join_table, source_col_name, _add_m2m_items(self, superclass, rel_model, join_table, source_col_name,
target_col_name, instance._get_pk_val(), *objs, **kwargs) target_col_name, instance._get_pk_val(), *objs, **kwargs)
# If this is an m2m relation to self, add the mirror entry in the m2m table # If this is a symmmetrical m2m relation to self, add the mirror entry in the m2m table
if instance.__class__ == rel_model: if instance.__class__ == rel_model and symmetrical:
_add_m2m_items(self, superclass, rel_model, join_table, target_col_name, _add_m2m_items(self, superclass, rel_model, join_table, target_col_name,
source_col_name, instance._get_pk_val(), *objs, **kwargs) source_col_name, instance._get_pk_val(), *objs, **kwargs)
@ -312,8 +313,8 @@ class ReverseManyRelatedObjectsDescriptor(object):
_remove_m2m_items(rel_model, join_table, source_col_name, _remove_m2m_items(rel_model, join_table, source_col_name,
target_col_name, instance._get_pk_val(), *objs) target_col_name, instance._get_pk_val(), *objs)
# If this is an m2m relation to self, remove the mirror entry in the m2m table # If this is a symmmetrical m2m relation to self, remove the mirror entry in the m2m table
if instance.__class__ == rel_model: if instance.__class__ == rel_model and symmetrical:
_remove_m2m_items(rel_model, join_table, target_col_name, _remove_m2m_items(rel_model, join_table, target_col_name,
source_col_name, instance._get_pk_val(), *objs) source_col_name, instance._get_pk_val(), *objs)
@ -322,8 +323,8 @@ class ReverseManyRelatedObjectsDescriptor(object):
def clear(self): def clear(self):
_clear_m2m_items(join_table, source_col_name, instance._get_pk_val()) _clear_m2m_items(join_table, source_col_name, instance._get_pk_val())
# If this is an m2m relation to self, clear the mirror entry in the m2m table # If this is a symmmetrical m2m relation to self, clear the mirror entry in the m2m table
if instance.__class__ == rel_model: if instance.__class__ == rel_model and symmetrical:
_clear_m2m_items(join_table, target_col_name, instance._get_pk_val()) _clear_m2m_items(join_table, target_col_name, instance._get_pk_val())
clear.alters_data = True clear.alters_data = True
@ -472,7 +473,8 @@ class ManyToManyField(RelatedField, Field):
related_name=kwargs.pop('related_name', None), related_name=kwargs.pop('related_name', None),
filter_interface=kwargs.pop('filter_interface', None), filter_interface=kwargs.pop('filter_interface', None),
limit_choices_to=kwargs.pop('limit_choices_to', None), limit_choices_to=kwargs.pop('limit_choices_to', None),
raw_id_admin=kwargs.pop('raw_id_admin', False)) raw_id_admin=kwargs.pop('raw_id_admin', False),
symmetrical=kwargs.pop('symmetrical', True))
if kwargs["rel"].raw_id_admin: if kwargs["rel"].raw_id_admin:
kwargs.setdefault("validator_list", []).append(self.isValidIDList) kwargs.setdefault("validator_list", []).append(self.isValidIDList)
Field.__init__(self, **kwargs) Field.__init__(self, **kwargs)
@ -559,8 +561,12 @@ class ManyToManyField(RelatedField, Field):
self.m2m_db_table = curry(self._get_m2m_db_table, cls._meta) self.m2m_db_table = curry(self._get_m2m_db_table, cls._meta)
def contribute_to_related_class(self, cls, related): def contribute_to_related_class(self, cls, related):
setattr(cls, related.get_accessor_name(), ManyRelatedObjectsDescriptor(related, 'm2m')) # m2m relations to self do not have a ManyRelatedObjectsDescriptor,
# Add the descriptor for the m2m relation # as it would be redundant - unless the field is non-symmetrical.
if related.model != related.parent_model or not self.rel.symmetrical:
# Add the descriptor for the m2m relation
setattr(cls, related.get_accessor_name(), ManyRelatedObjectsDescriptor(related, 'm2m'))
self.rel.singular = self.rel.singular or self.rel.to._meta.object_name.lower() self.rel.singular = self.rel.singular or self.rel.to._meta.object_name.lower()
# Set up the accessors for the column names on the m2m table # Set up the accessors for the column names on the m2m table
@ -601,7 +607,7 @@ class OneToOne(ManyToOne):
class ManyToMany: class ManyToMany:
def __init__(self, to, singular=None, related_name=None, def __init__(self, to, singular=None, related_name=None,
filter_interface=None, limit_choices_to=None, raw_id_admin=False): filter_interface=None, limit_choices_to=None, raw_id_admin=False, symmetrical=True):
self.to = to self.to = to
self.singular = singular or None self.singular = singular or None
self.related_name = related_name self.related_name = related_name
@ -609,4 +615,5 @@ class ManyToMany:
self.limit_choices_to = limit_choices_to or {} self.limit_choices_to = limit_choices_to or {}
self.edit_inline = False self.edit_inline = False
self.raw_id_admin = raw_id_admin self.raw_id_admin = raw_id_admin
self.symmetrical = symmetrical
assert not (self.raw_id_admin and self.filter_interface), "ManyToMany relationships may not use both raw_id_admin and filter_interface" assert not (self.raw_id_admin and self.filter_interface), "ManyToMany relationships may not use both raw_id_admin and filter_interface"

View File

@ -4,6 +4,10 @@
In this example, A Person can have many friends, who are also people. Friendship is a In this example, A Person can have many friends, who are also people. Friendship is a
symmetrical relationshiup - if I am your friend, you are my friend. symmetrical relationshiup - if I am your friend, you are my friend.
A person can also have many idols - but while I may idolize you, you may not think
the same of me. 'Idols' is an example of a non-symmetrical m2m field. Only recursive
m2m fields may be non-symmetrical, and they are symmetrical by default.
This test validates that the m2m table will create a mangled name for the m2m table if This test validates that the m2m table will create a mangled name for the m2m table if
there will be a clash, and tests that symmetry is preserved where appropriate. there will be a clash, and tests that symmetry is preserved where appropriate.
""" """
@ -13,6 +17,7 @@ from django.db import models
class Person(models.Model): class Person(models.Model):
name = models.CharField(maxlength=20) name = models.CharField(maxlength=20)
friends = models.ManyToManyField('self') friends = models.ManyToManyField('self')
idols = models.ManyToManyField('self', symmetrical=False, related_name='stalkers')
def __repr__(self): def __repr__(self):
return self.name return self.name
@ -89,4 +94,99 @@ API_TESTS = """
[Chuck] [Chuck]
# Add some idols in the direction of field definition
# Anne idolizes Bill and Chuck
>>> a.idols.add(b,c)
# Bill idolizes Anne right back
>>> b.idols.add(a)
# David is idolized by Anne and Chuck - add in reverse direction
>>> d.stalkers.add(a,c)
# Who are Anne's idols?
>>> a.idols.all()
[Bill, Chuck, David]
# Who is stalking Anne?
>>> a.stalkers.all()
[Bill]
# Who are Bill's idols?
>>> b.idols.all()
[Anne]
# Who is stalking Bill?
>>> b.stalkers.all()
[Anne]
# Who are Chuck's idols?
>>> c.idols.all()
[David]
# Who is stalking Chuck?
>>> c.stalkers.all()
[Anne]
# Who are David's idols?
>>> d.idols.all()
[]
# Who is stalking David
>>> d.stalkers.all()
[Anne, Chuck]
# Bill is already being stalked by Anne - add Anne again, but in the reverse direction
>>> b.stalkers.add(a)
# Who are Anne's idols?
>>> a.idols.all()
[Bill, Chuck, David]
# Who is stalking Anne?
[Bill]
# Who are Bill's idols
>>> b.idols.all()
[Anne]
# Who is stalking Bill?
>>> b.stalkers.all()
[Anne]
# Remove Anne from Bill's list of stalkers
>>> b.stalkers.remove(a)
# Who are Anne's idols?
>>> a.idols.all()
[Chuck, David]
# Who is stalking Anne?
>>> a.stalkers.all()
[Bill]
# Who are Bill's idols?
>>> b.idols.all()
[Anne]
# Who is stalking Bill?
>>> b.stalkers.all()
[]
# Clear Anne's group of idols
>>> a.idols.clear()
# Who are Anne's idols
>>> a.idols.all()
[]
# Reverse relationships should also be gone
# Who is stalking Chuck?
>>> c.stalkers.all()
[]
# Who is friends with David?
>>> d.stalkers.all()
[Chuck]
""" """