mirror of
https://github.com/jumpserver/jumpserver.git
synced 2026-05-13 18:23:21 +00:00
* feat: 添加 RBAC 应用模块 * feat: 添加 RBAC Model、API * feat: 添加 RBAC Model、API 2 * feat: 添加 RBAC Model、API 3 * feat: 添加 RBAC Model、API 4 * feat: RBAC * feat: RBAC * feat: RBAC * feat: RBAC * feat: RBAC * feat: RBAC 整理权限位 * feat: RBAC 整理权限位2 * feat: RBAC 整理权限位2 * feat: RBAC 整理权限位 * feat: RBAC 添加默认角色 * feat: RBAC 添加迁移文件;迁移用户角色->用户角色绑定 * feat: RBAC 添加迁移文件;迁移用户角色->用户角色绑定 * feat: RBAC 修改用户模块API * feat: RBAC 添加组织模块迁移文件 & 修改组织模块API * feat: RBAC 添加组织模块迁移文件 & 修改组织模块API * feat: RBAC 修改用户角色属性的使用 * feat: RBAC No.1 * xxx * perf: 暂存 * perf: ... * perf(rbac): 添加 perms 到 profile serializer 中 * stash * perf: 使用init * perf: 修改migrations * perf: rbac * stash * stash * pref: 修改rbac * stash it * stash: 先去修复其他bug * perf: 修改 role 添加 users * pref: 修改 RBAC Model * feat: 添加权限的 tree api * stash: 暂存一下 * stash: 暂存一下 * perf: 修改 model verbose name * feat: 添加model各种 verbose name * perf: 生成 migrations * perf: 优化权限位 * perf: 添加迁移脚本 * feat: 添加组织角色迁移 * perf: 添加迁移脚本 * stash * perf: 添加migrateion * perf: 暂存一下 * perf: 修改rbac * perf: stash it * fix: 迁移冲突 * fix: 迁移冲突 * perf: 暂存一下 * perf: 修改 rbac 逻辑 * stash: 暂存一下 * perf: 修改内置角色 * perf: 解决 root 组织的问题 * perf: stash it * perf: 优化 rbac * perf: 优化 rolebinding 处理 * perf: 完成用户离开组织的问题 * perf: 暂存一下 * perf: 修改翻译 * perf: 去掉了 IsSuperUser * perf: IsAppUser 去掉完成 * perf: 修改 connection token 的权限 * perf: 去掉导入的问题 * perf: perms define 格式,修改 app 用户 的全新啊 * perf: 修改 permission * perf: 去掉一些 org admin * perf: 去掉部分 org admin * perf: 再去掉点 org admin role * perf: 再去掉部分 org admin * perf: user 角色搜索 * perf: 去掉很多 js * perf: 添加权限位 * perf: 修改权限 * perf: 去掉一个 todo * merge: with dev * fix: 修复冲突 Co-authored-by: Bai <bugatti_it@163.com> Co-authored-by: Michael Bai <baijiangjie@gmail.com> Co-authored-by: ibuler <ibuler@qq.com>
196 lines
6.3 KiB
Python
196 lines
6.3 KiB
Python
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):
|
||
if hasattr(self.__class__, f'refresh_{self.name}'):
|
||
getattr(self.__class__, f'refresh_{self.name}')()
|
||
else:
|
||
setattr(settings, self.name, self.cleaned_value)
|
||
self.refresh_keycloak_to_openid_if_need()
|
||
|
||
@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],
|
||
'AUTH_SAML2': [settings.AUTH_BACKEND_SAML2],
|
||
}
|
||
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_SAML2(cls):
|
||
cls.refresh_authentications('AUTH_SAML2')
|
||
|
||
def refresh_keycloak_to_openid_if_need(self):
|
||
watch_config_names = [
|
||
'AUTH_OPENID', 'AUTH_OPENID_REALM_NAME', 'AUTH_OPENID_SERVER_URL',
|
||
'AUTH_OPENID_PROVIDER_ENDPOINT', 'AUTH_OPENID_KEYCLOAK'
|
||
]
|
||
if self.name not in watch_config_names:
|
||
# 不在监听的配置中, 不需要刷新
|
||
return
|
||
auth_keycloak = self.__class__.objects.filter(name='AUTH_OPENID_KEYCLOAK').first()
|
||
if not auth_keycloak or not auth_keycloak.cleaned_value:
|
||
# 关闭 Keycloak 方式的配置, 不需要刷新
|
||
return
|
||
|
||
from jumpserver.conf import Config
|
||
config_names = [
|
||
'AUTH_OPENID', 'AUTH_OPENID_REALM_NAME',
|
||
'AUTH_OPENID_SERVER_URL', 'AUTH_OPENID_PROVIDER_ENDPOINT'
|
||
]
|
||
# 获取当前 keycloak 配置
|
||
keycloak_config = {}
|
||
for name in config_names:
|
||
setting = self.__class__.objects.filter(name=name).first()
|
||
if not setting:
|
||
continue
|
||
value = setting.cleaned_value
|
||
keycloak_config[name] = value
|
||
|
||
# 转化 keycloak 配置为 openid 配置
|
||
openid_config = Config.convert_keycloak_to_openid(keycloak_config)
|
||
if not openid_config:
|
||
return
|
||
# 刷新 settings
|
||
for key, value in openid_config.items():
|
||
setattr(settings, key, value)
|
||
|
||
@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 = _("System setting")
|