Debian packaging for django-tenant-schems
Go to file
Frédéric Péters 683c8ae58a misc: remove usage of django.utils.six 2022-04-15 17:42:02 +02:00
debian debian: use debhelper compat level 12 2021-12-12 13:30:29 +01:00
docs Add missing `post_schema_sync` signal documentation 2018-08-12 21:31:13 +02:00
dts_test_project backported django postgres backend fixes 2017-12-06 16:30:14 +01:00
examples/tenant_tutorial Explicitly define INSTALLED_APPS in settings.py 2017-06-02 18:40:03 +10:00
source debian: initial import 2018-08-12 21:40:09 +02:00
tenant_schemas misc: remove usage of django.utils.six 2022-04-15 17:42:02 +02:00
.editorconfig Refactor ImproperlyConfigured to system check framework 2016-11-11 07:57:14 +11:00
.gitignore Ignore IntelliJ hidden files 2013-10-20 14:24:03 +02:00
.travis.yml Remove Django 1.9 from testing matrix, unsupported version. 2017-06-02 06:57:55 +10:00
CHANGES.txt pypi should now correctly contain all files 2013-03-03 08:44:03 +01:00
LICENSE pypi should now correctly contain all files 2013-03-03 08:44:03 +01:00
MANIFEST.in Corrected invalid syntax on recursive-include for manifest 2014-12-29 18:37:40 +01:00
README.rst Update README to remove non-ASCII characters. Refs #434. 2017-02-22 20:07:46 +11:00
__init__.py pypi should now correctly contain all files 2013-03-03 08:44:03 +01:00
changelog debian: initial import 2018-08-12 21:40:09 +02:00
compat debian: initial import 2018-08-12 21:40:09 +02:00
control debian: only build for python2 2018-08-12 21:40:16 +02:00
files debian: initial import 2018-08-12 21:40:09 +02:00
python-django-tenant-schemas.postinst.debhelper debian: initial import 2018-08-12 21:40:09 +02:00
python-django-tenant-schemas.prerm.debhelper debian: initial import 2018-08-12 21:40:09 +02:00
python-django-tenant-schemas.substvars debian: initial import 2018-08-12 21:40:09 +02:00
python3-django-tenant-schemas.postinst.debhelper debian: initial import 2018-08-12 21:40:09 +02:00
python3-django-tenant-schemas.prerm.debhelper debian: initial import 2018-08-12 21:40:09 +02:00
python3-django-tenant-schemas.substvars debian: initial import 2018-08-12 21:40:09 +02:00
rules debian: only build for python2 2018-08-12 21:40:16 +02:00
setup.cfg Bump version: 1.8.0 → 1.9.0 2017-06-03 21:47:55 +10:00
setup.py misc: adapt version number for PEP 440 2019-01-03 13:35:50 +01:00
tox.ini Remove Django 1.9 from testing matrix, unsupported version. 2017-06-02 06:57:55 +10:00
version.py Refactor ImproperlyConfigured to system check framework 2016-11-11 07:57:14 +11:00

README.rst

django-tenant-schemas
=====================

|PyPi version| |PyPi downloads| |Python versions| |Travis CI| |PostgreSQL|

This application enables `django`_ powered websites to have multiple
tenants via `PostgreSQL schemas`_. A vital feature for every
Software-as-a-Service website.

Django provides currently no simple way to support multiple tenants
using the same project instance, even when only the data is different.
Because we don't want you running many copies of your project, you'll be
able to have:

-  Multiple customers running on the same instance
-  Shared and Tenant-Specific data
-  Tenant View-Routing

What are schemas
----------------

A schema can be seen as a directory in an operating system, each
directory (schema) with it's own set of files (tables and objects). This
allows the same table name and objects to be used in different schemas
without conflict. For an accurate description on schemas, see
`PostgreSQL's official documentation on schemas`_.

Why schemas
-----------

There are typically three solutions for solving the multitenancy
problem.

1. Isolated Approach: Separate Databases. Each tenant has it's own
   database.

2. Semi Isolated Approach: Shared Database, Separate Schemas. One
   database for all tenants, but one schema per tenant.

3. Shared Approach: Shared Database, Shared Schema. All tenants share
   the same database and schema. There is a main tenant-table, where all
   other tables have a foreign key pointing to.

This application implements the second approach, which in our opinion,
represents the ideal compromise between simplicity and performance.

-  Simplicity: barely make any changes to your current code to support
   multitenancy. Plus, you only manage one database.
-  Performance: make use of shared connections, buffers and memory.

