import logging import base64 import hashlib import requests import os from datetime import datetime from stl import mesh # from OCC.Core.GProp import GProp_GProps from OCC.Extend.DataExchange import read_step_file from OCC.Extend.DataExchange import write_stl_file from odoo.addons.sf_base.commons.common import Common from odoo import models, fields, api from odoo.modules import get_resource_path from odoo.exceptions import ValidationError, UserError class QuickEasyOrder(models.Model): _name = 'quick.easy.order' _inherit = ['mail.thread', 'mail.activity.mixin'] _description = '简易下单' _order = 'id desc' name = fields.Char('订单编号', default=lambda self: self.env['ir.sequence'].next_by_code('quick.easy.order')) model_length = fields.Float('长(mm)', digits=(16, 3)) model_width = fields.Float('宽(mm)', digits=(16, 3)) model_height = fields.Float('高(mm)', digits=(16, 3)) model_volume = fields.Float('体积(mm³)', digits=(16, 3)) model_processing_side = fields.Char('加工面', default='A') model_feature = fields.Char('特征') machining_precision = fields.Selection([ ('0.10', '±0.10mm'), ('0.05', '±0.05mm'), ('0.03', '±0.03mm'), ('0.02', '±0.02mm'), ('0.01', '±0.01mm')], string='加工精度', default='0.10') material_id = fields.Many2one('sf.production.materials', '材料', tracking=True) material_model_id = fields.Many2one('sf.materials.model', '型号', domain="[('materials_id', '=', material_id)]", tracking=True) # process_id = fields.Many2one('sf.production.process', string='表面工艺') parameter_ids = fields.Many2many('sf.production.process.parameter', 'process_item_order_rel', string='可选参数', tracking=True) quantity = fields.Integer('数量', default=1, tracking=True) unit_price = fields.Float('单价', tracking=True) price = fields.Float('总价', tracking=True) model_file = fields.Binary('glb模型文件') upload_model_file = fields.Many2many('ir.attachment', 'upload_qf_model_file_attachment_ref', string='上传模型文件') delivery_time = fields.Date('交货日期') customer_id = fields.Many2one('res.partner', string='客户', default=lambda self: self.env.user.partner_id.id) state = fields.Selection([('草稿', '草稿'), ('待派单', '待派单'), ('待接单', '待接单'), ('加工中', '加工中'), ('物流中', '物流中'), ('已交付', '已交付')], string='订单状态', default='草稿', readonly=True, tracking=True) model_color_state = fields.Selection([ ('success', '成功'), ('fail', '失败')], string='模型上色状态', tracking=True) processing_time = fields.Integer('加工时长(min)') sale_order_id = fields.Many2one('sale.order', '销售订单号') @api.depends('unit_price', 'quantity') def _compute_total_amount(self): for item in self: item.price = item.unit_price * item.quantity @api.depends('material_id', 'material_model_id') def _compute_material_model(self): for item in self: materials = self.env['sf.production.materials'].search([], limit=1, order='id desc') item.material_id = materials.id item.material_model_id = self.env['sf.materials.model'].search( [('materials_id', '=', materials.id)], limit=1, order='id desc') @api.model def create(self, vals): if vals.get('upload_model_file'): logging.info('create-attachment:%s' % vals['upload_model_file'][0]) for item in vals['upload_model_file']: print(len(item[2])) if len(item[2]) > 0: logging.info('create-attachment:%s' % int(item[2][0])) attachment = self.env['ir.attachment'].sudo().search([('id', '=', int(item[2][0]))]) base64_data = base64.b64encode(attachment.datas) base64_datas = base64_data.decode('utf-8') model_code = hashlib.sha1(base64_datas.encode('utf-8')).hexdigest() report_path = attachment._full_path(attachment.store_fname) obj = super(QuickEasyOrder, self).create(vals) logging.info('---------向cloud生成模型库记录-------') self.model_coloring(obj) logging.info('---------开始派单到工厂-------') sale_order = self.distribute_to_factory(obj) obj.sale_order_id = sale_order.id obj.state = '待接单' return obj # 将attach的datas内容转为glb文件 def transition_glb_file(self, report_path, model_code): shapes = read_step_file(report_path) output_file = os.path.join('/tmp', str(model_code) + '.stl') write_stl_file(shapes, output_file, 'binary', 0.03, 0.5) # 转化为glb output_glb_file = os.path.join('/tmp', str(model_code) + '.glb') util_path = get_resource_path('sf_base', 'static/util') cmd = 'python3 %s/stl2gltf.py %s %s -b' % (util_path, output_file, output_glb_file) os.system(cmd) # 转base64 with open(output_glb_file, 'rb') as fileObj: image_data = fileObj.read() base64_data = base64.b64encode(image_data) return base64_data # return False @api.onchange('upload_model_file') def onchange_model_file(self): for item in self: if len(item.upload_model_file) > 1: raise ValidationError('只允许上传一个文件') if item.upload_model_file: file_attachment_id = item.upload_model_file[0] # 附件路径 report_path = file_attachment_id._full_path(file_attachment_id.store_fname) logging.info("模型路径: %s" % report_path) base64_data = base64.b64encode(file_attachment_id.datas) base64_datas = base64_data.decode('utf-8') model_code = hashlib.sha1(base64_datas.encode('utf-8')).hexdigest() # 读取文件 shapes = read_step_file(report_path) # # 第一种获取体积方式 # props = GProp_GProps() # # brepgprop_VolumeProperties(shapes, props) # volume = props.Mass() # # print(volume) # item.model_volume = volume # # 长宽高/体积 # output_file = os.path.join('C:/Users/43484/Desktop/机企猫工作文档', str(model_code) + '.stl') output_file = os.path.join('/tmp', str(model_code) + '.stl') # your_mesh = mesh.Mesh.from_file(output_file) # volume, cog, inertia = your_mesh.get_mass_properties() # xyz = (your_mesh.max_ - your_mesh.min_) # item.model_length = xyz[0] # 长 单位mm # item.model_width = xyz[1] # 宽 # item.model_height = xyz[2] # 高 # item.model_volume = volume write_stl_file(shapes, output_file, 'binary', 0.03, 0.5) # 转化为glb # output_glb_file = os.path.join('C:/Users/43484/Desktop/机企猫工作文档', str(model_code) + '.glb') output_glb_file = os.path.join('/tmp', str(model_code) + '.glb') util_path = get_resource_path('sf_base', 'static/util') cmd = 'python3 %s/stl2gltf.py %s %s -b' % (util_path, output_file, output_glb_file) os.system(cmd) # 转base64 with open(output_glb_file, 'rb') as fileObj: image_data = fileObj.read() base64_data = base64.b64encode(image_data) item.model_file = base64_data else: item.model_length = False item.model_width = False item.model_height = False item.model_volume = False item.model_file = False def distribute_to_factory(self, obj): """ 派单到工厂 :return: """ try: logging.info('---------派单到工厂-------') res = {'bfm_process_order_list': []} for item in obj: attachment = item.upload_model_file[0] base64_data = base64.b64encode(attachment.datas) base64_datas = base64_data.decode('utf-8') barcode = hashlib.sha1(base64_datas.encode('utf-8')).hexdigest() # logging.info('model_file-size: %s' % len(item.model_file)) res['bfm_process_order_list'].append({ 'model_long': item.model_length, 'model_width': item.model_width, 'model_height': item.model_height, 'model_volume': item.model_volume, 'model_machining_precision': item.machining_precision, 'model_name': attachment.name, 'model_data': base64_datas, 'model_file': base64.b64encode(item.model_file).decode('utf-8'), 'texture_code': item.material_id.materials_no, 'texture_type_code': item.material_model_id.materials_no, # 'surface_process_code': self.env['jikimo.surface.process']._json_surface_process_code(item), 'process_parameters_code': self.env[ 'sf.production.process.parameter']._json_production_process_item_code( item), 'price': item.price, 'number': item.quantity, 'total_amount': item.price, 'remark': '', 'manual_quotation': True, 'barcode': barcode }) # res['bfm_process_order_list'] = json.dumps(res['bfm_process_order_list']) product_id = self.env.ref('sf_dlm.product_template_sf').sudo() self_machining_id = self.env.ref('sf_dlm.product_embryo_sf_self_machining').sudo() outsource_id = self.env.ref('sf_dlm.product_embryo_sf_outsource').sudo() purchase_id = self.env.ref('sf_dlm.product_embryo_sf_purchase').sudo() company_id = self.env.ref('base.main_company').sudo() # user_id = request.env.ref('base.user_admin').sudo() order_id = self.env['sale.order'].sale_order_create(company_id, 'XXXXX', 'XXXXX', 'XXXXX', str(datetime.now()), '现结', '支付宝') i = 1 # 给sale_order的default_code字段赋值 aa = self.env['sale.order'].sudo().search([('name', '=', order_id.name)]) logging.info('---------aa------- %s' % aa.name) aa.default_code = obj.name for item in res['bfm_process_order_list']: product = self.env['product.template'].sudo().product_create(product_id, item, order_id, obj.name, i) bom_data = self.env['mrp.bom'].get_bom(product) logging.info('bom_data:%s' % bom_data) if bom_data: bom = self.env['mrp.bom'].bom_create(product, 'normal', False) bom.bom_create_line_has(bom_data) else: if product.materials_type_id.gain_way == '自加工': # 创建坯料 self_machining_embryo = self.env['product.template'].sudo().no_bom_product_create( self_machining_id, item, order_id, 'self_machining', i, False) # 创建坯料的bom self_machining_bom = self.env['mrp.bom'].bom_create(self_machining_embryo, 'normal', False) # 创建坯料里bom的组件 self_machining_bom_line = self_machining_bom.bom_create_line(self_machining_embryo) if not self_machining_bom_line: self.cr.rollback() return UserError('该订单模型的材料型号在您分配的工厂里暂未有原材料,请先配置再进行分配') # 产品配置bom product_bom_self_machining = self.env['mrp.bom'].bom_create(product, 'normal', False) product_bom_self_machining.bom_create_line_has(self_machining_embryo) elif product.materials_type_id.gain_way == '外协': # 创建坯料 outsource_embryo = self.env['product.template'].sudo().no_bom_product_create(outsource_id, item, order_id, 'subcontract', i, False) if outsource_embryo == -3: self.cr.rollback() return UserError( '该订单模型的材料型号在您分配的工厂里暂未设置获取方式和供应商,请先配置再进行分配') # 创建坯料的bom outsource_bom = self.env['mrp.bom'].bom_create(outsource_embryo, 'subcontract', True) # 创建坯料的bom的组件 outsource_bom_line = outsource_bom.with_user( self.env.ref("base.user_admin")).bom_create_line(outsource_embryo) if not outsource_bom_line: self.cr.rollback() return UserError('该订单模型的材料型号在您分配的工厂里暂未有原材料,请先配置再进行分配') # 产品配置bom product_bom_outsource = self.env['mrp.bom'].bom_create(product, 'normal', False) product_bom_outsource.bom_create_line_has(outsource_embryo) elif product.materials_type_id.gain_way == '采购': purchase_embryo = self.env['product.template'].sudo().no_bom_product_create(purchase_id, item, order_id, 'purchase', i, False) if purchase_embryo == -3: self.cr.rollback() return UserError( '该订单模型的材料型号在您分配的工厂里暂未设置获取方式和供应商,请先配置再进行分配') # 产品配置bom product_bom_purchase = self.env['mrp.bom'].bom_create(product, 'normal', False) product_bom_purchase.bom_create_line_has(purchase_embryo) order_id.with_user(self.env.ref("base.user_admin")).sale_order_create_line(product, item) return order_id except Exception as e: # self.cr.rollback() return UserError('工厂创建销售订单和产品失败,请联系管理员') # 模型上色 def model_coloring(self, order): url = '/api/library_of_models/create' config = self.env['res.config.settings'].get_values() config_header = Common.get_headers(self, config['token'], config['sf_secret_key']) logging.info('order: %s' % order.name) if order: attachment = order.upload_model_file[0] base64_data = base64.b64encode(attachment.datas) base64_datas = base64_data.decode('utf-8') model_code = hashlib.sha1(base64_datas.encode('utf-8')).hexdigest() logging.info('model_file-size: %s' % len(order.model_file)) logging.info('attachment.datas-size: %s' % len(attachment.datas)) vals = { 'programme_way': 'manual operation', 'model_code': model_code, 'model_data': base64.b64decode(attachment.datas), 'model_color_data': '', 'model_name': attachment.name, 'model_long': order.model_length, 'model_width': order.model_width, 'model_height': order.model_height, 'model_volume': order.model_volume, 'model_color_path': '/tmp/' + str(model_code) + ".step", 'model_order_no': '%s-%s' % (order.name, 1), 'remark': '订单号:%s 客户:%s' % (order.name, order.customer_id.name) } try: ret = requests.post((config['sf_url'] + url), json={}, data=vals, headers=config_header, timeout=60) ret = ret.json() # result = json.loads(ret['result']) if ret['status'] == 1: self.model_color_state = 'success' else: self.model_color_state = 'fail' raise UserError(ret['message']) except Exception as e: self.model_color_state = 'fail' raise UserError("模型上色失败")