Fixed #3121 -- Made `get_or_create()` work for `RelatedManager` and `ManyRelatedManager`.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@8415 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
a2be52fd2a
commit
a949f9ec7d
|
@ -311,6 +311,13 @@ class ForeignRelatedObjectsDescriptor(object):
|
||||||
return new_obj
|
return new_obj
|
||||||
create.alters_data = True
|
create.alters_data = True
|
||||||
|
|
||||||
|
def get_or_create(self, **kwargs):
|
||||||
|
# Update kwargs with the related object that this
|
||||||
|
# ForeignRelatedObjectsDescriptor knows about.
|
||||||
|
kwargs.update({rel_field.name: instance})
|
||||||
|
return super(RelatedManager, self).get_or_create(**kwargs)
|
||||||
|
get_or_create.alters_data = True
|
||||||
|
|
||||||
# remove() and clear() are only provided if the ForeignKey can have a value of null.
|
# remove() and clear() are only provided if the ForeignKey can have a value of null.
|
||||||
if rel_field.null:
|
if rel_field.null:
|
||||||
def remove(self, *objs):
|
def remove(self, *objs):
|
||||||
|
@ -409,6 +416,16 @@ def create_many_related_manager(superclass, through=False):
|
||||||
return new_obj
|
return new_obj
|
||||||
create.alters_data = True
|
create.alters_data = True
|
||||||
|
|
||||||
|
def get_or_create(self, **kwargs):
|
||||||
|
obj, created = \
|
||||||
|
super(ManyRelatedManager, self).get_or_create(**kwargs)
|
||||||
|
# We only need to add() if created because if we got an object back
|
||||||
|
# from get() then the relationship already exists.
|
||||||
|
if created:
|
||||||
|
self.add(obj)
|
||||||
|
return obj, created
|
||||||
|
get_or_create.alters_data = True
|
||||||
|
|
||||||
def _add_items(self, source_col_name, target_col_name, *objs):
|
def _add_items(self, source_col_name, target_col_name, *objs):
|
||||||
# join_table: name of the m2m link table
|
# join_table: name of the m2m link table
|
||||||
# source_col_name: the PK colname in join_table for the source object
|
# source_col_name: the PK colname in join_table for the source object
|
||||||
|
|
|
@ -0,0 +1,91 @@
|
||||||
|
from django.db import models
|
||||||
|
|
||||||
|
class Publisher(models.Model):
|
||||||
|
name = models.CharField(max_length=100)
|
||||||
|
|
||||||
|
class Author(models.Model):
|
||||||
|
name = models.CharField(max_length=100)
|
||||||
|
|
||||||
|
class Book(models.Model):
|
||||||
|
name = models.CharField(max_length=100)
|
||||||
|
authors = models.ManyToManyField(Author, related_name='books')
|
||||||
|
publisher = models.ForeignKey(Publisher, related_name='books')
|
||||||
|
|
||||||
|
|
||||||
|
__test__ = {'one':"""
|
||||||
|
#
|
||||||
|
# RelatedManager
|
||||||
|
#
|
||||||
|
|
||||||
|
# First create a Publisher.
|
||||||
|
>>> p = Publisher.objects.create(name='Acme Publishing')
|
||||||
|
|
||||||
|
# Create a book through the publisher.
|
||||||
|
>>> book, created = p.books.get_or_create(name='The Book of Ed & Fred')
|
||||||
|
>>> created
|
||||||
|
True
|
||||||
|
|
||||||
|
# The publisher should have one book.
|
||||||
|
>>> p.books.count()
|
||||||
|
1
|
||||||
|
|
||||||
|
# Try get_or_create again, this time nothing should be created.
|
||||||
|
>>> book, created = p.books.get_or_create(name='The Book of Ed & Fred')
|
||||||
|
>>> created
|
||||||
|
False
|
||||||
|
|
||||||
|
# And the publisher should still have one book.
|
||||||
|
>>> p.books.count()
|
||||||
|
1
|
||||||
|
|
||||||
|
#
|
||||||
|
# ManyRelatedManager
|
||||||
|
#
|
||||||
|
|
||||||
|
# Add an author to the book.
|
||||||
|
>>> ed, created = book.authors.get_or_create(name='Ed')
|
||||||
|
>>> created
|
||||||
|
True
|
||||||
|
|
||||||
|
# Book should have one author.
|
||||||
|
>>> book.authors.count()
|
||||||
|
1
|
||||||
|
|
||||||
|
# Try get_or_create again, this time nothing should be created.
|
||||||
|
>>> ed, created = book.authors.get_or_create(name='Ed')
|
||||||
|
>>> created
|
||||||
|
False
|
||||||
|
|
||||||
|
# And the book should still have one author.
|
||||||
|
>>> book.authors.count()
|
||||||
|
1
|
||||||
|
|
||||||
|
# Add a second author to the book.
|
||||||
|
>>> fred, created = book.authors.get_or_create(name='Fred')
|
||||||
|
>>> created
|
||||||
|
True
|
||||||
|
|
||||||
|
# The book should have two authors now.
|
||||||
|
>>> book.authors.count()
|
||||||
|
2
|
||||||
|
|
||||||
|
# Create an Author not tied to any books.
|
||||||
|
>>> Author.objects.create(name='Ted')
|
||||||
|
<Author: Author object>
|
||||||
|
|
||||||
|
# There should be three Authors in total. The book object should have two.
|
||||||
|
>>> Author.objects.count()
|
||||||
|
3
|
||||||
|
>>> book.authors.count()
|
||||||
|
2
|
||||||
|
|
||||||
|
# Try creating a book through an author.
|
||||||
|
>>> ed.books.get_or_create(name="Ed's Recipies", publisher=p)
|
||||||
|
(<Book: Book object>, True)
|
||||||
|
|
||||||
|
# Now Ed has two Books, Fred just one.
|
||||||
|
>>> ed.books.count()
|
||||||
|
2
|
||||||
|
>>> fred.books.count()
|
||||||
|
1
|
||||||
|
"""}
|
Loading…
Reference in New Issue