2016-12-27 19:37:50 +01:00
|
|
|
import json
|
2016-11-14 21:15:20 +01:00
|
|
|
import mimetypes
|
|
|
|
|
2017-05-04 12:28:17 +02:00
|
|
|
import hashlib
|
|
|
|
from collections import OrderedDict
|
2016-12-27 19:37:50 +01:00
|
|
|
from django.http import Http404, HttpResponse, HttpResponseNotModified
|
2017-05-05 16:21:48 +02:00
|
|
|
from django.shortcuts import get_object_or_404
|
2017-01-13 21:52:44 +01:00
|
|
|
from rest_framework.decorators import detail_route, list_route
|
2016-11-14 21:15:20 +01:00
|
|
|
from rest_framework.response import Response
|
|
|
|
from rest_framework.viewsets import ReadOnlyModelViewSet, ViewSet
|
|
|
|
|
2017-05-01 18:10:46 +02:00
|
|
|
from c3nav.access.apply import filter_arealocations_by_access, filter_queryset_by_access
|
2016-12-27 19:37:50 +01:00
|
|
|
from c3nav.mapdata.lastupdate import get_last_mapdata_update
|
2017-05-05 12:32:35 +02:00
|
|
|
from c3nav.mapdata.models import GEOMETRY_FEATURE_TYPES, AreaLocation, Level, LocationGroup, Source
|
2017-05-04 12:28:17 +02:00
|
|
|
from c3nav.mapdata.models.geometry import Stair
|
2016-12-21 20:56:28 +01:00
|
|
|
from c3nav.mapdata.search import get_location
|
2017-05-01 18:10:46 +02:00
|
|
|
from c3nav.mapdata.serializers.main import LevelSerializer, SourceSerializer
|
2017-01-13 21:52:44 +01:00
|
|
|
from c3nav.mapdata.utils.cache import (CachedReadOnlyViewSetMixin, cache_mapdata_api_response, get_bssid_areas_cached,
|
2017-05-01 18:10:46 +02:00
|
|
|
get_levels_cached)
|
2016-11-14 21:15:20 +01:00
|
|
|
|
|
|
|
|
2016-12-06 20:47:17 +01:00
|
|
|
class GeometryTypeViewSet(ViewSet):
|
|
|
|
"""
|
|
|
|
Lists all geometry types.
|
|
|
|
"""
|
2016-12-07 16:11:33 +01:00
|
|
|
@cache_mapdata_api_response()
|
2016-12-06 20:47:17 +01:00
|
|
|
def list(self, request):
|
|
|
|
return Response([
|
|
|
|
OrderedDict((
|
|
|
|
('name', name),
|
|
|
|
('title', str(mapitemtype._meta.verbose_name)),
|
|
|
|
('title_plural', str(mapitemtype._meta.verbose_name_plural)),
|
2017-05-05 12:32:35 +02:00
|
|
|
)) for name, mapitemtype in GEOMETRY_FEATURE_TYPES.items()
|
2016-12-06 20:47:17 +01:00
|
|
|
])
|
|
|
|
|
|
|
|
|
2016-11-27 14:03:39 +01:00
|
|
|
class GeometryViewSet(ViewSet):
|
2016-11-14 21:15:20 +01:00
|
|
|
"""
|
2016-11-27 14:03:39 +01:00
|
|
|
List all geometries.
|
2017-05-01 18:10:46 +02:00
|
|
|
You can filter by adding a level GET parameter.
|
2016-11-14 21:15:20 +01:00
|
|
|
"""
|
|
|
|
def list(self, request):
|
2016-12-07 16:11:33 +01:00
|
|
|
types = set(request.GET.getlist('type'))
|
2017-05-05 12:32:35 +02:00
|
|
|
valid_types = list(GEOMETRY_FEATURE_TYPES.keys())
|
2016-11-27 14:03:39 +01:00
|
|
|
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:
|
2017-05-05 16:21:48 +02:00
|
|
|
level = get_object_or_404(Level, id=request.GET['level'])
|
2016-12-07 16:11:33 +01:00
|
|
|
|
|
|
|
cache_key = '__'.join((
|
|
|
|
','.join([str(i) for i in types]),
|
|
|
|
str(level.id) if level is not None else '',
|
|
|
|
))
|
|
|
|
|
2017-05-01 18:10:46 +02:00
|
|
|
return self._list(request, types=types, level=level, add_cache_key=cache_key)
|
2016-12-07 16:11:33 +01:00
|
|
|
|
2016-12-19 16:54:11 +01:00
|
|
|
@staticmethod
|
|
|
|
def compare_by_location_type(x: AreaLocation, y: AreaLocation):
|
|
|
|
return AreaLocation.LOCATION_TYPES.index(x.location_type) - AreaLocation.LOCATION_TYPES.index(y.location_type)
|
|
|
|
|
2016-12-07 16:11:33 +01:00
|
|
|
@cache_mapdata_api_response()
|
2017-05-01 18:10:46 +02:00
|
|
|
def _list(self, request, types, level):
|
2016-11-27 14:03:39 +01:00
|
|
|
results = []
|
|
|
|
for t in types:
|
2017-05-05 12:32:35 +02:00
|
|
|
mapitemtype = GEOMETRY_FEATURE_TYPES[t]
|
2016-11-27 14:03:39 +01:00
|
|
|
queryset = mapitemtype.objects.all()
|
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-12-22 03:29:07 +01:00
|
|
|
queryset = filter_queryset_by_access(request, queryset)
|
2017-05-05 16:21:48 +02:00
|
|
|
queryset = queryset.order_by('id')
|
2016-12-06 23:43:57 +01:00
|
|
|
|
2017-05-01 18:10:46 +02:00
|
|
|
for field_name in ('level', 'crop_to_level', 'elevator'):
|
2016-12-06 23:43:57 +01:00
|
|
|
if hasattr(mapitemtype, field_name):
|
|
|
|
queryset = queryset.select_related(field_name)
|
|
|
|
|
|
|
|
for field_name in ('levels', ):
|
2016-12-06 22:55:15 +01:00
|
|
|
if hasattr(mapitemtype, field_name):
|
2016-12-06 23:43:57 +01:00
|
|
|
queryset.prefetch_related(field_name)
|
|
|
|
|
2016-12-19 16:54:11 +01:00
|
|
|
if issubclass(mapitemtype, AreaLocation):
|
|
|
|
queryset = sorted(queryset, key=AreaLocation.get_sort_key)
|
|
|
|
|
2017-05-04 12:28:17 +02:00
|
|
|
if issubclass(mapitemtype, Stair):
|
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-11-14 21:15:20 +01:00
|
|
|
|
|
|
|
|
2016-12-07 16:11:33 +01:00
|
|
|
class LevelViewSet(CachedReadOnlyViewSetMixin, ReadOnlyModelViewSet):
|
2016-11-14 21:15:20 +01:00
|
|
|
"""
|
|
|
|
List and retrieve levels.
|
|
|
|
"""
|
|
|
|
queryset = Level.objects.all()
|
|
|
|
serializer_class = LevelSerializer
|
2017-05-05 16:21:48 +02:00
|
|
|
lookup_field = 'id'
|
2016-11-14 21:15:20 +01:00
|
|
|
|
|
|
|
|
2016-12-07 16:11:33 +01:00
|
|
|
class SourceViewSet(CachedReadOnlyViewSetMixin, ReadOnlyModelViewSet):
|
2016-11-14 21:15:20 +01:00
|
|
|
"""
|
|
|
|
List and retrieve source images (to use as a drafts).
|
|
|
|
"""
|
|
|
|
queryset = Source.objects.all()
|
|
|
|
serializer_class = SourceSerializer
|
2017-05-05 16:21:48 +02:00
|
|
|
lookup_field = 'id'
|
2016-11-14 21:15:20 +01:00
|
|
|
|
|
|
|
def get_queryset(self):
|
2016-12-22 11:50:12 +01:00
|
|
|
return filter_queryset_by_access(self.request, super().get_queryset().all())
|
2016-11-14 21:15:20 +01:00
|
|
|
|
|
|
|
@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):
|
2016-11-14 21:15:20 +01:00
|
|
|
source = self.get_object()
|
|
|
|
response = HttpResponse(content_type=mimetypes.guess_type(source.name)[0])
|
2017-05-01 16:50:36 +02:00
|
|
|
response.write(source.image)
|
2016-11-14 21:15:20 +01:00
|
|
|
return response
|
2016-12-15 13:18:46 +01:00
|
|
|
|
|
|
|
|
2016-12-23 21:13:04 +01:00
|
|
|
class LocationViewSet(ViewSet):
|
2016-12-15 13:18:46 +01:00
|
|
|
"""
|
|
|
|
List and retrieve locations
|
|
|
|
"""
|
2016-12-24 03:08:12 +01:00
|
|
|
# We don't cache this, because it depends on access_list
|
2017-05-05 16:21:48 +02:00
|
|
|
lookup_field = 'location_id'
|
2016-12-15 13:18:46 +01:00
|
|
|
|
2016-12-24 21:51:30 +01:00
|
|
|
@staticmethod
|
|
|
|
def _filter(queryset):
|
2017-05-05 16:21:48 +02:00
|
|
|
return queryset.filter(can_search=True).order_by('id')
|
2016-12-24 21:51:30 +01:00
|
|
|
|
2016-12-15 13:18:46 +01:00
|
|
|
def list(self, request, **kwargs):
|
2016-12-27 19:37:50 +01:00
|
|
|
etag = hashlib.sha256(json.dumps({
|
|
|
|
'full_access': request.c3nav_full_access,
|
|
|
|
'access_list': request.c3nav_access_list,
|
|
|
|
'last_update': get_last_mapdata_update().isoformat()
|
2016-12-27 19:42:46 +01:00
|
|
|
}).encode()).hexdigest()
|
2016-12-27 19:37:50 +01:00
|
|
|
|
|
|
|
if_none_match = request.META.get('HTTP_IF_NONE_MATCH')
|
|
|
|
if if_none_match:
|
|
|
|
if if_none_match == etag:
|
|
|
|
return HttpResponseNotModified()
|
|
|
|
|
2016-12-15 13:18:46 +01:00
|
|
|
locations = []
|
2016-12-24 21:51:30 +01:00
|
|
|
locations += list(filter_queryset_by_access(request, self._filter(LocationGroup.objects.all())))
|
|
|
|
locations += sorted(filter_arealocations_by_access(request, self._filter(AreaLocation.objects.all())),
|
2016-12-19 18:13:14 +01:00
|
|
|
key=AreaLocation.get_sort_key, reverse=True)
|
2016-12-27 19:37:50 +01:00
|
|
|
|
|
|
|
response = Response([location.to_location_json() for location in locations])
|
|
|
|
response['ETag'] = etag
|
|
|
|
response['Cache-Control'] = 'no-cache'
|
2016-12-27 19:42:46 +01:00
|
|
|
return response
|
2016-12-15 13:18:46 +01:00
|
|
|
|
2017-05-05 16:21:48 +02:00
|
|
|
def retrieve(self, request, location_id=None, **kwargs):
|
|
|
|
location = get_location(request, location_id)
|
2016-12-15 13:18:46 +01:00
|
|
|
if location is None:
|
|
|
|
raise Http404
|
2016-12-28 01:23:52 +01:00
|
|
|
return Response(location.to_location_json())
|
2016-12-27 23:39:14 +01:00
|
|
|
|
|
|
|
@list_route(methods=['POST'])
|
|
|
|
def wifilocate(self, request):
|
|
|
|
stations = json.loads(request.POST['stations'])[:200]
|
|
|
|
if not stations:
|
2016-12-28 13:46:04 +01:00
|
|
|
return Response({'location': None})
|
2016-12-27 23:39:14 +01:00
|
|
|
|
|
|
|
bssids = get_bssid_areas_cached()
|
|
|
|
stations = sorted(stations, key=lambda l: l['level'])
|
|
|
|
for station in stations:
|
2016-12-28 11:28:21 +01:00
|
|
|
area_name = bssids.get(station['bssid'].lower())
|
2016-12-27 23:39:14 +01:00
|
|
|
if area_name is not None:
|
|
|
|
location = get_location(request, area_name)
|
|
|
|
if location is not None:
|
2017-01-13 21:52:44 +01:00
|
|
|
return Response({'location': location.to_location_json()})
|
2016-12-27 23:39:14 +01:00
|
|
|
|
|
|
|
return Response({'location': None})
|