hangguan.go 7.5 KB

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