2019-04-27 10:35:14 +00:00
|
|
|
"""
|
2019-04-27 10:43:27 +00:00
|
|
|
Build database model definitions
|
2019-04-27 10:35:14 +00:00
|
|
|
"""
|
|
|
|
|
2018-04-16 14:32:02 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
2019-05-01 14:04:39 +00:00
|
|
|
from datetime import datetime
|
|
|
|
|
2019-05-02 21:58:46 +00:00
|
|
|
from django.contrib.auth.models import User
|
2018-04-17 06:58:37 +00:00
|
|
|
from django.utils.translation import ugettext as _
|
2019-04-29 12:56:40 +00:00
|
|
|
from django.core.exceptions import ValidationError
|
2018-04-17 06:58:37 +00:00
|
|
|
|
2019-04-24 13:42:50 +00:00
|
|
|
from django.urls import reverse
|
2019-05-01 12:21:13 +00:00
|
|
|
from django.db import models, transaction
|
2019-05-07 12:46:37 +00:00
|
|
|
from django.db.models import Sum
|
2020-04-25 13:17:07 +00:00
|
|
|
from django.db.models.functions import Coalesce
|
2018-04-16 14:32:02 +00:00
|
|
|
from django.core.validators import MinValueValidator
|
|
|
|
|
2020-02-01 13:00:19 +00:00
|
|
|
from markdownx.models import MarkdownxField
|
|
|
|
|
2020-04-25 05:25:17 +00:00
|
|
|
from mptt.models import MPTTModel, TreeForeignKey
|
|
|
|
|
2020-08-07 23:16:53 +00:00
|
|
|
from InvenTree.status_codes import BuildStatus
|
2020-10-19 21:41:08 +00:00
|
|
|
from InvenTree.helpers import increment, getSetting
|
2020-10-19 13:13:43 +00:00
|
|
|
from InvenTree.validators import validate_build_order_reference
|
|
|
|
|
2020-09-17 12:44:17 +00:00
|
|
|
import InvenTree.fields
|
2019-06-04 13:38:52 +00:00
|
|
|
|
2020-04-28 00:35:19 +00:00
|
|
|
from stock import models as StockModels
|
|
|
|
from part import models as PartModels
|
2019-04-30 22:08:50 +00:00
|
|
|
|
2018-04-17 14:03:42 +00:00
|
|
|
|
2020-04-25 05:25:17 +00:00
|
|
|
class Build(MPTTModel):
|
2020-10-20 09:42:29 +00:00
|
|
|
""" A Build object organises the creation of new StockItem objects from other existing StockItem objects.
|
2019-04-29 12:19:13 +00:00
|
|
|
|
|
|
|
Attributes:
|
|
|
|
part: The part to be built (from component BOM items)
|
2020-10-19 11:36:14 +00:00
|
|
|
reference: Build order reference (required, must be unique)
|
2019-04-29 12:19:13 +00:00
|
|
|
title: Brief title describing the build (required)
|
|
|
|
quantity: Number of units to be built
|
2020-04-25 05:25:17 +00:00
|
|
|
parent: Reference to a Build object for which this Build is required
|
2020-04-25 03:15:45 +00:00
|
|
|
sales_order: References to a SalesOrder object for which this Build is required (e.g. the output of this build will be used to fulfil a sales order)
|
2019-05-10 09:03:10 +00:00
|
|
|
take_from: Location to take stock from to make this build (if blank, can take from anywhere)
|
2019-04-29 12:19:13 +00:00
|
|
|
status: Build status code
|
|
|
|
batch: Batch code transferred to build parts (optional)
|
|
|
|
creation_date: Date the build was created (auto)
|
|
|
|
completion_date: Date the build was completed
|
2020-04-06 01:28:35 +00:00
|
|
|
link: External URL for extra information
|
2019-04-29 12:19:13 +00:00
|
|
|
notes: Text notes
|
2018-04-16 14:32:02 +00:00
|
|
|
"""
|
2019-05-05 21:58:20 +00:00
|
|
|
|
2020-10-19 11:40:19 +00:00
|
|
|
class Meta:
|
|
|
|
verbose_name = _("Build Order")
|
|
|
|
verbose_name_plural = _("Build Orders")
|
|
|
|
|
2019-04-29 12:59:42 +00:00
|
|
|
def __str__(self):
|
2020-10-19 12:22:09 +00:00
|
|
|
|
2020-10-19 21:41:08 +00:00
|
|
|
prefix = getSetting("BUILDORDER_REFERENCE_PREFIX")
|
|
|
|
|
|
|
|
return f"{prefix}{self.reference}"
|
2019-04-29 12:59:42 +00:00
|
|
|
|
2018-04-17 13:13:41 +00:00
|
|
|
def get_absolute_url(self):
|
2019-04-24 13:42:50 +00:00
|
|
|
return reverse('build-detail', kwargs={'pk': self.id})
|
2018-04-17 13:13:41 +00:00
|
|
|
|
2020-05-16 07:29:41 +00:00
|
|
|
def clean(self):
|
|
|
|
"""
|
|
|
|
Validation for Build object.
|
|
|
|
"""
|
|
|
|
|
|
|
|
super().clean()
|
|
|
|
|
2020-10-20 09:42:29 +00:00
|
|
|
# Build quantity must be an integer
|
|
|
|
# Maybe in the future this will be adjusted?
|
2020-05-16 07:29:41 +00:00
|
|
|
|
2020-10-20 09:42:29 +00:00
|
|
|
if not self.quantity == int(self.quantity):
|
|
|
|
raise ValidationError({
|
|
|
|
'quantity': _("Build quantity must be integer value for trackable parts")
|
|
|
|
})
|
|
|
|
|
2020-10-19 11:36:14 +00:00
|
|
|
reference = models.CharField(
|
|
|
|
unique=True,
|
|
|
|
max_length=64,
|
|
|
|
blank=False,
|
|
|
|
help_text=_('Build Order Reference'),
|
|
|
|
verbose_name=_('Reference'),
|
2020-10-19 13:13:43 +00:00
|
|
|
validators=[
|
|
|
|
validate_build_order_reference
|
|
|
|
]
|
2020-10-19 11:36:14 +00:00
|
|
|
)
|
|
|
|
|
2019-05-10 09:03:10 +00:00
|
|
|
title = models.CharField(
|
2020-10-19 13:26:26 +00:00
|
|
|
verbose_name=_('Description'),
|
2019-05-10 09:03:10 +00:00
|
|
|
blank=False,
|
|
|
|
max_length=100,
|
2020-04-25 03:15:45 +00:00
|
|
|
help_text=_('Brief description of the build')
|
|
|
|
)
|
|
|
|
|
2020-04-25 05:25:17 +00:00
|
|
|
parent = TreeForeignKey(
|
|
|
|
'self',
|
|
|
|
on_delete=models.DO_NOTHING,
|
|
|
|
blank=True, null=True,
|
2020-04-25 13:17:07 +00:00
|
|
|
related_name='children',
|
|
|
|
verbose_name=_('Parent Build'),
|
|
|
|
help_text=_('Parent build to which this build is allocated'),
|
2020-04-25 05:25:17 +00:00
|
|
|
)
|
|
|
|
|
2020-04-25 03:15:45 +00:00
|
|
|
part = models.ForeignKey(
|
|
|
|
'part.Part',
|
2020-04-25 13:17:07 +00:00
|
|
|
verbose_name=_('Part'),
|
2020-04-25 03:15:45 +00:00
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='builds',
|
|
|
|
limit_choices_to={
|
|
|
|
'is_template': False,
|
|
|
|
'assembly': True,
|
|
|
|
'active': True,
|
|
|
|
'virtual': False,
|
|
|
|
},
|
|
|
|
help_text=_('Select part to build'),
|
|
|
|
)
|
|
|
|
|
|
|
|
sales_order = models.ForeignKey(
|
|
|
|
'order.SalesOrder',
|
2020-04-25 13:17:07 +00:00
|
|
|
verbose_name=_('Sales Order Reference'),
|
2020-04-25 03:15:45 +00:00
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
related_name='builds',
|
|
|
|
null=True, blank=True,
|
|
|
|
help_text=_('SalesOrder to which this build is allocated')
|
|
|
|
)
|
2019-04-27 10:35:14 +00:00
|
|
|
|
2020-04-25 03:15:45 +00:00
|
|
|
take_from = models.ForeignKey(
|
|
|
|
'stock.StockLocation',
|
2020-04-25 13:17:07 +00:00
|
|
|
verbose_name=_('Source Location'),
|
2020-04-25 03:15:45 +00:00
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
related_name='sourcing_builds',
|
|
|
|
null=True, blank=True,
|
|
|
|
help_text=_('Select location to take stock from for this build (leave blank to take from any stock location)')
|
|
|
|
)
|
2019-04-27 10:35:14 +00:00
|
|
|
|
2019-04-29 12:19:13 +00:00
|
|
|
quantity = models.PositiveIntegerField(
|
2020-04-25 13:17:07 +00:00
|
|
|
verbose_name=_('Build Quantity'),
|
2019-04-29 12:19:13 +00:00
|
|
|
default=1,
|
|
|
|
validators=[MinValueValidator(1)],
|
2019-11-18 23:22:46 +00:00
|
|
|
help_text=_('Number of parts to build')
|
2019-04-29 12:19:13 +00:00
|
|
|
)
|
2019-06-04 13:38:52 +00:00
|
|
|
|
2020-04-25 13:17:07 +00:00
|
|
|
status = models.PositiveIntegerField(
|
|
|
|
verbose_name=_('Build Status'),
|
|
|
|
default=BuildStatus.PENDING,
|
|
|
|
choices=BuildStatus.items(),
|
|
|
|
validators=[MinValueValidator(0)],
|
|
|
|
help_text=_('Build status code')
|
|
|
|
)
|
2019-04-29 12:19:13 +00:00
|
|
|
|
2020-04-25 13:17:07 +00:00
|
|
|
batch = models.CharField(
|
|
|
|
verbose_name=_('Batch Code'),
|
|
|
|
max_length=100,
|
|
|
|
blank=True,
|
|
|
|
null=True,
|
|
|
|
help_text=_('Batch code for this build output')
|
|
|
|
)
|
2019-04-29 12:19:13 +00:00
|
|
|
|
2020-02-10 10:34:41 +00:00
|
|
|
creation_date = models.DateField(auto_now_add=True, editable=False)
|
2019-04-29 12:19:13 +00:00
|
|
|
|
2019-04-27 10:35:14 +00:00
|
|
|
completion_date = models.DateField(null=True, blank=True)
|
2019-05-02 21:58:46 +00:00
|
|
|
|
2020-04-25 13:17:07 +00:00
|
|
|
completed_by = models.ForeignKey(
|
|
|
|
User,
|
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
blank=True, null=True,
|
|
|
|
related_name='builds_completed'
|
|
|
|
)
|
2019-04-29 12:19:13 +00:00
|
|
|
|
2020-09-17 12:44:17 +00:00
|
|
|
link = InvenTree.fields.InvenTreeURLField(
|
2020-04-25 13:17:07 +00:00
|
|
|
verbose_name=_('External Link'),
|
|
|
|
blank=True, help_text=_('Link to external URL')
|
|
|
|
)
|
2018-04-17 10:25:43 +00:00
|
|
|
|
2020-04-25 13:17:07 +00:00
|
|
|
notes = MarkdownxField(
|
|
|
|
verbose_name=_('Notes'),
|
|
|
|
blank=True, help_text=_('Extra build notes')
|
|
|
|
)
|
2018-04-17 10:25:43 +00:00
|
|
|
|
2020-02-17 12:31:23 +00:00
|
|
|
@property
|
|
|
|
def output_count(self):
|
|
|
|
return self.build_outputs.count()
|
|
|
|
|
2020-10-19 12:22:09 +00:00
|
|
|
@classmethod
|
|
|
|
def getNextBuildNumber(cls):
|
|
|
|
"""
|
|
|
|
Try to predict the next Build Order reference:
|
|
|
|
"""
|
|
|
|
|
|
|
|
if cls.objects.count() == 0:
|
|
|
|
return None
|
|
|
|
|
|
|
|
build = cls.objects.last()
|
|
|
|
ref = build.reference
|
|
|
|
|
|
|
|
if not ref:
|
|
|
|
return None
|
|
|
|
|
|
|
|
tries = set()
|
|
|
|
|
|
|
|
while 1:
|
|
|
|
new_ref = increment(ref)
|
|
|
|
|
|
|
|
if new_ref in tries:
|
|
|
|
# We are potentially stuck in a loop - simply return the original reference
|
|
|
|
return ref
|
|
|
|
|
|
|
|
if cls.objects.filter(reference=new_ref).exists():
|
|
|
|
tries.add(new_ref)
|
|
|
|
new_ref = increment(new_ref)
|
|
|
|
else:
|
|
|
|
break
|
|
|
|
|
|
|
|
return new_ref
|
|
|
|
|
2019-05-01 12:21:13 +00:00
|
|
|
@transaction.atomic
|
2019-05-02 21:58:46 +00:00
|
|
|
def cancelBuild(self, user):
|
2019-04-30 10:39:01 +00:00
|
|
|
""" Mark the Build as CANCELLED
|
|
|
|
|
|
|
|
- Delete any pending BuildItem objects (but do not remove items from stock)
|
2019-05-01 12:21:13 +00:00
|
|
|
- Set build status to CANCELLED
|
|
|
|
- Save the Build object
|
2019-04-30 10:39:01 +00:00
|
|
|
"""
|
2019-05-01 12:21:13 +00:00
|
|
|
|
2019-05-01 14:04:39 +00:00
|
|
|
for item in self.allocated_stock.all():
|
2019-05-01 12:21:13 +00:00
|
|
|
item.delete()
|
|
|
|
|
2019-05-02 11:23:53 +00:00
|
|
|
# Date of 'completion' is the date the build was cancelled
|
|
|
|
self.completion_date = datetime.now().date()
|
2019-05-02 21:58:46 +00:00
|
|
|
self.completed_by = user
|
2019-05-02 11:23:53 +00:00
|
|
|
|
2019-06-17 13:02:44 +00:00
|
|
|
self.status = BuildStatus.CANCELLED
|
2019-05-01 12:21:13 +00:00
|
|
|
self.save()
|
2019-04-30 10:39:01 +00:00
|
|
|
|
2019-05-05 21:55:39 +00:00
|
|
|
def getAutoAllocations(self):
|
|
|
|
""" Return a list of parts which will be allocated
|
|
|
|
using the 'AutoAllocate' function.
|
|
|
|
|
|
|
|
For each item in the BOM for the attached Part:
|
|
|
|
|
|
|
|
- If there is a single StockItem, use that StockItem
|
|
|
|
- Take as many parts as available (up to the quantity required for the BOM)
|
|
|
|
- If there are multiple StockItems available, ignore (leave up to the user)
|
|
|
|
|
|
|
|
Returns:
|
2019-05-09 13:55:30 +00:00
|
|
|
A list object containing the StockItem objects to be allocated (and the quantities)
|
2019-05-05 21:55:39 +00:00
|
|
|
"""
|
|
|
|
|
2019-05-09 13:55:30 +00:00
|
|
|
allocations = []
|
2019-05-05 21:55:39 +00:00
|
|
|
|
2019-05-20 14:16:00 +00:00
|
|
|
for item in self.part.bom_items.all().prefetch_related('sub_part'):
|
2019-05-05 21:55:39 +00:00
|
|
|
|
|
|
|
# How many parts required for this build?
|
|
|
|
q_required = item.quantity * self.quantity
|
|
|
|
|
2020-04-27 22:44:10 +00:00
|
|
|
# Grab a list of StockItem objects which are "in stock"
|
2020-04-28 00:43:46 +00:00
|
|
|
stock = StockModels.StockItem.objects.filter(StockModels.StockItem.IN_STOCK_FILTER)
|
2020-04-27 22:44:10 +00:00
|
|
|
|
|
|
|
# Filter by part reference
|
|
|
|
stock = stock.filter(part=item.sub_part)
|
2019-05-05 21:55:39 +00:00
|
|
|
|
2019-05-10 09:03:45 +00:00
|
|
|
# Ensure that the available stock items are in the correct location
|
|
|
|
if self.take_from is not None:
|
|
|
|
# Filter for stock that is located downstream of the designated location
|
2019-05-10 09:12:56 +00:00
|
|
|
stock = stock.filter(location__in=[loc for loc in self.take_from.getUniqueChildren()])
|
2019-05-10 09:03:45 +00:00
|
|
|
|
2019-05-05 21:55:39 +00:00
|
|
|
# Only one StockItem to choose from? Default to that one!
|
|
|
|
if len(stock) == 1:
|
|
|
|
stock_item = stock[0]
|
|
|
|
|
2019-05-05 21:58:20 +00:00
|
|
|
# Check that we have not already allocated this stock-item against this build
|
|
|
|
build_items = BuildItem.objects.filter(build=self, stock_item=stock_item)
|
|
|
|
|
|
|
|
if len(build_items) > 0:
|
|
|
|
continue
|
|
|
|
|
2019-05-05 21:55:39 +00:00
|
|
|
# Are there any parts available?
|
|
|
|
if stock_item.quantity > 0:
|
2019-05-09 13:55:30 +00:00
|
|
|
|
2019-05-05 21:55:39 +00:00
|
|
|
# Only take as many as are available
|
|
|
|
if stock_item.quantity < q_required:
|
|
|
|
q_required = stock_item.quantity
|
|
|
|
|
2019-05-09 13:55:30 +00:00
|
|
|
allocation = {
|
|
|
|
'stock_item': stock_item,
|
|
|
|
'quantity': q_required,
|
|
|
|
}
|
|
|
|
|
|
|
|
allocations.append(allocation)
|
2019-05-05 21:55:39 +00:00
|
|
|
|
|
|
|
return allocations
|
|
|
|
|
2019-05-09 22:33:54 +00:00
|
|
|
@transaction.atomic
|
|
|
|
def unallocateStock(self):
|
|
|
|
""" Deletes all stock allocations for this build. """
|
|
|
|
|
|
|
|
BuildItem.objects.filter(build=self.id).delete()
|
|
|
|
|
2019-05-05 21:55:39 +00:00
|
|
|
@transaction.atomic
|
|
|
|
def autoAllocate(self):
|
|
|
|
""" Run auto-allocation routine to allocate StockItems to this Build.
|
|
|
|
|
2019-05-09 13:55:30 +00:00
|
|
|
Returns a list of dict objects with keys like:
|
|
|
|
|
|
|
|
{
|
|
|
|
'stock_item': item,
|
|
|
|
'quantity': quantity,
|
|
|
|
}
|
|
|
|
|
2019-05-05 21:55:39 +00:00
|
|
|
See: getAutoAllocations()
|
|
|
|
"""
|
|
|
|
|
|
|
|
allocations = self.getAutoAllocations()
|
|
|
|
|
|
|
|
for item in allocations:
|
|
|
|
# Create a new allocation
|
|
|
|
build_item = BuildItem(
|
|
|
|
build=self,
|
2019-05-09 13:55:30 +00:00
|
|
|
stock_item=item['stock_item'],
|
|
|
|
quantity=item['quantity'])
|
2019-05-05 21:55:39 +00:00
|
|
|
|
|
|
|
build_item.save()
|
|
|
|
|
2019-05-01 12:21:13 +00:00
|
|
|
@transaction.atomic
|
2019-07-22 05:55:36 +00:00
|
|
|
def completeBuild(self, location, serial_numbers, user):
|
2019-04-30 10:39:01 +00:00
|
|
|
""" Mark the Build as COMPLETE
|
|
|
|
|
|
|
|
- Takes allocated items from stock
|
|
|
|
- Delete pending BuildItem objects
|
|
|
|
"""
|
|
|
|
|
2020-04-25 14:32:09 +00:00
|
|
|
# Complete the build allocation for each BuildItem
|
|
|
|
for build_item in self.allocated_stock.all().prefetch_related('stock_item'):
|
|
|
|
build_item.complete_allocation(user)
|
2019-05-02 21:58:46 +00:00
|
|
|
|
2020-04-27 10:46:34 +00:00
|
|
|
# Check that the stock-item has been assigned to this build, and remove the builditem from the database
|
|
|
|
if build_item.stock_item.build_order == self:
|
|
|
|
build_item.delete()
|
2020-04-26 06:44:35 +00:00
|
|
|
|
2019-07-22 05:55:36 +00:00
|
|
|
notes = 'Built {q} on {now}'.format(
|
|
|
|
q=self.quantity,
|
|
|
|
now=str(datetime.now().date())
|
2019-05-01 14:04:39 +00:00
|
|
|
)
|
|
|
|
|
2020-04-25 14:32:09 +00:00
|
|
|
# Generate the build outputs
|
2020-02-17 12:31:23 +00:00
|
|
|
if self.part.trackable and serial_numbers:
|
2019-07-22 05:55:36 +00:00
|
|
|
# Add new serial numbers
|
|
|
|
for serial in serial_numbers:
|
2020-04-28 00:43:46 +00:00
|
|
|
item = StockModels.StockItem.objects.create(
|
2019-07-22 05:55:36 +00:00
|
|
|
part=self.part,
|
2019-09-01 13:18:28 +00:00
|
|
|
build=self,
|
2019-07-22 05:55:36 +00:00
|
|
|
location=location,
|
|
|
|
quantity=1,
|
|
|
|
serial=serial,
|
|
|
|
batch=str(self.batch) if self.batch else '',
|
|
|
|
notes=notes
|
|
|
|
)
|
|
|
|
|
|
|
|
item.save()
|
|
|
|
|
|
|
|
else:
|
|
|
|
# Add stock of the newly created item
|
2020-04-28 00:43:46 +00:00
|
|
|
item = StockModels.StockItem.objects.create(
|
2019-07-22 05:55:36 +00:00
|
|
|
part=self.part,
|
2019-09-01 13:18:28 +00:00
|
|
|
build=self,
|
2019-07-22 05:55:36 +00:00
|
|
|
location=location,
|
|
|
|
quantity=self.quantity,
|
|
|
|
batch=str(self.batch) if self.batch else '',
|
|
|
|
notes=notes
|
|
|
|
)
|
|
|
|
|
|
|
|
item.save()
|
2019-05-01 14:04:39 +00:00
|
|
|
|
|
|
|
# Finally, mark the build as complete
|
2020-04-25 14:32:09 +00:00
|
|
|
self.completion_date = datetime.now().date()
|
|
|
|
self.completed_by = user
|
2019-06-06 00:43:34 +00:00
|
|
|
self.status = BuildStatus.COMPLETE
|
2019-05-01 14:04:39 +00:00
|
|
|
self.save()
|
2019-04-30 10:39:01 +00:00
|
|
|
|
2020-04-25 14:32:09 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-25 13:44:03 +00:00
|
|
|
def isFullyAllocated(self):
|
|
|
|
"""
|
|
|
|
Return True if this build has been fully allocated.
|
|
|
|
"""
|
|
|
|
|
|
|
|
bom_items = self.part.bom_items.all()
|
|
|
|
|
|
|
|
for item in bom_items:
|
|
|
|
part = item.sub_part
|
|
|
|
|
|
|
|
if not self.isPartFullyAllocated(part):
|
|
|
|
return False
|
|
|
|
|
2020-04-27 10:16:41 +00:00
|
|
|
return True
|
2020-04-25 13:44:03 +00:00
|
|
|
|
|
|
|
def isPartFullyAllocated(self, part):
|
|
|
|
"""
|
|
|
|
Check if a given Part is fully allocated for this Build
|
|
|
|
"""
|
|
|
|
|
|
|
|
return self.getAllocatedQuantity(part) >= self.getRequiredQuantity(part)
|
|
|
|
|
2019-05-07 13:03:05 +00:00
|
|
|
def getRequiredQuantity(self, part):
|
|
|
|
""" Calculate the quantity of <part> required to make this build.
|
|
|
|
"""
|
|
|
|
|
|
|
|
try:
|
2020-04-28 00:43:46 +00:00
|
|
|
item = PartModels.BomItem.objects.get(part=self.part.id, sub_part=part.id)
|
2020-04-25 13:17:07 +00:00
|
|
|
q = item.quantity
|
2020-04-28 00:43:46 +00:00
|
|
|
except PartModels.BomItem.DoesNotExist:
|
2020-04-25 13:17:07 +00:00
|
|
|
q = 0
|
|
|
|
|
|
|
|
return q * self.quantity
|
2019-05-07 13:03:05 +00:00
|
|
|
|
2019-05-07 12:46:37 +00:00
|
|
|
def getAllocatedQuantity(self, part):
|
|
|
|
""" Calculate the total number of <part> currently allocated to this build
|
|
|
|
"""
|
|
|
|
|
2020-04-25 13:17:07 +00:00
|
|
|
allocated = BuildItem.objects.filter(build=self.id, stock_item__part=part.id).aggregate(q=Coalesce(Sum('quantity'), 0))
|
2019-05-07 12:46:37 +00:00
|
|
|
|
2020-04-25 13:17:07 +00:00
|
|
|
return allocated['q']
|
2019-05-07 12:46:37 +00:00
|
|
|
|
|
|
|
def getUnallocatedQuantity(self, part):
|
|
|
|
""" Calculate the quantity of <part> which still needs to be allocated to this build.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
Part - the part to be tested
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
The remaining allocated quantity
|
|
|
|
"""
|
|
|
|
|
2019-05-07 13:03:05 +00:00
|
|
|
return max(self.getRequiredQuantity(part) - self.getAllocatedQuantity(part), 0)
|
2019-05-07 12:46:37 +00:00
|
|
|
|
2018-04-30 22:55:51 +00:00
|
|
|
@property
|
|
|
|
def required_parts(self):
|
2019-04-27 10:35:14 +00:00
|
|
|
""" Returns a dict of parts required to build this part (BOM) """
|
2018-04-30 22:55:51 +00:00
|
|
|
parts = []
|
|
|
|
|
2019-05-20 14:16:00 +00:00
|
|
|
for item in self.part.bom_items.all().prefetch_related('sub_part'):
|
2020-04-27 11:28:44 +00:00
|
|
|
part = {
|
|
|
|
'part': item.sub_part,
|
|
|
|
'per_build': item.quantity,
|
|
|
|
'quantity': item.quantity * self.quantity,
|
|
|
|
'allocated': self.getAllocatedQuantity(item.sub_part)
|
|
|
|
}
|
2018-04-30 22:55:51 +00:00
|
|
|
|
|
|
|
parts.append(part)
|
|
|
|
|
|
|
|
return parts
|
|
|
|
|
2018-04-30 23:00:09 +00:00
|
|
|
@property
|
|
|
|
def can_build(self):
|
2019-04-27 10:35:14 +00:00
|
|
|
""" Return true if there are enough parts to supply build """
|
2018-04-30 23:00:09 +00:00
|
|
|
|
|
|
|
for item in self.required_parts:
|
|
|
|
if item['part'].total_stock < item['quantity']:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
2018-04-17 10:25:43 +00:00
|
|
|
@property
|
|
|
|
def is_active(self):
|
2019-04-27 10:35:14 +00:00
|
|
|
""" Is this build active? An active build is either:
|
|
|
|
|
|
|
|
- PENDING
|
|
|
|
- HOLDING
|
2018-04-17 10:25:43 +00:00
|
|
|
"""
|
|
|
|
|
2019-06-04 13:38:52 +00:00
|
|
|
return self.status in BuildStatus.ACTIVE_CODES
|
2018-04-17 10:25:43 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_complete(self):
|
2019-04-27 10:35:14 +00:00
|
|
|
""" Returns True if the build status is COMPLETE """
|
2019-06-04 14:00:05 +00:00
|
|
|
return self.status == BuildStatus.COMPLETE
|
2019-04-29 12:19:13 +00:00
|
|
|
|
|
|
|
|
2019-04-29 12:30:21 +00:00
|
|
|
class BuildItem(models.Model):
|
|
|
|
""" A BuildItem links multiple StockItem objects to a Build.
|
2019-04-29 12:19:13 +00:00
|
|
|
These are used to allocate part stock to a build.
|
|
|
|
Once the Build is completed, the parts are removed from stock and the
|
|
|
|
BuildItemAllocation objects are removed.
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
build: Link to a Build object
|
2019-05-10 10:11:52 +00:00
|
|
|
stock_item: Link to a StockItem object
|
2019-04-29 12:19:13 +00:00
|
|
|
quantity: Number of units allocated
|
|
|
|
"""
|
|
|
|
|
2019-04-30 05:48:07 +00:00
|
|
|
def get_absolute_url(self):
|
|
|
|
# TODO - Fix!
|
|
|
|
return '/build/item/{pk}/'.format(pk=self.id)
|
2019-04-30 09:17:54 +00:00
|
|
|
# return reverse('build-detail', kwargs={'pk': self.id})
|
2019-04-30 05:48:07 +00:00
|
|
|
|
2019-04-29 12:56:40 +00:00
|
|
|
class Meta:
|
|
|
|
unique_together = [
|
|
|
|
('build', 'stock_item'),
|
|
|
|
]
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
""" Check validity of the BuildItem model.
|
|
|
|
The following checks are performed:
|
|
|
|
|
|
|
|
- StockItem.part must be in the BOM of the Part object referenced by Build
|
2019-04-30 06:35:55 +00:00
|
|
|
- Allocation quantity cannot exceed available quantity
|
2019-04-29 12:56:40 +00:00
|
|
|
"""
|
2019-04-30 06:35:55 +00:00
|
|
|
|
|
|
|
super(BuildItem, self).clean()
|
|
|
|
|
|
|
|
errors = {}
|
|
|
|
|
2019-05-25 12:01:30 +00:00
|
|
|
try:
|
2019-04-30 06:35:55 +00:00
|
|
|
if self.stock_item.part not in self.build.part.required_parts():
|
2019-05-12 02:16:04 +00:00
|
|
|
errors['stock_item'] = [_("Selected stock item not found in BOM for part '{p}'".format(p=self.build.part.full_name))]
|
2019-04-30 06:35:55 +00:00
|
|
|
|
2019-05-25 12:01:30 +00:00
|
|
|
if self.quantity > self.stock_item.quantity:
|
|
|
|
errors['quantity'] = [_("Allocated quantity ({n}) must not exceed available quantity ({q})".format(
|
|
|
|
n=self.quantity,
|
|
|
|
q=self.stock_item.quantity
|
|
|
|
))]
|
|
|
|
|
2020-04-27 10:05:02 +00:00
|
|
|
if self.stock_item.quantity - self.stock_item.allocation_count() + self.quantity < self.quantity:
|
2020-04-22 03:11:19 +00:00
|
|
|
errors['quantity'] = _('StockItem is over-allocated')
|
|
|
|
|
|
|
|
if self.quantity <= 0:
|
|
|
|
errors['quantity'] = _('Allocation quantity must be greater than zero')
|
|
|
|
|
|
|
|
if self.stock_item.serial and not self.quantity == 1:
|
|
|
|
errors['quantity'] = _('Quantity must be 1 for serialized stock')
|
|
|
|
|
2020-04-28 00:35:19 +00:00
|
|
|
except (StockModels.StockItem.DoesNotExist, PartModels.Part.DoesNotExist):
|
2019-05-25 12:01:30 +00:00
|
|
|
pass
|
2019-04-30 06:35:55 +00:00
|
|
|
|
|
|
|
if len(errors) > 0:
|
|
|
|
raise ValidationError(errors)
|
2019-04-29 12:56:40 +00:00
|
|
|
|
2020-04-25 14:32:09 +00:00
|
|
|
def complete_allocation(self, user):
|
|
|
|
|
|
|
|
item = self.stock_item
|
|
|
|
|
|
|
|
# Split the allocated stock if there are more available than allocated
|
|
|
|
if item.quantity > self.quantity:
|
|
|
|
item = item.splitStock(self.quantity, None, user)
|
|
|
|
|
|
|
|
# Update our own reference to the new item
|
|
|
|
self.stock_item = item
|
|
|
|
self.save()
|
|
|
|
|
2020-04-26 22:58:18 +00:00
|
|
|
# TODO - If the item__part object is not trackable, delete the stock item here
|
|
|
|
|
2020-04-25 14:32:09 +00:00
|
|
|
item.build_order = self.build
|
|
|
|
item.save()
|
|
|
|
|
2019-04-29 12:19:13 +00:00
|
|
|
build = models.ForeignKey(
|
|
|
|
Build,
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='allocated_stock',
|
2019-11-18 23:22:46 +00:00
|
|
|
help_text=_('Build to allocate parts')
|
2019-04-29 12:19:13 +00:00
|
|
|
)
|
|
|
|
|
2019-04-29 12:33:39 +00:00
|
|
|
stock_item = models.ForeignKey(
|
2019-04-29 12:19:13 +00:00
|
|
|
'stock.StockItem',
|
|
|
|
on_delete=models.CASCADE,
|
|
|
|
related_name='allocations',
|
2020-10-20 09:37:57 +00:00
|
|
|
help_text=_('Source stock item'),
|
2020-04-26 06:38:29 +00:00
|
|
|
limit_choices_to={
|
|
|
|
'build_order': None,
|
|
|
|
'sales_order': None,
|
|
|
|
'belongs_to': None,
|
|
|
|
}
|
2019-04-29 12:19:13 +00:00
|
|
|
)
|
|
|
|
|
2019-11-18 23:22:46 +00:00
|
|
|
quantity = models.DecimalField(
|
|
|
|
decimal_places=5,
|
|
|
|
max_digits=15,
|
2019-04-29 12:19:13 +00:00
|
|
|
default=1,
|
2020-04-25 13:59:28 +00:00
|
|
|
validators=[MinValueValidator(0)],
|
2019-11-18 23:22:46 +00:00
|
|
|
help_text=_('Stock quantity to allocate to build')
|
2019-04-29 12:19:13 +00:00
|
|
|
)
|
2020-10-20 09:37:57 +00:00
|
|
|
|
|
|
|
install_into = models.ForeignKey(
|
|
|
|
'stock.StockItem',
|
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
blank=True, null=True,
|
|
|
|
related_name='items_to_install',
|
|
|
|
help_text=_('Destination stock item'),
|
|
|
|
limit_choices_to={
|
|
|
|
'is_building': True,
|
|
|
|
}
|
|
|
|
)
|