2018-04-16 14:32:02 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
2019-04-24 13:42:50 +00:00
|
|
|
from django.test import TestCase
|
2019-08-15 11:35:16 +00:00
|
|
|
from django.urls import reverse
|
|
|
|
from django.contrib.auth import get_user_model
|
2020-10-06 09:29:16 +00:00
|
|
|
from django.contrib.auth.models import Group
|
2019-08-15 11:35:16 +00:00
|
|
|
|
|
|
|
from rest_framework.test import APITestCase
|
|
|
|
from rest_framework import status
|
|
|
|
|
|
|
|
import json
|
2021-01-19 20:49:14 +00:00
|
|
|
from datetime import datetime, timedelta
|
2019-04-24 13:42:50 +00:00
|
|
|
|
|
|
|
from .models import Build
|
2020-11-03 09:19:24 +00:00
|
|
|
from stock.models import StockItem
|
2018-04-16 14:32:02 +00:00
|
|
|
|
2021-04-21 07:16:07 +00:00
|
|
|
from InvenTree.status_codes import BuildStatus, StockStatus
|
2019-06-04 13:42:48 +00:00
|
|
|
|
2019-04-24 13:42:50 +00:00
|
|
|
|
|
|
|
class BuildTestSimple(TestCase):
|
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
fixtures = [
|
|
|
|
'category',
|
|
|
|
'part',
|
|
|
|
'location',
|
|
|
|
'build',
|
|
|
|
]
|
|
|
|
|
2019-04-24 13:42:50 +00:00
|
|
|
def setUp(self):
|
2019-08-15 11:35:16 +00:00
|
|
|
# Create a user for auth
|
2020-11-12 10:53:04 +00:00
|
|
|
user = get_user_model()
|
|
|
|
user.objects.create_user('testuser', 'test@testing.com', 'password')
|
2019-08-15 11:35:16 +00:00
|
|
|
|
2020-11-12 10:53:04 +00:00
|
|
|
self.user = user.objects.get(username='testuser')
|
2020-10-06 09:29:16 +00:00
|
|
|
|
|
|
|
g = Group.objects.create(name='builders')
|
|
|
|
self.user.groups.add(g)
|
|
|
|
|
|
|
|
for rule in g.rule_sets.all():
|
|
|
|
if rule.name == 'build':
|
|
|
|
rule.can_change = True
|
|
|
|
rule.can_add = True
|
|
|
|
rule.can_delete = True
|
|
|
|
|
|
|
|
rule.save()
|
|
|
|
|
|
|
|
g.save()
|
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
self.client.login(username='testuser', password='password')
|
2019-04-24 13:42:50 +00:00
|
|
|
|
|
|
|
def test_build_objects(self):
|
|
|
|
# Ensure the Build objects were correctly created
|
2020-11-03 09:19:24 +00:00
|
|
|
self.assertEqual(Build.objects.count(), 5)
|
2019-04-24 13:42:50 +00:00
|
|
|
b = Build.objects.get(pk=2)
|
|
|
|
self.assertEqual(b.batch, 'B2')
|
|
|
|
self.assertEqual(b.quantity, 21)
|
|
|
|
|
2020-10-19 12:31:52 +00:00
|
|
|
self.assertEqual(str(b), 'BO0002')
|
2019-06-17 12:41:44 +00:00
|
|
|
|
2019-04-24 13:42:50 +00:00
|
|
|
def test_url(self):
|
|
|
|
b1 = Build.objects.get(pk=1)
|
|
|
|
self.assertEqual(b1.get_absolute_url(), '/build/1/')
|
|
|
|
|
|
|
|
def test_is_complete(self):
|
|
|
|
b1 = Build.objects.get(pk=1)
|
|
|
|
b2 = Build.objects.get(pk=2)
|
|
|
|
|
|
|
|
self.assertEqual(b1.is_complete, False)
|
|
|
|
self.assertEqual(b2.is_complete, True)
|
|
|
|
|
2019-06-17 12:41:44 +00:00
|
|
|
self.assertEqual(b2.status, BuildStatus.COMPLETE)
|
|
|
|
|
2021-01-19 20:49:14 +00:00
|
|
|
def test_overdue(self):
|
|
|
|
"""
|
|
|
|
Test overdue status functionality
|
|
|
|
"""
|
|
|
|
|
|
|
|
today = datetime.now().date()
|
|
|
|
|
|
|
|
build = Build.objects.get(pk=1)
|
|
|
|
self.assertFalse(build.is_overdue)
|
|
|
|
|
|
|
|
build.target_date = today - timedelta(days=1)
|
|
|
|
build.save()
|
|
|
|
self.assertTrue(build.is_overdue)
|
|
|
|
|
|
|
|
build.target_date = today + timedelta(days=80)
|
|
|
|
build.save()
|
|
|
|
self.assertFalse(build.is_overdue)
|
|
|
|
|
2019-04-24 13:42:50 +00:00
|
|
|
def test_is_active(self):
|
|
|
|
b1 = Build.objects.get(pk=1)
|
|
|
|
b2 = Build.objects.get(pk=2)
|
|
|
|
|
|
|
|
self.assertEqual(b1.is_active, True)
|
|
|
|
self.assertEqual(b2.is_active, False)
|
|
|
|
|
|
|
|
def test_required_parts(self):
|
|
|
|
# TODO - Generate BOM for test part
|
|
|
|
pass
|
2019-06-17 12:41:44 +00:00
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
def test_cancel_build(self):
|
2019-06-17 12:41:44 +00:00
|
|
|
""" Test build cancellation function """
|
|
|
|
|
|
|
|
build = Build.objects.get(id=1)
|
|
|
|
|
|
|
|
self.assertEqual(build.status, BuildStatus.PENDING)
|
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
build.cancelBuild(self.user)
|
2019-06-17 12:41:44 +00:00
|
|
|
|
|
|
|
self.assertEqual(build.status, BuildStatus.CANCELLED)
|
2019-08-15 11:35:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TestBuildAPI(APITestCase):
|
|
|
|
"""
|
|
|
|
Series of tests for the Build DRF API
|
|
|
|
- Tests for Build API
|
|
|
|
- Tests for BuildItem API
|
|
|
|
"""
|
|
|
|
|
|
|
|
fixtures = [
|
|
|
|
'category',
|
|
|
|
'part',
|
|
|
|
'location',
|
|
|
|
'build',
|
|
|
|
]
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
# Create a user for auth
|
2020-11-12 10:53:04 +00:00
|
|
|
user = get_user_model()
|
|
|
|
self.user = user.objects.create_user('testuser', 'test@testing.com', 'password')
|
2020-10-06 09:29:16 +00:00
|
|
|
|
|
|
|
g = Group.objects.create(name='builders')
|
2020-11-12 10:53:04 +00:00
|
|
|
self.user.groups.add(g)
|
2020-10-06 09:29:16 +00:00
|
|
|
|
|
|
|
for rule in g.rule_sets.all():
|
|
|
|
if rule.name == 'build':
|
|
|
|
rule.can_change = True
|
|
|
|
rule.can_add = True
|
|
|
|
rule.can_delete = True
|
|
|
|
|
|
|
|
rule.save()
|
|
|
|
|
|
|
|
g.save()
|
2019-08-15 11:35:16 +00:00
|
|
|
|
|
|
|
self.client.login(username='testuser', password='password')
|
|
|
|
|
|
|
|
def test_get_build_list(self):
|
2020-11-03 09:19:24 +00:00
|
|
|
"""
|
|
|
|
Test that we can retrieve list of build objects
|
|
|
|
"""
|
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
url = reverse('api-build-list')
|
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
|
2020-11-03 09:19:24 +00:00
|
|
|
self.assertEqual(len(response.data), 5)
|
|
|
|
|
|
|
|
# Filter query by build status
|
|
|
|
response = self.client.get(url, {'status': 40}, format='json')
|
|
|
|
|
|
|
|
self.assertEqual(len(response.data), 4)
|
|
|
|
|
|
|
|
# Filter by "active" status
|
|
|
|
response = self.client.get(url, {'active': True}, format='json')
|
|
|
|
self.assertEqual(len(response.data), 1)
|
|
|
|
self.assertEqual(response.data[0]['pk'], 1)
|
|
|
|
|
|
|
|
response = self.client.get(url, {'active': False}, format='json')
|
|
|
|
self.assertEqual(len(response.data), 4)
|
|
|
|
|
|
|
|
# Filter by 'part' status
|
|
|
|
response = self.client.get(url, {'part': 25}, format='json')
|
|
|
|
self.assertEqual(len(response.data), 2)
|
|
|
|
|
|
|
|
# Filter by an invalid part
|
|
|
|
response = self.client.get(url, {'part': 99999}, format='json')
|
|
|
|
self.assertEqual(len(response.data), 0)
|
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
def test_get_build_item_list(self):
|
|
|
|
""" Test that we can retrieve list of BuildItem objects """
|
|
|
|
url = reverse('api-build-item-list')
|
|
|
|
|
|
|
|
response = self.client.get(url, format='json')
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
|
|
|
|
# Test again, filtering by park ID
|
|
|
|
response = self.client.get(url, {'part': '1'}, format='json')
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
|
|
|
|
|
|
|
|
class TestBuildViews(TestCase):
|
|
|
|
""" Tests for Build app views """
|
|
|
|
|
|
|
|
fixtures = [
|
|
|
|
'category',
|
|
|
|
'part',
|
|
|
|
'location',
|
|
|
|
'build',
|
|
|
|
]
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
super().setUp()
|
|
|
|
|
|
|
|
# Create a user
|
2020-11-12 10:53:04 +00:00
|
|
|
user = get_user_model()
|
|
|
|
self.user = user.objects.create_user('username', 'user@email.com', 'password')
|
2020-10-06 09:29:16 +00:00
|
|
|
|
|
|
|
g = Group.objects.create(name='builders')
|
2020-11-12 10:53:04 +00:00
|
|
|
self.user.groups.add(g)
|
2020-10-06 09:29:16 +00:00
|
|
|
|
|
|
|
for rule in g.rule_sets.all():
|
|
|
|
if rule.name == 'build':
|
|
|
|
rule.can_change = True
|
|
|
|
rule.can_add = True
|
|
|
|
rule.can_delete = True
|
|
|
|
|
|
|
|
rule.save()
|
|
|
|
|
|
|
|
g.save()
|
2019-08-15 11:35:16 +00:00
|
|
|
|
|
|
|
self.client.login(username='username', password='password')
|
|
|
|
|
2020-11-03 09:19:24 +00:00
|
|
|
# Create a build output for build # 1
|
|
|
|
self.build = Build.objects.get(pk=1)
|
|
|
|
|
|
|
|
self.output = StockItem.objects.create(
|
|
|
|
part=self.build.part,
|
|
|
|
quantity=self.build.quantity,
|
|
|
|
build=self.build,
|
|
|
|
is_building=True,
|
|
|
|
)
|
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
def test_build_index(self):
|
|
|
|
""" test build index view """
|
|
|
|
|
|
|
|
response = self.client.get(reverse('build-index'))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_build_detail(self):
|
|
|
|
""" Test the detail view for a Build object """
|
|
|
|
|
|
|
|
pk = 1
|
|
|
|
|
|
|
|
response = self.client.get(reverse('build-detail', args=(pk,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
build = Build.objects.get(pk=pk)
|
|
|
|
|
|
|
|
content = str(response.content)
|
|
|
|
|
|
|
|
self.assertIn(build.title, content)
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
def test_build_create(self):
|
|
|
|
""" Test the build creation view (ajax form) """
|
|
|
|
|
|
|
|
url = reverse('build-create')
|
|
|
|
|
|
|
|
# Create build without specifying part
|
|
|
|
response = self.client.get(url, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
# Create build with valid part
|
|
|
|
response = self.client.get(url, {'part': 1}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
# Create build with invalid part
|
|
|
|
response = self.client.get(url, {'part': 9999}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_build_allocate(self):
|
|
|
|
""" Test the part allocation view for a Build """
|
|
|
|
|
|
|
|
url = reverse('build-allocate', args=(1,))
|
|
|
|
|
|
|
|
# Get the page normally
|
|
|
|
response = self.client.get(url)
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
# Get the page in editing mode
|
|
|
|
response = self.client.get(url, {'edit': 1})
|
|
|
|
self.assertEqual(response.status_code, 200)
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
def test_build_item_create(self):
|
|
|
|
""" Test the BuildItem creation view (ajax form) """
|
|
|
|
|
|
|
|
url = reverse('build-item-create')
|
|
|
|
|
|
|
|
# Try without a part specified
|
2019-08-15 11:50:42 +00:00
|
|
|
response = self.client.get(url, {'build': 1}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
2019-08-15 11:35:16 +00:00
|
|
|
self.assertEqual(response.status_code, 200)
|
2019-08-15 11:50:42 +00:00
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
# Try with an invalid build ID
|
2019-08-15 11:50:42 +00:00
|
|
|
response = self.client.get(url, {'build': 9999}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
2019-08-15 11:35:16 +00:00
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
# Try with a valid part specified
|
|
|
|
response = self.client.get(url, {'build': 1, 'part': 1}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
# Try with an invalid part specified
|
|
|
|
response = self.client.get(url, {'build': 1, 'part': 9999}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_build_item_edit(self):
|
|
|
|
""" Test the BuildItem edit view (ajax form) """
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
# TODO
|
|
|
|
# url = reverse('build-item-edit')
|
|
|
|
pass
|
|
|
|
|
2020-11-03 09:19:24 +00:00
|
|
|
def test_build_output_complete(self):
|
|
|
|
"""
|
|
|
|
Test the build output completion form
|
|
|
|
"""
|
2019-08-15 11:49:40 +00:00
|
|
|
|
2020-11-03 09:19:24 +00:00
|
|
|
# Firstly, check that the build cannot be completed!
|
|
|
|
self.assertFalse(self.build.can_complete)
|
|
|
|
|
2020-11-03 11:14:02 +00:00
|
|
|
url = reverse('build-output-complete', args=(1,))
|
2019-08-15 11:49:40 +00:00
|
|
|
|
|
|
|
# Test without confirmation
|
|
|
|
response = self.client.post(url, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-08-15 11:49:40 +00:00
|
|
|
data = json.loads(response.content)
|
|
|
|
self.assertFalse(data['form_valid'])
|
|
|
|
|
|
|
|
# Test with confirmation, valid location
|
2020-11-03 09:19:24 +00:00
|
|
|
response = self.client.post(
|
|
|
|
url,
|
|
|
|
{
|
|
|
|
'confirm': 1,
|
|
|
|
'confirm_incomplete': 1,
|
|
|
|
'location': 1,
|
|
|
|
'output': self.output.pk,
|
2021-04-21 07:16:07 +00:00
|
|
|
'stock_status': StockStatus.DAMAGED
|
2020-11-03 09:19:24 +00:00
|
|
|
},
|
|
|
|
HTTP_X_REQUESTED_WITH='XMLHttpRequest'
|
|
|
|
)
|
|
|
|
|
2019-08-15 11:49:40 +00:00
|
|
|
self.assertEqual(response.status_code, 200)
|
2020-11-03 09:19:24 +00:00
|
|
|
|
2019-08-15 11:49:40 +00:00
|
|
|
data = json.loads(response.content)
|
2021-04-21 07:16:07 +00:00
|
|
|
|
2019-08-15 11:49:40 +00:00
|
|
|
self.assertTrue(data['form_valid'])
|
|
|
|
|
2020-11-03 09:19:24 +00:00
|
|
|
# Now the build should be able to be completed
|
|
|
|
self.build.refresh_from_db()
|
|
|
|
self.assertTrue(self.build.can_complete)
|
|
|
|
|
2019-08-15 11:49:40 +00:00
|
|
|
# Test with confirmation, invalid location
|
|
|
|
response = self.client.post(url, {'confirm': 1, 'location': 9999}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-08-15 11:49:40 +00:00
|
|
|
data = json.loads(response.content)
|
|
|
|
self.assertFalse(data['form_valid'])
|
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
def test_build_cancel(self):
|
|
|
|
""" Test the build cancellation form """
|
|
|
|
|
|
|
|
url = reverse('build-cancel', args=(1,))
|
|
|
|
|
|
|
|
# Test without confirmation
|
|
|
|
response = self.client.post(url, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-08-15 11:35:16 +00:00
|
|
|
data = json.loads(response.content)
|
|
|
|
self.assertFalse(data['form_valid'])
|
|
|
|
|
|
|
|
b = Build.objects.get(pk=1)
|
|
|
|
self.assertEqual(b.status, 10) # Build status is still PENDING
|
|
|
|
|
|
|
|
# Test with confirmation
|
|
|
|
response = self.client.post(url, {'confirm_cancel': 1}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
data = json.loads(response.content)
|
|
|
|
self.assertTrue(data['form_valid'])
|
|
|
|
|
|
|
|
b = Build.objects.get(pk=1)
|
2019-08-15 11:49:40 +00:00
|
|
|
self.assertEqual(b.status, 30) # Build status is now CANCELLED
|
|
|
|
|
|
|
|
def test_build_unallocate(self):
|
|
|
|
""" Test the build unallocation view (ajax form) """
|
|
|
|
|
|
|
|
url = reverse('build-unallocate', args=(1,))
|
|
|
|
|
|
|
|
# Test without confirmation
|
|
|
|
response = self.client.post(url, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
data = json.loads(response.content)
|
|
|
|
self.assertFalse(data['form_valid'])
|
2021-05-06 10:11:38 +00:00
|
|
|
|
2019-08-15 11:49:40 +00:00
|
|
|
# Test with confirmation
|
|
|
|
response = self.client.post(url, {'confirm': 1}, HTTP_X_REQUESTED_WITH='XMLHttpRequest')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
data = json.loads(response.content)
|
|
|
|
self.assertTrue(data['form_valid'])
|