trade_position_analysis.go 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511
  1. package data
  2. import (
  3. "context"
  4. "fmt"
  5. "hongze/hongze_task/models/data_manage"
  6. "hongze/hongze_task/utils"
  7. "sort"
  8. "strconv"
  9. "time"
  10. )
  11. // InitPositionTask 统计今日交易所的持仓分析数据
  12. func InitPositionTask(cont context.Context) (err error) {
  13. exchanges := []string{"zhengzhou","dalian","shanghai","cffex","ine"} //郑商所,大商所,上期所,中金所,上期能源
  14. startDate := time.Now().Format(utils.FormatDate)
  15. endDate := startDate
  16. for _, v := range exchanges {
  17. exchange := v
  18. err = nil
  19. fmt.Println("InitPositionTask: 启动:"+exchange)
  20. utils.FileLog.Info("InitPositionTask: 启动:"+exchange)
  21. fmt.Println("开始"+startDate+"结束"+endDate)
  22. utils.FileLog.Info(fmt.Sprintf("InitTradePosition:开始:%s; 结束:%s", startDate, endDate))
  23. tErr, errMsg := InitTradePosition(exchange, startDate, endDate)
  24. if tErr != nil {
  25. err = tErr
  26. fmt.Println("InitTradePosition: 操作失败:"+errMsg+tErr.Error())
  27. utils.FileLog.Info(fmt.Sprintf("InitTradePosition: 操作失败:%s:%s", errMsg, tErr.Error()))
  28. continue
  29. }
  30. fmt.Println("InitTradePosition:"+exchange+"已完成")
  31. utils.FileLog.Info("InitTradePosition:"+exchange+"已完成")
  32. }
  33. return
  34. }
  35. func InitTradePosition(exchange, startDate, endDate string) (err error, errMsg string) {
  36. // 批量插入今日的初始值
  37. num, err := data_manage.GetTradePositionTopCountByExchangeDataTime(exchange, startDate, endDate)
  38. if err != nil {
  39. errMsg = "查询原始数据失败,GetTradePositionTopCountByExchangeDataTime() Err: "
  40. return
  41. }
  42. if num > 0 {
  43. err = fmt.Errorf("数据已存在,无需处理")
  44. return
  45. }
  46. err = data_manage.MultiInsertTradeBaseDataToTop(exchange, startDate, endDate)
  47. if err != nil {
  48. errMsg = "新增原始数据失败,MultiInsertTradeBaseDataToTop() Err: "
  49. return
  50. }
  51. originList, err := data_manage.GetTradePositionTopByExchangeDataTime(exchange, startDate, endDate)
  52. if err != nil {
  53. errMsg = "查询原始数据失败, GetTradePositionTopByExchangeDataTime() Err: "
  54. return
  55. }
  56. if len(originList) <= 0 {
  57. err = fmt.Errorf("原始数据没有值")
  58. return
  59. }
  60. now := time.Now()
  61. dataTimeMap := make(map[string]*data_manage.TradePositionTop)
  62. onlyEmptyMap := make(map[string]bool)
  63. onlyEmptyNameMap := make(map[string]*data_manage.TradePositionTop)
  64. topLastMap := make(map[string]int)
  65. topLastRankMap := make(map[string]int)
  66. list := make([]*data_manage.TradePositionTop, 0)
  67. for _, v := range originList {
  68. tmp0, tmpErr := dealTradeOriginData(dataTimeMap, onlyEmptyMap, onlyEmptyNameMap, v, topLastMap, topLastRankMap, startDate, now)
  69. if tmpErr != nil {
  70. err = tmpErr
  71. errMsg = "处理原始数据失败 dealTradeOriginData() Err: "
  72. return
  73. }
  74. if tmp0 != nil {
  75. list = append(list, tmp0)
  76. }
  77. if len(list) >= 1000 {
  78. err = data_manage.InsertMultiTradePositionTop(exchange, list)
  79. if err != nil {
  80. errMsg = "批量新增昨日数据失败,InsertMultiTradePositionTop() Err: "
  81. return
  82. }
  83. list = make([]*data_manage.TradePositionTop, 0)
  84. }
  85. }
  86. if len(list) > 0 {
  87. err = data_manage.InsertMultiTradePositionTop(exchange, list)
  88. if err != nil {
  89. errMsg = "批量新增昨日数据失败,InsertMultiTradePositionTop() Err: "
  90. return
  91. }
  92. list = make([]*data_manage.TradePositionTop, 0)
  93. }
  94. // 处理某个期货公司只有买单没有卖单,或者只有卖单没有买单的情况
  95. for k, v := range onlyEmptyNameMap {
  96. _, ok1 := onlyEmptyMap[k+"_1"]
  97. _, ok2 := onlyEmptyMap[k+"_2"]
  98. var dealType int
  99. if ok1 && !ok2 {
  100. dealType = 2 //只有买单没有卖单
  101. } else if !ok1 && ok2 {
  102. dealType = 1 //只有卖单没有买单的情况
  103. } else {
  104. continue
  105. }
  106. if dealType > 0 {
  107. str := v.ClassifyName + "_" + v.ClassifyType + "_" + v.DataTime + "_" + strconv.Itoa(dealType)
  108. dealValue := 0
  109. if lastVal, ok := topLastMap[str]; ok {
  110. dealValue = int(float64(lastVal)*0.7 + 0.5)
  111. }
  112. tmp := &data_manage.TradePositionTop{
  113. ClassifyName: v.ClassifyName,
  114. ClassifyType: v.ClassifyType,
  115. DealShortName: v.DealShortName,
  116. DataTime: v.DataTime,
  117. DealValue: dealValue,
  118. CreateTime: now,
  119. ModifyTime: now,
  120. DealType: dealType,
  121. SourceType: 2,
  122. }
  123. list = append(list, tmp)
  124. if len(list) >= 1000 {
  125. err = data_manage.InsertMultiTradePositionTop(exchange, list)
  126. if err != nil {
  127. errMsg = "批量新增前日数据失败,InsertMultiTradePositionTop() Err: "
  128. return
  129. }
  130. list = make([]*data_manage.TradePositionTop, 0)
  131. }
  132. }
  133. }
  134. if len(list) > 0 {
  135. err = data_manage.InsertMultiTradePositionTop(exchange, list)
  136. if err != nil {
  137. errMsg = "批量新增前日数据失败,InsertMultiTradePositionTop() Err: "
  138. return
  139. }
  140. }
  141. //生成净多单,净空单榜单
  142. err = createAnalysisCleanTop(exchange, startDate, endDate)
  143. if err != nil {
  144. errMsg = "创建净多单,净空单数据失败,createAnalysisCleanTop() Err: "
  145. return
  146. }
  147. // 特殊处理起始日期前一天的数据
  148. err = DealYesterdayData(exchange, startDate)
  149. if err != nil {
  150. errMsg = "处理昨日数据失败,DealYesterdayData() Err: "
  151. return
  152. }
  153. return
  154. }
  155. func dealTradeOriginData(dataTimeMap map[string]*data_manage.TradePositionTop, onlyEmptyMap map[string]bool, onlyEmptyNameMap map[string]*data_manage.TradePositionTop, currentItem *data_manage.TradePositionTop, topLastMap map[string]int, topLastRankMap map[string]int, startDate string, now time.Time) (tmp0 *data_manage.TradePositionTop, err error) {
  156. classifyName := currentItem.ClassifyName
  157. classifyType := currentItem.ClassifyType
  158. dealShortName := currentItem.DealShortName
  159. dealValue := currentItem.DealValue
  160. dealChange := currentItem.DealChange
  161. dataTime := currentItem.DataTime
  162. dealType := currentItem.DealType
  163. dealTypeStr := strconv.Itoa(dealType)
  164. dataTimeMap[classifyName+"_"+classifyType+"_"+dealTypeStr+"_"+dealShortName+"_"+dataTime] = currentItem
  165. onlyEmptyMap[classifyName+"_"+classifyType+"_"+dataTime+"_"+dealShortName+"_"+dealTypeStr] = true
  166. onlyEmptyNameMap[classifyName+"_"+classifyType+"_"+dataTime+"_"+dealShortName] = currentItem
  167. if currentItem.Rank > topLastRankMap[classifyName+"_"+classifyType+"_"+dataTime+"_"+dealTypeStr] {
  168. topLastMap[classifyName+"_"+classifyType+"_"+dataTime+"_"+dealTypeStr] = dealValue
  169. topLastRankMap[classifyName+"_"+classifyType+"_"+dataTime+"_"+dealTypeStr] = currentItem.Rank
  170. }
  171. if dataTime > startDate {
  172. tmpTimeStr, tErr := getYesterdayDate(dataTime)
  173. if tErr != nil {
  174. err = tErr
  175. return
  176. }
  177. if tmpTimeStr < startDate {
  178. return
  179. }
  180. // 判断T-1日是否有值, 如果T-1日为空,则根据T日的值计算出T-1的值
  181. if _, ok := dataTimeMap[classifyName+"_"+classifyType+"_"+dealTypeStr+"_"+dealShortName+"_"+tmpTimeStr]; !ok {
  182. yesterdayVal := dealValue - dealChange
  183. yesterdayChange := 0
  184. beforeYesterday, _ := getYesterdayDate(tmpTimeStr)
  185. beforeYesterdayItem, ok1 := dataTimeMap[classifyName+"_"+classifyType+"_"+dealTypeStr+"_"+dealShortName+"_"+beforeYesterday]
  186. if ok1 {
  187. yesterdayChange = yesterdayVal - beforeYesterdayItem.DealValue
  188. }
  189. tmp0 = &data_manage.TradePositionTop{
  190. ClassifyName: classifyName,
  191. ClassifyType: classifyType,
  192. DealShortName: dealShortName,
  193. DealValue: yesterdayVal,
  194. DealChange: yesterdayChange,
  195. DataTime: tmpTimeStr,
  196. CreateTime: now,
  197. ModifyTime: now,
  198. DealType: dealType,
  199. SourceType: 1,
  200. }
  201. dataTimeMap[classifyName+"_"+classifyType+"_"+dealTypeStr+"_"+dealShortName+"_"+tmpTimeStr] = tmp0
  202. onlyEmptyMap[classifyName+"_"+classifyType+"_"+tmpTimeStr+"_"+dealShortName+"_"+dealTypeStr] = true
  203. onlyEmptyNameMap[classifyName+"_"+classifyType+"_"+tmpTimeStr+"_"+dealShortName] = tmp0
  204. }
  205. }
  206. return
  207. }
  208. // 更新昨日数据
  209. func DealYesterdayData(exchange, startDate string) (err error) {
  210. // 查询最早的日期
  211. firstItem, err := data_manage.GetFirstBaseFromTradeIndexByDate(exchange)
  212. if err != nil {
  213. return
  214. }
  215. if startDate == firstItem.DataTime { //如果当前是起始日,则无需统计修改前一天的数据
  216. return
  217. }
  218. yesterdayStr, err := getYesterdayDate(startDate)
  219. if err != nil {
  220. return
  221. }
  222. //查找前日的值,并更新对应的更改
  223. beforeYesterdayStr, err := getYesterdayDate(yesterdayStr)
  224. if err != nil {
  225. return
  226. }
  227. // 先查出T日最原始的数据
  228. originList, err := data_manage.GetTradePositionTopByExchangeDataTime(exchange, startDate, startDate)
  229. if err != nil {
  230. return
  231. }
  232. originBuyMap := make(map[string]*data_manage.TradePositionTop)
  233. originSoldMap := make(map[string]*data_manage.TradePositionTop)
  234. for _, v := range originList {
  235. if v.SourceType != 0 {
  236. continue
  237. }
  238. str := v.ClassifyName + "_" + v.ClassifyType + "_" + v.DealShortName
  239. if v.DealType == 1 {
  240. originBuyMap[str] = v
  241. } else if v.DealType == 2 {
  242. originSoldMap[str] = v
  243. }
  244. }
  245. // 然后查询T-1中数据来源类型是2的数据
  246. changeList, err := data_manage.GetTradePositionTopByExchangeSourceType(exchange, yesterdayStr, 2)
  247. if err != nil {
  248. return
  249. }
  250. if len(changeList) <= 0 {
  251. //err = fmt.Errorf("前天的数据无需修改")
  252. return
  253. }
  254. // 查询出前日的成交量
  255. beforeYesterdayList, err := data_manage.GetTradePositionTopByExchangeDataTime(exchange, beforeYesterdayStr, beforeYesterdayStr)
  256. if err != nil {
  257. return
  258. }
  259. beforeYesterdayMap1 := make(map[string]int)
  260. beforeYesterdayMap2 := make(map[string]int)
  261. if len(beforeYesterdayList) > 0 {
  262. for _, v := range beforeYesterdayList {
  263. if v.SourceType == 2 {
  264. continue
  265. }
  266. str := v.ClassifyName + "_" + v.ClassifyType + "_" + v.DealShortName
  267. if v.DealType == 1 {
  268. beforeYesterdayMap1[str] = v.DealValue
  269. } else if v.DealType == 2 {
  270. beforeYesterdayMap2[str] = v.DealValue
  271. }
  272. }
  273. }
  274. // 根据原始数据中的值推算出最新的值
  275. now := time.Now()
  276. // 批量更新到分析表中,
  277. var updateAnalysisData []data_manage.UpdateDealValueChange
  278. for _, v := range changeList {
  279. str := v.ClassifyName + "_" + v.ClassifyType + "_" + v.DealShortName
  280. dealValue := 0
  281. dealChange := 0
  282. if v.DealType == 1 {
  283. if n, ok := originBuyMap[str]; ok {
  284. dealValue = n.DealValue - n.DealChange
  285. if beforeVal, ok1 := beforeYesterdayMap1[str]; ok1 {
  286. dealChange = dealValue - beforeVal
  287. }
  288. tmp := data_manage.UpdateDealValueChange{
  289. Id: v.Id,
  290. DealValue: dealValue,
  291. DealChange: dealChange,
  292. SourceType: 1,
  293. ModifyTime: now,
  294. }
  295. updateAnalysisData = append(updateAnalysisData, tmp)
  296. }
  297. } else if v.DealType == 2 {
  298. if n, ok := originSoldMap[str]; ok {
  299. dealValue = n.DealValue - n.DealChange
  300. if beforeVal, ok1 := beforeYesterdayMap2[str]; ok1 {
  301. dealChange = dealValue - beforeVal
  302. }
  303. tmp := data_manage.UpdateDealValueChange{
  304. Id: v.Id,
  305. DealValue: dealValue,
  306. DealChange: dealChange,
  307. SourceType: 1,
  308. ModifyTime: now,
  309. }
  310. updateAnalysisData = append(updateAnalysisData, tmp)
  311. }
  312. }
  313. }
  314. if len(updateAnalysisData) > 0 {
  315. err = data_manage.MultiUpdatePositionTop(exchange, updateAnalysisData)
  316. if err != nil {
  317. return
  318. }
  319. //删除T-1日净多单和净空单的榜单
  320. err = data_manage.DeletePositionTopByDataTime(exchange, yesterdayStr, 3)
  321. if err != nil {
  322. return
  323. }
  324. err = data_manage.DeletePositionTopByDataTime(exchange, yesterdayStr, 4)
  325. if err != nil {
  326. return
  327. }
  328. //重新生成净多单和净空单的榜单
  329. err = createAnalysisCleanTop(exchange, yesterdayStr, yesterdayStr)
  330. if err != nil {
  331. return
  332. }
  333. }
  334. return
  335. }
  336. // createAnalysisCleanTop 生成净多单,净空单榜单
  337. func createAnalysisCleanTop(exchange, startDate, endDate string) (err error) {
  338. topList := make([]*data_manage.TradePositionTop, 0)
  339. now := time.Now()
  340. var subDataList data_manage.TradePositionSubList
  341. subChangeMap1 := make(map[string]int) //净多单map
  342. subChangeMap2 := make(map[string]int) //净空单map
  343. //查询所有差值数据,
  344. yesterday, err := getYesterdayDate(startDate)
  345. if err != nil {
  346. return
  347. }
  348. yesterdayTopList1, tErr := data_manage.GetTradePositionTopByExchangeDataTimeType(exchange, yesterday, 3)
  349. if tErr != nil {
  350. err = tErr
  351. return
  352. }
  353. if len(yesterdayTopList1) > 0 {
  354. for _, v := range yesterdayTopList1 {
  355. nameStr := v.ClassifyName + "_" + v.ClassifyType + "_" + v.DataTime + "_" + v.DealShortName
  356. subChangeMap1[nameStr] = v.DealValue
  357. }
  358. }
  359. yesterdayTopList2, tErr := data_manage.GetTradePositionTopByExchangeDataTimeType(exchange, yesterday, 4)
  360. if tErr != nil {
  361. err = tErr
  362. return
  363. }
  364. if len(yesterdayTopList2) > 0 {
  365. for _, v := range yesterdayTopList2 {
  366. nameStr := v.ClassifyName + "_" + v.ClassifyType + "_" + v.DataTime + "_" + v.DealShortName
  367. subChangeMap2[nameStr] = v.DealValue
  368. }
  369. }
  370. originDataList, err := data_manage.GetTradePositionTopByExchangeDataTime(exchange, startDate, endDate)
  371. if err != nil {
  372. return
  373. }
  374. buyDataMap := make(map[string]int)
  375. for _, v := range originDataList {
  376. str := v.ClassifyName + "_" + v.ClassifyType + "_" + v.DataTime + "_" + v.DealShortName
  377. if v.DealType == 1 {
  378. buyDataMap[str] = v.DealValue
  379. } else if v.DealType == 2 {
  380. subValue := 0
  381. dealType := 0
  382. if buy, ok := buyDataMap[str]; ok {
  383. subValue = buy - v.DealValue
  384. if subValue >= 0 {
  385. dealType = 3
  386. } else {
  387. subValue = -subValue
  388. dealType = 4
  389. }
  390. }
  391. tmp := &data_manage.TradePositionSub{
  392. ClassifyName: v.ClassifyName,
  393. ClassifyType: v.ClassifyType,
  394. DataTime: v.DataTime,
  395. DealShortName: v.DealShortName,
  396. SubValue: subValue,
  397. DealType: dealType,
  398. }
  399. subDataList = append(subDataList, tmp)
  400. }
  401. }
  402. if len(subDataList) > 0 {
  403. sort.Sort(subDataList)
  404. }
  405. var dealType int
  406. rankMap := make(map[string]int)
  407. for _, v := range subDataList {
  408. subValue := v.SubValue
  409. nameStr := v.ClassifyName + "_" + v.ClassifyType + "_" + v.DataTime + "_" + v.DealShortName
  410. if v.DealType == 3 {
  411. subChangeMap1[nameStr] = subValue
  412. dealType = 3
  413. if _, ok := rankMap[v.ClassifyName+"_"+v.ClassifyType+"_"+v.DataTime+"_3"]; !ok {
  414. rankMap[v.ClassifyName+"_"+v.ClassifyType+"_"+v.DataTime+"_3"] = 1
  415. } else {
  416. rankMap[v.ClassifyName+"_"+v.ClassifyType+"_"+v.DataTime+"_3"]++
  417. }
  418. } else if v.DealType == 4 {
  419. subChangeMap2[nameStr] = subValue
  420. dealType = 4
  421. if _, ok := rankMap[v.ClassifyName+"_"+v.ClassifyType+"_"+v.DataTime+"_4"]; !ok {
  422. rankMap[v.ClassifyName+"_"+v.ClassifyType+"_"+v.DataTime+"_4"] = 1
  423. } else {
  424. rankMap[v.ClassifyName+"_"+v.ClassifyType+"_"+v.DataTime+"_4"]++
  425. }
  426. }
  427. //和T-1日比较差值
  428. var tmpTimeStr string
  429. tmpTimeStr, err = getYesterdayDate(v.DataTime)
  430. if err != nil {
  431. return
  432. }
  433. yesterdayStr := v.ClassifyName + "_" + v.ClassifyType + "_" + tmpTimeStr + "_" + v.DealShortName
  434. dealChange := 0
  435. if dealType == 3 {
  436. if c, ok := subChangeMap1[yesterdayStr]; ok {
  437. dealChange = subValue - c
  438. }
  439. } else if dealType == 4 {
  440. if c, ok := subChangeMap2[yesterdayStr]; ok {
  441. dealChange = subValue - c
  442. }
  443. }
  444. tmp := &data_manage.TradePositionTop{
  445. ClassifyName: v.ClassifyName,
  446. ClassifyType: v.ClassifyType,
  447. DataTime: v.DataTime,
  448. CreateTime: now,
  449. ModifyTime: now,
  450. DealShortName: v.DealShortName,
  451. DealValue: subValue,
  452. DealChange: dealChange,
  453. DealType: dealType,
  454. Rank: rankMap[v.ClassifyName+"_"+v.ClassifyType+"_"+v.DataTime+"_"+strconv.Itoa(dealType)],
  455. }
  456. topList = append(topList, tmp)
  457. if len(topList) >= 1000 {
  458. err = data_manage.InsertMultiTradePositionTop(exchange, topList)
  459. if err != nil {
  460. return
  461. }
  462. topList = make([]*data_manage.TradePositionTop, 0)
  463. }
  464. }
  465. if len(topList) >= 0 {
  466. err = data_manage.InsertMultiTradePositionTop(exchange, topList)
  467. if err != nil {
  468. return
  469. }
  470. }
  471. return
  472. }
  473. func getYesterdayDate(today string) (yesterday string, err error) {
  474. i := 1
  475. tmpTime, err := time.ParseInLocation(utils.FormatDate, today, time.Local)
  476. if err != nil {
  477. return
  478. }
  479. tmpTimeDate := tmpTime.AddDate(0, 0, -i)
  480. weekStr := tmpTimeDate.Weekday().String()
  481. if weekStr == "Sunday" {
  482. i += 2
  483. } else if weekStr == "Saturday" {
  484. i += 1
  485. }
  486. tmpTimeDate = tmpTime.AddDate(0, 0, -i)
  487. yesterday = tmpTimeDate.Format(utils.FormatDate)
  488. return
  489. }