| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049 |
- from rest_framework import viewsets
- from utils.page import MyPageNumberPagination
- from utils.datasolve import sumOfList, transportation_calculate
- from utils.md5 import Md5
- from rest_framework.filters import OrderingFilter
- from django_filters.rest_framework import DjangoFilterBackend
- from django.db import transaction
- from rest_framework.response import Response
- from rest_framework.exceptions import APIException
- from django.utils import timezone
- from django.db.models import Sum
- from .models import BoundListModel, BoundDetailModel,BoundBatchModel, BatchOperateLogModel, OutBatchModel,OutBoundDetailModel,MaterialStatistics,OutBoundDemandModel
- # from .files import FileListRenderCN, FileDetailRenderCN
- from .serializers import BoundListGetSerializer,BoundListPostSerializer,BoundBatchGetSerializer,BoundBatchPostSerializer,BoundDetailGetSerializer,BoundDetailPostSerializer
- from .serializers import OutBoundDetailGetSerializer,OutBoundDetailPostSerializer,OutBatchGetSerializer,OutBatchPostSerializer,BatchLogGetSerializer
- from .serializers import MaterialStatisticsSerializer,MaterialStatisticsSerializer_items
- from .serializers import OutBoundDemandModelSerializer
- from .filter import BoundListFilter, BoundDetailFilter,BoundBatchFilter
- from .filter import OutBatchFilter,OutBoundDetailFilter,BatchlogFilter
- from .filter import MaterialStatisticsFilter
- from .filter import OutBoundDemandFilter
- # 以后添加模块检验
- from warehouse.models import ListModel as warehouse
- from staff.models import ListModel as staff
- from rest_framework.permissions import AllowAny
- from rest_framework.views import APIView
- # 出库需求视图类
- class OutBoundDemandViewSet(viewsets.ModelViewSet):
- """
- retrieve:
- Response a data list(get)
- list:
- Response a data list(all)
- create:
- Create a data line(post)
- delete:
- Delete a data line(delete)
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
-
- pagination_class = MyPageNumberPagination
- filter_backends = [DjangoFilterBackend, OrderingFilter, ]
- ordering_fields = ['id', "create_time", "update_time", ]
- filter_class = OutBoundDemandFilter
- def get_project(self):
- try:
- id = self.kwargs.get('pk')
- return id
- except:
- return None
- def get_queryset(self):
- id = self.get_project()
- if self.request.user:
- if id is None:
- return OutBoundDemandModel.objects.filter( is_delete=False)
- else:
- return OutBoundDemandModel.objects.filter( id=id, is_delete=False)
- else:
- return OutBoundDemandModel.objects.none()
- def get_serializer_class(self):
- if self.action in ['list' ]:
- return OutBoundDemandModelSerializer
- else:
- return OutBoundDemandModelSerializer
-
- def batch_list(self, request):
- data =self.request.data
- OutBoundDemand_all = OutBoundDemandModel.objects.filter(bound_list_id=data['bound_list_id'], is_delete=False).all()
- data = OutBoundDemandModelSerializer(OutBoundDemand_all, many=True).data
- return_data ={
- "code": 200,
- "msg": "Success Create",
- "data": data
- }
-
- return Response(return_data,status=200,headers={})
-
- def create(self, request, *args, **kwargs):
- data = self.request.data
- data['openid'] = self.request.auth.openid
- data['create_time'] = str(timezone.now().strftime('%Y-%m-%d %H:%M:%S'))
- data['update_time'] = str(timezone.now().strftime('%Y-%m-%d %H:%M:%S'))
- data['working'] = True
- bound_list_obj = BoundListModel.objects.get(id=data['bound_list_id'])
- OutBoundDemand_obj =OutBoundDemandModel.objects.create(
- bound_list=bound_list_obj,
- goods_code=data['goods_code'],
- goods_desc=data['goods_desc'],
- goods_std=data['goods_std'],
- goods_unit=data['goods_unit'],
- goods_qty=data['goods_out_qty'],
- out_type = data['out_type'],
- creater=data['creater'],
- create_time=data['create_time'],
- update_time=data['update_time'],
- working=data['working']
- )
- return_data = OutBoundDemandModelSerializer(OutBoundDemand_obj).data
- headers = self.get_success_headers(return_data)
- return Response(return_data, status=200, headers=headers)
-
- def batch_demanded_list(self, request):
- data =self.request.data
- OutBoundDemand_all = OutBatchModel.objects.filter(bound_list_id=data['bound_list_id'], is_delete=False).all()
- data = OutBatchGetSerializer(OutBoundDemand_all, many=True).data
- return_data = {
- "code": 200,
- "msg": "Success Create",
- "data": data
- }
- return Response(return_data,status=200,headers={})
- def distribute(self, request):
- """主分配入口"""
- try:
- with transaction.atomic():
- bound_list_id, demands,out_type, creater= self.validate_distribute_request(request)
- if OutBatchModel.objects.filter(bound_list_id=bound_list_id, is_delete=False).exists():
- return_data = {
- "code": 200,
- "msg": "Success Create",
- "data": {
- "msg": "该订单已分配,请勿重复分配"
- }
- }
- return Response(return_data, status=200, headers={})
- aggregated_demands = self.aggregate_demands(demands)
- result = self.process_all_goods(aggregated_demands, request, out_type, creater,bound_list_id)
- return self.build_success_response(result)
- except APIException as e:
- return self.build_error_response(e.detail, 200)
- except Exception as e:
- return self.build_error_response(str(e), 200)
- # 验证层方法
- def validate_distribute_request(self, request):
- """验证请求参数"""
- bound_list_id = request.data.get('bound_list_id')
- if not bound_list_id:
- raise APIException({"detail": "Missing bound_list_id"})
- demands = OutBoundDemandModel.objects.filter(
- bound_list_id=bound_list_id,
- is_delete=False
- )
- if not demands.exists():
- raise APIException({"detail": "No demands found"})
- base_info = OutBoundDemandModel.objects.filter(
- bound_list_id=bound_list_id,
- is_delete=False
- ).first()
- out_type = base_info.out_type
- creater = base_info.creater
- return bound_list_id, demands, out_type, creater
- # 数据处理层方法
- def aggregate_demands(self, demands):
- """合并相同物料需求"""
- return demands.values('goods_code').annotate(
- total_demand=Sum('goods_qty')
- )
- # 核心分配逻辑
- def process_all_goods(self, aggregated_demands, request, out_type, creater,bound_list_id):
- """处理所有物料分配"""
- return [
- self.process_single_goods(
- goods_code=item['goods_code'],
- total_demand=item['total_demand'],
- request=request,
- out_type=out_type,
- creater=creater,
- bound_list_id=bound_list_id
- )
- for item in aggregated_demands
- ]
- def process_single_goods(self, goods_code, total_demand, request,out_type, creater,bound_list_id):
- """处理单个物料分配"""
- batches = self.get_available_batches(goods_code)
- remaining, allocations = self.allocate_batches(total_demand, batches, request,out_type, creater,bound_list_id)
-
- if remaining > 0:
- raise APIException({
- "detail": f"Insufficient stock for {goods_code}",
- "required": total_demand,
- "allocated": total_demand - remaining
- })
-
- return {
- "goods_code": goods_code,
- "total_demand": total_demand,
- "allocations": allocations
- }
- def get_available_batches(self, goods_code):
- """获取可用入库批次"""
- return BoundBatchModel.objects.filter(
- goods_code=goods_code,
- is_delete=False
- ).order_by('bound_batch_order')
- # 批次分配逻辑
- def allocate_batches(self, total_demand, batches, request,out_type, creater,bound_list_id):
- """分配具体批次"""
- remaining = total_demand
- allocations = []
-
- for batch in batches:
- if remaining <= 0:
- break
-
- allocated = self.allocate_single_batch(
- batch=batch,
- remaining=remaining,
- request=request,
- out_type=out_type,
- creater=creater,
- bound_list_id=bound_list_id
- )
-
- if allocated == 0:
- continue
-
- allocations.append(allocated)
- remaining -= allocated['allocated']
-
- return remaining, allocations
- def allocate_single_batch(self, batch, remaining, request,out_type, creater,bound_list_id):
- """单个批次分配逻辑"""
- available = batch.goods_in_location_qty - batch.goods_out_qty
- if available <= 0:
- return 0
- allocate_qty = min(remaining, available)
- self.update_batch_status(batch, allocate_qty)
- out_batch = self.create_out_batch(batch, allocate_qty, request,out_type, creater,bound_list_id)
-
- return {
- "batch": batch.bound_number,
- "allocated": allocate_qty,
- "out_batch": out_batch.out_number
- }
- # 数据操作层方法
- def update_batch_status(self, batch, allocate_qty):
- """更新批次状态和数量"""
- batch.goods_out_qty += allocate_qty
-
- if batch.goods_out_qty == batch.goods_in_location_qty:
- batch.status = 6 # 已出库
- elif batch.goods_out_qty > 0:
- batch.status = 5 # 部分出库
-
- batch.save()
- def create_out_batch(self, batch, allocate_qty, request,out_type, creater,bound_list_id):
- """创建出库记录"""
- out_data = {
- 'bound_list': bound_list_id,
- 'out_number': self.generate_out_number(batch.goods_code),
- 'batch_number': batch.id,
- 'out_date': timezone.now(),
- 'warehouse_code': batch.warehouse_code,
- 'warehouse_name': batch.warehouse_name,
- 'goods_code': batch.goods_code,
- 'goods_desc': batch.goods_desc,
- 'goods_out_qty': allocate_qty,
- 'status': 0,
- 'openid': request.auth.openid,
- 'out_type': out_type,
- 'creater': creater,
- }
-
- serializer = OutBatchPostSerializer(data=out_data)
- if not serializer.is_valid():
- raise APIException({
- "detail": f"Serialization error for {batch.goods_code}",
- "errors": serializer.errors
- })
-
- return serializer.save()
- # 工具方法
- def generate_out_number(self, goods_code):
- """生成唯一出库单号"""
- timestamp = timezone.now().strftime("%Y%m%d%H%M%S")
- import uuid
- return f"OUT-{goods_code}-{timestamp}-{uuid.uuid4().hex[:6]}"
- def build_success_response(self, data):
- """构建成功响应"""
- return Response({
- "code": 200,
- "msg": "Distribution completed",
- "data": data
- })
- def build_error_response(self, error, status_code):
- """构建错误响应"""
- return Response({
- "code": status_code,
- "msg": "Distribution failed" if status_code == 400 else "Server error",
- "error": error
- }, status=status_code)
- # 物料统计视图类
- class MaterialStatisticsViewSet(viewsets.ModelViewSet):
- """
- retrieve:
- Response a data list(get)
- list:
- Response a data list(all)
- create:
- Create a data line(post)
- delete:
- Delete a data line(delete)
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
-
- pagination_class = MyPageNumberPagination
- filter_backends = [DjangoFilterBackend, OrderingFilter, ]
- ordering_fields = ['id', "create_time", "update_time", ]
- filter_class = MaterialStatisticsFilter
- def get_project(self):
- try:
- id = self.kwargs.get('pk')
- return id
- except:
- return None
- def get_queryset(self):
- id = self.get_project()
- if self.request.user:
- if id is None:
- return MaterialStatistics.objects.filter()
- else:
- return MaterialStatistics.objects.filter(id=id)
- else:
- return MaterialStatistics.objects.none()
- def get_serializer_class(self):
- if self.action in ['list' ]:
- return MaterialStatisticsSerializer
- elif self.action in ['retrieve']:
- return MaterialStatisticsSerializer_items
- else:
- return self.http_method_not_allowed(request=self.request)
- # 汇报单类视图
- class BoundListViewSet(viewsets.ModelViewSet):
- """
- retrieve:
- Response a data list(get)
- list:
- Response a data list(all)
- create:
- Create a data line(post)
- delete:
- Delete a data line(delete)
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
-
- pagination_class = MyPageNumberPagination
- filter_backends = [DjangoFilterBackend, OrderingFilter, ]
- ordering_fields = ['id', "create_time", "update_time", ]
- filter_class = BoundListFilter
- def get_project(self):
- try:
- id = self.kwargs.get('pk')
- return id
- except:
- return None
- def get_queryset(self):
- id = self.get_project()
- if self.request.user:
- if id is None:
- return BoundListModel.objects.filter( is_delete=False)
- else:
- return BoundListModel.objects.filter( id=id, is_delete=False)
- else:
- return BoundListModel.objects.none()
- def get_serializer_class(self):
- if self.action in ['list', 'destroy','retrieve']:
- return BoundListGetSerializer
- elif self.action in ['create', 'update']:
- return BoundListPostSerializer
- else:
- return self.http_method_not_allowed(request=self.request)
- def create(self, request, *args, **kwargs):
- data = self.request.data
- # if BoundListModel.objects.filter(code=data['code'], is_delete=False).exists():
- # raise APIException({"detail": "Data exists"})
- # else:
- data['openid'] = self.request.auth.openid
- data['bound_date'] =str(timezone.now().strftime('%Y-%m-%d'))
- order_day=str(timezone.now().strftime('%Y-%m-'))
- data['bound_month'] =str(timezone.now().strftime('%Y%m'))
- if data['bound_type'] == 'in':
- data['bound_status'] = '100'
- else:
- data['bound_status'] = '200'
-
- qs_set = BoundListModel.objects.filter(bound_month=data['bound_month'], bound_code_type=data['bound_code_type'], is_delete=False)
- print('qs_set是:', len(qs_set))
- if len(qs_set) > 0:
- bound_last_code = qs_set.order_by('-id').first().bound_code
- data['bound_code'] = data['bound_code_type'] +'-'+ order_day + str(int(bound_last_code.split('-')[-1])+1).zfill(4)
- else:
- data['bound_code'] = data['bound_code_type'] +'-'+ order_day + '0001'
- serializer = self.get_serializer(data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
-
- def update(self, request, pk):
- qs = self.get_object()
- data = self.request.data
- serializer = self.get_serializer(qs, data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- def destroy(self, request, pk):
- qs = self.get_object()
- # if qs.openid != self.request.auth.openid:
- # raise APIException({"detail": "该入库非您所属,禁止删除,您可以进行编辑"})
- # else:
- qs.is_delete = True
- qs.bound_code =qs.bound_code+'-delete'+str(timezone.now().strftime('%Y%m%d%H%M%S'))
- qs.save()
- if qs.bound_type == 'in':
- BoundDetailModel.objects.filter(bound_list=qs.id).update(is_delete=True)
- if qs.relate_bill:
- qs.relate_bill.bound_status = 0
- qs.relate_bill.save()
- else:
- OutBoundDemandModel.objects.filter(bound_list=qs.id).update(is_delete=True)
- if qs.relate_out_bill:
- qs.relate_out_bill.bound_status = 0
- qs.relate_out_bill.save()
- serializer = self.get_serializer(qs, many=False)
- headers = self.get_success_headers(serializer.data)
-
- return Response(serializer.data, status=200, headers=headers)
- # 入库批次类视图
- class BoundBatchViewSet(viewsets.ModelViewSet):
- """
- retrieve:
- Response a data list(get)
- list:
- Response a data list(all)
- create:
- Create a data line(post)
- delete:
- Delete a data line(delete)
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
- pagination_class = MyPageNumberPagination
- filter_backends = [DjangoFilterBackend, OrderingFilter, ]
- ordering_fields = ['id', "create_time", "update_time", ]
- filter_class = BoundBatchFilter
- def get_project(self):
- try:
- id = self.kwargs.get('pk')
- return id
- except:
- return None
- def get_queryset(self):
- id = self.get_project()
- if self.request.user:
- if id is None:
- return BoundBatchModel.objects.filter( is_delete=False)
- else:
- return BoundBatchModel.objects.filter( id=id, is_delete=False)
- else:
- return BoundBatchModel.objects.none()
- def get_serializer_class(self):
- if self.action in ['list', 'destroy','retrieve']:
- return BoundBatchGetSerializer
- elif self.action in ['create', 'update']:
- return BoundBatchPostSerializer
- else:
- return self.http_method_not_allowed(request=self.request)
- def create(self, request, *args, **kwargs):
- data = self.request.data
- try:
- data['openid'] = self.request.auth.openid
- data['goods_total_weight'] = data['goods_weight']*data['goods_qty']
-
- order_day=str(timezone.now().strftime('-%Y%m'))
- order_month=str(timezone.now().strftime('%Y%m'))
- data['bound_month'] =str(timezone.now().strftime('%Y%m'))
- print(data['order'])
- if data['order'] == 'true':
- qs_set = BoundBatchModel.objects.filter( goods_code=data['goods_code'], bound_month=order_month, is_delete=False)
- print('qs_set是:', len(qs_set))
- if len(qs_set) > 0:
- bound_last_code = qs_set.order_by('-bound_batch_order').first().bound_number
- data['bound_batch_order'] = int(bound_last_code.split('-')[-1])+1
- data['bound_number'] = data['goods_code'] + '-' + str(int(bound_last_code.split('-')[-1])+1)
- else:
- data['bound_batch_order'] = int(order_day.split('-')[-1])*1000 +1
- data['bound_number'] = data['goods_code'] + order_day + '001'
- else:
- data['bound_number'] = data['goods_code'] + '-' + str(data['bound_batch_order'])
- serializer = self.get_serializer(data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- headers = self.get_success_headers(serializer.data)
- self.add_batch_log(serializer.data, 0, data['goods_qty'])
- return Response(serializer.data, status=200, headers=headers)
- except Exception as e:
- print(e)
- raise APIException({"detail": "{}".format(e)})
-
- def add_batch_log(self, data, log_type, goods_qty):
- choices_dict = dict(BatchOperateLogModel.BATCH_LOG_TYPE)
- log_type_name = choices_dict.get(log_type, "未知类型")
-
- try:
- # 获取 BoundBatchModel 实例
- batch_obj = BoundBatchModel.objects.get(id=data['id'])
- except BoundBatchModel.DoesNotExist:
- return False
- log_data = {
- 'batch_id': batch_obj,
- 'log_type': log_type,
- 'log_date': timezone.now().strftime('%Y-%m-%d-%H:%M'), # 直接格式化时间,无需转字符串
- 'goods_code': data['goods_code'],
- 'goods_desc': data['goods_desc'],
- 'goods_qty': data['goods_qty'],
- 'log_content': f"{log_type_name} {data['goods_qty']}件",
- 'creater': data['creater'],
- 'openid': data['openid'],
- 'is_delete': False,
-
- }
-
- # 创建日志记录
- BatchOperateLogModel.objects.create(**log_data)
- return True
- def update(self, request, pk):
- qs = self.get_object()
- data = self.request.data
- serializer = self.get_serializer(qs, data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- def destroy(self, request, pk):
- qs = self.get_object()
- if qs.openid != self.request.auth.openid:
- raise APIException({"detail": "该入库非您所属,禁止删除,您可以进行编辑"})
- else:
- qs.is_delete = True
- qs.save()
- serializer = self.get_serializer(qs, many=False)
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- def batchinout(self, request, *args, **kwargs):
- from container.models import ContainerDetailModel, ContainerListModel
- from decimal import Decimal
-
- batch_number = request.data.get('batch_number')
- batch_in_qty = request.data.get('batch_in_qty')
-
- if not batch_number:
- return Response({"code": 200,"message": "批次号不能为空","data": "批次进出库数目:{batch_in_qty}"}, status=200)
-
- try:
- batch_obj = BoundBatchModel.objects.get(bound_number=batch_number, is_delete=False)
- except BoundBatchModel.DoesNotExist:
- return Response({"code": 200,"message": "批次号不存在","data": "批次进出库数目:{batch_in_qty}"}, status=200)
-
- if batch_in_qty:
- # 获取或创建虚拟容器
- virtual_container, created = ContainerListModel.objects.get_or_create(
- container_code=1
-
- )
-
- # 使用 update_or_create 避免并发问题
- virtual_container_detail = ContainerDetailModel.objects.create(
- month=timezone.now().strftime('%Y%m'),
- container= virtual_container,
- batch= batch_obj,
- goods_class= 1, # 成品
- goods_code= batch_obj.goods_code,
- goods_desc= batch_obj.goods_desc,
- goods_qty= Decimal(batch_in_qty),
- goods_out_qty= Decimal(0),
- goods_weight= batch_obj.goods_weight,
- status= 2, # 在库
- creater= "system",
- create_time= timezone.now(),
- update_time= timezone.now(),
- is_delete= False )
- virtual_container_detail.save()
- virtual_container_detail.goods_out_qty = batch_in_qty
- virtual_container_detail.save()
-
- return Response({"code": 200,"message": "操作成功","data": "批次进出库数目:{batch_in_qty}"}, status=200)
-
- # 入库明细类视图
- class BoundDetailViewSet(viewsets.ModelViewSet):
- """
- retrieve:
- Response a data list(get)
- list:
- Response a data list(all)
- create:
- Create a data line(post)
- delete:
- Delete a data line(delete)
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
- pagination_class = MyPageNumberPagination
- filter_backends = [DjangoFilterBackend, OrderingFilter, ]
- ordering_fields = ['id', "create_time", "update_time", ]
- filter_class = BoundDetailFilter
- def get_project(self):
- try:
- id = self.kwargs.get('pk')
- return id
- except:
- return None
-
- def get_queryset(self):
- id = self.get_project()
- if self.request.user:
- if id is None:
- return BoundDetailModel.objects.filter( is_delete=False)
- else:
- return BoundDetailModel.objects.filter( id=id, is_delete=False)
- else:
- return BoundDetailModel.objects.none()
- def get_serializer_class(self):
- if self.action in ['list', 'destroy','retrieve']:
- return BoundDetailGetSerializer
- elif self.action in ['create', 'update']:
- return BoundDetailPostSerializer
- else:
- return self.http_method_not_allowed(request=self.request)
- def create(self, request, *args, **kwargs):
- data = self.request.data
- data['openid'] = self.request.auth.openid
- data.setdefault('is_delete', False)
- # 验证并保存数据
- data['detail_code'] = f"DC-{data['bound_list']:02}{data['bound_batch']:02}"
- print(data['detail_code'])
- if BoundDetailModel.objects.filter(detail_code=data['detail_code'], is_delete=False).exists():
- raise APIException({"detail": "Data exists"})
- else:
- serializer = self.get_serializer(data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- # 返回响应
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
-
- def update(self, request, pk):
- qs = self.get_object()
- data = self.request.data
- serializer = self.get_serializer(qs, data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- def destroy(self, request, pk):
- qs = self.get_object()
- if qs.openid != self.request.auth.openid:
- raise APIException({"detail": "该入库非您所属,禁止删除,您可以进行编辑"})
- else:
- qs.is_delete = True
- qs.save()
- serializer = self.get_serializer(qs, many=False)
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- # 出库明细类视图
- class OutBoundDetailViewSet(viewsets.ModelViewSet):
- """
- retrieve:
- Response a data list(get)
- list: Response a data list(all)
- create:
- Create a data line(post)
- delete:
- Delete a data line(delete)
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
- pagination_class = MyPageNumberPagination
- filter_backends = [DjangoFilterBackend, OrderingFilter, ]
- ordering_fields = ['id', "create_time", "update_time", ]
- filter_class = OutBoundDetailFilter
- def get_project(self):
- try:
- id = self.kwargs.get('pk')
- return id
- except:
- return None
- def get_queryset(self):
- id = self.get_project()
- if self.request.user:
- if id is None:
- return OutBoundDetailModel.objects.filter( is_delete=False)
- else:
- return OutBoundDetailModel.objects.filter( id=id, is_delete=False)
- else:
- return OutBoundDetailModel.objects.none()
- def get_serializer_class(self):
- if self.action in ['list', 'destroy','retrieve']:
- return OutBoundDetailGetSerializer
- elif self.action in ['create', 'update']:
- return OutBoundDetailPostSerializer
- else:
- return self.http_method_not_allowed(request=self.request)
- def create(self, request, *args, **kwargs):
- data = self.request.data
- data['openid'] = self.request.auth.openid
- data.setdefault('is_delete', False)
- data['bound_batch_number'] = OutBatchModel.objects.get(id=data['bound_batch']).batch_number.id
- # 验证并保存数据
- data['detail_code'] = f"DC-{data['bound_list']:02}{data['bound_batch']:02}"
- print(data['detail_code'])
- if OutBoundDetailModel.objects.filter(detail_code=data['detail_code'], is_delete=False).exists():
- raise APIException({"detail": "Data exists"})
- else:
- serializer = self.get_serializer(data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- # 返回响应
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- def update(self, request, pk):
- qs = self.get_object()
- data = self.request.data
- serializer = self.get_serializer(qs, data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- def destroy(self, request, pk):
- qs = self.get_object()
- if qs.openid != self.request.auth.openid:
- raise APIException({"detail": "该入库非您所属,禁止删除,您可以进行编辑"})
- else:
- qs.is_delete = True
- qs.save()
- serializer = self.get_serializer(qs, many=False)
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- # 出库批次类视图
- class OutBoundBatchViewSet(viewsets.ModelViewSet):
- """
- retrieve:
- Response a data list(get)
- list:
- Response a data list(all)
- create:
- Create a data line(post)
- delete:
- Delete a data line(delete)
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
- pagination_class = MyPageNumberPagination
- filter_backends = [DjangoFilterBackend, OrderingFilter, ]
- ordering_fields = ['id', "create_time", "update_time", ]
- filter_class = OutBatchFilter
- def get_project(self):
- try:
- id = self.kwargs.get('pk')
- return id
- except:
- return None
- def get_queryset(self):
- id = self.get_project()
- if self.request.user:
- if id is None:
- return OutBatchModel.objects.filter( is_delete=False)
- else:
- return OutBatchModel.objects.filter( id=id, is_delete=False)
- else:
- return OutBatchModel.objects.none()
- def get_serializer_class(self):
- if self.action in ['list', 'destroy','retrieve']:
- return OutBatchGetSerializer
- elif self.action in ['create', 'update']:
- return OutBatchPostSerializer
- else:
- return self.http_method_not_allowed(request=self.request)
- def create(self, request, *args, **kwargs):
- data = self.request.data
- batch_obj = BoundBatchModel.objects.get(bound_number=data['out_number'])
- if batch_obj is None:
- raise APIException({"detail": "批次不存在"})
- bound_obj = BoundListModel.objects.get(id=data['bound_list_id'])
- data['bound_list'] = bound_obj.id
- data['batch_number'] = batch_obj.id
- data['out_date'] = str(timezone.now().strftime('%Y-%m-%d %H:%M:%S'))
- data['openid'] = self.request.auth.openid
- data.setdefault('is_delete', False)
- data['goods_total_weight'] = data['goods_weight']*data['goods_out_qty']
- from decimal import Decimal
- # data['goods_out_qty'] 是一个字符串或浮点数
- data['goods_out_qty'] = Decimal(str(data['goods_out_qty']))
- # 现在 data['goods_out_qty'] 是 Decimal 类型,可以进行减法操作
- from decimal import Decimal
- # 假设 data['goods_out_qty'] 是一个 float 类型的数值
- data['goods_out_qty'] = Decimal(str(data['goods_out_qty']))
- # 现在所有数值都是 Decimal 类型,可以安全地进行减法运算
- data['goods_qty'] = batch_obj.goods_qty - batch_obj.goods_reserve_qty - data['goods_out_qty']
- # data['goods_qty'] = batch_obj.goods_qty - batch_obj.goods_reserve_qty - data['goods_out_qty']
- data['status'] = 0 #现在处于出库申请状态
-
- serializer = self.get_serializer(data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- headers = self.get_success_headers(serializer.data)
- self.add_batch_log(serializer.data, 1, data['goods_out_qty'])
-
- return Response(serializer.data, status=200, headers=headers)
- def update(self, request, pk):
- qs = self.get_object()
- data = self.request.data
- data['openid'] = self.request.auth.openid
- data.setdefault('is_delete', False)
- data['goods_total_weight'] = data['goods_weight']*data['goods_qty']
- serializer = self.get_serializer(qs, data=data)
- serializer.is_valid(raise_exception=True)
- serializer.save()
- headers = self.get_success_headers(serializer.data)
- self.add_batch_log(serializer.data, 1, data['goods_qty'])
- return Response(serializer.data, status=200, headers=headers)
- def destroy(self, request, pk):
- qs = self.get_object()
- if qs.openid != self.request.auth.openid:
- raise APIException({"detail": "该出库非您所属,禁止删除,您可以进行编辑"})
- else:
- qs.is_delete = True
- qs.save()
- serializer = self.get_serializer(qs, many=False)
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- def add_batch_log(self, data, log_type, goods_qty):
- choices_dict = dict(BatchOperateLogModel.BATCH_LOG_TYPE)
- log_type_name = choices_dict.get(log_type, "未知类型")
- try:
- # 获取 BoundBatchModel 实例
- batch_obj = BoundBatchModel.objects.get(id=data['batch_number'])
- except BoundBatchModel.DoesNotExist:
- # 处理批次不存在的情况(如记录日志或抛出异常)
- return False
- log_data = {
- 'batch_id': batch_obj, # 关键修复:传入实例对象,而不是 batch_obj.id
- 'log_type': log_type,
- 'log_date': timezone.now().strftime('%Y-%m-%d-%H:%M'), # 直接格式化时间,无需转字符串
- 'goods_code': data['goods_code'],
- 'goods_desc': data['goods_desc'],
- 'goods_qty': data['goods_qty'],
- 'log_content': f"{log_type_name} {goods_qty}件",
- 'creater': data['creater'],
- 'openid': data['openid'],
- 'is_delete': False,
- # 注意:create_time 和 update_time 由模型的 auto_now_add 和 auto_now 自动处理,无需手动赋值
- }
- # 创建日志记录
- BatchOperateLogModel.objects.create(**log_data)
- return True
- # 批次操作日志类视图
- class BoundBatchLogViewSet(viewsets.ModelViewSet):
-
- """
- retrieve:
- Response a data list(get)
- list:
- Response a data list(all)
-
- delete:
- Delete a data line(delete)
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
- pagination_class = MyPageNumberPagination
- filter_backends = [DjangoFilterBackend, OrderingFilter, ]
- ordering_fields = ['id', "create_time", "update_time", ]
- filter_class = BatchlogFilter
- def get_queryset(self):
- return BatchOperateLogModel.objects.filter( is_delete=False)
-
- def get_serializer_class(self):
- if self.action in ['list', 'destroy','retrieve']:
- return BatchLogGetSerializer
- else:
- return self.http_method_not_allowed(request=self.request)
- def destroy(self, request, pk):
- qs = self.get_object()
- qs.is_delete = True
- qs.save()
- serializer = self.get_serializer(qs, many=False)
- headers = self.get_success_headers(serializer.data)
- return Response(serializer.data, status=200, headers=headers)
- # 托盘类视图
- class BatchContainerAPIView(APIView):
- """
- post:
- 返回批次对应的container列表
- """
- # authentication_classes = [] # 禁用所有认证类
- # permission_classes = [AllowAny] # 允许任意访问
- def post(self, request):
- data = request.data
- batch_id = data.get('batch_id')
- from container.models import ContainerDetailModel
- from bin.models import LocationContainerLink
- container_detail_all = ContainerDetailModel.objects.filter(batch_id=batch_id, is_delete=False).exclude(status=3).all()
- container_dict = {}
- for container_detail in container_detail_all:
- container_id = container_detail.container_id
- link_obj = LocationContainerLink.objects.filter(container_id=container_id,is_active=True).first()
- if container_id not in container_dict:
- container_dict[container_id] = {
- 'id': container_id,
- 'goods_code': container_detail.goods_code,
- 'goods_desc': container_detail.goods_desc,
- 'container_code': container_detail.container.container_code,
- 'current_location': container_detail.container.current_location,
- 'goods_qty': container_detail.goods_qty-container_detail.goods_out_qty,
- 'class':container_detail.goods_class,
- 'location_c_number' : link_obj.location.c_number if link_obj else None,
- 'location_max_capacity': link_obj.location.max_capacity if link_obj else None,
- 'location_group': link_obj.location.location_group if link_obj else None,
- 'location_code': link_obj.location.location_code if link_obj else None,
- }
- else:
- container_dict[container_id]['goods_qty'] += container_detail.goods_qty-container_detail.goods_out_qty
- container_dict = list(container_dict.values())
- return_data = {'code': 200,'msg': 'Success Create', 'data': container_dict}
- return Response(return_data)
|