mirror of
https://github.com/jumpserver/jumpserver.git
synced 2025-04-28 19:34:53 +00:00
263 lines
8.6 KiB
Python
263 lines
8.6 KiB
Python
# -*- coding: utf-8 -*-
|
||
#
|
||
from collections import defaultdict
|
||
from contextlib import nullcontext
|
||
from itertools import chain
|
||
from typing import Callable
|
||
|
||
from django.db import models
|
||
from django.db.models.signals import m2m_changed
|
||
from rest_framework.request import Request
|
||
from rest_framework.response import Response
|
||
from rest_framework.settings import api_settings
|
||
|
||
from common.drf.filters import (
|
||
IDSpmFilterBackend, CustomFilterBackend, IDInFilterBackend,
|
||
IDNotFilterBackend, NotOrRelFilterBackend, LabelFilterBackend
|
||
)
|
||
from common.utils import get_logger, lazyproperty
|
||
from orgs.utils import tmp_to_org, tmp_to_root_org
|
||
from .action import RenderToJsonMixin
|
||
from .serializer import SerializerMixin
|
||
|
||
__all__ = [
|
||
'CommonApiMixin', 'PaginatedResponseMixin', 'RelationMixin',
|
||
'ExtraFilterFieldsMixin'
|
||
]
|
||
|
||
logger = get_logger(__name__)
|
||
|
||
|
||
class PaginatedResponseMixin:
|
||
|
||
def get_paginated_response_from_queryset(self, queryset):
|
||
page = self.paginate_queryset(queryset)
|
||
if page is not None:
|
||
serializer = self.get_serializer(page, many=True)
|
||
return self.get_paginated_response(serializer.data)
|
||
|
||
serializer = self.get_serializer(queryset, many=True)
|
||
return Response(serializer.data)
|
||
|
||
|
||
class RelationMixin:
|
||
m2m_field = None
|
||
from_field = None
|
||
to_field = None
|
||
to_model = None
|
||
|
||
def __init__(self, *args, **kwargs):
|
||
super().__init__(*args, **kwargs)
|
||
|
||
assert self.m2m_field is not None, '''
|
||
`m2m_field` should not be `None`
|
||
'''
|
||
|
||
self.from_field = self.m2m_field.m2m_field_name()
|
||
self.to_field = self.m2m_field.m2m_reverse_field_name()
|
||
self.to_model = self.m2m_field.related_model
|
||
self.through = getattr(self.m2m_field.model, self.m2m_field.attname).through
|
||
|
||
def get_queryset(self):
|
||
# 注意,此处拦截了 `get_queryset` 没有 `super`
|
||
queryset = self.through.objects.all()
|
||
return queryset
|
||
|
||
def send_m2m_changed_signal(self, instances, action):
|
||
if not isinstance(instances, list):
|
||
instances = [instances]
|
||
|
||
from_to_mapper = defaultdict(list)
|
||
|
||
for i in instances:
|
||
to_id = getattr(i, self.to_field).id
|
||
# TODO 优化,不应该每次都查询数据库
|
||
from_obj = getattr(i, self.from_field)
|
||
from_to_mapper[from_obj].append(to_id)
|
||
|
||
for from_obj, to_ids in from_to_mapper.items():
|
||
m2m_changed.send(
|
||
sender=self.through, instance=from_obj, action=action,
|
||
reverse=False, model=self.to_model, pk_set=to_ids
|
||
)
|
||
|
||
def perform_create(self, serializer):
|
||
instance = serializer.save()
|
||
self.send_m2m_changed_signal(instance, 'post_add')
|
||
|
||
def perform_destroy(self, instance):
|
||
instance.delete()
|
||
self.send_m2m_changed_signal(instance, 'post_remove')
|
||
|
||
|
||
class QuerySetMixin:
|
||
action: str
|
||
request: Request
|
||
get_serializer_class: Callable
|
||
get_queryset: Callable
|
||
|
||
def get_queryset(self):
|
||
return super().get_queryset()
|
||
|
||
def filter_queryset(self, queryset):
|
||
queryset = super().filter_queryset(queryset)
|
||
if not hasattr(self, 'action'):
|
||
return queryset
|
||
if self.action == 'metadata':
|
||
queryset = queryset.none()
|
||
queryset = self.setup_eager_loading(queryset)
|
||
return queryset
|
||
|
||
def setup_eager_loading(self, queryset, is_paginated=False):
|
||
is_export_request = self.request.query_params.get('format') in ['csv', 'xlsx']
|
||
no_request_page = self.request.query_params.get('limit') is None
|
||
# 不分页不走一般这个,是因为会消耗多余的 sql 查询, 不如分页的时候查询一次
|
||
if not is_export_request and not is_paginated and not no_request_page:
|
||
return queryset
|
||
|
||
serializer_class = self.get_serializer_class()
|
||
if not serializer_class:
|
||
return queryset
|
||
|
||
if hasattr(serializer_class, 'setup_eager_loading'):
|
||
queryset = serializer_class.setup_eager_loading(queryset)
|
||
|
||
if hasattr(serializer_class, 'setup_eager_labels'):
|
||
queryset = serializer_class.setup_eager_labels(queryset)
|
||
return queryset
|
||
|
||
def paginate_queryset(self, queryset):
|
||
page = super().paginate_queryset(queryset)
|
||
model = getattr(queryset, 'model', None)
|
||
if not model or hasattr(queryset, 'custom'):
|
||
return page
|
||
|
||
serializer_class = self.get_serializer_class()
|
||
if page and serializer_class:
|
||
# 必须要返回 ids,用于排序
|
||
queryset, ids = self._get_page_again(page, model)
|
||
page = self.setup_eager_loading(queryset, is_paginated=True)
|
||
page_mapper = {str(obj.id): obj for obj in page}
|
||
page = [page_mapper.get(_id) for _id in ids if _id in page_mapper]
|
||
return page
|
||
|
||
def _get_page_again(self, page, model):
|
||
"""
|
||
因为 setup_eager_loading 需要是 queryset 结构, 所以必须要重新构造
|
||
"""
|
||
id_org_mapper = {str(obj.id): getattr(obj, 'org_id', None) for obj in page}
|
||
ids = list(id_org_mapper.keys())
|
||
org_ids = list(set(id_org_mapper.values()) - {None})
|
||
|
||
if not org_ids:
|
||
context = nullcontext()
|
||
elif len(org_ids) == 1:
|
||
context = tmp_to_org(org_ids[0])
|
||
else:
|
||
context = tmp_to_root_org()
|
||
|
||
with context:
|
||
page = model.objects.filter(id__in=ids)
|
||
return page, ids
|
||
|
||
|
||
class ExtraFilterFieldsMixin:
|
||
"""
|
||
额外的 api filter
|
||
"""
|
||
default_added_filters = (
|
||
CustomFilterBackend, IDSpmFilterBackend, IDInFilterBackend,
|
||
IDNotFilterBackend, LabelFilterBackend
|
||
)
|
||
filter_backends = api_settings.DEFAULT_FILTER_BACKENDS
|
||
extra_filter_fields = []
|
||
extra_filter_backends = []
|
||
|
||
def set_compatible_fields(self):
|
||
"""
|
||
兼容老的 filter_fields
|
||
"""
|
||
if not hasattr(self, 'filter_fields') and hasattr(self, 'filterset_fields'):
|
||
self.filter_fields = self.filterset_fields
|
||
|
||
def get_filter_backends(self):
|
||
self.set_compatible_fields()
|
||
if self.filter_backends != self.__class__.filter_backends:
|
||
return self.filter_backends
|
||
backends = list(chain(
|
||
self.filter_backends,
|
||
self.default_added_filters,
|
||
self.extra_filter_backends,
|
||
))
|
||
# 这个要放在最后
|
||
backends.append(NotOrRelFilterBackend)
|
||
return backends
|
||
|
||
def filter_queryset(self, queryset):
|
||
for backend in self.get_filter_backends():
|
||
queryset = backend().filter_queryset(self.request, queryset, self)
|
||
return queryset
|
||
|
||
|
||
class OrderingFielderFieldsMixin:
|
||
"""
|
||
额外的 api ordering
|
||
"""
|
||
ordering_fields = None
|
||
extra_ordering_fields = []
|
||
|
||
@lazyproperty
|
||
def ordering_fields(self):
|
||
return self._get_ordering_fields()
|
||
|
||
def _get_ordering_fields(self):
|
||
if isinstance(self.__class__.ordering_fields, (list, tuple)):
|
||
return self.__class__.ordering_fields
|
||
|
||
try:
|
||
valid_fields = self.get_valid_ordering_fields()
|
||
except Exception as e:
|
||
logger.debug('get_valid_ordering_fields error: %s, pass' % e)
|
||
# 这里千万不要这么用,会让 logging 重复,至于为什么,我也不知道
|
||
# logging.debug('get_valid_ordering_fields error: %s' % e)
|
||
valid_fields = []
|
||
|
||
fields = list(chain(
|
||
valid_fields,
|
||
self.extra_ordering_fields
|
||
))
|
||
return fields
|
||
|
||
def get_valid_ordering_fields(self):
|
||
if getattr(self, 'model', None):
|
||
model = self.model
|
||
elif getattr(self, 'queryset', None):
|
||
model = self.queryset.model
|
||
else:
|
||
queryset = self.get_queryset()
|
||
model = None if isinstance(queryset, list) else queryset.model
|
||
|
||
if not model:
|
||
return []
|
||
|
||
excludes_fields = (
|
||
models.UUIDField, models.Model, models.ForeignKey,
|
||
models.FileField, models.JSONField, models.ManyToManyField,
|
||
models.DurationField,
|
||
)
|
||
valid_fields = []
|
||
for field in model._meta.fields:
|
||
if isinstance(field, excludes_fields):
|
||
continue
|
||
valid_fields.append(field.name)
|
||
return valid_fields
|
||
|
||
|
||
class CommonApiMixin(
|
||
SerializerMixin, QuerySetMixin, ExtraFilterFieldsMixin,
|
||
OrderingFielderFieldsMixin, RenderToJsonMixin, PaginatedResponseMixin
|
||
):
|
||
def is_swagger_request(self):
|
||
return getattr(self, 'swagger_fake_view', False) or \
|
||
getattr(self, 'raw_action', '') == 'metadata'
|