2019-04-26 15:03:11 +00:00
|
|
|
from rest_framework import status
|
2020-09-05 13:28:54 +00:00
|
|
|
|
2019-04-26 15:03:11 +00:00
|
|
|
from django.urls import reverse
|
|
|
|
|
2020-09-05 13:28:54 +00:00
|
|
|
from part.models import Part
|
|
|
|
from stock.models import StockItem
|
|
|
|
from company.models import Company
|
|
|
|
|
2021-02-26 10:45:37 +00:00
|
|
|
from InvenTree.api_tester import InvenTreeAPITestCase
|
2020-09-05 13:28:54 +00:00
|
|
|
from InvenTree.status_codes import StockStatus
|
|
|
|
|
2019-04-26 15:03:11 +00:00
|
|
|
|
2021-02-26 10:45:37 +00:00
|
|
|
class PartAPITest(InvenTreeAPITestCase):
|
2019-04-27 04:50:49 +00:00
|
|
|
"""
|
|
|
|
Series of tests for the Part DRF API
|
|
|
|
- Tests for Part API
|
|
|
|
- Tests for PartCategory API
|
|
|
|
"""
|
2019-04-26 15:03:11 +00:00
|
|
|
|
2019-05-04 14:15:25 +00:00
|
|
|
fixtures = [
|
|
|
|
'category',
|
|
|
|
'part',
|
|
|
|
'location',
|
|
|
|
'bom',
|
2020-05-17 04:14:54 +00:00
|
|
|
'test_templates',
|
2019-05-04 14:15:25 +00:00
|
|
|
]
|
|
|
|
|
2021-02-26 10:45:37 +00:00
|
|
|
roles = [
|
|
|
|
'part.change',
|
|
|
|
'part.add',
|
|
|
|
'part.delete',
|
|
|
|
'part_category.change',
|
|
|
|
'part_category.add',
|
|
|
|
]
|
2019-04-26 15:03:11 +00:00
|
|
|
|
2021-02-26 10:45:37 +00:00
|
|
|
def setUp(self):
|
|
|
|
super().setUp()
|
2019-04-27 04:50:49 +00:00
|
|
|
|
|
|
|
def test_get_categories(self):
|
2019-05-04 14:15:25 +00:00
|
|
|
""" Test that we can retrieve list of part categories """
|
2019-04-26 15:03:11 +00:00
|
|
|
url = reverse('api-part-category-list')
|
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
2019-05-04 14:15:25 +00:00
|
|
|
self.assertEqual(len(response.data), 8)
|
2019-04-26 15:03:11 +00:00
|
|
|
|
|
|
|
def test_add_categories(self):
|
2019-05-04 14:15:25 +00:00
|
|
|
""" Check that we can add categories """
|
2019-04-26 15:03:11 +00:00
|
|
|
data = {
|
|
|
|
'name': 'Animals',
|
|
|
|
'description': 'All animals go here'
|
|
|
|
}
|
|
|
|
|
|
|
|
url = reverse('api-part-category-list')
|
2019-04-27 04:50:49 +00:00
|
|
|
response = self.client.post(url, data, format='json')
|
2019-04-26 15:03:11 +00:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
2019-05-04 14:15:25 +00:00
|
|
|
|
|
|
|
parent = response.data['pk']
|
2019-04-26 15:03:11 +00:00
|
|
|
|
|
|
|
# Add some sub-categories to the top-level 'Animals' category
|
|
|
|
for animal in ['cat', 'dog', 'zebra']:
|
|
|
|
data = {
|
|
|
|
'name': animal,
|
|
|
|
'description': 'A sort of animal',
|
2019-05-04 14:15:25 +00:00
|
|
|
'parent': parent,
|
2019-04-26 15:03:11 +00:00
|
|
|
}
|
2019-04-27 04:50:49 +00:00
|
|
|
response = self.client.post(url, data, format='json')
|
2019-04-26 15:03:11 +00:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
2019-05-04 14:15:25 +00:00
|
|
|
self.assertEqual(response.data['parent'], parent)
|
2019-04-26 15:03:11 +00:00
|
|
|
self.assertEqual(response.data['name'], animal)
|
|
|
|
self.assertEqual(response.data['pathstring'], 'Animals/' + animal)
|
|
|
|
|
2019-04-27 04:50:49 +00:00
|
|
|
# There should be now 8 categories
|
2019-04-26 15:03:11 +00:00
|
|
|
response = self.client.get(url, format='json')
|
2019-05-04 14:15:25 +00:00
|
|
|
self.assertEqual(len(response.data), 12)
|
2019-04-26 15:03:11 +00:00
|
|
|
|
2019-04-27 04:50:49 +00:00
|
|
|
def test_cat_detail(self):
|
|
|
|
url = reverse('api-part-category-detail', kwargs={'pk': 4})
|
|
|
|
response = self.client.get(url, format='json')
|
2019-04-26 15:03:11 +00:00
|
|
|
|
2019-04-27 04:50:49 +00:00
|
|
|
# Test that we have retrieved the category
|
2019-05-04 14:15:25 +00:00
|
|
|
self.assertEqual(response.data['description'], 'Integrated Circuits')
|
2019-04-27 04:50:49 +00:00
|
|
|
self.assertEqual(response.data['parent'], 1)
|
2019-04-26 15:03:11 +00:00
|
|
|
|
2019-04-27 04:50:49 +00:00
|
|
|
# Change some data and post it back
|
|
|
|
data = response.data
|
|
|
|
data['name'] = 'Changing category'
|
|
|
|
data['parent'] = None
|
|
|
|
data['description'] = 'Changing the description'
|
|
|
|
response = self.client.patch(url, data, format='json')
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(response.data['description'], 'Changing the description')
|
2019-05-04 14:15:25 +00:00
|
|
|
self.assertIsNone(response.data['parent'])
|
2019-04-26 15:03:11 +00:00
|
|
|
|
|
|
|
def test_get_all_parts(self):
|
|
|
|
url = reverse('api-part-list')
|
2020-04-11 14:10:33 +00:00
|
|
|
data = {'cascade': True}
|
|
|
|
response = self.client.get(url, data, format='json')
|
2019-04-26 15:03:11 +00:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
2020-05-15 22:43:57 +00:00
|
|
|
self.assertEqual(len(response.data), 13)
|
2019-04-26 15:03:11 +00:00
|
|
|
|
|
|
|
def test_get_parts_by_cat(self):
|
|
|
|
url = reverse('api-part-list')
|
2019-05-04 14:15:25 +00:00
|
|
|
data = {'category': 2}
|
2019-04-27 04:50:49 +00:00
|
|
|
response = self.client.get(url, data, format='json')
|
2019-04-26 15:03:11 +00:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
|
|
|
|
# There should only be 2 objects in category C
|
|
|
|
self.assertEqual(len(response.data), 2)
|
|
|
|
|
|
|
|
for part in response.data:
|
2019-05-04 14:15:25 +00:00
|
|
|
self.assertEqual(part['category'], 2)
|
2019-04-26 15:03:11 +00:00
|
|
|
|
|
|
|
def test_include_children(self):
|
|
|
|
""" Test the special 'include_child_categories' flag
|
|
|
|
If provided, parts are provided for ANY child category (recursive)
|
|
|
|
"""
|
|
|
|
url = reverse('api-part-list')
|
2020-04-03 01:30:58 +00:00
|
|
|
data = {'category': 1, 'cascade': True}
|
2019-04-26 15:03:11 +00:00
|
|
|
|
|
|
|
# Now request to include child categories
|
2019-04-27 04:50:49 +00:00
|
|
|
response = self.client.get(url, data, format='json')
|
2019-04-26 15:03:11 +00:00
|
|
|
|
|
|
|
# Now there should be 5 total parts
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
2019-05-04 14:15:25 +00:00
|
|
|
self.assertEqual(len(response.data), 3)
|
2019-04-27 04:50:49 +00:00
|
|
|
|
|
|
|
def test_get_bom_list(self):
|
2019-05-04 14:15:25 +00:00
|
|
|
""" There should be 4 BomItem objects in the database """
|
2019-04-27 04:50:49 +00:00
|
|
|
url = reverse('api-bom-list')
|
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
self.assertEqual(len(response.data), 4)
|
|
|
|
|
|
|
|
def test_get_bom_detail(self):
|
|
|
|
# Get the detail for a single BomItem
|
2019-09-05 09:29:51 +00:00
|
|
|
url = reverse('api-bom-item-detail', kwargs={'pk': 3})
|
2019-04-27 04:50:49 +00:00
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
2019-11-18 21:59:56 +00:00
|
|
|
self.assertEqual(int(float(response.data['quantity'])), 25)
|
2019-04-27 04:50:49 +00:00
|
|
|
|
|
|
|
# Increase the quantity
|
|
|
|
data = response.data
|
|
|
|
data['quantity'] = 57
|
|
|
|
data['note'] = 'Added a note'
|
|
|
|
|
|
|
|
response = self.client.patch(url, data, format='json')
|
|
|
|
|
|
|
|
# Check that the quantity was increased and a note added
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
2019-11-18 21:59:56 +00:00
|
|
|
self.assertEqual(int(float(response.data['quantity'])), 57)
|
2019-04-27 04:50:49 +00:00
|
|
|
self.assertEqual(response.data['note'], 'Added a note')
|
|
|
|
|
|
|
|
def test_add_bom_item(self):
|
|
|
|
url = reverse('api-bom-list')
|
|
|
|
|
|
|
|
data = {
|
2019-05-04 14:15:25 +00:00
|
|
|
'part': 100,
|
2019-04-27 04:50:49 +00:00
|
|
|
'sub_part': 4,
|
|
|
|
'quantity': 777,
|
|
|
|
}
|
|
|
|
|
|
|
|
response = self.client.post(url, data, format='json')
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
|
|
|
|
2019-06-02 09:46:30 +00:00
|
|
|
# Now try to create a BomItem which points to a non-assembly part (should fail)
|
2019-04-27 04:50:49 +00:00
|
|
|
data['part'] = 3
|
|
|
|
response = self.client.post(url, data, format='json')
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST)
|
|
|
|
|
2019-05-04 14:15:25 +00:00
|
|
|
# TODO - Now try to create a BomItem which references itself
|
2019-04-27 04:50:49 +00:00
|
|
|
data['part'] = 2
|
|
|
|
data['sub_part'] = 2
|
|
|
|
response = self.client.post(url, data, format='json')
|
2020-05-17 04:14:54 +00:00
|
|
|
|
|
|
|
def test_test_templates(self):
|
|
|
|
|
|
|
|
url = reverse('api-part-test-template-list')
|
|
|
|
|
|
|
|
# List ALL items
|
|
|
|
response = self.client.get(url)
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(len(response.data), 7)
|
|
|
|
|
|
|
|
# Request for a particular part
|
|
|
|
response = self.client.get(url, data={'part': 10000})
|
|
|
|
self.assertEqual(len(response.data), 5)
|
|
|
|
|
|
|
|
response = self.client.get(url, data={'part': 10004})
|
|
|
|
self.assertEqual(len(response.data), 7)
|
|
|
|
|
|
|
|
# Try to post a new object (should succeed)
|
|
|
|
response = self.client.post(
|
|
|
|
url,
|
|
|
|
data={
|
|
|
|
'part': 10000,
|
|
|
|
'test_name': 'New Test',
|
|
|
|
'required': True,
|
|
|
|
},
|
|
|
|
format='json',
|
|
|
|
)
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
|
|
|
|
|
|
|
# Try to post a new test with the same name (should fail)
|
|
|
|
response = self.client.post(
|
|
|
|
url,
|
|
|
|
data={
|
|
|
|
'part': 10004,
|
|
|
|
'test_name': " newtest"
|
|
|
|
},
|
|
|
|
format='json',
|
|
|
|
)
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST)
|
|
|
|
|
|
|
|
# Try to post a new test against a non-trackable part (should fail)
|
|
|
|
response = self.client.post(
|
|
|
|
url,
|
|
|
|
data={
|
|
|
|
'part': 1,
|
|
|
|
'test_name': 'A simple test',
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST)
|
2020-09-05 13:28:54 +00:00
|
|
|
|
2020-11-04 04:41:17 +00:00
|
|
|
def test_get_thumbs(self):
|
|
|
|
"""
|
|
|
|
Return list of part thumbnails
|
|
|
|
"""
|
|
|
|
|
|
|
|
url = reverse('api-part-thumbs')
|
|
|
|
|
|
|
|
response = self.client.get(url)
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
|
2021-02-28 09:31:01 +00:00
|
|
|
def test_paginate(self):
|
|
|
|
"""
|
|
|
|
Test pagination of the Part list API
|
|
|
|
"""
|
|
|
|
|
|
|
|
for n in [1, 5, 10]:
|
|
|
|
response = self.get(reverse('api-part-list'), {'limit': n})
|
|
|
|
|
2021-03-01 00:21:18 +00:00
|
|
|
data = response.data
|
|
|
|
|
|
|
|
self.assertIn('count', data)
|
|
|
|
self.assertIn('results', data)
|
2021-02-28 09:31:01 +00:00
|
|
|
|
2021-03-01 00:21:18 +00:00
|
|
|
self.assertEqual(len(data['results']), n)
|
2021-02-28 09:31:01 +00:00
|
|
|
|
2020-09-05 13:28:54 +00:00
|
|
|
|
2021-02-26 10:45:37 +00:00
|
|
|
class PartAPIAggregationTest(InvenTreeAPITestCase):
|
2020-09-05 13:28:54 +00:00
|
|
|
"""
|
|
|
|
Tests to ensure that the various aggregation annotations are working correctly...
|
|
|
|
"""
|
|
|
|
|
|
|
|
fixtures = [
|
|
|
|
'category',
|
|
|
|
'company',
|
|
|
|
'part',
|
|
|
|
'location',
|
|
|
|
'bom',
|
|
|
|
'test_templates',
|
|
|
|
]
|
|
|
|
|
2021-02-26 10:45:37 +00:00
|
|
|
roles = [
|
|
|
|
'part.view',
|
|
|
|
'part.change',
|
|
|
|
]
|
|
|
|
|
2020-09-05 13:28:54 +00:00
|
|
|
def setUp(self):
|
|
|
|
|
2021-02-26 10:45:37 +00:00
|
|
|
super().setUp()
|
2020-09-05 13:28:54 +00:00
|
|
|
|
|
|
|
# Add a new part
|
|
|
|
self.part = Part.objects.create(
|
|
|
|
name='Banana',
|
|
|
|
)
|
|
|
|
|
|
|
|
# Create some stock items associated with the part
|
|
|
|
|
|
|
|
# First create 600 units which are OK
|
|
|
|
StockItem.objects.create(part=self.part, quantity=100)
|
|
|
|
StockItem.objects.create(part=self.part, quantity=200)
|
|
|
|
StockItem.objects.create(part=self.part, quantity=300)
|
|
|
|
|
|
|
|
# Now create another 400 units which are LOST
|
|
|
|
StockItem.objects.create(part=self.part, quantity=400, status=StockStatus.LOST)
|
|
|
|
|
|
|
|
def get_part_data(self):
|
|
|
|
url = reverse('api-part-list')
|
|
|
|
|
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
|
|
|
|
for part in response.data:
|
|
|
|
if part['pk'] == self.part.pk:
|
|
|
|
return part
|
|
|
|
|
|
|
|
# We should never get here!
|
|
|
|
self.assertTrue(False)
|
|
|
|
|
|
|
|
def test_stock_quantity(self):
|
|
|
|
"""
|
|
|
|
Simple test for the stock quantity
|
|
|
|
"""
|
|
|
|
|
|
|
|
data = self.get_part_data()
|
|
|
|
|
|
|
|
self.assertEqual(data['in_stock'], 600)
|
|
|
|
self.assertEqual(data['stock_item_count'], 4)
|
|
|
|
|
|
|
|
# Add some more stock items!!
|
|
|
|
for i in range(100):
|
|
|
|
StockItem.objects.create(part=self.part, quantity=5)
|
|
|
|
|
|
|
|
# Add another stock item which is assigned to a customer (and shouldn't count)
|
|
|
|
customer = Company.objects.get(pk=4)
|
|
|
|
StockItem.objects.create(part=self.part, quantity=9999, customer=customer)
|
|
|
|
|
|
|
|
data = self.get_part_data()
|
|
|
|
|
|
|
|
self.assertEqual(data['in_stock'], 1100)
|
|
|
|
self.assertEqual(data['stock_item_count'], 105)
|
2021-04-11 09:28:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
class PartParameterTest(InvenTreeAPITestCase):
|
|
|
|
"""
|
|
|
|
Tests for the ParParameter API
|
|
|
|
"""
|
|
|
|
|
|
|
|
superuser = True
|
|
|
|
|
|
|
|
fixtures = [
|
|
|
|
'category',
|
|
|
|
'part',
|
|
|
|
'location',
|
|
|
|
'params',
|
|
|
|
]
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
|
|
|
|
super().setUp()
|
|
|
|
|
|
|
|
def test_list_params(self):
|
|
|
|
"""
|
|
|
|
Test for listing part parameters
|
|
|
|
"""
|
|
|
|
|
|
|
|
url = reverse('api-part-param-list')
|
|
|
|
|
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
|
|
|
|
self.assertEqual(len(response.data), 5)
|
|
|
|
|
|
|
|
# Filter by part
|
|
|
|
response = self.client.get(
|
|
|
|
url,
|
|
|
|
{
|
|
|
|
'part': 3,
|
|
|
|
},
|
|
|
|
format='json'
|
|
|
|
)
|
|
|
|
|
|
|
|
self.assertEqual(len(response.data), 3)
|
|
|
|
|
|
|
|
# Filter by template
|
|
|
|
response = self.client.get(
|
|
|
|
url,
|
|
|
|
{
|
|
|
|
'template': 1,
|
|
|
|
},
|
|
|
|
format='json',
|
|
|
|
)
|
|
|
|
|
|
|
|
self.assertEqual(len(response.data), 3)
|
|
|
|
|
|
|
|
def test_create_param(self):
|
|
|
|
"""
|
|
|
|
Test that we can create a param via the API
|
|
|
|
"""
|
|
|
|
|
|
|
|
url = reverse('api-part-param-list')
|
|
|
|
|
|
|
|
response = self.client.post(
|
|
|
|
url,
|
|
|
|
{
|
|
|
|
'part': '2',
|
|
|
|
'template': '3',
|
|
|
|
'data': 70
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 201)
|
|
|
|
|
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
|
|
|
|
self.assertEqual(len(response.data), 6)
|
|
|
|
|
|
|
|
def test_param_detail(self):
|
|
|
|
"""
|
|
|
|
Tests for the PartParameter detail endpoint
|
|
|
|
"""
|
|
|
|
|
|
|
|
url = reverse('api-part-param-detail', kwargs={'pk': 5})
|
|
|
|
|
|
|
|
response = self.client.get(url)
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
data = response.data
|
|
|
|
|
|
|
|
self.assertEqual(data['pk'], 5)
|
|
|
|
self.assertEqual(data['part'], 3)
|
|
|
|
self.assertEqual(data['data'], '12')
|
|
|
|
|
|
|
|
# PATCH data back in
|
|
|
|
response = self.client.patch(url, {'data': '15'}, format='json')
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
# Check that the data changed!
|
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
|
|
|
|
data = response.data
|
|
|
|
|
|
|
|
self.assertEqual(data['data'], '15')
|