Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Refactor webhooks #5916

Merged
merged 38 commits into from
Apr 7, 2023
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
38 commits
Select commit Hold shift + click to select a range
632bd92
use built-in django signals for sending webhooks
sizov-kirill Mar 23, 2023
af4c460
fix tests
sizov-kirill Mar 23, 2023
2bbcf8f
send webhooks after deleting project or organization
sizov-kirill Mar 23, 2023
b5fd22e
rest api tests: rebuild images
sizov-kirill Mar 23, 2023
d195da0
add migration file
sizov-kirill Mar 23, 2023
ea5da16
fix black warnings
sizov-kirill Mar 23, 2023
eb1a2c3
fix isort warnings
sizov-kirill Mar 23, 2023
1cf847a
fix isort warnings
sizov-kirill Mar 23, 2023
8d7c472
Merge branch 'develop' into sk/refactor-webhooks
sizov-kirill Mar 23, 2023
e5e23d3
fix migration
sizov-kirill Mar 23, 2023
7ee387f
update schema
sizov-kirill Mar 23, 2023
721a33a
update schema
sizov-kirill Mar 23, 2023
0956912
update assets
sizov-kirill Mar 24, 2023
eb5709b
fix deleting webhooks
sizov-kirill Mar 24, 2023
79fdd7e
use docker-compose v2
sizov-kirill Mar 24, 2023
787104b
update tests
sizov-kirill Mar 24, 2023
1ab6b7c
fix black warnings
sizov-kirill Mar 24, 2023
62110fb
revert changes for webhooks ping view
sizov-kirill Mar 24, 2023
f2b68a7
send information between pre and post signals
sizov-kirill Mar 24, 2023
31099c5
Merge branch 'develop' into sk/refactor-webhooks
sizov-kirill Mar 24, 2023
ad4c66e
extend logs for e2e tests
sizov-kirill Mar 25, 2023
69fc37f
Merge branch 'develop' into sk/refactor-webhooks
nmanovic Mar 27, 2023
10a1239
fix bug with project deleting
sizov-kirill Mar 27, 2023
2ac2386
Merge branch 'sk/refactor-webhooks' of https://github.com/opencv/cvat…
sizov-kirill Mar 27, 2023
d111e36
Merge branch 'develop' into sk/refactor-webhooks
yasakova-anastasia Mar 28, 2023
8032617
fix comments
sizov-kirill Apr 6, 2023
8d6c08f
fixes
sizov-kirill Apr 6, 2023
d4112d7
merge develop with resolving conflicts
sizov-kirill Apr 6, 2023
2f079fe
fix imports
sizov-kirill Apr 6, 2023
8e56962
Update tests/python/shared/fixtures/init.py
Apr 6, 2023
602d0b8
remove invitation update event
sizov-kirill Apr 6, 2023
64ca96a
apply review comments
sizov-kirill Apr 6, 2023
db2aaeb
fix test
sizov-kirill Apr 6, 2023
aa03260
Merge branch 'sk/refactor-webhooks' of https://github.com/opencv/cvat…
sizov-kirill Apr 6, 2023
69fed67
fix black warning
sizov-kirill Apr 6, 2023
757b3a1
delete label event, raise error if _deleted_object not accessible
sizov-kirill Apr 6, 2023
ba72abe
update schema
sizov-kirill Apr 6, 2023
d8a0d10
fix tests
sizov-kirill Apr 7, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 0 additions & 24 deletions cvat/apps/engine/mixins.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,6 @@
from cvat.apps.engine.models import Location
from cvat.apps.engine.location import StorageType, get_location_configuration
from cvat.apps.engine.serializers import DataSerializer, LabeledDataSerializer
from cvat.apps.webhooks.signals import signal_update, signal_create, signal_delete

class TusFile:
_tus_cache_timeout = 3600
Expand Down Expand Up @@ -336,11 +335,6 @@ def deserialize(self, request, import_func):
return self.upload_data(request)


class CreateModelMixin(mixins.CreateModelMixin):
def perform_create(self, serializer, **kwargs):
serializer.save(**kwargs)
signal_create.send(self, instance=serializer.instance)

