views.py 40 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049
  1. from rest_framework import viewsets
  2. from utils.page import MyPageNumberPagination
  3. from utils.datasolve import sumOfList, transportation_calculate
  4. from utils.md5 import Md5
  5. from rest_framework.filters import OrderingFilter
  6. from django_filters.rest_framework import DjangoFilterBackend
  7. from django.db import transaction
  8. from rest_framework.response import Response
  9. from rest_framework.exceptions import APIException
  10. from django.utils import timezone
  11. from django.db.models import Sum
  12. from .models import BoundListModel, BoundDetailModel,BoundBatchModel, BatchOperateLogModel, OutBatchModel,OutBoundDetailModel,MaterialStatistics,OutBoundDemandModel
  13. # from .files import FileListRenderCN, FileDetailRenderCN
  14. from .serializers import BoundListGetSerializer,BoundListPostSerializer,BoundBatchGetSerializer,BoundBatchPostSerializer,BoundDetailGetSerializer,BoundDetailPostSerializer
  15. from .serializers import OutBoundDetailGetSerializer,OutBoundDetailPostSerializer,OutBatchGetSerializer,OutBatchPostSerializer,BatchLogGetSerializer
  16. from .serializers import MaterialStatisticsSerializer,MaterialStatisticsSerializer_items
  17. from .serializers import OutBoundDemandModelSerializer
  18. from .filter import BoundListFilter, BoundDetailFilter,BoundBatchFilter
  19. from .filter import OutBatchFilter,OutBoundDetailFilter,BatchlogFilter
  20. from .filter import MaterialStatisticsFilter
  21. from .filter import OutBoundDemandFilter
  22. # 以后添加模块检验
  23. from warehouse.models import ListModel as warehouse
  24. from staff.models import ListModel as staff
  25. from rest_framework.permissions import AllowAny
  26. from rest_framework.views import APIView
  27. # 出库需求视图类
  28. class OutBoundDemandViewSet(viewsets.ModelViewSet):
  29. """
  30. retrieve:
  31. Response a data list(get)
  32. list:
  33. Response a data list(all)
  34. create:
  35. Create a data line(post)
  36. delete:
  37. Delete a data line(delete)
  38. """
  39. # authentication_classes = [] # 禁用所有认证类
  40. # permission_classes = [AllowAny] # 允许任意访问
  41. pagination_class = MyPageNumberPagination
  42. filter_backends = [DjangoFilterBackend, OrderingFilter, ]
  43. ordering_fields = ['id', "create_time", "update_time", ]
  44. filter_class = OutBoundDemandFilter
  45. def get_project(self):
  46. try:
  47. id = self.kwargs.get('pk')
  48. return id
  49. except:
  50. return None
  51. def get_queryset(self):
  52. id = self.get_project()
  53. if self.request.user:
  54. if id is None:
  55. return OutBoundDemandModel.objects.filter( is_delete=False)
  56. else:
  57. return OutBoundDemandModel.objects.filter( id=id, is_delete=False)
  58. else:
  59. return OutBoundDemandModel.objects.none()
  60. def get_serializer_class(self):
  61. if self.action in ['list' ]:
  62. return OutBoundDemandModelSerializer
  63. else:
  64. return OutBoundDemandModelSerializer
  65. def batch_list(self, request):
  66. data =self.request.data
  67. OutBoundDemand_all = OutBoundDemandModel.objects.filter(bound_list_id=data['bound_list_id'], is_delete=False).all()
  68. data = OutBoundDemandModelSerializer(OutBoundDemand_all, many=True).data
  69. return_data ={
  70. "code": 200,
  71. "msg": "Success Create",
  72. "data": data
  73. }
  74. return Response(return_data,status=200,headers={})
  75. def create(self, request, *args, **kwargs):
  76. data = self.request.data
  77. data['openid'] = self.request.auth.openid
  78. data['create_time'] = str(timezone.now().strftime('%Y-%m-%d %H:%M:%S'))
  79. data['update_time'] = str(timezone.now().strftime('%Y-%m-%d %H:%M:%S'))
  80. data['working'] = True
  81. bound_list_obj = BoundListModel.objects.get(id=data['bound_list_id'])
  82. OutBoundDemand_obj =OutBoundDemandModel.objects.create(
  83. bound_list=bound_list_obj,
  84. goods_code=data['goods_code'],
  85. goods_desc=data['goods_desc'],
  86. goods_std=data['goods_std'],
  87. goods_unit=data['goods_unit'],
  88. goods_qty=data['goods_out_qty'],
  89. out_type = data['out_type'],
  90. creater=data['creater'],
  91. create_time=data['create_time'],
  92. update_time=data['update_time'],
  93. working=data['working']
  94. )
  95. return_data = OutBoundDemandModelSerializer(OutBoundDemand_obj).data
  96. headers = self.get_success_headers(return_data)
  97. return Response(return_data, status=200, headers=headers)
  98. def batch_demanded_list(self, request):
  99. data =self.request.data
  100. OutBoundDemand_all = OutBatchModel.objects.filter(bound_list_id=data['bound_list_id'], is_delete=False).all()
  101. data = OutBatchGetSerializer(OutBoundDemand_all, many=True).data
  102. return_data = {
  103. "code": 200,
  104. "msg": "Success Create",
  105. "data": data
  106. }
  107. return Response(return_data,status=200,headers={})
  108. def distribute(self, request):
  109. """主分配入口"""
  110. try:
  111. with transaction.atomic():
  112. bound_list_id, demands,out_type, creater= self.validate_distribute_request(request)
  113. if OutBatchModel.objects.filter(bound_list_id=bound_list_id, is_delete=False).exists():
  114. return_data = {
  115. "code": 200,
  116. "msg": "Success Create",
  117. "data": {
  118. "msg": "该订单已分配,请勿重复分配"
  119. }
  120. }
  121. return Response(return_data, status=200, headers={})
  122. aggregated_demands = self.aggregate_demands(demands)
  123. result = self.process_all_goods(aggregated_demands, request, out_type, creater,bound_list_id)
  124. return self.build_success_response(result)
  125. except APIException as e:
  126. return self.build_error_response(e.detail, 200)
  127. except Exception as e:
  128. return self.build_error_response(str(e), 200)
  129. # 验证层方法
  130. def validate_distribute_request(self, request):
  131. """验证请求参数"""
  132. bound_list_id = request.data.get('bound_list_id')
  133. if not bound_list_id:
  134. raise APIException({"detail": "Missing bound_list_id"})
  135. demands = OutBoundDemandModel.objects.filter(
  136. bound_list_id=bound_list_id,
  137. is_delete=False
  138. )
  139. if not demands.exists():
  140. raise APIException({"detail": "No demands found"})
  141. base_info = OutBoundDemandModel.objects.filter(
  142. bound_list_id=bound_list_id,
  143. is_delete=False
  144. ).first()
  145. out_type = base_info.out_type
  146. creater = base_info.creater
  147. return bound_list_id, demands, out_type, creater
  148. # 数据处理层方法
  149. def aggregate_demands(self, demands):
  150. """合并相同物料需求"""
  151. return demands.values('goods_code').annotate(
  152. total_demand=Sum('goods_qty')
  153. )
  154. # 核心分配逻辑
  155. def process_all_goods(self, aggregated_demands, request, out_type, creater,bound_list_id):
  156. """处理所有物料分配"""
  157. return [
  158. self.process_single_goods(
  159. goods_code=item['goods_code'],
  160. total_demand=item['total_demand'],
  161. request=request,
  162. out_type=out_type,
  163. creater=creater,
  164. bound_list_id=bound_list_id
  165. )
  166. for item in aggregated_demands
  167. ]
  168. def process_single_goods(self, goods_code, total_demand, request,out_type, creater,bound_list_id):
  169. """处理单个物料分配"""
  170. batches = self.get_available_batches(goods_code)
  171. remaining, allocations = self.allocate_batches(total_demand, batches, request,out_type, creater,bound_list_id)
  172. if remaining > 0:
  173. raise APIException({
  174. "detail": f"Insufficient stock for {goods_code}",
  175. "required": total_demand,
  176. "allocated": total_demand - remaining
  177. })
  178. return {
  179. "goods_code": goods_code,
  180. "total_demand": total_demand,
  181. "allocations": allocations
  182. }
  183. def get_available_batches(self, goods_code):
  184. """获取可用入库批次"""
  185. return BoundBatchModel.objects.filter(
  186. goods_code=goods_code,
  187. is_delete=False
  188. ).order_by('bound_batch_order')
  189. # 批次分配逻辑
  190. def allocate_batches(self, total_demand, batches, request,out_type, creater,bound_list_id):
  191. """分配具体批次"""
  192. remaining = total_demand
  193. allocations = []
  194. for batch in batches:
  195. if remaining <= 0:
  196. break
  197. allocated = self.allocate_single_batch(
  198. batch=batch,
  199. remaining=remaining,
  200. request=request,
  201. out_type=out_type,
  202. creater=creater,
  203. bound_list_id=bound_list_id
  204. )
  205. if allocated == 0:
  206. continue
  207. allocations.append(allocated)
  208. remaining -= allocated['allocated']
  209. return remaining, allocations
  210. def allocate_single_batch(self, batch, remaining, request,out_type, creater,bound_list_id):
  211. """单个批次分配逻辑"""
  212. available = batch.goods_in_location_qty - batch.goods_out_qty
  213. if available <= 0:
  214. return 0
  215. allocate_qty = min(remaining, available)
  216. self.update_batch_status(batch, allocate_qty)
  217. out_batch = self.create_out_batch(batch, allocate_qty, request,out_type, creater,bound_list_id)
  218. return {
  219. "batch": batch.bound_number,
  220. "allocated": allocate_qty,
  221. "out_batch": out_batch.out_number
  222. }
  223. # 数据操作层方法
  224. def update_batch_status(self, batch, allocate_qty):
  225. """更新批次状态和数量"""
  226. batch.goods_out_qty += allocate_qty
  227. if batch.goods_out_qty == batch.goods_in_location_qty:
  228. batch.status = 6 # 已出库
  229. elif batch.goods_out_qty > 0:
  230. batch.status = 5 # 部分出库
  231. batch.save()
  232. def create_out_batch(self, batch, allocate_qty, request,out_type, creater,bound_list_id):
  233. """创建出库记录"""
  234. out_data = {
  235. 'bound_list': bound_list_id,
  236. 'out_number': self.generate_out_number(batch.goods_code),
  237. 'batch_number': batch.id,
  238. 'out_date': timezone.now(),
  239. 'warehouse_code': batch.warehouse_code,
  240. 'warehouse_name': batch.warehouse_name,
  241. 'goods_code': batch.goods_code,
  242. 'goods_desc': batch.goods_desc,
  243. 'goods_out_qty': allocate_qty,
  244. 'status': 0,
  245. 'openid': request.auth.openid,
  246. 'out_type': out_type,
  247. 'creater': creater,
  248. }
  249. serializer = OutBatchPostSerializer(data=out_data)
  250. if not serializer.is_valid():
  251. raise APIException({
  252. "detail": f"Serialization error for {batch.goods_code}",
  253. "errors": serializer.errors
  254. })
  255. return serializer.save()
  256. # 工具方法
  257. def generate_out_number(self, goods_code):
  258. """生成唯一出库单号"""
  259. timestamp = timezone.now().strftime("%Y%m%d%H%M%S")
  260. import uuid
  261. return f"OUT-{goods_code}-{timestamp}-{uuid.uuid4().hex[:6]}"
  262. def build_success_response(self, data):
  263. """构建成功响应"""
  264. return Response({
  265. "code": 200,
  266. "msg": "Distribution completed",
  267. "data": data
  268. })
  269. def build_error_response(self, error, status_code):
  270. """构建错误响应"""
  271. return Response({
  272. "code": status_code,
  273. "msg": "Distribution failed" if status_code == 400 else "Server error",
  274. "error": error
  275. }, status=status_code)
  276. # 物料统计视图类
  277. class MaterialStatisticsViewSet(viewsets.ModelViewSet):
  278. """
  279. retrieve:
  280. Response a data list(get)
  281. list:
  282. Response a data list(all)
  283. create:
  284. Create a data line(post)
  285. delete:
  286. Delete a data line(delete)
  287. """
  288. # authentication_classes = [] # 禁用所有认证类
  289. # permission_classes = [AllowAny] # 允许任意访问
  290. pagination_class = MyPageNumberPagination
  291. filter_backends = [DjangoFilterBackend, OrderingFilter, ]
  292. ordering_fields = ['id', "create_time", "update_time", ]
  293. filter_class = MaterialStatisticsFilter
  294. def get_project(self):
  295. try:
  296. id = self.kwargs.get('pk')
  297. return id
  298. except:
  299. return None
  300. def get_queryset(self):
  301. id = self.get_project()
  302. if self.request.user:
  303. if id is None:
  304. return MaterialStatistics.objects.filter()
  305. else:
  306. return MaterialStatistics.objects.filter(id=id)
  307. else:
  308. return MaterialStatistics.objects.none()
  309. def get_serializer_class(self):
  310. if self.action in ['list' ]:
  311. return MaterialStatisticsSerializer
  312. elif self.action in ['retrieve']:
  313. return MaterialStatisticsSerializer_items
  314. else:
  315. return self.http_method_not_allowed(request=self.request)
  316. # 汇报单类视图
  317. class BoundListViewSet(viewsets.ModelViewSet):
  318. """
  319. retrieve:
  320. Response a data list(get)
  321. list:
  322. Response a data list(all)
  323. create:
  324. Create a data line(post)
  325. delete:
  326. Delete a data line(delete)
  327. """
  328. # authentication_classes = [] # 禁用所有认证类
  329. # permission_classes = [AllowAny] # 允许任意访问
  330. pagination_class = MyPageNumberPagination
  331. filter_backends = [DjangoFilterBackend, OrderingFilter, ]
  332. ordering_fields = ['id', "create_time", "update_time", ]
  333. filter_class = BoundListFilter
  334. def get_project(self):
  335. try:
  336. id = self.kwargs.get('pk')
  337. return id
  338. except:
  339. return None
  340. def get_queryset(self):
  341. id = self.get_project()
  342. if self.request.user:
  343. if id is None:
  344. return BoundListModel.objects.filter( is_delete=False)
  345. else:
  346. return BoundListModel.objects.filter( id=id, is_delete=False)
  347. else:
  348. return BoundListModel.objects.none()
  349. def get_serializer_class(self):
  350. if self.action in ['list', 'destroy','retrieve']:
  351. return BoundListGetSerializer
  352. elif self.action in ['create', 'update']:
  353. return BoundListPostSerializer
  354. else:
  355. return self.http_method_not_allowed(request=self.request)
  356. def create(self, request, *args, **kwargs):
  357. data = self.request.data
  358. # if BoundListModel.objects.filter(code=data['code'], is_delete=False).exists():
  359. # raise APIException({"detail": "Data exists"})
  360. # else:
  361. data['openid'] = self.request.auth.openid
  362. data['bound_date'] =str(timezone.now().strftime('%Y-%m-%d'))
  363. order_day=str(timezone.now().strftime('%Y-%m-'))
  364. data['bound_month'] =str(timezone.now().strftime('%Y%m'))
  365. if data['bound_type'] == 'in':
  366. data['bound_status'] = '100'
  367. else:
  368. data['bound_status'] = '200'
  369. qs_set = BoundListModel.objects.filter(bound_month=data['bound_month'], bound_code_type=data['bound_code_type'], is_delete=False)
  370. print('qs_set是:', len(qs_set))
  371. if len(qs_set) > 0:
  372. bound_last_code = qs_set.order_by('-id').first().bound_code
  373. data['bound_code'] = data['bound_code_type'] +'-'+ order_day + str(int(bound_last_code.split('-')[-1])+1).zfill(4)
  374. else:
  375. data['bound_code'] = data['bound_code_type'] +'-'+ order_day + '0001'
  376. serializer = self.get_serializer(data=data)
  377. serializer.is_valid(raise_exception=True)
  378. serializer.save()
  379. headers = self.get_success_headers(serializer.data)
  380. return Response(serializer.data, status=200, headers=headers)
  381. def update(self, request, pk):
  382. qs = self.get_object()
  383. data = self.request.data
  384. serializer = self.get_serializer(qs, data=data)
  385. serializer.is_valid(raise_exception=True)
  386. serializer.save()
  387. headers = self.get_success_headers(serializer.data)
  388. return Response(serializer.data, status=200, headers=headers)
  389. def destroy(self, request, pk):
  390. qs = self.get_object()
  391. # if qs.openid != self.request.auth.openid:
  392. # raise APIException({"detail": "该入库非您所属,禁止删除,您可以进行编辑"})
  393. # else:
  394. qs.is_delete = True
  395. qs.bound_code =qs.bound_code+'-delete'+str(timezone.now().strftime('%Y%m%d%H%M%S'))
  396. qs.save()
  397. if qs.bound_type == 'in':
  398. BoundDetailModel.objects.filter(bound_list=qs.id).update(is_delete=True)
  399. if qs.relate_bill:
  400. qs.relate_bill.bound_status = 0
  401. qs.relate_bill.save()
  402. else:
  403. OutBoundDemandModel.objects.filter(bound_list=qs.id).update(is_delete=True)
  404. if qs.relate_out_bill:
  405. qs.relate_out_bill.bound_status = 0
  406. qs.relate_out_bill.save()
  407. serializer = self.get_serializer(qs, many=False)
  408. headers = self.get_success_headers(serializer.data)
  409. return Response(serializer.data, status=200, headers=headers)
  410. # 入库批次类视图
  411. class BoundBatchViewSet(viewsets.ModelViewSet):
  412. """
  413. retrieve:
  414. Response a data list(get)
  415. list:
  416. Response a data list(all)
  417. create:
  418. Create a data line(post)
  419. delete:
  420. Delete a data line(delete)
  421. """
  422. # authentication_classes = [] # 禁用所有认证类
  423. # permission_classes = [AllowAny] # 允许任意访问
  424. pagination_class = MyPageNumberPagination
  425. filter_backends = [DjangoFilterBackend, OrderingFilter, ]
  426. ordering_fields = ['id', "create_time", "update_time", ]
  427. filter_class = BoundBatchFilter
  428. def get_project(self):
  429. try:
  430. id = self.kwargs.get('pk')
  431. return id
  432. except:
  433. return None
  434. def get_queryset(self):
  435. id = self.get_project()
  436. if self.request.user:
  437. if id is None:
  438. return BoundBatchModel.objects.filter( is_delete=False)
  439. else:
  440. return BoundBatchModel.objects.filter( id=id, is_delete=False)
  441. else:
  442. return BoundBatchModel.objects.none()
  443. def get_serializer_class(self):
  444. if self.action in ['list', 'destroy','retrieve']:
  445. return BoundBatchGetSerializer
  446. elif self.action in ['create', 'update']:
  447. return BoundBatchPostSerializer
  448. else:
  449. return self.http_method_not_allowed(request=self.request)
  450. def create(self, request, *args, **kwargs):
  451. data = self.request.data
  452. try:
  453. data['openid'] = self.request.auth.openid
  454. data['goods_total_weight'] = data['goods_weight']*data['goods_qty']
  455. order_day=str(timezone.now().strftime('-%Y%m'))
  456. order_month=str(timezone.now().strftime('%Y%m'))
  457. data['bound_month'] =str(timezone.now().strftime('%Y%m'))
  458. print(data['order'])
  459. if data['order'] == 'true':
  460. qs_set = BoundBatchModel.objects.filter( goods_code=data['goods_code'], bound_month=order_month, is_delete=False)
  461. print('qs_set是:', len(qs_set))
  462. if len(qs_set) > 0:
  463. bound_last_code = qs_set.order_by('-bound_batch_order').first().bound_number
  464. data['bound_batch_order'] = int(bound_last_code.split('-')[-1])+1
  465. data['bound_number'] = data['goods_code'] + '-' + str(int(bound_last_code.split('-')[-1])+1)
  466. else:
  467. data['bound_batch_order'] = int(order_day.split('-')[-1])*1000 +1
  468. data['bound_number'] = data['goods_code'] + order_day + '001'
  469. else:
  470. data['bound_number'] = data['goods_code'] + '-' + str(data['bound_batch_order'])
  471. serializer = self.get_serializer(data=data)
  472. serializer.is_valid(raise_exception=True)
  473. serializer.save()
  474. headers = self.get_success_headers(serializer.data)
  475. self.add_batch_log(serializer.data, 0, data['goods_qty'])
  476. return Response(serializer.data, status=200, headers=headers)
  477. except Exception as e:
  478. print(e)
  479. raise APIException({"detail": "{}".format(e)})
  480. def add_batch_log(self, data, log_type, goods_qty):
  481. choices_dict = dict(BatchOperateLogModel.BATCH_LOG_TYPE)
  482. log_type_name = choices_dict.get(log_type, "未知类型")
  483. try:
  484. # 获取 BoundBatchModel 实例
  485. batch_obj = BoundBatchModel.objects.get(id=data['id'])
  486. except BoundBatchModel.DoesNotExist:
  487. return False
  488. log_data = {
  489. 'batch_id': batch_obj,
  490. 'log_type': log_type,
  491. 'log_date': timezone.now().strftime('%Y-%m-%d-%H:%M'), # 直接格式化时间,无需转字符串
  492. 'goods_code': data['goods_code'],
  493. 'goods_desc': data['goods_desc'],
  494. 'goods_qty': data['goods_qty'],
  495. 'log_content': f"{log_type_name} {data['goods_qty']}件",
  496. 'creater': data['creater'],
  497. 'openid': data['openid'],
  498. 'is_delete': False,
  499. }
  500. # 创建日志记录
  501. BatchOperateLogModel.objects.create(**log_data)
  502. return True
  503. def update(self, request, pk):
  504. qs = self.get_object()
  505. data = self.request.data
  506. serializer = self.get_serializer(qs, data=data)
  507. serializer.is_valid(raise_exception=True)
  508. serializer.save()
  509. headers = self.get_success_headers(serializer.data)
  510. return Response(serializer.data, status=200, headers=headers)
  511. def destroy(self, request, pk):
  512. qs = self.get_object()
  513. if qs.openid != self.request.auth.openid:
  514. raise APIException({"detail": "该入库非您所属,禁止删除,您可以进行编辑"})
  515. else:
  516. qs.is_delete = True
  517. qs.save()
  518. serializer = self.get_serializer(qs, many=False)
  519. headers = self.get_success_headers(serializer.data)
  520. return Response(serializer.data, status=200, headers=headers)
  521. def batchinout(self, request, *args, **kwargs):
  522. from container.models import ContainerDetailModel, ContainerListModel
  523. from decimal import Decimal
  524. batch_number = request.data.get('batch_number')
  525. batch_in_qty = request.data.get('batch_in_qty')
  526. if not batch_number:
  527. return Response({"code": 200,"message": "批次号不能为空","data": "批次进出库数目:{batch_in_qty}"}, status=200)
  528. try:
  529. batch_obj = BoundBatchModel.objects.get(bound_number=batch_number, is_delete=False)
  530. except BoundBatchModel.DoesNotExist:
  531. return Response({"code": 200,"message": "批次号不存在","data": "批次进出库数目:{batch_in_qty}"}, status=200)
  532. if batch_in_qty:
  533. # 获取或创建虚拟容器
  534. virtual_container, created = ContainerListModel.objects.get_or_create(
  535. container_code=1
  536. )
  537. # 使用 update_or_create 避免并发问题
  538. virtual_container_detail = ContainerDetailModel.objects.create(
  539. month=timezone.now().strftime('%Y%m'),
  540. container= virtual_container,
  541. batch= batch_obj,
  542. goods_class= 1, # 成品
  543. goods_code= batch_obj.goods_code,
  544. goods_desc= batch_obj.goods_desc,
  545. goods_qty= Decimal(batch_in_qty),
  546. goods_out_qty= Decimal(0),
  547. goods_weight= batch_obj.goods_weight,
  548. status= 2, # 在库
  549. creater= "system",
  550. create_time= timezone.now(),
  551. update_time= timezone.now(),
  552. is_delete= False )
  553. virtual_container_detail.save()
  554. virtual_container_detail.goods_out_qty = batch_in_qty
  555. virtual_container_detail.save()
  556. return Response({"code": 200,"message": "操作成功","data": "批次进出库数目:{batch_in_qty}"}, status=200)
  557. # 入库明细类视图
  558. class BoundDetailViewSet(viewsets.ModelViewSet):
  559. """
  560. retrieve:
  561. Response a data list(get)
  562. list:
  563. Response a data list(all)
  564. create:
  565. Create a data line(post)
  566. delete:
  567. Delete a data line(delete)
  568. """
  569. # authentication_classes = [] # 禁用所有认证类
  570. # permission_classes = [AllowAny] # 允许任意访问
  571. pagination_class = MyPageNumberPagination
  572. filter_backends = [DjangoFilterBackend, OrderingFilter, ]
  573. ordering_fields = ['id', "create_time", "update_time", ]
  574. filter_class = BoundDetailFilter
  575. def get_project(self):
  576. try:
  577. id = self.kwargs.get('pk')
  578. return id
  579. except:
  580. return None
  581. def get_queryset(self):
  582. id = self.get_project()
  583. if self.request.user:
  584. if id is None:
  585. return BoundDetailModel.objects.filter( is_delete=False)
  586. else:
  587. return BoundDetailModel.objects.filter( id=id, is_delete=False)
  588. else:
  589. return BoundDetailModel.objects.none()
  590. def get_serializer_class(self):
  591. if self.action in ['list', 'destroy','retrieve']:
  592. return BoundDetailGetSerializer
  593. elif self.action in ['create', 'update']:
  594. return BoundDetailPostSerializer
  595. else:
  596. return self.http_method_not_allowed(request=self.request)
  597. def create(self, request, *args, **kwargs):
  598. data = self.request.data
  599. data['openid'] = self.request.auth.openid
  600. data.setdefault('is_delete', False)
  601. # 验证并保存数据
  602. data['detail_code'] = f"DC-{data['bound_list']:02}{data['bound_batch']:02}"
  603. print(data['detail_code'])
  604. if BoundDetailModel.objects.filter(detail_code=data['detail_code'], is_delete=False).exists():
  605. raise APIException({"detail": "Data exists"})
  606. else:
  607. serializer = self.get_serializer(data=data)
  608. serializer.is_valid(raise_exception=True)
  609. serializer.save()
  610. # 返回响应
  611. headers = self.get_success_headers(serializer.data)
  612. return Response(serializer.data, status=200, headers=headers)
  613. def update(self, request, pk):
  614. qs = self.get_object()
  615. data = self.request.data
  616. serializer = self.get_serializer(qs, data=data)
  617. serializer.is_valid(raise_exception=True)
  618. serializer.save()
  619. headers = self.get_success_headers(serializer.data)
  620. return Response(serializer.data, status=200, headers=headers)
  621. def destroy(self, request, pk):
  622. qs = self.get_object()
  623. if qs.openid != self.request.auth.openid:
  624. raise APIException({"detail": "该入库非您所属,禁止删除,您可以进行编辑"})
  625. else:
  626. qs.is_delete = True
  627. qs.save()
  628. serializer = self.get_serializer(qs, many=False)
  629. headers = self.get_success_headers(serializer.data)
  630. return Response(serializer.data, status=200, headers=headers)
  631. # 出库明细类视图
  632. class OutBoundDetailViewSet(viewsets.ModelViewSet):
  633. """
  634. retrieve:
  635. Response a data list(get)
  636. list: Response a data list(all)
  637. create:
  638. Create a data line(post)
  639. delete:
  640. Delete a data line(delete)
  641. """
  642. # authentication_classes = [] # 禁用所有认证类
  643. # permission_classes = [AllowAny] # 允许任意访问
  644. pagination_class = MyPageNumberPagination
  645. filter_backends = [DjangoFilterBackend, OrderingFilter, ]
  646. ordering_fields = ['id', "create_time", "update_time", ]
  647. filter_class = OutBoundDetailFilter
  648. def get_project(self):
  649. try:
  650. id = self.kwargs.get('pk')
  651. return id
  652. except:
  653. return None
  654. def get_queryset(self):
  655. id = self.get_project()
  656. if self.request.user:
  657. if id is None:
  658. return OutBoundDetailModel.objects.filter( is_delete=False)
  659. else:
  660. return OutBoundDetailModel.objects.filter( id=id, is_delete=False)
  661. else:
  662. return OutBoundDetailModel.objects.none()
  663. def get_serializer_class(self):
  664. if self.action in ['list', 'destroy','retrieve']:
  665. return OutBoundDetailGetSerializer
  666. elif self.action in ['create', 'update']:
  667. return OutBoundDetailPostSerializer
  668. else:
  669. return self.http_method_not_allowed(request=self.request)
  670. def create(self, request, *args, **kwargs):
  671. data = self.request.data
  672. data['openid'] = self.request.auth.openid
  673. data.setdefault('is_delete', False)
  674. data['bound_batch_number'] = OutBatchModel.objects.get(id=data['bound_batch']).batch_number.id
  675. # 验证并保存数据
  676. data['detail_code'] = f"DC-{data['bound_list']:02}{data['bound_batch']:02}"
  677. print(data['detail_code'])
  678. if OutBoundDetailModel.objects.filter(detail_code=data['detail_code'], is_delete=False).exists():
  679. raise APIException({"detail": "Data exists"})
  680. else:
  681. serializer = self.get_serializer(data=data)
  682. serializer.is_valid(raise_exception=True)
  683. serializer.save()
  684. # 返回响应
  685. headers = self.get_success_headers(serializer.data)
  686. return Response(serializer.data, status=200, headers=headers)
  687. def update(self, request, pk):
  688. qs = self.get_object()
  689. data = self.request.data
  690. serializer = self.get_serializer(qs, data=data)
  691. serializer.is_valid(raise_exception=True)
  692. serializer.save()
  693. headers = self.get_success_headers(serializer.data)
  694. return Response(serializer.data, status=200, headers=headers)
  695. def destroy(self, request, pk):
  696. qs = self.get_object()
  697. if qs.openid != self.request.auth.openid:
  698. raise APIException({"detail": "该入库非您所属,禁止删除,您可以进行编辑"})
  699. else:
  700. qs.is_delete = True
  701. qs.save()
  702. serializer = self.get_serializer(qs, many=False)
  703. headers = self.get_success_headers(serializer.data)
  704. return Response(serializer.data, status=200, headers=headers)
  705. # 出库批次类视图
  706. class OutBoundBatchViewSet(viewsets.ModelViewSet):
  707. """
  708. retrieve:
  709. Response a data list(get)
  710. list:
  711. Response a data list(all)
  712. create:
  713. Create a data line(post)
  714. delete:
  715. Delete a data line(delete)
  716. """
  717. # authentication_classes = [] # 禁用所有认证类
  718. # permission_classes = [AllowAny] # 允许任意访问
  719. pagination_class = MyPageNumberPagination
  720. filter_backends = [DjangoFilterBackend, OrderingFilter, ]
  721. ordering_fields = ['id', "create_time", "update_time", ]
  722. filter_class = OutBatchFilter
  723. def get_project(self):
  724. try:
  725. id = self.kwargs.get('pk')
  726. return id
  727. except:
  728. return None
  729. def get_queryset(self):
  730. id = self.get_project()
  731. if self.request.user:
  732. if id is None:
  733. return OutBatchModel.objects.filter( is_delete=False)
  734. else:
  735. return OutBatchModel.objects.filter( id=id, is_delete=False)
  736. else:
  737. return OutBatchModel.objects.none()
  738. def get_serializer_class(self):
  739. if self.action in ['list', 'destroy','retrieve']:
  740. return OutBatchGetSerializer
  741. elif self.action in ['create', 'update']:
  742. return OutBatchPostSerializer
  743. else:
  744. return self.http_method_not_allowed(request=self.request)
  745. def create(self, request, *args, **kwargs):
  746. data = self.request.data
  747. batch_obj = BoundBatchModel.objects.get(bound_number=data['out_number'])
  748. if batch_obj is None:
  749. raise APIException({"detail": "批次不存在"})
  750. bound_obj = BoundListModel.objects.get(id=data['bound_list_id'])
  751. data['bound_list'] = bound_obj.id
  752. data['batch_number'] = batch_obj.id
  753. data['out_date'] = str(timezone.now().strftime('%Y-%m-%d %H:%M:%S'))
  754. data['openid'] = self.request.auth.openid
  755. data.setdefault('is_delete', False)
  756. data['goods_total_weight'] = data['goods_weight']*data['goods_out_qty']
  757. from decimal import Decimal
  758. # data['goods_out_qty'] 是一个字符串或浮点数
  759. data['goods_out_qty'] = Decimal(str(data['goods_out_qty']))
  760. # 现在 data['goods_out_qty'] 是 Decimal 类型,可以进行减法操作
  761. from decimal import Decimal
  762. # 假设 data['goods_out_qty'] 是一个 float 类型的数值
  763. data['goods_out_qty'] = Decimal(str(data['goods_out_qty']))
  764. # 现在所有数值都是 Decimal 类型,可以安全地进行减法运算
  765. data['goods_qty'] = batch_obj.goods_qty - batch_obj.goods_reserve_qty - data['goods_out_qty']
  766. # data['goods_qty'] = batch_obj.goods_qty - batch_obj.goods_reserve_qty - data['goods_out_qty']
  767. data['status'] = 0 #现在处于出库申请状态
  768. serializer = self.get_serializer(data=data)
  769. serializer.is_valid(raise_exception=True)
  770. serializer.save()
  771. headers = self.get_success_headers(serializer.data)
  772. self.add_batch_log(serializer.data, 1, data['goods_out_qty'])
  773. return Response(serializer.data, status=200, headers=headers)
  774. def update(self, request, pk):
  775. qs = self.get_object()
  776. data = self.request.data
  777. data['openid'] = self.request.auth.openid
  778. data.setdefault('is_delete', False)
  779. data['goods_total_weight'] = data['goods_weight']*data['goods_qty']
  780. serializer = self.get_serializer(qs, data=data)
  781. serializer.is_valid(raise_exception=True)
  782. serializer.save()
  783. headers = self.get_success_headers(serializer.data)
  784. self.add_batch_log(serializer.data, 1, data['goods_qty'])
  785. return Response(serializer.data, status=200, headers=headers)
  786. def destroy(self, request, pk):
  787. qs = self.get_object()
  788. if qs.openid != self.request.auth.openid:
  789. raise APIException({"detail": "该出库非您所属,禁止删除,您可以进行编辑"})
  790. else:
  791. qs.is_delete = True
  792. qs.save()
  793. serializer = self.get_serializer(qs, many=False)
  794. headers = self.get_success_headers(serializer.data)
  795. return Response(serializer.data, status=200, headers=headers)
  796. def add_batch_log(self, data, log_type, goods_qty):
  797. choices_dict = dict(BatchOperateLogModel.BATCH_LOG_TYPE)
  798. log_type_name = choices_dict.get(log_type, "未知类型")
  799. try:
  800. # 获取 BoundBatchModel 实例
  801. batch_obj = BoundBatchModel.objects.get(id=data['batch_number'])
  802. except BoundBatchModel.DoesNotExist:
  803. # 处理批次不存在的情况(如记录日志或抛出异常)
  804. return False
  805. log_data = {
  806. 'batch_id': batch_obj, # 关键修复:传入实例对象,而不是 batch_obj.id
  807. 'log_type': log_type,
  808. 'log_date': timezone.now().strftime('%Y-%m-%d-%H:%M'), # 直接格式化时间,无需转字符串
  809. 'goods_code': data['goods_code'],
  810. 'goods_desc': data['goods_desc'],
  811. 'goods_qty': data['goods_qty'],
  812. 'log_content': f"{log_type_name} {goods_qty}件",
  813. 'creater': data['creater'],
  814. 'openid': data['openid'],
  815. 'is_delete': False,
  816. # 注意:create_time 和 update_time 由模型的 auto_now_add 和 auto_now 自动处理,无需手动赋值
  817. }
  818. # 创建日志记录
  819. BatchOperateLogModel.objects.create(**log_data)
  820. return True
  821. # 批次操作日志类视图
  822. class BoundBatchLogViewSet(viewsets.ModelViewSet):
  823. """
  824. retrieve:
  825. Response a data list(get)
  826. list:
  827. Response a data list(all)
  828. delete:
  829. Delete a data line(delete)
  830. """
  831. # authentication_classes = [] # 禁用所有认证类
  832. # permission_classes = [AllowAny] # 允许任意访问
  833. pagination_class = MyPageNumberPagination
  834. filter_backends = [DjangoFilterBackend, OrderingFilter, ]
  835. ordering_fields = ['id', "create_time", "update_time", ]
  836. filter_class = BatchlogFilter
  837. def get_queryset(self):
  838. return BatchOperateLogModel.objects.filter( is_delete=False)
  839. def get_serializer_class(self):
  840. if self.action in ['list', 'destroy','retrieve']:
  841. return BatchLogGetSerializer
  842. else:
  843. return self.http_method_not_allowed(request=self.request)
  844. def destroy(self, request, pk):
  845. qs = self.get_object()
  846. qs.is_delete = True
  847. qs.save()
  848. serializer = self.get_serializer(qs, many=False)
  849. headers = self.get_success_headers(serializer.data)
  850. return Response(serializer.data, status=200, headers=headers)
  851. # 托盘类视图
  852. class BatchContainerAPIView(APIView):
  853. """
  854. post:
  855. 返回批次对应的container列表
  856. """
  857. # authentication_classes = [] # 禁用所有认证类
  858. # permission_classes = [AllowAny] # 允许任意访问
  859. def post(self, request):
  860. data = request.data
  861. batch_id = data.get('batch_id')
  862. from container.models import ContainerDetailModel
  863. from bin.models import LocationContainerLink
  864. container_detail_all = ContainerDetailModel.objects.filter(batch_id=batch_id, is_delete=False).exclude(status=3).all()
  865. container_dict = {}
  866. for container_detail in container_detail_all:
  867. container_id = container_detail.container_id
  868. link_obj = LocationContainerLink.objects.filter(container_id=container_id,is_active=True).first()
  869. if container_id not in container_dict:
  870. container_dict[container_id] = {
  871. 'id': container_id,
  872. 'goods_code': container_detail.goods_code,
  873. 'goods_desc': container_detail.goods_desc,
  874. 'container_code': container_detail.container.container_code,
  875. 'current_location': container_detail.container.current_location,
  876. 'goods_qty': container_detail.goods_qty-container_detail.goods_out_qty,
  877. 'class':container_detail.goods_class,
  878. 'location_c_number' : link_obj.location.c_number if link_obj else None,
  879. 'location_max_capacity': link_obj.location.max_capacity if link_obj else None,
  880. 'location_group': link_obj.location.location_group if link_obj else None,
  881. 'location_code': link_obj.location.location_code if link_obj else None,
  882. }
  883. else:
  884. container_dict[container_id]['goods_qty'] += container_detail.goods_qty-container_detail.goods_out_qty
  885. container_dict = list(container_dict.values())
  886. return_data = {'code': 200,'msg': 'Success Create', 'data': container_dict}
  887. return Response(return_data)