c_platform.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. package config
  2. import (
  3. "cicv-data-closedloop/common/config/c_log"
  4. "cicv-data-closedloop/common/util"
  5. "crypto/md5"
  6. "encoding/hex"
  7. "encoding/json"
  8. "fmt"
  9. "sort"
  10. "strings"
  11. "time"
  12. )
  13. type TaskTrigger struct {
  14. TriggerId string `json:"triggerId"` // 触发器id
  15. TriggerName string `json:"triggerName"` // 触发器名称
  16. TriggerScriptPath string `json:"triggerScriptPath"` // 脚本在oss上的path
  17. ListenTopic string `json:"listenTopic"` // 触发器监听的topic
  18. CollectionTopic string `json:"collectionTopic"` // 触发器采集的topic
  19. CollectionPreSeconds int `json:"collectionPreSeconds"` // 采集前几秒的数据
  20. CollectionAfterSeconds int `json:"collectionAfterSeconds"` // 采集后几秒的数据
  21. Priority int `json:"priority"` // 优先级, 数值越小越优先
  22. }
  23. type PlatformConfigStruct struct {
  24. TaskConfigId string `json:"taskConfigId"` // 任务ID
  25. TaskConfigName string `json:"taskConfigName"` // 任务名称
  26. TaskType string `json:"taskType"` // 任务类型
  27. StartTime string `json:"startTime"` // 开始时间(taskType为“single”有效)
  28. EndTime string `json:"endTime"` // 结束时间(taskType为“single”有效)
  29. DailyStartTime string `json:"dailyStartTime"` // 开始时间(taskType为“daily/weekly/monthly”有效)
  30. DailyEndTime string `json:"dailyEndTime"` // 结束时间(taskType为“daily/weekly/monthly”有效)
  31. CollectionStrategy int `json:"collectionStrategy"` // 采集策略
  32. UploadStrategy int `json:"uploadStrategy"` // 上传策略
  33. CompressionType int `json:"compressionType"` // 压缩类型
  34. TaskTriggers []TaskTrigger `json:"taskTriggers"` // 触发器信息
  35. }
  36. type response struct {
  37. Data []PlatformConfigStruct `json:"data"`
  38. Success bool `json:"success"`
  39. Message string `json:"message"`
  40. Code int `json:"code"`
  41. NowTime string `json:"nowTime"`
  42. }
  43. var (
  44. PlatformConfig PlatformConfigStruct
  45. PlatformTasks []PlatformConfigStruct
  46. RecordTopics []string
  47. CurrTask PlatformConfigStruct // 正在运行中的任务
  48. )
  49. // InitPlatformConfig 初始化数据闭环平台的配置
  50. func InitPlatformConfig() {
  51. if !LocalConfig.Internet {
  52. PlatformConfig = PlatformConfigStruct{}
  53. return
  54. }
  55. c_log.GlobalLogger.Info("获取数据闭环平台配置 - 开始")
  56. // 1 如果车辆没有配置任务,则阻塞在这里,不启动任务
  57. for {
  58. time.Sleep(time.Duration(2) * time.Second)
  59. config, err := GetConfig()
  60. if err != nil {
  61. c_log.GlobalLogger.Error("获取今日任务信息失败:", err)
  62. PlatformConfig = PlatformConfigStruct{}
  63. continue
  64. }
  65. PlatformConfig = config
  66. break
  67. }
  68. c_log.GlobalLogger.Infof("获取数据闭环平台配置 - 成功。")
  69. //c_log.GlobalLogger.Infof("获取数据闭环平台配置 - 成功。【触发前】=【%v 秒】,触发后=【%v 秒】,【最大窗口】=【%v 秒】。", PlatformConfig.TaskBeforeTime, PlatformConfig.TaskAfterTime, PlatformConfig.TaskMaxTime)
  70. }
  71. func GetConfig() (PlatformConfigStruct, error) {
  72. var err error
  73. PlatformTasks, err = GetPlatformTasks()
  74. if err != nil {
  75. return PlatformConfigStruct{}, fmt.Errorf("请求任务信息失败,", err)
  76. }
  77. if !checkPlatformTasks() {
  78. return PlatformConfigStruct{}, fmt.Errorf("配置未通过校验。")
  79. }
  80. for _, task := range PlatformTasks {
  81. flag := shouldExecuteTask(task.StartTime, task.EndTime)
  82. fmt.Println("flag", flag)
  83. // 当前时间不执行此任务,跳过
  84. if !flag {
  85. continue
  86. }
  87. RecordTopics = getUniqueCollectionTopics(task.TaskTriggers)
  88. return task, nil
  89. }
  90. return PlatformConfigStruct{}, fmt.Errorf("当前时刻不存在待执行的任务。")
  91. }
  92. func GetPlatformTasks() ([]PlatformConfigStruct, error) {
  93. queryParams := map[string]string{
  94. "equipmentNo": LocalConfig.SecretKey,
  95. }
  96. // 1. 生成签名
  97. sign := GenerateSign(CloudConfig.Platform.Ak, CloudConfig.Platform.Sk, queryParams)
  98. fmt.Println("sign:", sign)
  99. if sign == "" {
  100. return []PlatformConfigStruct{}, fmt.Errorf("生成签名失败。")
  101. }
  102. headers := map[string]string{
  103. "ak": CloudConfig.Platform.Ak,
  104. "timestamp": fmt.Sprintf("%d", time.Now().Unix()),
  105. "sign": sign,
  106. }
  107. // 2 访问任务配置获取接口
  108. resp, err := util.HttpGetStringAddHeadersResponseString(
  109. CloudConfig.Platform.UrlTask,
  110. headers,
  111. queryParams,
  112. )
  113. if err != nil {
  114. c_log.GlobalLogger.Error("访问接口", CloudConfig.Platform.UrlTask, "失败:", err)
  115. return []PlatformConfigStruct{}, err
  116. }
  117. var result response
  118. err = json.Unmarshal([]byte(resp), &result)
  119. fmt.Println("result", result)
  120. if err != nil {
  121. c_log.GlobalLogger.Error("解析【返回结果】", resp, "失败:", err)
  122. return []PlatformConfigStruct{}, err
  123. }
  124. return result.Data, nil
  125. }
  126. func checkPlatformTasks() bool {
  127. if len(PlatformTasks) == 0 {
  128. c_log.GlobalLogger.Error("数据闭环平台中该车辆今天没有配置任务。")
  129. return false
  130. }
  131. return true
  132. }
  133. // GenerateSign 生成签名
  134. func GenerateSign(ak, sk string, params map[string]string) string {
  135. // 1. 添加时间戳
  136. params["timestamp"] = fmt.Sprintf("%d", time.Now().Unix())
  137. // 2. 参数排序
  138. keys := make([]string, 0, len(params))
  139. for k := range params {
  140. keys = append(keys, k)
  141. }
  142. sort.Strings(keys)
  143. // 3. 拼接参数
  144. var buffer strings.Builder
  145. for _, k := range keys {
  146. buffer.WriteString(k)
  147. buffer.WriteByte('=')
  148. buffer.WriteString(params[k])
  149. buffer.WriteByte('&')
  150. }
  151. paramStr := buffer.String()[:len(buffer.String())-1]
  152. // 4. 计算MD5
  153. signStr := paramStr + sk
  154. hash := md5.Sum([]byte(signStr))
  155. return hex.EncodeToString(hash[:])
  156. }
  157. // shouldExecuteTask 给定任务起止时间,判断当前时刻此任务是否应该执行
  158. func shouldExecuteTask(startTimeStr, endTimeStr string) bool {
  159. // 定义时间格式(根据输入的时间字符串格式)
  160. timeFormat := "2006-01-02 15:04:05"
  161. // 注意: time.Parse 时间没有考虑时区,当做默认的 +0 时区 UTC 时间来解析了,就造成了实际时间比预期的 +8 小时
  162. // 解析 startTime 和 endTime
  163. startTime, err := time.ParseInLocation(timeFormat, startTimeStr, time.Local)
  164. //fmt.Println("startTime", startTime)
  165. if err != nil {
  166. fmt.Println("Error parsing startTime:", err)
  167. return false
  168. }
  169. endTime, err := time.ParseInLocation(timeFormat, endTimeStr, time.Local)
  170. //fmt.Println("endTime", endTime)
  171. if err != nil {
  172. fmt.Println("Error parsing endTime:", err)
  173. return false
  174. }
  175. // 获取当前时间
  176. currentTime := time.Now()
  177. //fmt.Println("currentTime", currentTime)
  178. // 判断当前时间是否在 startTime 和 endTime 之间
  179. return !currentTime.Before(startTime) && !currentTime.After(endTime)
  180. }
  181. // getUniqueCollectionTopics 统计需要采集的 topic 并去重
  182. func getUniqueCollectionTopics(taskTriggers []TaskTrigger) []string {
  183. // 使用 map 去重
  184. topicMap := make(map[string]struct{})
  185. // 遍历 taskTriggers
  186. for _, trigger := range taskTriggers {
  187. // 分割 collectionTopic 字段
  188. topics := strings.Split(trigger.CollectionTopic, ",")
  189. for _, topic := range topics {
  190. trimmedTopic := strings.TrimSpace(topic) // 去除空格
  191. if trimmedTopic != "" { // 忽略空字符串
  192. topicMap[trimmedTopic] = struct{}{}
  193. }
  194. }
  195. }
  196. // 将 map 中的 key 转换为切片
  197. var uniqueTopics []string
  198. for topic := range topicMap {
  199. uniqueTopics = append(uniqueTopics, topic)
  200. }
  201. return uniqueTopics
  202. }