Project

General

Profile

Download (6.82 KB) Statistics
| Branch: | Tag: | Revision:
ec0613c1 Benjamin Dauvergne
import os
import json
aae80ef0 Benjamin Dauvergne
import glob
ec0613c1 Benjamin Dauvergne
e62d56fc Benjamin Dauvergne
from django.conf import settings, UserSettingsHolder
aae80ef0 Benjamin Dauvergne
from django.db import connection
from django.http import Http404
from django.contrib.contenttypes.models import ContentType
from tenant_schemas.utils import get_tenant_model, remove_www_and_dev, get_public_schema_name
e62d56fc Benjamin Dauvergne
SENTINEL = object()

aae80ef0 Benjamin Dauvergne
class TenantNotFound(RuntimeError):
pass

class TenantMiddleware(object):
"""
This middleware should be placed at the very top of the middleware stack.
Selects the proper database schema using the request host. Can fail in
various ways which is better than corrupting or revealing data...
"""
@classmethod
def base(cls):
return settings.TENANT_BASE

@classmethod
def hostname2schema(cls, hostname):
'''Convert hostname to PostgreSQL schema name'''
if hostname in getattr(settings, 'TENANT_MAPPING', {}):
return settings.TENANT_MAPPING[hostname]
return hostname.replace('.', '_').replace('-', '_')

@classmethod
def get_tenant_by_hostname(cls, hostname):
'''Retrieve a tenant object for this hostname'''
bb762ce4 Thomas NOEL
if not os.path.exists(os.path.join(cls.base(), hostname)):
aae80ef0 Benjamin Dauvergne
raise TenantNotFound
bb762ce4 Thomas NOEL
schema = cls.hostname2schema(hostname)
aae80ef0 Benjamin Dauvergne
return get_tenant_model()(schema_name=schema, domain_url=hostname)

@classmethod
def get_tenants(cls):
self = cls()
for path in glob.glob(os.path.join(cls.base(), '*')):
hostname = os.path.basename(path)
yield get_tenant_model()(
schema_name=self.hostname2schema(hostname),
domain_url=hostname)
e62d56fc Benjamin Dauvergne
def process_request(self, request):
aae80ef0 Benjamin Dauvergne
# connection needs first to be at the public schema, as this is where the
# tenant informations are saved
connection.set_schema_to_public()
hostname_without_port = remove_www_and_dev(request.get_host().split(':')[0])

try:
request.tenant = self.get_tenant_by_hostname(hostname_without_port)
except TenantNotFound:
raise Http404
connection.set_tenant(request.tenant)

# content type can no longer be cached as public and tenant schemas have different
# models. if someone wants to change this, the cache needs to be separated between
# public and shared schemas. if this cache isn't cleared, this can cause permission
# problems. for example, on public, a particular model has id 14, but on the tenants
# it has the id 15. if 14 is cached instead of 15, the permissions for the wrong
# model will be fetched.
ContentType.objects.clear_cache()

# do we have a public-specific token?
if hasattr(settings, 'PUBLIC_SCHEMA_URLCONF') and request.tenant.schema_name == get_public_schema_name():
request.urlconf = settings.PUBLIC_SCHEMA_URLCONF



ec0613c1 Benjamin Dauvergne
class TenantSettingBaseMiddleware(object):
'''Base middleware classe for loading settings based on tenants

Child classes MUST override the load_tenant_settings() method.
'''
def __init__(self, *args, **kwargs):
self.tenants_settings = {}

def get_tenant_settings(self, wrapped, tenant):
'''Get last loaded settings for tenant, try to update it by loading
settings again is last loading time is less recent thant settings data
store. Compare with last modification time is done in the
load_tenant_settings() method.
'''
tenant_settings, last_time = self.tenants_settings.get(tenant.schema_name, (None,None))
if tenant_settings is None:
tenant_settings = UserSettingsHolder(wrapped)
tenant_settings, last_time = self.load_tenant_settings(wrapped, tenant, tenant_settings, last_time)
self.tenants_settings[tenant.schema_name] = tenant_settings, last_time
return tenant_settings

def load_tenant_settings(self, wrapped, tenant, tenant_settings, last_time):
'''Load tenant settings into tenant_settings object, eventually skip if
last_time is more recent than last update time for settings and return
the new value for tenant_settings and last_time'''
raise NotImplemented

def process_request(self, request):
if not hasattr(request, '_old_settings_wrapped'):
request._old_settings_wrapped = []
request._old_settings_wrapped.append(settings._wrapped)
settings._wrapped = self.get_tenant_settings(settings._wrapped, request.tenant)

def process_response(self, request, response):
if hasattr(request, '_old_settings_wrapped') and request._old_settings_wrapped:
settings._wrapped = request._old_settings_wrapped.pop()
return response


class FileBasedTenantSettingBaseMiddleware(TenantSettingBaseMiddleware):
FILENAME = None

def load_tenant_settings(self, wrapped, tenant, tenant_settings, last_time):
bb762ce4 Thomas NOEL
path = os.path.join(settings.TENANT_BASE, tenant.domain_url, self.FILENAME)
ec0613c1 Benjamin Dauvergne
try:
new_time = os.stat(path).st_mtime
except OSError:
# file was removed
if not last_time is None:
return UserSettingsHolder(wrapped), None
else:
if last_time is None or new_time >= last_time:
# file is new
tenant_settings = UserSettingsHolder(wrapped)
self.load_file(tenant_settings, path)
return tenant_settings, new_time
# nothing has changed
return tenant_settings, last_time


class JSONSettingsMiddleware(FileBasedTenantSettingBaseMiddleware):
'''Load settings from a JSON file whose path is given by:

os.path.join(settings.TENANT_BASE % schema_name, 'settings.json')

The JSON file must be a dictionnary whose key/value will override
current settings.
'''
FILENAME = 'settings.json'

def load_file(sef, tenant_settings, path):
with file(path) as f:
json_settings = json.load(f)
for key in json_settings:
setattr(tenant_settings, key, json_settings[key])


class DictAdapter(dict):
'''Give dict interface to plain objects'''
def __init__(self, wrapped):
self.wrapped = wrapped

def __setitem__(self, key, value):
setattr(self.wrapped, key, value)

def __getitem__(self, key):
try:
return getattr(self.wrapped, key)
except AttributeError:
raise KeyError


1b18c2d9 Jérôme Schneider
class PythonSettingsMiddleware(FileBasedTenantSettingBaseMiddleware):
ec0613c1 Benjamin Dauvergne
'''Load settings from a file whose path is given by:

os.path.join(settings.TENANT_BASE % schema_name, 'settings.py')

The file is executed in the same context as the classic settings file
using execfile.
'''
FILENAME = 'settings.py'

def load_file(self, tenant_settings, path):
execfile(path, DictAdapter(tenant_settings))