team-3/src/c3nav/mapdata/api.py

152 lines
5.6 KiB
Python
Raw Normal View History

import mimetypes
import os
from collections import OrderedDict
from django.conf import settings
from django.core.files import File
2016-11-27 14:03:39 +01:00
from django.http import HttpResponse
from rest_framework.decorators import detail_route
from rest_framework.response import Response
from rest_framework.viewsets import ReadOnlyModelViewSet, ViewSet
2016-11-27 14:03:39 +01:00
from c3nav.mapdata.models import GEOMETRY_MAPITEM_TYPES, Level, Package, Source
from c3nav.mapdata.models.geometry import DirectedLineGeometryMapItemWithLevel
2016-12-07 16:11:33 +01:00
from c3nav.mapdata.permissions import filter_queryset_by_package_access, get_unlocked_packages_names
from c3nav.mapdata.serializers.main import LevelSerializer, PackageSerializer, SourceSerializer
2016-12-07 16:11:33 +01:00
from c3nav.mapdata.utils.cache import (CachedReadOnlyViewSetMixin, cache_mapdata_api_response, get_levels_cached,
get_packages_cached)
class GeometryTypeViewSet(ViewSet):
"""
Lists all geometry types.
"""
2016-12-07 16:11:33 +01:00
@cache_mapdata_api_response()
def list(self, request):
return Response([
OrderedDict((
('name', name),
('title', str(mapitemtype._meta.verbose_name)),
('title_plural', str(mapitemtype._meta.verbose_name_plural)),
)) for name, mapitemtype in GEOMETRY_MAPITEM_TYPES.items()
])
2016-11-27 14:03:39 +01:00
class GeometryViewSet(ViewSet):
"""
2016-11-27 14:03:39 +01:00
List all geometries.
2016-12-07 16:11:33 +01:00
You can filter by adding a level GET parameter or one or more package or type GET parameters.
"""
def list(self, request):
2016-12-07 16:11:33 +01:00
types = set(request.GET.getlist('type'))
2016-11-27 14:03:39 +01:00
valid_types = list(GEOMETRY_MAPITEM_TYPES.keys())
if not types:
types = valid_types
else:
2016-12-07 16:11:33 +01:00
types = [t for t in valid_types if t in types]
level = None
if 'level' in request.GET:
levels_cached = get_levels_cached()
level_name = request.GET['level']
if level_name in levels_cached:
level = levels_cached[level_name]
packages_cached = get_packages_cached()
package_names = set(request.GET.getlist('package')) & set(get_unlocked_packages_names(request))
packages = [packages_cached[name] for name in package_names if name in packages_cached]
if len(packages) == len(packages_cached):
packages = []
package_ids = sorted([package.id for package in packages])
cache_key = '__'.join((
','.join([str(i) for i in types]),
str(level.id) if level is not None else '',
','.join([str(i) for i in package_ids]),
))
return self._list(request, types=types, level=level, packages=packages, add_cache_key=cache_key)
@cache_mapdata_api_response()
def _list(self, request, types, level, packages):
2016-11-27 14:03:39 +01:00
results = []
for t in types:
mapitemtype = GEOMETRY_MAPITEM_TYPES[t]
queryset = mapitemtype.objects.all()
if packages:
2016-12-06 23:43:57 +01:00
queryset = queryset.filter(package__in=packages)
2016-12-07 16:11:33 +01:00
if level:
2016-12-01 12:25:02 +01:00
if hasattr(mapitemtype, 'level'):
2016-12-07 16:11:33 +01:00
queryset = queryset.filter(level=level)
2016-12-01 12:25:02 +01:00
elif hasattr(mapitemtype, 'levels'):
2016-12-07 16:11:33 +01:00
queryset = queryset.filter(levels=level)
2016-12-01 12:25:02 +01:00
else:
queryset = queryset.none()
2016-11-27 14:03:39 +01:00
queryset = filter_queryset_by_package_access(request, queryset)
queryset = queryset.order_by('name')
2016-12-06 23:43:57 +01:00
for field_name in ('package', 'level', 'crop_to_level', 'elevator'):
if hasattr(mapitemtype, field_name):
queryset = queryset.select_related(field_name)
for field_name in ('levels', ):
if hasattr(mapitemtype, field_name):
2016-12-06 23:43:57 +01:00
queryset.prefetch_related(field_name)
if issubclass(mapitemtype, DirectedLineGeometryMapItemWithLevel):
2016-12-08 12:36:09 +01:00
results.extend(obj.to_shadow_geojson() for obj in queryset)
results.extend(obj.to_geojson() for obj in queryset)
2016-12-06 23:43:57 +01:00
2016-11-27 14:03:39 +01:00
return Response(results)
2016-12-07 16:11:33 +01:00
class PackageViewSet(CachedReadOnlyViewSetMixin, ReadOnlyModelViewSet):
"""
Retrieve packages the map consists of.
"""
queryset = Package.objects.all()
serializer_class = PackageSerializer
lookup_field = 'name'
lookup_value_regex = '[^/]+'
ordering = ('name',)
2016-12-07 16:11:33 +01:00
class LevelViewSet(CachedReadOnlyViewSetMixin, ReadOnlyModelViewSet):
"""
List and retrieve levels.
"""
queryset = Level.objects.all()
serializer_class = LevelSerializer
lookup_field = 'name'
lookup_value_regex = '[^/]+'
ordering = ('altitude',)
2016-12-07 16:11:33 +01:00
class SourceViewSet(CachedReadOnlyViewSetMixin, ReadOnlyModelViewSet):
"""
List and retrieve source images (to use as a drafts).
"""
queryset = Source.objects.all()
serializer_class = SourceSerializer
lookup_field = 'name'
lookup_value_regex = '[^/]+'
ordering = ('name',)
2016-12-07 16:11:33 +01:00
include_package_access = True
def get_queryset(self):
2016-11-27 14:03:39 +01:00
return filter_queryset_by_package_access(self.request, super().get_queryset())
@detail_route(methods=['get'])
def image(self, request, name=None):
2016-12-07 16:11:33 +01:00
return self._image(request, name=name, add_cache_key=self._get_add_cache_key(request))
@cache_mapdata_api_response()
def _image(self, request, name=None):
source = self.get_object()
response = HttpResponse(content_type=mimetypes.guess_type(source.name)[0])
image_path = os.path.join(settings.MAP_ROOT, source.package.directory, 'sources', source.name)
for chunk in File(open(image_path, 'rb')).chunks():
response.write(chunk)
return response