factory/apps/inm/views.py

277 lines
9.8 KiB
Python

from django.shortcuts import render
from rest_framework.mixins import ListModelMixin, DestroyModelMixin
from rest_framework.exceptions import ParseError, PermissionDenied
from rest_framework.decorators import action
from django.db import transaction
from rest_framework import serializers
from django.utils import timezone
from rest_framework.response import Response
from django.db.models import Sum
from apps.inm.models import WareHouse, MaterialBatch, MIO, MIOItem
from apps.inm.serializers import (
MaterialBatchSerializer, WareHourseSerializer, MIOListSerializer, MIOItemSerializer, MioItemAnaSerializer,
MIODoSerializer, MIOSaleSerializer, MIOPurSerializer, MIOOtherSerializer, MIOItemCreateSerializer, MaterialBatchDetailSerializer, MIODetailSerializer, MIOItemTestSerializer, MIOItemPurInTestSerializer)
from apps.utils.viewsets import CustomGenericViewSet, CustomModelViewSet
from apps.inm.services import InmService
from apps.utils.mixins import BulkCreateModelMixin, BulkDestroyModelMixin, BulkUpdateModelMixin
from apps.utils.permission import has_perm
from .filters import MaterialBatchFilter
# Create your views here.
class WarehouseVIewSet(CustomModelViewSet):
"""
list: 仓库信息
仓库信息
"""
queryset = WareHouse.objects.all()
serializer_class = WareHourseSerializer
search_fields = ['name', 'number', 'place']
ordering = ['create_time']
def perform_destroy(self, instance):
if MaterialBatch.objects.filter(warehouse=instance).exclude(count=0).exists():
raise ParseError('该仓库存在物料')
instance.delete()
class MaterialBatchViewSet(ListModelMixin, CustomGenericViewSet):
"""
list: 物料批次
物料批次
"""
perms_map = {'get': '*'}
queryset = MaterialBatch.objects.all()
serializer_class = MaterialBatchSerializer
retrieve_serializer_class = MaterialBatchDetailSerializer
select_related_fields = ['warehouse', 'material']
filterset_class = MaterialBatchFilter
search_fields = ['material__name', 'material__number',
'material__model', 'material__specification', 'batch']
class MioDoViewSet(BulkCreateModelMixin, BulkUpdateModelMixin, CustomGenericViewSet):
perms_map = {'post': '*', 'put': 'mio.do'}
queryset = MIO.objects.filter(
type__in=[MIO.MIO_TYPE_DO_IN, MIO.MIO_TYPE_DO_OUT])
serializer_class = MIODoSerializer
def create(self, request, *args, **kwargs):
"""
生产领料/入库
生产领料/入库
"""
return super().create(request, *args, **kwargs)
class MioSaleViewSet(BulkCreateModelMixin, BulkUpdateModelMixin, CustomGenericViewSet):
perms_map = {'post': '*', 'put': 'mio.sale'}
queryset = MIO.objects.filter(
type__in=[MIO.MIO_TYPE_DO_IN, MIO.MIO_TYPE_DO_OUT])
serializer_class = MIOSaleSerializer
def create(self, request, *args, **kwargs):
"""
销售出库
销售出库
"""
return super().create(request, *args, **kwargs)
class MioPurViewSet(BulkCreateModelMixin, BulkUpdateModelMixin, CustomGenericViewSet):
perms_map = {'post': '*', 'put': 'mio.pur'}
queryset = MIO.objects.filter(
type__in=[MIO.MIO_TYPE_PUR_IN])
serializer_class = MIOPurSerializer
def create(self, request, *args, **kwargs):
"""
采购入库
采购入库
"""
return super().create(request, *args, **kwargs)
class MioOtherViewSet(BulkCreateModelMixin, BulkUpdateModelMixin, CustomGenericViewSet):
perms_map = {'post': '*', 'put': 'mio.other'}
queryset = MIO.objects.filter(
type__in=[MIO.MIO_TYPE_OTHER_OUT, MIO.MIO_TYPE_OTHER_IN])
serializer_class = MIOOtherSerializer
def create(self, request, *args, **kwargs):
"""
其他出入库
其他出入库
"""
return super().create(request, *args, **kwargs)
class MIOViewSet(CustomModelViewSet):
"""
list: 出入库记录
出入库记录
"""
queryset = MIO.objects.all()
select_related_fields = ['create_by', 'belong_dept', 'do_user',
'submit_user', 'supplier', 'order', 'customer', 'pu_order']
serializer_class = MIOListSerializer
retrieve_serializer_class = MIODetailSerializer
filterset_fields = {
'state': ["exact", "in"],
"type": ["exact", "in"],
"pu_order": ["exact"],
"order": ["exact"]
}
search_fields = ['number']
data_filter = True
def get_serializer_class(self):
if self.action in ['create', 'update', 'partial_update']:
type = self.request.data.get('type')
user = self.request.user
if type in [MIO.MIO_TYPE_DO_IN, MIO.MIO_TYPE_DO_OUT]:
if not has_perm(user, ['mio.do']):
raise PermissionDenied
return MIODoSerializer
elif type in [MIO.MIO_TYPE_OTHER_IN, MIO.MIO_TYPE_OTHER_OUT]:
if not has_perm(user, ['mio.other']):
raise PermissionDenied
return MIOOtherSerializer
elif type == MIO.MIO_TYPE_SALE_OUT:
if not has_perm(user, ['mio.sale']):
raise PermissionDenied
return MIOSaleSerializer
elif type == MIO.MIO_TYPE_PUR_IN:
if not has_perm(user, ['mio.pur']):
raise PermissionDenied
return MIOPurSerializer
return super().get_serializer_class()
def perform_destroy(self, instance):
if instance.state != MIO.MIO_CREATE:
raise ParseError('非创建中不可删除')
return super().perform_destroy(instance)
@action(methods=['post'], detail=True, perms_map={'post': 'mio.submit'}, serializer_class=serializers.Serializer)
@transaction.atomic
def submit(self, request, *args, **kwargs):
"""提交
提交
"""
ins = self.get_object()
if ins.inout_date is None:
raise ParseError('出入库日期未填写')
if ins.state != MIO.MIO_CREATE:
raise ParseError('记录状态异常')
ins.submit_time = timezone.now()
ins.state = MIO.MIO_SUBMITED
ins.submit_user = request.user
ins.save()
InmService.update_inm(ins)
return Response(MIOListSerializer(instance=ins).data)
class MIOItemViewSet(ListModelMixin, BulkCreateModelMixin, BulkDestroyModelMixin, CustomGenericViewSet):
"""
list: 出入库明细
出入库明细
"""
perms_map = {'get': '*', 'post': '*', 'delete': '*'}
queryset = MIOItem.objects.all()
serializer_class = MIOItemSerializer
create_serializer_class = MIOItemCreateSerializer
select_related_fields = ['warehouse', 'mio', 'material']
filterset_fields = {
"warehouse": ["exact"],
"mio": ["exact"],
"mio__state": ["exact"],
"mio__type": ["exact", "in"],
"material": ["exact"],
"test_date": ["isnull", "exact"]
}
ordering = ['create_time']
def perform_destroy(self, instance):
if instance.mio.state != MIO.MIO_CREATE:
raise ParseError('出入库记录非创建中不可删除')
if has_perm(self.request.user, ['mio.update']) is False and instance.mio.create_by != self.request.user:
raise PermissionDenied('无权限删除')
return super().perform_destroy(instance)
@action(methods=['post'], detail=True, perms_map={'post': 'mioitem.test'}, serializer_class=MIOItemTestSerializer)
@transaction.atomic
def test(self, request, *args, **kwargs):
"""半成品检验
半成品检验
"""
ins: MIOItem = self.get_object()
if ins.test_date:
raise ParseError('该明细已检验')
sr = MIOItemTestSerializer(instance=ins, data=request.data)
sr.is_valid(raise_exception=True)
sr.save()
# 开始变动库存
count_notok = ins.count_notok
batch = ins.batch
material = ins.material
warehouse = ins.warehouse
mb = MaterialBatch.objects.get(
material=material, batch=batch, warehouse=warehouse)
count_new = mb.count - count_notok
if count_new < 0:
raise ParseError('库存扣减失败,请确认!')
mb.count = count_new
mb.save()
return Response()
@action(methods=['post'], detail=True, perms_map={'post': 'mioitem.test'}, serializer_class=MIOItemPurInTestSerializer)
@transaction.atomic
def test_pur_in(self, request, *args, **kwargs):
"""入厂检验
入厂检验
"""
ins: MIOItem = self.get_object()
if ins.test_date:
raise ParseError('该明细已检验')
sr = MIOItemPurInTestSerializer(instance=ins, data=request.data)
sr.is_valid(raise_exception=True)
sr.save()
return Response()
@action(methods=['post'], detail=False, perms_map={'post': '*'}, serializer_class=MioItemAnaSerializer)
def sale_out_ana(self, request, *args, **kwargs):
"""交付统计数据
交付统计数据
"""
sr = MIOItemTestSerializer(data=request.data)
sr.is_valid(raise_exception=True)
vdata = sr.validated_data
mioitems = MIOItem.objects.filter(
mio__type='sale_out', mio__state=MIO.MIO_SUBMITED)
if vdata.get('material_cate'):
mioitems = mioitems.filter(
material__cate=vdata['material_cate'])
if vdata.get('start_date', ''):
mioitems = mioitems.filter(
mio__inout_date__gte=vdata['start_date'])
if vdata.get('end_date', ''):
mioitems = mioitems.filter(mio__inout_date__lte=vdata['end_date'])
res = mioitems.aggregate(count=Sum('count'))
for i in res:
if res[i] is None:
res[i] = 0
return Response(res)