I've had my share of confusion over delete.  Here's a thread that I
had with Malcolm recently:

http://groups.google.com/group/django-users/browse_thread/thread/582f21486c1073eb/7f5743953047d9fa?lnk=raot&fwc=1

However, I guess I'm still quite confused.  One thing I don't
understand is when the "bulk delete" happens.  I thought, for example,
that it happened when one uses the queryset delete.  So I just wrote
myself a little test.

Here's the model.  I have a 'Book' and it has readers (that have
ForeignKey back to Book)

class Publisher(models.Model):
    name = models.CharField(max_length=100)

class Book(models.Model):
    title = models.CharField(max_length=100)
    publisher = models.ForeignKey(Publisher, blank=True, null=True)

class Reader(models.Model):
    book = models.ForeignKey(Book, blank=True, null=True)
    name = models.CharField(max_length=100)

Here's a small snippet of code that creates a book with two readers.
I then delete the book using a queryset delete.  I thought that in
this case neither Book.delete() nor Reader.delete() would get called.
However, the output indicates that preDeleteCallBack() does get
called, once for the deletion of book1, and once for each of the two
reader objects.

  def preDeleteCallBack(sender, instance=None, **kwargs):
      print "PreDelete called for instance", instance

  pre_delete.connect(preDeleteCallBack, sender=Reader)
  pre_delete.connect(preDeleteCallBack, sender=Book)

  pub1 = Publisher.objects.create(name="pub1")
  book1 = Book.objects.create(title="book1", publisher=pub1)
  reader1 = Reader.objects.create(name="reader1", book=book1)
  reader2 = Reader.objects.create(name="reader2", book=book1)

  Book.objects.filter(title="book1").delete()

The output from this call to delete() is:

PreDelete called for instance book1
PreDelete called for instance Reader object
PreDelete called for instance Reader object


In my app, I use a lot of formsets.  I thought that bulk delete got
used for those.  Yet again, when I tested it using a little debug app
I find that the preDeleteCallBack() function gets called.  For
example:

    bookPostDict = {
    'form-INITIAL_FORMS': u'1', 'form-TOTAL_FORMS': u'1', 'form-0-
title': u'book1', 'form-0-DELETE': u'on', 'form-0-id': u'1', 'form-0-
publisher': u'1',
    }

    bookFormSet = BookFormSet(data=bookPostDict)
    if bookFormSet.is_valid():
        bookFormSet.save()

When bookFormSet.save() gets executed, I get this output:

PreDelete called for instance book1
PreDelete called for instance Reader object
PreDelete called for instance Reader object

So I guess I am totally confused now.   I know I have had a ton of
problems with my foreignKey objects getting deleted when the thing
they point to gets deleted.  Based on Malcolm's response in the thread
above, it certainly seemed like I was in bulk delete territory, but
using these pruned down examples I find I can't even get a bulk delete
to happen.

George, would you be able to clarify when the bulk delete happens?  I
see in the doc that it says:

"Keep in mind that this will, whenever possible, be executed purely in
SQL, and so the delete() methods of individual object instances will
not necessarily be called during the process."

Perhaps in my queryset delete I am not hitting the case where bulk
delete will be called?  I am also interested particularly in formsets
and what happens when can_delete is True and what happens when the
delete occurs as a result of doing a formset save() when some objects
have been targeted for deletion.

Sorry for the length of this.  I've spent all morning trying to grock
this and am disappointed that my debugging work has resulted in me
being even more confused than before!

Margie



On May 9, 8:33 am, George Song <geo...@damacy.net> wrote:
> On 5/9/2009 7:14 AM, Rune Bromer wrote:
>
>
>
> > Hi,
>
> > I run a small project handling payments from users. Each user has the
> > possibility to delete their account and if they do I have to keep
> > their payment data. Also, I would like to disable admins from deleting
> > the data as there are NEVER a valid reason to do so. The payment model
> > has foreign keys so obviously I need to set these to None before I
> > delete the instances ref'ed by the payment. I do this in a pre_delete
> > () signal for all the models ref'ed by the payment model.
>
> > Also, I created a delete() method on the payment model that will not
> > call the supers delete() when the payment is complete.
>
> > The only thing missing is the admin. As I understand the admin will
> > send a pre_delete() signal so hopefully it will not find the payment
> > objects when deleting a user because the link will go away in the
> > signal. What I'm more worried about is if a stupid admin tries to
> > delete the payment objects themselves. I can remove the actions of
> > cause, but how about the per-object delete functionality? Removing the
> > permission will not work for people having is_superuser = True.
>
> > Hope the problem is understandable. Hopefully I'm overlooking
> > something simple?
>
> If you already did an override on `Payment.delete()`, then you should be
> fine.
>
> Although be mindful of bulk delete, as it doesn't use `Model.delete()`:
> <http://code.djangoproject.com/ticket/11022>
>
> Also you might want to think about setting up a group in auth to have
> finer grained permission control. It's pretty simple to just leave out
> delete for this specific model. Is there a reason they need to be
> superusers?
>
> --
> George
--~--~---------~--~----~------------~-------~--~----~
You received this message because you are subscribed to the Google Groups 
"Django users" group.
To post to this group, send email to django-users@googlegroups.com
To unsubscribe from this group, send email to 
django-users+unsubscr...@googlegroups.com
For more options, visit this group at 
http://groups.google.com/group/django-users?hl=en
-~----------~----~----~----~------~----~------~--~---

Reply via email to