682 lines
37 KiB
Python
682 lines
37 KiB
Python
# -*- coding: utf-8 -*-
|
||
import logging
|
||
import json
|
||
from datetime import datetime
|
||
|
||
from odoo.addons.sf_manufacturing.models.agv_scheduling import RepeatTaskException
|
||
from odoo import http
|
||
from odoo.http import request
|
||
from odoo.addons.sf_base.decorators.api_log import api_log
|
||
|
||
|
||
class Manufacturing_Connect(http.Controller):
|
||
|
||
@http.route('/AutoDeviceApi/GetWoInfo', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('获取工单', requester='中控系统')
|
||
def get_Work_Info(self, **kw):
|
||
"""
|
||
自动化传递工单号获取工单信息
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('get_Work_Info:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True, 'Datas': []}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/GetWoInfo'})
|
||
logging.info('RfidCode:%s' % ret['RfidCode'])
|
||
if 'RfidCode' in ret:
|
||
workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('rfid_code', '=', ret['RfidCode']), ('state', '!=', 'rework')])
|
||
if workorder:
|
||
for item in workorder:
|
||
res['Datas'].append({
|
||
'BillId': item.production_id.name,
|
||
'ProductionLine': item.production_id.production_line_id.id,
|
||
'SortId': item.sequence,
|
||
'CraftName': item.name,
|
||
'Quantity': 1,
|
||
'MaterialId': item.product_id.default_code,
|
||
'MaterialName': item.product_id.name,
|
||
'Spec': '%s×%s×%s' % (item.move_raw_ids.materiel_length, item.move_raw_ids.materiel_width,
|
||
item.move_raw_ids.materiel_height),
|
||
'Material': item.product_id.materials_type_id.name
|
||
})
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 203, 'Error': '该rfid暂未有对应的工单'}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 201, 'Error': '未传RfidCode字段'}
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('get_Work_Info error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/GetShiftPlan', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('获取日计划', requester='中控系统')
|
||
def get_ShiftPlan(self, **kw):
|
||
"""
|
||
自动化每天获取机台日计划
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('get_ShiftPlan:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True, 'Datas': []}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/GetShiftPlan'})
|
||
if 'ProductionLine' in ret:
|
||
workorder_ids = request.env['mrp.workorder'].sudo().get_plan_workorder(ret['ProductionLine'])
|
||
else:
|
||
ProductionLine = False
|
||
workorder_ids = request.env['mrp.workorder'].sudo().get_plan_workorder(ProductionLine)
|
||
# todo 需要筛选出CNC工单
|
||
logging.info('RfidCode:%s' % ret)
|
||
logging.info('workorder_ids:%s' % workorder_ids)
|
||
workorder = request.env['mrp.workorder'].sudo().search(workorder_ids)
|
||
if workorder:
|
||
for item in workorder:
|
||
date_planned_start = ''
|
||
date_planned_finished = ''
|
||
if item.date_planned_start is not False:
|
||
logging.info('date_planned_start:%s' % item.date_planned_start)
|
||
planned_start = item.date_planned_start.strftime("%Y-%m-%d %H:%M:%S")
|
||
date_planned_start = request.env['sf.sync.common'].sudo().get_add_time(planned_start)
|
||
if item.date_planned_finished is not False:
|
||
logging.info('date_planned_finished:%s' % item.date_planned_finished)
|
||
planned_finished = item.date_planned_finished.strftime("%Y-%m-%d %H:%M:%S")
|
||
date_planned_finished = request.env['sf.sync.common'].sudo().get_add_time(planned_finished)
|
||
res['Datas'].append({
|
||
'BillId': item.production_id.name,
|
||
'RfidCode': item.rfid_code,
|
||
'CraftName': item.name,
|
||
'Quantity': 1,
|
||
'WortkStart': date_planned_start,
|
||
'WorkEnd': date_planned_finished,
|
||
'MaterialId': item.product_id.default_code,
|
||
'MaterialName': item.product_id.name,
|
||
# 'Spec':item.mat,
|
||
'Material': item.product_id.materials_type_id.name
|
||
})
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('get_ShiftPlan error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/QcCheck', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('工件预调(前置三元检测)', requester='中控系统')
|
||
def get_qcCheck(self, **kw):
|
||
"""
|
||
工件预调(前置三元检测)
|
||
1、前置三元检测在产线外:三元检测设备把测量信息上传给MES,
|
||
MES生成检测定位数据。中控系统传递RFID编号给MES获取测量偏置结果。(来源为三元检测工单上的字段)
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('get_qcCheck:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True, 'Datas': []}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/QcCheck'})
|
||
logging.info('RfidCode:%s' % ret['RfidCode'])
|
||
if 'RfidCode' in ret:
|
||
workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('routing_type', '=', '装夹预调'), ('rfid_code', '=', ret['RfidCode']), ('state', '!=', 'rework')],
|
||
limit=1, order='id asc')
|
||
if workorder:
|
||
for item in workorder:
|
||
if item.material_center_point:
|
||
offset = item.material_center_point[1:-1].split(",")
|
||
res['Datas'].append({
|
||
'XOffset': 0 if not item.material_center_point else offset[0],
|
||
'YOffset': 0 if not item.material_center_point else offset[1],
|
||
'ZOffset': 0 if not item.material_center_point else offset[2],
|
||
'COffset': 0,
|
||
'Coordinate': 'G54'
|
||
})
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 203, 'Error': '该rfid暂未有对应的工件预调(前置三元检测)工单'}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 201, 'Error': '未传RfidCode字段'}
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('get_qcCheck error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/FeedBackStart', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('工单开始', requester='中控系统')
|
||
def button_Work_START(self, **kw):
|
||
"""
|
||
工单任务开始
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('button_Work_START:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True, 'Datas': ['工单已开始']}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/FeedBackStart'})
|
||
production_id = ret['BillId']
|
||
routing_type = ret['CraftId']
|
||
equipment_id = ret["DeviceId"]
|
||
workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('production_id', '=', production_id), ('routing_type', '=', routing_type),
|
||
('rfid_code', '!=', False), ('state', '!=', 'rework')], limit=1)
|
||
if not workorder:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': '该工单不存在'}
|
||
return json.JSONEncoder().encode(res)
|
||
logging.info('workorder_state:%s' % workorder.state)
|
||
if workorder.state != 'ready':
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': '工单未就绪'}
|
||
return json.JSONEncoder().encode(res)
|
||
work_equipment_id = request.env['maintenance.equipment'].sudo().search([('name', '=', equipment_id)],
|
||
limit=1)
|
||
logging.info('work_equipment_id:%s' % work_equipment_id.name)
|
||
if not work_equipment_id:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': '没有找到该加工设备'}
|
||
return json.JSONEncoder().encode(res)
|
||
workorder.equipment_id = work_equipment_id
|
||
workorder.button_start()
|
||
|
||
# 根据工单的实际开始时间修改排程单的开始时间、状态
|
||
if workorder.date_start:
|
||
request.env['sf.production.plan'].sudo().search([('production_id', '=', production_id)]).write(
|
||
{'actual_start_time': workorder.date_start,
|
||
'state': 'processing'})
|
||
res.update({'workorder_id': workorder.id})
|
||
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('button_Work_START error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/FeedBackEnd', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('工单结束', requester='中控系统')
|
||
def button_Work_End(self, **kw):
|
||
"""
|
||
工单任务结束
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('button_Work_End:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True, 'Datas': ['工单已结束']}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
logging.info('button_Work_End:%s' % ret)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/FeedBackEnd'})
|
||
production_id = ret['BillId']
|
||
routing_type = ret['CraftId']
|
||
workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('production_id', '=', production_id), ('routing_type', '=', routing_type),
|
||
('rfid_code', '!=', False), ('state', '!=', 'rework')], limit=1)
|
||
if not workorder:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': '该工单不存在'}
|
||
return json.JSONEncoder().encode(res)
|
||
if workorder.state != 'progress':
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': '该工单未开始'}
|
||
return json.JSONEncoder().encode(res)
|
||
# workorder.write({'date_finished': datetime.now()})
|
||
if ret['IsComplete'] is True:
|
||
workorder.write({'date_finished': datetime.now()})
|
||
|
||
# workorder.process_state = '待解除装夹'
|
||
# workorder.sudo().production_id.process_state = '待解除装夹'
|
||
|
||
# 根据工单的实际结束时间修改排程单的结束时间、状态,同时修改销售订单的状态
|
||
# if workorder.date_finished:
|
||
# request.env['sf.production.plan'].sudo().search([('production_id', '=', production_id)]).write(
|
||
# {'actual_end_time': workorder.date_finished,
|
||
# 'state': 'finished'})
|
||
# production_obj = request.env['mrp.production'].sudo().search([('name', '=', production_id)])
|
||
# if production_obj:
|
||
# production_obj.sudo().work_order_state = '已完成'
|
||
# production_obj.write({'state': 'done'})
|
||
# request.env['sale.order'].sudo().search(
|
||
# [('name', '=', production_obj.origin)]).write({'schedule_status': 'to deliver'})
|
||
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('button_Work_End error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/PartQualityInspect', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('零件检测(后置三元检测)', requester='中控系统')
|
||
def PartQualityInspect(self, **kw):
|
||
"""
|
||
零件质检
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('PartQualityInspect:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/PartQualityInspect'})
|
||
production_id = ret['BillId']
|
||
# routing_type = ret['CraftId']
|
||
workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('production_id', '=', production_id), ('routing_type', '=', 'CNC加工'),
|
||
('state', 'not in', ['rework', 'done', 'cancel'])], order='sequence asc',
|
||
limit=1)
|
||
if workorder:
|
||
# workorder.test_results = ret['Quality']
|
||
logging.info('制造订单:%s' % workorder.production_id.name)
|
||
if 'ReportPaht' in ret:
|
||
if ret['ReportPaht'].find('.pdf') != -1:
|
||
download_state = request.env['mrp.workorder'].with_user(
|
||
request.env.ref("base.user_admin")).download_reportfile_tmp(workorder,
|
||
ret['ReportPaht'])
|
||
if download_state is True:
|
||
detection_ret = request.env['mrp.workorder'].with_user(
|
||
request.env.ref("base.user_admin")).get_detection_file(workorder, ret['ReportPaht'])
|
||
logging.info('detection_ret:%s' % detection_ret)
|
||
if detection_ret is False:
|
||
res = {'Succeed': False, 'ErrorCode': 205, 'Error': '检测报告文件读取失败'}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 204, 'Error': '检测报告文件从FTP拉取失败'}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 203, 'Error': '未传ReportPaht字段'}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 206, 'Error': '未查询到工单'}
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('PartQualityInspect error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/CMMProgDolod', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('CMM测量程序下载', requester='中控系统')
|
||
def CMMProgDolod(self, **kw):
|
||
"""
|
||
中控系统传递RFID编号给MES,获取测量程序文件。Ftp下载文件
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('CMMProgDolod:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True, 'Datas': []}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/CMMProgDolod'})
|
||
if 'RfidCode' in ret:
|
||
logging.info('RfidCode:%s' % ret['RfidCode'])
|
||
workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('rfid_code', '=', ret['RfidCode']), ('routing_type', '=', 'CNC加工'), ('state', '!=', 'rework')])
|
||
if workorder:
|
||
for item in workorder.cmm_ids:
|
||
if item.program_create_date is not False:
|
||
program_create_date = item.program_create_date.strftime("%Y-%m-%d %H:%M:%S")
|
||
program_date_str = request.env['sf.sync.common'].sudo().get_add_time(program_create_date)
|
||
res['Datas'].append({
|
||
'CraftId': workorder.id,
|
||
'CraftKey': workorder.name,
|
||
'ProgramDate': '' if not item.program_create_date else program_date_str,
|
||
'ProgramPath': item.program_path,
|
||
'PostProcessing': item.program_name,
|
||
})
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 203, 'Error': '暂无工单及对应的CNC程序数据'}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 201, 'Error': '未传RfidCode字段'}
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('CMMProgDolod error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/NCProgDolod', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('CAM加工程序下载', requester='中控系统')
|
||
def NCProgDolod(self, **kw):
|
||
"""
|
||
中控系统传递RFID编号给MES,获取程序单及程序文件。Ftp下载文件
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('NCProgDolod:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True, 'Datas': []}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/NCProgDolod'})
|
||
if 'RfidCode' in ret:
|
||
logging.info('RfidCode:%s' % ret['RfidCode'])
|
||
workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('rfid_code', '=', ret['RfidCode']), ('routing_type', '=', 'CNC加工'), ('state', '!=', 'rework')])
|
||
if workorder:
|
||
for item in workorder.cnc_ids:
|
||
res['Datas'].append({
|
||
'CraftId': workorder.id,
|
||
'CraftName': workorder.name,
|
||
'SortId': item.sequence_number,
|
||
'ProgramName': item.program_name,
|
||
'ToolId': item.cutting_tool_no,
|
||
'ToolName': item.cutting_tool_name,
|
||
'Depth': item.depth_of_processing_z,
|
||
'ProgramPath': item.program_path,
|
||
'ProgramTime': item.estimated_processing_time,
|
||
})
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 203, 'Error': '暂无工单及对应的CNC程序数据'}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 201, 'Error': '未传RfidCode字段'}
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('NCProgDolod error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/LocationChange', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('库位变更', requester='中控系统')
|
||
def LocationChange(self, **kw):
|
||
"""
|
||
库位变更
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('LocationChange:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True, 'Datas': []}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/LocationChange'})
|
||
logging.info('库位变更LocationChange_ret:%s' % ret)
|
||
RfidCode = ret['RfidCode']
|
||
ChangeType = ret['ChangeType']
|
||
OldDeciveId = ret['OldDeciveId']
|
||
OldPosition = ret['OldPosition']
|
||
NewDeciveId = ret['NewDeciveId']
|
||
NewPosition = ret['NewPosition']
|
||
OldDeciveStart = ret['OldDeciveStart']
|
||
OldDeciveEnd = ret['OldDeciveEnd']
|
||
|
||
if ChangeType == 'Part':
|
||
temp_val_sn_id = None
|
||
old_localtion = None
|
||
# if ChangeType == 'Part' or ChangeType == 'Tool':
|
||
stock_lot_obj = request.env['stock.lot'].sudo().search(
|
||
[('rfid', '=', RfidCode)], limit=1)
|
||
logging.info('stock_lot_obj===========:%s' % stock_lot_obj)
|
||
if not stock_lot_obj:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': '未根据RfidCode找到该产品'}
|
||
return json.JSONEncoder().encode(res)
|
||
if OldPosition:
|
||
old_localtion = request.env['sf.shelf.location'].sudo().search(
|
||
[('barcode', '=', OldPosition)], limit=1)
|
||
logging.info('old_localtion===========:%s' % old_localtion)
|
||
new_localtion = request.env['sf.shelf.location'].sudo().search(
|
||
[('barcode', '=', NewPosition)], limit=1)
|
||
logging.info('new_localtion===========:%s' % new_localtion)
|
||
if not new_localtion:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': '没有该目标位置'}
|
||
return json.JSONEncoder().encode(res)
|
||
if old_localtion:
|
||
temp_val_sn_id = old_localtion.product_sn_id
|
||
logging.info('temp_val_sn_id===========:%s' % temp_val_sn_id)
|
||
old_localtion.product_sn_id = None
|
||
new_localtion.product_sn_id = temp_val_sn_id
|
||
logging.info('====1======')
|
||
else:
|
||
new_localtion.product_sn_id = stock_lot_obj.id
|
||
logging.info('=====2======')
|
||
elif ChangeType == 'Tool':
|
||
# 对功能刀具库位变更信息进行更改
|
||
def write_tool(DeciveId):
|
||
if 'Tool' in DeciveId:
|
||
shelfinfo = list(filter(lambda x: x.get('DeviceId') == DeciveId,
|
||
request.env['sf.shelf.location'].sudo().get_sf_shelf_location_info(
|
||
DeciveId)))
|
||
request.env['sf.shelf.location.datasync'].sudo().set_shelf_location(shelfinfo)
|
||
else:
|
||
equipment_id = request.env['maintenance.equipment'].sudo().search([('name', '=', DeciveId)])
|
||
if equipment_id:
|
||
equipment_id.sudo().register_equipment_tool()
|
||
else:
|
||
res_1 = {'Succeed': False, 'ErrorCode': 202, 'Error': f'设备【{DeciveId}】不存在'}
|
||
return json.JSONEncoder().encode(res_1)
|
||
|
||
if OldDeciveId:
|
||
write_tool(OldDeciveId)
|
||
elif NewDeciveId:
|
||
write_tool(NewDeciveId)
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('LocationChange error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/AGVToProduct', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('AGV运送上产线', requester='中控系统')
|
||
def AGVToProduct(self, **kw):
|
||
"""
|
||
AGV运送上产线(完成)
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('AGVToProduct:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/AGVToProduct'})
|
||
logging.info('ret:%s' % ret)
|
||
if 'DeviceId' in ret:
|
||
logging.info('DeviceId:%s' % ret['DeviceId'])
|
||
if 'IsComplete' in ret:
|
||
rfid_codes = []
|
||
workorder_ids = []
|
||
if ret['IsComplete'] is True or ret['IsComplete'] is False:
|
||
for i in range(1, 5):
|
||
logging.info('F-RfidCode:%s' % i)
|
||
if f'RfidCode{i}' in ret:
|
||
rfid_code = ret[f'RfidCode{i}']
|
||
logging.info('RfidCode:%s' % rfid_code)
|
||
if rfid_code is not None and rfid_code != '':
|
||
rfid_codes.append(rfid_code)
|
||
domain = [
|
||
('rfid_code', '=', rfid_code),
|
||
('routing_type', '=', 'CNC加工'), ('state', '!=', 'rework')
|
||
]
|
||
workorder = request.env['mrp.workorder'].sudo().search(domain, order='id asc')
|
||
if workorder:
|
||
for order in workorder:
|
||
workorder_ids.append(order.id)
|
||
if order.production_line_state == '待上产线':
|
||
logging.info(
|
||
'工单产线状态:%s' % order.production_line_state)
|
||
panel_workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('rfid_code', '=', rfid_code), ('state', '!=', 'rework'),
|
||
('processing_panel', '=', order.processing_panel)])
|
||
if panel_workorder:
|
||
panel_workorder.write({'production_line_state': '已上产线'})
|
||
# workpiece_delivery = request.env['sf.workpiece.delivery'].sudo().search(
|
||
# [
|
||
# ('rfid_code', '=', rfid_code), ('type', '=', '上产线'),
|
||
# ('production_id', '=', order.production_id.id),
|
||
# ('workorder_id', '=', order.id),
|
||
# ('workorder_state', '=', 'done')])
|
||
# if workpiece_delivery.status == '待下发':
|
||
# workpiece_delivery.write({'is_manual_work': True})
|
||
# 下发
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 204,
|
||
'Error': 'DeviceId为%s没有对应的已配送工件数据' % ret['DeviceId']}
|
||
if ret['IsComplete'] is True:
|
||
# 向AGV任务调度下发运送空料架任务
|
||
workorders = request.env['mrp.workorder'].browse(workorder_ids)
|
||
request.env['sf.agv.scheduling'].add_scheduling(ret['DeviceId'], '运送空料架', workorders)
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 203, 'Error': '未传IsComplete字段'}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 201, 'Error': '未传DeviceId字段'}
|
||
except RepeatTaskException as e:
|
||
logging.info('AGVToProduct error:%s' % e)
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': str(e)}
|
||
logging.info('AGVToProduct error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/AGVDownProduct', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
@api_log('AGV运送下产线', requester='中控系统')
|
||
def AGVDownProduct(self, **kw):
|
||
"""
|
||
MES调度AGV,搬运零件AGV托盘到产线接驳站。
|
||
生产线接受到零件AGV托盘到位信号后,把生产合格或特采的零件,机器人搬运零件到AGV接驳站中,触发AGV运送下产线接口。
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('AGVDownProduct:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/AGVDownProduct'})
|
||
logging.info('ret:%s' % ret)
|
||
if 'DeviceId' in ret:
|
||
logging.info('DeviceId:%s' % ret['DeviceId'])
|
||
# delivery_Arr = []
|
||
workorder_ids = []
|
||
if 'IsComplete' in ret:
|
||
if ret['IsComplete'] is True or ret['IsComplete'] is False:
|
||
for i in range(1, 5):
|
||
logging.info('F-RfidCode:%s' % i)
|
||
if f'RfidCode{i}' in ret:
|
||
rfid_code = ret[f'RfidCode{i}']
|
||
logging.info('RfidCode:%s' % rfid_code)
|
||
if rfid_code is not None and rfid_code != '':
|
||
domain = [
|
||
('rfid_code', '=', rfid_code),
|
||
('routing_type', '=', 'CNC加工'), ('state', '!=', 'rework')
|
||
]
|
||
workorder = request.env['mrp.workorder'].sudo().search(domain, order='id asc')
|
||
if workorder:
|
||
for order in workorder:
|
||
workorder_ids.append(order.id)
|
||
if order.production_line_state == '已上产线':
|
||
logging.info(
|
||
'工单产线状态:%s' % order.production_line_state)
|
||
panel_workorder = request.env['mrp.workorder'].sudo().search(
|
||
[('rfid_code', '=', rfid_code), ('state', '!=', 'rework'),
|
||
('processing_panel', '=', order.processing_panel)])
|
||
if panel_workorder:
|
||
panel_workorder.write({'production_line_state': '已下产线'})
|
||
workorder.write({'state': 'to be detected'})
|
||
workorder.check_ids.filtered(
|
||
lambda ch: ch.quality_state == 'waiting').write(
|
||
{'quality_state': 'none'})
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 204,
|
||
'Error': 'DeviceId为%s没有对应的已配送工件数据' % ret['DeviceId']}
|
||
# if delivery_Arr:
|
||
# logging.info('delivery_Arr:%s' % delivery_Arr)
|
||
# delivery_workpiece = request.env['sf.workpiece.delivery'].sudo().search(
|
||
# [('id', 'in', delivery_Arr)])
|
||
# if delivery_workpiece:
|
||
# logging.info('开始向agv下发下产线任务')
|
||
# agv_site = request.env['sf.agv.site'].sudo().search([])
|
||
# if agv_site:
|
||
# has_site = agv_site.update_site_state()
|
||
# if has_site is True:
|
||
# is_free = delivery_workpiece._check_avgsite_state()
|
||
# if is_free is True:
|
||
# delivery_workpiece._delivery_avg()
|
||
# logging.info('agv下发下产线任务下发完成')
|
||
if ret['IsComplete'] is True:
|
||
# 向AGV任务调度下发下产线任务
|
||
workorders = request.env['mrp.workorder'].browse(workorder_ids)
|
||
request.env['sf.agv.scheduling'].add_scheduling(ret['DeviceId'], '下产线', workorders)
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 203, 'Error': '未传IsComplete字段'}
|
||
except RepeatTaskException as e:
|
||
logging.info('AGVToProduct error:%s' % e)
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': str(e)}
|
||
logging.info('AGVDownProduct error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/EquipmentBaseCoordinate', type='json', auth='sf_token', methods=['GET', 'POST'],
|
||
csrf=False,
|
||
cors="*")
|
||
def PutEquipmentBaseCoordinate(self, **kw):
|
||
"""
|
||
获取机床基坐标
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('PutEquipmentBaseCoordinate:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/EquipmentBaseCoordinate'})
|
||
if 'DeviceId' in ret:
|
||
equipment = request.env['maintenance.equipment'].sudo().search('name', '=', ret['DeviceId'])
|
||
if equipment:
|
||
equipment.sudo().write({
|
||
'base_coordinate_fixture_model_id': ret['base_coordinate_fixture_model_id'],
|
||
'base_coordinate_g_coordinate': ret['base_coordinate_g_coordinate'],
|
||
'base_coordinate_x': ret['base_coordinate_x'],
|
||
'base_coordinate_y': ret['base_coordinate_y'],
|
||
'base_coordinate_z': ret['base_coordinate_z'],
|
||
})
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 203, 'Error': 'DeviceId为%s的设备不存在!' % ret['DeviceId']}
|
||
else:
|
||
res = {'Succeed': False, 'ErrorCode': 201, 'Error': '未传DeviceId字段'}
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||
logging.info('AGVDownProduct error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|
||
|
||
@http.route('/AutoDeviceApi/AgvStationState', type='json', auth='public', methods=['GET', 'POST'], csrf=False,
|
||
cors="*")
|
||
def AGVStationState(self, **kw):
|
||
"""
|
||
中控推送接驳站状态
|
||
:param kw:
|
||
:return:
|
||
"""
|
||
logging.info('AGVStationState:%s' % kw)
|
||
try:
|
||
res = {'Succeed': True}
|
||
datas = request.httprequest.data
|
||
ret = json.loads(datas)
|
||
request.env['center_control.interface.log'].sudo().create(
|
||
{'content': ret, 'name': 'AutoDeviceApi/AGVStationState'})
|
||
logging.info('ret:%s' % ret)
|
||
ret = ret['param']
|
||
params = {}
|
||
for i in range(len(ret)):
|
||
if 'DeviceId' in ret[i] and 'AtHome' in ret[i]:
|
||
logging.info('DeviceId:%s, AtHome:%s' % (ret[i]['DeviceId'], ret[i]['AtHome']))
|
||
params[ret[i]['DeviceId']] = '占用' if ret[i]['AtHome'] else '空闲'
|
||
if params:
|
||
request.env['sf.agv.site'].update_site_state(params)
|
||
except Exception as e:
|
||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': str(e)}
|
||
logging.info('AGVDownProduct error:%s' % e)
|
||
return json.JSONEncoder().encode(res)
|