Opened 6 weeks ago

Last modified 10 days ago

#35956 new New feature

Add composite foreign keys

Reported by: Csirmaz Bendegúz Owned by:
Component: Database layer (models, ORM) Version: dev
Severity: Normal Keywords:
Cc: Csirmaz Bendegúz, Mariusz Felisiak Triage Stage: Accepted
Has patch: no Needs documentation: no
Needs tests: no Patch needs improvement: no
Easy pickings: no UI/UX: no

Description

This is a follow up to #373 (CompositePrimaryKey).

Now that composite primary keys are merged, it would be great to be able to create foreign keys referencing them through the Django ORM.

My proposal is to add 2 parameters to ForeignKey: from_fields and to_fields.
They would map to the underlying ForeignObject's from_fields, to_fields parameters.

If a ForeignKey has multiple fields, it acts as a virtual field, meaning it doesn't create a database column automatically.

Change History (4)

comment:1 by Sarah Boyce, 5 weeks ago

Triage Stage: UnreviewedAccepted

in reply to:  description ; comment:2 by Mariusz Felisiak, 2 weeks ago

Replying to Csirmaz Bendegúz:

My proposal is to add 2 parameters to ForeignKey: from_fields and to_fields.

Do we need these parameters? CompositePrimaryKey is always a primary key so its fields should be detected automatically, IMO, the following example should work:

class Release(models.Model):
    pk = models.CompositePrimaryKey("version", "name")
    version = models.IntegerField()
    name = models.CharField(max_length=20)


class RefRelease(models.Model):
    release = models.ForeignKey("Release", models.CASCADE)

comment:3 by Mariusz Felisiak, 2 weeks ago

Cc: Mariusz Felisiak added

in reply to:  2 comment:4 by Csirmaz Bendegúz, 10 days ago

Replying to Mariusz Felisiak:

Do we need these parameters? CompositePrimaryKey is always a primary key so its fields should be detected automatically, IMO, the following example should work:

class Release(models.Model):
    pk = models.CompositePrimaryKey("version", "name")
    version = models.IntegerField()
    name = models.CharField(max_length=20)


class RefRelease(models.Model):
    release = models.ForeignKey("Release", models.CASCADE)

The issue with this example is RefRelease must have 2 fields matching the CompositePrimaryKey (an IntegerField() and a CharField(max_length=20)).

So release would need to create 2 fields implicitly and I think that's too restrictive.

We should allow sharing fields between composite foreign keys.

For example, in a multi-tenant app where each model has a CompositePrimaryKey("tenant_id", "id"), the tenant_id field should be shared between composite foreign keys.

Let me know what you think, I'm happy to consider any alternatives.

Version 1, edited 10 days ago by Csirmaz Bendegúz (previous) (next) (diff)
Note: See TracTickets for help on using tickets.
Back to Top