-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'task-9-get-tasks-api' into 'develop'
Added /api/cvat-tasks/ list endpoint See merge request epmc-mlcv/model_garden!16
- Loading branch information
Showing
14 changed files
with
508 additions
and
69 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -9,3 +9,5 @@ __pycache__/ | |
|
||
# OSX | ||
.DS_Store | ||
|
||
.coverage |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,9 +1,32 @@ | ||
from rest_framework import serializers | ||
|
||
|
||
class TaskSerializer(serializers.Serializer): | ||
class CvatTaskCreateSerializer(serializers.Serializer): | ||
task_name = serializers.CharField() | ||
dataset_id = serializers.CharField() | ||
assignee_id = serializers.IntegerField() | ||
files_in_task = serializers.IntegerField() | ||
count_of_tasks = serializers.IntegerField() | ||
|
||
|
||
class CvatTaskSerializer(serializers.Serializer): | ||
id = serializers.IntegerField() | ||
url = serializers.URLField() | ||
name = serializers.CharField() | ||
mode = serializers.CharField() | ||
size = serializers.IntegerField(required=False) | ||
owner = serializers.IntegerField(required=False) | ||
assignee = serializers.IntegerField(allow_null=True) | ||
created_date = serializers.DateTimeField(required=False) | ||
updated_date = serializers.DateTimeField(required=False) | ||
overlap = serializers.IntegerField(required=False) | ||
segment_size = serializers.IntegerField(required=False) | ||
z_order = serializers.BooleanField(required=False) | ||
status = serializers.CharField() | ||
labels = serializers.JSONField(required=False) | ||
segments = serializers.JSONField(required=False) | ||
image_quality = serializers.IntegerField(required=False) | ||
start_frame = serializers.IntegerField(required=False) | ||
stop_frame = serializers.IntegerField(required=False) | ||
frame_filter = serializers.CharField(required=False) | ||
project = serializers.IntegerField(required=False, allow_null=True) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,13 +1,13 @@ | ||
from .bucket import BucketViewSet | ||
from .cvat_tasks import CvatTaskViewSet | ||
from .cvat_user import CvatUserViewSet | ||
from .dataset import DatasetViewSet | ||
from .media_asset import MediaAssetViewSet | ||
from .task import TaskViewSet | ||
|
||
__all__ = ( | ||
"BucketViewSet", | ||
"CvatTaskViewSet", | ||
"CvatUserViewSet", | ||
"DatasetViewSet", | ||
"MediaAssetViewSet", | ||
"TaskViewSet", | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,138 @@ | ||
from typing import Iterable, Iterator | ||
|
||
from django import forms | ||
from rest_framework import status | ||
from rest_framework.request import Request | ||
from rest_framework.response import Response | ||
from rest_framework.viewsets import ViewSet | ||
from rest_framework.pagination import PageNumberPagination | ||
from rest_framework.exceptions import ValidationError | ||
|
||
from model_garden.serializers import ( | ||
CvatTaskSerializer, CvatTaskCreateSerializer | ||
) | ||
from model_garden.services.cvat import ( | ||
CvatService, CVATServiceException, ListRequest | ||
) | ||
|
||
|
||
class CvatTaskViewSet(ViewSet): | ||
serializer_class = CvatTaskCreateSerializer | ||
|
||
def create(self, request): | ||
cvat_service = CvatService() | ||
serializer = self.serializer_class(data=request.data) | ||
serializer.is_valid(raise_exception=True) | ||
data = serializer.data | ||
try: | ||
cvat_service.create_task( | ||
name=data['task_name'], | ||
assignee_id=data['assignee_id'], | ||
owner_id=cvat_service.get_root_user()['id'], | ||
) | ||
except CVATServiceException as e: | ||
return Response(data={'message': str(e)}, status=400) | ||
|
||
return Response() | ||
|
||
def list(self, request: Request): | ||
queryset = CvatTasksQuerySet(CvatService()) | ||
# iterating gets single value as a string instead of a list | ||
queryset.filter(**{k: v for k, v in request.query_params.items()}) | ||
queryset.order_by(request.query_params.get('ordering', 'id')) | ||
|
||
paginator = CvatTaskPagination() | ||
page = paginator.paginate_queryset(queryset, request) | ||
if not page: | ||
return Response( | ||
data={'message': 'Tasks was not found'}, | ||
status=status.HTTP_404_NOT_FOUND, | ||
) | ||
|
||
serializer = CvatTaskSerializer(page, many=True) | ||
return paginator.get_paginated_response(serializer.data) | ||
|
||
|
||
class CvatTaskPagination(PageNumberPagination): | ||
page_size = 100 | ||
page_size_query_param = 'page_size' | ||
max_page_size = 1000 | ||
|
||
|
||
class CvatTaskFilter(forms.Form): | ||
id = forms.IntegerField(required=False) | ||
project = forms.CharField(required=False, max_length=256) | ||
name = forms.CharField(required=False, max_length=256) | ||
mode = forms.ChoiceField( | ||
choices=( | ||
('annotation', 'annotation'), | ||
('interpolation', 'interpolation') | ||
), | ||
required=False | ||
) | ||
status = forms.ChoiceField( | ||
choices=( | ||
('annotation', 'annotation'), | ||
('validation', 'validation'), | ||
('completed', 'completed'), | ||
), | ||
required=False | ||
) | ||
assignee = forms.CharField(required=False, max_length=256) | ||
|
||
|
||
class CvatTasksQuerySet: | ||
ORDERING_FIELDS = ("id", "name", "status", "assignee",) | ||
|
||
def __init__(self, cvat_service: CvatService): | ||
self.service = cvat_service | ||
self.service_request = ListRequest() | ||
self._tasks = [] | ||
|
||
def filter(self, **kwargs) -> 'CvatTasksQuerySet': | ||
filter_form = CvatTaskFilter(kwargs) | ||
if not filter_form.is_valid(): | ||
raise ValidationError( | ||
filter_form.errors, code=status.HTTP_400_BAD_REQUEST, | ||
) | ||
|
||
self.service_request.filters.update( | ||
{k: v for k, v in filter_form.cleaned_data.items() if v} | ||
) | ||
return self | ||
|
||
def order_by(self, field_name: str) -> 'CvatTasksQuerySet': | ||
self.service_request.ordering = field_name | ||
return self | ||
|
||
def count(self) -> int: | ||
self.service_request.page = 1 | ||
self.service_request.page_size = 1 | ||
tasks = self.service.tasks(self.service_request) | ||
return tasks.count | ||
|
||
def __len__(self) -> int: | ||
return self.count() | ||
|
||
def __getitem__(self, key) -> Iterable[dict]: | ||
if not isinstance(key, (int, slice)): | ||
raise TypeError | ||
|
||
start = key.start if isinstance(key, slice) else key | ||
stop = key.stop if isinstance(key, slice) else key + 1 | ||
page_size = stop - start | ||
|
||
if page_size == 0: | ||
self._tasks = [] | ||
return self._tasks | ||
|
||
self.service_request.page = start // page_size + 1 | ||
self.service_request.page_size = page_size | ||
self._tasks = self.service.tasks(self.service_request).results | ||
return self._tasks | ||
|
||
def __iter__(self) -> Iterator[dict]: | ||
"""Django instatiate iterator only after slicing, | ||
so iterator doesn't have to make request | ||
""" | ||
return iter(self._tasks) |
This file was deleted.
Oops, something went wrong.
Oops, something went wrong.