Gives the ability to dynamically configure SQL For clause for models. This give you ability to wrap any sql into models and use ORM features on it.
Project description
IDEA
Be able to define the sql FROM clause dynamically and fill it with args. On django models the sql FROM clause is the db table name or other static name (configured in Meta).
The idea is to change that!. By that we are able to map a tabular functions, any sql/queries outputs, and other, to Django models!
It is what we are trying to do here.
Anything which have tabular interface output, like: table, view, function, queries, and so on, should be able to map to dedicated django model and be able to use the orm methods (like select related, prefetch, annotations and others).
Examples:
Wrap aggregation result
# regular models
class Owner(models.Model):
name = models.CharField(max_length=512)
class InventoryRecord(models.Model):
count = models.IntegerField()
owner = models.ForeignKey(Owner, related_name='inventory_records', on_delete=models.CASCADE)
# Our perspective for the InventoryRecordQuerySet
class AggregatedInventoryPerspective(DynamicFromClauseBaseModel):
count_sum = models.IntegerField()
owner = models.ForeignKey(Owner, related_name='+', on_delete=models.DO_NOTHING, primary_key=True)
# Lets make some aggregations
aggr_inv_records_queryset = InventoryRecord.objects.values("owner").annotate(count_sum=models.Sum("count"))
# Let use ORM on the results from the aggr_inv_records_queryset
aggregated_inv_records = AggregatedInventoryPerspective.objects.set_source_from_queryset(
aggr_inv_records_queryset
).select_related('owner')
Filter trough results of the window annotation on same queryset
# Regular django model, with extra objects manager
class Human(models.Model):
objects = QuerySet.as_manager()
dynamic_from_clause_objects = DynamicFromClauseQuerySet.as_manager()
weight = models.IntegerField()
height = models.IntegerField()
# We would like to annotate rank, and filter trought it,
# which is imposible in regular django without raw query.
# we can easy solve it here:
humans_with_rank = Human.objects.all().annotate(rank=Window(
expression=Rank(),
order_by=[F('height'), F('weight')]
))
# Now we can use our manager, to make query from the query
human_with_rank_equal_two = Human.dynamic_from_clause_objects.set_source_from_queryset(
humans_with_rank, forward_fields=['rank']
).filter(rank=2)
Let's use some database functions - check which rows are lock-ed on provided table
class PGRowLocks(Func):
function = 'pgrowlocks'
template = "%(function)s('%(expressions)s')"
# This model maps to the pgrowslocks function output which is all locks on provided table
class PgRowsLocks(DynamicBaseModel):
EXPRESSION_CLASS = PGRowLocks
locked_row = ArrayField(models.PositiveIntegerField(), size=2, primary_key=True)
locker = models.PositiveBigIntegerField()
multi = models.BooleanField()
xids = ArrayField(models.PositiveIntegerField())
modes = models.PositiveIntegerField(models.TextField())
pids = ArrayField(models.SmallIntegerField())
# Now we can easy check what is locked on which table :)
locked_rows = PgRowsLocks.objects.fill_expression_with_parameters(
SomeMode._meta.db_table
).all()
My tabular function
cooming soon, for now check tests
Note:
We have to specify which field is the primary key on the model
How it works?
The Code is easy. The only thing which we do here is to extend the django SQL compiler and change how it creates the from_clause. The library has very little code.
Motivation
I think that this approach has sense cus I saw a lot of problems or ugly solutions which have tried to:
- use table functions,
- serialize objects on aggregated queryset,
- make selects over nested queries,
- replacing what database should do with python code,
- "manually" prefetching on serializers lvl,
- and others ugly things.
I think that this library contains a good idea, and a reasonable attempt, to solve issues like the above.
TODO:
- Add tests across multiple django versions
- Migrations (here or in other library like the django-db-views - db functions can be a good replacement for views, cus views always calculate the whole dataset which can raise performance issues).
How to work with repo
add your .env file in the main directory, which set up POSTGRES env variables. See conftest.py file.
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file django-dynamic-from-clause-0.0.1.tar.gz
.
File metadata
- Download URL: django-dynamic-from-clause-0.0.1.tar.gz
- Upload date:
- Size: 5.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.1 CPython/3.10.4
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 5659cabede8d5e69626a28b4bb3b7e23d9a2114e240d366cacfa22fd5270b53f |
|
MD5 | e766a06277771dab8a38564200209850 |
|
BLAKE2b-256 | b9081c9caa904fb52385220515001cdba84e3dc9b4dfa5d60e6f329f32283e26 |
File details
Details for the file django_dynamic_from_clause-0.0.1-py3-none-any.whl
.
File metadata
- Download URL: django_dynamic_from_clause-0.0.1-py3-none-any.whl
- Upload date:
- Size: 6.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.1 CPython/3.10.4
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 0ac40d4919d6a6cd81beafffb1e58c58bc887dccbc930784cb4e6e0b7d61c87e |
|
MD5 | b89d49089efca980ca9107061ed6071f |
|
BLAKE2b-256 | 11fa60447cc46b785354dc2af44b5cf0c3e03950d87803ca2fbb114ff813377f |