jumpserver/apps/notifications/api/notifications.py

219 lines
7.8 KiB
Python

import os
from django.template.loader import render_to_string
from rest_framework import status
from rest_framework.decorators import action
from rest_framework.mixins import ListModelMixin, UpdateModelMixin, RetrieveModelMixin
from rest_framework.response import Response
from rest_framework.views import APIView
from common.api import JMSGenericViewSet
from common.permissions import OnlySuperUser, IsValidUser
from common.views.template import _get_data_template_path, _get_edit_template_path
from notifications.backends import BACKEND
from notifications.models import SystemMsgSubscription, UserMsgSubscription
from notifications.notifications import CustomMsgTemplateBase
from notifications.notifications import system_msgs
from notifications.serializers import (
SystemMsgSubscriptionSerializer, SystemMsgSubscriptionByCategorySerializer,
UserMsgSubscriptionSerializer,
)
from notifications.serializers import TemplateEditSerializer
__all__ = (
'BackendListView', 'SystemMsgSubscriptionViewSet',
'UserMsgSubscriptionViewSet', 'get_all_test_messages', 'TemplateViewSet',
)
class BackendListView(APIView):
permission_classes = [IsValidUser]
def get(self, request):
data = [
{
'name': backend,
'name_display': backend.label
}
for backend in BACKEND if backend.is_enable
]
return Response(data=data)
class SystemMsgSubscriptionViewSet(
ListModelMixin, UpdateModelMixin, JMSGenericViewSet
):
lookup_field = 'message_type'
queryset = SystemMsgSubscription.objects.all()
serializer_classes = {
'list': SystemMsgSubscriptionByCategorySerializer,
'update': SystemMsgSubscriptionSerializer,
'partial_update': SystemMsgSubscriptionSerializer
}
rbac_perms = {
'*': 'settings.change_systemmsgsubscription'
}
def list(self, request, *args, **kwargs):
data = []
category_children_mapper = {}
subscriptions = self.get_queryset()
msgtype_sub_mapper = {}
for sub in subscriptions:
msgtype_sub_mapper[sub.message_type] = sub
for msg in system_msgs:
message_type = msg['message_type']
message_type_label = msg['message_type_label']
category = msg['category']
category_label = msg['category_label']
if category not in category_children_mapper:
children = []
data.append({
'category': category,
'category_label': category_label,
'children': children
})
category_children_mapper[category] = children
sub = msgtype_sub_mapper[message_type]
sub.message_type_label = message_type_label
category_children_mapper[category].append(sub)
serializer = self.get_serializer(data, many=True)
return Response(data=serializer.data)
class UserMsgSubscriptionViewSet(ListModelMixin,
RetrieveModelMixin,
UpdateModelMixin,
JMSGenericViewSet):
lookup_field = 'user_id'
serializer_class = UserMsgSubscriptionSerializer
permission_classes = (IsValidUser,)
def get_queryset(self):
queryset = UserMsgSubscription.objects.filter(user=self.request.user)
return queryset
def get_all_test_messages(request):
import textwrap
from ..notifications import Message
from django.shortcuts import HttpResponse
if not request.user.is_superuser:
return HttpResponse('没有权限', status=401)
msgs_cls = Message.get_all_sub_messages()
html_data = '<h3>HTML 格式 </h3>'
text_data = '<h3>Text 格式</h3>'
for msg_cls in msgs_cls:
try:
msg = msg_cls.gen_test_msg()
if not msg:
continue
msg_html = msg.html_msg_with_sign['message']
msg_text = msg.text_msg_with_sign['message']
except NotImplementedError:
msg_html = msg_text = '没有实现方法'
except Exception as e:
msg_html = msg_text = 'Error: ' + str(e)
html_data += """
<h3>{}</h3>
{}
<hr />
""".format(msg_cls.__name__, msg_html)
text_data += textwrap.dedent("""
<h3>{}</h3>
<pre>
{}
</pre>
<br/>
<hr />
""").format(msg_cls.__name__, msg_text)
return HttpResponse(html_data + text_data)
class TemplateViewSet(JMSGenericViewSet):
permission_classes = [OnlySuperUser]
def list(self, request):
result = []
metas = [cls.as_dict() for cls in CustomMsgTemplateBase._registry]
for meta in metas:
item = {
'template_name': meta['template_name'],
'subject': meta.get('subject', ''),
'contexts': meta.get('contexts', []),
'content': None,
'content_error': None,
'source': None,
}
edit_path = _get_edit_template_path(meta['template_name'])
try:
if os.path.exists(edit_path):
with open(edit_path, 'r', encoding='utf-8') as f:
item['content'] = f.read()
item['source'] = 'data'
else:
ctx = {x.get('name'): x.get('default') for x in item['contexts']}
try:
rendered = render_to_string(meta['template_name'], ctx)
item['content'] = rendered
item['source'] = 'original'
except Exception as e:
item['content_error'] = str(e)
item['source'] = 'original_error'
except Exception as e:
item['content_error'] = str(e)
result.append(item)
return Response(result)
@action(detail=False, methods=['patch'], url_path='edit', name='edit')
def edit(self, request):
"""保存前端编辑的模板内容到 data/template/<template_name> 目录"""
serializer = TemplateEditSerializer(data=request.data)
serializer.is_valid(raise_exception=True)
template_name = serializer.validated_data['template_name']
content = serializer.validated_data['template_content']
render_html = serializer.validated_data['render_html']
data_path = _get_data_template_path(template_name)
edit_path = _get_edit_template_path(template_name)
data_dir = os.path.dirname(data_path)
try:
os.makedirs(data_dir, exist_ok=True)
with open(data_path, 'w', encoding='utf-8') as f:
f.write(render_html)
with open(edit_path, 'w', encoding='utf-8') as f:
f.write(content)
except Exception as e:
return Response({'ok': False, 'error': str(e)}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
return Response({'ok': True, 'path': data_path})
@action(detail=False, methods=['post'], url_path='reset', name='reset')
def reset(self, request):
template_name = request.data.get('template_name')
if not template_name:
return Response({'ok': False, 'error': 'template_name is required'}, status=status.HTTP_400_BAD_REQUEST)
data_path = _get_data_template_path(template_name)
edit_path = _get_edit_template_path(template_name)
try:
if os.path.exists(data_path) and os.path.isfile(data_path):
os.remove(data_path)
if os.path.exists(edit_path) and os.path.isfile(edit_path):
os.remove(edit_path)
except Exception as e:
return Response({'ok': False, 'error': str(e)}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
return Response({'ok': True, 'path': data_path})