class PartialUpdateModelMixin:
"""
Update fields of a model instance.
Expand All @@ -353,26 +347,8 @@ def _update(self, request, *args, **kwargs):
return mixins.UpdateModelMixin.update(self, request, *args, **kwargs)

def perform_update(self, serializer):
instance = serializer.instance
data = serializer.to_representation(instance)
old_values = {
attr: data[attr] if attr in data else getattr(instance, attr, None)
for attr in self.request.data.keys()
}

mixins.UpdateModelMixin.perform_update(self, serializer=serializer)

if getattr(serializer.instance, '_prefetched_objects_cache', None):
serializer.instance._prefetched_objects_cache = {}

signal_update.send(self, instance=serializer.instance, old_values=old_values)

def partial_update(self, request, *args, **kwargs):
with mock.patch.object(self, 'update', new=self._update, create=True):
return mixins.UpdateModelMixin.partial_update(self, request=request, *args, **kwargs)


class DestroyModelMixin(mixins.DestroyModelMixin):
def perform_destroy(self, instance):
signal_delete.send(self, instance=instance)
super().perform_destroy(instance)
7 changes: 7 additions & 0 deletions cvat/apps/engine/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -534,6 +534,13 @@ def create(cls, **kwargs):
except IntegrityError:
raise InvalidLabel("All label names must be unique")

def get_organization_id(self):
if self.project is not None:
return self.project.organization.id
if self.task is not None:
return self.task.organization.id
return None

class Meta:
default_permissions = ()
constraints = [
Expand Down
121 changes: 67 additions & 54 deletions cvat/apps/engine/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,77 +6,92 @@
import io
import os
import os.path as osp
import pytz
import traceback
from datetime import datetime
from distutils.util import strtobool
from tempfile import mkstemp

from django.db.models.query import Prefetch
from django.shortcuts import get_object_or_404
import django.db.models as dj_models
import django_rq
import pytz
from django.apps import apps
from django.conf import settings
from django.contrib.auth.models import User
from django.db import IntegrityError
from django.http import HttpResponse, HttpResponseNotFound, HttpResponseBadRequest
from django.db.models.query import Prefetch
from django.http import (HttpResponse, HttpResponseBadRequest,
HttpResponseNotFound)
from django.shortcuts import get_object_or_404
from django.utils import timezone
import django.db.models as dj_models

from drf_spectacular.types import OpenApiTypes
from drf_spectacular.utils import (
OpenApiParameter, OpenApiResponse, PolymorphicProxySerializer,
extend_schema_view, extend_schema
)
from django_sendfile import sendfile
from drf_spectacular.plumbing import build_array_type, build_basic_type

from drf_spectacular.types import OpenApiTypes
from drf_spectacular.utils import (OpenApiParameter, OpenApiResponse,
PolymorphicProxySerializer, extend_schema,
extend_schema_view)
from rest_framework import mixins, serializers, status, viewsets
from rest_framework.decorators import action
from rest_framework.exceptions import APIException, NotFound, ValidationError, PermissionDenied
from rest_framework.exceptions import (APIException, NotFound,
PermissionDenied, ValidationError)
from rest_framework.permissions import SAFE_METHODS
from rest_framework.response import Response
from django_sendfile import sendfile

import cvat.apps.dataset_manager as dm
import cvat.apps.dataset_manager.views # pylint: disable=unused-import
from cvat.apps.engine.cloud_provider import db_storage_to_storage_instance
from cvat.apps.dataset_manager.bindings import CvatImportError
from cvat.apps.dataset_manager.serializers import DatasetFormatsSerializer
from cvat.apps.engine import backup
from cvat.apps.engine.cache import MediaCache
from cvat.apps.engine.cloud_provider import db_storage_to_storage_instance
from cvat.apps.engine.frame_provider import FrameProvider
from cvat.apps.engine.location import StorageType, get_location_configuration
from cvat.apps.engine.media_extractors import get_mime
from cvat.apps.engine.models import (
Job, Label, Task, Project, Issue, Data,
Comment, StorageMethodChoice, StorageChoice,
CloudProviderChoice, Location
)
from cvat.apps.engine.mixins import (AnnotationMixin, PartialUpdateModelMixin,
SerializeMixin, UploadMixin)
from cvat.apps.engine.models import CloudProviderChoice
from cvat.apps.engine.models import CloudStorage as CloudStorageModel
from cvat.apps.engine.serializers import (
AboutSerializer, AnnotationFileSerializer, BasicUserSerializer,
DataMetaReadSerializer, DataMetaWriteSerializer, DataSerializer,
FileInfoSerializer, JobReadSerializer, JobWriteSerializer, LabelSerializer,
LabeledDataSerializer,
ProjectReadSerializer, ProjectWriteSerializer,
RqStatusSerializer, TaskReadSerializer, TaskWriteSerializer,
UserSerializer, PluginsSerializer, IssueReadSerializer,
IssueWriteSerializer, CommentReadSerializer, CommentWriteSerializer, CloudStorageWriteSerializer,
CloudStorageReadSerializer, DatasetFileSerializer,
ProjectFileSerializer, TaskFileSerializer)

from cvat.apps.engine.models import (Comment, Data, Issue, Job, Label,
Location, Project, StorageChoice,
StorageMethodChoice, Task)
from cvat.apps.engine.serializers import (AboutSerializer,
AnnotationFileSerializer,
BasicUserSerializer,
CloudStorageReadSerializer,
CloudStorageWriteSerializer,
CommentReadSerializer,
CommentWriteSerializer,
DataMetaReadSerializer,
DataMetaWriteSerializer,
DataSerializer,
DatasetFileSerializer,
FileInfoSerializer,
IssueReadSerializer,
IssueWriteSerializer,
JobReadSerializer,
JobWriteSerializer,
LabeledDataSerializer,
LabelSerializer, PluginsSerializer,
ProjectFileSerializer,
ProjectReadSerializer,
ProjectWriteSerializer,
RqStatusSerializer,
TaskFileSerializer,
TaskReadSerializer,
TaskWriteSerializer, UserSerializer)
from cvat.apps.engine.utils import (av_scan_paths, configure_dependent_job,
get_rq_job_meta, parse_exception_message,
process_failed_job)
from cvat.apps.events.handlers import handle_annotations_patch
from cvat.apps.iam.permissions import (CloudStoragePermission,
CommentPermission, IssuePermission,
JobPermission, LabelPermission,
ProjectPermission, TaskPermission,
UserPermission)
from utils.dataset_manifest import ImageManifestManager
from cvat.apps.engine.utils import (
av_scan_paths, process_failed_job, configure_dependent_job, parse_exception_message, get_rq_job_meta
)
from cvat.apps.engine import backup
from cvat.apps.engine.mixins import PartialUpdateModelMixin, UploadMixin, AnnotationMixin, SerializeMixin, DestroyModelMixin, CreateModelMixin
from cvat.apps.engine.location import get_location_configuration, StorageType

from . import models, task
from .log import slogger
from cvat.apps.iam.permissions import (CloudStoragePermission,
CommentPermission, IssuePermission, JobPermission, LabelPermission, ProjectPermission,
TaskPermission, UserPermission)
from cvat.apps.engine.cache import MediaCache
from cvat.apps.events.handlers import handle_annotations_patch


@extend_schema(tags=['server'])
class ServerViewSet(viewsets.ViewSet):
Expand Down Expand Up @@ -215,7 +230,7 @@ def plugins(request):
})
)
class ProjectViewSet(viewsets.GenericViewSet, mixins.ListModelMixin,
mixins.RetrieveModelMixin, CreateModelMixin, DestroyModelMixin,
mixins.RetrieveModelMixin, mixins.CreateModelMixin, mixins.DestroyModelMixin,
PartialUpdateModelMixin, UploadMixin, AnnotationMixin, SerializeMixin
):
queryset = models.Project.objects.select_related(
Expand Down Expand Up @@ -250,8 +265,7 @@ def get_queryset(self):
return queryset

def perform_create(self, serializer, **kwargs):
super().perform_create(
serializer,
serializer.save(
owner=self.request.user,
organization=self.request.iam_context['organization']
)
Expand Down Expand Up @@ -666,7 +680,7 @@ def __call__(self, request, start, stop, db_data):
})
)
class TaskViewSet(viewsets.GenericViewSet, mixins.ListModelMixin,
mixins.RetrieveModelMixin, CreateModelMixin, DestroyModelMixin,
mixins.RetrieveModelMixin, mixins.CreateModelMixin, mixins.DestroyModelMixin,
PartialUpdateModelMixin, UploadMixin, AnnotationMixin, SerializeMixin
):
queryset = Task.objects.select_related(
Expand Down Expand Up @@ -787,8 +801,7 @@ def perform_update(self, serializer):
updated_instance.project.save()

def perform_create(self, serializer, **kwargs):
super().perform_create(
serializer,
serializer.save(
owner=self.request.user,
organization=self.request.iam_context['organization']
)
Expand Down Expand Up @@ -945,7 +958,7 @@ def data(self, request, pk):
@extend_schema(methods=['HEAD'],
summary="Implements TUS file uploading protocol."
)
@action(detail=True, methods=['HEAD', 'PATCH'], url_path='data/'+UploadMixin.file_id_regex)
@action(detail=True, methods=['HEAD', 'PATCH'], url_path='data/' + UploadMixin.file_id_regex)
def append_data_chunk(self, request, pk, file_id):
self._object = self.get_object()
return self.append_tus_chunk(request, file_id)
Expand Down Expand Up @@ -1697,7 +1710,7 @@ def preview(self, request, pk):
})
)
class IssueViewSet(viewsets.GenericViewSet, mixins.ListModelMixin,
mixins.RetrieveModelMixin, CreateModelMixin, DestroyModelMixin,
mixins.RetrieveModelMixin, mixins.CreateModelMixin, mixins.DestroyModelMixin,
PartialUpdateModelMixin
):
queryset = Issue.objects.prefetch_related(
Expand Down Expand Up @@ -1734,7 +1747,7 @@ def get_serializer_class(self):
return IssueWriteSerializer

def perform_create(self, serializer, **kwargs):
super().perform_create(serializer, owner=self.request.user)
serializer.save(owner=self.request.user)

@extend_schema(tags=['comments'])
@extend_schema_view(
Expand Down Expand Up @@ -1767,7 +1780,7 @@ def perform_create(self, serializer, **kwargs):
})
)
class CommentViewSet(viewsets.GenericViewSet, mixins.ListModelMixin,
mixins.RetrieveModelMixin, CreateModelMixin, DestroyModelMixin,
mixins.RetrieveModelMixin, mixins.CreateModelMixin, mixins.DestroyModelMixin,
PartialUpdateModelMixin
):
queryset = Comment.objects.prefetch_related(
Expand Down Expand Up @@ -1803,7 +1816,7 @@ def get_serializer_class(self):
return CommentWriteSerializer

def perform_create(self, serializer, **kwargs):
super().perform_create(serializer, owner=self.request.user)
serializer.save(owner=self.request.user)


@extend_schema(tags=['labels'])
Expand Down
54 changes: 43 additions & 11 deletions cvat/apps/events/handlers.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,6 @@
from crum import get_current_user, get_current_request

from cvat.apps.engine.models import (
Organization,
Project,
Task,
Job,
Expand All @@ -34,12 +33,37 @@
LabelSerializer,
)
from cvat.apps.engine.models import ShapeType
from cvat.apps.organizations.serializers import OrganizationReadSerializer
from cvat.apps.webhooks.signals import project_id, organization_id
from cvat.apps.organizations.models import Membership, Organization, Invitation
from cvat.apps.organizations.serializers import OrganizationReadSerializer, MembershipReadSerializer, InvitationReadSerializer
from cvat.apps.engine.log import vlogger

from .event import event_scope, create_event

def project_id(instance):
if isinstance(instance, Project):
return instance.id

try:
pid = getattr(instance, "project_id", None)
if pid is None:
return instance.get_project_id()
return pid
except Exception:
return None


def organization_id(instance):
if isinstance(instance, Organization):
return instance.id

try:
oid = getattr(instance, "organization_id", None)
if oid is None:
return instance.get_organization_id()
return oid
except Exception:
return None


def task_id(instance):
if isinstance(instance, Task):
Expand Down Expand Up @@ -134,13 +158,13 @@ def organization_slug(instance):
except Exception:
return None

def _get_instance_diff(old_data, data):
ingone_related_fields = (
def get_instance_diff(old_data, data):
ignore_related_fields = (
"labels",
)
diff = {}
for prop, value in data.items():
if prop in ingone_related_fields:
if prop in ignore_related_fields:
continue
old_value = old_data.get(prop)
if old_value != value:
Expand Down Expand Up @@ -228,8 +252,16 @@ def _get_serializer(instance):
serializer = CommentReadSerializer(instance=instance, context=context)
if isinstance(instance, Label):
serializer = LabelSerializer(instance=instance, context=context)
if isinstance(instance, Membership):
serializer = MembershipReadSerializer(instance=instance, context=context)
if isinstance(instance, Invitation):
serializer = InvitationReadSerializer(instance=instance, context=context)

return serializer

if serializer :
def get_serializer(instance):
sizov-kirill marked this conversation as resolved.
Show resolved Hide resolved
serializer = _get_serializer(instance)
if serializer:
serializer.fields.pop("url", None)
return serializer

Expand All @@ -253,7 +285,7 @@ def handle_create(scope, instance, **kwargs):
uname = user_name(instance)
uemail = user_email(instance)

serializer = _get_serializer(instance=instance)
serializer = get_serializer(instance=instance)
try:
payload = serializer.data
except Exception:
Expand Down Expand Up @@ -289,9 +321,9 @@ def handle_update(scope, instance, old_instance, **kwargs):
uname = user_name(instance)
uemail = user_email(instance)

old_serializer = _get_serializer(instance=old_instance)
serializer = _get_serializer(instance=instance)
diff = _get_instance_diff(old_data=old_serializer.data, data=serializer.data)
old_serializer = get_serializer(instance=old_instance)
serializer = get_serializer(instance=instance)
diff = get_instance_diff(old_data=old_serializer.data, data=serializer.data)

timestamp = str(datetime.now(timezone.utc).timestamp())
for prop, change in diff.items():
Expand Down
Loading