mirror of
https://github.com/inventree/InvenTree
synced 2024-08-30 18:33:04 +00:00
606 lines
18 KiB
Python
606 lines
18 KiB
Python
"""
|
|
JSON API for the Stock app
|
|
"""
|
|
|
|
from django_filters.rest_framework import FilterSet, DjangoFilterBackend
|
|
from django_filters import NumberFilter
|
|
|
|
from django.conf import settings
|
|
from django.conf.urls import url, include
|
|
from django.urls import reverse
|
|
|
|
from .models import StockLocation, StockItem
|
|
from .models import StockItemTracking
|
|
|
|
from part.models import Part, PartCategory
|
|
|
|
from .serializers import StockItemSerializer, StockQuantitySerializer
|
|
from .serializers import LocationSerializer
|
|
from .serializers import StockTrackingSerializer
|
|
|
|
from InvenTree.views import TreeSerializer
|
|
from InvenTree.helpers import str2bool, isNull
|
|
from InvenTree.status_codes import StockStatus
|
|
|
|
import os
|
|
|
|
from rest_framework.serializers import ValidationError
|
|
from rest_framework.views import APIView
|
|
from rest_framework.response import Response
|
|
from rest_framework import generics, response, filters, permissions
|
|
|
|
|
|
class StockCategoryTree(TreeSerializer):
|
|
title = 'Stock'
|
|
model = StockLocation
|
|
|
|
@property
|
|
def root_url(self):
|
|
return reverse('stock-index')
|
|
|
|
def get_items(self):
|
|
return StockLocation.objects.all().prefetch_related('stock_items', 'children')
|
|
|
|
|
|
class StockDetail(generics.RetrieveUpdateDestroyAPIView):
|
|
""" API detail endpoint for Stock object
|
|
|
|
get:
|
|
Return a single StockItem object
|
|
|
|
post:
|
|
Update a StockItem
|
|
|
|
delete:
|
|
Remove a StockItem
|
|
"""
|
|
|
|
queryset = StockItem.objects.all()
|
|
serializer_class = StockItemSerializer
|
|
permission_classes = (permissions.IsAuthenticated,)
|
|
|
|
def get_serializer(self, *args, **kwargs):
|
|
|
|
try:
|
|
kwargs['part_detail'] = str2bool(self.request.GET.get('part_detail', False))
|
|
except AttributeError:
|
|
pass
|
|
|
|
try:
|
|
kwargs['location_detail'] = str2bool(self.request.GET.get('location_detail', False))
|
|
except AttributeError:
|
|
pass
|
|
|
|
try:
|
|
kwargs['supplier_detail'] = str2bool(self.request.GET.get('supplier_detail', False))
|
|
except AttributeError:
|
|
pass
|
|
|
|
kwargs['context'] = self.get_serializer_context()
|
|
|
|
return self.serializer_class(*args, **kwargs)
|
|
|
|
|
|
class StockFilter(FilterSet):
|
|
""" FilterSet for advanced stock filtering.
|
|
|
|
Allows greater-than / less-than filtering for stock quantity
|
|
"""
|
|
|
|
min_stock = NumberFilter(name='quantity', lookup_expr='gte')
|
|
max_stock = NumberFilter(name='quantity', lookup_expr='lte')
|
|
|
|
class Meta:
|
|
model = StockItem
|
|
fields = ['quantity', 'part', 'location']
|
|
|
|
|
|
class StockStocktake(APIView):
|
|
""" Stocktake API endpoint provides stock update of multiple items simultaneously.
|
|
The 'action' field tells the type of stock action to perform:
|
|
- stocktake: Count the stock item(s)
|
|
- remove: Remove the quantity provided from stock
|
|
- add: Add the quantity provided from stock
|
|
"""
|
|
|
|
permission_classes = [
|
|
permissions.IsAuthenticated,
|
|
]
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
if 'action' not in request.data:
|
|
raise ValidationError({'action': 'Stocktake action must be provided'})
|
|
|
|
action = request.data['action']
|
|
|
|
ACTIONS = ['stocktake', 'remove', 'add']
|
|
|
|
if action not in ACTIONS:
|
|
raise ValidationError({'action': 'Action must be one of ' + ','.join(ACTIONS)})
|
|
|
|
elif 'items[]' not in request.data:
|
|
raise ValidationError({'items[]:' 'Request must contain list of items'})
|
|
|
|
items = []
|
|
|
|
# Ensure each entry is valid
|
|
for entry in request.data['items[]']:
|
|
if 'pk' not in entry:
|
|
raise ValidationError({'pk': 'Each entry must contain pk field'})
|
|
elif 'quantity' not in entry:
|
|
raise ValidationError({'quantity': 'Each entry must contain quantity field'})
|
|
|
|
item = {}
|
|
try:
|
|
item['item'] = StockItem.objects.get(pk=entry['pk'])
|
|
except StockItem.DoesNotExist:
|
|
raise ValidationError({'pk': 'No matching StockItem found for pk={pk}'.format(pk=entry['pk'])})
|
|
try:
|
|
item['quantity'] = int(entry['quantity'])
|
|
except ValueError:
|
|
raise ValidationError({'quantity': 'Quantity must be an integer'})
|
|
|
|
if item['quantity'] < 0:
|
|
raise ValidationError({'quantity': 'Quantity must be >= 0'})
|
|
|
|
items.append(item)
|
|
|
|
# Stocktake notes
|
|
notes = ''
|
|
|
|
if 'notes' in request.data:
|
|
notes = request.data['notes']
|
|
|
|
n = 0
|
|
|
|
for item in items:
|
|
quantity = int(item['quantity'])
|
|
|
|
if action == u'stocktake':
|
|
if item['item'].stocktake(quantity, request.user, notes=notes):
|
|
n += 1
|
|
elif action == u'remove':
|
|
if item['item'].take_stock(quantity, request.user, notes=notes):
|
|
n += 1
|
|
elif action == u'add':
|
|
if item['item'].add_stock(quantity, request.user, notes=notes):
|
|
n += 1
|
|
|
|
return Response({'success': 'Updated stock for {n} items'.format(n=n)})
|
|
|
|
|
|
class StockMove(APIView):
|
|
""" API endpoint for performing stock movements """
|
|
|
|
permission_classes = [
|
|
permissions.IsAuthenticated,
|
|
]
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
data = request.data
|
|
|
|
if 'location' not in data:
|
|
raise ValidationError({'location': 'Destination must be specified'})
|
|
|
|
try:
|
|
loc_id = int(data.get('location'))
|
|
except ValueError:
|
|
raise ValidationError({'location': 'Integer ID required'})
|
|
|
|
try:
|
|
location = StockLocation.objects.get(pk=loc_id)
|
|
except StockLocation.DoesNotExist:
|
|
raise ValidationError({'location': 'Location does not exist'})
|
|
|
|
if 'stock' not in data:
|
|
raise ValidationError({'stock': 'Stock list must be specified'})
|
|
|
|
stock_list = data.get('stock')
|
|
|
|
if type(stock_list) is not list:
|
|
raise ValidationError({'stock': 'Stock must be supplied as a list'})
|
|
|
|
if 'notes' not in data:
|
|
raise ValidationError({'notes': 'Notes field must be supplied'})
|
|
|
|
for item in stock_list:
|
|
try:
|
|
stock_id = int(item['pk'])
|
|
if 'quantity' in item:
|
|
quantity = int(item['quantity'])
|
|
else:
|
|
# If quantity not supplied, we'll move the entire stock
|
|
quantity = None
|
|
except ValueError:
|
|
# Ignore this one
|
|
continue
|
|
|
|
# Ignore a zero quantity movement
|
|
if quantity <= 0:
|
|
continue
|
|
|
|
try:
|
|
stock = StockItem.objects.get(pk=stock_id)
|
|
except StockItem.DoesNotExist:
|
|
continue
|
|
|
|
if quantity is None:
|
|
quantity = stock.quantity
|
|
|
|
stock.move(location, data.get('notes'), request.user, quantity=quantity)
|
|
|
|
return Response({'success': 'Moved parts to {loc}'.format(
|
|
loc=str(location)
|
|
)})
|
|
|
|
|
|
class StockLocationList(generics.ListCreateAPIView):
|
|
""" API endpoint for list view of StockLocation objects:
|
|
|
|
- GET: Return list of StockLocation objects
|
|
- POST: Create a new StockLocation
|
|
"""
|
|
|
|
queryset = StockLocation.objects.all()
|
|
serializer_class = LocationSerializer
|
|
|
|
def get_queryset(self):
|
|
"""
|
|
Custom filtering:
|
|
- Allow filtering by "null" parent to retrieve top-level stock locations
|
|
"""
|
|
|
|
queryset = super().get_queryset()
|
|
|
|
loc_id = self.request.query_params.get('parent', None)
|
|
|
|
if loc_id is not None:
|
|
|
|
# Look for top-level locations
|
|
if isNull(loc_id):
|
|
queryset = queryset.filter(parent=None)
|
|
|
|
else:
|
|
try:
|
|
loc_id = int(loc_id)
|
|
queryset = queryset.filter(parent=loc_id)
|
|
except ValueError:
|
|
pass
|
|
|
|
return queryset
|
|
|
|
permission_classes = [
|
|
permissions.IsAuthenticated,
|
|
]
|
|
|
|
filter_backends = [
|
|
DjangoFilterBackend,
|
|
filters.SearchFilter,
|
|
filters.OrderingFilter,
|
|
]
|
|
|
|
filter_fields = [
|
|
]
|
|
|
|
search_fields = [
|
|
'name',
|
|
'description',
|
|
]
|
|
|
|
|
|
class StockList(generics.ListCreateAPIView):
|
|
""" API endpoint for list view of Stock objects
|
|
|
|
- GET: Return a list of all StockItem objects (with optional query filters)
|
|
- POST: Create a new StockItem
|
|
|
|
Additional query parameters are available:
|
|
- aggregate: If 'true' then stock items are aggregated by Part and Location
|
|
- location: Filter stock by location
|
|
- category: Filter by parts belonging to a certain category
|
|
- supplier: Filter by supplier
|
|
- ancestor: Filter by an 'ancestor' StockItem
|
|
- status: Filter by the StockItem status
|
|
"""
|
|
|
|
queryset = StockItem.objects.all()
|
|
|
|
def get_serializer(self, *args, **kwargs):
|
|
|
|
try:
|
|
part_detail = str2bool(self.request.GET.get('part_detail', None))
|
|
location_detail = str2bool(self.request.GET.get('location_detail', None))
|
|
except AttributeError:
|
|
part_detail = None
|
|
location_detail = None
|
|
|
|
kwargs['part_detail'] = part_detail
|
|
kwargs['location_detail'] = location_detail
|
|
|
|
kwargs['context'] = self.get_serializer_context()
|
|
return self.serializer_class(*args, **kwargs)
|
|
|
|
def list(self, request, *args, **kwargs):
|
|
|
|
queryset = self.filter_queryset(self.get_queryset())
|
|
|
|
# Instead of using the DRF serializer to LIST,
|
|
# we will serialize the objects manually.
|
|
# This is significantly faster
|
|
|
|
data = queryset.values(
|
|
'pk',
|
|
'parent',
|
|
'quantity',
|
|
'serial',
|
|
'batch',
|
|
'status',
|
|
'notes',
|
|
'link',
|
|
'location',
|
|
'location__name',
|
|
'location__description',
|
|
'part',
|
|
'part__IPN',
|
|
'part__name',
|
|
'part__revision',
|
|
'part__description',
|
|
'part__image',
|
|
'part__category',
|
|
'part__category__name',
|
|
'part__category__description',
|
|
'supplier_part',
|
|
)
|
|
|
|
# Reduce the number of lookups we need to do for categories
|
|
# Cache location lookups for this query
|
|
locations = {}
|
|
|
|
for item in data:
|
|
|
|
img = item['part__image']
|
|
|
|
if img:
|
|
# Use the thumbnail image instead
|
|
fn, ext = os.path.splitext(img)
|
|
|
|
thumb = "{fn}.thumbnail{ext}".format(fn=fn, ext=ext)
|
|
|
|
thumb = os.path.join(settings.MEDIA_URL, thumb)
|
|
else:
|
|
thumb = ''
|
|
|
|
item['part__thumbnail'] = thumb
|
|
|
|
del item['part__image']
|
|
|
|
loc_id = item['location']
|
|
|
|
if loc_id:
|
|
if loc_id not in locations:
|
|
locations[loc_id] = StockLocation.objects.get(pk=loc_id).pathstring
|
|
|
|
item['location__path'] = locations[loc_id]
|
|
else:
|
|
item['location__path'] = None
|
|
|
|
item['status_text'] = StockStatus.label(item['status'])
|
|
|
|
return Response(data)
|
|
|
|
def get_queryset(self):
|
|
"""
|
|
If the query includes a particular location,
|
|
we may wish to also request stock items from all child locations.
|
|
"""
|
|
|
|
# Start with all objects
|
|
stock_list = super(StockList, self).get_queryset()
|
|
|
|
stock_list = stock_list.filter(customer=None, belongs_to=None)
|
|
|
|
# Does the client wish to filter by the Part ID?
|
|
part_id = self.request.query_params.get('part', None)
|
|
|
|
if part_id:
|
|
try:
|
|
part = Part.objects.get(pk=part_id)
|
|
|
|
# If the part is a Template part, select stock items for any "variant" parts under that template
|
|
if part.is_template:
|
|
stock_list = stock_list.filter(part__in=[part.id for part in Part.objects.filter(variant_of=part_id)])
|
|
else:
|
|
stock_list = stock_list.filter(part=part_id)
|
|
|
|
except (ValueError, Part.DoesNotExist):
|
|
pass
|
|
|
|
# Does the client wish to filter by the 'ancestor'?
|
|
anc_id = self.request.query_params.get('ancestor', None)
|
|
|
|
if anc_id:
|
|
try:
|
|
ancestor = StockItem.objects.get(pk=anc_id)
|
|
|
|
# Only allow items which are descendants of the specified StockItem
|
|
stock_list = stock_list.filter(id__in=[item.pk for item in ancestor.children.all()])
|
|
|
|
except (ValueError, Part.DoesNotExist):
|
|
pass
|
|
|
|
# Does the client wish to filter by stock location?
|
|
loc_id = self.request.query_params.get('location', None)
|
|
|
|
cascade = str2bool(self.request.query_params.get('cascade', False))
|
|
|
|
if loc_id is not None:
|
|
|
|
# Filter by 'null' location (i.e. top-level items)
|
|
if isNull(loc_id):
|
|
stock_list = stock_list.filter(location=None)
|
|
else:
|
|
try:
|
|
# If '?cascade=true' then include items which exist in sub-locations
|
|
if cascade:
|
|
location = StockLocation.objects.get(pk=loc_id)
|
|
stock_list = stock_list.filter(location__in=location.getUniqueChildren())
|
|
else:
|
|
stock_list = stock_list.filter(location=loc_id)
|
|
|
|
except (ValueError, StockLocation.DoesNotExist):
|
|
pass
|
|
|
|
# Does the client wish to filter by part category?
|
|
cat_id = self.request.query_params.get('category', None)
|
|
|
|
if cat_id:
|
|
try:
|
|
category = PartCategory.objects.get(pk=cat_id)
|
|
stock_list = stock_list.filter(part__category__in=category.getUniqueChildren())
|
|
|
|
except (ValueError, PartCategory.DoesNotExist):
|
|
pass
|
|
|
|
# Filter by StockItem status
|
|
status = self.request.query_params.get('status', None)
|
|
|
|
if status:
|
|
stock_list = stock_list.filter(status=status)
|
|
|
|
# Filter by supplier_part ID
|
|
supplier_part_id = self.request.query_params.get('supplier_part', None)
|
|
|
|
if supplier_part_id:
|
|
stock_list = stock_list.filter(supplier_part=supplier_part_id)
|
|
|
|
# Filter by supplier ID
|
|
supplier_id = self.request.query_params.get('supplier', None)
|
|
|
|
if supplier_id:
|
|
stock_list = stock_list.filter(supplier_part__supplier=supplier_id)
|
|
|
|
# Also ensure that we pre-fecth all the related items
|
|
stock_list = stock_list.prefetch_related(
|
|
'part',
|
|
'part__category',
|
|
'location'
|
|
)
|
|
|
|
stock_list = stock_list.order_by('part__name')
|
|
|
|
return stock_list
|
|
|
|
serializer_class = StockItemSerializer
|
|
|
|
permission_classes = [
|
|
permissions.IsAuthenticated,
|
|
]
|
|
|
|
filter_backends = [
|
|
DjangoFilterBackend,
|
|
filters.SearchFilter,
|
|
filters.OrderingFilter,
|
|
]
|
|
|
|
filter_fields = [
|
|
'supplier_part',
|
|
'customer',
|
|
'belongs_to',
|
|
'build'
|
|
]
|
|
|
|
|
|
class StockStocktakeEndpoint(generics.UpdateAPIView):
|
|
""" API endpoint for performing stocktake """
|
|
|
|
queryset = StockItem.objects.all()
|
|
serializer_class = StockQuantitySerializer
|
|
permission_classes = (permissions.IsAuthenticated,)
|
|
|
|
def update(self, request, *args, **kwargs):
|
|
object = self.get_object()
|
|
object.stocktake(request.data['quantity'], request.user)
|
|
|
|
serializer = self.get_serializer(object)
|
|
|
|
return response.Response(serializer.data)
|
|
|
|
|
|
class StockTrackingList(generics.ListCreateAPIView):
|
|
""" API endpoint for list view of StockItemTracking objects.
|
|
|
|
StockItemTracking objects are read-only
|
|
(they are created by internal model functionality)
|
|
|
|
- GET: Return list of StockItemTracking objects
|
|
"""
|
|
|
|
queryset = StockItemTracking.objects.all()
|
|
serializer_class = StockTrackingSerializer
|
|
permission_classes = [permissions.IsAuthenticated]
|
|
|
|
filter_backends = [
|
|
DjangoFilterBackend,
|
|
filters.SearchFilter,
|
|
filters.OrderingFilter,
|
|
]
|
|
|
|
filter_fields = [
|
|
'item',
|
|
'user',
|
|
]
|
|
|
|
ordering = '-date'
|
|
|
|
ordering_fields = [
|
|
'date',
|
|
]
|
|
|
|
search_fields = [
|
|
'title',
|
|
'notes',
|
|
]
|
|
|
|
|
|
class LocationDetail(generics.RetrieveUpdateDestroyAPIView):
|
|
""" API endpoint for detail view of StockLocation object
|
|
|
|
- GET: Return a single StockLocation object
|
|
- PATCH: Update a StockLocation object
|
|
- DELETE: Remove a StockLocation object
|
|
"""
|
|
|
|
queryset = StockLocation.objects.all()
|
|
serializer_class = LocationSerializer
|
|
permission_classes = (permissions.IsAuthenticated,)
|
|
|
|
|
|
stock_endpoints = [
|
|
url(r'^$', StockDetail.as_view(), name='api-stock-detail'),
|
|
]
|
|
|
|
location_endpoints = [
|
|
url(r'^(?P<pk>\d+)/', LocationDetail.as_view(), name='api-location-detail'),
|
|
|
|
url(r'^.*$', StockLocationList.as_view(), name='api-location-list'),
|
|
]
|
|
|
|
stock_api_urls = [
|
|
url(r'location/', include(location_endpoints)),
|
|
|
|
# These JSON endpoints have been replaced (for now) with server-side form rendering - 02/06/2019
|
|
# url(r'stocktake/?', StockStocktake.as_view(), name='api-stock-stocktake'),
|
|
# url(r'move/?', StockMove.as_view(), name='api-stock-move'),
|
|
|
|
url(r'track/?', StockTrackingList.as_view(), name='api-stock-track'),
|
|
|
|
url(r'^tree/?', StockCategoryTree.as_view(), name='api-stock-tree'),
|
|
|
|
# Detail for a single stock item
|
|
url(r'^(?P<pk>\d+)/', include(stock_endpoints)),
|
|
|
|
url(r'^.*$', StockList.as_view(), name='api-stock-list'),
|
|
]
|