django-constance/constance/admin.py

190 lines
6.8 KiB
Python
Raw Normal View History

from datetime import datetime, date, time
from decimal import Decimal
import hashlib
2010-08-23 15:23:38 +00:00
from operator import itemgetter
2010-08-23 14:18:44 +00:00
2010-11-10 03:07:53 +00:00
from django import forms
from django.contrib import admin, messages
2010-11-10 03:07:53 +00:00
from django.contrib.admin import widgets
from django.contrib.admin.options import csrf_protect_m
from django.core.exceptions import PermissionDenied, ImproperlyConfigured
2010-08-23 14:18:44 +00:00
from django.forms import fields
2010-11-10 03:07:53 +00:00
from django.http import HttpResponseRedirect
from django.shortcuts import render_to_response
from django.template.context import RequestContext
from django.utils import six
2010-11-10 03:07:53 +00:00
from django.utils.formats import localize
from django.utils.functional import SimpleLazyObject
from django.utils.translation import ugettext_lazy as _
try:
from django.utils.encoding import smart_bytes
except ImportError:
from django.utils.encoding import smart_str as smart_bytes
2010-08-23 12:49:54 +00:00
2013-04-12 15:53:54 +00:00
try:
from django.conf.urls import patterns, url
except ImportError: # Django < 1.4
from django.conf.urls.defaults import patterns, url
from . import settings
from .base import Config as ConfigClass
2014-07-08 15:06:40 +00:00
config = SimpleLazyObject(ConfigClass)
2010-08-23 12:49:54 +00:00
2010-09-03 14:18:17 +00:00
NUMERIC_WIDGET = forms.TextInput(attrs={'size': 10})
2010-08-23 12:49:54 +00:00
INTEGER_LIKE = (fields.IntegerField, {'widget': NUMERIC_WIDGET})
2013-04-12 15:39:10 +00:00
STRING_LIKE = (fields.CharField, {
'widget': forms.Textarea(attrs={'rows': 3}),
'required': False,
})
2010-08-23 14:18:44 +00:00
FIELDS = {
bool: (fields.BooleanField, {'required': False}),
int: INTEGER_LIKE,
2010-09-03 14:18:17 +00:00
Decimal: (fields.DecimalField, {'widget': NUMERIC_WIDGET}),
str: STRING_LIKE,
datetime: (fields.DateTimeField, {'widget': widgets.AdminSplitDateTime}),
date: (fields.DateField, {'widget': widgets.AdminDateWidget}),
time: (fields.TimeField, {'widget': widgets.AdminTimeWidget}),
2010-09-03 14:18:17 +00:00
float: (fields.FloatField, {'widget': NUMERIC_WIDGET}),
2010-08-23 14:18:44 +00:00
}
2013-04-12 15:25:11 +00:00
if not six.PY3:
FIELDS.update({
long: INTEGER_LIKE,
unicode: STRING_LIKE,
})
2010-08-23 14:18:44 +00:00
class ConstanceForm(forms.Form):
version = forms.CharField(widget=forms.HiddenInput)
def __init__(self, initial, *args, **kwargs):
super(ConstanceForm, self).__init__(*args, initial=initial, **kwargs)
version_hash = hashlib.md5()
for name, (default, help_text) in settings.CONFIG.items():
config_type = type(default)
if config_type not in FIELDS:
raise ImproperlyConfigured(_("Constance doesn't support "
2014-11-27 18:05:42 +00:00
"config values of the type "
"%(config_type)s. Please fix "
"the value of '%(name)s'.")
% {'config_type': config_type,
'name': name})
field_class, kwargs = FIELDS[config_type]
self.fields[name] = field_class(label=name, **kwargs)
2010-08-23 14:18:44 +00:00
version_hash.update(smart_bytes(initial.get(name, '')))
self.initial['version'] = version_hash.hexdigest()
2010-08-23 14:40:01 +00:00
def save(self):
for name in settings.CONFIG:
2010-08-23 14:40:01 +00:00
setattr(config, name, self.cleaned_data[name])
def clean_version(self):
value = self.cleaned_data['version']
if value != self.initial['version']:
raise forms.ValidationError(_('The settings have been modified '
'by someone else. Please reload the '
'form and resubmit your changes.'))
return value
2010-08-23 14:18:44 +00:00
2010-08-23 12:49:54 +00:00
class ConstanceAdmin(admin.ModelAdmin):
2010-11-10 03:10:49 +00:00
def get_urls(self):
2010-08-23 12:49:54 +00:00
info = self.model._meta.app_label, self.model._meta.module_name
2010-11-10 03:10:49 +00:00
return patterns('',
2010-08-23 12:49:54 +00:00
url(r'^$',
2010-11-10 03:10:49 +00:00
self.admin_site.admin_view(self.changelist_view),
2013-04-12 15:25:11 +00:00
name='%s_%s_changelist' % info),
2011-09-21 14:24:15 +00:00
url(r'^$',
self.admin_site.admin_view(self.changelist_view),
2013-04-12 15:25:11 +00:00
name='%s_%s_add' % info),
2010-08-23 12:49:54 +00:00
)
2010-11-10 03:10:49 +00:00
@csrf_protect_m
def changelist_view(self, request, extra_context=None):
# First load a mapping between config name and default value
if not self.has_change_permission(request, None):
raise PermissionDenied
default_initial = ((name, default)
2013-04-12 15:25:11 +00:00
for name, (default, help_text) in settings.CONFIG.items())
# Then update the mapping with actually values from the backend
initial = dict(default_initial,
**dict(config._backend.mget(settings.CONFIG.keys())))
form = ConstanceForm(initial=initial)
2010-08-23 14:40:01 +00:00
if request.method == 'POST':
form = ConstanceForm(data=request.POST, initial=initial)
2010-08-23 14:40:01 +00:00
if form.is_valid():
form.save()
# In django 1.5 this can be replaced with self.message_user
messages.add_message(
request,
messages.SUCCESS,
_('Live settings updated successfully.'),
)
return HttpResponseRedirect('.')
2010-08-23 12:49:54 +00:00
context = {
'config': [],
'title': _('Constance config'),
2010-08-23 12:49:54 +00:00
'app_label': 'constance',
'opts': Config._meta,
2010-08-23 14:18:44 +00:00
'form': form,
2010-08-25 11:16:58 +00:00
'media': self.media + form.media,
2010-08-23 12:49:54 +00:00
}
2013-04-12 15:25:11 +00:00
for name, (default, help_text) in settings.CONFIG.items():
# First try to load the value from the actual backend
value = initial.get(name)
# Then if the returned value is None, get the default
if value is None:
value = getattr(config, name)
2010-08-23 12:49:54 +00:00
context['config'].append({
'name': name,
'default': localize(default),
'help_text': _(help_text),
'value': localize(value),
'modified': value != default,
2013-04-12 15:39:10 +00:00
'form_field': form[name],
2010-08-23 12:49:54 +00:00
})
2010-08-23 15:23:38 +00:00
context['config'].sort(key=itemgetter('name'))
2013-04-12 15:39:10 +00:00
context_instance = RequestContext(request,
current_app=self.admin_site.name)
2010-11-10 03:10:49 +00:00
return render_to_response('admin/constance/change_list.html',
context, context_instance=context_instance)
2010-08-23 12:49:54 +00:00
def has_add_permission(self, *args, **kwargs):
return False
def has_delete_permission(self, *args, **kwargs):
return False
def has_change_permission(self, request, obj=None):
if settings.SUPERUSER_ONLY:
return request.user.is_superuser
2013-04-12 15:39:10 +00:00
return super(ConstanceAdmin, self).has_change_permission(request, obj)
2010-08-23 12:49:54 +00:00
class Config(object):
class Meta(object):
app_label = 'constance'
object_name = 'Config'
model_name = module_name = 'config'
verbose_name_plural = _('config')
2010-11-10 03:12:39 +00:00
get_ordered_objects = lambda x: False
2011-02-23 15:39:58 +00:00
abstract = False
2013-01-19 16:27:35 +00:00
swapped = False
def get_change_permission(self):
return 'change_%s' % self.model_name
2010-08-23 12:49:54 +00:00
_meta = Meta()
admin.site.register([Config], ConstanceAdmin)