market.go 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. package market
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "mtp2_if/controllers/quote"
  6. "mtp2_if/global/app"
  7. "mtp2_if/global/e"
  8. "mtp2_if/logger"
  9. "mtp2_if/models"
  10. "net/http"
  11. "sort"
  12. "time"
  13. "github.com/gin-gonic/gin"
  14. )
  15. // 通用市场
  16. // QueryMarketRunReq 查询市场运行信息请求参数
  17. type QueryMarketRunReq struct {
  18. MarketID int `form:"marketID"`
  19. }
  20. // QueryMarketRun 查询市场运行信息
  21. // @Summary 查询市场运行信息
  22. // @Produce json
  23. // @Param marketID query int false "市场ID,不传返回所有"
  24. // @Success 200 {object} models.Marketrun
  25. // @Failure 500 {object} app.Response
  26. // @Router /Market/QueryMarketRun [get]
  27. // @Tags 通用市场
  28. func QueryMarketRun(c *gin.Context) {
  29. appG := app.Gin{C: c}
  30. // 获取请求参数
  31. var req QueryMarketRunReq
  32. if err := appG.C.ShouldBindQuery(&req); err != nil {
  33. logger.GetLogger().Errorf("QueryMarketRun failed: %s", err.Error())
  34. appG.Response(http.StatusBadRequest, e.INVALID_PARAMS, nil)
  35. return
  36. }
  37. marketRuns, err := models.GetMarketRuns(req.MarketID)
  38. if err != nil {
  39. // 查询失败
  40. logger.GetLogger().Errorf("QueryMarketRun failed: %s", err.Error())
  41. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_FAIL, nil)
  42. return
  43. }
  44. // 获取真实开休市时间
  45. for i := range marketRuns {
  46. marketRun := &marketRuns[i]
  47. quoteTradeDate := marketRun.Tradedate2
  48. // 获取市场
  49. market, err := models.GetMarket(int(marketRun.Marketid))
  50. if err != nil {
  51. logger.GetLogger().Errorf("QueryMarketRun failed: %s", err.Error())
  52. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_FAIL, nil)
  53. return
  54. }
  55. if market.Markettype != 2 {
  56. continue
  57. }
  58. if len(quoteTradeDate) == 0 {
  59. logger.GetLogger().Errorf("QueryMarketRun failed: %s", err.Error())
  60. appG.Response(http.StatusBadRequest, e.ERROR_GET_MARKETRUN_FAILED, nil)
  61. return
  62. }
  63. // 获取目标品种的开休市计划
  64. // 非外部市场或外部市场没有配置QuoteSourceGroupRunStep表数据的情况下,都从MarketRunStepDetail中获取数据
  65. var runSteps []map[string]interface{}
  66. sourceRunSteps, err := models.FindMarketRunStepDetails(int(marketRun.Marketid))
  67. if err != nil {
  68. logger.GetLogger().Errorf("QueryMarketRun failed: %s", err.Error())
  69. appG.Response(http.StatusBadRequest, e.ERROR_GET_RUNSTEP_FAILED, nil)
  70. return
  71. }
  72. for _, v := range sourceRunSteps {
  73. // struct -> json
  74. if jsonBytes, err := json.Marshal(v); err == nil {
  75. // json -> struct
  76. var runStepMap map[string]interface{}
  77. json.Unmarshal(jsonBytes, &runStepMap)
  78. runSteps = append(runSteps, runStepMap)
  79. }
  80. }
  81. // 构建分时图可直接使用的开休市数据
  82. // 这里有一个知识点:TRADEWEEKDAY 与 STARTWEEKDAY,以及 TRADEWEEKDAY 与 ENDWEEKDAY 之间相差最多一天(管理端限制),
  83. // 所以目前并不支持正真的周五夜盘模式。我们在实现时不用做得太复杂。
  84. // 当前交易日(周几)对应的开休市计划
  85. tradeDate, err := time.ParseInLocation("20060102", quoteTradeDate, time.Local)
  86. if err != nil {
  87. logger.GetLogger().Errorf("QueryMarketRun failed: %s", err.Error())
  88. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_FAIL, nil)
  89. return
  90. }
  91. // !!!开休市计划明细
  92. curWeekRunSteps := make([]map[string]interface{}, 0)
  93. for _, v := range runSteps {
  94. tradeWeekDay := int(v["tradeweekday"].(float64))
  95. if tradeWeekDay == int(tradeDate.Weekday()) {
  96. curWeekRunSteps = append(curWeekRunSteps, v)
  97. }
  98. }
  99. // 获取不到可用的开休市计划
  100. if len(curWeekRunSteps) == 0 {
  101. logger.GetLogger().Errorf("QueryMarketRun failed: %s", err.Error())
  102. appG.Response(http.StatusBadRequest, e.ERROR_GET_RUNSTEP_FAILED, nil)
  103. return
  104. }
  105. // 按 SECTIONID 顺序排序
  106. sort.Slice(curWeekRunSteps, func(i int, j int) bool {
  107. return curWeekRunSteps[i]["sectionid"].(float64) < curWeekRunSteps[j]["sectionid"].(float64)
  108. })
  109. // 把各开休市时间段转化为真实的时间
  110. // 关于开休市计划的时间顺序:管理端会按时间顺序添加开休市计划,所以交易日开始时间为第一条记录的开始时间,结束时间为最后一条记录的结束时间
  111. // 关于目标商品的交易日问题:目前只能从商品所属市场获取当前交易日,这样有两个问题,一是不能按常规显示最后一个有历史数据的交易日;二是目前所有外部商品的开休市计划都是一样的。
  112. timeFormat := "20060102 15:04"
  113. // 开始时间
  114. startInterval := quote.GetTradeDay(int(curWeekRunSteps[0]["tradeweekday"].(float64)), int(curWeekRunSteps[0]["startweekday"].(float64)))
  115. marketRun.StartTime, _ = time.ParseInLocation(timeFormat, fmt.Sprintf("%s %s", quoteTradeDate, curWeekRunSteps[0]["starttime"].(string)), time.Local)
  116. if startInterval != 0 {
  117. duration, _ := time.ParseDuration(fmt.Sprintf("%dh", startInterval*24))
  118. marketRun.StartTime = marketRun.StartTime.Add(duration)
  119. }
  120. // 结束时间
  121. index := len(curWeekRunSteps) - 1
  122. endInterval := quote.GetTradeDay(int(curWeekRunSteps[index]["tradeweekday"].(float64)), int(curWeekRunSteps[index]["endweekday"].(float64)))
  123. marketRun.EndTime, _ = time.ParseInLocation(timeFormat, fmt.Sprintf("%s %s", quoteTradeDate, curWeekRunSteps[index]["endtime"].(string)), time.Local)
  124. if endInterval != 0 {
  125. duration, _ := time.ParseDuration(fmt.Sprintf("%dh", endInterval*24))
  126. marketRun.EndTime = marketRun.EndTime.Add(duration)
  127. }
  128. }
  129. // 查询成功返回
  130. logger.GetLogger().Debugln("QueryMarketRun successed: %v", marketRuns)
  131. appG.Response(http.StatusOK, e.SUCCESS, marketRuns)
  132. }
  133. // QueryMarketsByLoginIDReq 获取登录账号有权限的市场信息请求参数
  134. type QueryMarketsByLoginIDReq struct {
  135. LoginID int `form:"loginID" binding:"required"`
  136. }
  137. // QueryMarketsByLoginID 获取登录账号有权限的市场信息
  138. // @Summary 获取登录账号有权限的市场信息
  139. // @Produce json
  140. // @Security ApiKeyAuth
  141. // @Param loginID query int true "登录账号"
  142. // @Success 200 {object} models.Market
  143. // @Failure 500 {object} app.Response
  144. // @Router /Market/QueryMarketsByLoginID [get]
  145. // @Tags 通用市场
  146. func QueryMarketsByLoginID(c *gin.Context) {
  147. appG := app.Gin{C: c}
  148. // 获取请求参数
  149. var req QueryMarketsByLoginIDReq
  150. if err := appG.C.ShouldBindQuery(&req); err != nil {
  151. logger.GetLogger().Errorf("QueryMarketsByLoginID failed: %s", err.Error())
  152. appG.Response(http.StatusBadRequest, e.INVALID_PARAMS, nil)
  153. return
  154. }
  155. markets, err := models.GetMarketsByLoginID(req.LoginID)
  156. if err != nil {
  157. // 查询失败
  158. logger.GetLogger().Errorf("QueryMarketsByLoginID failed: %s", err.Error())
  159. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_FAIL, nil)
  160. return
  161. }
  162. // 查询成功返回
  163. logger.GetLogger().Debugln("QueryMarketsByLoginID successed: %v", markets)
  164. appG.Response(http.StatusOK, e.SUCCESS, markets)
  165. }
  166. // QueryGoodsesByLoginIDReq 获取登录账号有权限的商品信息请求参数
  167. type QueryGoodsesByLoginIDReq struct {
  168. LoginID int `form:"loginID" binding:"required"`
  169. MarketIDs string `form:"marketIDs"`
  170. }
  171. // QueryGoodsesByLoginID 获取登录账号有权限的商品信息
  172. // @Summary 获取登录账号有权限的商品信息
  173. // @Produce json
  174. // @Security ApiKeyAuth
  175. // @Param loginID query int true "登录账号"
  176. // @Param marketIDs query string false "市场ID列表,格式:1,2,3"
  177. // @Success 200 {object} models.Goods
  178. // @Failure 500 {object} app.Response
  179. // @Router /Market/QueryGoodsesByLoginID [get]
  180. // @Tags 通用市场
  181. func QueryGoodsesByLoginID(c *gin.Context) {
  182. appG := app.Gin{C: c}
  183. // 获取请求参数
  184. var req QueryGoodsesByLoginIDReq
  185. if err := appG.C.ShouldBindQuery(&req); err != nil {
  186. logger.GetLogger().Errorf("QueryGoodsesByLoginID failed: %s", err.Error())
  187. appG.Response(http.StatusBadRequest, e.INVALID_PARAMS, nil)
  188. return
  189. }
  190. goodses, err := models.GetGoodsByLoginID(req.LoginID, req.MarketIDs)
  191. if err != nil {
  192. // 查询失败
  193. logger.GetLogger().Errorf("QueryGoodsesByLoginID failed: %s", err.Error())
  194. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_FAIL, nil)
  195. return
  196. }
  197. // 查询成功返回
  198. logger.GetLogger().Debugln("QueryGoodsesByLoginID successed:", goodses)
  199. appG.Response(http.StatusOK, e.SUCCESS, goodses)
  200. }
  201. // GetAllExExchanges 获取所有外部交易所信息
  202. // @Summary 获取所有外部交易所信息
  203. // @Produce json
  204. // @Security ApiKeyAuth
  205. // @Success 200 {object} models.Goods
  206. // @Failure 500 {object} app.Response
  207. // @Router /Market/GetAllExExchanges [get]
  208. // @Tags 通用市场
  209. func GetAllExExchanges(c *gin.Context) {
  210. appG := app.Gin{C: c}
  211. rst, err := models.GetAllExExchanges()
  212. if err != nil {
  213. // 查询失败
  214. logger.GetLogger().Errorf("GetAllExExchanges failed: %s", err.Error())
  215. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_FAIL, nil)
  216. return
  217. }
  218. // 查询成功返回
  219. logger.GetLogger().Debugln("GetAllExExchanges successed: %v", rst)
  220. appG.Response(http.StatusOK, e.SUCCESS, rst)
  221. }
  222. // GetMarketSections 查询新板块设置
  223. // @Summary 查询新板块设置
  224. // @Produce json
  225. // @Success 200 {array} models.GetMarketSectionsRsp
  226. // @Failure 500 {object} app.Response
  227. // @Router /Market/GetMarketSections [get]
  228. // @Tags 通用市场
  229. func GetMarketSections(c *gin.Context) {
  230. appG := app.Gin{C: c}
  231. rst, err := models.GetMarketSections()
  232. if err != nil {
  233. // 查询失败
  234. logger.GetLogger().Errorf("GetMarketSections failed: %s", err.Error())
  235. appG.Response(http.StatusBadRequest, e.ERROR_QUERY_FAIL, nil)
  236. return
  237. }
  238. // 查询成功返回
  239. logger.GetLogger().Debugln("GetMarketSections successed: %v", rst)
  240. appG.Response(http.StatusOK, e.SUCCESS, rst)
  241. }