5

我有一个遗留数据库,其中一些表包含复合主键。通过运行manage.py inspectdb命令获得的模型看起来像这样。在遗留数据库中使用django项目中的复合主键

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) 

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

我可以正常取数据。但是,当我想在某个模型实例上运行save()命令时出现问题。查询django执行的是不正确的。例如:

>>> 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'} 

但我需要:

{'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'} 

由于Django不支持复合主键,这将是对解决这一问题的最佳解决方案?请注意,这是传统数据库表,并没有AutoField

编辑:增加了传统的表结构

+------------+------------+------+-----+---------+-------+ 
| Field  | Type  | Null | Key | Default | Extra | 
+------------+------------+------+-----+---------+-------+ 
| userid  | int(11) | NO | PRI | NULL |  | 
| field1id | int(11) | NO | PRI | NULL |  | 
| isfavorite | int(11) | NO |  | 0  |  | 
| isadmin | int(11) | NO |  | 0  |  | 
| role  | int(11) | YES |  | NULL |  | 
| field2  | tinyint(1) | NO |  | 1  |  | 
| field3  | tinyint(1) | NO |  | 1  |  | 
| isactive | tinyint(4) | NO |  | 1  |  | 
+------------+------------+------+-----+---------+-------+ 
+0

如果您添加表格结构,也是最好的。 – e4c5

+0

@ e4c5添加了遗留表格结构 – AmirM

回答

1

不幸的是Django的不not yet have a composite primary key(复合主键也被称为多列主键)

有一个第三方库,毫不奇怪的名字django-compositekey那使创建具有多列主键的模型成为可能。但是该项目现在保持不变,并且与最新版本的django不兼容。

最好的办法是在你的表格中添加一个新的列,它是一个AutoField,并将它作为新的主键。然后可以将构成主键的字段标记为unique_together。虽然这可能不是理想的少数查询。这对大多数人来说已经够用了。

如果使用当前表结构更新您的问题(如sql控制台中所示),我将更新我的答案以显示模型的外观。

更新 有很多不同的方法可以放弃旧的复合主键并将其替换为新的自动增量主键。这是最容易的,它只涉及SQL

CREATE TABLE newtable LIKE mytable; 
ALTER TABLE newtable DROP PRIMARY KEY; 
ALTER TABLE newtable ADD `id` INT NOT NULL AUTO_INCREMENT PRIMARY KEY; 
RENAME TABLE mytable to mytable_old; 
RENAME TABLE newtable to mytable; 
INSERT INTO mytable(userid,field1id, rest of the fields) 
    SELECT * FROM mytable_old; 

然后编辑您的模型并从这些字段中删除primary_key = True标志。

脚注:
像sqlite的一些数据库,例如不支持在CREATE TABLELIKE条款。在这些情况下,您必须查找原始表的create table语句,并在编辑表名后复制粘贴。从您的表格结构来看,您似乎在使用mysql,它支持LIKE子句。

+0

虽然它是'纯粹的SQL'?我不确定'CREATE TABLE'中的'LIKE'是标准SQL,可能是添加到'纯粹MySQL'或类似的东西?(MySQL/MyISAM支持LIKE,但Firebird不支持,不知道其他引擎/ RDMBS) –

+0

@JanSegre你是对的,它不被所有数据库支持。当我在上面的答案中写纯sql的时候,我只是想不涉及django而不是使用ansi标准。更新我的答案。 – e4c5

0

您必须从模型中删除managed = False命令才能编辑您的表格

相关问题