hangguan.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. // services/hangguan.go
  2. package services
  3. import (
  4. "context"
  5. "encoding/json"
  6. "fmt"
  7. "go-policy-service/config"
  8. "go-policy-service/models"
  9. "go-policy-service/structs"
  10. "go-policy-service/utils"
  11. "math"
  12. "strconv"
  13. "strings"
  14. )
  15. type HangguanService struct {
  16. cfg *config.Config
  17. httpClient utils.HTTPClient
  18. tokenMgr *TokenManager
  19. }
  20. // 更新后的New方法
  21. func NewHangguanService(
  22. cfg *config.Config,
  23. httpClient utils.HTTPClient,
  24. tokenMgr *TokenManager,
  25. ) *HangguanService {
  26. return &HangguanService{
  27. cfg: cfg,
  28. httpClient: httpClient,
  29. tokenMgr: tokenMgr,
  30. }
  31. }
  32. func (s *HangguanService) requestSFDataConvert(task models.HgFlightSearchTask, startDate string) ([]byte, error) {
  33. // 将 uint 日期转换为 YYYY-MM-DD 格式字符串)
  34. req := structs.ReqSearchFlightData{
  35. DepCode: task.Dep,
  36. ArrCode: task.Arr,
  37. Date: startDate,
  38. FlyNo: task.FlightNo,
  39. AcCode: task.Flight,
  40. }
  41. // 将结构体转换为 JSON
  42. jsonData, err := utils.StringJsonConvert(req)
  43. if err != nil {
  44. return []byte{}, fmt.Errorf("SF 请求数据转化失败: %w", err)
  45. }
  46. return jsonData, nil
  47. }
  48. func (h *HangguanService) RequestSFlightData(ctx context.Context, task models.HgFlightSearchTask) {
  49. token, err := h.tokenMgr.GetAccessToken()
  50. if err != nil {
  51. utils.Logger.WithField("task_id", task.ID).Error("Failed to get access token: ", err)
  52. }
  53. //获取起飞时间列表
  54. startDateList, err := utils.ParseDateRange(task.DateRange)
  55. if err != nil {
  56. utils.Logger.WithField("task_id", task.ID).Error("获取起飞日期时间列表错误: ", err)
  57. return
  58. }
  59. for _, startDate := range startDateList {
  60. // 构建请求数据
  61. reqData, _ := h.requestSFDataConvert(task, startDate)
  62. // url
  63. url := fmt.Sprintf("%s%s?access_token=%s", h.cfg.HgApiUrl, "/distribution/api/search/flight/list", token)
  64. // 记录请求数据
  65. //utils.Logger.WithField("task_id", task.ID).Info("请求数据: ", string(reqData))
  66. // 调用第三方接口
  67. //resp, err := h.httpClient.RequestWithProxy(ctx, "POST", url, reqData, h.cfg.ProxyUrl) //测试使用
  68. resp, err := h.httpClient.RequestJSON(ctx, "POST", url, reqData)
  69. // 记录响应数据
  70. //utils.Logger.WithField("task_id", task.ID).Info("响应数据: ", string(resp))
  71. if err != nil {
  72. utils.Logger.WithField("task_id", task.ID).Error("API request failed: ", err)
  73. continue
  74. }
  75. //解析数据
  76. var srchResp structs.ResSearchFlight
  77. if err = json.Unmarshal(resp, &srchResp); err != nil {
  78. utils.Logger.WithField("task_id", task.ID).Error("数据解析失败: ", err)
  79. continue
  80. }
  81. if !srchResp.Success {
  82. utils.Logger.WithField("task_id", task.ID).Error("数据请求失败: ", srchResp.Msg)
  83. continue
  84. }
  85. // 处理响应数据
  86. err = h.processResponse(ctx, &task, &srchResp.Data, startDate)
  87. if err != nil {
  88. utils.Logger.WithField("task_id", task.ID).Error("Response processing failed: ", err)
  89. continue
  90. }
  91. }
  92. }
  93. func (h *HangguanService) processResponse(ctx context.Context, task *models.HgFlightSearchTask, response *structs.ResSearchFlightData, startDate string) error {
  94. // 实现具体的响应处理逻辑
  95. if response.Total <= 0 {
  96. return fmt.Errorf("返回数据为空")
  97. }
  98. pushDataList := make([]structs.Datum, 0)
  99. for _, fltData := range response.Datas {
  100. if fltData.FlightInfo.IsCancel == 1 ||
  101. len(fltData.CabinInfos) == 0 ||
  102. fltData.FlightInfo.AcCode != task.Flight { //航班取消 或者 无舱位信息 或者 无实际承运航司
  103. continue
  104. }
  105. var minPrice float64 = 0
  106. var minPricePushData structs.Datum
  107. CabinLoop:
  108. for _, cabinPrice := range fltData.CabinInfos {
  109. //无售卖控制 或者 产品类型为 py
  110. if cabinPrice.AdtPrice.ProductType == "py" {
  111. continue
  112. }
  113. if len(cabinPrice.SaleControls) > 0 {
  114. for _, saleControls := range cabinPrice.SaleControls {
  115. // ... existing code ...
  116. if saleControls.PsNum != "" ||
  117. saleControls.PsType == "ADT+CHD" ||
  118. saleControls.PsIDType != "" ||
  119. saleControls.PsIDNo != "" ||
  120. saleControls.PsAdIdType != "" ||
  121. saleControls.NewMember == 1 ||
  122. saleControls.CheckThreeElement != 0 ||
  123. saleControls.HasAge != "" {
  124. continue CabinLoop
  125. }
  126. if saleControls.PsAge != "" ||
  127. !isRangeClose(saleControls.PsAge, 12, 100, 2, 0) {
  128. continue CabinLoop
  129. }
  130. }
  131. }
  132. if minPrice == 0 || cabinPrice.AdtPrice.Price < minPrice { //取最低价格的舱位信息
  133. minPrice = cabinPrice.AdtPrice.Price
  134. minPricePushData = structs.Datum{
  135. DepAir: fltData.FlightInfo.DepCode,
  136. ArrAir: fltData.FlightInfo.ArrCode,
  137. FlightStartDate: utils.DateFormmat(fltData.FlightInfo.DepDateTime, "2006-01-02"),
  138. PrintPrice: cabinPrice.AdtPrice.Price,
  139. Stock: cabinPrice.Left,
  140. FlightNo: fltData.FlightInfo.FlyNo,
  141. Cabin: cabinPrice.AdtPrice.Cabin,
  142. FlightEndDate: utils.DateFormmat(fltData.FlightInfo.ArrDateTime, "2006-01-02"),
  143. }
  144. }
  145. }
  146. if minPricePushData == (structs.Datum{}) { //无可用舱位
  147. continue
  148. }
  149. pushDataList = append(pushDataList, minPricePushData)
  150. }
  151. // 构建推送数据
  152. pushData := structs.PushData{
  153. Data: pushDataList,
  154. IsPublishImmediately: 1,
  155. Query: structs.Query{
  156. DepAir: task.Dep,
  157. ArrAir: task.Arr,
  158. DepDate: startDate,
  159. },
  160. ServiceTag: "hgSpecail",
  161. }
  162. // 推送数据打印成 json 格式
  163. pushDataJson, _ := json.Marshal(pushData)
  164. utils.Logger.WithField("task_id", task.ID).Info("推送数据: ", string(pushDataJson))
  165. // 推送数据
  166. if err := h.PushPolicyData(ctx, pushData); err != nil {
  167. utils.Logger.WithField("task_id", task.ID).Error("推送失败: ", err)
  168. return err
  169. }
  170. return nil
  171. }
  172. func (h *HangguanService) PushPolicyData(ctx context.Context, data structs.PushData) error {
  173. pushData, err := json.Marshal(data)
  174. if err != nil {
  175. return fmt.Errorf("推送数据JSON 编码失败: %w", err)
  176. }
  177. url := h.cfg.PushUrl
  178. resp, err := h.httpClient.RequestJSON(ctx, "POST", url+"/api/policy/batch-policy-import", pushData)
  179. if err != nil {
  180. return fmt.Errorf("推送请求失败: %w", err)
  181. }
  182. var result struct {
  183. Code int `json:"code"`
  184. Message string `json:"message"`
  185. }
  186. if err := json.Unmarshal(resp, &result); err != nil {
  187. return fmt.Errorf("响应解析失败: %w", err)
  188. }
  189. if result.Code != 200 {
  190. return fmt.Errorf("推送接口返回错误: %s", result.Message)
  191. }
  192. return nil
  193. }
  194. /**
  195. * 年龄限制
  196. * 儿童期 1-12岁
  197. * 青年 15-24
  198. * 老年 60/65+
  199. * 成年 18岁+
  200. * 目标范围:[10, 100]
  201. */
  202. // isRangeClose 检查 ranges 字符串中,是否存在一个子区间 [s,e]:
  203. // 1) 与 [targetStart,targetEnd] 有交集;
  204. // 2) |s - targetStart| ≤ startTol 且 |e - targetEnd| ≤ endTol
  205. //
  206. // ranges 格式示例:"[25,60]|[23,60]|[70,80]"
  207. // startTol、endTol 单位同年龄(比如 startTol=5 表示起点最多相差 5 岁,endTol=10 表示终点最多相差 10 岁)
  208. func isRangeClose(ranges string, targetStart, targetEnd, startTol, endTol int) bool {
  209. parts := strings.Split(ranges, "|")
  210. if len(parts) >= 4 { // 有4段年龄范围以上的过滤
  211. return false
  212. }
  213. for _, part := range parts {
  214. part = strings.Trim(part, "[]")
  215. bounds := strings.Split(part, ",")
  216. if len(bounds) != 2 {
  217. continue
  218. }
  219. s, err1 := strconv.Atoi(strings.TrimSpace(bounds[0]))
  220. e, err2 := strconv.Atoi(strings.TrimSpace(bounds[1]))
  221. if err1 != nil || err2 != nil {
  222. continue
  223. }
  224. // 1) 必须有交集
  225. if e < targetStart || s > targetEnd {
  226. continue
  227. }
  228. // 2) 分别判断起点、终点容差
  229. if math.Abs(float64(s-targetStart)) <= float64(startTol) &&
  230. math.Abs(float64(e-targetEnd)) <= float64(endTol) {
  231. return true
  232. }
  233. }
  234. return false
  235. }