2018-12-10 02:11:54 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
2019-01-15 02:23:30 +00:00
|
|
|
import os
|
2019-02-13 07:55:11 +00:00
|
|
|
import re
|
|
|
|
|
2023-01-31 03:00:21 +00:00
|
|
|
from celery.result import AsyncResult
|
2023-02-06 11:16:24 +00:00
|
|
|
from rest_framework import generics, viewsets, mixins, status
|
2022-10-25 12:02:23 +00:00
|
|
|
from django.shortcuts import get_object_or_404
|
2019-02-20 09:51:53 +00:00
|
|
|
from django.utils.translation import ugettext as _
|
2019-12-05 07:09:25 +00:00
|
|
|
from django_celery_beat.models import PeriodicTask
|
2023-02-06 11:16:24 +00:00
|
|
|
from rest_framework.response import Response
|
|
|
|
|
2023-02-16 07:52:21 +00:00
|
|
|
from common.exceptions import JMSException
|
2022-02-17 12:13:31 +00:00
|
|
|
from common.permissions import IsValidUser
|
2023-01-31 03:00:21 +00:00
|
|
|
from common.api import LogTailApi, CommonApiMixin
|
2023-02-06 11:16:24 +00:00
|
|
|
from ops.celery import app
|
2022-10-24 12:14:18 +00:00
|
|
|
from ..models import CeleryTaskExecution, CeleryTask
|
2019-01-15 02:23:30 +00:00
|
|
|
from ..celery.utils import get_celery_task_log_path
|
2020-12-29 13:05:18 +00:00
|
|
|
from ..ansible.utils import get_ansible_task_log_path
|
2023-01-31 03:00:21 +00:00
|
|
|
from ..serializers import CeleryResultSerializer, CeleryPeriodTaskSerializer
|
|
|
|
from ..serializers.celery import CeleryTaskSerializer, CeleryTaskExecutionSerializer
|
2018-12-10 02:11:54 +00:00
|
|
|
|
2020-12-29 13:05:18 +00:00
|
|
|
__all__ = [
|
2022-10-24 12:14:18 +00:00
|
|
|
'CeleryTaskExecutionLogApi', 'CeleryResultApi', 'CeleryPeriodTaskViewSet',
|
|
|
|
'AnsibleTaskLogApi', 'CeleryTaskViewSet', 'CeleryTaskExecutionViewSet'
|
2020-12-29 13:05:18 +00:00
|
|
|
]
|
2018-12-10 02:11:54 +00:00
|
|
|
|
|
|
|
|
2022-10-24 12:14:18 +00:00
|
|
|
class CeleryTaskExecutionLogApi(LogTailApi):
|
2018-12-10 02:11:54 +00:00
|
|
|
permission_classes = (IsValidUser,)
|
2019-01-15 02:23:30 +00:00
|
|
|
task = None
|
|
|
|
task_id = ''
|
2019-02-13 07:55:11 +00:00
|
|
|
pattern = re.compile(r'Task .* succeeded in \d+\.\d+s.*')
|
2018-12-10 02:11:54 +00:00
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
2019-01-15 02:23:30 +00:00
|
|
|
self.task_id = str(kwargs.get('pk'))
|
|
|
|
self.task = AsyncResult(self.task_id)
|
|
|
|
return super().get(request, *args, **kwargs)
|
|
|
|
|
2019-02-13 07:55:11 +00:00
|
|
|
def filter_line(self, line):
|
|
|
|
if self.pattern.match(line):
|
|
|
|
line = self.pattern.sub(line, '')
|
|
|
|
return line
|
|
|
|
|
2019-01-15 02:23:30 +00:00
|
|
|
def get_log_path(self):
|
|
|
|
new_path = get_celery_task_log_path(self.task_id)
|
|
|
|
if new_path and os.path.isfile(new_path):
|
|
|
|
return new_path
|
|
|
|
try:
|
2022-10-24 12:14:18 +00:00
|
|
|
task = CeleryTaskExecution.objects.get(id=self.task_id)
|
|
|
|
except CeleryTaskExecution.DoesNotExist:
|
2019-01-15 02:23:30 +00:00
|
|
|
return None
|
|
|
|
return task.full_log_path
|
|
|
|
|
|
|
|
def is_file_finish_write(self):
|
|
|
|
return self.task.ready()
|
2018-12-10 02:11:54 +00:00
|
|
|
|
2019-02-20 09:51:53 +00:00
|
|
|
def get_no_file_message(self, request):
|
|
|
|
if self.mark == 'undefined':
|
|
|
|
return '.'
|
|
|
|
else:
|
|
|
|
return _('Waiting task start')
|
|
|
|
|
2018-12-10 02:11:54 +00:00
|
|
|
|
2020-12-29 13:05:18 +00:00
|
|
|
class AnsibleTaskLogApi(LogTailApi):
|
|
|
|
permission_classes = (IsValidUser,)
|
|
|
|
|
|
|
|
def get_log_path(self):
|
|
|
|
new_path = get_ansible_task_log_path(self.kwargs.get('pk'))
|
|
|
|
if new_path and os.path.isfile(new_path):
|
|
|
|
return new_path
|
|
|
|
|
|
|
|
def get_no_file_message(self, request):
|
|
|
|
if self.mark == 'undefined':
|
|
|
|
return '.'
|
|
|
|
else:
|
|
|
|
return _('Waiting task start')
|
|
|
|
|
|
|
|
|
2018-12-10 02:11:54 +00:00
|
|
|
class CeleryResultApi(generics.RetrieveAPIView):
|
|
|
|
permission_classes = (IsValidUser,)
|
|
|
|
serializer_class = CeleryResultSerializer
|
|
|
|
|
|
|
|
def get_object(self):
|
|
|
|
pk = self.kwargs.get('pk')
|
|
|
|
return AsyncResult(pk)
|
|
|
|
|
2019-12-05 07:09:25 +00:00
|
|
|
|
|
|
|
class CeleryPeriodTaskViewSet(CommonApiMixin, viewsets.ModelViewSet):
|
|
|
|
queryset = PeriodicTask.objects.all()
|
|
|
|
serializer_class = CeleryPeriodTaskSerializer
|
|
|
|
http_method_names = ('get', 'head', 'options', 'patch')
|
|
|
|
|
|
|
|
def get_queryset(self):
|
|
|
|
queryset = super().get_queryset()
|
|
|
|
queryset = queryset.exclude(description='')
|
|
|
|
return queryset
|
2022-10-24 12:14:18 +00:00
|
|
|
|
|
|
|
|
2022-11-11 11:20:17 +00:00
|
|
|
class CelerySummaryAPIView(generics.RetrieveAPIView):
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2023-01-31 03:00:21 +00:00
|
|
|
class CeleryTaskViewSet(
|
|
|
|
CommonApiMixin, mixins.RetrieveModelMixin,
|
|
|
|
mixins.ListModelMixin, mixins.DestroyModelMixin,
|
|
|
|
viewsets.GenericViewSet
|
|
|
|
):
|
2023-02-09 03:49:16 +00:00
|
|
|
filterset_fields = ('id', 'name')
|
|
|
|
search_fields = filterset_fields
|
2022-10-24 12:14:18 +00:00
|
|
|
serializer_class = CeleryTaskSerializer
|
|
|
|
|
2022-11-15 08:29:40 +00:00
|
|
|
def get_queryset(self):
|
|
|
|
return CeleryTask.objects.exclude(name__startswith='celery')
|
|
|
|
|
2022-10-24 12:14:18 +00:00
|
|
|
|
2023-02-06 11:16:24 +00:00
|
|
|
class CeleryTaskExecutionViewSet(CommonApiMixin, viewsets.ModelViewSet):
|
2022-10-24 12:14:18 +00:00
|
|
|
serializer_class = CeleryTaskExecutionSerializer
|
2023-02-06 11:16:24 +00:00
|
|
|
http_method_names = ('get', 'post', 'head', 'options',)
|
2022-11-11 11:20:17 +00:00
|
|
|
queryset = CeleryTaskExecution.objects.all()
|
2023-02-09 09:44:35 +00:00
|
|
|
search_fields = ('name',)
|
2022-10-25 12:02:23 +00:00
|
|
|
|
|
|
|
def get_queryset(self):
|
2022-11-11 11:20:17 +00:00
|
|
|
task_id = self.request.query_params.get('task_id')
|
2022-10-25 12:02:23 +00:00
|
|
|
if task_id:
|
2022-11-11 11:20:17 +00:00
|
|
|
task = get_object_or_404(CeleryTask, id=task_id)
|
|
|
|
self.queryset = self.queryset.filter(name=task.name)
|
|
|
|
return self.queryset
|
2023-02-06 11:16:24 +00:00
|
|
|
|
|
|
|
def create(self, request, *args, **kwargs):
|
|
|
|
form_id = self.request.query_params.get('from', None)
|
|
|
|
if not form_id:
|
|
|
|
return Response(status=status.HTTP_400_BAD_REQUEST)
|
|
|
|
execution = get_object_or_404(CeleryTaskExecution, id=form_id)
|
|
|
|
task = app.tasks.get(execution.name, None)
|
|
|
|
if not task:
|
2023-02-16 07:52:21 +00:00
|
|
|
msg = _("Task {} not found").format(execution.name)
|
|
|
|
raise JMSException(code='task_not_found_error', detail=msg)
|
|
|
|
try:
|
|
|
|
t = task.delay(*execution.args, **execution.kwargs)
|
|
|
|
except TypeError:
|
|
|
|
msg = _("Task {} args or kwargs error").format(execution.name)
|
|
|
|
raise JMSException(code='task_args_error', detail=msg)
|
2023-02-06 11:16:24 +00:00
|
|
|
return Response(status=status.HTTP_201_CREATED, data={'task_id': t.id})
|