mirror of
				https://github.com/inventree/InvenTree.git
				synced 2025-10-25 18:37:38 +00:00 
			
		
		
		
	Merge branch 'master' into make-fields-filterable
This commit is contained in:
		| @@ -11,6 +11,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | |||||||
|  |  | ||||||
| - Added `order_queryset` report helper function in [#10439](https://github.com/inventree/InvenTree/pull/10439) | - Added `order_queryset` report helper function in [#10439](https://github.com/inventree/InvenTree/pull/10439) | ||||||
| - Added much more detailed status information for machines to the API endpoint (including backend and frontend changes) in [#10381](https://github.com/inventree/InvenTree/pull/10381) | - Added much more detailed status information for machines to the API endpoint (including backend and frontend changes) in [#10381](https://github.com/inventree/InvenTree/pull/10381) | ||||||
|  | - Added ability to partially complete and partially scrap build outputs in [#10499](https://github.com/inventree/InvenTree/pull/10499) | ||||||
|  |  | ||||||
| ### Changed | ### Changed | ||||||
|  |  | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 106 KiB After Width: | Height: | Size: 36 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 141 KiB After Width: | Height: | Size: 42 KiB | 
| @@ -70,6 +70,18 @@ The following options are available when completing a build output: | |||||||
| | Notes | Any additional notes associated with the completion of these outputs | | | Notes | Any additional notes associated with the completion of these outputs | | ||||||
| | Accept Incomplete Allocation | If selected, this option allows [tracked build outputs](./allocate.md#tracked-build-outputs) to be completed in the case where required BOM items have not been fully allocated | | | Accept Incomplete Allocation | If selected, this option allows [tracked build outputs](./allocate.md#tracked-build-outputs) to be completed in the case where required BOM items have not been fully allocated | | ||||||
|  |  | ||||||
|  | ### Partial Completion | ||||||
|  |  | ||||||
|  | A build output may be *partially completed* by specifying a quantity less than the total quantity of that build output. In such a case, the following actions are performed: | ||||||
|  |  | ||||||
|  | - The incomplete build output is "split" into two separate build outputs | ||||||
|  | - The specified quantity is marked as completed, and the completed build quantity is increased accordingly | ||||||
|  | - The remaining quantity is left as an incomplete build output, available for future completion | ||||||
|  |  | ||||||
|  | !!! note "Serialized Outputs" | ||||||
|  |     Serialized build outputs cannot be partially completed. | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Scrap Build Output | ## Scrap Build Output | ||||||
|  |  | ||||||
| *Scrapping* a build output marks the particular output as rejected, in the context of the given build order. | *Scrapping* a build output marks the particular output as rejected, in the context of the given build order. | ||||||
| @@ -95,6 +107,17 @@ The following options are available when scrapping a build order: | |||||||
| | Notes | Any additional notes associated with the scrapping of these outputs | | | Notes | Any additional notes associated with the scrapping of these outputs | | ||||||
| | Discard Allocations | If selected, any installed BOM items will be removed first, before marking the build output as scrapped. Use this option if the installed items are recoverable and can be used elsewhere | | | Discard Allocations | If selected, any installed BOM items will be removed first, before marking the build output as scrapped. Use this option if the installed items are recoverable and can be used elsewhere | | ||||||
|  |  | ||||||
|  | ### Partial Scrapping | ||||||
|  |  | ||||||
|  | A build output may be *partially scrapped* by specifying a quantity less than the total quantity of that build output. In such a case, the following actions are performed: | ||||||
|  |  | ||||||
|  | - The incomplete build output is "split" into two separate build outputs | ||||||
|  | - The specified quantity is marked as scrapped, and the completed build quantity is *not* increased | ||||||
|  | - The remaining quantity is left as an incomplete build output, available for future completion | ||||||
|  |  | ||||||
|  | !!! note "Serialized Outputs" | ||||||
|  |     Serialized build outputs cannot be partially scrapped. | ||||||
|  |  | ||||||
| ## Cancel Build Output | ## Cancel Build Output | ||||||
|  |  | ||||||
| *Cancelling* a build output causes the build output to be deleted, and removed from the database entirely. Use this option when the build output does not physically exist (or was never built) and should not be tracked in the database. | *Cancelling* a build output causes the build output to be deleted, and removed from the database entirely. Use this option when the build output does not physically exist (or was never built) and should not be tracked in the database. | ||||||
|   | |||||||
| @@ -1,13 +1,22 @@ | |||||||
| """InvenTree API version information.""" | """InvenTree API version information.""" | ||||||
|  |  | ||||||
| # InvenTree API version | # InvenTree API version | ||||||
| INVENTREE_API_VERSION = 401 | INVENTREE_API_VERSION = 403 | ||||||
|  |  | ||||||
| """Increment this API version number whenever there is a significant change to the API that any clients need to know about.""" | """Increment this API version number whenever there is a significant change to the API that any clients need to know about.""" | ||||||
|  |  | ||||||
| INVENTREE_API_TEXT = """ | INVENTREE_API_TEXT = """ | ||||||
|  |  | ||||||
| v401 -> 2025-10-04 : https://github.com/inventree/InvenTree/pull/10381 | v403 -> 2025-10-06: https://github.com/inventree/InvenTree/pull/10499 | ||||||
|  |     - Adds ability to partially scrap a build output | ||||||
|  |     - Adds ability to partially complete a build output | ||||||
|  |  | ||||||
|  | v402 -> 2025-10-05 : https://github.com/inventree/InvenTree/pull/10495 | ||||||
|  |     - Refactors 'part_detail' param in BuildList API endpoint | ||||||
|  |     - Refactors 'order_detail' param in GeneralExtraLineList API endpoint | ||||||
|  |     - Refactors 'part_detail', 'template_detail' param in PartParameterList / PartParameterDetail API endpoint | ||||||
|  |  | ||||||
|  | v401 -> 2025-10-05 : https://github.com/inventree/InvenTree/pull/10381 | ||||||
|     - Adds machine properties to machine API endpoints |     - Adds machine properties to machine API endpoints | ||||||
|  |  | ||||||
| v400 -> 2025-10-05 : https://github.com/inventree/InvenTree/pull/10486 | v400 -> 2025-10-05 : https://github.com/inventree/InvenTree/pull/10486 | ||||||
|   | |||||||
| @@ -218,7 +218,7 @@ class InvenTreeOutputOption: | |||||||
|     """Represents an available output option with description, flag name, and default value.""" |     """Represents an available output option with description, flag name, and default value.""" | ||||||
|  |  | ||||||
|     DEFAULT_DESCRIPTIONS = { |     DEFAULT_DESCRIPTIONS = { | ||||||
|         'part_detail': 'Include detailed information about the related part  in the response', |         'part_detail': 'Include detailed information about the related part in the response', | ||||||
|         'item_detail': 'Include detailed information about the item in the response', |         'item_detail': 'Include detailed information about the item in the response', | ||||||
|         'order_detail': 'Include detailed information about the sales order in the response', |         'order_detail': 'Include detailed information about the sales order in the response', | ||||||
|         'location_detail': 'Include detailed information about the stock location in the response', |         'location_detail': 'Include detailed information about the stock location in the response', | ||||||
| @@ -231,7 +231,7 @@ class InvenTreeOutputOption: | |||||||
|         self.flag = flag |         self.flag = flag | ||||||
|         self.default = default |         self.default = default | ||||||
|  |  | ||||||
|         if description is None: |         if description is None or description == '': | ||||||
|             self.description = self.DEFAULT_DESCRIPTIONS.get(flag, '') |             self.description = self.DEFAULT_DESCRIPTIONS.get(flag, '') | ||||||
|         else: |         else: | ||||||
|             self.description = description |             self.description = description | ||||||
|   | |||||||
| @@ -332,17 +332,22 @@ class BuildMixin: | |||||||
|         return queryset |         return queryset | ||||||
|  |  | ||||||
|  |  | ||||||
| class BuildList(DataExportViewMixin, BuildMixin, ListCreateAPI): | class BuildListOutputOptions(OutputConfiguration): | ||||||
|  |     """Output options for the BuildList endpoint.""" | ||||||
|  |  | ||||||
|  |     OPTIONS = [InvenTreeOutputOption('part_detail', default=True)] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BuildList(DataExportViewMixin, BuildMixin, OutputOptionsMixin, ListCreateAPI): | ||||||
|     """API endpoint for accessing a list of Build objects. |     """API endpoint for accessing a list of Build objects. | ||||||
|  |  | ||||||
|     - GET: Return list of objects (with filters) |     - GET: Return list of objects (with filters) | ||||||
|     - POST: Create a new Build object |     - POST: Create a new Build object | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     output_options = BuildListOutputOptions | ||||||
|     filterset_class = BuildFilter |     filterset_class = BuildFilter | ||||||
|  |  | ||||||
|     filter_backends = SEARCH_ORDER_FILTER_ALIAS |     filter_backends = SEARCH_ORDER_FILTER_ALIAS | ||||||
|  |  | ||||||
|     ordering_fields = [ |     ordering_fields = [ | ||||||
|         'reference', |         'reference', | ||||||
|         'part__name', |         'part__name', | ||||||
| @@ -360,14 +365,11 @@ class BuildList(DataExportViewMixin, BuildMixin, ListCreateAPI): | |||||||
|         'level', |         'level', | ||||||
|         'external', |         'external', | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
|     ordering_field_aliases = { |     ordering_field_aliases = { | ||||||
|         'reference': ['reference_int', 'reference'], |         'reference': ['reference_int', 'reference'], | ||||||
|         'project_code': ['project_code__code'], |         'project_code': ['project_code__code'], | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ordering = '-reference' |     ordering = '-reference' | ||||||
|  |  | ||||||
|     search_fields = [ |     search_fields = [ | ||||||
|         'reference', |         'reference', | ||||||
|         'title', |         'title', | ||||||
|   | |||||||
| @@ -1115,7 +1115,9 @@ class Build( | |||||||
|         items.all().delete() |         items.all().delete() | ||||||
|  |  | ||||||
|     @transaction.atomic |     @transaction.atomic | ||||||
|     def scrap_build_output(self, output, quantity, location, **kwargs): |     def scrap_build_output( | ||||||
|  |         self, output: stock.models.StockItem, quantity, location, **kwargs | ||||||
|  |     ): | ||||||
|         """Mark a particular build output as scrapped / rejected. |         """Mark a particular build output as scrapped / rejected. | ||||||
|  |  | ||||||
|         - Mark the output as "complete" |         - Mark the output as "complete" | ||||||
| @@ -1126,6 +1128,10 @@ class Build( | |||||||
|         if not output: |         if not output: | ||||||
|             raise ValidationError(_('No build output specified')) |             raise ValidationError(_('No build output specified')) | ||||||
|  |  | ||||||
|  |         # If quantity is not specified, assume the entire output quantity | ||||||
|  |         if quantity is None: | ||||||
|  |             quantity = output.quantity | ||||||
|  |  | ||||||
|         if quantity <= 0: |         if quantity <= 0: | ||||||
|             raise ValidationError({'quantity': _('Quantity must be greater than zero')}) |             raise ValidationError({'quantity': _('Quantity must be greater than zero')}) | ||||||
|  |  | ||||||
| @@ -1140,7 +1146,9 @@ class Build( | |||||||
|  |  | ||||||
|         if quantity < output.quantity: |         if quantity < output.quantity: | ||||||
|             # Split output into two items |             # Split output into two items | ||||||
|             output = output.splitStock(quantity, location=location, user=user) |             output = output.splitStock( | ||||||
|  |                 quantity, location=location, user=user, allow_production=True | ||||||
|  |             ) | ||||||
|             output.build = self |             output.build = self | ||||||
|  |  | ||||||
|         # Update build output item |         # Update build output item | ||||||
| @@ -1171,20 +1179,29 @@ class Build( | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     @transaction.atomic |     @transaction.atomic | ||||||
|     def complete_build_output(self, output, user, **kwargs): |     def complete_build_output( | ||||||
|  |         self, | ||||||
|  |         output: stock.models.StockItem, | ||||||
|  |         user: User, | ||||||
|  |         quantity: Optional[decimal.Decimal] = None, | ||||||
|  |         **kwargs, | ||||||
|  |     ): | ||||||
|         """Complete a particular build output. |         """Complete a particular build output. | ||||||
|  |  | ||||||
|         - Remove allocated StockItems |         Arguments: | ||||||
|         - Mark the output as complete |             output: The StockItem instance (build output) to complete | ||||||
|  |             user: The user who is completing the build output | ||||||
|  |             quantity: The quantity to complete (defaults to entire output quantity) | ||||||
|  |  | ||||||
|  |         Notes: | ||||||
|  |             - Remove allocated StockItems | ||||||
|  |             - Mark the output as complete | ||||||
|         """ |         """ | ||||||
|         # Select the location for the build output |         # Select the location for the build output | ||||||
|         location = kwargs.get('location', self.destination) |         location = kwargs.get('location', self.destination) | ||||||
|         status = kwargs.get('status', StockStatus.OK.value) |         status = kwargs.get('status', StockStatus.OK.value) | ||||||
|         notes = kwargs.get('notes', '') |         notes = kwargs.get('notes', '') | ||||||
|  |  | ||||||
|         # List the allocated BuildItem objects for the given output |  | ||||||
|         allocated_items = output.items_to_install.all() |  | ||||||
|  |  | ||||||
|         required_tests = kwargs.get('required_tests', output.part.getRequiredTests()) |         required_tests = kwargs.get('required_tests', output.part.getRequiredTests()) | ||||||
|         prevent_on_incomplete = kwargs.get( |         prevent_on_incomplete = kwargs.get( | ||||||
|             'prevent_on_incomplete', |             'prevent_on_incomplete', | ||||||
| @@ -1201,6 +1218,30 @@ class Build( | |||||||
|  |  | ||||||
|             raise ValidationError(msg) |             raise ValidationError(msg) | ||||||
|  |  | ||||||
|  |         # List the allocated BuildItem objects for the given output | ||||||
|  |         allocated_items = output.items_to_install.all() | ||||||
|  |  | ||||||
|  |         # If a partial quantity is provided, split the stock output | ||||||
|  |         if quantity is not None and quantity != output.quantity: | ||||||
|  |             # Cannot split a build output with allocated items | ||||||
|  |             if allocated_items.count() > 0: | ||||||
|  |                 raise ValidationError( | ||||||
|  |                     _('Cannot partially complete a build output with allocated items') | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |             if quantity <= 0: | ||||||
|  |                 raise ValidationError({ | ||||||
|  |                     'quantity': _('Quantity must be greater than zero') | ||||||
|  |                 }) | ||||||
|  |  | ||||||
|  |             if quantity > output.quantity: | ||||||
|  |                 raise ValidationError({ | ||||||
|  |                     'quantity': _('Quantity cannot be greater than the output quantity') | ||||||
|  |                 }) | ||||||
|  |  | ||||||
|  |             # Split the stock item | ||||||
|  |             output = output.splitStock(quantity, user=user, allow_production=True) | ||||||
|  |  | ||||||
|         for build_item in allocated_items: |         for build_item in allocated_items: | ||||||
|             # Complete the allocation of stock for that item |             # Complete the allocation of stock for that item | ||||||
|             build_item.complete_allocation(user=user) |             build_item.complete_allocation(user=user) | ||||||
|   | |||||||
| @@ -270,7 +270,7 @@ class BuildOutputQuantitySerializer(BuildOutputSerializer): | |||||||
|         max_digits=15, |         max_digits=15, | ||||||
|         decimal_places=5, |         decimal_places=5, | ||||||
|         min_value=Decimal(0), |         min_value=Decimal(0), | ||||||
|         required=True, |         required=False, | ||||||
|         label=_('Quantity'), |         label=_('Quantity'), | ||||||
|         help_text=_('Enter quantity for build output'), |         help_text=_('Enter quantity for build output'), | ||||||
|     ) |     ) | ||||||
| @@ -282,13 +282,16 @@ class BuildOutputQuantitySerializer(BuildOutputSerializer): | |||||||
|         output = data.get('output') |         output = data.get('output') | ||||||
|         quantity = data.get('quantity') |         quantity = data.get('quantity') | ||||||
|  |  | ||||||
|         if quantity <= 0: |         if quantity is not None: | ||||||
|             raise ValidationError({'quantity': _('Quantity must be greater than zero')}) |             if quantity <= 0: | ||||||
|  |                 raise ValidationError({ | ||||||
|  |                     'quantity': _('Quantity must be greater than zero') | ||||||
|  |                 }) | ||||||
|  |  | ||||||
|         if quantity > output.quantity: |             if quantity > output.quantity: | ||||||
|             raise ValidationError({ |                 raise ValidationError({ | ||||||
|                 'quantity': _('Quantity cannot be greater than the output quantity') |                     'quantity': _('Quantity cannot be greater than the output quantity') | ||||||
|             }) |                 }) | ||||||
|  |  | ||||||
|         return data |         return data | ||||||
|  |  | ||||||
| @@ -533,7 +536,7 @@ class BuildOutputScrapSerializer(serializers.Serializer): | |||||||
|         with transaction.atomic(): |         with transaction.atomic(): | ||||||
|             for item in outputs: |             for item in outputs: | ||||||
|                 output = item['output'] |                 output = item['output'] | ||||||
|                 quantity = item['quantity'] |                 quantity = item.get('quantity', None) | ||||||
|                 build.scrap_build_output( |                 build.scrap_build_output( | ||||||
|                     output, |                     output, | ||||||
|                     quantity, |                     quantity, | ||||||
| @@ -558,7 +561,7 @@ class BuildOutputCompleteSerializer(serializers.Serializer): | |||||||
|             'notes', |             'notes', | ||||||
|         ] |         ] | ||||||
|  |  | ||||||
|     outputs = BuildOutputSerializer(many=True, required=True) |     outputs = BuildOutputQuantitySerializer(many=True, required=True) | ||||||
|  |  | ||||||
|     location = serializers.PrimaryKeyRelatedField( |     location = serializers.PrimaryKeyRelatedField( | ||||||
|         queryset=StockLocation.objects.all(), |         queryset=StockLocation.objects.all(), | ||||||
| @@ -637,10 +640,12 @@ class BuildOutputCompleteSerializer(serializers.Serializer): | |||||||
|         with transaction.atomic(): |         with transaction.atomic(): | ||||||
|             for item in outputs: |             for item in outputs: | ||||||
|                 output = item['output'] |                 output = item['output'] | ||||||
|  |                 quantity = item.get('quantity', None) | ||||||
|  |  | ||||||
|                 build.complete_build_output( |                 build.complete_build_output( | ||||||
|                     output, |                     output, | ||||||
|                     request.user if request else None, |                     request.user if request else None, | ||||||
|  |                     quantity=quantity, | ||||||
|                     location=location, |                     location=location, | ||||||
|                     status=status, |                     status=status, | ||||||
|                     notes=notes, |                     notes=notes, | ||||||
|   | |||||||
| @@ -1349,6 +1349,77 @@ class BuildOutputScrapTest(BuildAPITest): | |||||||
|             self.assertEqual(output.status, StockStatus.REJECTED) |             self.assertEqual(output.status, StockStatus.REJECTED) | ||||||
|             self.assertFalse(output.is_building) |             self.assertFalse(output.is_building) | ||||||
|  |  | ||||||
|  |     def test_partial_scrap(self): | ||||||
|  |         """Test partial scrapping of a build output.""" | ||||||
|  |         # Create a build output | ||||||
|  |         build = Build.objects.get(pk=1) | ||||||
|  |         output = build.create_build_output(10).first() | ||||||
|  |  | ||||||
|  |         self.assertEqual(build.build_outputs.count(), 1) | ||||||
|  |  | ||||||
|  |         data = { | ||||||
|  |             'outputs': [{'output': output.pk, 'quantity': 3}], | ||||||
|  |             'location': 1, | ||||||
|  |             'notes': 'Invalid scrap', | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         # Ensure that an invalid quantity raises an error | ||||||
|  |         for q in [-3, 0, 99]: | ||||||
|  |             data['outputs'][0]['quantity'] = q | ||||||
|  |             self.scrap(build.pk, data, expected_code=400) | ||||||
|  |  | ||||||
|  |         # Partially scrap the output (with a valid quantity) | ||||||
|  |         data['outputs'][0]['quantity'] = 3 | ||||||
|  |         self.scrap(build.pk, data) | ||||||
|  |  | ||||||
|  |         self.assertEqual(build.build_outputs.count(), 2) | ||||||
|  |         output.refresh_from_db() | ||||||
|  |         self.assertEqual(output.quantity, 7) | ||||||
|  |         self.assertTrue(output.is_building) | ||||||
|  |  | ||||||
|  |         scrapped = output.children.first() | ||||||
|  |         self.assertEqual(scrapped.quantity, 3) | ||||||
|  |         self.assertEqual(scrapped.status, StockStatus.REJECTED) | ||||||
|  |         self.assertFalse(scrapped.is_building) | ||||||
|  |  | ||||||
|  |     def test_partial_complete(self): | ||||||
|  |         """Test partial completion of a build output.""" | ||||||
|  |         build = Build.objects.get(pk=1) | ||||||
|  |         output = build.create_build_output(10).first() | ||||||
|  |         self.assertEqual(build.build_outputs.count(), 1) | ||||||
|  |         self.assertEqual(output.quantity, 10) | ||||||
|  |         self.assertTrue(output.is_building) | ||||||
|  |         self.assertEqual(build.completed, 0) | ||||||
|  |  | ||||||
|  |         url = reverse('api-build-output-complete', kwargs={'pk': build.pk}) | ||||||
|  |  | ||||||
|  |         data = { | ||||||
|  |             'outputs': [{'output': output.pk, 'quantity': 4}], | ||||||
|  |             'location': 1, | ||||||
|  |             'notes': 'Partial complete', | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         # Ensure that an invalid quantity raises an error | ||||||
|  |         for q in [-4, 0, 999]: | ||||||
|  |             data['outputs'][0]['quantity'] = q | ||||||
|  |             self.post(url, data, expected_code=400) | ||||||
|  |  | ||||||
|  |         # Partially complete the output (with a valid quantity) | ||||||
|  |         data['outputs'][0]['quantity'] = 4 | ||||||
|  |         self.post(url, data, expected_code=201) | ||||||
|  |  | ||||||
|  |         build.refresh_from_db() | ||||||
|  |         output.refresh_from_db() | ||||||
|  |         self.assertEqual(build.completed, 4) | ||||||
|  |         self.assertEqual(build.build_outputs.count(), 2) | ||||||
|  |         self.assertEqual(output.quantity, 6) | ||||||
|  |         self.assertTrue(output.is_building) | ||||||
|  |  | ||||||
|  |         completed_output = output.children.first() | ||||||
|  |         self.assertEqual(completed_output.quantity, 4) | ||||||
|  |         self.assertEqual(completed_output.status, StockStatus.OK) | ||||||
|  |         self.assertFalse(completed_output.is_building) | ||||||
|  |  | ||||||
|  |  | ||||||
| class BuildLineTests(BuildAPITest): | class BuildLineTests(BuildAPITest): | ||||||
|     """Unit tests for the BuildLine API endpoints.""" |     """Unit tests for the BuildLine API endpoints.""" | ||||||
|   | |||||||
| @@ -58,6 +58,12 @@ from part.models import Part | |||||||
| from users.models import Owner | from users.models import Owner | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GeneralExtraLineListOutputOptions(OutputConfiguration): | ||||||
|  |     """Output options for the GeneralExtraLineList endpoint.""" | ||||||
|  |  | ||||||
|  |     OPTIONS = [InvenTreeOutputOption('order_detail')] | ||||||
|  |  | ||||||
|  |  | ||||||
| class GeneralExtraLineList(SerializerContextMixin, DataExportViewMixin): | class GeneralExtraLineList(SerializerContextMixin, DataExportViewMixin): | ||||||
|     """General template for ExtraLine API classes.""" |     """General template for ExtraLine API classes.""" | ||||||
|  |  | ||||||
| @@ -69,6 +75,8 @@ class GeneralExtraLineList(SerializerContextMixin, DataExportViewMixin): | |||||||
|  |  | ||||||
|         return queryset |         return queryset | ||||||
|  |  | ||||||
|  |     output_options = GeneralExtraLineListOutputOptions | ||||||
|  |  | ||||||
|     filter_backends = SEARCH_ORDER_FILTER |     filter_backends = SEARCH_ORDER_FILTER | ||||||
|  |  | ||||||
|     ordering_fields = ['quantity', 'notes', 'reference'] |     ordering_fields = ['quantity', 'notes', 'reference'] | ||||||
| @@ -733,7 +741,9 @@ class PurchaseOrderLineItemDetail( | |||||||
|     output_options = PurchaseOrderLineItemOutputOptions |     output_options = PurchaseOrderLineItemOutputOptions | ||||||
|  |  | ||||||
|  |  | ||||||
| class PurchaseOrderExtraLineList(GeneralExtraLineList, ListCreateAPI): | class PurchaseOrderExtraLineList( | ||||||
|  |     GeneralExtraLineList, OutputOptionsMixin, ListCreateAPI | ||||||
|  | ): | ||||||
|     """API endpoint for accessing a list of PurchaseOrderExtraLine objects.""" |     """API endpoint for accessing a list of PurchaseOrderExtraLine objects.""" | ||||||
|  |  | ||||||
|     queryset = models.PurchaseOrderExtraLine.objects.all() |     queryset = models.PurchaseOrderExtraLine.objects.all() | ||||||
| @@ -1070,7 +1080,7 @@ class SalesOrderLineItemDetail( | |||||||
|     output_options = SalesOrderLineItemOutputOptions |     output_options = SalesOrderLineItemOutputOptions | ||||||
|  |  | ||||||
|  |  | ||||||
| class SalesOrderExtraLineList(GeneralExtraLineList, ListCreateAPI): | class SalesOrderExtraLineList(GeneralExtraLineList, OutputOptionsMixin, ListCreateAPI): | ||||||
|     """API endpoint for accessing a list of SalesOrderExtraLine objects.""" |     """API endpoint for accessing a list of SalesOrderExtraLine objects.""" | ||||||
|  |  | ||||||
|     queryset = models.SalesOrderExtraLine.objects.all() |     queryset = models.SalesOrderExtraLine.objects.all() | ||||||
| @@ -1660,7 +1670,7 @@ class ReturnOrderLineItemDetail( | |||||||
|     output_options = ReturnOrderLineItemOutputOptions |     output_options = ReturnOrderLineItemOutputOptions | ||||||
|  |  | ||||||
|  |  | ||||||
| class ReturnOrderExtraLineList(GeneralExtraLineList, ListCreateAPI): | class ReturnOrderExtraLineList(GeneralExtraLineList, OutputOptionsMixin, ListCreateAPI): | ||||||
|     """API endpoint for accessing a list of ReturnOrderExtraLine objects.""" |     """API endpoint for accessing a list of ReturnOrderExtraLine objects.""" | ||||||
|  |  | ||||||
|     queryset = models.ReturnOrderExtraLine.objects.all() |     queryset = models.ReturnOrderExtraLine.objects.all() | ||||||
|   | |||||||
| @@ -1365,11 +1365,25 @@ class PartParameterTemplateDetail(PartParameterTemplateMixin, RetrieveUpdateDest | |||||||
|     """API endpoint for accessing the detail view for a PartParameterTemplate object.""" |     """API endpoint for accessing the detail view for a PartParameterTemplate object.""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PartParameterOutputOptions(OutputConfiguration): | ||||||
|  |     """Output options for the PartParameter endpoints.""" | ||||||
|  |  | ||||||
|  |     OPTIONS = [ | ||||||
|  |         InvenTreeOutputOption('part_detail'), | ||||||
|  |         InvenTreeOutputOption( | ||||||
|  |             'template_detail', | ||||||
|  |             default=True, | ||||||
|  |             description='Include detailed information about the part parameter template.', | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |  | ||||||
| class PartParameterAPIMixin: | class PartParameterAPIMixin: | ||||||
|     """Mixin class for PartParameter API endpoints.""" |     """Mixin class for PartParameter API endpoints.""" | ||||||
|  |  | ||||||
|     queryset = PartParameter.objects.all() |     queryset = PartParameter.objects.all() | ||||||
|     serializer_class = part_serializers.PartParameterSerializer |     serializer_class = part_serializers.PartParameterSerializer | ||||||
|  |     output_options = PartParameterOutputOptions | ||||||
|  |  | ||||||
|     def get_queryset(self, *args, **kwargs): |     def get_queryset(self, *args, **kwargs): | ||||||
|         """Override get_queryset method to prefetch related fields.""" |         """Override get_queryset method to prefetch related fields.""" | ||||||
| @@ -1414,7 +1428,9 @@ class PartParameterFilter(FilterSet): | |||||||
|             return queryset.filter(part=part) |             return queryset.filter(part=part) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PartParameterList(PartParameterAPIMixin, DataExportViewMixin, ListCreateAPI): | class PartParameterList( | ||||||
|  |     PartParameterAPIMixin, OutputOptionsMixin, DataExportViewMixin, ListCreateAPI | ||||||
|  | ): | ||||||
|     """API endpoint for accessing a list of PartParameter objects. |     """API endpoint for accessing a list of PartParameter objects. | ||||||
|  |  | ||||||
|     - GET: Return list of PartParameter objects |     - GET: Return list of PartParameter objects | ||||||
| @@ -1442,7 +1458,9 @@ class PartParameterList(PartParameterAPIMixin, DataExportViewMixin, ListCreateAP | |||||||
|     ] |     ] | ||||||
|  |  | ||||||
|  |  | ||||||
| class PartParameterDetail(PartParameterAPIMixin, RetrieveUpdateDestroyAPI): | class PartParameterDetail( | ||||||
|  |     PartParameterAPIMixin, OutputOptionsMixin, RetrieveUpdateDestroyAPI | ||||||
|  | ): | ||||||
|     """API endpoint for detail view of a single PartParameter object.""" |     """API endpoint for detail view of a single PartParameter object.""" | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2209,6 +2209,7 @@ class StockItem( | |||||||
|             batch: If provided, override the batch (default = existing batch) |             batch: If provided, override the batch (default = existing batch) | ||||||
|             status: If provided, override the status (default = existing status) |             status: If provided, override the status (default = existing status) | ||||||
|             packaging: If provided, override the packaging (default = existing packaging) |             packaging: If provided, override the packaging (default = existing packaging) | ||||||
|  |             allow_production: If True, allow splitting of stock which is in production (default = False) | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             The new StockItem object |             The new StockItem object | ||||||
| @@ -2221,8 +2222,10 @@ class StockItem( | |||||||
|         """ |         """ | ||||||
|         # Run initial checks to test if the stock item can actually be "split" |         # Run initial checks to test if the stock item can actually be "split" | ||||||
|  |  | ||||||
|  |         allow_production = kwargs.get('allow_production', False) | ||||||
|  |  | ||||||
|         # Cannot split a stock item which is in production |         # Cannot split a stock item which is in production | ||||||
|         if self.is_building: |         if self.is_building and not allow_production: | ||||||
|             raise ValidationError(_('Stock item is currently in production')) |             raise ValidationError(_('Stock item is currently in production')) | ||||||
|  |  | ||||||
|         notes = kwargs.get('notes', '') |         notes = kwargs.get('notes', '') | ||||||
|   | |||||||
| @@ -51,7 +51,11 @@ export function BuiltinQueryCountWidgets(): DashboardWidgetProps[] { | |||||||
|       label: 'low-stk', |       label: 'low-stk', | ||||||
|       description: t`Show the number of parts which are low on stock`, |       description: t`Show the number of parts which are low on stock`, | ||||||
|       modelType: ModelType.part, |       modelType: ModelType.part, | ||||||
|       params: { low_stock: true, active: true } |       params: { | ||||||
|  |         active: true, | ||||||
|  |         low_stock: true, | ||||||
|  |         virtual: false | ||||||
|  |       } | ||||||
|     }), |     }), | ||||||
|     QueryCountDashboardWidget({ |     QueryCountDashboardWidget({ | ||||||
|       title: t`Required for Build Orders`, |       title: t`Required for Build Orders`, | ||||||
|   | |||||||
| @@ -236,7 +236,7 @@ function BuildOutputFormRow({ | |||||||
|   props: TableFieldRowProps; |   props: TableFieldRowProps; | ||||||
|   record: any; |   record: any; | ||||||
| }>) { | }>) { | ||||||
|   const serial = useMemo(() => { |   const stockItemColumn = useMemo(() => { | ||||||
|     if (record.serial) { |     if (record.serial) { | ||||||
|       return `# ${record.serial}`; |       return `# ${record.serial}`; | ||||||
|     } else { |     } else { | ||||||
| @@ -244,15 +244,39 @@ function BuildOutputFormRow({ | |||||||
|     } |     } | ||||||
|   }, [record]); |   }, [record]); | ||||||
|  |  | ||||||
|  |   const quantityColumn = useMemo(() => { | ||||||
|  |     // Serialized output - quantity cannot be changed | ||||||
|  |     if (record.serial) { | ||||||
|  |       return '1'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Non-serialized output - quantity can be changed | ||||||
|  |     return ( | ||||||
|  |       <StandaloneField | ||||||
|  |         fieldName='quantity' | ||||||
|  |         fieldDefinition={{ | ||||||
|  |           field_type: 'number', | ||||||
|  |           required: true, | ||||||
|  |           value: props.item.quantity, | ||||||
|  |           onValueChange: (value: any) => { | ||||||
|  |             props.changeFn(props.idx, 'quantity', value); | ||||||
|  |           } | ||||||
|  |         }} | ||||||
|  |         error={props.rowErrors?.quantity?.message} | ||||||
|  |       /> | ||||||
|  |     ); | ||||||
|  |   }, [props, record]); | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <> |     <> | ||||||
|       <Table.Tr> |       <Table.Tr> | ||||||
|         <Table.Td> |         <Table.Td> | ||||||
|           <RenderPartColumn part={record.part_detail} /> |           <RenderPartColumn part={record.part_detail} /> | ||||||
|         </Table.Td> |         </Table.Td> | ||||||
|  |         <Table.Td>{stockItemColumn}</Table.Td> | ||||||
|         <Table.Td> |         <Table.Td> | ||||||
|           <TableFieldErrorWrapper props={props} errorKey='output'> |           <TableFieldErrorWrapper props={props} errorKey='output'> | ||||||
|             {serial} |             {quantityColumn} | ||||||
|           </TableFieldErrorWrapper> |           </TableFieldErrorWrapper> | ||||||
|         </Table.Td> |         </Table.Td> | ||||||
|         <Table.Td>{record.batch}</Table.Td> |         <Table.Td>{record.batch}</Table.Td> | ||||||
| @@ -297,7 +321,8 @@ export function useCompleteBuildOutputsForm({ | |||||||
|         field_type: 'table', |         field_type: 'table', | ||||||
|         value: outputs.map((output: any) => { |         value: outputs.map((output: any) => { | ||||||
|           return { |           return { | ||||||
|             output: output.pk |             output: output.pk, | ||||||
|  |             quantity: output.quantity | ||||||
|           }; |           }; | ||||||
|         }), |         }), | ||||||
|         modelRenderer: (row: TableFieldRowProps) => { |         modelRenderer: (row: TableFieldRowProps) => { | ||||||
| @@ -309,6 +334,7 @@ export function useCompleteBuildOutputsForm({ | |||||||
|         headers: [ |         headers: [ | ||||||
|           { title: t`Part` }, |           { title: t`Part` }, | ||||||
|           { title: t`Build Output` }, |           { title: t`Build Output` }, | ||||||
|  |           { title: t`Quantity to Complete`, style: { width: '200px' } }, | ||||||
|           { title: t`Batch` }, |           { title: t`Batch` }, | ||||||
|           { title: t`Status` }, |           { title: t`Status` }, | ||||||
|           { title: '', style: { width: '50px' } } |           { title: '', style: { width: '50px' } } | ||||||
| @@ -382,7 +408,8 @@ export function useScrapBuildOutputsForm({ | |||||||
|         }, |         }, | ||||||
|         headers: [ |         headers: [ | ||||||
|           { title: t`Part` }, |           { title: t`Part` }, | ||||||
|           { title: t`Stock Item` }, |           { title: t`Build Output` }, | ||||||
|  |           { title: t`Quantity to Scrap`, style: { width: '200px' } }, | ||||||
|           { title: t`Batch` }, |           { title: t`Batch` }, | ||||||
|           { title: t`Status` }, |           { title: t`Status` }, | ||||||
|           { title: '', style: { width: '50px' } } |           { title: '', style: { width: '50px' } } | ||||||
|   | |||||||
| @@ -206,7 +206,8 @@ test('Build Order - Build Outputs', async ({ browser }) => { | |||||||
|  |  | ||||||
|   await page.getByLabel('text-field-batch_code').fill('BATCH12345'); |   await page.getByLabel('text-field-batch_code').fill('BATCH12345'); | ||||||
|   await page.getByLabel('related-field-location').click(); |   await page.getByLabel('related-field-location').click(); | ||||||
|   await page.getByText('Reel Storage').click(); |   await page.getByLabel('related-field-location').fill('Reel'); | ||||||
|  |   await page.getByText('- Electronics Lab/Reel Storage').click(); | ||||||
|   await page.getByRole('button', { name: 'Submit' }).click(); |   await page.getByRole('button', { name: 'Submit' }).click(); | ||||||
|  |  | ||||||
|   // Should be an error as the number of serial numbers doesn't match the quantity |   // Should be an error as the number of serial numbers doesn't match the quantity | ||||||
| @@ -246,6 +247,20 @@ test('Build Order - Build Outputs', async ({ browser }) => { | |||||||
|   await page.waitForTimeout(250); |   await page.waitForTimeout(250); | ||||||
|   await page.getByRole('button', { name: 'Submit' }).click(); |   await page.getByRole('button', { name: 'Submit' }).click(); | ||||||
|   await page.getByText('Build outputs have been completed').waitFor(); |   await page.getByText('Build outputs have been completed').waitFor(); | ||||||
|  |  | ||||||
|  |   // Check for expected UI elements in the "scrap output" dialog | ||||||
|  |   const cell3 = await page.getByRole('cell', { name: '16' }); | ||||||
|  |   const row3 = await getRowFromCell(cell3); | ||||||
|  |   await row3.getByLabel(/row-action-menu-/i).click(); | ||||||
|  |   await page.getByRole('menuitem', { name: 'Scrap' }).click(); | ||||||
|  |  | ||||||
|  |   await page | ||||||
|  |     .getByText( | ||||||
|  |       'Selected build outputs will be completed, but marked as scrapped' | ||||||
|  |     ) | ||||||
|  |     .waitFor(); | ||||||
|  |   await page.getByRole('cell', { name: 'Quantity: 16' }).waitFor(); | ||||||
|  |   await page.getByRole('button', { name: 'Cancel', exact: true }).click(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| test('Build Order - Allocation', async ({ browser }) => { | test('Build Order - Allocation', async ({ browser }) => { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user