Merge remote-tracking branch 'inventree/master'

This commit is contained in:
Oliver Walters 2019-05-21 14:13:30 +10:00
commit a15f46c972
10 changed files with 188 additions and 18 deletions

View File

@ -0,0 +1,14 @@
# Sample company data
- model: company.company
fields:
name: ACME
description: A Cool Military Enterprise
- model: company.company
fields:
name: Appel Computers
description: Think more differenter
- model: company.company
fields:
name: Zerg Corp
description: We eat the competition

View File

@ -0,0 +1,47 @@
# Price breaks for supplier parts
# Price breaks for ACME0001
- model: company.supplierpricebreak
fields:
part: 1
quantity: 1
cost: 10
- model: company.supplierpricebreak
fields:
part: 1
quantity: 5
cost: 7.50
- model: company.supplierpricebreak
fields:
part: 1
quantity: 25
cost: 3.50
# Price breaks for ACME0002
- model: company.supplierpricebreak
fields:
part: 2
quantity: 5
cost: 7.00
- model: company.supplierpricebreak
fields:
part: 2
quantity: 50
cost: 1.25
# Price breaks for ZERGLPHS
- model: company.supplierpricebreak
fields:
part: 4
quantity: 25
cost: 8
- model: company.supplierpricebreak
fields:
part: 4
quantity: 100
cost: 1.25

View File

@ -0,0 +1,31 @@
# Supplier Parts
# M2x4 LPHS from ACME
- model: company.supplierpart
pk: 1
fields:
part: 1
supplier: 1
SKU: 'ACME0001'
- model: company.supplierpart
pk: 2
fields:
part: 1
supplier: 1
SKU: 'ACME0002'
# M2x4 LPHS from Zerg Corp
- model: company.supplierpart
pk: 3
fields:
part: 1
supplier: 3
SKU: 'ZERGLPHS'
- model: company.supplierpart
pk: 4
fields:
part: 2
supplier: 2
SKU: 'ZERGM312'

View File

@ -57,7 +57,6 @@ class EditSupplierPartForm(HelperForm):
'note', 'note',
'base_cost', 'base_cost',
'multiple', 'multiple',
'minimum',
'packaging', 'packaging',
'lead_time' 'lead_time'
] ]

View File

@ -0,0 +1,17 @@
# Generated by Django 2.2 on 2019-05-21 03:51
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('company', '0002_auto_20190520_2204'),
]
operations = [
migrations.RemoveField(
model_name='supplierpart',
name='minimum',
),
]

View File

@ -173,7 +173,6 @@ class SupplierPart(models.Model):
note: Longer form note field note: Longer form note field
base_cost: Base charge added to order independent of quantity e.g. "Reeling Fee" base_cost: Base charge added to order independent of quantity e.g. "Reeling Fee"
multiple: Multiple that the part is provided in multiple: Multiple that the part is provided in
minimum: MOQ (minimum order quantity) required for purchase
lead_time: Supplier lead time lead_time: Supplier lead time
packaging: packaging that the part is supplied in, e.g. "Reel" packaging: packaging that the part is supplied in, e.g. "Reel"
""" """
@ -217,8 +216,6 @@ class SupplierPart(models.Model):
multiple = models.PositiveIntegerField(default=1, validators=[MinValueValidator(1)], help_text='Order multiple') multiple = models.PositiveIntegerField(default=1, validators=[MinValueValidator(1)], help_text='Order multiple')
minimum = models.PositiveIntegerField(default=1, validators=[MinValueValidator(1)], help_text='Minimum order quantity (MOQ)')
lead_time = models.DurationField(blank=True, null=True) lead_time = models.DurationField(blank=True, null=True)
@property @property
@ -256,10 +253,6 @@ class SupplierPart(models.Model):
if len(price_breaks) == 0: if len(price_breaks) == 0:
return None return None
# Minimum ordering requirement
if moq and self.minimum > quantity:
quantity = self.minimum
# Order multiples # Order multiples
if multiples: if multiples:
quantity = int(math.ceil(quantity / self.multiple) * self.multiple) quantity = int(math.ceil(quantity / self.multiple) * self.multiple)

View File

@ -66,9 +66,6 @@ InvenTree | {{ company.name }} - Parts
{% if part.base_cost > 0 %} {% if part.base_cost > 0 %}
<tr><td>Base Price (Flat Fee)</td><td>{{ part.base_cost }}</td></tr> <tr><td>Base Price (Flat Fee)</td><td>{{ part.base_cost }}</td></tr>
{% endif %} {% endif %}
{% if part.minimum > 1 %}
<tr><td>Minimum Order Quantity</td><td>{{ part.minimum }}</td></tr>
{% endif %}
<tr> <tr>
<th>Price Breaks</th> <th>Price Breaks</th>
<th> <th>

View File

