trade_analysis_data.go 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924
  1. package trade_analysis
  2. import (
  3. "eta/eta_api/models/data_manage"
  4. tradeAnalysisModel "eta/eta_api/models/data_manage/trade_analysis"
  5. "eta/eta_api/services/data"
  6. "eta/eta_api/utils"
  7. "fmt"
  8. "sort"
  9. "strings"
  10. "time"
  11. )
  12. // FormatCompanyTradeData2EdbMappings [公司-合约加总]转为指标数据
  13. func FormatCompanyTradeData2EdbMappings(companyTradeData []*tradeAnalysisModel.ContractCompanyTradeData, tradeType, dateType, dateTypeNum int, startDate, endDate string, chartEdbList []*data_manage.ChartSaveItem) (edbMappings []*data_manage.ChartEdbInfoMapping, chartName string, err error) {
  14. edbMappings = make([]*data_manage.ChartEdbInfoMapping, 0)
  15. if dateType <= 0 {
  16. dateType = utils.DateTypeOneMonth
  17. }
  18. // 期货公司名称作为标识进行匹配
  19. edbMap := make(map[string]*data_manage.ChartSaveItem)
  20. if len(chartEdbList) > 0 {
  21. for _, v := range chartEdbList {
  22. edbMap[v.UniqueFlag] = v
  23. }
  24. }
  25. for k, v := range companyTradeData {
  26. mapping := new(data_manage.ChartEdbInfoMapping)
  27. mapping.EdbName = v.CompanyName
  28. mapping.EdbNameEn = v.CompanyName
  29. mapping.EdbAliasName = v.CompanyName
  30. mapping.EdbAliasNameEn = v.CompanyName
  31. mapping.Frequency = "日度"
  32. mapping.FrequencyEn = data.GetFrequencyEn(mapping.Frequency)
  33. mapping.SourceName = utils.SourceNameTradeAnalysis
  34. mapping.Source = utils.CHART_SOURCE_TRADE_ANALYSIS_PROCESS
  35. mapping.IsAxis = 1
  36. mapping.EdbInfoType = 1
  37. mapping.StartDate = v.StartDate.Format(utils.FormatDate)
  38. mapping.EndDate = v.EndDate.Format(utils.FormatDate)
  39. mapping.ConvertUnit = tradeAnalysisModel.WarehouseDefaultUnit // 固定单位
  40. mapping.UniqueFlag = v.CompanyName // 期货公司名称作为每条曲线的唯一标识
  41. // 有配置那么取配置中的图例名称和左右轴
  42. edbConf := edbMap[mapping.UniqueFlag]
  43. if edbConf != nil {
  44. mapping.EdbName = edbConf.EdbAliasName
  45. mapping.EdbNameEn = edbConf.EdbAliasName
  46. mapping.EdbAliasName = edbConf.EdbAliasName
  47. mapping.EdbAliasNameEn = edbConf.EdbAliasName
  48. mapping.IsAxis = edbConf.IsAxis
  49. }
  50. // 根据参数取日期范围
  51. var startTime, endTime time.Time
  52. if dateType > 0 {
  53. st, ed := utils.GetDateByDateTypeV2(dateType, startDate, endDate, dateTypeNum, time.Time{})
  54. if st != "" {
  55. startTime, _ = time.ParseInLocation(utils.FormatDate, st, time.Local)
  56. }
  57. if startTime.IsZero() {
  58. startTime = v.StartDate
  59. }
  60. if ed != "" {
  61. endTime, _ = time.ParseInLocation(utils.FormatDate, ed, time.Local)
  62. }
  63. if endTime.IsZero() {
  64. endTime = v.EndDate
  65. }
  66. }
  67. // 指标数据和最值
  68. edbData := make([]*data_manage.EdbDataList, 0)
  69. var minData, maxData float64
  70. var setMinMax bool
  71. for _, dv := range v.DataList {
  72. if dv.Date.Before(startTime) || dv.Date.After(endTime) {
  73. continue
  74. }
  75. // 交易方向
  76. var (
  77. val float64
  78. hasVal bool
  79. )
  80. if tradeType == tradeAnalysisModel.WarehouseBuyChartType {
  81. if dv.BuyValType == tradeAnalysisModel.TradeDataTypeNull {
  82. continue
  83. }
  84. hasVal = true
  85. val = float64(dv.BuyVal)
  86. }
  87. if tradeType == tradeAnalysisModel.WarehouseSoldChartType {
  88. if dv.SoldValType == tradeAnalysisModel.TradeDataTypeNull {
  89. continue
  90. }
  91. hasVal = true
  92. val = float64(dv.SoldVal)
  93. }
  94. if tradeType == tradeAnalysisModel.WarehousePureBuyChartType {
  95. if dv.PureBuyValType == tradeAnalysisModel.TradeDataTypeNull {
  96. continue
  97. }
  98. hasVal = true
  99. val = float64(dv.PureBuyVal)
  100. }
  101. if !hasVal {
  102. continue
  103. }
  104. if !setMinMax {
  105. minData = val
  106. maxData = val
  107. setMinMax = true
  108. }
  109. if val < minData {
  110. minData = val
  111. }
  112. if val > maxData {
  113. maxData = val
  114. }
  115. edbData = append(edbData, &data_manage.EdbDataList{
  116. DataTime: dv.Date.Format(utils.FormatDate),
  117. DataTimestamp: dv.Date.UnixNano() / 1e6,
  118. Value: val,
  119. })
  120. }
  121. mapping.MinData = minData
  122. mapping.MaxData = maxData
  123. mapping.DataList = edbData
  124. edbMappings = append(edbMappings, mapping)
  125. // 图表默认名称
  126. if k == 0 {
  127. chartName += strings.ReplaceAll(v.ClassifyType, ",", "")
  128. }
  129. chartName += v.CompanyName
  130. }
  131. // 图表名称后缀
  132. chartName += tradeAnalysisModel.WarehouseTypeSuffixNames[tradeType]
  133. return
  134. }
  135. func GetWarehouseTradeData(exchange, classifyName string, contracts, companies []string, predictRatio float64) (companyTradeData []*tradeAnalysisModel.ContractCompanyTradeData, err error) {
  136. // 获取合约持仓数据
  137. contractTradeData, lastBuyVal, lastSoldVal, e := GetContractCompanyTradeData(exchange, []string{classifyName}, contracts, companies, time.Time{}, time.Time{})
  138. if e != nil {
  139. err = fmt.Errorf("获取合约-持仓数据失败, %v", e)
  140. return
  141. }
  142. // 填充[合约-公司]预估数据, 并根据[公司-多合约]分组, [公司]算作一个指标, 指标值为[多个合约]的计算加总
  143. companyContracts := make(map[string][]*tradeAnalysisModel.ContractCompanyTradeData)
  144. for _, v := range contractTradeData {
  145. td, fd, ed, e := PredictingTradeData(v.DataList, lastBuyVal[v.ClassifyType], lastSoldVal[v.ClassifyType], predictRatio)
  146. if e != nil {
  147. err = fmt.Errorf("数据补全失败, %v", e)
  148. return
  149. }
  150. v.DataList = td
  151. v.StartDate = fd
  152. v.EndDate = ed
  153. if companyContracts[v.CompanyName] == nil {
  154. companyContracts[v.CompanyName] = make([]*tradeAnalysisModel.ContractCompanyTradeData, 0)
  155. }
  156. companyContracts[v.CompanyName] = append(companyContracts[v.CompanyName], v)
  157. }
  158. // 以[公司]为组, 计算合约加总
  159. mussyTradeData := make(map[string]*tradeAnalysisModel.ContractCompanyTradeData)
  160. for k, v := range companyContracts {
  161. companyData := new(tradeAnalysisModel.ContractCompanyTradeData)
  162. companyData.CompanyName = k
  163. companyData.DataList = make([]*tradeAnalysisModel.ContractCompanyTradeDataList, 0)
  164. contractArr := make([]string, 0)
  165. // 合约加总
  166. sumDateData := make(map[time.Time]*tradeAnalysisModel.ContractCompanyTradeDataList)
  167. for _, vv := range v {
  168. contractArr = append(contractArr, vv.ClassifyType)
  169. for _, dv := range vv.DataList {
  170. if sumDateData[dv.Date] == nil {
  171. sumDateData[dv.Date] = new(tradeAnalysisModel.ContractCompanyTradeDataList)
  172. sumDateData[dv.Date].Date = dv.Date
  173. }
  174. // 数据类型以第一个非零值为准, 只处理多空和净多, 变化就不管了
  175. if sumDateData[dv.Date].BuyValType == tradeAnalysisModel.TradeDataTypeNull && dv.BuyValType > tradeAnalysisModel.TradeDataTypeNull {
  176. sumDateData[dv.Date].BuyValType = dv.BuyValType
  177. }
  178. if sumDateData[dv.Date].BuyValType == tradeAnalysisModel.TradeDataTypeOrigin && dv.BuyValType == tradeAnalysisModel.TradeDataTypeCalculate {
  179. sumDateData[dv.Date].BuyValType = dv.BuyValType
  180. }
  181. if dv.BuyValType > tradeAnalysisModel.TradeDataTypeNull {
  182. sumDateData[dv.Date].BuyVal += dv.BuyVal
  183. }
  184. // 空单
  185. if sumDateData[dv.Date].SoldValType == tradeAnalysisModel.TradeDataTypeNull && dv.SoldValType > tradeAnalysisModel.TradeDataTypeNull {
  186. sumDateData[dv.Date].SoldValType = dv.SoldValType
  187. }
  188. if sumDateData[dv.Date].SoldValType == tradeAnalysisModel.TradeDataTypeOrigin && dv.SoldValType == tradeAnalysisModel.TradeDataTypeCalculate {
  189. sumDateData[dv.Date].SoldValType = dv.SoldValType
  190. }
  191. if dv.SoldValType > tradeAnalysisModel.TradeDataTypeNull {
  192. sumDateData[dv.Date].SoldVal += dv.SoldVal
  193. }
  194. // 净多单
  195. if sumDateData[dv.Date].PureBuyValType == tradeAnalysisModel.TradeDataTypeNull && dv.PureBuyValType > tradeAnalysisModel.TradeDataTypeNull {
  196. sumDateData[dv.Date].PureBuyValType = dv.PureBuyValType
  197. }
  198. if sumDateData[dv.Date].PureBuyValType == tradeAnalysisModel.TradeDataTypeOrigin && dv.PureBuyValType == tradeAnalysisModel.TradeDataTypeCalculate {
  199. sumDateData[dv.Date].PureBuyValType = dv.PureBuyValType
  200. }
  201. if dv.PureBuyValType > tradeAnalysisModel.TradeDataTypeNull {
  202. sumDateData[dv.Date].PureBuyVal += dv.PureBuyVal
  203. }
  204. }
  205. // 多个合约比对开始结束时间
  206. if companyData.StartDate.IsZero() {
  207. companyData.StartDate = vv.StartDate
  208. }
  209. if vv.StartDate.Before(companyData.StartDate) {
  210. companyData.StartDate = vv.StartDate
  211. }
  212. if companyData.EndDate.IsZero() {
  213. companyData.EndDate = vv.EndDate
  214. }
  215. if vv.EndDate.Before(companyData.EndDate) {
  216. companyData.EndDate = vv.EndDate
  217. }
  218. }
  219. for _, sv := range sumDateData {
  220. companyData.DataList = append(companyData.DataList, sv)
  221. }
  222. sort.Slice(companyData.DataList, func(i, j int) bool {
  223. return companyData.DataList[i].Date.Before(companyData.DataList[j].Date)
  224. })
  225. companyData.ClassifyType = strings.Join(contractArr, ",")
  226. mussyTradeData[k] = companyData
  227. }
  228. // 数据根据公司排序, 不然会随机乱
  229. companyTradeData = make([]*tradeAnalysisModel.ContractCompanyTradeData, 0)
  230. for _, v := range companies {
  231. // 没数据也需要加进去, 不然edbList会少
  232. if mussyTradeData[v] == nil {
  233. companyData := new(tradeAnalysisModel.ContractCompanyTradeData)
  234. companyData.CompanyName = v
  235. companyData.DataList = make([]*tradeAnalysisModel.ContractCompanyTradeDataList, 0)
  236. companyTradeData = append(companyTradeData, companyData)
  237. continue
  238. }
  239. companyTradeData = append(companyTradeData, mussyTradeData[v])
  240. }
  241. return
  242. }
  243. // PredictingTradeData 根据数据库中的多空数据填充预估数据
  244. func PredictingTradeData(originData []*tradeAnalysisModel.ContractCompanyTradeDataList, lastBuyDateVal, lastSoldDateVal map[time.Time]int, predictRatio float64) (newData []*tradeAnalysisModel.ContractCompanyTradeDataList, firstDate, endDate time.Time, err error) {
  245. // 测试用的验证数据
  246. //lastBuyDateVal, lastSoldDateVal = make(map[time.Time]int), make(map[time.Time]int)
  247. //lastBuyDateVal[time.Date(2024, 7, 16, 0, 0, 0, 0, time.Local)] = 4602
  248. //lastBuyDateVal[time.Date(2024, 7, 17, 0, 0, 0, 0, time.Local)] = 5116
  249. //lastBuyDateVal[time.Date(2024, 7, 18, 0, 0, 0, 0, time.Local)] = 5130
  250. //lastBuyDateVal[time.Date(2024, 7, 19, 0, 0, 0, 0, time.Local)] = 5354
  251. //lastBuyDateVal[time.Date(2024, 7, 22, 0, 0, 0, 0, time.Local)] = 5916
  252. //lastBuyDateVal[time.Date(2024, 7, 23, 0, 0, 0, 0, time.Local)] = 6524
  253. //lastBuyDateVal[time.Date(2024, 7, 26, 0, 0, 0, 0, time.Local)] = 6575
  254. //lastBuyDateVal[time.Date(2024, 7, 29, 0, 0, 0, 0, time.Local)] = 7461
  255. //lastBuyDateVal[time.Date(2024, 7, 30, 0, 0, 0, 0, time.Local)] = 8488
  256. //
  257. //lastSoldDateVal[time.Date(2024, 7, 11, 0, 0, 0, 0, time.Local)] = 5467
  258. //lastSoldDateVal[time.Date(2024, 7, 12, 0, 0, 0, 0, time.Local)] = 5248
  259. //lastSoldDateVal[time.Date(2024, 7, 15, 0, 0, 0, 0, time.Local)] = 5102
  260. //lastSoldDateVal[time.Date(2024, 7, 16, 0, 0, 0, 0, time.Local)] = 4771
  261. //lastSoldDateVal[time.Date(2024, 7, 23, 0, 0, 0, 0, time.Local)] = 5989
  262. //lastSoldDateVal[time.Date(2024, 7, 26, 0, 0, 0, 0, time.Local)] = 6745
  263. //lastSoldDateVal[time.Date(2024, 7, 30, 0, 0, 0, 0, time.Local)] = 7272
  264. //
  265. //originData = make([]*tradeAnalysisModel.ContractCompanyTradeDataList, 0)
  266. //originData = append(originData, &tradeAnalysisModel.ContractCompanyTradeDataList{
  267. // Date: time.Date(2024, 7, 10, 0, 0, 0, 0, time.Local),
  268. // BuyVal: 14324,
  269. // BuyValType: tradeAnalysisModel.TradeDataTypeOrigin,
  270. // BuyChange: -1107,
  271. // BuyChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  272. // SoldVal: 0,
  273. // SoldValType: tradeAnalysisModel.TradeDataTypeNull,
  274. // SoldChange: 0,
  275. // SoldChangeType: tradeAnalysisModel.TradeDataTypeNull,
  276. //}, &tradeAnalysisModel.ContractCompanyTradeDataList{
  277. // Date: time.Date(2024, 7, 11, 0, 0, 0, 0, time.Local),
  278. // BuyVal: 14280,
  279. // BuyValType: tradeAnalysisModel.TradeDataTypeOrigin,
  280. // BuyChange: -44,
  281. // BuyChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  282. //}, &tradeAnalysisModel.ContractCompanyTradeDataList{
  283. // Date: time.Date(2024, 7, 12, 0, 0, 0, 0, time.Local),
  284. // BuyVal: 14214,
  285. // BuyValType: tradeAnalysisModel.TradeDataTypeOrigin,
  286. // BuyChange: -66,
  287. // BuyChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  288. //}, &tradeAnalysisModel.ContractCompanyTradeDataList{
  289. // Date: time.Date(2024, 7, 15, 0, 0, 0, 0, time.Local),
  290. // BuyVal: 14269,
  291. // BuyValType: tradeAnalysisModel.TradeDataTypeOrigin,
  292. // BuyChange: 55,
  293. // BuyChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  294. //}, &tradeAnalysisModel.ContractCompanyTradeDataList{
  295. // Date: time.Date(2024, 7, 17, 0, 0, 0, 0, time.Local),
  296. // SoldVal: 5254,
  297. // SoldValType: tradeAnalysisModel.TradeDataTypeOrigin,
  298. // SoldChange: 708,
  299. // SoldChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  300. //}, &tradeAnalysisModel.ContractCompanyTradeDataList{
  301. // Date: time.Date(2024, 7, 18, 0, 0, 0, 0, time.Local),
  302. // SoldVal: 6595,
  303. // SoldValType: tradeAnalysisModel.TradeDataTypeOrigin,
  304. // SoldChange: 1341,
  305. // SoldChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  306. //}, &tradeAnalysisModel.ContractCompanyTradeDataList{
  307. // Date: time.Date(2024, 7, 19, 0, 0, 0, 0, time.Local),
  308. // SoldVal: 5938,
  309. // SoldValType: tradeAnalysisModel.TradeDataTypeOrigin,
  310. // SoldChange: -657,
  311. // SoldChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  312. //}, &tradeAnalysisModel.ContractCompanyTradeDataList{
  313. // Date: time.Date(2024, 7, 22, 0, 0, 0, 0, time.Local),
  314. // SoldVal: 6131,
  315. // SoldValType: tradeAnalysisModel.TradeDataTypeOrigin,
  316. // SoldChange: 193,
  317. // SoldChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  318. //}, &tradeAnalysisModel.ContractCompanyTradeDataList{
  319. // Date: time.Date(2024, 7, 29, 0, 0, 0, 0, time.Local),
  320. // SoldVal: 6679,
  321. // SoldValType: tradeAnalysisModel.TradeDataTypeOrigin,
  322. // SoldChange: 312,
  323. // SoldChangeType: tradeAnalysisModel.TradeDataTypeOrigin,
  324. //})
  325. if len(originData) == 0 {
  326. return
  327. }
  328. if predictRatio < 0 || predictRatio > 1 {
  329. err = fmt.Errorf("估计参数不在0-1之间")
  330. return
  331. }
  332. sort.Slice(originData, func(i, j int) bool {
  333. return originData[i].Date.Before(originData[j].Date)
  334. })
  335. dateVal := make(map[time.Time]*tradeAnalysisModel.ContractCompanyTradeDataList)
  336. for _, v := range originData {
  337. dateVal[v.Date] = v
  338. }
  339. // 生成开始日期-1d(可能会往前面推算一天)至结束日期间的交易日, 以交易日为时间序列遍历
  340. tradeDays := utils.GetTradingDays(originData[0].Date.AddDate(0, 0, -1), originData[len(originData)-1].Date)
  341. for k, v := range tradeDays {
  342. // T日多空均无的情况
  343. //bothLast := false
  344. if dateVal[v] == nil {
  345. // T-1和T+1[原始数据]均无值, 那么T日无数据
  346. hasPrev, hasNext := false, false
  347. if k-1 >= 0 {
  348. hasPrev = true
  349. }
  350. if k+1 <= len(tradeDays)-1 {
  351. hasNext = true
  352. }
  353. if !hasPrev && !hasNext {
  354. continue
  355. }
  356. // T+1有值, 优先从T+1推, 然后继续走下面计算净多单的逻辑
  357. if hasNext {
  358. nextDay := tradeDays[k+1]
  359. if dateVal[nextDay] != nil {
  360. // T+1有多/空及多空变化, 且是原始数据, 那么推出数据并在map中新加一日数据
  361. if dateVal[nextDay].BuyValType == tradeAnalysisModel.TradeDataTypeOrigin && dateVal[nextDay].BuyChangeType == tradeAnalysisModel.TradeDataTypeOrigin {
  362. if _, ok := dateVal[v]; !ok {
  363. dateVal[v] = new(tradeAnalysisModel.ContractCompanyTradeDataList)
  364. dateVal[v].Date = v
  365. }
  366. dateVal[v].BuyVal = dateVal[nextDay].BuyVal - dateVal[nextDay].BuyChange
  367. dateVal[v].BuyValType = tradeAnalysisModel.TradeDataTypeOrigin
  368. }
  369. if dateVal[nextDay].SoldValType == tradeAnalysisModel.TradeDataTypeOrigin && dateVal[nextDay].SoldChangeType == tradeAnalysisModel.TradeDataTypeOrigin {
  370. if _, ok := dateVal[v]; !ok {
  371. dateVal[v] = new(tradeAnalysisModel.ContractCompanyTradeDataList)
  372. dateVal[v].Date = v
  373. }
  374. dateVal[v].SoldVal = dateVal[nextDay].SoldVal - dateVal[nextDay].SoldChange
  375. dateVal[v].SoldValType = tradeAnalysisModel.TradeDataTypeOrigin
  376. }
  377. }
  378. }
  379. // T+1没推出来而T-1有值, 那么T多空均取末位, 计算净多单
  380. _, has := dateVal[v]
  381. if hasPrev && !has {
  382. sv, sok := lastSoldDateVal[v]
  383. bv, bok := lastBuyDateVal[v]
  384. if !sok && !bok {
  385. continue
  386. }
  387. dateVal[v] = new(tradeAnalysisModel.ContractCompanyTradeDataList)
  388. dateVal[v].Date = v
  389. if sok {
  390. dateVal[v].SoldVal = int(predictRatio*float64(sv) + 0.5)
  391. dateVal[v].SoldValType = tradeAnalysisModel.TradeDataTypeCalculate
  392. }
  393. if bok {
  394. dateVal[v].BuyVal = int(predictRatio*float64(bv) + 0.5)
  395. dateVal[v].BuyValType = tradeAnalysisModel.TradeDataTypeCalculate
  396. }
  397. if dateVal[v].BuyValType > tradeAnalysisModel.TradeDataTypeNull && dateVal[v].SoldValType > tradeAnalysisModel.TradeDataTypeNull {
  398. dateVal[v].PureBuyVal = dateVal[v].BuyVal - dateVal[v].SoldVal
  399. dateVal[v].PureBuyValType = tradeAnalysisModel.TradeDataTypeCalculate
  400. }
  401. continue
  402. }
  403. }
  404. // 多空均有的情况下计算净多单
  405. if dateVal[v].BuyValType == tradeAnalysisModel.TradeDataTypeOrigin && dateVal[v].SoldValType == tradeAnalysisModel.TradeDataTypeOrigin {
  406. dateVal[v].PureBuyVal = dateVal[v].BuyVal - dateVal[v].SoldVal
  407. dateVal[v].PureBuyValType = tradeAnalysisModel.TradeDataTypeOrigin // 原始值算出来的也作原始值
  408. }
  409. // 仅有多单, 空单取末位, 计算净多单
  410. if dateVal[v].BuyValType == tradeAnalysisModel.TradeDataTypeOrigin && dateVal[v].SoldValType == tradeAnalysisModel.TradeDataTypeNull {
  411. if sv, ok := lastSoldDateVal[v]; ok {
  412. dateVal[v].SoldVal = int(predictRatio*float64(sv) + 0.5) // 估计参数*末位值, 向上取整
  413. dateVal[v].SoldValType = tradeAnalysisModel.TradeDataTypeCalculate
  414. dateVal[v].PureBuyVal = dateVal[v].BuyVal - dateVal[v].SoldVal
  415. dateVal[v].PureBuyValType = tradeAnalysisModel.TradeDataTypeCalculate
  416. }
  417. }
  418. // 仅有空单, 多单取末位, 计算净多单
  419. if dateVal[v].SoldValType == tradeAnalysisModel.TradeDataTypeOrigin && dateVal[v].BuyValType == tradeAnalysisModel.TradeDataTypeNull {
  420. if sv, ok := lastBuyDateVal[v]; ok {
  421. dateVal[v].BuyVal = int(predictRatio*float64(sv) + 0.5)
  422. dateVal[v].BuyValType = tradeAnalysisModel.TradeDataTypeCalculate
  423. dateVal[v].PureBuyVal = dateVal[v].BuyVal - dateVal[v].SoldVal
  424. dateVal[v].PureBuyValType = tradeAnalysisModel.TradeDataTypeCalculate
  425. }
  426. }
  427. }
  428. // 二次遍历, 计算与T-1的变化值
  429. for k, v := range tradeDays {
  430. // 无T/T-1数据, 忽略
  431. if dateVal[v] == nil {
  432. continue
  433. }
  434. if k-1 < 0 {
  435. continue
  436. }
  437. beforeDay := tradeDays[k-1]
  438. if dateVal[beforeDay] == nil {
  439. continue
  440. }
  441. // 多单变化
  442. if dateVal[v].BuyChangeType == tradeAnalysisModel.TradeDataTypeNull {
  443. if dateVal[v].BuyValType > tradeAnalysisModel.TradeDataTypeNull && dateVal[beforeDay].BuyValType > tradeAnalysisModel.TradeDataTypeNull {
  444. dateVal[v].BuyChange = dateVal[v].BuyVal - dateVal[beforeDay].BuyVal
  445. // 如果当日多单或者前日多单是估计值, 那么多单变化也为估计值
  446. if dateVal[v].BuyValType == tradeAnalysisModel.TradeDataTypeCalculate || dateVal[beforeDay].BuyValType == tradeAnalysisModel.TradeDataTypeCalculate {
  447. dateVal[v].BuyChangeType = tradeAnalysisModel.TradeDataTypeCalculate
  448. }
  449. }
  450. }
  451. // 空单变化
  452. if dateVal[v].SoldChangeType == tradeAnalysisModel.TradeDataTypeNull {
  453. if dateVal[v].SoldValType > tradeAnalysisModel.TradeDataTypeNull && dateVal[beforeDay].SoldValType > tradeAnalysisModel.TradeDataTypeNull {
  454. dateVal[v].SoldChange = dateVal[v].SoldVal - dateVal[beforeDay].SoldVal
  455. // 如果当日空单或者前日空单是估计值, 那么空单变化也为估计值
  456. if dateVal[v].SoldValType == tradeAnalysisModel.TradeDataTypeCalculate || dateVal[beforeDay].SoldValType == tradeAnalysisModel.TradeDataTypeCalculate {
  457. dateVal[v].SoldChangeType = tradeAnalysisModel.TradeDataTypeCalculate
  458. }
  459. }
  460. }
  461. // 净多变化
  462. if dateVal[v].PureBuyChangeType == tradeAnalysisModel.TradeDataTypeNull {
  463. if dateVal[v].PureBuyValType > tradeAnalysisModel.TradeDataTypeNull && dateVal[beforeDay].PureBuyValType > tradeAnalysisModel.TradeDataTypeNull {
  464. dateVal[v].PureBuyChange = dateVal[v].PureBuyVal - dateVal[beforeDay].PureBuyVal
  465. dateVal[v].PureBuyChangeType = tradeAnalysisModel.TradeDataTypeOrigin
  466. // 如果当日净多单或者前日净多单是估计值, 那么净多单变化也为估计值
  467. if dateVal[v].PureBuyValType == tradeAnalysisModel.TradeDataTypeCalculate || dateVal[beforeDay].PureBuyValType == tradeAnalysisModel.TradeDataTypeCalculate {
  468. dateVal[v].PureBuyChangeType = tradeAnalysisModel.TradeDataTypeCalculate
  469. }
  470. }
  471. }
  472. }
  473. // 重新遍历map, 生成数据序列并排序
  474. newData = make([]*tradeAnalysisModel.ContractCompanyTradeDataList, 0)
  475. for _, v := range dateVal {
  476. if v.BuyValType == tradeAnalysisModel.TradeDataTypeNull && v.SoldValType == tradeAnalysisModel.TradeDataTypeNull {
  477. continue
  478. }
  479. newData = append(newData, v)
  480. }
  481. sort.Slice(newData, func(i, j int) bool {
  482. return newData[i].Date.Before(newData[j].Date)
  483. })
  484. if len(newData) > 0 {
  485. firstDate = newData[0].Date
  486. endDate = newData[len(newData)-1].Date
  487. }
  488. return
  489. }
  490. // GetTopContractRank 获取TOP20根据成交量的合约排名
  491. func GetTopContractRank(exchange string, classifyNames []string, dataDate time.Time) (items []*tradeAnalysisModel.ContractTopRankData, err error) {
  492. // 郑商所/广期所查询方式不一样
  493. var tradeAnalysis TradeAnalysisInterface
  494. switch exchange {
  495. case tradeAnalysisModel.TradeExchangeZhengzhou:
  496. tradeAnalysis = &ZhengzhouTradeAnalysis{}
  497. case tradeAnalysisModel.TradeExchangeGuangzhou:
  498. tradeAnalysis = &GuangzhouTradeAnalysis{}
  499. default:
  500. tradeAnalysis = &BaseTradeAnalysis{}
  501. }
  502. // 郑商所-需要把所选品种转为实际合约进行后续的查询
  503. if exchange == tradeAnalysisModel.TradeExchangeZhengzhou {
  504. classifies, e := GetZhengzhouContractsByClassifyNames(classifyNames)
  505. if e != nil {
  506. err = fmt.Errorf("获取郑商所实际合约失败, %v", e)
  507. return
  508. }
  509. classifyNames = classifies
  510. }
  511. // 获取多单/空单原始数据
  512. rankData, e := tradeAnalysis.GetContractTopRankData(exchange, classifyNames, dataDate)
  513. if e != nil {
  514. err = fmt.Errorf("获取多空单原始数据失败, %v", e)
  515. return
  516. }
  517. items = make([]*tradeAnalysisModel.ContractTopRankData, 0)
  518. for _, v := range rankData {
  519. v.Exchange = exchange
  520. // 郑商所-这里注意把查出来的品种和合约赋值,不然后续是乱的
  521. if v.Exchange == tradeAnalysisModel.TradeExchangeZhengzhou {
  522. v.ClassifyType = v.ClassifyName
  523. v.ClassifyName = GetZhengzhouClassifyName(v.ClassifyName)
  524. }
  525. items = append(items, v)
  526. }
  527. return
  528. }
  529. // GetTableTradeData 获取多空分析表格持仓数据
  530. func GetTableTradeData(exchange string, classifyName string, contracts []string, companyName string, predictRatio float64, startDate, endDate time.Time, contractType int) (companyTradeData []*tradeAnalysisModel.ContractCompanyTradeData, err error) {
  531. companyTradeData = make([]*tradeAnalysisModel.ContractCompanyTradeData, 0)
  532. // 获取合约持仓数据
  533. contractTradeData, lastBuyVal, lastSoldVal, e := GetContractCompanyTradeData(exchange, []string{classifyName}, contracts, []string{companyName}, startDate, endDate)
  534. if e != nil {
  535. err = fmt.Errorf("获取合约-持仓数据失败, %v", e)
  536. return
  537. }
  538. // 填充[合约-公司]预估数据, 并根据[公司-多合约]分组, [公司]算作一个指标, 指标值为[多个合约]的计算加总
  539. companyContracts := make(map[string][]*tradeAnalysisModel.ContractCompanyTradeData)
  540. for _, v := range contractTradeData {
  541. td, fd, ed, e := PredictingTradeData(v.DataList, lastBuyVal[v.ClassifyType], lastSoldVal[v.ClassifyType], predictRatio)
  542. if e != nil {
  543. err = fmt.Errorf("数据补全失败, %v", e)
  544. return
  545. }
  546. v.DataList = td
  547. v.StartDate = fd
  548. v.EndDate = ed
  549. // 合约类型参数不为合约加总时, 每个合约算一行数据
  550. if contractType != tradeAnalysisModel.ContractQueryTypeTotal {
  551. companyTradeData = append(companyTradeData, v)
  552. continue
  553. }
  554. // 往下计算合约加总
  555. if companyContracts[v.CompanyName] == nil {
  556. companyContracts[v.CompanyName] = make([]*tradeAnalysisModel.ContractCompanyTradeData, 0)
  557. }
  558. companyContracts[v.CompanyName] = append(companyContracts[v.CompanyName], v)
  559. }
  560. // 类型为合约加总才往下合并
  561. if contractType != tradeAnalysisModel.ContractQueryTypeTotal {
  562. return
  563. }
  564. // 以[公司]为组, 计算合约加总
  565. for k, v := range companyContracts {
  566. companyData := new(tradeAnalysisModel.ContractCompanyTradeData)
  567. companyData.CompanyName = k
  568. companyData.DataList = make([]*tradeAnalysisModel.ContractCompanyTradeDataList, 0)
  569. contractArr := make([]string, 0)
  570. // 合约加总
  571. sumDateData := make(map[time.Time]*tradeAnalysisModel.ContractCompanyTradeDataList)
  572. for _, vv := range v {
  573. contractArr = append(contractArr, vv.ClassifyType)
  574. for _, dv := range vv.DataList {
  575. if sumDateData[dv.Date] == nil {
  576. sumDateData[dv.Date] = new(tradeAnalysisModel.ContractCompanyTradeDataList)
  577. sumDateData[dv.Date].Date = dv.Date
  578. }
  579. // 数据类型以第一个非零值为准, 只处理多空和净多, 变化就不管了
  580. if sumDateData[dv.Date].BuyValType == tradeAnalysisModel.TradeDataTypeNull && dv.BuyValType > tradeAnalysisModel.TradeDataTypeNull {
  581. sumDateData[dv.Date].BuyValType = dv.BuyValType
  582. }
  583. if sumDateData[dv.Date].BuyValType == tradeAnalysisModel.TradeDataTypeOrigin && dv.BuyValType == tradeAnalysisModel.TradeDataTypeCalculate {
  584. sumDateData[dv.Date].BuyValType = dv.BuyValType
  585. }
  586. if dv.BuyValType > tradeAnalysisModel.TradeDataTypeNull {
  587. sumDateData[dv.Date].BuyVal += dv.BuyVal
  588. }
  589. // 空单
  590. if sumDateData[dv.Date].SoldValType == tradeAnalysisModel.TradeDataTypeNull && dv.SoldValType > tradeAnalysisModel.TradeDataTypeNull {
  591. sumDateData[dv.Date].SoldValType = dv.SoldValType
  592. }
  593. if sumDateData[dv.Date].SoldValType == tradeAnalysisModel.TradeDataTypeOrigin && dv.SoldValType == tradeAnalysisModel.TradeDataTypeCalculate {
  594. sumDateData[dv.Date].SoldValType = dv.SoldValType
  595. }
  596. if dv.SoldValType > tradeAnalysisModel.TradeDataTypeNull {
  597. sumDateData[dv.Date].SoldVal += dv.SoldVal
  598. }
  599. // 净多单
  600. if sumDateData[dv.Date].PureBuyValType == tradeAnalysisModel.TradeDataTypeNull && dv.PureBuyValType > tradeAnalysisModel.TradeDataTypeNull {
  601. sumDateData[dv.Date].PureBuyValType = dv.PureBuyValType
  602. }
  603. if sumDateData[dv.Date].PureBuyValType == tradeAnalysisModel.TradeDataTypeOrigin && dv.PureBuyValType == tradeAnalysisModel.TradeDataTypeCalculate {
  604. sumDateData[dv.Date].PureBuyValType = dv.PureBuyValType
  605. }
  606. if dv.PureBuyValType > tradeAnalysisModel.TradeDataTypeNull {
  607. sumDateData[dv.Date].PureBuyVal += dv.PureBuyVal
  608. }
  609. }
  610. // 多个合约比对开始结束时间
  611. if companyData.StartDate.IsZero() {
  612. companyData.StartDate = vv.StartDate
  613. }
  614. if vv.StartDate.Before(companyData.StartDate) {
  615. companyData.StartDate = vv.StartDate
  616. }
  617. if companyData.EndDate.IsZero() {
  618. companyData.EndDate = vv.EndDate
  619. }
  620. if vv.EndDate.Before(companyData.EndDate) {
  621. companyData.EndDate = vv.EndDate
  622. }
  623. }
  624. for _, sv := range sumDateData {
  625. companyData.DataList = append(companyData.DataList, sv)
  626. }
  627. sort.Slice(companyData.DataList, func(i, j int) bool {
  628. return companyData.DataList[i].Date.Before(companyData.DataList[j].Date)
  629. })
  630. //companyData.ClassifyType = strings.Join(contractArr, ",")
  631. companyData.Exchange = exchange
  632. companyData.CompanyName = classifyName
  633. companyData.ClassifyType = classifyName
  634. companyTradeData = append(companyTradeData, companyData)
  635. }
  636. return
  637. }
  638. // GetCorrelationTableTradeData 获取相关性表格持仓数据
  639. func GetCorrelationTableTradeData(exchange string, classifyNames, contracts, companies []string, predictRatio float64) (tradeData []*tradeAnalysisModel.ContractCompanyTradeData, err error) {
  640. // 获取合约持仓数据
  641. contractTradeData, lastBuyVal, lastSoldVal, e := GetContractCompanyTradeData(exchange, classifyNames, contracts, companies, time.Time{}, time.Time{})
  642. if e != nil {
  643. err = fmt.Errorf("获取合约-持仓数据失败, %v", e)
  644. return
  645. }
  646. // 填充预估数据
  647. tradeData = make([]*tradeAnalysisModel.ContractCompanyTradeData, 0)
  648. for _, v := range contractTradeData {
  649. td, fd, ed, e := PredictingTradeData(v.DataList, lastBuyVal[v.ClassifyType], lastSoldVal[v.ClassifyType], predictRatio)
  650. if e != nil {
  651. err = fmt.Errorf("数据补全失败, %v", e)
  652. return
  653. }
  654. v.DataList = td
  655. v.StartDate = fd
  656. v.EndDate = ed
  657. tradeData = append(tradeData, v)
  658. }
  659. return
  660. }
  661. // TransTradeData2EdbData 持仓数据转为指标数据
  662. func TransTradeData2EdbData(tradeData []*tradeAnalysisModel.ContractCompanyTradeData, contractPosition int) (edbData []*tradeAnalysisModel.ContractCompanyTradeEdb, edbDataMap []map[time.Time]int, err error) {
  663. if len(tradeData) == 0 {
  664. return
  665. }
  666. edbData = make([]*tradeAnalysisModel.ContractCompanyTradeEdb, 0)
  667. edbDataMap = make([]map[time.Time]int, 0)
  668. for _, v := range tradeData {
  669. newEdb := new(tradeAnalysisModel.ContractCompanyTradeEdb)
  670. newEdb.Exchange = v.Exchange
  671. newEdb.ClassifyName = v.ClassifyName
  672. newEdb.ClassifyType = v.ClassifyType
  673. newEdb.CompanyName = v.CompanyName
  674. newEdb.IsTotal = v.IsTotal
  675. newEdb.ContractPosition = contractPosition
  676. newEdb.StartDate = v.StartDate
  677. newEdb.EndDate = v.EndDate
  678. newEdb.DataList = make([]*tradeAnalysisModel.ContractCompanyTradeEdbData, 0)
  679. dataMap := make(map[time.Time]int)
  680. for _, d := range v.DataList {
  681. var vd int
  682. switch contractPosition {
  683. case tradeAnalysisModel.ContractPositionBuy:
  684. if d.BuyValType == tradeAnalysisModel.TradeDataTypeNull {
  685. continue
  686. }
  687. vd = d.BuyVal
  688. case tradeAnalysisModel.ContractPositionSold:
  689. if d.SoldValType == tradeAnalysisModel.TradeDataTypeNull {
  690. continue
  691. }
  692. vd = d.SoldVal
  693. case tradeAnalysisModel.ContractPositionPureBuy:
  694. if d.PureBuyValType == tradeAnalysisModel.TradeDataTypeNull {
  695. continue
  696. }
  697. vd = d.PureBuyVal
  698. default:
  699. continue
  700. }
  701. newEdb.DataList = append(newEdb.DataList, &tradeAnalysisModel.ContractCompanyTradeEdbData{
  702. DataTime: d.Date,
  703. Val: vd,
  704. })
  705. dataMap[d.Date] = vd
  706. }
  707. edbData = append(edbData, newEdb)
  708. edbDataMap = append(edbDataMap, dataMap)
  709. }
  710. return
  711. }
  712. // GetContractCompanyTradeData 获取合约持仓数据
  713. func GetContractCompanyTradeData(exchange string, classifyNames, contracts, companies []string, startDate, endDate time.Time) (contractTradeData map[string]*tradeAnalysisModel.ContractCompanyTradeData, lastBuyVal, lastSoldVal map[string]map[time.Time]int, err error) {
  714. // 各原始数据表期货公司名称不一致
  715. companyMap := make(map[string]string)
  716. {
  717. ob := new(tradeAnalysisModel.TradeFuturesCompany)
  718. list, e := ob.GetItemsByCondition(``, make([]interface{}, 0), []string{}, "")
  719. if e != nil {
  720. err = fmt.Errorf("获取期货公司名称失败: %v", e)
  721. return
  722. }
  723. switch exchange {
  724. case "zhengzhou":
  725. for _, v := range list {
  726. companyMap[v.CompanyName] = v.ZhengzhouName
  727. }
  728. case "dalian":
  729. for _, v := range list {
  730. companyMap[v.CompanyName] = v.DalianName
  731. }
  732. case "shanghai":
  733. for _, v := range list {
  734. companyMap[v.CompanyName] = v.ShanghaiName
  735. }
  736. case "cffex":
  737. for _, v := range list {
  738. companyMap[v.CompanyName] = v.CffexName
  739. }
  740. case "ine":
  741. for _, v := range list {
  742. companyMap[v.CompanyName] = v.IneName
  743. }
  744. case "guangzhou":
  745. for _, v := range list {
  746. companyMap[v.CompanyName] = v.GuangzhouName
  747. }
  748. }
  749. }
  750. var queryCompanies []string
  751. for _, v := range companies {
  752. if v == tradeAnalysisModel.TradeFuturesCompanyTop20 {
  753. queryCompanies = append(queryCompanies, tradeAnalysisModel.TradeFuturesCompanyTop20)
  754. continue
  755. }
  756. companyName, ok := companyMap[v]
  757. if !ok {
  758. utils.FileLog.Info(fmt.Sprintf("交易所%s公司名称映射不存在: %s", exchange, v))
  759. continue
  760. }
  761. queryCompanies = append(queryCompanies, companyName)
  762. }
  763. // 郑商所/广期所查询方式不一样
  764. var tradeAnalysis TradeAnalysisInterface
  765. switch exchange {
  766. case tradeAnalysisModel.TradeExchangeZhengzhou:
  767. tradeAnalysis = &ZhengzhouTradeAnalysis{}
  768. case tradeAnalysisModel.TradeExchangeGuangzhou:
  769. tradeAnalysis = &GuangzhouTradeAnalysis{}
  770. default:
  771. tradeAnalysis = &BaseTradeAnalysis{}
  772. }
  773. // 获取多单/空单原始数据
  774. originList, _, lastOriginList, e := tradeAnalysis.GetTradeDataByContracts(exchange, classifyNames, contracts, queryCompanies, startDate, endDate)
  775. if e != nil {
  776. err = fmt.Errorf("获取多空单原始数据失败, %v", e)
  777. return
  778. }
  779. // [合约-期货公司]数据分组
  780. contractTradeData = make(map[string]*tradeAnalysisModel.ContractCompanyTradeData)
  781. {
  782. keyDateData := make(map[string]*tradeAnalysisModel.ContractCompanyTradeDataList)
  783. keyDateDataExist := make(map[string]bool)
  784. for _, v := range originList {
  785. companyName := v.CompanyName
  786. k := fmt.Sprintf("%s-%s", v.ClassifyType, companyName)
  787. if contractTradeData[k] == nil {
  788. contractTradeData[k] = new(tradeAnalysisModel.ContractCompanyTradeData)
  789. contractTradeData[k].Exchange = exchange
  790. contractTradeData[k].CompanyName = companyName
  791. contractTradeData[k].ClassifyName = v.ClassifyName
  792. contractTradeData[k].ClassifyType = v.ClassifyType
  793. contractTradeData[k].DataList = make([]*tradeAnalysisModel.ContractCompanyTradeDataList, 0)
  794. }
  795. kd := fmt.Sprintf("%s-%s", k, v.DataTime.Format(utils.FormatDate))
  796. if keyDateData[kd] == nil {
  797. keyDateData[kd] = new(tradeAnalysisModel.ContractCompanyTradeDataList)
  798. keyDateData[kd].Date = v.DataTime
  799. }
  800. if v.ValType == 1 {
  801. keyDateData[kd].BuyVal = v.Val
  802. keyDateData[kd].BuyValType = tradeAnalysisModel.TradeDataTypeOrigin
  803. keyDateData[kd].BuyChange = v.ValChange
  804. keyDateData[kd].BuyChangeType = tradeAnalysisModel.TradeDataTypeOrigin
  805. }
  806. if v.ValType == 2 {
  807. keyDateData[kd].SoldVal = v.Val
  808. keyDateData[kd].SoldValType = tradeAnalysisModel.TradeDataTypeOrigin
  809. keyDateData[kd].SoldChange = v.ValChange
  810. keyDateData[kd].SoldChangeType = tradeAnalysisModel.TradeDataTypeOrigin
  811. }
  812. if !keyDateDataExist[kd] {
  813. contractTradeData[k].DataList = append(contractTradeData[k].DataList, keyDateData[kd])
  814. keyDateDataExist[kd] = true
  815. }
  816. }
  817. }
  818. // 合约的[日期-末位值]
  819. lastBuyVal = make(map[string]map[time.Time]int)
  820. lastSoldVal = make(map[string]map[time.Time]int)
  821. {
  822. for _, v := range lastOriginList {
  823. if v.ValType == 1 {
  824. if lastBuyVal[v.ClassifyType] == nil {
  825. lastBuyVal[v.ClassifyType] = make(map[time.Time]int)
  826. }
  827. lastBuyVal[v.ClassifyType][v.DataTime] = v.Val
  828. continue
  829. }
  830. if lastSoldVal[v.ClassifyType] == nil {
  831. lastSoldVal[v.ClassifyType] = make(map[time.Time]int)
  832. }
  833. lastSoldVal[v.ClassifyType][v.DataTime] = v.Val
  834. }
  835. }
  836. return
  837. }
  838. // GetTradeClassifyNewestDataTime 获取数据最新日期
  839. func GetTradeClassifyNewestDataTime(exchange string, classifyNames []string) (dataTime time.Time, err error) {
  840. var tradeAnalysis TradeAnalysisInterface
  841. switch exchange {
  842. case tradeAnalysisModel.TradeExchangeZhengzhou:
  843. tradeAnalysis = &ZhengzhouTradeAnalysis{}
  844. case tradeAnalysisModel.TradeExchangeGuangzhou:
  845. tradeAnalysis = &GuangzhouTradeAnalysis{}
  846. default:
  847. tradeAnalysis = &BaseTradeAnalysis{}
  848. }
  849. if exchange == tradeAnalysisModel.TradeExchangeZhengzhou {
  850. classifies, e := GetZhengzhouContractsByClassifyNames(classifyNames)
  851. if e != nil {
  852. err = fmt.Errorf("获取郑商所实际合约失败, %v", e)
  853. return
  854. }
  855. classifyNames = classifies
  856. }
  857. d, e := tradeAnalysis.GetClassifyNewestDataTime(exchange, classifyNames)
  858. if e != nil && e.Error() != utils.ErrNoRow() {
  859. err = fmt.Errorf("获取品种最新数据日期失败, %v", e)
  860. return
  861. }
  862. if !d.IsZero() {
  863. dataTime = d
  864. } else {
  865. dataTime = time.Now().Local()
  866. }
  867. return
  868. }