2016-05-12 35 views
1

In meinem Django-Projekt arbeite ich mit Legacy-Datenbank, die eine Tabelle mit mehreren Primärschlüsseln hat. Ich stehe vor dem Problem, dass Django ORM falsch SQL-Abfragen erstellt, wenn ich die Methode save() für eine bestimmte Modellinstanz aufruft.Mehrere Primärschlüssel in Django-Projekt mit Legacy-Datenbank

Zum Beispiel habe ich folgendes Modell:

class MyTable(models.Model): 
    field_1 = models.IntegerField(primary_key=True) 
    field_2 = models.BooleanField(default=False, null=False) 
    field_3 = models.BooleanField(default=False, null=False) 
    user = models.ForeignKey(
     CustomUser, models.DO_NOTHING, db_column='userid', primary_key=True) 

    class Meta: 
     managed = False 
     db_table = 'my_table' 
     unique_together = (('user', 'field_1'),) 

ich der Instanz Modell wie folgt aus: mytable_instance = MyTable.objects.all()[0]. Dann möchte ich diese Instanz ändern:

mytable_instance.field_2 = True 
mytable_instance.field_2.save() 

Allerdings führt Django ORM diese Abfrage:

{'sql': 'UPDATE `my_table` SET `field_2` = 1, `field_3` = 0 WHERE `my_table`.`field_1` = 123', 'time': '0.000'} 

, die nicht korrekt ist, weil es neue Werte für field_2 und field_3 für alle Zeilen in my_table gesetzt wird mit field_1 = 123

Wie kann ich dieses Problem überwinden? Ich brauche eine SQL-Abfrage wie folgt aus:

'UPDATE `my_table` SET `field_2` = 1, `field_3` = 0 WHERE `my_table`.`field_1` = 123 AND `my_table`.`user_id` = 1' 

EDIT: vollständige Code

Modell für Legacy-Datenbanktabelle:

class MyTable(models.Model): 
    field1_id = models.IntegerField(db_column='field1id', primary_key=True) 
    is_favorite = models.BooleanField(db_column='isfavorite', default=False, null=False) 
    is_admin = models.BooleanField(db_column='isadmin', default=False, null=False) 
    role = models.IntegerField(default=USER_GUEST, null=False) 
    field2 = models.BooleanField(null=False, default=True) 
    field3 = models.BooleanField(null=False, default=True) 
    is_active = models.BooleanField(db_column='isactive', null=False, default=True) 

    user = models.ForeignKey(
     CustomUser, models.DO_NOTHING, db_column='userid', primary_key=True) 

    DB_NAME = 'default' 

    class Meta: 
     managed = False 
     db_table = 'mytable' 
     unique_together = (('user', 'field1_id'),) 

Abfragen:

>>> from web_services.apps.my_app.models import MyTable 
>>> g = MyTable.objects.get(field1_id=12) 
>>> g.is_active = True 
>>> g.save() 
>>> connection.queries[-1] 
{'time': '0.000', 'sql': 'UPDATE `mytable` SET `isfavorite` = 0, `isadmin` = 1, `role` = 3, `field2` = 1, `field3` = 1, `isactive` = 1 WHERE `mytable`.`field1id` = 12'} 

Aber ich brauche :

{'time': '0.000', 'sql': 'UPDATE `mytable` SET `isfavorite` = 0, `isadmin` = 1, `role` = 3, `field2` = 1, `field3` = 1, `isactive` = 1 WHERE `mytable`.`field1id` = 12' AND `mytable`.`userid` = 1'} 
+0

Dies ist kaum plausibel. Bitte posten Sie Ihren vollständigen Code. Ich vermute, dass ein Problem in Ihrem Code dazu führt. – e4c5

+0

@ e4c5 Ich habe Update oben hinzugefügt, danke! – AmirM

Antwort

1

Blick Lassen Sie uns an diesen beiden Anfragen:

UPDATE `mytable` 
SET `isfavorite` = 0, `isadmin` = 1, `role` = 3, `field2` = 1, `field3` = 1, `isactive` = 1 
WHERE `mytable`.`field1id` = 12 

Und

UPDATE `mytable` 
SET `isfavorite` = 0,`isadmin` = 1, `role` = 3, `field2` = 1, `field3` = 1, `isactive` = 1 
WHERE `mytable`.`field1id` = 12' AND `mytable`.`userid` = 1 

Der einzige Unterschied zwischen den beiden ist die AND userid=1 Bit.

Es ist in der Tat richtig, wenn Sie sagen, dass

es neue Werte für field_2 und field_3 für alle Zeilen in my_table mit field1_id = 12

Es gibt genau eine solche Reihe gesetzt wird. Warum? weil Sie field_id als Primärschlüssel definiert haben. Dies garantiert, dass es eine einzige Zeile mit field1_id = N gibt, wobei N eine ganze Zahl ist.

So django Abfrage verhält sich perfekt.

+0

Das Problem, dass es sich um eine Legacy-Datenbank handelt und diese möglichen Kombinationen von '(field1_id, user)': '(12, 1)', '(12,2)', '(12, 3)', ' (2, 1) ',' (3, 1) 'und usw. Also mit' 'mytable' '.'field1id' = 12' es aktualisiert alle '(12, 1)', '(12,2)', '(12, 3)' – AmirM

+0

Also, meine Frage sollte wahrscheinlich sein, wie Composite-Key zu setzen? – AmirM

+0

Nein, nicht nach dem von Ihnen geposteten Modell. Es ** kann nicht sein (12, 1), (12,2), (12, 3) ** – e4c5