InvenTree/InvenTree/part/models.py

236 lines
6.5 KiB
Python
Raw Normal View History

from __future__ import unicode_literals
2017-03-29 12:19:53 +00:00
from django.utils.translation import ugettext as _
from django.db import models
from django.db.models import Sum
2017-04-16 07:05:02 +00:00
from django.core.validators import MinValueValidator
from InvenTree.models import InvenTreeTree
import os
from django.db.models.signals import pre_delete
from django.dispatch import receiver
2017-03-28 12:17:56 +00:00
2017-03-28 12:31:41 +00:00
class PartCategory(InvenTreeTree):
2017-03-27 11:55:21 +00:00
""" PartCategory provides hierarchical organization of Part objects.
"""
2017-03-29 12:19:53 +00:00
def get_absolute_url(self):
return '/part/category/{id}/'.format(id=self.id)
class Meta:
verbose_name = "Part Category"
verbose_name_plural = "Part Categories"
2017-03-29 12:19:53 +00:00
@property
def partcount(self):
""" Return the total part count under this category
(including children of child categories)
"""
count = self.parts.count()
for child in self.children.all():
count += child.partcount
return count
"""
2017-04-10 23:41:03 +00:00
@property
def parts(self):
2017-04-11 13:07:02 +00:00
return self.part_set.all()
"""
@receiver(pre_delete, sender=PartCategory, dispatch_uid='partcategory_delete_log')
def before_delete_part_category(sender, instance, using, **kwargs):
# Update each part in this category to point to the parent category
for part in instance.parts.all():
part.category = instance.parent
part.save()
2017-04-10 23:41:03 +00:00
# Update each child category
for child in instance.children.all():
child.parent = instance.parent
child.save()
# Function to automatically rename a part image on upload
# Format: part_pk.<img>
def rename_part_image(instance, filename):
base = 'part_images'
if filename.count('.') > 0:
ext = filename.split('.')[-1]
else:
ext = ''
fn = 'part_{pk}_img'.format(pk=instance.pk)
if ext:
fn += '.' + ext
return os.path.join(base, fn)
2017-03-29 12:19:53 +00:00
class Part(models.Model):
""" Represents an abstract part
Parts can be "stocked" in multiple warehouses,
and can be combined to form other parts
"""
2017-03-29 12:19:53 +00:00
def get_absolute_url(self):
return '/part/{id}/'.format(id=self.id)
2017-04-01 02:31:48 +00:00
# Short name of the part
name = models.CharField(max_length=100, unique=True)
2017-04-01 02:31:48 +00:00
# Longer description of the part (optional)
description = models.CharField(max_length=250)
2017-04-01 02:31:48 +00:00
# Internal Part Number (optional)
# Potentially multiple parts map to the same internal IPN (variants?)
# So this does not have to be unique
IPN = models.CharField(max_length=100, blank=True)
2017-04-01 02:31:48 +00:00
# Provide a URL for an external link
URL = models.URLField(blank=True)
2017-04-01 02:31:48 +00:00
# Part category - all parts must be assigned to a category
category = models.ForeignKey(PartCategory, related_name='parts',
null=True, blank=True,
on_delete=models.DO_NOTHING)
2017-04-01 02:31:48 +00:00
image = models.ImageField(upload_to=rename_part_image, max_length=255, null=True, blank=True)
2017-04-01 02:31:48 +00:00
# Minimum "allowed" stock level
2017-04-16 07:05:02 +00:00
minimum_stock = models.PositiveIntegerField(default=0, validators=[MinValueValidator(0)])
2017-04-01 02:31:48 +00:00
# Units of quantity for this part. Default is "pcs"
2017-03-28 06:49:01 +00:00
units = models.CharField(max_length=20, default="pcs", blank=True)
2017-04-01 02:31:48 +00:00
# Is this part "trackable"?
# Trackable parts can have unique instances which are assigned serial numbers
# and can have their movements tracked
trackable = models.BooleanField(default=False)
2017-03-29 12:19:53 +00:00
# Is this part "purchaseable"?
purchaseable = models.BooleanField(default=True)
def __str__(self):
if self.IPN:
return "{name} ({ipn})".format(
2017-03-28 12:31:41 +00:00
ipn=self.IPN,
name=self.name)
else:
return self.name
2017-03-29 12:19:53 +00:00
class Meta:
verbose_name = "Part"
verbose_name_plural = "Parts"
#unique_together = (("name", "category"),)
2017-03-29 12:19:53 +00:00
@property
def tracked_parts(self):
return self.serials.order_by('serial')
@property
2017-03-28 11:27:46 +00:00
def stock(self):
""" Return the total stock quantity for this part.
Part may be stored in multiple locations
"""
2017-03-29 12:19:53 +00:00
stocks = self.locations.all()
if len(stocks) == 0:
return 0
2017-03-29 12:19:53 +00:00
result = stocks.aggregate(total=Sum('quantity'))
return result['total']
2017-03-29 12:19:53 +00:00
@property
def bomItemCount(self):
return self.bom_items.all().count()
@property
def usedInCount(self):
return self.used_in.all().count()
"""
@property
def projects(self):
" Return a list of unique projects that this part is associated with.
2017-04-01 02:31:48 +00:00
A part may be used in zero or more projects.
"
2017-03-29 12:19:53 +00:00
project_ids = set()
project_parts = self.projectpart_set.all()
2017-03-29 12:19:53 +00:00
projects = []
2017-03-29 12:19:53 +00:00
for pp in project_parts:
if pp.project.id not in project_ids:
project_ids.add(pp.project.id)
projects.append(pp.project)
2017-03-29 12:19:53 +00:00
return projects
"""
2017-03-29 04:12:14 +00:00
def attach_file(instance, filename):
base='part_files'
# TODO - For a new PartAttachment object, PK is NULL!!
# Prefix the attachment ID to the filename
fn = "{id}_{fn}".format(id=instance.pk, fn=filename)
return os.path.join(base, fn)
class PartAttachment(models.Model):
""" A PartAttachment links a file to a part
Parts can have multiple files such as datasheets, etc
"""
part = models.ForeignKey(Part, on_delete=models.CASCADE,
related_name='attachments')
attachment = models.FileField(upload_to=attach_file, null=True, blank=True)
2017-03-29 04:45:50 +00:00
class BomItem(models.Model):
""" A BomItem links a part to its component items.
A part can have a BOM (bill of materials) which defines
which parts are required (and in what quatity) to make it
"""
def get_absolute_url(self):
return '/part/bom/{id}/'.format(id=self.id)
# A link to the parent part
# Each part will get a reverse lookup field 'bom_items'
part = models.ForeignKey(Part, on_delete=models.CASCADE, related_name='bom_items')
# A link to the child item (sub-part)
# Each part will get a reverse lookup field 'used_in'
sub_part = models.ForeignKey(Part, on_delete=models.CASCADE, related_name='used_in')
# Quantity required
quantity = models.PositiveIntegerField(default=1, validators=[MinValueValidator(0)])
class Meta:
verbose_name = "BOM Item"
# Prevent duplication of parent/child rows
unique_together = ('part', 'sub_part')
def __str__(self):
return "{par} -> {child} ({n})".format(
par=self.part.name,
child=self.sub_part.name,
n=self.quantity)