BuildSxTestServiceImpl.java 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. package com.fdkankan.contro.mq.service.impl;
  2. import cn.hutool.core.collection.CollUtil;
  3. import cn.hutool.core.io.FileUtil;
  4. import com.alibaba.fastjson.JSON;
  5. import com.alibaba.fastjson.JSONArray;
  6. import com.alibaba.fastjson.JSONObject;
  7. import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
  8. import com.fdkankan.common.constant.*;
  9. import com.fdkankan.common.util.FileUtils;
  10. import com.fdkankan.contro.entity.ScenePlus;
  11. import com.fdkankan.contro.entity.ScenePlusExt;
  12. import com.fdkankan.contro.mq.service.IBuildSceneService;
  13. import com.fdkankan.contro.service.*;
  14. import com.fdkankan.fyun.config.FYunFileConfig;
  15. import com.fdkankan.fyun.face.FYunFileServiceInterface;
  16. import com.fdkankan.model.constants.ConstantFilePath;
  17. import com.fdkankan.model.constants.UploadFilePath;
  18. import com.fdkankan.model.enums.ModelTypeEnums;
  19. import com.fdkankan.model.utils.CreateObjUtil;
  20. import com.fdkankan.model.utils.SceneUtil;
  21. import com.fdkankan.rabbitmq.bean.BuildSceneCallMessage;
  22. import com.fdkankan.rabbitmq.bean.BuildSceneResultMqMessage;
  23. import com.fdkankan.rabbitmq.util.RabbitMqProducer;
  24. import lombok.extern.slf4j.Slf4j;
  25. import org.springframework.beans.factory.annotation.Autowired;
  26. import org.springframework.beans.factory.annotation.Value;
  27. import org.springframework.cloud.context.config.annotation.RefreshScope;
  28. import org.springframework.stereotype.Service;
  29. import javax.annotation.Resource;
  30. import java.io.File;
  31. import java.nio.charset.StandardCharsets;
  32. import java.util.*;
  33. /**
  34. * <p>
  35. * TODO
  36. * </p>
  37. *
  38. * @author dengsixing
  39. * @since 2022/4/20
  40. **/
  41. @Slf4j
  42. @Service
  43. @RefreshScope
  44. public class BuildSxTestServiceImpl implements IBuildSceneService {
  45. @Value("${queue.modeling.modeling-call}")
  46. private String queueModelingCall;
  47. @Value("${queue.modeling.sx.modeling-sx-post:modeling-sx-post}")
  48. private String queueModelingPost;
  49. @Value("${model.type:#{null}}")
  50. private String modelType;
  51. @Value("${env:gn}")
  52. private String env;
  53. @Value("#{'${build.scene.post.not-delete-nas-nums:}'.split(',')}")
  54. private List<String> notDeleteNasNumList;
  55. @Autowired
  56. private RabbitMqProducer mqProducer;
  57. @Resource
  58. private FYunFileServiceInterface fYunFileService;
  59. @Autowired
  60. private FYunFileConfig fYunFileConfig;
  61. @Autowired
  62. private IScenePlusService scenePlusService;
  63. @Autowired
  64. private IScenePlusExtService scenePlusExtService;
  65. @Autowired
  66. private IBuildSceneDTService buildSceneDTService;
  67. @Autowired
  68. private ICommonService commonService;
  69. @Autowired
  70. private ISceneColdStorageService sceneColdStorageService;
  71. @Autowired
  72. private IntermitSceneService intermitSceneService;
  73. @Autowired
  74. private IBuildService buildService;
  75. @Override
  76. public void buildScenePre(BuildSceneCallMessage message) throws Exception{
  77. String num = message.getSceneNum();
  78. try {
  79. //重新计算时需要删除文件夹,否知使用缓存
  80. if(new File(message.getPath() + File.separator + "results").exists()){
  81. FileUtils.deleteDirectory(message.getPath() + File.separator + "results");
  82. }
  83. //由于刘强说caches会影响计算结果,所以这里删除caches
  84. if(new File(message.getPath() + File.separator + "caches").exists()){
  85. FileUtils.deleteDirectory(message.getPath() + File.separator + "caches");
  86. }
  87. //下载文件
  88. // //删除点位校准数据
  89. // if (Objects.nonNull(message.getExt())
  90. // && message.getExt().containsKey("deleteExtras")
  91. // && (Boolean) message.getExt().get("deleteExtras")) {
  92. // String extras = String.format(UploadFilePath.scene_result_data_path, num).concat("extras");
  93. // if(CollUtil.isNotEmpty(fYunFileService.listRemoteFiles(extras))){
  94. // fYunFileService.deleteFolder(extras);
  95. // }
  96. // }
  97. //用户相机重新全量上传,需要解冻结
  98. // sceneColdStorageService.unfreeze(num, "用户相机重新全量上传", message.getPath());
  99. //根据相机类型,组装资源路径
  100. //下载资源到本地
  101. // this.downLoadSource(message, message.getPath());
  102. //TODO 等算法文档出来 下载文件
  103. //发送mq,就进行计算
  104. message.setResultReceiverMqName(queueModelingPost);
  105. JSONObject fdageData = commonService.getFdageData(message.getPath() + File.separator + "capture" + File.separator + "data.fdage");
  106. //TODO 等算法文档出来 修改参数
  107. buildService.writeDataJson(message, fdageData, null, null);
  108. log.info("场景计算资源准备结束,场景码:{}", message.getSceneNum());
  109. }catch (Exception e){
  110. log.error("场景计算前置处理出错,num"+num, e);
  111. buildSceneDTService.handBaseFail("场景计算资源准备异常!", message.getPath(), message.getSceneNum(), "计算控制服务器");
  112. throw e;
  113. }
  114. }
  115. private String getOssPath(String path) {
  116. String ossPath = ConstantFilePath.OSS_PREFIX
  117. + path.replace(ConstantFilePath.BUILD_MODEL_PATH, "")
  118. .replace(ConstantFilePath.BUILD_MODEL_LASER_PATH, "");
  119. if (!ossPath.endsWith("/")) {
  120. ossPath = ossPath.concat("/");
  121. }
  122. return ossPath;
  123. }
  124. @Override
  125. public void downLoadSource(BuildSceneCallMessage buildSceneMqMessage,String path){
  126. String ossPath = getOssPath(path);
  127. fYunFileService.downloadFileByCommand(path + File.separator + "capture", ossPath);
  128. }
  129. @Override
  130. public void buildScenePost(BuildSceneResultMqMessage message) throws Exception {
  131. String sceneCode = message.getBuildContext().get("sceneNum").toString();
  132. String path = message.getPath();
  133. try {
  134. // 上传计算日志
  135. //如果是重复计算,没有走到计算逻辑,不需要上传日志文件
  136. log.info("开始上传计算日志");
  137. String buildLogPath = String.format(UploadFilePath.BUILD_LOG_PATH, sceneCode);
  138. fYunFileService.uploadFile(path + File.separator + "console.log", buildLogPath + "console.log");
  139. log.info("计算日志上传完成");
  140. JSONObject fdageData = commonService.getFdageData(path + File.separator + "capture" +File.separator+"data.fdage");
  141. if (!message.getBuildSuccess()) {
  142. log.error("建模失败,修改状态为失败状态");
  143. scenePlusService.update(new LambdaUpdateWrapper<ScenePlus>()
  144. .set(ScenePlus::getSceneStatus, SceneStatus.FAILD.code())
  145. .eq(ScenePlus::getNum, sceneCode));
  146. //推送到全景看看
  147. intermitSceneService.sendMq(sceneCode, fdageData, CommonSuccessStatus.FAIL.code());
  148. // 发送钉钉消息,计算失败
  149. buildSceneDTService.handModelFail("计算失败", message.getPath(), sceneCode, message.getHostName());
  150. return;
  151. }
  152. ScenePlus scenePlus = scenePlusService.getScenePlusByNum(sceneCode);
  153. Integer cameraType = Integer.parseInt(message.getBuildContext().get("cameraType").toString());
  154. Map<String, String> uploadFiles = getUploadFiles(scenePlus,path);
  155. scenePlus.setPayStatus(PayStatus.PAY.code());
  156. scenePlus.setUpdateTime(new Date());
  157. scenePlus.setSceneStatus(SceneStatus.NO_DISPLAY.code());
  158. Integer videoVersion = fdageData.getInteger("videoVersion");
  159. //读取计算结果文件生成videosJson
  160. JSONObject videosJson = commonService.getVideosJson(path, videoVersion, sceneCode, cameraType);
  161. ScenePlusExt scenePlusExt = scenePlusExtService.getScenePlusExtByPlusId(scenePlus.getId());
  162. log.info("开始上传场景计算结果数据,num:{}", sceneCode);
  163. //上传文件
  164. fYunFileService.uploadMulFiles(uploadFiles);
  165. //容量统计
  166. Long space = commonService.getSpace(sceneCode);
  167. //写入数据库
  168. this.updateDbPlus(scenePlus.getSceneSource(), space, videosJson.toJSONString(), message.getComputeTime(),false,scenePlusExt);
  169. Object[] editInfoArr = commonService.updateEditInfo(scenePlus);
  170. //统计原始资源大小
  171. scenePlusExt.setOrigSpace(FileUtil.size(new File(path.concat(File.separator).concat("capture"))));
  172. //删除计算目录
  173. if(CollUtil.isEmpty(notDeleteNasNumList) || !notDeleteNasNumList.contains(sceneCode)){
  174. CreateObjUtil.deleteFile(path.replace(ConstantFilePath.BUILD_MODEL_PATH, "/"));
  175. }
  176. //如果相机容量不足,需要把场景的paystatus改为容量不足状态
  177. scenePlus.setPayStatus(commonService.getPayStatus(scenePlus.getCameraId(), space, fdageData));
  178. this.uploadStatusJson(scenePlus, scenePlusExt);
  179. scenePlusService.updateById(scenePlus);
  180. scenePlusExtService.updateById(scenePlusExt);
  181. //推送到全景看看
  182. intermitSceneService.sendMq(sceneCode, fdageData, CommonSuccessStatus.SUCCESS.code());
  183. log.info("场景计算结果处理结束,场景码:{}", sceneCode);
  184. }catch (Exception e){
  185. log.error("场景计算结果处理出错,num"+sceneCode, e);
  186. buildSceneDTService.handBaseFail("场景计算结果处理出错!", message.getPath(), sceneCode, "计算控制服务器");
  187. throw e;
  188. }
  189. }
  190. private Map<String, String> getUploadFiles(ScenePlus scenePlus,String path) throws Exception {
  191. String projectNum = scenePlus.getNum();
  192. String dataViewPath = String.format(UploadFilePath.DATA_VIEW_PATH, projectNum);
  193. String imagesPath = String.format(UploadFilePath.IMG_VIEW_PATH, projectNum);
  194. String videoPath = String.format(UploadFilePath.VIDEOS_VIEW_PATH, projectNum);
  195. String resultsPath = path + File.separator + "results" + File.separator;
  196. String uploadData = FileUtils.readFile(resultsPath + "upload.json");
  197. JSONArray array = JSONObject.parseObject(uploadData).getJSONArray("upload");
  198. JSONObject fileJson = null;
  199. String fileName = "";
  200. Map<String, String> map = new HashMap();
  201. for (int i = 0; i < array.size(); ++i) {
  202. fileJson = array.getJSONObject(i);
  203. fileName = fileJson.getString("file");
  204. String filePath = resultsPath + fileName;
  205. if (!(new File(filePath)).exists()) {
  206. throw new Exception(filePath + "文件不存在");
  207. }
  208. if(fileJson.getIntValue("clazz") == 1 || fileJson.getIntValue("clazz") == 22 || fileJson.getIntValue("clazz") == 3 || fileJson.getIntValue("clazz") == 5){
  209. map.put(filePath, imagesPath + fileName);
  210. }
  211. }
  212. return map;
  213. }
  214. private void uploadStatusJson(ScenePlus scenePlus, ScenePlusExt scenePlusExt){
  215. String num = scenePlus.getNum();
  216. String dataViewPath = String.format(UploadFilePath.DATA_VIEW_PATH, num);
  217. Integer status = 1;
  218. // 上传status JSON.
  219. JSONObject statusJson = new JSONObject();
  220. //临时将-2改成1,app还没完全更新
  221. statusJson.put("status", status);
  222. statusJson.put("webSite", scenePlusExt.getWebSite());
  223. statusJson.put("sceneNum", num);
  224. statusJson.put("thumb", scenePlusExt.getThumb());
  225. statusJson.put("payStatus", scenePlus.getPayStatus());
  226. statusJson.put("sceneScheme", scenePlusExt.getSceneScheme());
  227. FileUtils.writeFile(ConstantFilePath.SCENE_PATH + "data/data" + num + File.separator + "status.json", statusJson.toString());
  228. fYunFileService.uploadFile(statusJson.toJSONString().getBytes(StandardCharsets.UTF_8), dataViewPath + "status.json");
  229. }
  230. private void updateDbPlus(int sceneSource,Long space,String videosJson, Long computeTime,boolean isObj,ScenePlusExt scenePlusExt){
  231. scenePlusExt.setSpace(space);
  232. scenePlusExt.setComputeTime(computeTime);
  233. scenePlusExt.setAlgorithmTime(new Date());
  234. scenePlusExt.setVideos(videosJson);
  235. scenePlusExt.setIsObj(isObj ? 1 : 0);
  236. if(ModelTypeEnums.TILE_CODE.equals(modelType)){
  237. scenePlusExt.setSceneScheme(3);
  238. }
  239. switch (SceneSource.get(sceneSource)){
  240. case BM:
  241. scenePlusExt.setSceneResolution(SceneResolution.two_K.code());
  242. scenePlusExt.setSceneFrom(SceneFrom.PRO.code());
  243. break;
  244. case SM:
  245. scenePlusExt.setSceneResolution(SceneResolution.one_k.code());
  246. scenePlusExt.setSceneFrom(SceneFrom.LITE.code());
  247. break;
  248. case ZT:
  249. scenePlusExt.setSceneResolution(SceneResolution.four_K.code());
  250. scenePlusExt.setSceneFrom(SceneFrom.MINION.code());
  251. break;
  252. case JG:
  253. scenePlusExt.setSceneResolution(SceneResolution.four_K.code());
  254. scenePlusExt.setSceneFrom(SceneFrom.LASER.code());
  255. break;
  256. case SG:
  257. scenePlusExt.setSceneResolution(SceneResolution.four_K.code());
  258. scenePlusExt.setSceneFrom(SceneFrom.LASER.code());
  259. break;
  260. }
  261. String sceneKind = scenePlusExt.getSceneScheme() == 3 ? SceneKind.FACE.code():SceneKind.TILES.code();
  262. scenePlusExt.setSceneKind(sceneKind);
  263. // scenePlusExt.setModelKind(modelKind);
  264. //统计点位数量
  265. scenePlusExt.setShootCount(this.getShootCount(scenePlusExt));
  266. scenePlusExtService.updateById(scenePlusExt);
  267. }
  268. private Integer getShootCount(ScenePlusExt scenePlusExt){
  269. Integer shootCount = null;
  270. String homePath = SceneUtil.getHomePath(scenePlusExt.getDataSource());
  271. JSONObject dataFdageObj = JSON.parseObject(fYunFileService.getFileContent(homePath.concat("data.fdage")));
  272. if(Objects.nonNull(dataFdageObj)){
  273. JSONArray points = dataFdageObj.getJSONArray("points");
  274. if(CollUtil.isNotEmpty(points)){
  275. shootCount = points.size();
  276. }
  277. }
  278. if(Objects.nonNull(shootCount) && shootCount > 0){
  279. return shootCount;
  280. }
  281. String slamDataStr = fYunFileService.getFileContent(homePath.concat("slam_data.json"));
  282. JSONObject slamDataObj = JSON.parseObject(slamDataStr);
  283. if(Objects.nonNull(slamDataObj)){
  284. JSONArray viewsInfo = slamDataObj.getJSONArray("views_info");
  285. if(CollUtil.isNotEmpty(viewsInfo)){
  286. shootCount = viewsInfo.stream().mapToInt(info -> {
  287. return ((JSONObject) info).getJSONArray("list_pose").size();
  288. }).sum();
  289. }
  290. }
  291. return shootCount;
  292. }
  293. }