Skip to main content

A framework for managing database views based on Django, without the migrations

Project description

django_orm_views

The package for managing database views using django, without the migrations. It can be used for generating database views and reading from them using django orm system.

What does this support?

  • This package adds support to Django for writing Postgres views using:
    • Raw SQL
    • Django Querysets

They look something like this:

class MySQLView(PostgresViewFromSQL):
    sql = """
       SELECT col_a, col_b FROM table_1;
    """
class MyQuerysetView(PostgresViewFromQueryset):
    @classmethod
    def get_queryset(cls):
        return (
            Table1
            .objects
            .values('col_a', 'col_b')
        )
  • The package allows reading from a view using django ORM. For a view to be readable, it has to be inherited from one of the readable view abstractions - ReadableViewFromQueryset or ReadableViewFromSQL. It would look like this:
class MyReadableView(ReadableViewFromQueryset):
    @classmethod
    def get_queryset(cls):
        return (
            Table1
            .objects
            .values('col_a', 'col_b')
        )
    
result = MyReadableView.objects.all()
result = MyReadableView.objects.filter(name="test")
result = MyReadableView.objects.get(id=1)

The name for the view in the database is generated automatically in the base class - BasePostgresView. It's not possible at the moment to define a custom name in the database for a readable view. Though it's possible if the view is not readable (not inherited from the readable view abstraction).

Since a readable view is basically a not managed django model, you might want to set some custom meta data for it by defining a Meta class inside the readable view ( For example changing the data source - db_table).

! Note that defining a custom Meta class is not allowed in the current version of the package and will lead to an error!

See some more examples in the tests here

This also supports the construction of materialised views via PostgresMaterialisedViewMixin. Note that the function refresh_materialized_view will need to be managed by the user in order to keep these up to date where required.

What does this not support?

  • Any database engines aside from Postgres (unless syntax happens to be the same!)
  • Defining custom name for a readable view in the database. Although it's possible for a view that is not readable.
  • Defining custom Meta class in a readable view as it can be done for a django model to define additional settings E.g. db_table

When should I use this?

Our use-case is for a database which is managed by Django in which we would like to provide an analytics-friendly representation of some of our data. This involves giving analytics direct access to our database (whilst using a permissions framework), but using views to expose the data in a more simple way, as well as obscuring data which we consider personally identifiable/sensitive.

There are other frameworks existing which do similar things, usually including reads via the ORM. We found that these packages all generate migrations (despite being unmanaged) and we wanted to remove this from the django migrations process altogether - there seemed to be no value add by including migrations and they would just muddy our migration states.

Cool! But how do I use this?

  • pip install django-orm-views
  • Add 'django_orm_views' to your INSTALLED_APPS
  • Create a postgres_views.py (file or package) inside any app
  • Add a PostgresViewFromQueryset or PostgresViewFromSQL to your postgres_views.py (as above)
  • run ./manage.py sync_views

A postgres_views.py file might look something like the following:

class ComplexViewFromQueryset(PostgresViewFromQueryset):

    prefix = 'test'  # This is optional, but allows you to prefix the table names for views
    
    @classmethod
    def get_queryset(cls):  # Return a `.values` with the query you desire
        return (
            TestModelWithForeignKey
            .objects
            .all()
            .annotate(
                double_integer_col=F('foreign_key__integer_col') * 2
            )
            .values(
                'id',
                'foreign_key__id',
                'foreign_key__integer_col',
                'foreign_key__character_col',
                'foreign_key__date_col',
                'foreign_key__datetime_col',
                'double_integer_col',
            )
        )

When we run the ./manage.py sync_views, we'll create a view called test_complexviewfromqueryset under the views schema.

Note, you can put ./manage.py sync_views into your CI/CD. It works by:

  • Opening a transaction
  • Dropping the views schema
  • Recreating the views schema
  • Recreating all views under that schema
  • Committing the transaction

What's still to come?

  • Support for more database engines. This currently only supports Postgres, but should be a reasonably light shift to support other database engines.
  • Making the package more configurable using settings.
  • Consideration of 0 downtime deployments with views.
    • Note, this can still be achieved with the current implementation, but a bad migration (with a view depending) could cascade a view and create downtime. Ideally migrations + view creation should happen in a single transaction.

Contributing

Feel free to fork the package and propose changes. The repo comes with a test django project which can be used to effectively test changes. It also demonstrates the functionality pretty well.

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

django-orm-views-2.0.2.tar.gz (12.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

django_orm_views-2.0.2-py3-none-any.whl (12.9 kB view details)

Uploaded Python 3

File details

Details for the file django-orm-views-2.0.2.tar.gz.

File metadata

  • Download URL: django-orm-views-2.0.2.tar.gz
  • Upload date:
  • Size: 12.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.17

File hashes

Hashes for django-orm-views-2.0.2.tar.gz
Algorithm Hash digest
SHA256 67fdedd3261b72ff1bcc1838f566fbc228bd881a50b3ab1db85db10c87bb7256
MD5 93a4631ccda0f7dab80de92c3e2f201f
BLAKE2b-256 20145c7e0f137f7102971bd745174d6f9b23483cfbc79a8dc88b02bf849320a4

See more details on using hashes here.

File details

Details for the file django_orm_views-2.0.2-py3-none-any.whl.

File metadata

File hashes

Hashes for django_orm_views-2.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 6a0e17674f11f272e1fdfef5bc6ea94db726c58990266a76038278572176b52e
MD5 684703b536694bea6452bc3ec31ad2a6
BLAKE2b-256 c9f8d40f9a2a4d5830211a79ede4b5736d67cd05f1167ad6d8839e4b0d58024b

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page