simulation_service.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. package simulation_service
  2. import (
  3. "archive/zip"
  4. "context"
  5. "fmt"
  6. "github.com/cloudwego/hertz/pkg/app"
  7. "github.com/cloudwego/hertz/pkg/protocol/consts"
  8. "io"
  9. "net/http"
  10. "os"
  11. "path/filepath"
  12. "pji_desktop_http/biz/dal/mysql"
  13. "pji_desktop_http/common/config"
  14. "pji_desktop_http/common/config/c_log"
  15. "pji_desktop_http/common/entity"
  16. "pji_desktop_http/common/util"
  17. )
  18. // DownloadSimulationZipFile 根据请求id从oss拉取并打包仿真测试所需要的文件
  19. // @router /simulation/download/zipfile [GET]
  20. func DownloadSimulationZipFile(ctx context.Context, c *app.RequestContext) {
  21. id := c.Query("id")
  22. fmt.Println(id)
  23. // 根据id生成用于地图更新的压缩包
  24. filePath, tmpDir, err := generateSimulationZipById(ctx, id)
  25. if err != nil {
  26. c.JSON(http.StatusInternalServerError, map[string]string{"error": err.Error()})
  27. }
  28. fmt.Println("filePath", filePath)
  29. // 检查文件是否存在
  30. if _, err := os.Stat(filePath); os.IsNotExist(err) {
  31. c.JSON(http.StatusNotFound, map[string]string{"error": "File not found"})
  32. return
  33. }
  34. // 打开文件
  35. f, err := os.Open(filePath)
  36. if err != nil {
  37. c.JSON(http.StatusInternalServerError, map[string]string{"error": "Failed to open file"})
  38. return
  39. }
  40. defer f.Close()
  41. // 设置响应头
  42. c.Response.Header.Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"`, filepath.Base(filePath)))
  43. c.Response.Header.Set("Content-Type", "binary/octet-stream")
  44. // 将文件流式传输回客户端
  45. data, err := io.ReadAll(f)
  46. if err != nil {
  47. panic(err)
  48. }
  49. if _, err := c.Write(data); err != nil {
  50. c.JSON(http.StatusInternalServerError, map[string]string{"error": err.Error()})
  51. return
  52. }
  53. defer os.RemoveAll(tmpDir)
  54. }
  55. // 根据id生成用于仿真测试的压缩包
  56. func generateSimulationZipById(ctx context.Context, id string) (file string, tmpDir string, err error) {
  57. // 根据id获取对应的oss文件列表
  58. allFileList, err := util.GetExactedMapFileById(id)
  59. //fmt.Println("Filtered Strings:", fileList)
  60. if err != nil {
  61. return
  62. }
  63. // 创建临时文件夹
  64. tmpDir, err = os.MkdirTemp("", "temp-download-*")
  65. fmt.Println("tmpDir:", tmpDir)
  66. if err != nil {
  67. fmt.Println("Error creating temporary directory:", err)
  68. return "", "", err
  69. }
  70. c_log.GlobalLogger.Info("创建下载-临时文件夹:", tmpDir)
  71. // 创建根文件夹(文件打包的根目录)
  72. baseDir := filepath.Join(tmpDir, "data")
  73. if err := os.Mkdir(baseDir, 0755); err != nil {
  74. fmt.Println("Error creating subdirectory:", err)
  75. return "", "", err
  76. }
  77. c_log.GlobalLogger.Info("创建文件打包根目录:", baseDir)
  78. // 从oss下载data.zip, map.pgm, map.yaml文件到根目录
  79. // 过滤特定后缀的文件列表
  80. simulationFileList := util.FilterBySuffixes(allFileList, config.SimulationFiltersuffixes...)
  81. fmt.Println("simulationFileList", simulationFileList)
  82. // 从oss下载文件到 根目录
  83. for _, file := range simulationFileList {
  84. err = config.OssBucket.GetObjectToFile(file, filepath.Join(baseDir, filepath.Base(file)))
  85. if err != nil {
  86. fmt.Println("Error downloading file - data.zip, map.pgm, map.yaml, map.bag:", err)
  87. return "", "", err
  88. }
  89. }
  90. c_log.GlobalLogger.Info("下载data.zip, map.pgm, map.yaml, map.bag文件到根目录 - 成功")
  91. // 下载world文件
  92. // 查询状态
  93. world, err := mysql.QueryWorld(ctx, id)
  94. if err != nil || world == nil { // 文件不存在
  95. c_log.GlobalLogger.Info("world文件不存在 - 跳过")
  96. } else { // 文件存在
  97. // 下载文件
  98. worldURL := world.WorldURL
  99. err = config.OssBucket.GetObjectToFile(worldURL, filepath.Join(baseDir, filepath.Base(worldURL)))
  100. if err != nil {
  101. fmt.Println("Error downloading world file:", err)
  102. return "", "", err
  103. }
  104. c_log.GlobalLogger.Info("下载world文件到根目录 - 成功")
  105. }
  106. // 下载原始bag
  107. rosField, err := util.GetRosFileById(id)
  108. if err != nil {
  109. fmt.Println("Error querying origin map's bag file:", err)
  110. return "", "", err
  111. }
  112. err = config.OssBucket.GetObjectToFile(rosField, filepath.Join(baseDir, "origin_map.bag"))
  113. if err != nil {
  114. fmt.Println("Error downloading origin map's bag file:", err)
  115. return "", "", err
  116. }
  117. c_log.GlobalLogger.Info("下载origin_map.bag文件到根目录 - 成功")
  118. // 创建压缩文件
  119. zipPath := filepath.Join(tmpDir, "simulationFile-"+id+".zip")
  120. zipFile, err := os.Create(zipPath)
  121. if err != nil {
  122. fmt.Println("Error creating ZIP file:", err)
  123. return "", "", err
  124. }
  125. defer zipFile.Close()
  126. zipWriter := zip.NewWriter(zipFile)
  127. defer zipWriter.Close()
  128. // 压缩文件夹
  129. if err := util.AddDirToZip(baseDir, zipWriter); err != nil {
  130. fmt.Println("Error adding directory to ZIP:", err)
  131. return "", "", err
  132. }
  133. fmt.Println("ZIP file created successfully.")
  134. c_log.GlobalLogger.Info("创建压缩文件 - 成功")
  135. return zipPath, tmpDir, nil
  136. }
  137. // CheckDataFileStatus 检查data目录是否存在
  138. // @router /simulation/check/file/data/status [GET]
  139. func CheckDataFileStatus(ctx context.Context, c *app.RequestContext) {
  140. sceneID := c.Query("id")
  141. fmt.Println("id", sceneID)
  142. // 根据id获取对应的oss文件列表
  143. allFileList, err := util.GetExactedMapFileById(sceneID)
  144. if err != nil {
  145. return
  146. }
  147. // 过滤特定后缀的文件列表(data.zip)
  148. simulationFileList := util.FilterBySuffixes(allFileList, config.DataFiltersuffixes...)
  149. if len(simulationFileList) == 0 {
  150. c.JSON(consts.StatusOK, entity.HttpResult{Status: false, Code: "", Message: "data目录不存在"})
  151. return
  152. }
  153. c.JSON(consts.StatusOK, entity.HttpResult{Status: true, Code: "", Message: "data目录存在"})
  154. }