Opened 15 years ago

Closed 13 years ago

#10829 closed (duplicate)

QuerySet.delete() attempts to delete unmanaged models.

Reported by: Justin Lilly Owned by: nobody
Component: Database layer (models, ORM) Version: 1.1-beta
Severity: Keywords: unmanaged
Cc: Triage Stage: Design decision needed
Has patch: no Needs documentation: no
Needs tests: no Patch needs improvement: no
Easy pickings: no UI/UX: no

Description

When there is an unmanaged model which has a ForeignKey to an object that you're attempting to delete, the ORM attempts to delete it. In my case, this is a database view which doesn't support deletion.

I'll update this with a bit more information and a method for reproducing the error later today, just want to get it on the radar.

I'm also assigning this to the 1.1 milestone as not being able to delete an object associated with a managed model is kind of a big deal. Feel free to override me on it though.

Attachments (1)

django.db.models.query.py.patch (2.0 KB ) - added by philroche 15 years ago.
Patch for MySQL unmanaged view (discussed in comment#7)

Download all attachments as: .zip

Change History (11)

comment:1 by Alex Gaynor, 15 years ago

Hrm, this will require some thought, as an unmanaged model *could* indicate something like a db view that you can't delete from, however that semantic is in no way implied by the documentation.

comment:2 by Alex Gaynor, 15 years ago

Triage Stage: UnreviewedDesign decision needed

comment:3 by anonymous, 15 years ago

I would say that the proper way to handle this would be to have a kwarg for the delete method. QuerySet.delete(cascade=False) Seems reasonable, at least.

comment:4 by Malcolm Tredinnick, 15 years ago

Urgh. This is messy. The idea in comment:3 isn't a solution, since this isn't a property of the delete() call -- it's a property of the model definition (and cascading delete behaviour is the subject of another ticket and unresolved as yet because it's a hard problem to cover all use-cases). The current behaviour is correct (unmanaged refers to syncdb and other SQL-altering statements), but we should be able to handle Justin's case, too.

Would be nice to have a solution for 1.1, so I'll give this some thought.

comment:5 by Justin Lilly, 15 years ago

For any looking for a stop-gap, I've gotten past it by writing a delete() method which invalidates the FK used to populate my view which then allows it to delete cleanly. Code is:

    def delete(self):
        baseid = self.baseobject_ptr_id
        BaseRelation.objects.filter(models.Q(base_obj=baseid) | models.Q(related_obj=baseid)).delete()
        super(Video, self).delete()

comment:6 by Jacob, 15 years ago

milestone: 1.1

This is probably a subset of the "deletes always cascade" bug, but even if not this won't be fixed in the 1.1 timeframe.

comment:7 by philroche, 15 years ago

This is a show stopper for us so I have a small patch

line 1002 django\db\models\query.py

perform_delete = True

if cls._meta.managed == False:
    check_view = "select count(TABLE_NAME) from INFORMATION_SCHEMA.VIEWS WHERE TABLE_NAME = '%s';" % cls._meta.db_table
    cursor = connection.cursor()
    cursor.execute(check_view, [])
    if not cursor.rowcount == 0:
        perform_delete = False
    
if perform_delete:
   ...... the rest of the deletion of code

It's ugly as hell I know but a start.

by philroche, 15 years ago

Patch for MySQL unmanaged view (discussed in comment#7)

in reply to:  1 comment:8 by dvainsencher, 14 years ago

Replying to Alex:

Hrm, this will require some thought, as an unmanaged model *could* indicate something like a db view that you can't delete from, however that semantic is in no way implied by the documentation.

I think that if you create a structure that can represent a view it's API should know how to deal with database changes operations. It's look like a bug to me.

comment:9 by ticcky, 14 years ago

Yet another dirty solution... =)

--- django/db/models/.svn/text-base/query.py.svn-base	2010-06-21 13:58:01.000000000 +0200
+++ django/db/models/query.py	2010-07-13 13:38:44.000000000 +0200
@@ -1304,6 +1304,8 @@
     obj_pairs = {}
     try:
         for cls in ordered_classes:
+            if cls._meta.managed == False:
+                continue
             items = seen_objs[cls].items()
             items.sort()
             obj_pairs[cls] = items
@@ -1327,6 +1329,8 @@
 
         # Now delete the actual data.
         for cls in ordered_classes:
+            if cls._meta.managed == False:
+                continue
             items = obj_pairs[cls]
             items.reverse()
 

comment:10 by Carl Meyer, 13 years ago

Resolution: duplicate
Status: newclosed

Marking as duplicate of #7539; being able to turn off cascade per-FK is the correct solution here, since unmanaged does not imply unmodifiable.

Note: See TracTickets for help on using tickets.
Back to Top