API improvements for Stock app

This commit is contained in:
Oliver Walters 2017-04-14 12:54:34 +10:00
parent 2299cd0700
commit 0441eb4c38
3 changed files with 76 additions and 29 deletions

View File

@ -35,18 +35,10 @@ class LocationDetailSerializer(serializers.ModelSerializer):
""" Detailed information about a stock location """ Detailed information about a stock location
""" """
# List of all stock items in this location
items = StockItemSerializer(many=True, read_only=True)
# List of all child locations under this one
children = LocationBriefSerializer(many=True, read_only=True)
class Meta: class Meta:
model = StockLocation model = StockLocation
fields = ('pk', fields = ('pk',
'name', 'name',
'description', 'description',
'parent', 'parent',
'path', 'path')
'children',
'items')

View File

@ -1,15 +1,20 @@
from django.conf.urls import url from django.conf.urls import url, include
from . import views from . import views
urlpatterns = [ locpatterns = [
# List all stock quantities for a given part url(r'^(?P<pk>[0-9]+)/?$', views.LocationDetail.as_view()),
url(r'^part/(?P<part>[0-9]+)$', views.PartStockDetail.as_view()),
# List all stock items in a given location url(r'^\?*[^/]*/?$', views.LocationList.as_view())
url(r'^location/(?P<pk>[0-9]+)$', views.LocationDetail.as_view()), ]
# List all top-level locations urlpatterns = [
url(r'^location/$', views.LocationList.as_view()), # Stock location urls
url(r'^$', views.LocationList.as_view()) url(r'^location/?', include(locpatterns)),
# Detail for a single stock item
url(r'^(?P<pk>[0-9]+)$', views.StockDetail.as_view()),
# List all stock items, with optional filters
url(r'^\?*[^/]*/?$', views.StockList.as_view()),
] ]

View File

@ -1,22 +1,55 @@
from rest_framework import generics from rest_framework import generics, permissions
import django_filters
from .models import StockLocation, StockItem from .models import StockLocation, StockItem
from .serializers import StockItemSerializer, LocationDetailSerializer from .serializers import StockItemSerializer, LocationDetailSerializer
class PartStockDetail(generics.ListCreateAPIView): class StockDetail(generics.RetrieveUpdateDestroyAPIView):
""" Return a list of all stockitems for a given part
"""
queryset = StockItem.objects.all()
serializer_class = StockItemSerializer serializer_class = StockItemSerializer
class StockFilter(django_filters.rest_framework.FilterSet):
min_stock = django_filters.NumberFilter(name='quantity', lookup_expr='gte')
max_stock = django_filters.NumberFilter(name='quantity', lookup_expr='lte')
class Meta:
model = StockItem
fields = ['quantity']
class StockList(generics.ListCreateAPIView):
serializer_class = StockItemSerializer
permission_classes = (permissions.IsAuthenticatedOrReadOnly,)
filter_backends = (django_filters.rest_framework.DjangoFilterBackend,)
filter_class = StockFilter
def get_queryset(self): def get_queryset(self):
part_id = self.kwargs['part'] items = StockItem.objects.all()
return StockItem.objects.filter(part=part_id)
# Specify a particular part
part_id = self.request.query_params.get('part', None)
if part_id:
items = items.filter(part=part_id)
class LocationDetail(generics.RetrieveAPIView): # Specify a particular location
loc_id = self.request.query_params.get('location', None)
if loc_id:
items = items.filter(location=loc_id)
return items
def create(self, request, *args, **kwargs):
# If the PART parameter is passed in the URL, use that
part_id = self.request.query_params.get('part', None)
if part_id:
request.data['part'] = part_id
return super(StockList, self).create(request, *args, **kwargs)
class LocationDetail(generics.RetrieveUpdateDestroyAPIView):
""" Return information on a specific stock location """ Return information on a specific stock location
""" """
@ -24,10 +57,27 @@ class LocationDetail(generics.RetrieveAPIView):
serializer_class = LocationDetailSerializer serializer_class = LocationDetailSerializer
class LocationList(generics.ListAPIView): class LocationList(generics.ListCreateAPIView):
""" Return a list of top-level locations """ Return a list of top-level locations
Locations are considered "top-level" if they do not have a parent Locations are considered "top-level" if they do not have a parent
""" """
queryset = StockLocation.objects.filter(parent=None) def get_queryset(self):
params = self.request.query_params
locations = StockLocation.objects.all()
parent_id = params.get('parent', None)
if parent_id and parent_id.lower() in ['none', 'false', 'null', 'top']:
locations = locations.filter(parent=None)
else:
try:
parent_id_num = int(parent_id)
locations = locations.filter(parent=parent_id_num)
except:
pass
return locations
serializer_class = LocationDetailSerializer serializer_class = LocationDetailSerializer