BuildIntermitSceneServiceImpl.java 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  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.RelocationInit;
  11. import com.fdkankan.contro.entity.ScenePlus;
  12. import com.fdkankan.contro.entity.ScenePlusExt;
  13. import com.fdkankan.contro.mq.service.IBuildSceneService;
  14. import com.fdkankan.contro.service.*;
  15. import com.fdkankan.fyun.config.FYunFileConfig;
  16. import com.fdkankan.fyun.face.FYunFileServiceInterface;
  17. import com.fdkankan.model.constants.ConstantFilePath;
  18. import com.fdkankan.model.constants.UploadFilePath;
  19. import com.fdkankan.model.enums.ModelTypeEnums;
  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 BuildIntermitSceneServiceImpl implements IBuildSceneService {
  45. @Value("${queue.modeling.modeling-call}")
  46. private String queueModelingCall;
  47. @Value("${queue.modeling.intermit.modeling-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. @Autowired
  76. private IRelocationInitService relocationInitService;
  77. @Override
  78. public void buildScenePre(BuildSceneCallMessage message) throws Exception{
  79. String num = message.getSceneNum();
  80. try {
  81. //重新计算时需要删除文件夹,否知使用缓存
  82. if(new File(message.getPath() + File.separator + "results").exists()){
  83. FileUtils.deleteDirectory(message.getPath() + File.separator + "results");
  84. }
  85. //由于刘强说caches会影响计算结果,所以这里删除caches
  86. if(new File(message.getPath() + File.separator + "caches").exists()){
  87. FileUtils.deleteDirectory(message.getPath() + File.separator + "caches");
  88. }
  89. //删除点位校准数据
  90. if (Objects.nonNull(message.getExt())
  91. && message.getExt().containsKey("deleteExtras")
  92. && (Boolean) message.getExt().get("deleteExtras")) {
  93. String extras = String.format(UploadFilePath.scene_result_data_path, num).concat("extras");
  94. if(CollUtil.isNotEmpty(fYunFileService.listRemoteFiles(extras))){
  95. fYunFileService.deleteFolder(extras);
  96. }
  97. }
  98. //用户相机重新全量上传,需要解冻结
  99. sceneColdStorageService.unfreeze(num, "用户相机重新全量上传", message.getPath());
  100. //根据相机类型,组装资源路径
  101. //下载资源到本地
  102. this.downLoadSource(message, message.getPath());
  103. message.getBuildContext().put("cameraType",message.getCameraType());
  104. //发送mq,就进行计算
  105. message.setResultReceiverMqName(queueModelingPost);
  106. message.setBizType("intermit");
  107. JSONObject fdageData = commonService.getFdageData(message.getPath() + File.separator + "capture" + File.separator + "data.fdage");
  108. buildService.writeDataJson(message, fdageData, null, null);
  109. log.info("场景计算资源准备结束,场景码:{}", message.getSceneNum());
  110. }catch (Exception e){
  111. log.error("场景计算前置处理出错,num"+num, e);
  112. buildSceneDTService.handBaseFail("场景计算资源准备异常!", message.getPath(), message.getSceneNum(), "计算控制服务器");
  113. throw e;
  114. }
  115. }
  116. private String getOssPath(String path) {
  117. String ossPath = ConstantFilePath.OSS_PREFIX
  118. + path.replace(ConstantFilePath.BUILD_MODEL_PATH, "")
  119. .replace(ConstantFilePath.BUILD_MODEL_LASER_PATH, "");
  120. if (!ossPath.endsWith("/")) {
  121. ossPath = ossPath.concat("/");
  122. }
  123. return ossPath;
  124. }
  125. @Override
  126. public void downLoadSource(BuildSceneCallMessage buildSceneMqMessage,String path){
  127. String ossPath = getOssPath(path);
  128. fYunFileService.downloadFileByCommand(path + File.separator + "capture", ossPath);
  129. }
  130. @Override
  131. public void buildScenePost(BuildSceneResultMqMessage message) throws Exception {
  132. String sceneCode = message.getBuildContext().get("sceneNum").toString();
  133. String path = message.getPath();
  134. try {
  135. // 上传计算日志
  136. //如果是重复计算,没有走到计算逻辑,不需要上传日志文件
  137. log.info("开始上传计算日志");
  138. String buildLogPath = String.format(UploadFilePath.BUILD_LOG_PATH, sceneCode);
  139. fYunFileService.uploadFile(path + File.separator + "console.log", buildLogPath + "console.log");
  140. log.info("计算日志上传完成");
  141. JSONObject fdageData = commonService.getFdageData(path + File.separator + "capture" +File.separator+"data.fdage");
  142. if (!message.getBuildSuccess()) {
  143. log.error("建模失败,修改状态为失败状态");
  144. scenePlusService.update(new LambdaUpdateWrapper<ScenePlus>()
  145. .set(ScenePlus::getSceneStatus, SceneStatus.FAILD.code())
  146. .eq(ScenePlus::getNum, sceneCode));
  147. //推送到全景看看
  148. intermitSceneService.sendMq(sceneCode, fdageData, CommonSuccessStatus.FAIL.code());
  149. // 发送钉钉消息,计算失败
  150. buildSceneDTService.handModelFail("计算失败", message.getPath(), sceneCode, message.getHostName());
  151. return;
  152. }
  153. ScenePlus scenePlus = scenePlusService.getScenePlusByNum(sceneCode);
  154. Integer cameraType = Integer.parseInt(message.getBuildContext().get("cameraType").toString());
  155. Map<String, String> uploadFiles = getUploadFiles(scenePlus,path);
  156. scenePlus.setPayStatus(PayStatus.PAY.code());
  157. scenePlus.setUpdateTime(new Date());
  158. scenePlus.setSceneStatus(SceneStatus.NO_DISPLAY.code());
  159. Integer videoVersion = fdageData.getInteger("videoVersion");
  160. //读取计算结果文件生成videosJson
  161. JSONObject videosJson = commonService.getVideosJson(path, videoVersion, sceneCode, cameraType);
  162. ScenePlusExt scenePlusExt = scenePlusExtService.getScenePlusExtByPlusId(scenePlus.getId());
  163. log.info("开始上传场景计算结果数据,num:{}", sceneCode);
  164. //上传结算结果之前需要删除oss上的结果数据
  165. String panoramaKey = String.format(UploadFilePath.IMG_VIEW_PATH, sceneCode) + "panoramas";
  166. if(fYunFileService.fileExist(panoramaKey)){
  167. fYunFileService.deleteFolder(panoramaKey);
  168. }
  169. //上传文件
  170. fYunFileService.uploadMulFiles(uploadFiles);
  171. //容量统计
  172. Long space = commonService.getSpace(sceneCode);
  173. //写入数据库
  174. this.updateDbPlus(scenePlus.getSceneSource(), space, videosJson.toJSONString(), message.getComputeTime(),false,scenePlusExt);
  175. Object[] editInfoArr = commonService.updateEditInfo(scenePlus);
  176. //统计原始资源大小
  177. scenePlusExt.setOrigSpace(FileUtil.size(new File(path.concat(File.separator).concat("capture"))));
  178. // //删除计算目录
  179. // if(CollUtil.isEmpty(notDeleteNasNumList) || !notDeleteNasNumList.contains(sceneCode)){
  180. // CreateObjUtil.deleteFile(path.replace(ConstantFilePath.BUILD_MODEL_PATH, "/"));
  181. // }
  182. //如果相机容量不足,需要把场景的paystatus改为容量不足状态
  183. scenePlus.setPayStatus(commonService.getPayStatus(scenePlus.getCameraId(), space, fdageData));
  184. this.uploadStatusJson(scenePlus, scenePlusExt);
  185. scenePlusService.updateById(scenePlus);
  186. scenePlusExtService.updateById(scenePlusExt);
  187. //推送到全景看看
  188. intermitSceneService.sendMq(sceneCode, fdageData, CommonSuccessStatus.SUCCESS.code());
  189. //判断是否有重定位关系,如果有,推送到激光系统去做初始化处理
  190. String sxUuid = fdageData.getString("parentScene");
  191. ScenePlus sxScenePlus = scenePlusService.getByFileId(sxUuid);
  192. RelocationInit relocationInit = new RelocationInit();
  193. relocationInit.setUnicode(sxUuid);
  194. if(Objects.nonNull(sxScenePlus)){
  195. relocationInit.setNum(sxScenePlus.getNum());
  196. }
  197. relocationInit.setPanNum(sceneCode);
  198. relocationInitService.save(relocationInit);
  199. relocationInitService.pushInit(relocationInit);
  200. log.info("场景计算结果处理结束,场景码:{}", sceneCode);
  201. }catch (Exception e){
  202. log.error("场景计算结果处理出错,num"+sceneCode, e);
  203. buildSceneDTService.handBaseFail("场景计算结果处理出错!", message.getPath(), sceneCode, "计算控制服务器");
  204. throw e;
  205. }
  206. }
  207. private Map<String, String> getUploadFiles(ScenePlus scenePlus,String path) throws Exception {
  208. String projectNum = scenePlus.getNum();
  209. String dataViewPath = String.format(UploadFilePath.DATA_VIEW_PATH, projectNum);
  210. String imagesPath = String.format(UploadFilePath.IMG_VIEW_PATH, projectNum);
  211. String videoPath = String.format(UploadFilePath.VIDEOS_VIEW_PATH, projectNum);
  212. String resultsPath = path + File.separator + "results" + File.separator;
  213. String uploadData = FileUtils.readFile(resultsPath + "upload.json");
  214. JSONArray array = JSONObject.parseObject(uploadData).getJSONArray("upload");
  215. JSONObject fileJson = null;
  216. String fileName = "";
  217. Map<String, String> map = new HashMap();
  218. for (int i = 0; i < array.size(); ++i) {
  219. fileJson = array.getJSONObject(i);
  220. fileName = fileJson.getString("file");
  221. String filePath = resultsPath + fileName;
  222. if (!(new File(filePath)).exists()) {
  223. throw new Exception(filePath + "文件不存在");
  224. }
  225. if(fileJson.getIntValue("clazz") == 1 || fileJson.getIntValue("clazz") == 22 || fileJson.getIntValue("clazz") == 3 || fileJson.getIntValue("clazz") == 5){
  226. map.put(filePath, imagesPath + fileName);
  227. }
  228. }
  229. return map;
  230. }
  231. private void uploadStatusJson(ScenePlus scenePlus, ScenePlusExt scenePlusExt){
  232. String num = scenePlus.getNum();
  233. String dataViewPath = String.format(UploadFilePath.DATA_VIEW_PATH, num);
  234. Integer status = 1;
  235. // 上传status JSON.
  236. JSONObject statusJson = new JSONObject();
  237. //临时将-2改成1,app还没完全更新
  238. statusJson.put("status", status);
  239. statusJson.put("webSite", scenePlusExt.getWebSite());
  240. statusJson.put("sceneNum", num);
  241. statusJson.put("thumb", scenePlusExt.getThumb());
  242. statusJson.put("payStatus", scenePlus.getPayStatus());
  243. statusJson.put("sceneScheme", scenePlusExt.getSceneScheme());
  244. FileUtils.writeFile(ConstantFilePath.SCENE_PATH + "data/data" + num + File.separator + "status.json", statusJson.toString());
  245. fYunFileService.uploadFile(statusJson.toJSONString().getBytes(StandardCharsets.UTF_8), dataViewPath + "status.json");
  246. }
  247. private void updateDbPlus(int sceneSource,Long space,String videosJson, Long computeTime,boolean isObj,ScenePlusExt scenePlusExt){
  248. scenePlusExt.setSpace(space);
  249. scenePlusExt.setComputeTime(computeTime);
  250. scenePlusExt.setAlgorithmTime(new Date());
  251. scenePlusExt.setVideos(videosJson);
  252. scenePlusExt.setIsObj(isObj ? 1 : 0);
  253. if(ModelTypeEnums.TILE_CODE.equals(modelType)){
  254. scenePlusExt.setSceneScheme(3);
  255. }
  256. switch (SceneSource.get(sceneSource)){
  257. case BM:
  258. scenePlusExt.setSceneResolution(SceneResolution.two_K.code());
  259. scenePlusExt.setSceneFrom(SceneFrom.PRO.code());
  260. break;
  261. case SM:
  262. scenePlusExt.setSceneResolution(SceneResolution.one_k.code());
  263. scenePlusExt.setSceneFrom(SceneFrom.LITE.code());
  264. break;
  265. case ZT:
  266. scenePlusExt.setSceneResolution(SceneResolution.four_K.code());
  267. scenePlusExt.setSceneFrom(SceneFrom.MINION.code());
  268. break;
  269. case JG:
  270. scenePlusExt.setSceneResolution(SceneResolution.four_K.code());
  271. scenePlusExt.setSceneFrom(SceneFrom.LASER.code());
  272. break;
  273. case SG:
  274. scenePlusExt.setSceneResolution(SceneResolution.four_K.code());
  275. scenePlusExt.setSceneFrom(SceneFrom.LASER.code());
  276. break;
  277. }
  278. String sceneKind = scenePlusExt.getSceneScheme() == 3 ? SceneKind.FACE.code():SceneKind.TILES.code();
  279. scenePlusExt.setSceneKind(sceneKind);
  280. // scenePlusExt.setModelKind(modelKind);
  281. //统计点位数量
  282. scenePlusExt.setShootCount(this.getShootCount(scenePlusExt));
  283. scenePlusExtService.updateById(scenePlusExt);
  284. }
  285. private Integer getShootCount(ScenePlusExt scenePlusExt){
  286. Integer shootCount = null;
  287. String homePath = SceneUtil.getHomePath(scenePlusExt.getDataSource());
  288. JSONObject dataFdageObj = JSON.parseObject(fYunFileService.getFileContent(homePath.concat("data.fdage")));
  289. if(Objects.nonNull(dataFdageObj)){
  290. JSONArray points = dataFdageObj.getJSONArray("points");
  291. if(CollUtil.isNotEmpty(points)){
  292. shootCount = points.size();
  293. }
  294. }
  295. if(Objects.nonNull(shootCount) && shootCount > 0){
  296. return shootCount;
  297. }
  298. String slamDataStr = fYunFileService.getFileContent(homePath.concat("slam_data.json"));
  299. JSONObject slamDataObj = JSON.parseObject(slamDataStr);
  300. if(Objects.nonNull(slamDataObj)){
  301. JSONArray viewsInfo = slamDataObj.getJSONArray("views_info");
  302. if(CollUtil.isNotEmpty(viewsInfo)){
  303. shootCount = viewsInfo.stream().mapToInt(info -> {
  304. return ((JSONObject) info).getJSONArray("list_pose").size();
  305. }).sum();
  306. }
  307. }
  308. return shootCount;
  309. }
  310. }