|
|
|
import json
|
|
|
|
|
|
|
|
from django.db import models
|
|
|
|
from django.db.utils import ProgrammingError, OperationalError
|
|
|
|
from django.utils.translation import ugettext_lazy as _
|
|
|
|
from django.conf import settings
|
|
|
|
|
|
|
|
from common.utils import signer, get_logger
|
|
|
|
|
|
|
|
logger = get_logger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class SettingQuerySet(models.QuerySet):
|
|
|
|
def __getattr__(self, item):
|
|
|
|
queryset = list(self)
|
|
|
|
instances = [i for i in queryset if i.name == item]
|
|
|
|
if len(instances) == 1:
|
|
|
|
return instances[0]
|
|
|
|
else:
|
|
|
|
return Setting()
|
|
|
|
|
|
|
|
|
|
|
|
class SettingManager(models.Manager):
|
|
|
|
def get_queryset(self):
|
|
|
|
return SettingQuerySet(self.model, using=self._db)
|
|
|
|
|
|
|
|
|
|
|
|
class Setting(models.Model):
|
|
|
|
name = models.CharField(max_length=128, unique=True, verbose_name=_("Name"))
|
|
|
|
value = models.TextField(verbose_name=_("Value"), null=True, blank=True)
|
|
|
|
category = models.CharField(max_length=128, default="default")
|
|
|
|
encrypted = models.BooleanField(default=False)
|
|
|
|
enabled = models.BooleanField(verbose_name=_("Enabled"), default=True)
|
|
|
|
comment = models.TextField(verbose_name=_("Comment"))
|
|
|
|
|
|
|
|
objects = SettingManager()
|
|
|
|
cache_key_prefix = '_SETTING_'
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def cleaned_value(self):
|
|
|
|
try:
|
|
|
|
value = self.value
|
|
|
|
if self.encrypted:
|
|
|
|
value = signer.unsign(value)
|
|
|
|
if not value:
|
|
|
|
return None
|
|
|
|
value = json.loads(value)
|
|
|
|
return value
|
|
|
|
except json.JSONDecodeError:
|
|
|
|
return None
|
|
|
|
|
|
|
|
@cleaned_value.setter
|
|
|
|
def cleaned_value(self, item):
|
|
|
|
try:
|
|
|
|
v = json.dumps(item)
|
|
|
|
if self.encrypted:
|
|
|
|
v = signer.sign(v)
|
|
|
|
self.value = v
|
|
|
|
except json.JSONDecodeError as e:
|
|
|
|
raise ValueError("Json dump error: {}".format(str(e)))
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def refresh_all_settings(cls):
|
|
|
|
try:
|
|
|
|
settings_list = cls.objects.all()
|
|
|
|
for setting in settings_list:
|
|
|
|
setting.refresh_setting()
|
|
|
|
except (ProgrammingError, OperationalError):
|
|
|
|
pass
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def refresh_item(cls, name):
|
|
|
|
item = cls.objects.filter(name=name).first()
|
|
|
|
if not item:
|
|
|
|
return
|
|
|
|
item.refresh_setting()
|
|
|
|
|
|
|
|
def refresh_setting(self):
|
|
|
|
logger.debug(f"Refresh setting: {self.name}")
|
|
|
|
if hasattr(self.__class__, f'refresh_{self.name}'):
|
|
|
|
getattr(self.__class__, f'refresh_{self.name}')()
|
|
|
|
else:
|
|
|
|
setattr(settings, self.name, self.cleaned_value)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def refresh_authentications(cls, name):
|
|
|
|
setting = cls.objects.filter(name=name).first()
|
|
|
|
if not setting:
|
|
|
|
return
|
|
|
|
|
|
|
|
backends_map = {
|
|
|
|
'AUTH_LDAP': [settings.AUTH_BACKEND_LDAP],
|
|
|
|
'AUTH_OPENID': [settings.AUTH_BACKEND_OIDC_CODE, settings.AUTH_BACKEND_OIDC_PASSWORD],
|
|
|
|
'AUTH_RADIUS': [settings.AUTH_BACKEND_RADIUS],
|
|
|
|
'AUTH_CAS': [settings.AUTH_BACKEND_CAS],
|
|
|
|
}
|
|
|
|
setting_backends = backends_map[name]
|
|
|
|
auth_backends = settings.AUTHENTICATION_BACKENDS
|
|
|
|
|
|
|
|
for backend in setting_backends:
|
|
|
|
has = backend in auth_backends
|
|
|
|
|
|
|
|
# 添加
|
|
|
|
if setting.cleaned_value and not has:
|
|
|
|
logger.debug('Add auth backend: {}'.format(name))
|
|
|
|
settings.AUTHENTICATION_BACKENDS.insert(0, backend)
|
|
|
|
|
|
|
|
# 去掉
|
|
|
|
if not setting.cleaned_value and has:
|
|
|
|
index = auth_backends.index(backend)
|
|
|
|
logger.debug('Pop auth backend: {}'.format(name))
|
|
|
|
auth_backends.pop(index)
|
|
|
|
|
|
|
|
# 设置内存值
|
|
|
|
setattr(settings, name, setting.cleaned_value)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def refresh_AUTH_CAS(cls):
|
|
|
|
cls.refresh_authentications('AUTH_CAS')
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def refresh_AUTH_LDAP(cls):
|
|
|
|
cls.refresh_authentications('AUTH_LDAP')
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def refresh_AUTH_OPENID(cls):
|
|
|
|
cls.refresh_authentications('AUTH_OPENID')
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def refresh_AUTH_RADIUS(cls):
|
|
|
|
cls.refresh_authentications('AUTH_RADIUS')
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def update_or_create(cls, name='', value='', encrypted=False, category=''):
|
|
|
|
"""
|
|
|
|
不能使用 Model 提供的,update_or_create 因为这里有 encrypted 和 cleaned_value
|
|
|
|
:return: (changed, instance)
|
|
|
|
"""
|
|
|
|
setting = cls.objects.filter(name=name).first()
|
|
|
|
changed = False
|
|
|
|
if not setting:
|
|
|
|
setting = Setting(name=name, encrypted=encrypted, category=category)
|
|
|
|
if setting.cleaned_value != value:
|
|
|
|
setting.encrypted = encrypted
|
|
|
|
setting.cleaned_value = value
|
|
|
|
setting.save()
|
|
|
|
changed = True
|
|
|
|
return changed, setting
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
db_table = "settings_setting"
|
|
|
|
verbose_name = _("Setting")
|