2019-04-27 12:18:07 +00:00
|
|
|
"""
|
|
|
|
Provides a JSON API for the Part app
|
|
|
|
"""
|
|
|
|
|
2018-04-23 11:10:13 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
from __future__ import unicode_literals
|
2018-04-14 04:11:46 +00:00
|
|
|
|
2018-04-23 11:18:35 +00:00
|
|
|
from django_filters.rest_framework import DjangoFilterBackend
|
2020-05-02 04:03:17 +00:00
|
|
|
from django.http import JsonResponse
|
2020-09-22 16:24:09 +00:00
|
|
|
from django.db.models import Q, F, Count, Prefetch, Sum
|
2019-05-04 23:14:12 +00:00
|
|
|
|
2019-05-05 00:47:57 +00:00
|
|
|
from rest_framework import status
|
|
|
|
from rest_framework.response import Response
|
2019-09-05 09:29:51 +00:00
|
|
|
from rest_framework import filters, serializers
|
2019-04-13 23:25:46 +00:00
|
|
|
from rest_framework import generics, permissions
|
2018-04-14 04:11:46 +00:00
|
|
|
|
2018-05-04 13:54:57 +00:00
|
|
|
from django.conf.urls import url, include
|
2019-05-09 12:23:56 +00:00
|
|
|
from django.urls import reverse
|
2018-04-14 04:11:46 +00:00
|
|
|
|
2019-05-04 23:05:44 +00:00
|
|
|
from .models import Part, PartCategory, BomItem, PartStar
|
2019-09-07 09:44:10 +00:00
|
|
|
from .models import PartParameter, PartParameterTemplate
|
2020-05-17 03:56:49 +00:00
|
|
|
from .models import PartAttachment, PartTestTemplate
|
2020-09-18 12:11:51 +00:00
|
|
|
from .models import PartSellPriceBreak
|
2018-05-02 14:47:03 +00:00
|
|
|
|
2020-09-22 16:24:09 +00:00
|
|
|
from build.models import Build
|
|
|
|
|
2020-02-10 11:03:06 +00:00
|
|
|
from . import serializers as part_serializers
|
2018-04-14 04:11:46 +00:00
|
|
|
|
2018-04-28 13:22:12 +00:00
|
|
|
from InvenTree.views import TreeSerializer
|
2020-04-02 22:31:26 +00:00
|
|
|
from InvenTree.helpers import str2bool, isNull
|
2020-05-12 11:40:42 +00:00
|
|
|
from InvenTree.api import AttachmentMixin
|
2020-09-22 16:24:09 +00:00
|
|
|
from InvenTree.status_codes import BuildStatus
|
2018-04-28 13:22:12 +00:00
|
|
|
|
2019-04-13 23:25:46 +00:00
|
|
|
|
2018-04-28 13:22:12 +00:00
|
|
|
class PartCategoryTree(TreeSerializer):
|
|
|
|
|
|
|
|
title = "Parts"
|
|
|
|
model = PartCategory
|
2019-05-09 12:23:56 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def root_url(self):
|
|
|
|
return reverse('part-index')
|
2018-04-28 13:22:12 +00:00
|
|
|
|
2019-06-17 13:52:49 +00:00
|
|
|
def get_items(self):
|
|
|
|
return PartCategory.objects.all().prefetch_related('parts', 'children')
|
|
|
|
|
2020-10-06 00:29:38 +00:00
|
|
|
permission_classes = [
|
|
|
|
permissions.IsAuthenticated,
|
|
|
|
]
|
|
|
|
|
2018-04-14 04:11:46 +00:00
|
|
|
|
2018-05-04 13:54:57 +00:00
|
|
|
class CategoryList(generics.ListCreateAPIView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" API endpoint for accessing a list of PartCategory objects.
|
|
|
|
|
|
|
|
- GET: Return a list of PartCategory objects
|
|
|
|
- POST: Create a new PartCategory object
|
|
|
|
"""
|
|
|
|
|
2018-05-04 13:54:57 +00:00
|
|
|
queryset = PartCategory.objects.all()
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.CategorySerializer
|
2018-05-04 13:54:57 +00:00
|
|
|
|
2020-04-02 22:25:58 +00:00
|
|
|
def get_queryset(self):
|
|
|
|
"""
|
|
|
|
Custom filtering:
|
|
|
|
- Allow filtering by "null" parent to retrieve top-level part categories
|
|
|
|
"""
|
|
|
|
|
|
|
|
cat_id = self.request.query_params.get('parent', None)
|
|
|
|
|
|
|
|
queryset = super().get_queryset()
|
|
|
|
|
|
|
|
if cat_id is not None:
|
|
|
|
|
2020-04-02 22:31:26 +00:00
|
|
|
# Look for top-level categories
|
|
|
|
if isNull(cat_id):
|
|
|
|
queryset = queryset.filter(parent=None)
|
|
|
|
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
cat_id = int(cat_id)
|
|
|
|
queryset = queryset.filter(parent=cat_id)
|
|
|
|
except ValueError:
|
|
|
|
pass
|
2020-04-02 22:25:58 +00:00
|
|
|
|
|
|
|
return queryset
|
|
|
|
|
2018-05-04 13:54:57 +00:00
|
|
|
filter_backends = [
|
|
|
|
DjangoFilterBackend,
|
2019-06-02 09:15:05 +00:00
|
|
|
filters.SearchFilter,
|
2018-05-04 13:54:57 +00:00
|
|
|
filters.OrderingFilter,
|
|
|
|
]
|
|
|
|
|
|
|
|
filter_fields = [
|
|
|
|
]
|
|
|
|
|
|
|
|
ordering_fields = [
|
|
|
|
'name',
|
|
|
|
]
|
|
|
|
|
|
|
|
ordering = 'name'
|
|
|
|
|
|
|
|
search_fields = [
|
|
|
|
'name',
|
|
|
|
'description',
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2019-04-26 15:03:11 +00:00
|
|
|
class CategoryDetail(generics.RetrieveUpdateDestroyAPIView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" API endpoint for detail view of a single PartCategory object """
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.CategorySerializer
|
2019-04-26 15:03:11 +00:00
|
|
|
queryset = PartCategory.objects.all()
|
|
|
|
|
|
|
|
|
2020-09-18 12:11:51 +00:00
|
|
|
class PartSalePriceList(generics.ListCreateAPIView):
|
|
|
|
"""
|
|
|
|
API endpoint for list view of PartSalePriceBreak model
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = PartSellPriceBreak.objects.all()
|
|
|
|
serializer_class = part_serializers.PartSalePriceSerializer
|
|
|
|
|
|
|
|
filter_backends = [
|
|
|
|
DjangoFilterBackend
|
|
|
|
]
|
|
|
|
|
|
|
|
filter_fields = [
|
|
|
|
'part',
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2020-05-12 11:40:42 +00:00
|
|
|
class PartAttachmentList(generics.ListCreateAPIView, AttachmentMixin):
|
2020-05-11 13:25:55 +00:00
|
|
|
"""
|
|
|
|
API endpoint for listing (and creating) a PartAttachment (file upload).
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = PartAttachment.objects.all()
|
|
|
|
serializer_class = part_serializers.PartAttachmentSerializer
|
|
|
|
|
|
|
|
filter_fields = [
|
|
|
|
'part',
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2020-05-17 03:56:49 +00:00
|
|
|
class PartTestTemplateList(generics.ListCreateAPIView):
|
|
|
|
"""
|
|
|
|
API endpoint for listing (and creating) a PartTestTemplate.
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = PartTestTemplate.objects.all()
|
|
|
|
serializer_class = part_serializers.PartTestTemplateSerializer
|
|
|
|
|
|
|
|
def filter_queryset(self, queryset):
|
|
|
|
"""
|
|
|
|
Filter the test list queryset.
|
|
|
|
|
|
|
|
If filtering by 'part', we include results for any parts "above" the specified part.
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = super().filter_queryset(queryset)
|
|
|
|
|
|
|
|
params = self.request.query_params
|
|
|
|
|
|
|
|
part = params.get('part', None)
|
|
|
|
|
|
|
|
# Filter by part
|
|
|
|
if part:
|
|
|
|
try:
|
|
|
|
part = Part.objects.get(pk=part)
|
|
|
|
queryset = queryset.filter(part__in=part.get_ancestors(include_self=True))
|
|
|
|
except (ValueError, Part.DoesNotExist):
|
|
|
|
pass
|
|
|
|
|
2020-05-17 06:07:55 +00:00
|
|
|
# Filter by 'required' status
|
|
|
|
required = params.get('required', None)
|
|
|
|
|
|
|
|
if required is not None:
|
|
|
|
queryset = queryset.filter(required=required)
|
|
|
|
|
2020-05-17 03:56:49 +00:00
|
|
|
return queryset
|
|
|
|
|
|
|
|
filter_backends = [
|
|
|
|
DjangoFilterBackend,
|
|
|
|
filters.OrderingFilter,
|
|
|
|
filters.SearchFilter,
|
|
|
|
]
|
|
|
|
|
2020-05-17 04:15:13 +00:00
|
|
|
|
2020-02-10 11:03:06 +00:00
|
|
|
class PartThumbs(generics.ListAPIView):
|
|
|
|
""" API endpoint for retrieving information on available Part thumbnails """
|
|
|
|
|
|
|
|
serializer_class = part_serializers.PartThumbSerializer
|
|
|
|
|
2020-05-01 23:49:05 +00:00
|
|
|
def list(self, request, *args, **kwargs):
|
2020-02-10 11:03:06 +00:00
|
|
|
"""
|
|
|
|
Serialize the available Part images.
|
|
|
|
- Images may be used for multiple parts!
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Get all Parts which have an associated image
|
|
|
|
queryset = Part.objects.all().exclude(image='')
|
|
|
|
|
2020-04-19 13:56:16 +00:00
|
|
|
# TODO - We should return the thumbnails here, not the full image!
|
|
|
|
|
2020-02-10 12:48:45 +00:00
|
|
|
# Return the most popular parts first
|
2020-02-10 11:03:06 +00:00
|
|
|
data = queryset.values(
|
|
|
|
'image',
|
|
|
|
).annotate(count=Count('image')).order_by('-count')
|
|
|
|
|
|
|
|
return Response(data)
|
|
|
|
|
|
|
|
|
2020-07-20 21:16:26 +00:00
|
|
|
class PartThumbsUpdate(generics.RetrieveUpdateAPIView):
|
|
|
|
""" API endpoint for updating Part thumbnails"""
|
|
|
|
|
|
|
|
queryset = Part.objects.all()
|
|
|
|
serializer_class = part_serializers.PartThumbSerializerUpdate
|
|
|
|
|
|
|
|
filter_backends = [
|
|
|
|
DjangoFilterBackend
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2020-07-08 14:43:09 +00:00
|
|
|
class PartDetail(generics.RetrieveUpdateDestroyAPIView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" API endpoint for detail view of a single Part object """
|
2020-02-10 12:48:45 +00:00
|
|
|
|
2018-05-04 14:51:17 +00:00
|
|
|
queryset = Part.objects.all()
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.PartSerializer
|
2020-04-19 22:10:59 +00:00
|
|
|
|
|
|
|
starred_parts = None
|
|
|
|
|
|
|
|
def get_queryset(self, *args, **kwargs):
|
|
|
|
queryset = super().get_queryset(*args, **kwargs)
|
|
|
|
|
|
|
|
queryset = part_serializers.PartSerializer.prefetch_queryset(queryset)
|
|
|
|
queryset = part_serializers.PartSerializer.annotate_queryset(queryset)
|
2020-05-01 23:49:05 +00:00
|
|
|
|
2020-04-19 22:10:59 +00:00
|
|
|
return queryset
|
2018-05-04 14:51:17 +00:00
|
|
|
|
2020-04-19 22:10:59 +00:00
|
|
|
def get_serializer(self, *args, **kwargs):
|
|
|
|
|
|
|
|
try:
|
2020-05-01 23:49:05 +00:00
|
|
|
kwargs['category_detail'] = str2bool(self.request.query_params.get('category_detail', False))
|
2020-04-19 22:10:59 +00:00
|
|
|
except AttributeError:
|
2020-05-01 23:49:05 +00:00
|
|
|
pass
|
2020-04-19 22:10:59 +00:00
|
|
|
|
|
|
|
# Ensure the request context is passed through
|
|
|
|
kwargs['context'] = self.get_serializer_context()
|
|
|
|
|
|
|
|
# Pass a list of "starred" parts fo the current user to the serializer
|
|
|
|
# We do this to reduce the number of database queries required!
|
|
|
|
if self.starred_parts is None and self.request is not None:
|
|
|
|
self.starred_parts = [star.part for star in self.request.user.starred_parts.all()]
|
|
|
|
|
|
|
|
kwargs['starred_parts'] = self.starred_parts
|
|
|
|
|
|
|
|
return self.serializer_class(*args, **kwargs)
|
2019-04-13 23:25:46 +00:00
|
|
|
|
2020-07-08 14:43:09 +00:00
|
|
|
def destroy(self, request, *args, **kwargs):
|
|
|
|
# Retrieve part
|
|
|
|
part = Part.objects.get(pk=int(kwargs['pk']))
|
|
|
|
# Check if inactive
|
|
|
|
if not part.active:
|
|
|
|
# Delete
|
|
|
|
return super(PartDetail, self).destroy(request, *args, **kwargs)
|
|
|
|
else:
|
|
|
|
# Return 405 error
|
|
|
|
message = f'Part \'{part.name}\' (pk = {part.pk}) is active: cannot delete'
|
|
|
|
return Response(status=status.HTTP_405_METHOD_NOT_ALLOWED, data=message)
|
|
|
|
|
2020-04-19 22:25:24 +00:00
|
|
|
|
2018-04-14 04:11:46 +00:00
|
|
|
class PartList(generics.ListCreateAPIView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" API endpoint for accessing a list of Part objects
|
|
|
|
|
|
|
|
- GET: Return list of objects
|
|
|
|
- POST: Create a new Part object
|
2020-04-11 12:57:16 +00:00
|
|
|
|
|
|
|
The Part object list can be filtered by:
|
|
|
|
- category: Filter by PartCategory reference
|
|
|
|
- cascade: If true, include parts from sub-categories
|
2020-04-13 11:24:36 +00:00
|
|
|
- starred: Is the part "starred" by the current user?
|
2020-04-11 12:57:16 +00:00
|
|
|
- is_template: Is the part a template part?
|
|
|
|
- variant_of: Filter by variant_of Part reference
|
|
|
|
- assembly: Filter by assembly field
|
2020-04-11 13:02:18 +00:00
|
|
|
- component: Filter by component field
|
2020-04-11 13:03:03 +00:00
|
|
|
- trackable: Filter by trackable field
|
|
|
|
- purchaseable: Filter by purcahseable field
|
|
|
|
- salable: Filter by salable field
|
|
|
|
- active: Filter by active field
|
2020-09-02 13:18:26 +00:00
|
|
|
- ancestor: Filter parts by 'ancestor' (template / variant tree)
|
2019-04-27 12:18:07 +00:00
|
|
|
"""
|
2018-04-14 04:11:46 +00:00
|
|
|
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.PartSerializer
|
2018-04-15 15:02:17 +00:00
|
|
|
|
2020-04-19 12:54:46 +00:00
|
|
|
queryset = Part.objects.all()
|
|
|
|
|
2020-04-19 13:50:41 +00:00
|
|
|
starred_parts = None
|
|
|
|
|
|
|
|
def get_serializer(self, *args, **kwargs):
|
|
|
|
|
|
|
|
# Ensure the request context is passed through
|
|
|
|
kwargs['context'] = self.get_serializer_context()
|
|
|
|
|
|
|
|
# Pass a list of "starred" parts fo the current user to the serializer
|
|
|
|
# We do this to reduce the number of database queries required!
|
2020-04-19 15:23:05 +00:00
|
|
|
if self.starred_parts is None and self.request is not None:
|
2020-04-19 13:50:41 +00:00
|
|
|
self.starred_parts = [star.part for star in self.request.user.starred_parts.all()]
|
|
|
|
|
|
|
|
kwargs['starred_parts'] = self.starred_parts
|
|
|
|
|
|
|
|
return self.serializer_class(*args, **kwargs)
|
|
|
|
|
2020-05-01 23:49:05 +00:00
|
|
|
def list(self, request, *args, **kwargs):
|
|
|
|
"""
|
|
|
|
Overide the 'list' method, as the PartCategory objects are
|
|
|
|
very expensive to serialize!
|
|
|
|
|
|
|
|
So we will serialize them first, and keep them in memory,
|
|
|
|
so that they do not have to be serialized multiple times...
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = self.filter_queryset(self.get_queryset())
|
|
|
|
|
|
|
|
page = self.paginate_queryset(queryset)
|
|
|
|
if page is not None:
|
|
|
|
serializer = self.get_serializer(page, many=True)
|
|
|
|
return self.get_paginated_response(serializer.data)
|
|
|
|
|
|
|
|
serializer = self.get_serializer(queryset, many=True)
|
|
|
|
|
|
|
|
data = serializer.data
|
|
|
|
|
|
|
|
# Do we wish to include PartCategory detail?
|
|
|
|
if str2bool(request.query_params.get('category_detail', False)):
|
|
|
|
|
|
|
|
# Work out which part categorie we need to query
|
|
|
|
category_ids = set()
|
|
|
|
|
|
|
|
for part in data:
|
|
|
|
cat_id = part['category']
|
|
|
|
|
|
|
|
if cat_id is not None:
|
2020-05-02 00:05:35 +00:00
|
|
|
category_ids.add(cat_id)
|
2020-05-01 23:49:05 +00:00
|
|
|
|
|
|
|
# Fetch only the required PartCategory objects from the database
|
|
|
|
categories = PartCategory.objects.filter(pk__in=category_ids).prefetch_related(
|
|
|
|
'parts',
|
|
|
|
'parent',
|
|
|
|
'children',
|
|
|
|
)
|
|
|
|
|
|
|
|
category_map = {}
|
|
|
|
|
|
|
|
# Serialize each PartCategory object
|
|
|
|
for category in categories:
|
|
|
|
category_map[category.pk] = part_serializers.CategorySerializer(category).data
|
|
|
|
|
|
|
|
for part in data:
|
|
|
|
cat_id = part['category']
|
|
|
|
|
|
|
|
if cat_id is not None and cat_id in category_map.keys():
|
2020-05-02 00:05:35 +00:00
|
|
|
detail = category_map[cat_id]
|
2020-05-01 23:49:05 +00:00
|
|
|
else:
|
|
|
|
detail = None
|
|
|
|
|
|
|
|
part['category_detail'] = detail
|
|
|
|
|
2020-05-02 04:03:17 +00:00
|
|
|
"""
|
|
|
|
Determine the response type based on the request.
|
|
|
|
a) For HTTP requests (e.g. via the browseable API) return a DRF response
|
|
|
|
b) For AJAX requests, simply return a JSON rendered response.
|
|
|
|
"""
|
|
|
|
if request.is_ajax():
|
|
|
|
return JsonResponse(data, safe=False)
|
|
|
|
else:
|
|
|
|
return Response(data)
|
2020-05-01 23:49:05 +00:00
|
|
|
|
2020-05-12 11:11:38 +00:00
|
|
|
def perform_create(self, serializer):
|
|
|
|
"""
|
2020-03-18 11:00:32 +00:00
|
|
|
We wish to save the user who created this part!
|
|
|
|
|
2020-05-12 11:11:38 +00:00
|
|
|
Note: Implementation copied from DRF class CreateModelMixin
|
2020-03-18 11:00:32 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
part = serializer.save()
|
2020-05-12 11:11:38 +00:00
|
|
|
part.creation_user = self.request.user
|
2020-03-18 11:00:32 +00:00
|
|
|
part.save()
|
|
|
|
|
2020-04-19 14:49:13 +00:00
|
|
|
def get_queryset(self, *args, **kwargs):
|
|
|
|
|
|
|
|
queryset = super().get_queryset(*args, **kwargs)
|
2020-09-05 12:35:19 +00:00
|
|
|
|
2020-04-19 14:49:13 +00:00
|
|
|
queryset = part_serializers.PartSerializer.prefetch_queryset(queryset)
|
2020-05-02 04:03:17 +00:00
|
|
|
queryset = part_serializers.PartSerializer.annotate_queryset(queryset)
|
2020-04-19 14:49:13 +00:00
|
|
|
|
|
|
|
return queryset
|
|
|
|
|
2020-04-19 12:54:46 +00:00
|
|
|
def filter_queryset(self, queryset):
|
|
|
|
"""
|
2020-05-02 03:46:19 +00:00
|
|
|
Perform custom filtering of the queryset.
|
2020-05-02 03:51:29 +00:00
|
|
|
We overide the DRF filter_fields here because
|
2020-04-19 12:54:46 +00:00
|
|
|
"""
|
|
|
|
|
2020-09-02 13:18:26 +00:00
|
|
|
params = self.request.query_params
|
|
|
|
|
2020-04-19 12:54:46 +00:00
|
|
|
queryset = super().filter_queryset(queryset)
|
|
|
|
|
2020-09-02 13:18:26 +00:00
|
|
|
# Filter by 'ancestor'?
|
|
|
|
ancestor = params.get('ancestor', None)
|
|
|
|
|
|
|
|
if ancestor is not None:
|
|
|
|
# If an 'ancestor' part is provided, filter to match only children
|
|
|
|
try:
|
|
|
|
ancestor = Part.objects.get(pk=ancestor)
|
|
|
|
descendants = ancestor.get_descendants(include_self=False)
|
|
|
|
queryset = queryset.filter(pk__in=[d.pk for d in descendants])
|
|
|
|
except (ValueError, Part.DoesNotExist):
|
|
|
|
pass
|
|
|
|
|
2020-09-30 13:33:58 +00:00
|
|
|
# Filter by whether the part has an IPN (internal part number) defined
|
|
|
|
has_ipn = params.get('has_ipn', None)
|
|
|
|
|
|
|
|
if has_ipn is not None:
|
|
|
|
has_ipn = str2bool(has_ipn)
|
|
|
|
|
|
|
|
if has_ipn:
|
|
|
|
queryset = queryset.exclude(IPN='')
|
|
|
|
else:
|
|
|
|
queryset = queryset.filter(IPN='')
|
|
|
|
|
2020-09-19 11:03:49 +00:00
|
|
|
# Filter by whether the BOM has been validated (or not)
|
|
|
|
bom_valid = params.get('bom_valid', None)
|
2020-09-16 22:02:24 +00:00
|
|
|
|
2020-09-19 11:18:29 +00:00
|
|
|
# TODO: Querying bom_valid status may be quite expensive
|
|
|
|
# TODO: (It needs to be profiled!)
|
|
|
|
# TODO: It might be worth caching the bom_valid status to a database column
|
|
|
|
|
2020-09-19 11:03:49 +00:00
|
|
|
if bom_valid is not None:
|
2020-09-16 22:02:24 +00:00
|
|
|
|
2020-09-19 11:03:49 +00:00
|
|
|
bom_valid = str2bool(bom_valid)
|
2020-09-16 22:02:24 +00:00
|
|
|
|
2020-09-19 11:03:49 +00:00
|
|
|
# Limit queryset to active assemblies
|
|
|
|
queryset = queryset.filter(active=True, assembly=True)
|
|
|
|
|
|
|
|
pks = []
|
|
|
|
|
|
|
|
for part in queryset:
|
|
|
|
if part.is_bom_valid() == bom_valid:
|
|
|
|
pks.append(part.pk)
|
|
|
|
|
|
|
|
queryset = queryset.filter(pk__in=pks)
|
2020-09-16 22:02:24 +00:00
|
|
|
|
2020-04-19 12:54:46 +00:00
|
|
|
# Filter by 'starred' parts?
|
2020-09-02 13:18:26 +00:00
|
|
|
starred = params.get('starred', None)
|
2020-04-19 12:54:46 +00:00
|
|
|
|
|
|
|
if starred is not None:
|
2020-06-28 09:08:13 +00:00
|
|
|
starred = str2bool(starred)
|
2020-04-19 12:54:46 +00:00
|
|
|
starred_parts = [star.part.pk for star in self.request.user.starred_parts.all()]
|
|
|
|
|
|
|
|
if starred:
|
|
|
|
queryset = queryset.filter(pk__in=starred_parts)
|
|
|
|
else:
|
|
|
|
queryset = queryset.exclude(pk__in=starred_parts)
|
|
|
|
|
|
|
|
# Cascade?
|
2020-09-02 13:18:26 +00:00
|
|
|
cascade = str2bool(params.get('cascade', None))
|
2020-04-19 12:54:46 +00:00
|
|
|
|
|
|
|
# Does the user wish to filter by category?
|
2020-09-02 13:18:26 +00:00
|
|
|
cat_id = params.get('category', None)
|
2020-04-19 12:54:46 +00:00
|
|
|
|
|
|
|
if cat_id is None:
|
|
|
|
# No category filtering if category is not specified
|
|
|
|
pass
|
|
|
|
|
|
|
|
else:
|
|
|
|
# Category has been specified!
|
|
|
|
if isNull(cat_id):
|
|
|
|
# A 'null' category is the top-level category
|
|
|
|
if cascade is False:
|
|
|
|
# Do not cascade, only list parts in the top-level category
|
|
|
|
queryset = queryset.filter(category=None)
|
|
|
|
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
category = PartCategory.objects.get(pk=cat_id)
|
|
|
|
|
|
|
|
# If '?cascade=true' then include parts which exist in sub-categories
|
|
|
|
if cascade:
|
|
|
|
queryset = queryset.filter(category__in=category.getUniqueChildren())
|
|
|
|
# Just return parts directly in the requested category
|
|
|
|
else:
|
|
|
|
queryset = queryset.filter(category=cat_id)
|
|
|
|
except (ValueError, PartCategory.DoesNotExist):
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Annotate calculated data to the queryset
|
|
|
|
# (This will be used for further filtering)
|
|
|
|
queryset = part_serializers.PartSerializer.annotate_queryset(queryset)
|
|
|
|
|
|
|
|
# Filter by whether the part has stock
|
2020-09-02 13:18:26 +00:00
|
|
|
has_stock = params.get("has_stock", None)
|
2020-09-19 11:03:49 +00:00
|
|
|
|
2020-04-19 12:54:46 +00:00
|
|
|
if has_stock is not None:
|
|
|
|
has_stock = str2bool(has_stock)
|
|
|
|
|
|
|
|
if has_stock:
|
|
|
|
queryset = queryset.filter(Q(in_stock__gt=0))
|
|
|
|
else:
|
2020-04-19 13:50:41 +00:00
|
|
|
queryset = queryset.filter(Q(in_stock__lte=0))
|
2020-04-19 12:54:46 +00:00
|
|
|
|
|
|
|
# If we are filtering by 'low_stock' status
|
2020-09-02 13:18:26 +00:00
|
|
|
low_stock = params.get('low_stock', None)
|
2020-04-19 12:54:46 +00:00
|
|
|
|
|
|
|
if low_stock is not None:
|
|
|
|
low_stock = str2bool(low_stock)
|
|
|
|
|
|
|
|
if low_stock:
|
|
|
|
# Ignore any parts which do not have a specified 'minimum_stock' level
|
|
|
|
queryset = queryset.exclude(minimum_stock=0)
|
|
|
|
# Filter items which have an 'in_stock' level lower than 'minimum_stock'
|
|
|
|
queryset = queryset.filter(Q(in_stock__lt=F('minimum_stock')))
|
|
|
|
else:
|
|
|
|
# Filter items which have an 'in_stock' level higher than 'minimum_stock'
|
|
|
|
queryset = queryset.filter(Q(in_stock__gte=F('minimum_stock')))
|
|
|
|
|
2020-09-18 16:40:50 +00:00
|
|
|
# Filter by "parts which need stock to complete build"
|
|
|
|
stock_to_build = params.get('stock_to_build', None)
|
|
|
|
|
2020-09-19 11:18:29 +00:00
|
|
|
# TODO: This is super expensive, database query wise...
|
|
|
|
# TODO: Need to figure out a cheaper way of making this filter query
|
|
|
|
|
2020-09-18 16:40:50 +00:00
|
|
|
if stock_to_build is not None:
|
|
|
|
# Filter only active parts
|
|
|
|
queryset = queryset.filter(active=True)
|
2020-09-22 16:24:09 +00:00
|
|
|
# Prefetch current active builds
|
|
|
|
build_active_queryset = Build.objects.filter(status__in=BuildStatus.ACTIVE_CODES)
|
|
|
|
build_active_prefetch = Prefetch('builds',
|
|
|
|
queryset=build_active_queryset,
|
|
|
|
to_attr='current_builds')
|
|
|
|
parts = queryset.prefetch_related(build_active_prefetch)
|
|
|
|
|
|
|
|
# Store parts with builds needing stock
|
2020-09-18 16:40:50 +00:00
|
|
|
parts_need_stock = []
|
|
|
|
|
|
|
|
# Find parts with active builds
|
|
|
|
# where any subpart's stock is lower than quantity being built
|
2020-09-22 16:24:09 +00:00
|
|
|
for part in parts:
|
|
|
|
if part.current_builds:
|
|
|
|
builds_ids = [build.id for build in part.current_builds]
|
|
|
|
total_build_quantity = build_active_queryset.filter(pk__in=builds_ids).aggregate(quantity=Sum('quantity'))['quantity']
|
|
|
|
|
|
|
|
if part.can_build < total_build_quantity:
|
|
|
|
parts_need_stock.append(part.pk)
|
2020-09-18 16:40:50 +00:00
|
|
|
|
|
|
|
queryset = queryset.filter(pk__in=parts_need_stock)
|
|
|
|
|
2020-09-19 10:26:17 +00:00
|
|
|
# Limit choices
|
|
|
|
limit = params.get('limit', None)
|
|
|
|
|
|
|
|
if limit is not None:
|
|
|
|
try:
|
|
|
|
limit = int(limit)
|
|
|
|
if limit > 0:
|
|
|
|
queryset = queryset[:limit]
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
2020-04-19 12:54:46 +00:00
|
|
|
return queryset
|
|
|
|
|
2018-04-23 11:10:13 +00:00
|
|
|
filter_backends = [
|
|
|
|
DjangoFilterBackend,
|
|
|
|
filters.SearchFilter,
|
|
|
|
filters.OrderingFilter,
|
|
|
|
]
|
2018-04-14 04:19:03 +00:00
|
|
|
|
2018-04-23 11:10:13 +00:00
|
|
|
filter_fields = [
|
2019-05-26 01:15:41 +00:00
|
|
|
'is_template',
|
|
|
|
'variant_of',
|
2019-06-02 09:46:30 +00:00
|
|
|
'assembly',
|
|
|
|
'component',
|
2019-04-15 14:01:15 +00:00
|
|
|
'trackable',
|
|
|
|
'purchaseable',
|
|
|
|
'salable',
|
2019-04-28 13:15:30 +00:00
|
|
|
'active',
|
2018-04-23 11:10:13 +00:00
|
|
|
]
|
2018-04-14 04:19:03 +00:00
|
|
|
|
2018-04-23 11:10:13 +00:00
|
|
|
ordering_fields = [
|
|
|
|
'name',
|
2020-09-19 10:26:17 +00:00
|
|
|
'creation_date',
|
2018-04-23 11:10:13 +00:00
|
|
|
]
|
2018-04-14 04:19:03 +00:00
|
|
|
|
2020-04-19 13:50:41 +00:00
|
|
|
# Default ordering
|
2018-04-23 11:10:13 +00:00
|
|
|
ordering = 'name'
|
2018-04-14 04:19:03 +00:00
|
|
|
|
2018-04-23 11:18:35 +00:00
|
|
|
search_fields = [
|
2019-04-16 22:19:40 +00:00
|
|
|
'$name',
|
2018-04-23 11:18:35 +00:00
|
|
|
'description',
|
2019-05-06 01:23:55 +00:00
|
|
|
'$IPN',
|
2019-05-14 07:23:20 +00:00
|
|
|
'keywords',
|
2018-04-23 11:18:35 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
|
2019-05-05 00:36:48 +00:00
|
|
|
class PartStarDetail(generics.RetrieveDestroyAPIView):
|
|
|
|
""" API endpoint for viewing or removing a PartStar object """
|
|
|
|
|
|
|
|
queryset = PartStar.objects.all()
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.PartStarSerializer
|
2019-05-05 00:36:48 +00:00
|
|
|
|
|
|
|
|
2019-05-04 23:05:44 +00:00
|
|
|
class PartStarList(generics.ListCreateAPIView):
|
|
|
|
""" API endpoint for accessing a list of PartStar objects.
|
|
|
|
|
|
|
|
- GET: Return list of PartStar objects
|
|
|
|
- POST: Create a new PartStar object
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = PartStar.objects.all()
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.PartStarSerializer
|
2019-05-04 23:05:44 +00:00
|
|
|
|
2019-05-04 23:14:12 +00:00
|
|
|
def create(self, request, *args, **kwargs):
|
|
|
|
|
2019-05-05 00:47:57 +00:00
|
|
|
# Override the user field (with the logged-in user)
|
2019-05-05 00:36:48 +00:00
|
|
|
data = request.data.copy()
|
|
|
|
data['user'] = str(request.user.id)
|
|
|
|
|
|
|
|
serializer = self.get_serializer(data=data)
|
2019-05-04 23:14:12 +00:00
|
|
|
|
2019-05-05 00:36:48 +00:00
|
|
|
serializer.is_valid(raise_exception=True)
|
|
|
|
self.perform_create(serializer)
|
|
|
|
headers = self.get_success_headers(serializer.data)
|
|
|
|
return Response(serializer.data, status=status.HTTP_201_CREATED, headers=headers)
|
2019-05-04 23:14:12 +00:00
|
|
|
|
2019-05-04 23:05:44 +00:00
|
|
|
permission_classes = [
|
2019-07-08 09:20:00 +00:00
|
|
|
permissions.IsAuthenticated,
|
2019-05-04 23:05:44 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
filter_backends = [
|
|
|
|
DjangoFilterBackend,
|
|
|
|
filters.SearchFilter
|
|
|
|
]
|
|
|
|
|
|
|
|
filter_fields = [
|
|
|
|
'part',
|
|
|
|
'user',
|
|
|
|
]
|
|
|
|
|
|
|
|
search_fields = [
|
|
|
|
'partname'
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2019-09-07 09:44:10 +00:00
|
|
|
class PartParameterTemplateList(generics.ListCreateAPIView):
|
|
|
|
""" API endpoint for accessing a list of PartParameterTemplate objects.
|
|
|
|
|
|
|
|
- GET: Return list of PartParameterTemplate objects
|
|
|
|
- POST: Create a new PartParameterTemplate object
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = PartParameterTemplate.objects.all()
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.PartParameterTemplateSerializer
|
2019-09-07 09:44:10 +00:00
|
|
|
|
|
|
|
filter_backends = [
|
|
|
|
filters.OrderingFilter,
|
|
|
|
]
|
|
|
|
|
|
|
|
filter_fields = [
|
|
|
|
'name',
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
class PartParameterList(generics.ListCreateAPIView):
|
|
|
|
""" API endpoint for accessing a list of PartParameter objects
|
|
|
|
|
|
|
|
- GET: Return list of PartParameter objects
|
|
|
|
- POST: Create a new PartParameter object
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = PartParameter.objects.all()
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.PartParameterSerializer
|
2019-09-07 09:44:10 +00:00
|
|
|
|
|
|
|
filter_backends = [
|
|
|
|
DjangoFilterBackend
|
|
|
|
]
|
|
|
|
|
|
|
|
filter_fields = [
|
|
|
|
'part',
|
|
|
|
'template',
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2019-04-25 14:29:53 +00:00
|
|
|
class BomList(generics.ListCreateAPIView):
|
2019-05-04 23:05:44 +00:00
|
|
|
""" API endpoint for accessing a list of BomItem objects.
|
2019-04-27 12:18:07 +00:00
|
|
|
|
|
|
|
- GET: Return list of BomItem objects
|
|
|
|
- POST: Create a new BomItem object
|
|
|
|
"""
|
2019-05-23 12:36:19 +00:00
|
|
|
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.BomItemSerializer
|
2020-08-15 11:52:32 +00:00
|
|
|
|
|
|
|
def list(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
queryset = self.filter_queryset(self.get_queryset())
|
|
|
|
|
|
|
|
serializer = self.get_serializer(queryset, many=True)
|
|
|
|
|
|
|
|
data = serializer.data
|
|
|
|
|
|
|
|
if request.is_ajax():
|
|
|
|
return JsonResponse(data, safe=False)
|
|
|
|
else:
|
|
|
|
return Response(data)
|
|
|
|
|
2019-05-23 12:36:19 +00:00
|
|
|
def get_serializer(self, *args, **kwargs):
|
|
|
|
|
|
|
|
# Do we wish to include extra detail?
|
2019-07-07 23:39:58 +00:00
|
|
|
try:
|
2020-04-28 13:17:15 +00:00
|
|
|
kwargs['part_detail'] = str2bool(self.request.GET.get('part_detail', None))
|
2019-07-07 23:39:58 +00:00
|
|
|
except AttributeError:
|
2020-04-28 13:17:15 +00:00
|
|
|
pass
|
2019-05-23 12:36:19 +00:00
|
|
|
|
2020-04-28 13:17:15 +00:00
|
|
|
try:
|
|
|
|
kwargs['sub_part_detail'] = str2bool(self.request.GET.get('sub_part_detail', None))
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
|
2020-04-19 13:50:41 +00:00
|
|
|
# Ensure the request context is passed through!
|
2019-05-23 12:36:19 +00:00
|
|
|
kwargs['context'] = self.get_serializer_context()
|
2020-04-19 13:50:41 +00:00
|
|
|
|
2019-05-23 12:36:19 +00:00
|
|
|
return self.serializer_class(*args, **kwargs)
|
2018-05-02 13:42:57 +00:00
|
|
|
|
2020-08-15 11:52:32 +00:00
|
|
|
def get_queryset(self, *args, **kwargs):
|
|
|
|
|
2019-05-19 22:31:03 +00:00
|
|
|
queryset = BomItem.objects.all()
|
2020-08-15 11:52:32 +00:00
|
|
|
|
2019-05-19 22:31:03 +00:00
|
|
|
queryset = self.get_serializer_class().setup_eager_loading(queryset)
|
2020-04-11 14:20:29 +00:00
|
|
|
|
2020-04-28 13:17:15 +00:00
|
|
|
return queryset
|
|
|
|
|
|
|
|
def filter_queryset(self, queryset):
|
|
|
|
|
2020-04-28 13:17:59 +00:00
|
|
|
queryset = super().filter_queryset(queryset)
|
2020-04-28 13:17:15 +00:00
|
|
|
|
2020-10-04 13:42:09 +00:00
|
|
|
params = self.request.query_params
|
|
|
|
|
|
|
|
# Filter by "optional" status?
|
|
|
|
optional = params.get('optional', None)
|
|
|
|
|
|
|
|
if optional is not None:
|
|
|
|
optional = str2bool(optional)
|
|
|
|
|
|
|
|
queryset = queryset.filter(optional=optional)
|
|
|
|
|
2020-04-11 14:20:29 +00:00
|
|
|
# Filter by part?
|
2020-10-04 13:42:09 +00:00
|
|
|
part = params.get('part', None)
|
2020-04-11 14:20:29 +00:00
|
|
|
|
|
|
|
if part is not None:
|
|
|
|
queryset = queryset.filter(part=part)
|
|
|
|
|
|
|
|
# Filter by sub-part?
|
2020-10-04 13:42:09 +00:00
|
|
|
sub_part = params.get('sub_part', None)
|
2020-04-11 14:20:29 +00:00
|
|
|
|
|
|
|
if sub_part is not None:
|
|
|
|
queryset = queryset.filter(sub_part=sub_part)
|
|
|
|
|
2020-10-04 02:51:52 +00:00
|
|
|
# Filter by "trackable" status of the sub-part
|
2020-10-04 13:42:09 +00:00
|
|
|
trackable = params.get('trackable', None)
|
2020-10-04 02:51:52 +00:00
|
|
|
|
|
|
|
if trackable is not None:
|
|
|
|
trackable = str2bool(trackable)
|
|
|
|
queryset = queryset.filter(sub_part__trackable=trackable)
|
|
|
|
|
2019-05-19 22:31:03 +00:00
|
|
|
return queryset
|
|
|
|
|
2018-05-02 13:42:57 +00:00
|
|
|
filter_backends = [
|
|
|
|
DjangoFilterBackend,
|
|
|
|
filters.SearchFilter,
|
|
|
|
filters.OrderingFilter,
|
|
|
|
]
|
|
|
|
|
|
|
|
filter_fields = [
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2019-04-25 14:29:53 +00:00
|
|
|
class BomDetail(generics.RetrieveUpdateDestroyAPIView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" API endpoint for detail view of a single BomItem object """
|
2019-04-25 14:29:53 +00:00
|
|
|
|
|
|
|
queryset = BomItem.objects.all()
|
2020-02-10 11:03:06 +00:00
|
|
|
serializer_class = part_serializers.BomItemSerializer
|
2019-04-25 14:29:53 +00:00
|
|
|
|
|
|
|
|
2019-09-05 09:29:51 +00:00
|
|
|
class BomItemValidate(generics.UpdateAPIView):
|
|
|
|
""" API endpoint for validating a BomItem """
|
|
|
|
|
|
|
|
# Very simple serializers
|
|
|
|
class BomItemValidationSerializer(serializers.Serializer):
|
|
|
|
|
|
|
|
valid = serializers.BooleanField(default=False)
|
|
|
|
|
|
|
|
queryset = BomItem.objects.all()
|
|
|
|
serializer_class = BomItemValidationSerializer
|
|
|
|
|
|
|
|
def update(self, request, *args, **kwargs):
|
|
|
|
""" Perform update request """
|
|
|
|
|
|
|
|
partial = kwargs.pop('partial', False)
|
|
|
|
|
|
|
|
valid = request.data.get('valid', False)
|
|
|
|
|
|
|
|
instance = self.get_object()
|
|
|
|
|
|
|
|
serializer = self.get_serializer(instance, data=request.data, partial=partial)
|
|
|
|
serializer.is_valid(raise_exception=True)
|
|
|
|
|
|
|
|
if type(instance) == BomItem:
|
|
|
|
instance.validate_hash(valid)
|
|
|
|
|
|
|
|
return Response(serializer.data)
|
|
|
|
|
2020-05-11 13:32:40 +00:00
|
|
|
|
2018-04-23 11:10:13 +00:00
|
|
|
part_api_urls = [
|
2018-04-28 13:22:12 +00:00
|
|
|
url(r'^tree/?', PartCategoryTree.as_view(), name='api-part-tree'),
|
|
|
|
|
2020-05-11 13:25:55 +00:00
|
|
|
# Base URL for PartCategory API endpoints
|
|
|
|
url(r'^category/', include([
|
|
|
|
url(r'^(?P<pk>\d+)/?', CategoryDetail.as_view(), name='api-part-category-detail'),
|
2020-05-11 13:44:22 +00:00
|
|
|
url(r'^$', CategoryList.as_view(), name='api-part-category-list'),
|
2020-05-11 13:25:55 +00:00
|
|
|
])),
|
|
|
|
|
2020-05-17 03:56:49 +00:00
|
|
|
# Base URL for PartTestTemplate API endpoints
|
|
|
|
url(r'^test-template/', include([
|
|
|
|
url(r'^$', PartTestTemplateList.as_view(), name='api-part-test-template-list'),
|
|
|
|
])),
|
|
|
|
|
2020-05-11 13:25:55 +00:00
|
|
|
# Base URL for PartAttachment API endpoints
|
2020-05-17 03:56:49 +00:00
|
|
|
url(r'^attachment/', include([
|
2020-05-11 13:44:22 +00:00
|
|
|
url(r'^$', PartAttachmentList.as_view(), name='api-part-attachment-list'),
|
2020-05-11 13:25:55 +00:00
|
|
|
])),
|
|
|
|
|
|
|
|
# Base URL for PartStar API endpoints
|
|
|
|
url(r'^star/', include([
|
|
|
|
url(r'^(?P<pk>\d+)/?', PartStarDetail.as_view(), name='api-part-star-detail'),
|
2020-05-11 13:44:22 +00:00
|
|
|
url(r'^$', PartStarList.as_view(), name='api-part-star-list'),
|
2020-05-11 13:25:55 +00:00
|
|
|
])),
|
2020-09-18 12:11:51 +00:00
|
|
|
|
|
|
|
# Base URL for part sale pricing
|
|
|
|
url(r'^sale-price/', include([
|
|
|
|
url(r'^.*$', PartSalePriceList.as_view(), name='api-part-sale-price-list'),
|
|
|
|
])),
|
2020-05-11 13:25:55 +00:00
|
|
|
|
|
|
|
# Base URL for PartParameter API endpoints
|
|
|
|
url(r'^parameter/', include([
|
|
|
|
url(r'^template/$', PartParameterTemplateList.as_view(), name='api-part-param-template-list'),
|
|
|
|
url(r'^.*$', PartParameterList.as_view(), name='api-part-param-list'),
|
|
|
|
])),
|
2019-05-04 23:05:44 +00:00
|
|
|
|
2020-07-20 21:16:26 +00:00
|
|
|
url(r'^thumbs/', include([
|
|
|
|
url(r'^$', PartThumbs.as_view(), name='api-part-thumbs'),
|
|
|
|
url(r'^(?P<pk>\d+)/?', PartThumbsUpdate.as_view(), name='api-part-thumbs-update'),
|
|
|
|
])),
|
2020-02-10 11:03:06 +00:00
|
|
|
|
2019-06-23 14:01:34 +00:00
|
|
|
url(r'^(?P<pk>\d+)/?', PartDetail.as_view(), name='api-part-detail'),
|
2018-05-04 14:51:17 +00:00
|
|
|
|
2018-04-23 11:10:13 +00:00
|
|
|
url(r'^.*$', PartList.as_view(), name='api-part-list'),
|
|
|
|
]
|
2019-04-25 14:29:53 +00:00
|
|
|
|
|
|
|
bom_api_urls = [
|
|
|
|
# BOM Item Detail
|
2020-05-11 13:25:55 +00:00
|
|
|
url(r'^(?P<pk>\d+)/', include([
|
|
|
|
url(r'^validate/?', BomItemValidate.as_view(), name='api-bom-item-validate'),
|
|
|
|
url(r'^.*$', BomDetail.as_view(), name='api-bom-item-detail'),
|
|
|
|
])),
|
2019-04-25 14:29:53 +00:00
|
|
|
|
|
|
|
# Catch-all
|
|
|
|
url(r'^.*$', BomList.as_view(), name='api-bom-list'),
|
2019-04-26 13:34:15 +00:00
|
|
|
]
|