c_platform.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  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 int `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 string `json:"collectionPreSeconds"` // 采集前几秒的数据
  20. CollectionAfterSeconds string `json:"collectionAfterSeconds"` // 采集后几秒的数据
  21. Priority string `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. PlatformConfig1 []PlatformConfigStruct
  46. RecordTopics []string
  47. )
  48. // InitPlatformConfig 初始化数据闭环平台的配置
  49. func InitPlatformConfig() {
  50. if !LocalConfig.Internet {
  51. return
  52. }
  53. var err error
  54. c_log.GlobalLogger.Info("获取数据闭环平台配置 - 开始")
  55. // 1 如果车辆没有配置任务,则阻塞在这里,不启动任务
  56. for {
  57. time.Sleep(time.Duration(2) * time.Second)
  58. PlatformConfig1, err = GetConfig()
  59. if err != nil {
  60. c_log.GlobalLogger.Error("获取配置status失败:", err)
  61. continue
  62. }
  63. //if checkPlatformConfig() {
  64. // RecordTopics = strings.Split(PlatformConfig.EquipmentTopic, ",")
  65. // // 去掉首尾空格
  66. // for i, topic := range RecordTopics {
  67. // RecordTopics[i] = strings.TrimSpace(topic)
  68. // }
  69. // break
  70. //}
  71. }
  72. //c_log.GlobalLogger.Infof("获取数据闭环平台配置 - 成功。【触发前】=【%v 秒】,触发后=【%v 秒】,【最大窗口】=【%v 秒】。", PlatformConfig.TaskBeforeTime, PlatformConfig.TaskAfterTime, PlatformConfig.TaskMaxTime)
  73. }
  74. /*
  75. {
  76. "data": {
  77. "accessToken": "YWRmYWRzZmFzZGZhZHNmYWRmYWRm=",
  78. "expireTime": "28800",
  79. "equipmentNo": "robot-001"
  80. },
  81. "success": true,
  82. "message": "ok",
  83. "code": 1,
  84. "nowTime": "2023-12-09 22:41:00"
  85. }
  86. */
  87. // 认证接口,获取access_token
  88. func GetAccessToken() (string, error) {
  89. url := &CloudConfig.Platform.UrlDeviceAuth
  90. param := &map[string]string{
  91. "equipmentNo": LocalConfig.EquipmentNo,
  92. "secretKey": LocalConfig.SecretKey,
  93. }
  94. respJson, err := util.HttpPostJsonResponseString(
  95. *url,
  96. *param,
  97. )
  98. if err != nil {
  99. return "", nil
  100. }
  101. respMap, err := util.JsonStringToMap(respJson)
  102. if err != nil {
  103. c_log.GlobalLogger.Errorf("解析返回结果【%v】失败,请求地址为【%v】,请求参数为【%v】:%v", respJson, *url, *param, err)
  104. return "", nil
  105. }
  106. dataMap, ok := respMap["data"].(map[string]interface{})
  107. if !ok {
  108. c_log.GlobalLogger.Error("解析返回结果.data", dataMap, "失败:", err)
  109. return "", nil
  110. }
  111. return dataMap["accessToken"].(string), nil
  112. }
  113. /*
  114. {
  115. "data": {
  116. "status": "UNCHANGE"
  117. "taskConfigld": "xxx"
  118. },
  119. "success": true,
  120. "message": "ok",
  121. "code": 1,
  122. "nowTime": "2023-12-09 21:08:49"
  123. }
  124. */
  125. //GetStatus 根据taskConfigId获取任务status,如果传入空代表车端没有配置,直接获取新的配置
  126. func GetStatus(taskConfigId string) (string, error) {
  127. token, err := GetAccessToken()
  128. if err != nil {
  129. return "", err
  130. }
  131. resp, err := util.HttpGetStringAddHeadersResponseString(
  132. CloudConfig.Platform.UrlTaskPoll,
  133. map[string]string{
  134. "authorization": token,
  135. },
  136. map[string]string{
  137. "equipmentNo": LocalConfig.EquipmentNo,
  138. "taskConfigId": taskConfigId,
  139. },
  140. )
  141. if err != nil {
  142. c_log.GlobalLogger.Error("访问接口", CloudConfig.Platform.UrlTask, "失败:", err)
  143. return "", err
  144. }
  145. respMap, err := util.JsonStringToMap(resp)
  146. if err != nil {
  147. c_log.GlobalLogger.Error("解析【返回结果1】", resp, "失败:", err)
  148. return "", err
  149. }
  150. dataMap, ok := respMap["data"].(map[string]interface{})
  151. if !ok {
  152. c_log.GlobalLogger.Errorf("解析【返回结果.data】的类型不是(map[string]interface{}),【dataMap】=%v", dataMap)
  153. return "", err
  154. }
  155. return dataMap["status"].(string), nil
  156. }
  157. func GetConfig() ([]PlatformConfigStruct, error) {
  158. queryParams := map[string]string{
  159. "equipmentNo": LocalConfig.SecretKey,
  160. }
  161. // 1. 生成签名
  162. sign := GenerateSign(CloudConfig.Platform.Ak, CloudConfig.Platform.Sk, queryParams)
  163. fmt.Println("sign:", sign)
  164. if sign == "" {
  165. return []PlatformConfigStruct{}, fmt.Errorf("生成签名失败。")
  166. }
  167. headers := map[string]string{
  168. "ak": CloudConfig.Platform.Ak,
  169. "timestamp": fmt.Sprintf("%d", time.Now().Unix()),
  170. "sign": sign,
  171. }
  172. // 2 访问任务配置获取接口
  173. resp, err := util.HttpGetStringAddHeadersResponseString(
  174. CloudConfig.Platform.UrlTask,
  175. headers,
  176. queryParams,
  177. )
  178. if err != nil {
  179. c_log.GlobalLogger.Error("访问接口", CloudConfig.Platform.UrlTask, "失败:", err)
  180. return []PlatformConfigStruct{}, err
  181. }
  182. var result response
  183. err = json.Unmarshal([]byte(resp), &result)
  184. fmt.Println("result", result)
  185. if err != nil {
  186. c_log.GlobalLogger.Error("解析【返回结果】", resp, "失败:", err)
  187. return []PlatformConfigStruct{}, err
  188. }
  189. return result.Data, nil
  190. }
  191. func checkPlatformConfig() bool {
  192. if PlatformConfig.TaskConfigId == "" {
  193. c_log.GlobalLogger.Error("数据闭环平台没有配置任务。")
  194. return false
  195. }
  196. //if PlatformConfig.EquipmentTopic == "" {
  197. // c_log.GlobalLogger.Error("数据闭环平台没有配置topic序列。")
  198. // return false
  199. //}
  200. return true
  201. }
  202. // GenerateSign 生成签名
  203. func GenerateSign(ak, sk string, params map[string]string) string {
  204. // 1. 添加时间戳
  205. params["timestamp"] = fmt.Sprintf("%d", time.Now().Unix())
  206. // 2. 参数排序
  207. keys := make([]string, 0, len(params))
  208. for k := range params {
  209. keys = append(keys, k)
  210. }
  211. sort.Strings(keys)
  212. // 3. 拼接参数
  213. var buffer strings.Builder
  214. for _, k := range keys {
  215. buffer.WriteString(k)
  216. buffer.WriteByte('=')
  217. buffer.WriteString(params[k])
  218. buffer.WriteByte('&')
  219. }
  220. paramStr := buffer.String()[:len(buffer.String())-1]
  221. // 4. 计算MD5
  222. signStr := paramStr + sk
  223. hash := md5.Sum([]byte(signStr))
  224. return hex.EncodeToString(hash[:])
  225. }