Each solution has it's up and down sides, for a more in-depth
discussion, see Microsoft's excellent article on `Multi-Tenant Data
Architecture`_.

How it works
------------

Tenants are identified via their host name (i.e tenant.domain.com). This
information is stored on a table on the ``public`` schema. Whenever a
request is made, the host name is used to match a tenant in the
database. If there's a match, the search path is updated to use this
tenant's schema. So from now on all queries will take place at the
tenant's schema. For example, suppose you have a tenant ``customer`` at
http://customer.example.com. Any request incoming at
``customer.example.com`` will automatically use ``customer``\ 's schema
and make the tenant available at the request. If no tenant is found, a
404 error is raised. This also means you should have a tenant for your
main domain, typically using the ``public`` schema. For more information
please read the `setup`_ section.

What can this app do?
---------------------

As many tenants as you want
~~~~~~~~~~~~~~~~~~~~~~~~~~~

Each tenant has its data on a specific schema. Use a single project
instance to serve as many as you want.

Tenant-specific and shared apps
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

Tenant-specific apps do not share their data between tenants, but you
can also have shared apps where the information is always available and
shared between all.

Tenant View-Routing
~~~~~~~~~~~~~~~~~~~

You can have different views for ``http://customer.example.com/`` and
``http://example.com/``, even though Django only uses the string after
the host name to identify which view to serve.

Magic
~~~~~

Everyone loves magic! You'll be able to have all this barely having to
change your code!

Setup & Documentation
---------------------

**This is just a short setup guide**, it is **strongly** recommended
that you read the complete version at
`django-tenant-schemas.readthedocs.io`_.

Your ``DATABASE_ENGINE`` setting needs to be changed to

.. code-block:: python

    DATABASES = {
        'default': {
            'ENGINE': 'tenant_schemas.postgresql_backend',
            # ..
        }
    }

Add the middleware ``tenant_schemas.middleware.TenantMiddleware`` to the
top of ``MIDDLEWARE_CLASSES``, so that each request can be set to use
the correct schema.

.. code-block:: python

    MIDDLEWARE_CLASSES = (
        'tenant_schemas.middleware.TenantMiddleware',
        #...
    )

Add ``tenant_schemas.routers.TenantSyncRouter`` to your `DATABASE_ROUTERS`
setting, so that the correct apps can be synced, depending on what's
being synced (shared or tenant).

.. code-block:: python

    DATABASE_ROUTERS = (
        'tenant_schemas.routers.TenantSyncRouter',
    )

Add ``tenant_schemas`` to your ``INSTALLED_APPS``.

Create your tenant model
~~~~~~~~~~~~~~~~~~~~~~~~

.. code-block:: python

    from django.db import models
    from tenant_schemas.models import TenantMixin

    class Client(TenantMixin):
        name = models.CharField(max_length=100)
        paid_until =  models.DateField()
        on_trial = models.BooleanField()
        created_on = models.DateField(auto_now_add=True)

Define on ``settings.py`` which model is your tenant model. Assuming you
created ``Client`` inside an app named ``customers``, your
``TENANT_MODEL`` should look like this:

.. code-block:: python

    TENANT_MODEL = "customers.Client" # app.Model

Now run ``migrate_schemas`` to sync your apps to the ``public`` schema.

::

    python manage.py migrate_schemas --shared

Create your tenants just like a normal django model. Calling ``save``
will automatically create and sync/migrate the schema.

.. code-block:: python

    from customers.models import Client

    # create your public tenant
    tenant = Client(domain_url='tenant.my-domain.com',
                    schema_name='tenant1',
                    name='My First Tenant',
                    paid_until='2014-12-05',
                    on_trial=True)
    tenant.save()

Any request made to ``tenant.my-domain.com`` will now automatically set
your PostgreSQL's ``search_path`` to ``tenant1`` and ``public``, making
shared apps available too. This means that any call to the methods
``filter``, ``get``, ``save``, ``delete`` or any other function
involving a database connection will now be done at the tenant's schema,
so you shouldn't need to change anything at your views.

You're all set, but we have left key details outside of this short
tutorial, such as creating the public tenant and configuring shared and
tenant specific apps. Complete instructions can be found at
`django-tenant-schemas.readthedocs.io`_.



.. _django: https://www.djangoproject.com/
.. _PostgreSQL schemas: http://www.postgresql.org/docs/9.1/static/ddl-schemas.html
.. _PostgreSQL's official documentation on schemas: http://www.postgresql.org/docs/9.1/static/ddl-schemas.html
.. _Multi-Tenant Data Architecture: http://msdn.microsoft.com/en-us/library/aa479086.aspx

.. |PyPi version| image:: https://img.shields.io/pypi/v/django-tenant-schemas.svg
   :target: https://pypi.python.org/pypi/django-tenant-schemas
.. |PyPi downloads| image:: https://img.shields.io/pypi/dm/django-tenant-schemas.svg
   :target: https://pypi.python.org/pypi/django-tenant-schemas
.. |Python versions| image:: https://img.shields.io/pypi/pyversions/django-tenant-schemas.svg
.. |Travis CI| image:: https://travis-ci.org/bernardopires/django-tenant-schemas.svg?branch=master
   :target: https://travis-ci.org/bernardopires/django-tenant-schemas
.. |PostgreSQL| image:: https://img.shields.io/badge/PostgreSQL-9.2%2C%209.3%2C%209.4%2C%209.5%2C%209.6-blue.svg
.. _setup: https://django-tenant-schemas.readthedocs.io/en/latest/install.html
.. _django-tenant-schemas.readthedocs.io: https://django-tenant-schemas.readthedocs.io/en/latest/