mirror of
https://github.com/inventree/InvenTree
synced 2024-08-30 18:33:04 +00:00
Added stockhistory
using django-simple-history
This commit is contained in:
parent
81a42c1dff
commit
7478371d0c
@ -34,6 +34,7 @@ ALLOWED_HOSTS = []
|
|||||||
INSTALLED_APPS = [
|
INSTALLED_APPS = [
|
||||||
'django_filters',
|
'django_filters',
|
||||||
'rest_framework',
|
'rest_framework',
|
||||||
|
'simple_history',
|
||||||
|
|
||||||
# Core django modules
|
# Core django modules
|
||||||
'django.contrib.admin',
|
'django.contrib.admin',
|
||||||
@ -59,6 +60,7 @@ MIDDLEWARE = [
|
|||||||
'django.contrib.auth.middleware.AuthenticationMiddleware',
|
'django.contrib.auth.middleware.AuthenticationMiddleware',
|
||||||
'django.contrib.messages.middleware.MessageMiddleware',
|
'django.contrib.messages.middleware.MessageMiddleware',
|
||||||
'django.middleware.clickjacking.XFrameOptionsMiddleware',
|
'django.middleware.clickjacking.XFrameOptionsMiddleware',
|
||||||
|
'simple_history.middleware.HistoryRequestMiddleware',
|
||||||
]
|
]
|
||||||
|
|
||||||
ROOT_URLCONF = 'InvenTree.urls'
|
ROOT_URLCONF = 'InvenTree.urls'
|
||||||
|
@ -4,7 +4,7 @@ from django.contrib import admin
|
|||||||
from rest_framework.documentation import include_docs_urls
|
from rest_framework.documentation import include_docs_urls
|
||||||
|
|
||||||
from part.urls import part_urls, part_cat_urls, part_param_urls, part_param_template_urls
|
from part.urls import part_urls, part_cat_urls, part_param_urls, part_param_template_urls
|
||||||
from stock.urls import stock_urls, stock_loc_urls, stock_track_urls
|
from stock.urls import stock_urls, stock_loc_urls
|
||||||
from project.urls import prj_urls, prj_part_urls, prj_cat_urls, prj_run_urls
|
from project.urls import prj_urls, prj_part_urls, prj_cat_urls, prj_run_urls
|
||||||
from supplier.urls import cust_urls, manu_urls, supplier_part_urls, price_break_urls, supplier_urls
|
from supplier.urls import cust_urls, manu_urls, supplier_part_urls, price_break_urls, supplier_urls
|
||||||
from track.urls import unique_urls, part_track_urls
|
from track.urls import unique_urls, part_track_urls
|
||||||
@ -17,7 +17,6 @@ apipatterns = [
|
|||||||
# Stock URLs
|
# Stock URLs
|
||||||
url(r'^stock/', include(stock_urls)),
|
url(r'^stock/', include(stock_urls)),
|
||||||
url(r'^stock-location/', include(stock_loc_urls)),
|
url(r'^stock-location/', include(stock_loc_urls)),
|
||||||
url(r'^stock-track/', include(stock_track_urls)),
|
|
||||||
|
|
||||||
# Part URLs
|
# Part URLs
|
||||||
url(r'^part/', include(part_urls)),
|
url(r'^part/', include(part_urls)),
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
|
from simple_history.admin import SimpleHistoryAdmin
|
||||||
|
|
||||||
from .models import StockLocation, StockItem
|
from .models import StockLocation, StockItem
|
||||||
|
|
||||||
@ -7,7 +8,7 @@ class LocationAdmin(admin.ModelAdmin):
|
|||||||
list_display = ('name', 'path', 'description')
|
list_display = ('name', 'path', 'description')
|
||||||
|
|
||||||
|
|
||||||
class StockItemAdmin(admin.ModelAdmin):
|
class StockItemAdmin(SimpleHistoryAdmin):
|
||||||
list_display = ('part', 'quantity', 'location', 'status', 'updated')
|
list_display = ('part', 'quantity', 'location', 'status', 'updated')
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,8 +1,9 @@
|
|||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
from django.utils.translation import ugettext as _
|
from django.utils.translation import ugettext as _
|
||||||
from django.db import models
|
from django.db import models, transaction
|
||||||
from django.core.validators import MinValueValidator
|
from django.core.validators import MinValueValidator
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
|
from simple_history.models import HistoricalRecords
|
||||||
|
|
||||||
from supplier.models import SupplierPart
|
from supplier.models import SupplierPart
|
||||||
from part.models import Part
|
from part.models import Part
|
||||||
@ -65,6 +66,10 @@ class StockItem(models.Model):
|
|||||||
|
|
||||||
infinite = models.BooleanField(default=False)
|
infinite = models.BooleanField(default=False)
|
||||||
|
|
||||||
|
# History of this item
|
||||||
|
history = HistoricalRecords()
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def stocktake(self, count, user):
|
def stocktake(self, count, user):
|
||||||
""" Perform item stocktake.
|
""" Perform item stocktake.
|
||||||
When the quantity of an item is counted,
|
When the quantity of an item is counted,
|
||||||
@ -81,6 +86,7 @@ class StockItem(models.Model):
|
|||||||
self.stocktake_user = user
|
self.stocktake_user = user
|
||||||
self.save()
|
self.save()
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def add_stock(self, amount):
|
def add_stock(self, amount):
|
||||||
""" Add items to stock
|
""" Add items to stock
|
||||||
This function can be called by initiating a ProjectRun,
|
This function can be called by initiating a ProjectRun,
|
||||||
@ -101,6 +107,7 @@ class StockItem(models.Model):
|
|||||||
self.quantity = q
|
self.quantity = q
|
||||||
self.save()
|
self.save()
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
def take_stock(self, amount):
|
def take_stock(self, amount):
|
||||||
self.add_stock(-amount)
|
self.add_stock(-amount)
|
||||||
|
|
||||||
@ -109,15 +116,3 @@ class StockItem(models.Model):
|
|||||||
n=self.quantity,
|
n=self.quantity,
|
||||||
part=self.part.name,
|
part=self.part.name,
|
||||||
loc=self.location.name)
|
loc=self.location.name)
|
||||||
|
|
||||||
|
|
||||||
class StockTracking(models.Model):
|
|
||||||
""" Tracks a single movement of stock
|
|
||||||
- Used to track stock being taken from a location
|
|
||||||
- Used to track stock being added to a location
|
|
||||||
- "Pending" flag shows that stock WILL be taken / added
|
|
||||||
"""
|
|
||||||
|
|
||||||
item = models.ForeignKey(StockItem, on_delete=models.CASCADE, related_name='tracking')
|
|
||||||
quantity = models.IntegerField()
|
|
||||||
when = models.DateTimeField(auto_now=True)
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
|
|
||||||
from .models import StockItem, StockLocation, StockTracking
|
from .models import StockItem, StockLocation
|
||||||
|
|
||||||
|
|
||||||
class StockItemSerializer(serializers.HyperlinkedModelSerializer):
|
class StockItemSerializer(serializers.HyperlinkedModelSerializer):
|
||||||
@ -25,7 +25,10 @@ class StockItemSerializer(serializers.HyperlinkedModelSerializer):
|
|||||||
""" These fields are read-only in this context.
|
""" These fields are read-only in this context.
|
||||||
They can be updated by accessing the appropriate API endpoints
|
They can be updated by accessing the appropriate API endpoints
|
||||||
"""
|
"""
|
||||||
read_only_fields = ('stocktake_date', 'quantity',)
|
read_only_fields = ('stocktake_date',
|
||||||
|
'stocktake_user',
|
||||||
|
'updated',
|
||||||
|
'quantity',)
|
||||||
|
|
||||||
|
|
||||||
class StockQuantitySerializer(serializers.ModelSerializer):
|
class StockQuantitySerializer(serializers.ModelSerializer):
|
||||||
@ -46,13 +49,3 @@ class LocationSerializer(serializers.HyperlinkedModelSerializer):
|
|||||||
'description',
|
'description',
|
||||||
'parent',
|
'parent',
|
||||||
'path')
|
'path')
|
||||||
|
|
||||||
|
|
||||||
class StockTrackingSerializer(serializers.HyperlinkedModelSerializer):
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = StockTracking
|
|
||||||
fields = ('url',
|
|
||||||
'item',
|
|
||||||
'quantity',
|
|
||||||
'when')
|
|
||||||
|
@ -26,11 +26,3 @@ stock_loc_urls = [
|
|||||||
|
|
||||||
url(r'^$', views.LocationList.as_view())
|
url(r'^$', views.LocationList.as_view())
|
||||||
]
|
]
|
||||||
|
|
||||||
stock_track_urls = [
|
|
||||||
url(r'^(?P<pk>[0-9]+)/?$', views.StockTrackingDetail.as_view(), name='stocktracking-detail'),
|
|
||||||
|
|
||||||
url(r'^\?.*/?$', views.StockTrackingList.as_view()),
|
|
||||||
|
|
||||||
url(r'^$', views.StockTrackingList.as_view())
|
|
||||||
]
|
|
||||||
|
@ -4,9 +4,9 @@ from django_filters import NumberFilter
|
|||||||
from rest_framework import generics, permissions, response
|
from rest_framework import generics, permissions, response
|
||||||
|
|
||||||
# from InvenTree.models import FilterChildren
|
# from InvenTree.models import FilterChildren
|
||||||
from .models import StockLocation, StockItem, StockTracking
|
from .models import StockLocation, StockItem
|
||||||
from .serializers import StockItemSerializer, StockQuantitySerializer
|
from .serializers import StockItemSerializer, StockQuantitySerializer
|
||||||
from .serializers import LocationSerializer, StockTrackingSerializer
|
from .serializers import LocationSerializer
|
||||||
|
|
||||||
|
|
||||||
class StockDetail(generics.RetrieveUpdateDestroyAPIView):
|
class StockDetail(generics.RetrieveUpdateDestroyAPIView):
|
||||||
@ -127,47 +127,3 @@ class LocationList(generics.ListCreateAPIView):
|
|||||||
permission_classes = (permissions.IsAuthenticatedOrReadOnly,)
|
permission_classes = (permissions.IsAuthenticatedOrReadOnly,)
|
||||||
filter_backends = (DjangoFilterBackend,)
|
filter_backends = (DjangoFilterBackend,)
|
||||||
filter_class = StockLocationFilter
|
filter_class = StockLocationFilter
|
||||||
|
|
||||||
|
|
||||||
class StockTrackingDetail(generics.RetrieveUpdateDestroyAPIView):
|
|
||||||
"""
|
|
||||||
|
|
||||||
get:
|
|
||||||
Return a single StockTracking object
|
|
||||||
|
|
||||||
post:
|
|
||||||
Update a StockTracking object
|
|
||||||
|
|
||||||
delete:
|
|
||||||
Remove a StockTracking object
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
queryset = StockTracking.objects.all()
|
|
||||||
serializer_class = StockTrackingSerializer
|
|
||||||
permission_classes = (permissions.IsAuthenticatedOrReadOnly,)
|
|
||||||
|
|
||||||
|
|
||||||
class StockTrackingFilter(FilterSet):
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = StockTracking
|
|
||||||
fields = ['item']
|
|
||||||
|
|
||||||
|
|
||||||
class StockTrackingList(generics.ListCreateAPIView):
|
|
||||||
"""
|
|
||||||
|
|
||||||
get:
|
|
||||||
Return a list of all StockTracking items
|
|
||||||
|
|
||||||
post:
|
|
||||||
Create a new StockTracking item
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
queryset = StockTracking.objects.all()
|
|
||||||
serializer_class = StockTrackingSerializer
|
|
||||||
permission_classes = (permissions.IsAuthenticatedOrReadOnly,)
|
|
||||||
filter_backends = (DjangoFilterBackend,)
|
|
||||||
filter_class = StockTrackingFilter
|
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
Django==1.11
|
Django==1.11
|
||||||
djangorestframework==3.6.2
|
djangorestframework==3.6.2
|
||||||
django_filter==1.0.2
|
django_filter==1.0.2
|
||||||
|
django-simple-history==1.8.2
|
||||||
coreapi==2.3.0
|
coreapi==2.3.0
|
||||||
pygments==2.2.0
|
pygments==2.2.0
|
||||||
|
Loading…
Reference in New Issue
Block a user