@ -2,12 +2,23 @@ from django.test import TestCase
import os import os
from .models import Company, Contact from .models import Company, Contact, SupplierPart
from .models import rename_company_image from .models import rename_company_image
from part.models import Part
class CompanySimpleTest(TestCase): class CompanySimpleTest(TestCase):
fixtures = [
'company',
'category',
'part',
'location',
'bom',
'supplier_part',
'price_breaks',
]
def setUp(self): def setUp(self):
Company.objects.create(name='ABC Co.', Company.objects.create(name='ABC Co.',
description='Seller of ABC products', description='Seller of ABC products',
@ -16,8 +27,13 @@ class CompanySimpleTest(TestCase):
is_customer=False, is_customer=False,
is_supplier=True) is_supplier=True)
self.acme0001 = SupplierPart.objects.get(SKU='ACME0001')
self.acme0002 = SupplierPart.objects.get(SKU='ACME0002')
self.zerglphs = SupplierPart.objects.get(SKU='ZERGLPHS')
self.zergm312 = SupplierPart.objects.get(SKU='ZERGM312')
def test_company_model(self): def test_company_model(self):
c = Company.objects.get(pk=1) c = Company.objects.get(name='ABC Co.')
self.assertEqual(c.name, 'ABC Co.') self.assertEqual(c.name, 'ABC Co.')
self.assertEqual(str(c), 'ABC Co. - Seller of ABC products') self.assertEqual(str(c), 'ABC Co. - Seller of ABC products')
@ -34,11 +50,63 @@ class CompanySimpleTest(TestCase):
self.assertEqual(rn, 'company_images' + os.path.sep + 'company_1_img') self.assertEqual(rn, 'company_images' + os.path.sep + 'company_1_img')
def test_part_count(self): def test_part_count(self):
# Initially the company should have no associated parts
c = Company.objects.get(pk=1)
self.assertEqual(c.has_parts, False)
# TODO - Add some supplier parts here acme = Company.objects.get(pk=1)
appel = Company.objects.get(pk=2)
zerg = Company.objects.get(pk=3)
self.assertTrue(acme.has_parts)
self.assertEqual(acme.part_count, 2)
self.assertTrue(appel.has_parts)
self.assertEqual(appel.part_count, 1)
self.assertTrue(zerg.has_parts)
self.assertEqual(zerg.part_count, 1)
def test_price_breaks(self):
self.assertTrue(self.acme0001.has_price_breaks)
self.assertTrue(self.acme0002.has_price_breaks)
self.assertTrue(self.zergm312.has_price_breaks)
self.assertFalse(self.zerglphs.has_price_breaks)
self.assertEqual(self.acme0001.price_breaks.count(), 3)
self.assertEqual(self.acme0002.price_breaks.count(), 2)
self.assertEqual(self.zerglphs.price_breaks.count(), 0)
self.assertEqual(self.zergm312.price_breaks.count(), 2)
def test_quantity_pricing(self):
""" Simple test for quantity pricing """
p = self.acme0001.get_price
self.assertEqual(p(1), 10)
self.assertEqual(p(4), 40)
self.assertEqual(p(11), 82.5)
self.assertEqual(p(23), 172.5)
self.assertEqual(p(100), 350)
p = self.acme0002.get_price
self.assertEqual(p(1), None)
self.assertEqual(p(2), None)
self.assertEqual(p(5), 35)
self.assertEqual(p(45), 315)
self.assertEqual(p(55), 68.75)
def test_part_pricing(self):
m2x4 = Part.objects.get(name='M2x4 LPHS')
self.assertEqual(m2x4.get_price_info(10), "70.00000 - 75.00000")
self.assertEqual(m2x4.get_price_info(100), "125.00000 - 350.00000")
pmin, pmax = m2x4.get_price_range(5)
self.assertEqual(pmin, 35)
self.assertEqual(pmax, 37.5)
m3x12 = Part.objects.get(name='M3x12 SHCS')
self.assertIsNone(m3x12.get_price_info(3))
self.assertIsNotNone(m3x12.get_price_info(50))
class ContactSimpleTest(TestCase): class ContactSimpleTest(TestCase):

View File

@ -710,7 +710,7 @@ class Part(models.Model):
if kwargs.get('image', True): if kwargs.get('image', True):
if other.image: if other.image:
image_file = ContentFile(other.image.read()) image_file = ContentFile(other.image.read())
image_file.name = rename_part_image(self, 'test.png') image_file.name = rename_part_image(self, other.image.url)
self.image = image_file self.image = image_file

View File

@ -68,3 +68,7 @@ class PartTest(TestCase):
barcode = self.R1.format_barcode() barcode = self.R1.format_barcode()
self.assertIn('InvenTree', barcode) self.assertIn('InvenTree', barcode)
self.assertIn(self.R1.name, barcode) self.assertIn(self.R1.name, barcode)
def test_copy(self):
self.R2.deepCopy(self.R1, image=True, bom=True)