From 93191d3bebb564b2155c5b7c47a800fc5a539232 Mon Sep 17 00:00:00 2001 From: laodaming <11058467+laudamine@user.noreply.gitee.com> Date: Tue, 8 Aug 2023 12:22:15 +0800 Subject: [PATCH] fix --- initalize/rabbitmq.go | 2 +- model/gmodel/fs_cloud_render_log_logic.go | 6 + model/gmodel/fs_product_model3d_logic.go | 8 + model/gmodel/fs_product_template_v2_logic.go | 13 ++ .../render/consumer/assemble_render_data.go | 218 +++++++++++++++++- server/render/render.go | 1 + .../websocket/internal/logic/mq_consumer.go | 3 +- .../internal/logic/ws_render_image_logic.go | 3 +- server/websocket/websocket.go | 1 + utils/mq_consumer_factory/mq.go | 4 +- utils/websocket_data/render_data.go | 2 +- 11 files changed, 255 insertions(+), 6 deletions(-) diff --git a/initalize/rabbitmq.go b/initalize/rabbitmq.go index 004524c0..da7f4e7f 100644 --- a/initalize/rabbitmq.go +++ b/initalize/rabbitmq.go @@ -123,7 +123,7 @@ func (h *RabbitMqHandle) Consume(ctx context.Context, queueName constants.RABBIT <-limit wait.Done() }() - if err = handle.Run(m.Body); err != nil { + if err = handle.Run(ctx, m.Body); err != nil { logx.Error("failed to deal with MQ message:", string(m.Body)) return } diff --git a/model/gmodel/fs_cloud_render_log_logic.go b/model/gmodel/fs_cloud_render_log_logic.go index 35237a04..92f12587 100644 --- a/model/gmodel/fs_cloud_render_log_logic.go +++ b/model/gmodel/fs_cloud_render_log_logic.go @@ -1,3 +1,9 @@ package gmodel +import "context" + // TODO: 使用model的属性做你想做的 + +func (r *FsCloudRenderLogModel) Create(ctx context.Context, data *FsCloudRenderLog) error { + return r.db.WithContext(ctx).Model(&FsCloudRenderLog{}).Create(data).Error +} diff --git a/model/gmodel/fs_product_model3d_logic.go b/model/gmodel/fs_product_model3d_logic.go index 35270c57..9cb2fe91 100755 --- a/model/gmodel/fs_product_model3d_logic.go +++ b/model/gmodel/fs_product_model3d_logic.go @@ -93,3 +93,11 @@ func (d *FsProductModel3dModel) GetGroupPartListByProductIds(ctx context.Context Group("product_id").Find(&resp).Error return resp, err } +func (d *FsProductModel3dModel) FindOneJoinSize(ctx context.Context, productId int64) (resp FsProductModel3d, err error) { + err = d.db.WithContext(ctx).Table(d.name+"as m").Joins("left join fs_product_size as s on m.size_id = s.id"). + Select("m.*"). + Where("m.product_id = ?", productId). + Where("(s.status= ? and m.tag = ?)", 1, 1). + Order("s.sort ASC").Take(&resp).Error + return resp, err +} diff --git a/model/gmodel/fs_product_template_v2_logic.go b/model/gmodel/fs_product_template_v2_logic.go index 76bf7bf0..dc8a1b02 100755 --- a/model/gmodel/fs_product_template_v2_logic.go +++ b/model/gmodel/fs_product_template_v2_logic.go @@ -106,3 +106,16 @@ func (t *FsProductTemplateV2Model) GetProductTemplateListByParams(ctx context.Co err = db.Find(&resp).Error return resp, err } + +// 获取第一个尺寸下的模板 +func (t *FsProductTemplateV2Model) FindOneByProductIdTagIdWithSizeTable(ctx context.Context, productId int64, tagId string) (resp *FsProductTemplateV2, err error) { + err = t.db.WithContext(ctx).Table(t.name+" as t"). + Joins("left join fs_product_size as s on t.product_id = s.product_id"). + Select("t.*"). + Where("t.product_id = ? and t.tag = ? ", productId, tagId). + Where("t.status = ? and t.is_del = ?", 1, 0). + Where("s.status = ?", 1). + Order("s.sort ASC"). + Take(&resp).Error + return resp, err +} diff --git a/server/render/consumer/assemble_render_data.go b/server/render/consumer/assemble_render_data.go index 6e6cc6a3..4a1c6f75 100644 --- a/server/render/consumer/assemble_render_data.go +++ b/server/render/consumer/assemble_render_data.go @@ -1,22 +1,238 @@ package consumer import ( + "bytes" + "context" "encoding/json" + "errors" + "fmt" + "fusenapi/model/gmodel" "fusenapi/utils/websocket_data" "github.com/zeromicro/go-zero/core/logx" + "gorm.io/gorm" + "io/ioutil" + "net/http" + "time" ) +// 这里请求的py接口返回数据 +type pythonApiRsp struct { + Code int `json:"code"` + Msg string `json:"msg"` + Data []struct { + Tid int64 `json:"tid"` + Imgurl string `json:"imgurl"` + Costtime int64 `json:"costtime"` + } `json:"data"` +} + // 消费渲染需要组装的数据 type MqConsumerRenderAssemble struct { } -func (m *MqConsumerRenderAssemble) Run(data []byte) error { +func (m *MqConsumerRenderAssemble) Run(ctx context.Context, data []byte) error { logx.Info("收到需要组装的消息:", string(data)) var parseInfo websocket_data.AssembleRenderData if err := json.Unmarshal(data, &parseInfo); err != nil { logx.Error("MqConsumerRenderAssemble数据格式错误:", err) return nil //不返回错误就删除消息 } + val := ctx.Value("allmodels") + if val == nil { + return errors.New("allmodels is nil") + } + allmodels, ok := val.(*gmodel.AllModelsGen) + if !ok { + return errors.New("allmodels is nil!!") + } + timeSearchBegin := time.Now().UnixMilli() + //获取模板 + templateInfo, err := allmodels.FsProductTemplateV2.FindOneByProductIdTagIdWithSizeTable(ctx, parseInfo.RenderData.ProductId, fmt.Sprintf("%d", parseInfo.RenderData.TemplateTagId)) + if err != nil { + if errors.Is(err, gorm.ErrRecordNotFound) { + logx.Error("template info is not found") + return nil + } + logx.Error("failed to get template info:", err) + return err + } + renderLogTime := time.Now().UnixMilli() - timeSearchBegin + now := time.Now().Unix() + title := "1-组装模板数据" + //云渲染日志 + err = allmodels.FsCloudRenderLog.Create(ctx, &gmodel.FsCloudRenderLog{ + UserId: &parseInfo.RenderData.UserId, + Title: &title, + Time: &renderLogTime, + Tag: &parseInfo.RenderId, + Ctime: &now, + }) + if err != nil { + logx.Error(err) + } + pyapiBeginTime := time.Now().UnixMilli() + //这里curl post请求数据。获取处理好的贴图数据,用于贴model的贴图 + pythonPostData := map[string]interface{}{ + "tids": templateInfo.Id, + "data": parseInfo.RenderData.Data, + } + pyPostBytes, _ := json.Marshal(pythonPostData) + url := "http://110.41.19.98:8867/imgRender" + pyRsp, err := http.Post(url, "application/json;charset=UTF-8", bytes.NewReader(pyPostBytes)) + if err != nil { + logx.Error("request python render api err:", err) + return err + } + defer pyRsp.Body.Close() + pyRspBytes, err := ioutil.ReadAll(pyRsp.Body) + if err != nil { + logx.Error("failed to read python api rsp body,err=", err) + return err + } + var rspInfo pythonApiRsp + if err = json.Unmarshal(pyRspBytes, &rspInfo); err != nil { + logx.Error("failed to unmarshal python api rsp:", err) + return err + } + if rspInfo.Code != 200 { + logx.Error("python api 接口请求错误:", rspInfo.Msg) + return err + } + if len(rspInfo.Data) == 0 { + logx.Error("python api 接口没有数据:") + return err + } + //云渲染日志 + title = "2-请求->接收python合成刀版图接口" + now = time.Now().Unix() + pyRequestTime := time.Now().UnixMilli() - pyapiBeginTime + err = allmodels.FsCloudRenderLog.Create(ctx, &gmodel.FsCloudRenderLog{ + UserId: &parseInfo.RenderData.UserId, + Title: &title, + Time: &pyRequestTime, + Tag: &parseInfo.RenderId, + Ctime: &now, + }) + if err != nil { + logx.Error(err) + } + incTime := int64(0) + mapCurlData := make(map[int64]int) + for k, v := range rspInfo.Data { + mapCurlData[v.Tid] = k + incTime += v.Costtime + } + //云渲染日志 + title = "3-python合成刀版图" + now = time.Now().Unix() + postData := string(pyPostBytes) + pyRspStr := string(pyRspBytes) + err = allmodels.FsCloudRenderLog.Create(ctx, &gmodel.FsCloudRenderLog{ + UserId: &parseInfo.RenderData.UserId, + PostUrl: &url, + PostData: &postData, + Result: &pyRspStr, + Title: &title, + Time: &incTime, + Tag: &parseInfo.RenderId, + Ctime: &now, + }) + if err != nil { + logx.Error(err) + } + //获取渲染设置信息 + //element, err := allmodels.FsProductTemplateElement + /* + $element = ProductTemplateElement::find() + ->andFilterWhere(['in', 'product_template_id', $mids]) + ->asArray() + ->all(); + + $element = array_column($element, null, 'product_template_id'); + $elementTitles = array_column($element, 'title'); + + $result = []; + + $time_pinjie_begin = $render->getMillisecond(); + foreach ($templates as $key => $val) { + if(!isset($element[$val['model_id']]) || !isset($imageData[$val['id']])){ + continue; + } + //数据拼装 + $item = []; + + $item['light'] = $element[$val['model_id']]['light']; + $item['refletion'] = $element[$val['model_id']]['refletion'] == '' ? -1 : (int)$element[$val['model_id']]['refletion']; + $item['scale'] = $element[$val['model_id']]['scale']; + $item['sku_id'] = $val['product_id']; + $item['tid'] = $element[$val['model_id']]['title']; + $item['rotation'] = $element[$val['model_id']]['rotation']; + $item['filePath'] = '';//todo 文件路径,针对千人千面 + + //组装data数据 + $tempData = []; + //获取材质模式对应关系 + $mode = $element[$val['model_id']]['mode'] ? json_decode($element[$val['model_id']]['mode'], true) : []; + // $base_img = (new ImageService())->base64EncodeImageNoHeader(\Yii::$app->params['baseurl'].$imageData[$val['id']]['imgurl']); + $base_img = \Yii::$app->params['h5Url'].'/storage'.$imageData[$val['id']]['imgurl']; + //判断是否包含base数据 即对应建模那边的model + if($element[$val['model_id']]['base']){ + $tempData[] = [ + 'name' => 'model', + 'data' => '0,'.$base_img.','.$element[$val['model_id']]['base'], + 'type' => 'other', + 'layer' => '0', + 'is_update' => 1, + 'mode' => $mode['model'], + ]; + } + if($element[$val['model_id']]['shadow']){ + $tempData[] = [ + 'name' => 'shadow', + 'data' => $element[$val['model_id']]['shadow'], + 'type' => 'other', + 'layer' => '0', + 'is_update' => 0, + 'mode' => $mode['shadow'], + ]; + } + if($element[$val['model_id']]['model_p']){ + $tempData[] = [ + 'name' => 'model_P', + 'data' => '0,'.$element[$val['model_id']]['model_p'], + 'type' => 'other', + 'layer' => '0', + 'is_update' => 0, + 'mode' => $mode['model_P'], + ]; + } + $item['data'] = $tempData; + $result[] = $item; + + } + $log = new CloudRenderLog(); + $log->title = '接收到python刀版图 -> 3-组装MQ渲染任务队列'; + $log->time = $render->getMillisecond() - $time_pinjie_begin; + $log->user_id = $user_id; + $log->post_data = ''; + $log->post_url = ''; + $log->result = $res; + $log->tag = $inputData['id']; + $log->ctime = time(); + $log->save(false); + } + + $sendData = [ + 'id' => $inputData['id'], + 'order_id' => 0, + 'user_id' => \Yii::$app->user->id, + 'sku_ids' => $inputData['sku_ids'], + 'tids' => $elementTitles, + 'data' => $result, + 'is_thousand_face' => 0, + 'folder' => '',//todo 千人千面需要使用 + ]; + return $sendData;*/ return nil } diff --git a/server/render/render.go b/server/render/render.go index 1210715d..acb9d019 100644 --- a/server/render/render.go +++ b/server/render/render.go @@ -35,6 +35,7 @@ func main() { //消费渲染前组装数据队列 ctx1 := context.Background() ctx2, cancel := context.WithCancel(ctx1) + ctx2 = context.WithValue(ctx2, "allmodels", ctx.AllModels) defer cancel() go ctx.RabbitMq.Consume(ctx2, constants.RABBIT_MQ_ASSEMBLE_RENDER_DATA, &consumer.MqConsumerRenderAssemble{}) handler.RegisterHandlers(server, ctx) diff --git a/server/websocket/internal/logic/mq_consumer.go b/server/websocket/internal/logic/mq_consumer.go index c28ff2d6..7488e229 100644 --- a/server/websocket/internal/logic/mq_consumer.go +++ b/server/websocket/internal/logic/mq_consumer.go @@ -1,6 +1,7 @@ package logic import ( + "context" "encoding/json" "fusenapi/constants" "fusenapi/utils/websocket_data" @@ -11,7 +12,7 @@ import ( type MqConsumerRenderResult struct { } -func (m *MqConsumerRenderResult) Run(data []byte) error { +func (m *MqConsumerRenderResult) Run(ctx context.Context, data []byte) error { logx.Info("接收到MqConsumerRenderResult数据:", string(data)) var parseInfo websocket_data.RenderImageNotify if err := json.Unmarshal(data, &parseInfo); err != nil { diff --git a/server/websocket/internal/logic/ws_render_image_logic.go b/server/websocket/internal/logic/ws_render_image_logic.go index d4974870..978154d0 100644 --- a/server/websocket/internal/logic/ws_render_image_logic.go +++ b/server/websocket/internal/logic/ws_render_image_logic.go @@ -30,6 +30,7 @@ func (w *wsConnectItem) renderImage(data []byte) { return } logx.Info("收到请求云渲染图片数据:", renderImageData) + renderImageData.RenderData.UserId = w.userId //把需要渲染的图片任务加进去 taskId := hash.JsonHashKey(renderImageData.RenderData) w.renderProperty.renderImageTaskCtlChan <- renderImageControlChanItem{ @@ -39,7 +40,7 @@ func (w *wsConnectItem) renderImage(data []byte) { } tmpData := websocket_data.AssembleRenderData{ TaskId: taskId, - UserId: w.userId, + RenderId: renderImageData.RenderId, RenderData: renderImageData.RenderData, } d, _ := json.Marshal(tmpData) diff --git a/server/websocket/websocket.go b/server/websocket/websocket.go index 202e57fd..cfc775f7 100644 --- a/server/websocket/websocket.go +++ b/server/websocket/websocket.go @@ -33,6 +33,7 @@ func main() { //消费渲染结果队列 ctx1 := context.Background() ctx2, cancel := context.WithCancel(ctx1) + ctx2 = context.WithValue(ctx2, "allmodels", ctx.AllModels) defer cancel() go ctx.RabbitMq.Consume(ctx2, constants.RABBIT_MQ_RENDER_RESULT_DATA, &logic.MqConsumerRenderResult{}) handler.RegisterHandlers(server, ctx) diff --git a/utils/mq_consumer_factory/mq.go b/utils/mq_consumer_factory/mq.go index 0f3bc97d..d2b7e1d8 100644 --- a/utils/mq_consumer_factory/mq.go +++ b/utils/mq_consumer_factory/mq.go @@ -1,6 +1,8 @@ package mq_consumer_factory +import "context" + // 消费mq消息要实现对应Run方法 type MqHandle interface { - Run(data []byte) error + Run(ctx context.Context, data []byte) error } diff --git a/utils/websocket_data/render_data.go b/utils/websocket_data/render_data.go index 09a203da..5ec90785 100644 --- a/utils/websocket_data/render_data.go +++ b/utils/websocket_data/render_data.go @@ -37,6 +37,6 @@ type ThirdPartyLoginRspMsg struct { // 发送到渲染组装的mq数据 type AssembleRenderData struct { TaskId string `json:"task_id"` - UserId int64 `json:"user_id"` + RenderId string `json:"render_id"` RenderData RenderData `json:"render_data"` }