2018-04-17 06:58:37 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
2017-03-27 10:04:15 +00:00
|
|
|
from __future__ import unicode_literals
|
2018-04-17 06:58:37 +00:00
|
|
|
|
2018-04-27 14:06:39 +00:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
from django.core.exceptions import ValidationError
|
2018-04-17 06:58:37 +00:00
|
|
|
|
2017-04-21 13:47:04 +00:00
|
|
|
from django.db import models, transaction
|
2017-04-16 07:05:02 +00:00
|
|
|
from django.core.validators import MinValueValidator
|
2017-04-20 12:40:59 +00:00
|
|
|
from django.contrib.auth.models import User
|
2018-04-16 14:32:02 +00:00
|
|
|
from django.db.models.signals import pre_delete
|
|
|
|
from django.dispatch import receiver
|
|
|
|
|
|
|
|
from datetime import datetime
|
2017-03-27 10:04:15 +00:00
|
|
|
|
2017-03-27 11:55:21 +00:00
|
|
|
from InvenTree.models import InvenTreeTree
|
2017-03-28 12:25:38 +00:00
|
|
|
|
2018-04-27 14:06:39 +00:00
|
|
|
from part.models import Part
|
|
|
|
|
2018-04-15 15:02:17 +00:00
|
|
|
|
2017-04-11 08:58:44 +00:00
|
|
|
class StockLocation(InvenTreeTree):
|
|
|
|
""" Organization tree for StockItem objects
|
2018-04-12 06:27:26 +00:00
|
|
|
A "StockLocation" can be considered a warehouse, or storage location
|
|
|
|
Stock locations can be heirarchical as required
|
2017-04-11 08:58:44 +00:00
|
|
|
"""
|
|
|
|
|
2018-04-15 13:27:56 +00:00
|
|
|
def get_absolute_url(self):
|
|
|
|
return '/stock/location/{id}/'.format(id=self.id)
|
|
|
|
|
2017-04-11 08:58:44 +00:00
|
|
|
@property
|
|
|
|
def items(self):
|
2018-04-16 13:26:02 +00:00
|
|
|
return self.stockitem_set.all()
|
|
|
|
|
|
|
|
@property
|
|
|
|
def has_items(self):
|
|
|
|
return self.items.count() > 0
|
2017-03-29 11:55:28 +00:00
|
|
|
|
2017-03-28 12:25:38 +00:00
|
|
|
|
2018-04-14 10:33:53 +00:00
|
|
|
@receiver(pre_delete, sender=StockLocation, dispatch_uid='stocklocation_delete_log')
|
|
|
|
def before_delete_stock_location(sender, instance, using, **kwargs):
|
|
|
|
|
|
|
|
# Update each part in the stock location
|
|
|
|
for item in instance.items.all():
|
2018-04-16 13:28:53 +00:00
|
|
|
item.location = instance.parent
|
|
|
|
item.save()
|
2018-04-14 10:33:53 +00:00
|
|
|
|
2018-04-15 13:27:56 +00:00
|
|
|
# Update each child category
|
|
|
|
for child in instance.children.all():
|
|
|
|
child.parent = instance.parent
|
|
|
|
child.save()
|
|
|
|
|
2018-04-15 15:02:17 +00:00
|
|
|
|
2017-03-27 10:04:15 +00:00
|
|
|
class StockItem(models.Model):
|
2018-04-16 10:08:04 +00:00
|
|
|
"""
|
|
|
|
A 'StockItem' instance represents a quantity of physical instances of a part.
|
|
|
|
It may exist in a StockLocation, or as part of a sub-assembly installed into another StockItem
|
|
|
|
StockItems may be tracked using batch or serial numbers.
|
|
|
|
If a serial number is assigned, then StockItem cannot have a quantity other than 1
|
|
|
|
"""
|
2018-04-15 13:27:56 +00:00
|
|
|
|
2018-05-07 13:40:17 +00:00
|
|
|
def save(self, *args, **kwargs):
|
|
|
|
if not self.pk:
|
|
|
|
add_note = True
|
|
|
|
else:
|
|
|
|
add_note = False
|
|
|
|
|
|
|
|
super(StockItem, self).save(*args, **kwargs)
|
|
|
|
|
|
|
|
if add_note:
|
|
|
|
# This StockItem is being saved for the first time
|
|
|
|
self.add_transaction_note(
|
|
|
|
'Created stock item',
|
|
|
|
None,
|
|
|
|
system=True
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-04-27 14:06:39 +00:00
|
|
|
def clean(self):
|
|
|
|
|
|
|
|
# The 'supplier_part' field must point to the same part!
|
|
|
|
try:
|
|
|
|
if self.supplier_part is not None:
|
|
|
|
if not self.supplier_part.part == self.part:
|
2018-04-27 15:16:47 +00:00
|
|
|
raise ValidationError({'supplier_part': _("Part type ('{pf}') must be {pe}").format(
|
|
|
|
pf=str(self.supplier_part.part),
|
|
|
|
pe=str(self.part))
|
|
|
|
})
|
|
|
|
|
2018-04-29 07:06:22 +00:00
|
|
|
if self.part is not None:
|
|
|
|
if self.part.trackable and not self.serial:
|
|
|
|
raise ValidationError({
|
|
|
|
'serial': _('Serial number must be set for trackable items')
|
|
|
|
})
|
|
|
|
|
2018-04-27 14:06:39 +00:00
|
|
|
except Part.DoesNotExist:
|
|
|
|
# This gets thrown if self.supplier_part is null
|
|
|
|
# TODO - Find a test than can be perfomed...
|
|
|
|
pass
|
|
|
|
|
2018-05-07 08:16:05 +00:00
|
|
|
if self.belongs_to and self.belongs_to.pk == self.pk:
|
|
|
|
raise ValidationError({
|
|
|
|
'belongs_to': _('Item cannot belong to itself')
|
|
|
|
})
|
|
|
|
|
2018-04-29 03:20:02 +00:00
|
|
|
# Serial number cannot be set for items with quantity greater than 1
|
|
|
|
if not self.quantity == 1 and self.serial:
|
|
|
|
raise ValidationError({
|
|
|
|
'quantity': _("Quantity must be set to 1 for item with a serial number"),
|
|
|
|
'serial': _("Serial number cannot be set if quantity > 1")
|
|
|
|
})
|
|
|
|
|
2018-04-15 13:27:56 +00:00
|
|
|
def get_absolute_url(self):
|
|
|
|
return '/stock/item/{id}/'.format(id=self.id)
|
|
|
|
|
2018-04-16 13:09:45 +00:00
|
|
|
class Meta:
|
|
|
|
unique_together = [
|
|
|
|
('part', 'serial'),
|
|
|
|
]
|
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
# The 'master' copy of the part of which this stock item is an instance
|
2018-04-22 11:54:12 +00:00
|
|
|
part = models.ForeignKey('part.Part', on_delete=models.CASCADE, related_name='locations')
|
2018-04-14 05:26:42 +00:00
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
# The 'supplier part' used in this instance. May be null if no supplier parts are defined the master part
|
2018-04-22 11:54:12 +00:00
|
|
|
supplier_part = models.ForeignKey('part.SupplierPart', blank=True, null=True, on_delete=models.SET_NULL)
|
2018-04-14 05:26:42 +00:00
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
# Where the part is stored. If the part has been used to build another stock item, the location may not make sense
|
2018-04-14 10:33:53 +00:00
|
|
|
location = models.ForeignKey(StockLocation, on_delete=models.DO_NOTHING,
|
2018-04-16 13:09:45 +00:00
|
|
|
related_name='items', blank=True, null=True,
|
|
|
|
help_text='Where is this stock item located?')
|
2018-04-14 05:26:42 +00:00
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
# If this StockItem belongs to another StockItem (e.g. as part of a sub-assembly)
|
|
|
|
belongs_to = models.ForeignKey('self', on_delete=models.DO_NOTHING,
|
2018-04-16 13:09:45 +00:00
|
|
|
related_name='owned_parts', blank=True, null=True,
|
|
|
|
help_text='Is this item installed in another item?')
|
2018-04-16 10:08:04 +00:00
|
|
|
|
|
|
|
# The StockItem may be assigned to a particular customer
|
2018-04-18 23:01:07 +00:00
|
|
|
customer = models.ForeignKey('company.Company', on_delete=models.SET_NULL,
|
2018-04-18 22:31:31 +00:00
|
|
|
related_name='stockitems', blank=True, null=True,
|
|
|
|
help_text='Item assigned to customer?')
|
2018-04-16 10:08:04 +00:00
|
|
|
|
|
|
|
# Optional serial number
|
2018-04-16 13:09:45 +00:00
|
|
|
serial = models.PositiveIntegerField(blank=True, null=True,
|
|
|
|
help_text='Serial number for this item')
|
2018-04-16 10:08:04 +00:00
|
|
|
|
2018-04-16 11:07:57 +00:00
|
|
|
# Optional URL to link to external resource
|
|
|
|
URL = models.URLField(max_length=125, blank=True)
|
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
# Optional batch information
|
2018-04-16 13:09:45 +00:00
|
|
|
batch = models.CharField(max_length=100, blank=True,
|
|
|
|
help_text='Batch code for this stock item')
|
2018-04-16 10:08:04 +00:00
|
|
|
|
2018-04-16 14:32:02 +00:00
|
|
|
# If this part was produced by a build, point to that build here
|
2018-04-22 11:54:12 +00:00
|
|
|
# build = models.ForeignKey('build.Build', on_delete=models.SET_NULL, blank=True, null=True)
|
2018-04-16 14:32:02 +00:00
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
# Quantity of this stock item. Value may be overridden by other settings
|
2018-04-30 12:30:15 +00:00
|
|
|
quantity = models.PositiveIntegerField(validators=[MinValueValidator(0)], default=1)
|
2018-04-14 05:26:42 +00:00
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
# Last time this item was updated (set automagically)
|
2017-03-27 10:04:15 +00:00
|
|
|
updated = models.DateField(auto_now=True)
|
2017-03-29 11:55:28 +00:00
|
|
|
|
2017-03-29 12:36:06 +00:00
|
|
|
# last time the stock was checked / counted
|
2017-04-20 12:08:27 +00:00
|
|
|
stocktake_date = models.DateField(blank=True, null=True)
|
2018-04-14 05:26:42 +00:00
|
|
|
|
2017-04-20 12:40:59 +00:00
|
|
|
stocktake_user = models.ForeignKey(User, on_delete=models.SET_NULL, blank=True, null=True)
|
2017-03-29 12:36:06 +00:00
|
|
|
|
|
|
|
review_needed = models.BooleanField(default=False)
|
|
|
|
|
2018-04-12 06:27:26 +00:00
|
|
|
ITEM_OK = 10
|
2017-03-29 12:19:53 +00:00
|
|
|
ITEM_ATTENTION = 50
|
|
|
|
ITEM_DAMAGED = 55
|
|
|
|
ITEM_DESTROYED = 60
|
|
|
|
|
|
|
|
ITEM_STATUS_CODES = {
|
2018-04-12 06:27:26 +00:00
|
|
|
ITEM_OK: _("OK"),
|
2017-03-29 12:19:53 +00:00
|
|
|
ITEM_ATTENTION: _("Attention needed"),
|
|
|
|
ITEM_DAMAGED: _("Damaged"),
|
|
|
|
ITEM_DESTROYED: _("Destroyed")
|
|
|
|
}
|
2017-03-29 11:55:28 +00:00
|
|
|
|
|
|
|
status = models.PositiveIntegerField(
|
2018-04-12 06:27:26 +00:00
|
|
|
default=ITEM_OK,
|
2017-04-16 07:05:02 +00:00
|
|
|
choices=ITEM_STATUS_CODES.items(),
|
|
|
|
validators=[MinValueValidator(0)])
|
2017-03-29 11:55:28 +00:00
|
|
|
|
2018-04-17 15:16:30 +00:00
|
|
|
notes = models.TextField(blank=True)
|
2017-04-15 15:43:30 +00:00
|
|
|
|
2017-03-29 11:55:28 +00:00
|
|
|
# If stock item is incoming, an (optional) ETA field
|
2018-04-12 06:27:26 +00:00
|
|
|
# expected_arrival = models.DateField(null=True, blank=True)
|
2017-03-29 11:55:28 +00:00
|
|
|
|
2017-04-20 12:08:27 +00:00
|
|
|
infinite = models.BooleanField(default=False)
|
|
|
|
|
2018-05-06 12:38:39 +00:00
|
|
|
def can_delete(self):
|
|
|
|
# TODO - Return FALSE if this item cannot be deleted!
|
|
|
|
return True
|
|
|
|
|
2018-04-29 14:59:36 +00:00
|
|
|
@property
|
|
|
|
def in_stock(self):
|
|
|
|
if self.quantity == 0:
|
|
|
|
return False
|
|
|
|
|
|
|
|
if self.belongs_to or self.customer:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
2018-04-16 10:17:58 +00:00
|
|
|
@property
|
|
|
|
def has_tracking_info(self):
|
2018-04-16 13:26:02 +00:00
|
|
|
return self.tracking_info.count() > 0
|
2018-04-16 10:17:58 +00:00
|
|
|
|
2018-04-30 11:36:50 +00:00
|
|
|
def add_transaction_note(self, title, user, notes='', system=True):
|
|
|
|
track = StockItemTracking.objects.create(
|
|
|
|
item=self,
|
|
|
|
title=title,
|
|
|
|
user=user,
|
|
|
|
date=datetime.now().date(),
|
|
|
|
notes=notes,
|
|
|
|
system=system
|
|
|
|
)
|
|
|
|
|
|
|
|
track.save()
|
|
|
|
|
2018-05-06 11:39:33 +00:00
|
|
|
@transaction.atomic
|
|
|
|
def move(self, location, user):
|
|
|
|
|
|
|
|
if location == self.location:
|
|
|
|
return
|
|
|
|
|
|
|
|
note = "Moved to {loc}".format(loc=location.name)
|
|
|
|
|
|
|
|
self.location = location
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
self.add_transaction_note('Transfer',
|
|
|
|
user,
|
|
|
|
notes=note,
|
|
|
|
system=True)
|
|
|
|
|
|
|
|
|
2017-04-21 13:47:04 +00:00
|
|
|
@transaction.atomic
|
2018-05-08 12:06:28 +00:00
|
|
|
def stocktake(self, count, user, notes=''):
|
2017-04-20 12:08:27 +00:00
|
|
|
""" Perform item stocktake.
|
|
|
|
When the quantity of an item is counted,
|
|
|
|
record the date of stocktake
|
|
|
|
"""
|
|
|
|
|
|
|
|
count = int(count)
|
|
|
|
|
|
|
|
if count < 0 or self.infinite:
|
|
|
|
return
|
|
|
|
|
|
|
|
self.quantity = count
|
|
|
|
self.stocktake_date = datetime.now().date()
|
2017-04-20 12:40:59 +00:00
|
|
|
self.stocktake_user = user
|
2017-04-20 12:08:27 +00:00
|
|
|
self.save()
|
|
|
|
|
2018-05-08 12:06:28 +00:00
|
|
|
self.add_transaction_note('Stocktake - counted {n} items'.format(n=count),
|
2018-04-30 11:36:50 +00:00
|
|
|
user,
|
2018-05-08 12:06:28 +00:00
|
|
|
notes=notes,
|
2018-04-30 11:36:50 +00:00
|
|
|
system=True)
|
|
|
|
|
2017-04-21 13:47:04 +00:00
|
|
|
@transaction.atomic
|
2018-05-08 12:06:28 +00:00
|
|
|
def add_stock(self, quantity, user, notes=''):
|
2017-04-20 12:08:27 +00:00
|
|
|
""" Add items to stock
|
|
|
|
This function can be called by initiating a ProjectRun,
|
|
|
|
or by manually adding the items to the stock location
|
|
|
|
"""
|
|
|
|
|
2018-05-08 12:06:28 +00:00
|
|
|
quantity = int(quantity)
|
2017-04-20 12:08:27 +00:00
|
|
|
|
2018-05-08 12:06:28 +00:00
|
|
|
# Ignore amounts that do not make sense
|
|
|
|
if quantity <= 0 or self.infinite:
|
2017-04-20 12:08:27 +00:00
|
|
|
return
|
|
|
|
|
2018-05-08 12:06:28 +00:00
|
|
|
self.quantity += quantity
|
2017-04-20 12:08:27 +00:00
|
|
|
|
|
|
|
self.save()
|
|
|
|
|
2018-05-08 12:06:28 +00:00
|
|
|
self.add_transaction_note('Added {n} items to stock'.format(n=quantity),
|
|
|
|
user,
|
|
|
|
notes=notes,
|
|
|
|
system=True)
|
|
|
|
|
2017-04-21 13:47:04 +00:00
|
|
|
@transaction.atomic
|
2018-05-08 12:06:28 +00:00
|
|
|
def take_stock(self, quantity, user, notes=''):
|
|
|
|
""" Remove items from stock
|
|
|
|
"""
|
|
|
|
|
|
|
|
quantity = int(quantity)
|
|
|
|
|
|
|
|
if quantity <= 0 or self.infinite:
|
|
|
|
return
|
|
|
|
|
|
|
|
self.quantity -= quantity
|
|
|
|
|
|
|
|
if self.quantity < 0:
|
|
|
|
self.quantity = 0
|
|
|
|
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
self.add_transaction_note('Removed {n} items from stock'.format(n=quantity),
|
|
|
|
user,
|
|
|
|
notes=notes,
|
|
|
|
system=True)
|
2017-04-20 12:20:41 +00:00
|
|
|
|
2017-03-27 10:04:15 +00:00
|
|
|
def __str__(self):
|
2018-04-26 23:33:05 +00:00
|
|
|
s = '{n} x {part}'.format(
|
2018-04-27 15:16:47 +00:00
|
|
|
n=self.quantity,
|
2018-04-26 23:33:05 +00:00
|
|
|
part=self.part.name)
|
|
|
|
|
|
|
|
if self.location:
|
|
|
|
s += ' @ {loc}'.format(loc=self.location.name)
|
|
|
|
|
|
|
|
return s
|
2018-04-16 10:08:04 +00:00
|
|
|
|
2018-04-16 12:23:29 +00:00
|
|
|
@property
|
|
|
|
def is_trackable(self):
|
|
|
|
return self.part.trackable
|
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
|
|
|
|
class StockItemTracking(models.Model):
|
|
|
|
""" Stock tracking entry
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Stock item
|
|
|
|
item = models.ForeignKey(StockItem, on_delete=models.CASCADE,
|
|
|
|
related_name='tracking_info')
|
|
|
|
|
|
|
|
# Date this entry was created (cannot be edited)
|
|
|
|
date = models.DateField(auto_now_add=True, editable=False)
|
|
|
|
|
|
|
|
# Short-form title for this tracking entry
|
|
|
|
title = models.CharField(max_length=250)
|
|
|
|
|
|
|
|
# Optional longer description
|
2018-04-17 15:44:55 +00:00
|
|
|
notes = models.TextField(blank=True)
|
2018-04-16 10:08:04 +00:00
|
|
|
|
|
|
|
# Which user created this tracking entry?
|
|
|
|
user = models.ForeignKey(User, on_delete=models.SET_NULL, blank=True, null=True)
|
|
|
|
|
2018-04-17 22:44:08 +00:00
|
|
|
# Was this tracking note auto-generated by the system?
|
|
|
|
system = models.BooleanField(default=False)
|
|
|
|
|
2018-04-16 10:08:04 +00:00
|
|
|
# TODO
|
|
|
|
# image = models.ImageField(upload_to=func, max_length=255, null=True, blank=True)
|
|
|
|
|
|
|
|
# TODO
|
|
|
|
# file = models.FileField()
|