398 lines
17 KiB
Python
398 lines
17 KiB
Python
from apps.utils.constants import EXCLUDE_FIELDS
|
|
from apps.utils.serializers import CustomModelSerializer
|
|
from rest_framework import serializers
|
|
from rest_framework.exceptions import ValidationError, ParseError
|
|
from datetime import datetime
|
|
|
|
from .models import SfLog, StLog, SfLogExp, WMaterial, Mlog, Handover, Mlogb, AttLog, OtherLog
|
|
from apps.system.models import Dept, User
|
|
from apps.system.serializers import UserSimpleSerializer
|
|
from apps.pm.models import Mtask
|
|
from apps.wpm.tasks import cal_enstat_when_pcoal_heat_change, cal_enstat_when_team_change, cal_exp_duration_hour
|
|
from apps.wpm.services import get_sflog
|
|
from apps.mtm.models import Mgroup, TeamMember, Shift, Material
|
|
from apps.mtm.serializers import MaterialSimpleSerializer
|
|
from django.db import transaction
|
|
from django.utils import timezone
|
|
|
|
|
|
class OtherLogSerializer(CustomModelSerializer):
|
|
class Meta:
|
|
model = OtherLog
|
|
fields = '__all__'
|
|
read_only_fields = EXCLUDE_FIELDS
|
|
|
|
|
|
class StLogSerializer(CustomModelSerializer):
|
|
mgroup_name = serializers.CharField(source='mgroup.name', read_only=True)
|
|
current_sflog = serializers.CharField(label='当前处理值班', write_only=True)
|
|
current_note = serializers.CharField(
|
|
label='值班处理备注', write_only=True, allow_blank=True)
|
|
|
|
class Meta:
|
|
model = StLog
|
|
fields = '__all__'
|
|
read_only_fields = EXCLUDE_FIELDS + \
|
|
['is_shutdown', 'sflog', 'duration']
|
|
|
|
def create(self, validated_data):
|
|
current_sflog_id = validated_data.pop('current_sflog')
|
|
current_note = validated_data.pop('current_note', '')
|
|
with transaction.atomic():
|
|
sflog = get_sflog(
|
|
validated_data['mgroup'], validated_data['start_time'])
|
|
if current_sflog_id != sflog.id:
|
|
raise ParseError('值班时间与发生时间不一致')
|
|
instance = super().create(validated_data)
|
|
instance.sflog = sflog
|
|
instance.save()
|
|
SfLogExp.objects.create(
|
|
sflog=sflog, stlog=instance, create_by=self.context['request'].user, note=current_note)
|
|
return instance
|
|
|
|
def update(self, instance, validated_data):
|
|
if instance.is_shutdown: # 停机不可编辑end_time
|
|
validated_data.pop('end_time', None)
|
|
if instance.end_time:
|
|
raise ParseError('该异常已结束无需编辑')
|
|
with transaction.atomic():
|
|
validated_data.pop('mgroup', None)
|
|
validated_data.pop('start_time', None)
|
|
end_time = validated_data.pop('end_time', None)
|
|
current_sflog = validated_data.pop('current_sflog')
|
|
current_note = validated_data.pop('current_note', '')
|
|
instance = super().update(instance, validated_data)
|
|
if end_time: # 需要把涉及到的sflog都拉入
|
|
cal_exp_duration_hour(instance.id)
|
|
try:
|
|
sflogexp = SfLogExp.objects.get(
|
|
stlog=instance.stlog, sflog=current_sflog)
|
|
sflogexp.note = current_note
|
|
sflogexp.save()
|
|
except SfLogExp.DoesNotExist:
|
|
raise ParseError('该异常无需本班填写')
|
|
return instance
|
|
|
|
def validate(self, attrs):
|
|
now = timezone.now()
|
|
start_time: datetime = attrs['start_time']
|
|
end_time: datetime = attrs.get('end_time', None)
|
|
if start_time > now:
|
|
raise ParseError('开始时间应为历史时间')
|
|
if end_time:
|
|
if end_time > now:
|
|
raise ParseError('开始时间应为历史时间')
|
|
if end_time > start_time:
|
|
attrs['duration'] = (
|
|
end_time - start_time).total_seconds() / 3600
|
|
else:
|
|
raise ParseError('结束时间应大于开始时间')
|
|
return super().validate(attrs)
|
|
|
|
|
|
class SfLogSerializer(CustomModelSerializer):
|
|
mgroup_name = serializers.CharField(source='mgroup.name', read_only=True)
|
|
team_name = serializers.CharField(source='team.name', read_only=True)
|
|
shift_name = serializers.CharField(source='shift.name', read_only=True)
|
|
leader_name = serializers.CharField(source='leader.name', read_only=True)
|
|
|
|
class Meta:
|
|
model = SfLog
|
|
fields = '__all__'
|
|
read_only_fields = EXCLUDE_FIELDS + \
|
|
['mgroup', 'start_time', 'end_time', 'belong_dept']
|
|
extra_kwargs = {
|
|
'team': {'required': True},
|
|
'leader': {'required': True}
|
|
}
|
|
|
|
def update(self, instance, validated_data):
|
|
with transaction.atomic():
|
|
old_pcoal_heat = instance.pcoal_heat
|
|
old_team = instance.team
|
|
instance: SfLog = super().update(instance, validated_data)
|
|
new_pcoal_heat = instance.pcoal_heat
|
|
new_team = instance.team
|
|
mgroup: Mgroup = instance.mgroup
|
|
if new_pcoal_heat != old_pcoal_heat and mgroup.need_enm:
|
|
cal_enstat_when_pcoal_heat_change.delay(instance.id)
|
|
if new_team != old_team:
|
|
default_state = 'pending'
|
|
if timezone.now() > instance.end_time:
|
|
default_state = 'normal'
|
|
# 分配班组时创建人员到岗情况
|
|
for item in TeamMember.objects.filter(team=new_team, mgroup=instance.mgroup):
|
|
AttLog.objects.get_or_create(sflog=instance, user=item.user, defaults={
|
|
'sflog': instance, 'user': item.user, 'post': item.post, 'state': default_state, 'create_by': self.context['request'].user})
|
|
if mgroup.need_enm:
|
|
cal_enstat_when_team_change.delay(instance.id)
|
|
return instance
|
|
# def to_internal_value(self, data):
|
|
# if hasattr(self.Meta, 'update_fields') and self.context['request'].method in ['PUT', 'PATCH']:
|
|
# u_fields = self.Meta.update_fields
|
|
# new_data = {key: data[key] for key in u_fields if key in data}
|
|
# return super().to_internal_value(new_data)
|
|
# return super().to_internal_value(data)
|
|
|
|
|
|
class SflogExpSerializer(CustomModelSerializer):
|
|
# mgroup = serializers.CharField(source='sflog.mgroup.id', read_only=True)
|
|
# mgroup_name = serializers.CharField(
|
|
# source='sflog.mgroup.name', read_only=True)
|
|
# stlog_ = StLogSerializer(source='stlog', read_only=True)
|
|
# happen_time = serializers.DateTimeField(required=True, label='发生时间')
|
|
# cate = serializers.CharField(required=True, label='原因类别')
|
|
sflog_ = SfLogSerializer(source='sflog', read_only=True)
|
|
|
|
class Meta:
|
|
model = SfLogExp
|
|
fields = '__all__'
|
|
read_only_fields = EXCLUDE_FIELDS + ['sflog', 'stlog', 'duration']
|
|
|
|
|
|
class WMaterialSerializer(CustomModelSerializer):
|
|
material_ = MaterialSimpleSerializer(source='material', read_only=True)
|
|
material_name = serializers.StringRelatedField(
|
|
source='material', read_only=True)
|
|
belong_dept_name = serializers.CharField(
|
|
source='belong_dept.name', read_only=True)
|
|
|
|
class Meta:
|
|
model = WMaterial
|
|
fields = '__all__'
|
|
|
|
|
|
class MlogbSerializer(CustomModelSerializer):
|
|
material_out_ = MaterialSimpleSerializer(
|
|
source='material_out', read_only=True)
|
|
material_out_name = serializers.StringRelatedField(
|
|
source='material_out', read_only=True)
|
|
|
|
class Meta:
|
|
model = Mlogb
|
|
fields = ['id', 'material_out', 'count_ok',
|
|
'material_out_', 'material_out_name']
|
|
|
|
|
|
class MlogSerializer(CustomModelSerializer):
|
|
belong_dept = serializers.CharField(
|
|
source='mgroup.belong_dept.id', read_only=True)
|
|
belong_dept_name = serializers.CharField(
|
|
source='mgroup.belong_dept.name', read_only=True)
|
|
mgroup_name = serializers.CharField(
|
|
source='mgroup.name', read_only=True)
|
|
mtask_count = serializers.IntegerField(
|
|
source='mtask.count', read_only=True)
|
|
create_by_name = serializers.CharField(
|
|
source='create_by.name', read_only=True)
|
|
update_by_name = serializers.CharField(
|
|
source='update_by.name', read_only=True)
|
|
handovers = serializers.PrimaryKeyRelatedField(
|
|
source='handover_mlog', read_only=True, many=True)
|
|
material_out_ = MaterialSimpleSerializer(
|
|
source='material_out', read_only=True)
|
|
material_out_name = serializers.StringRelatedField(
|
|
source='material_out', read_only=True)
|
|
material_in_name = serializers.StringRelatedField(
|
|
source='material_in', read_only=True)
|
|
handle_user_name = serializers.CharField(
|
|
source='handle_user.name', read_only=True)
|
|
handle_user_2_name = serializers.CharField(
|
|
source='handle_user_2.name', read_only=True)
|
|
handle_leader_name = serializers.CharField(
|
|
source='handle_leader.name', read_only=True)
|
|
equipment_name = serializers.StringRelatedField(
|
|
source='equipment', read_only=True)
|
|
equipment_2_name = serializers.StringRelatedField(
|
|
source='equipment_2', read_only=True)
|
|
shift = serializers.PrimaryKeyRelatedField(
|
|
label='班次ID', queryset=Shift.objects.all(), required=True)
|
|
mgroup = serializers.PrimaryKeyRelatedField(
|
|
label='工段ID', queryset=Mgroup.objects.all(), required=True
|
|
)
|
|
material_out = serializers.PrimaryKeyRelatedField(
|
|
label='产物ID', queryset=Material.objects.all(), required=True
|
|
)
|
|
shift_name = serializers.CharField(source='shift.name', read_only=True)
|
|
mlogb = MlogbSerializer(
|
|
label='多产出件信息', many=True, required=False)
|
|
handle_users_ = UserSimpleSerializer(
|
|
source='handle_users', many=True, read_only=True)
|
|
equipments_name = serializers.StringRelatedField(
|
|
source='equipments', read_only=True, many=True)
|
|
|
|
class Meta:
|
|
model = Mlog
|
|
fields = '__all__'
|
|
read_only_fields = EXCLUDE_FIELDS + \
|
|
['submit_time', 'submit_user', 'material_outs']
|
|
|
|
def create(self, validated_data):
|
|
mtask: Mtask = validated_data.get('mtask', None)
|
|
# batch = validated_data['batch']
|
|
# handle_date = validated_data['handle_date']
|
|
# handle_user = validated_data.get('handle_user', None)
|
|
if mtask:
|
|
validated_data['mgroup'] = mtask.mgroup
|
|
validated_data['material_in'] = mtask.material_in
|
|
material_out = mtask.material_out
|
|
validated_data['material_out'] = material_out
|
|
validated_data['handle_date'] = mtask.start_date
|
|
# if not WMaterial.objects.filter(batch=batch).exists():
|
|
# raise ValidationError('批次号不存在')
|
|
else:
|
|
mgroup = validated_data['mgroup']
|
|
material_out = validated_data['material_out']
|
|
if not (mgroup and material_out):
|
|
raise ValidationError('缺少工段或产物!')
|
|
# if handle_user:
|
|
# if Mlog.objects.filter(mtask=mtask, batch=batch, handle_date=handle_date, handle_user=handle_user).exists():
|
|
# raise ValidationError('存在相同的日志记录')
|
|
# else:
|
|
# if Mlog.objects.filter(mtask=mtask, batch=batch, handle_date=handle_date).exists():
|
|
# raise ValidationError('存在相同的日志记录')
|
|
with transaction.atomic():
|
|
mlogb = validated_data.pop('mlogb', [])
|
|
instance = super().create(validated_data)
|
|
brotherId_should_list = material_out.brothers
|
|
if brotherId_should_list:
|
|
if mlogb:
|
|
for item in mlogb:
|
|
if item['material_out'].id in brotherId_should_list:
|
|
Mlogb.objects.create(
|
|
mlog=instance, material_out=item['material_out'], count_ok=item['count_ok'])
|
|
else:
|
|
raise ValidationError('缺少产出物信息')
|
|
return instance
|
|
|
|
def update(self, instance, validated_data):
|
|
validated_data.pop('mtask', None)
|
|
validated_data.pop('mgroup', None)
|
|
if instance.mtask:
|
|
validated_data.pop('handle_date', None)
|
|
# validated_data.pop('handle_user', None)
|
|
with transaction.atomic():
|
|
mlogb = validated_data.pop('mlogb', [])
|
|
instance = super().update(instance, validated_data)
|
|
if mlogb:
|
|
Mlogb.objects.filter(mlog=instance).update(count_ok=0)
|
|
for item in mlogb:
|
|
Mlogb.objects.filter(mlog=instance, material_out=item['material_out']).update(
|
|
count_ok=item['count_ok'])
|
|
return instance
|
|
|
|
def validate(self, attrs):
|
|
mtask = attrs.get('mtask', None)
|
|
count_notok = 0
|
|
for i in attrs:
|
|
if 'count_n_' in i:
|
|
count_notok = count_notok + attrs[i]
|
|
attrs['count_notok'] = count_notok
|
|
if attrs['count_real'] >= attrs['count_ok'] + attrs['count_notok']:
|
|
pass
|
|
else:
|
|
raise ValidationError('生产数量不能小于合格数量')
|
|
if mtask:
|
|
if mtask.start_date == mtask.end_date:
|
|
attrs['handle_date'] = mtask.start_date
|
|
else:
|
|
if attrs['handle_date'] >= mtask.start_date and attrs['handle_date'] <= mtask.end_date:
|
|
pass
|
|
else:
|
|
raise ValidationError('操作日期错误')
|
|
return attrs
|
|
|
|
|
|
class MlogRevertSerializer(serializers.Serializer):
|
|
change_reason = serializers.CharField(label='撤回原因')
|
|
|
|
|
|
class MlogRelatedSerializer(serializers.Serializer):
|
|
mtask = serializers.PrimaryKeyRelatedField(
|
|
label='小任务ID', queryset=Mtask.objects.all())
|
|
|
|
|
|
class DeptBatchSerializer(serializers.Serializer):
|
|
belong_dept_name = serializers.CharField(label='车间名称')
|
|
|
|
|
|
class HandoverSerializer(CustomModelSerializer):
|
|
wm = serializers.PrimaryKeyRelatedField(
|
|
label='车间库存ID', queryset=WMaterial.objects.all())
|
|
material = serializers.PrimaryKeyRelatedField(
|
|
required=True, label='物料ID', queryset=Material.objects.all())
|
|
send_user_name = serializers.CharField(
|
|
source='send_user.name', read_only=True)
|
|
recive_user_name = serializers.CharField(
|
|
source='recive_user.name', read_only=True)
|
|
recive_dept_name = serializers.CharField(
|
|
source='recive_dept', read_only=True)
|
|
material_ = MaterialSimpleSerializer(source='material', read_only=True)
|
|
material_name = serializers.StringRelatedField(
|
|
source='material', read_only=True)
|
|
|
|
def validate(self, attrs):
|
|
if attrs.get('mlog', None):
|
|
attrs['send_mgroup'] = attrs['mlog'].mgroup
|
|
attrs['send_dept'] = attrs['mlog'].mgroup.belong_dept
|
|
elif attrs.get('wm', None):
|
|
attrs['send_dept'] = attrs['wm'].belong_dept
|
|
return super().validate(attrs)
|
|
|
|
class Meta:
|
|
model = Handover
|
|
fields = '__all__'
|
|
read_only_fields = EXCLUDE_FIELDS
|
|
|
|
|
|
class GenHandoverSerializer(serializers.Serializer):
|
|
mlogs = serializers.PrimaryKeyRelatedField(
|
|
label='mlog的ID列表', queryset=Mlog.objects.all(), many=True)
|
|
recive_dept = serializers.PrimaryKeyRelatedField(
|
|
label='交送车间', queryset=Dept.objects.all())
|
|
recive_user = serializers.PrimaryKeyRelatedField(
|
|
label='接收人', queryset=User.objects.all())
|
|
send_user = serializers.PrimaryKeyRelatedField(
|
|
label='交送人', queryset=User.objects.all())
|
|
send_date = serializers.DateField(label='交送日期')
|
|
|
|
|
|
class GenHandoverWmSerializer(serializers.Serializer):
|
|
wm = serializers.PrimaryKeyRelatedField(
|
|
label='车间物料ID', queryset=WMaterial.objects.all())
|
|
send_mgroup = serializers.PrimaryKeyRelatedField(
|
|
label='送料工段ID', queryset=Mgroup.objects.all())
|
|
recive_dept = serializers.PrimaryKeyRelatedField(
|
|
label='交送车间', queryset=Dept.objects.all())
|
|
recive_user = serializers.PrimaryKeyRelatedField(
|
|
label='接收人', queryset=User.objects.all())
|
|
send_user = serializers.PrimaryKeyRelatedField(
|
|
label='交送人', queryset=User.objects.all())
|
|
send_date = serializers.DateField(label='交送日期')
|
|
count = serializers.IntegerField(label='交送数量')
|
|
|
|
def validate(self, attrs):
|
|
if attrs['count'] <= 1:
|
|
raise ValidationError('交送数量必须大于1')
|
|
return attrs
|
|
|
|
|
|
class MlogAnaSerializer(serializers.Serializer):
|
|
belong_dept_name = serializers.CharField(label='车间名', required=False)
|
|
start_date = serializers.DateField(label='开始日期', required=True)
|
|
end_date = serializers.DateField(label='结束日期', required=True)
|
|
material_cate = serializers.CharField(label='物料系列', required=False)
|
|
|
|
|
|
class AttLogSerializer(CustomModelSerializer):
|
|
mgroup_name = serializers.CharField(
|
|
source='sflog.mgroup.name', read_only=True)
|
|
user_name = serializers.CharField(source='user.name', read_only=True)
|
|
post_name = serializers.CharField(source='post.name', read_only=True)
|
|
belong_dept_name = serializers.CharField(
|
|
source='sflog.mgroup.belong_dept.name', read_only=True)
|
|
|
|
class Meta:
|
|
model = AttLog
|
|
fields = '__all__'
|