Opened 6 weeks ago
Last modified 4 weeks ago
#35941 assigned New feature
Add composite GenericForeignKey support — at Version 6
Reported by: | Csirmaz Bendegúz | Owned by: | Csirmaz Bendegúz |
---|---|---|---|
Component: | Database layer (models, ORM) | Version: | dev |
Severity: | Normal | Keywords: | |
Cc: | Csirmaz Bendegúz, Peter Thomassen | Triage Stage: | Accepted |
Has patch: | yes | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description (last modified by )
This is a follow up to #373 (CompositePrimaryKey).
Proposal:
My proposal is to implement GenericForeignKey support with JSON.
object_id
is aCharField
(orTextField
)- CompositePrimaryKey is stored as a JSON array in
object_id
- JOINs can be achieved with JSON functions (varies per db backend)
*Joins:*
After some experimentation with JSON functions, I believe the simplest solution is to construct JOINs like this:
JOIN ON ((object_id::jsonb)->>0)::integer = id AND ((object_id::jsonb)->>1)::uuid = uuid
Casting is a pain point, especially when joining on DateTimeField
s, as we need to make sure the two columns are in the same format.
Risks:
What if someone is using a JSON array as the primary key (but it's not a composite primary key)?
Before deserializing the JSON array, we need to check if the content type has a composite primary key or not.
What if the db backend doesn't support JSON functions?
All supported databases support JSON functions.
Notes:
- JOINs must work with Unicode characters
- int, date, datetime, uuid, text fields must be supported
- Django admin's
LogEntry
has its own implementation of "generic foreign keys". The approach we take withGenericForeignKey
should also apply toLogEntry
.
Any feedback is appreciated!
Change History (6)
comment:1 by , 6 weeks ago
Triage Stage: | Unreviewed → Accepted |
---|
comment:3 by , 6 weeks ago
I think the proposal looks great Ben, thanks for creating this ticket.
One thing that comes to mind as potentially problematic in JSON array comparison for JOIN
is the treatment of Unicode characters as we're using json.dumps(ensure_ascii=False)
as well as quotes and other characters that need to be escaped in JSON. It'd be great to add test coverage for these.
As for the risks I appreciate you raising them but I think that supporting composite generic foreign key composed of standard members (int, date, datetime, uuid, text) is already a good stretch and I would feel comfortable saying that anything that falls in the exotic primary key category should not necessarily be supported in the first place.
comment:4 by , 6 weeks ago
Description: | modified (diff) |
---|---|
Owner: | set to |
Status: | new → assigned |
Yes, thanks for raising these points Simon!
comment:5 by , 6 weeks ago
Cc: | added |
---|
comment:6 by , 5 weeks ago
Description: | modified (diff) |
---|---|
Has patch: | set |
Thank you Ben for creating this ticket. Accepting as described in ticket:373#comment:204.