chart_info.go 49 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527
  1. package correlation
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "eta/eta_api/models/data_manage"
  6. "eta/eta_api/models/system"
  7. "eta/eta_api/services/alarm_msg"
  8. "eta/eta_api/services/data"
  9. "eta/eta_api/utils"
  10. "fmt"
  11. "github.com/shopspring/decimal"
  12. "math"
  13. "sort"
  14. "strconv"
  15. "strings"
  16. "time"
  17. )
  18. // HandleDataByLinearRegression 线性方程插值法补全数据
  19. func HandleDataByLinearRegression(originList []*data_manage.EdbDataList, handleDataMap map[string]float64) (newList []*data_manage.EdbDataList, err error) {
  20. if len(originList) < 2 {
  21. return
  22. }
  23. var startEdbInfoData *data_manage.EdbDataList
  24. for _, v := range originList {
  25. handleDataMap[v.DataTime] = v.Value
  26. // 第一个数据就给过滤了,给后面的试用
  27. if startEdbInfoData == nil {
  28. startEdbInfoData = v
  29. newList = append(newList, &data_manage.EdbDataList{
  30. DataTime: v.DataTime,
  31. Value: v.Value,
  32. })
  33. continue
  34. }
  35. // 获取两条数据之间相差的天数
  36. startDataTime, _ := time.ParseInLocation(utils.FormatDate, startEdbInfoData.DataTime, time.Local)
  37. currDataTime, _ := time.ParseInLocation(utils.FormatDate, v.DataTime, time.Local)
  38. betweenHour := int(currDataTime.Sub(startDataTime).Hours())
  39. betweenDay := betweenHour / 24
  40. // 如果相差一天,那么过滤
  41. if betweenDay <= 1 {
  42. startEdbInfoData = v
  43. newList = append(newList, &data_manage.EdbDataList{
  44. DataTime: v.DataTime,
  45. Value: v.Value,
  46. })
  47. continue
  48. }
  49. // 生成线性方程式
  50. var a, b float64
  51. {
  52. coordinateData := make([]utils.Coordinate, 0)
  53. tmpCoordinate1 := utils.Coordinate{
  54. X: 1,
  55. Y: startEdbInfoData.Value,
  56. }
  57. coordinateData = append(coordinateData, tmpCoordinate1)
  58. tmpCoordinate2 := utils.Coordinate{
  59. X: float64(betweenDay) + 1,
  60. Y: v.Value,
  61. }
  62. coordinateData = append(coordinateData, tmpCoordinate2)
  63. a, b = utils.GetLinearResult(coordinateData)
  64. if math.IsNaN(a) || math.IsNaN(b) {
  65. err = fmt.Errorf("线性方程公式生成失败")
  66. return
  67. }
  68. }
  69. // 生成对应的值
  70. {
  71. for i := 1; i < betweenDay; i++ {
  72. tmpDataTime := startDataTime.AddDate(0, 0, i)
  73. aDecimal := decimal.NewFromFloat(a)
  74. xDecimal := decimal.NewFromInt(int64(i) + 1)
  75. bDecimal := decimal.NewFromFloat(b)
  76. val, _ := aDecimal.Mul(xDecimal).Add(bDecimal).Round(4).Float64()
  77. handleDataMap[tmpDataTime.Format(utils.FormatDate)] = val
  78. newList = append(newList, &data_manage.EdbDataList{
  79. DataTime: tmpDataTime.Format(utils.FormatDate),
  80. Value: val,
  81. })
  82. }
  83. }
  84. // 最后将自己赋值
  85. newList = append(newList, &data_manage.EdbDataList{
  86. EdbDataId: v.EdbDataId,
  87. DataTime: v.DataTime,
  88. Value: v.Value,
  89. })
  90. startEdbInfoData = v
  91. }
  92. return
  93. }
  94. // MoveDataDaysToNewDataList 平移指标数据生成新的数据序列
  95. func MoveDataDaysToNewDataList(dataList []*data_manage.EdbDataList, moveDay int) (newDataList []data_manage.EdbDataList, dateDataMap map[string]float64) {
  96. dateMap := make(map[time.Time]float64)
  97. var minDate, maxDate time.Time
  98. dateDataMap = make(map[string]float64)
  99. for _, v := range dataList {
  100. currDate, _ := time.ParseInLocation(utils.FormatDate, v.DataTime, time.Local)
  101. if minDate.IsZero() || currDate.Before(minDate) {
  102. minDate = currDate
  103. }
  104. if maxDate.IsZero() || currDate.After(maxDate) {
  105. maxDate = currDate
  106. }
  107. dateMap[currDate] = v.Value
  108. }
  109. // 处理领先、滞后数据
  110. newDateMap := make(map[time.Time]float64)
  111. for currDate, value := range dateMap {
  112. newDate := currDate.AddDate(0, 0, moveDay)
  113. newDateMap[newDate] = value
  114. }
  115. minDate = minDate.AddDate(0, 0, moveDay)
  116. maxDate = maxDate.AddDate(0, 0, moveDay)
  117. // 获取日期相差日
  118. dayNum := utils.GetTimeSubDay(minDate, maxDate)
  119. for i := 0; i <= dayNum; i++ {
  120. currDate := minDate.AddDate(0, 0, i)
  121. tmpValue, ok := newDateMap[currDate]
  122. if !ok {
  123. //找不到数据,那么就用前面的数据吧
  124. if len(newDataList)-1 < 0 {
  125. tmpValue = 0
  126. } else {
  127. tmpValue = newDataList[len(newDataList)-1].Value
  128. }
  129. }
  130. tmpData := data_manage.EdbDataList{
  131. DataTime: currDate.Format(utils.FormatDate),
  132. Value: tmpValue,
  133. }
  134. dateDataMap[tmpData.DataTime] = tmpData.Value
  135. newDataList = append(newDataList, tmpData)
  136. }
  137. return
  138. }
  139. // GetChartEdbInfoFormat 相关性图表-获取指标信息
  140. func GetChartEdbInfoFormat(chartInfoId int, edbInfoMappingA, edbInfoMappingB *data_manage.ChartEdbInfoMapping) (edbList []*data_manage.ChartEdbInfoMapping, err error) {
  141. edbList = make([]*data_manage.ChartEdbInfoMapping, 0)
  142. if edbInfoMappingA == nil || edbInfoMappingB == nil {
  143. err = fmt.Errorf("指标信息有误")
  144. return
  145. }
  146. edbInfoMappingA.FrequencyEn = data.GetFrequencyEn(edbInfoMappingA.Frequency)
  147. if edbInfoMappingA.Unit == `无` {
  148. edbInfoMappingA.Unit = ``
  149. }
  150. if edbInfoMappingB.Unit == `无` {
  151. edbInfoMappingB.Unit = ``
  152. }
  153. if chartInfoId <= 0 {
  154. edbInfoMappingA.IsAxis = 1
  155. edbInfoMappingA.LeadValue = 0
  156. edbInfoMappingA.LeadUnit = ""
  157. edbInfoMappingA.ChartEdbMappingId = 0
  158. edbInfoMappingA.ChartInfoId = 0
  159. edbInfoMappingA.IsOrder = false
  160. edbInfoMappingA.EdbInfoType = 1
  161. edbInfoMappingA.ChartStyle = ""
  162. edbInfoMappingA.ChartColor = ""
  163. edbInfoMappingA.ChartWidth = 0
  164. edbInfoMappingB.IsAxis = 1
  165. edbInfoMappingB.LeadValue = 0
  166. edbInfoMappingB.LeadUnit = ""
  167. edbInfoMappingB.ChartEdbMappingId = 0
  168. edbInfoMappingB.ChartInfoId = 0
  169. edbInfoMappingB.IsOrder = false
  170. edbInfoMappingB.EdbInfoType = 1
  171. edbInfoMappingB.ChartStyle = ""
  172. edbInfoMappingB.ChartColor = ""
  173. edbInfoMappingB.ChartWidth = 0
  174. } else {
  175. edbInfoMappingA.LeadUnitEn = data.GetLeadUnitEn(edbInfoMappingA.LeadUnit)
  176. edbInfoMappingB.LeadUnitEn = data.GetLeadUnitEn(edbInfoMappingB.LeadUnit)
  177. }
  178. edbList = append(edbList, edbInfoMappingA, edbInfoMappingB)
  179. return
  180. }
  181. // GetChartDataByEdbInfo 相关性图表-根据指标信息获取x轴和y轴
  182. func GetChartDataByEdbInfo(edbInfoMappingA, edbInfoMappingB *data_manage.ChartEdbInfoMapping, leadValue int, leadUnit, startDate, endDate, extraConfig string) (xEdbIdValue []int, yDataList []data_manage.YData, err error) {
  183. xData := make([]int, 0)
  184. yData := make([]float64, 0)
  185. if leadValue == 0 {
  186. xData = append(xData, 0)
  187. }
  188. if leadValue > 0 {
  189. leadMin := 0 - leadValue
  190. xLen := 2*leadValue + 1
  191. for i := 0; i < xLen; i++ {
  192. n := leadMin + i
  193. xData = append(xData, n)
  194. }
  195. }
  196. // 计算窗口,不包含第一天
  197. startDateTime, _ := time.ParseInLocation(utils.FormatDate, startDate, time.Local)
  198. startDate = startDateTime.AddDate(0, 0, 1).Format(utils.FormatDate)
  199. //// 2023-03-02 时间序列始终以指标B为基准, 始终是A进行平移
  200. //baseEdbInfo := edbInfoMappingB
  201. //changeEdbInfo := edbInfoMappingA
  202. // 2023-03-17 时间序列始终以指标A为基准, 始终是B进行平移
  203. baseEdbInfo := edbInfoMappingA
  204. changeEdbInfo := edbInfoMappingB
  205. // 获取时间基准指标在时间区间内的值
  206. aDataList := make([]*data_manage.EdbDataList, 0)
  207. switch baseEdbInfo.EdbInfoCategoryType {
  208. case 0:
  209. aDataList, err = data_manage.GetEdbDataList(baseEdbInfo.Source, baseEdbInfo.SubSource, baseEdbInfo.EdbInfoId, startDate, endDate)
  210. case 1:
  211. _, aDataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(baseEdbInfo.EdbInfoId, startDate, endDate, false)
  212. default:
  213. err = errors.New("指标base类型异常")
  214. return
  215. }
  216. // 获取变频指标所有日期的值, 插值法完善数据
  217. bDataList := make([]*data_manage.EdbDataList, 0)
  218. switch changeEdbInfo.EdbInfoCategoryType {
  219. case 0:
  220. bDataList, err = data_manage.GetEdbDataList(changeEdbInfo.Source, changeEdbInfo.SubSource, changeEdbInfo.EdbInfoId, "", "")
  221. case 1:
  222. _, bDataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(changeEdbInfo.EdbInfoId, "", "", false)
  223. default:
  224. err = errors.New("指标change类型异常")
  225. return
  226. }
  227. //changeDataMap := make(map[string]float64)
  228. //newChangeDataList, e := HandleDataByLinearRegression(bDataList, changeDataMap)
  229. //if e != nil {
  230. // err = fmt.Errorf("获取变频指标插值法Map失败, Err: %s", e.Error())
  231. // return
  232. //}
  233. // 2023-03-17 时间序列始终以指标A为基准, 始终是B进行平移
  234. baseDataList := make([]*data_manage.EdbDataList, 0)
  235. baseDataMap := make(map[string]float64)
  236. changeDataList := make([]*data_manage.EdbDataList, 0)
  237. changeDataMap := make(map[string]float64)
  238. // 先把低频指标升频为高频
  239. {
  240. frequencyIntMap := map[string]int{
  241. "日度": 1,
  242. "周度": 2,
  243. "旬度": 3,
  244. "月度": 4,
  245. "季度": 5,
  246. "年度": 6,
  247. }
  248. // 如果A指标是高频,那么就需要对B指标进行升频
  249. if frequencyIntMap[edbInfoMappingA.Frequency] < frequencyIntMap[edbInfoMappingB.Frequency] {
  250. tmpNewChangeDataList, e := HandleDataByLinearRegression(bDataList, changeDataMap)
  251. if e != nil {
  252. err = fmt.Errorf("获取变频指标插值法Map失败, Err: %s", e.Error())
  253. return
  254. }
  255. changeDataList = tmpNewChangeDataList
  256. baseDataList = aDataList
  257. for _, v := range baseDataList {
  258. baseDataMap[v.DataTime] = v.Value
  259. }
  260. } else if frequencyIntMap[edbInfoMappingA.Frequency] > frequencyIntMap[edbInfoMappingB.Frequency] {
  261. // 如果B指标是高频,那么就需要对A指标进行升频
  262. tmpNewChangeDataList, e := HandleDataByLinearRegression(aDataList, baseDataMap)
  263. if e != nil {
  264. err = fmt.Errorf("获取变频指标插值法Map失败, Err: %s", e.Error())
  265. return
  266. }
  267. baseDataList = tmpNewChangeDataList
  268. changeDataList = bDataList
  269. for _, v := range changeDataList {
  270. changeDataMap[v.DataTime] = v.Value
  271. }
  272. } else {
  273. baseDataList = aDataList
  274. for _, v := range baseDataList {
  275. baseDataMap[v.DataTime] = v.Value
  276. }
  277. changeDataList = bDataList
  278. for _, v := range changeDataList {
  279. changeDataMap[v.DataTime] = v.Value
  280. }
  281. }
  282. }
  283. // 计算不领先也不滞后时的相关系数
  284. baseCalculateData := make([]float64, 0)
  285. baseDataTimeArr := make([]string, 0)
  286. for i := range baseDataList {
  287. baseDataTimeArr = append(baseDataTimeArr, baseDataList[i].DataTime)
  288. baseCalculateData = append(baseCalculateData, baseDataList[i].Value)
  289. }
  290. //zeroBaseData := make([]float64, 0)
  291. //zeroCalculateData := make([]float64, 0)
  292. //for i := range baseDataTimeArr {
  293. // tmpBaseVal, ok1 := baseDataMap[baseDataTimeArr[i]]
  294. // tmpCalculateVal, ok2 := changeDataMap[baseDataTimeArr[i]]
  295. // if ok1 && ok2 {
  296. // zeroBaseData = append(zeroBaseData, tmpBaseVal)
  297. // zeroCalculateData = append(zeroCalculateData, tmpCalculateVal)
  298. // }
  299. //}
  300. //if len(zeroBaseData) != len(zeroCalculateData) {
  301. // err = fmt.Errorf("相关系数两组序列元素数不一致, %d-%d", len(baseCalculateData), len(zeroCalculateData))
  302. // return
  303. //}
  304. //zeroRatio := utils.CalculateCorrelationByIntArr(zeroBaseData, zeroCalculateData)
  305. //if leadValue == 0 {
  306. // yData = append(yData, zeroRatio)
  307. //}
  308. // 计算领先/滞后N期
  309. if leadValue > 0 {
  310. // 平移变频指标领先/滞后的日期(单位天)
  311. moveUnitDays := utils.FrequencyDaysMap[leadUnit]
  312. for i := range xData {
  313. //if xData[i] == 0 {
  314. // yData = append(yData, zeroRatio)
  315. // continue
  316. //}
  317. xCalculateData := make([]float64, 0)
  318. yCalculateData := make([]float64, 0)
  319. // 平移指定天数
  320. mDays := int(moveUnitDays) * xData[i]
  321. _, dMap := MoveDataDaysToNewDataList(changeDataList, mDays)
  322. // 取出对应的基准日期的值
  323. for i2 := range baseDataTimeArr {
  324. tmpDate := baseDataTimeArr[i2]
  325. if yVal, ok := dMap[tmpDate]; ok {
  326. xCalculateData = append(xCalculateData, baseCalculateData[i2])
  327. yCalculateData = append(yCalculateData, yVal)
  328. }
  329. }
  330. if len(yCalculateData) <= 0 {
  331. //err = fmt.Errorf("领先滞后相关系数两组序列元素数不一致, %d-%d", len(baseCalculateData), len(yCalculateData))
  332. //return
  333. // 领先滞后后,没有可以计算的数据了
  334. continue
  335. }
  336. // 公式计算出领先/滞后频度对应点的相关性系数
  337. ratio := utils.CalculateCorrelationByIntArr(xCalculateData, yCalculateData)
  338. yData = append(yData, ratio)
  339. }
  340. }
  341. // 图例
  342. var extra data_manage.CorrelationChartInfoExtraConfig
  343. legend := new(data_manage.CorrelationChartLegend)
  344. if extraConfig != "" {
  345. if e := json.Unmarshal([]byte(extraConfig), &extra); e != nil {
  346. err = fmt.Errorf("图例解析异常, err: %v", e)
  347. return
  348. }
  349. if len(extra.LegendConfig) > 0 {
  350. legend = extra.LegendConfig[0]
  351. }
  352. }
  353. xEdbIdValue = xData
  354. yDataList = make([]data_manage.YData, 0)
  355. yDate := "0000-00-00"
  356. var y data_manage.YData
  357. y.Date = yDate
  358. y.Value = yData
  359. if legend != nil {
  360. y.Name = legend.LegendName
  361. y.Color = legend.Color
  362. }
  363. yDataList = append(yDataList, y)
  364. return
  365. }
  366. // RollingCorrelationChartDataResp 滚动相关性图表数据
  367. type RollingCorrelationChartDataResp struct {
  368. MaxData float64
  369. MinData float64
  370. LatestDate string `description:"真实数据的最后日期"`
  371. EdbInfoCategoryType int
  372. ChartColor string
  373. ChartStyle string
  374. PredictChartColor string
  375. ChartType int
  376. ChartWidth int
  377. EdbName string
  378. EdbNameEn string
  379. Unit string
  380. UnitEn string
  381. IsAxis int
  382. DataList []data_manage.EdbDataList
  383. }
  384. // GetRollingCorrelationChartDataByEdbInfo 滚动相关性计算
  385. func GetRollingCorrelationChartDataByEdbInfo(edbInfoMappingA, edbInfoMappingB *data_manage.ChartEdbInfoMapping, leadValue int, leadUnit string, calculateValue int, calculateUnit string, startDate, endDate, chartName, chartNameEn string) (dataResp RollingCorrelationChartDataResp, err error) {
  386. dataResp = RollingCorrelationChartDataResp{
  387. DataList: make([]data_manage.EdbDataList, 0),
  388. MaxData: 0,
  389. MinData: 0,
  390. ChartColor: "#00f",
  391. ChartStyle: `spline`,
  392. PredictChartColor: `#00f`,
  393. ChartType: 0,
  394. ChartWidth: 3,
  395. EdbName: chartName,
  396. EdbNameEn: chartNameEn,
  397. IsAxis: 1,
  398. }
  399. dataList := make([]data_manage.EdbDataList, 0)
  400. // 计算窗口,不包含第一天
  401. startDateTime, _ := time.ParseInLocation(utils.FormatDate, startDate, time.Local)
  402. startDate = startDateTime.AddDate(0, 0, 1).Format(utils.FormatDate)
  403. baseEdbInfo := edbInfoMappingA
  404. changeEdbInfo := edbInfoMappingB
  405. // 获取时间基准指标在时间区间内的值
  406. aDataList := make([]*data_manage.EdbDataList, 0)
  407. switch baseEdbInfo.EdbInfoCategoryType {
  408. case 0:
  409. aDataList, err = data_manage.GetEdbDataList(baseEdbInfo.Source, baseEdbInfo.SubSource, baseEdbInfo.EdbInfoId, startDate, endDate)
  410. case 1:
  411. _, aDataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(baseEdbInfo.EdbInfoId, startDate, endDate, true)
  412. default:
  413. err = errors.New("指标base类型异常")
  414. return
  415. }
  416. // 获取变频指标所有日期的值, 插值法完善数据
  417. bDataList := make([]*data_manage.EdbDataList, 0)
  418. switch changeEdbInfo.EdbInfoCategoryType {
  419. case 0:
  420. bDataList, err = data_manage.GetEdbDataList(changeEdbInfo.Source, changeEdbInfo.SubSource, changeEdbInfo.EdbInfoId, "", "")
  421. case 1:
  422. _, bDataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(changeEdbInfo.EdbInfoId, "", "", false)
  423. default:
  424. err = errors.New("指标change类型异常")
  425. return
  426. }
  427. // 数据平移变频指标领先/滞后的日期(单位天)
  428. // 2023-03-17 时间序列始终以指标A为基准, 始终是B进行平移
  429. //baseDataList := make([]*data_manage.EdbDataList, 0)
  430. baseDataMap := make(map[string]float64)
  431. changeDataList := make([]*data_manage.EdbDataList, 0)
  432. changeDataMap := make(map[string]float64)
  433. // A指标不管三七二十一,先变个频再说
  434. {
  435. _, e := HandleDataByLinearRegression(aDataList, baseDataMap)
  436. if e != nil {
  437. err = fmt.Errorf("获取变频指标插值法Map失败, Err: %s", e.Error())
  438. return
  439. }
  440. //baseDataList = tmpNewChangeDataList
  441. }
  442. // B指标不管三七二十一,先变个频再说
  443. {
  444. tmpNewChangeDataList, e := HandleDataByLinearRegression(bDataList, changeDataMap)
  445. if e != nil {
  446. err = fmt.Errorf("获取变频指标插值法Map失败, Err: %s", e.Error())
  447. return
  448. }
  449. changeDataList = tmpNewChangeDataList
  450. // 平移下日期
  451. moveUnitDays := utils.FrequencyDaysMap[leadUnit]
  452. _, changeDataMap = MoveDataDaysToNewDataList(changeDataList, leadValue*moveUnitDays)
  453. }
  454. // 计算计算时,需要多少个日期内数据
  455. calculateDay := utils.FrequencyDaysMap[calculateUnit] * calculateValue
  456. // 计算 每个日期的相关性值
  457. {
  458. startDateTime, _ := time.ParseInLocation(utils.FormatDate, startDate, time.Local)
  459. if endDate == `` {
  460. endDate = baseEdbInfo.EndDate
  461. }
  462. endDateTime, _ := time.ParseInLocation(utils.FormatDate, endDate, time.Local)
  463. endDateTime = endDateTime.AddDate(0, 0, -(calculateDay - 1))
  464. // 是否开始第一条数据
  465. var isStart, isNotFirst bool
  466. for currDay := startDateTime; !currDay.After(endDateTime); currDay = currDay.AddDate(0, 0, 1) {
  467. yCalculateData := make([]float64, 0)
  468. baseCalculateData := make([]float64, 0)
  469. // 取出对应的基准日期的值
  470. for i := 0; i < calculateDay; i++ {
  471. iDay := currDay.AddDate(0, 0, i).Format(utils.FormatDate)
  472. tmpBaseValue, ok1 := baseDataMap[iDay]
  473. tmpChangeValue, ok2 := changeDataMap[iDay]
  474. if ok1 && ok2 {
  475. baseCalculateData = append(baseCalculateData, tmpBaseValue)
  476. yCalculateData = append(yCalculateData, tmpChangeValue)
  477. } else {
  478. continue
  479. }
  480. }
  481. // 公式计算出领先/滞后频度对应点的相关性系数
  482. var ratio float64
  483. if len(baseCalculateData) > 0 {
  484. ratio = utils.CalculateCorrelationByIntArr(baseCalculateData, yCalculateData)
  485. } else {
  486. // 没有数据的话,那就不返回
  487. continue
  488. }
  489. // 过滤前面都是0的数据
  490. {
  491. if ratio != 0 {
  492. isStart = true
  493. }
  494. if !isStart {
  495. continue
  496. }
  497. }
  498. dataTime := currDay.AddDate(0, 0, calculateDay-1)
  499. dataList = append(dataList, data_manage.EdbDataList{
  500. //EdbDataId: 0,
  501. EdbInfoId: 0,
  502. DataTime: dataTime.Format(utils.FormatDate),
  503. DataTimestamp: dataTime.UnixNano() / 1e6,
  504. Value: ratio,
  505. })
  506. if !isNotFirst {
  507. dataResp.MinData = ratio
  508. dataResp.MaxData = ratio
  509. isNotFirst = true
  510. }
  511. if dataResp.MinData > ratio {
  512. dataResp.MinData = ratio
  513. }
  514. if dataResp.MaxData < ratio {
  515. dataResp.MaxData = ratio
  516. }
  517. }
  518. dataResp.DataList = dataList
  519. }
  520. return
  521. }
  522. // ChartInfoRefresh 图表刷新
  523. func ChartInfoRefresh(chartInfoId int) (err error) {
  524. var errMsg string
  525. defer func() {
  526. if err != nil {
  527. go alarm_msg.SendAlarmMsg("CorrelationChartInfoRefresh: "+errMsg, 3)
  528. }
  529. }()
  530. correlationChart := new(data_manage.ChartInfoCorrelation)
  531. if err = correlationChart.GetItemById(chartInfoId); err != nil {
  532. errMsg = "获取相关性图表失败, Err: " + err.Error()
  533. return
  534. }
  535. // 批量刷新ETA指标
  536. err, _ = data.EdbInfoRefreshAllFromBaseV3([]int{correlationChart.EdbInfoIdFirst, correlationChart.EdbInfoIdSecond}, false, true, false)
  537. if err != nil {
  538. return
  539. }
  540. // 重新生成数据并更新
  541. edbInfoMappingA, err := data_manage.GetChartEdbMappingByEdbInfoId(correlationChart.EdbInfoIdFirst)
  542. if err != nil {
  543. errMsg = "获取相关性图表, A指标mapping信息失败, Err:" + err.Error()
  544. return
  545. }
  546. // TODO:多因子刷新
  547. edbInfoMappingB, err := data_manage.GetChartEdbMappingByEdbInfoId(correlationChart.EdbInfoIdSecond)
  548. if err != nil {
  549. errMsg = "获取相关性图表, B指标mapping信息失败, Err:" + err.Error()
  550. return
  551. }
  552. periodData, correlationData, err := GetChartDataByEdbInfo(edbInfoMappingA, edbInfoMappingB, correlationChart.LeadValue, correlationChart.LeadUnit, correlationChart.StartDate.Format(utils.FormatDate), correlationChart.EndDate.Format(utils.FormatDate), "")
  553. if err != nil {
  554. errMsg = "获取相关性图表, 图表计算值失败, Err:" + err.Error()
  555. return
  556. }
  557. periodDataByte, err := json.Marshal(periodData)
  558. if err != nil {
  559. errMsg = "相关性图表, X轴信息有误, Err:" + err.Error()
  560. return
  561. }
  562. correlationDataByte, err := json.Marshal(correlationData[0].Value)
  563. if err != nil {
  564. errMsg = "相关性图表, Y轴信息有误, Err:" + err.Error()
  565. return
  566. }
  567. correlationChart.PeriodData = string(periodDataByte)
  568. correlationChart.CorrelationData = string(correlationDataByte)
  569. correlationChart.ModifyTime = time.Now().Local()
  570. correlationUpdateCols := []string{"PeriodData", "CorrelationData", "ModifyTime"}
  571. if err = correlationChart.Update(correlationUpdateCols); err != nil {
  572. errMsg = "更新相关性图表失败, Err:" + err.Error()
  573. return
  574. }
  575. return
  576. }
  577. // GetChartAndCorrelationInfo 获取图表信息和相关信息信息
  578. func GetChartAndCorrelationInfo(chartInfoId int) (chartInfo *data_manage.ChartInfo, correlationInfo *data_manage.ChartInfoCorrelation, tips string, err error) {
  579. item, e := data_manage.GetChartInfoById(chartInfoId)
  580. if e != nil {
  581. if e.Error() == utils.ErrNoRow() {
  582. tips = "图表已被删除, 请刷新页面"
  583. err = fmt.Errorf("图表已被删除, 请刷新页面")
  584. return
  585. }
  586. err = fmt.Errorf("获取图表信息失败, Err: %s", e.Error())
  587. return
  588. }
  589. if item.Source != utils.CHART_SOURCE_CORRELATION {
  590. tips = "该图不是相关性图表"
  591. err = fmt.Errorf("该图不是相关性图表")
  592. return
  593. }
  594. chartInfo = item
  595. correlationInfo = new(data_manage.ChartInfoCorrelation)
  596. if e = correlationInfo.GetItemById(chartInfo.ChartInfoId); e != nil {
  597. err = fmt.Errorf("获取图表相关性信息失败, Err: %s", e.Error())
  598. return
  599. }
  600. return
  601. }
  602. // AddChartInfo 添加图表
  603. func AddChartInfo(req data_manage.AddChartInfoReq, source int, sysUser *system.Admin, lang string) (chartInfo *data_manage.ChartInfo, err error, errMsg string, isSendEmail bool) {
  604. isSendEmail = true
  605. req.ChartName = strings.Trim(req.ChartName, " ")
  606. if req.ChartName == "" {
  607. errMsg = "请填写图表名称!"
  608. err = errors.New(errMsg)
  609. isSendEmail = false
  610. return
  611. }
  612. if req.ChartClassifyId <= 0 {
  613. errMsg = "分类参数错误!"
  614. err = errors.New(errMsg)
  615. isSendEmail = false
  616. return
  617. }
  618. // 相关性图表配置
  619. if req.CorrelationChartInfo.LeadValue == 0 && source == utils.CHART_SOURCE_CORRELATION {
  620. errMsg = "请输入领先期数"
  621. err = errors.New(errMsg)
  622. isSendEmail = false
  623. return
  624. }
  625. if req.CorrelationChartInfo.LeadUnit == "" {
  626. errMsg = "请填写领先单位"
  627. err = errors.New(errMsg)
  628. isSendEmail = false
  629. return
  630. }
  631. //if req.CorrelationChartInfo.StartDate == "" || req.CorrelationChartInfo.EndDate == "" {
  632. // errMsg = "请填写开始结束日期"
  633. // err = errors.New(errMsg)
  634. // isSendEmail = false
  635. // return
  636. //}
  637. //startDate, e := time.Parse(utils.FormatDate, req.CorrelationChartInfo.StartDate)
  638. //if e != nil {
  639. // errMsg = "开始日期格式有误"
  640. // err = errors.New(errMsg)
  641. // isSendEmail = false
  642. // return
  643. //}
  644. //endDate, e := time.Parse(utils.FormatDate, req.CorrelationChartInfo.EndDate)
  645. //if e != nil {
  646. // errMsg = "结束日期格式有误"
  647. // err = errors.New(errMsg)
  648. // isSendEmail = false
  649. // return
  650. //}
  651. if len(req.CorrelationChartInfo.EdbInfoIdList) != 2 {
  652. errMsg = "请选择AB指标"
  653. err = errors.New(errMsg)
  654. isSendEmail = false
  655. return
  656. }
  657. chartClassify, err := data_manage.GetChartClassifyById(req.ChartClassifyId)
  658. if err != nil {
  659. if err.Error() == utils.ErrNoRow() {
  660. errMsg = "分类不存在"
  661. err = errors.New(errMsg)
  662. isSendEmail = false
  663. return
  664. }
  665. errMsg = "获取分类信息失败"
  666. err = errors.New("获取分类信息失败,Err:" + err.Error())
  667. return
  668. }
  669. if chartClassify == nil {
  670. errMsg = "分类不存在"
  671. err = errors.New(errMsg)
  672. isSendEmail = false
  673. return
  674. }
  675. var edbInfoIdArr []int
  676. for _, v := range req.CorrelationChartInfo.EdbInfoIdList {
  677. edbInfoId := v.EdbInfoId
  678. edbInfo, tmpErr := data_manage.GetEdbInfoById(edbInfoId)
  679. if tmpErr != nil {
  680. if tmpErr.Error() == utils.ErrNoRow() {
  681. errMsg = "指标不存在!"
  682. err = errors.New("指标不存在,edbInfoId:" + strconv.Itoa(edbInfoId))
  683. return
  684. } else {
  685. errMsg = "获取指标信息失败!"
  686. err = errors.New("获取图表的指标信息失败,Err:" + tmpErr.Error())
  687. return
  688. }
  689. }
  690. if edbInfo == nil {
  691. errMsg = "指标已被删除,请重新选择!"
  692. err = errors.New("指标不存在,ChartInfoId:" + strconv.Itoa(edbInfoId))
  693. return
  694. } else {
  695. if edbInfo.EdbInfoId <= 0 {
  696. errMsg = "指标已被删除,请重新选择!"
  697. err = errors.New("指标不存在,ChartInfoId:" + strconv.Itoa(edbInfoId))
  698. return
  699. }
  700. }
  701. edbInfoIdArr = append(edbInfoIdArr, edbInfoId)
  702. edbInfo.EdbNameSource = edbInfo.EdbName
  703. }
  704. sort.Ints(edbInfoIdArr)
  705. var edbInfoIdArrStr []string
  706. for _, v := range edbInfoIdArr {
  707. edbInfoIdArrStr = append(edbInfoIdArrStr, strconv.Itoa(v))
  708. }
  709. edbInfoIdStr := strings.Join(edbInfoIdArrStr, ",")
  710. var chartInfoId int
  711. // 判断图表是否存在
  712. {
  713. var condition string
  714. var pars []interface{}
  715. switch lang {
  716. case utils.EnLangVersion:
  717. condition += " AND chart_name_en = ? AND source = ? "
  718. default:
  719. condition += " AND chart_name=? AND source = ? "
  720. }
  721. pars = append(pars, req.ChartName, source)
  722. count, tmpErr := data_manage.GetChartInfoCountByCondition(condition, pars)
  723. if tmpErr != nil {
  724. errMsg = "判断图表名称是否存在失败"
  725. err = errors.New("判断图表名称是否存在失败,Err:" + tmpErr.Error())
  726. return
  727. }
  728. if count > 0 {
  729. errMsg = "图表已存在,请重新填写"
  730. err = errors.New(errMsg)
  731. isSendEmail = false
  732. return
  733. }
  734. }
  735. disableVal := data.CheckIsDisableChart(edbInfoIdArr)
  736. chartInfo = new(data_manage.ChartInfo)
  737. chartInfo.ChartName = req.ChartName
  738. chartInfo.ChartNameEn = req.ChartName
  739. chartInfo.EdbInfoIds = edbInfoIdStr
  740. chartInfo.ChartClassifyId = req.ChartClassifyId
  741. chartInfo.SysUserId = sysUser.AdminId
  742. chartInfo.SysUserRealName = sysUser.RealName
  743. chartInfo.CreateTime = time.Now()
  744. chartInfo.ModifyTime = time.Now()
  745. chartInfo.IsSetName = 0
  746. timestamp := strconv.FormatInt(time.Now().UnixNano(), 10)
  747. chartInfo.UniqueCode = utils.MD5(utils.CHART_PREFIX + "_" + timestamp)
  748. chartInfo.ChartType = 9 // 相关性图
  749. chartInfo.Calendar = "公历"
  750. chartInfo.DateType = 6
  751. chartInfo.StartDate = req.StartDate
  752. chartInfo.EndDate = req.EndDate
  753. chartInfo.SeasonStartDate = req.StartDate
  754. chartInfo.SeasonEndDate = req.EndDate
  755. chartInfo.LeftMin = req.LeftMin
  756. chartInfo.LeftMax = req.LeftMax
  757. chartInfo.RightMin = req.RightMin
  758. chartInfo.RightMax = req.RightMax
  759. chartInfo.Disabled = disableVal
  760. chartInfo.Source = source
  761. chartInfo.ChartThemeId = req.ChartThemeId
  762. chartInfo.SourcesFrom = req.SourcesFrom
  763. chartInfo.Instructions = req.Instructions
  764. chartInfo.MarkersLines = req.MarkersLines
  765. chartInfo.MarkersAreas = req.MarkersAreas
  766. if req.ExtraConfig != "" {
  767. chartInfo.ExtraConfig = req.ExtraConfig
  768. }
  769. // 指标信息
  770. mapList := make([]*data_manage.ChartEdbMapping, 0)
  771. for _, v := range req.CorrelationChartInfo.EdbInfoIdList {
  772. mapItem := new(data_manage.ChartEdbMapping)
  773. mapItem.EdbInfoId = v.EdbInfoId
  774. mapItem.CreateTime = time.Now()
  775. mapItem.ModifyTime = time.Now()
  776. edbTimestamp := strconv.FormatInt(time.Now().UnixNano(), 10)
  777. mapItem.UniqueCode = utils.MD5(utils.CHART_PREFIX + "_" + edbTimestamp + "_" + strconv.Itoa(v.EdbInfoId))
  778. mapItem.IsOrder = true
  779. mapItem.IsAxis = 1
  780. mapItem.EdbInfoType = 1
  781. mapItem.Source = utils.CHART_SOURCE_CORRELATION
  782. mapList = append(mapList, mapItem)
  783. }
  784. // 相关性图表扩展信息
  785. correlationChart := new(data_manage.ChartInfoCorrelation)
  786. correlationChart.LeadValue = req.CorrelationChartInfo.LeadValue
  787. correlationChart.LeadUnit = req.CorrelationChartInfo.LeadUnit
  788. correlationChart.CalculateValue = req.CorrelationChartInfo.CalculateValue
  789. correlationChart.CalculateUnit = req.CorrelationChartInfo.CalculateUnit
  790. correlationChart.BaseCalculateValue = req.CorrelationChartInfo.BaseCalculateValue
  791. correlationChart.BaseCalculateUnit = req.CorrelationChartInfo.BaseCalculateUnit
  792. // 滚动相关性会有日期等信息
  793. if source == utils.CHART_SOURCE_ROLLING_CORRELATION {
  794. correlationChart.DateType = req.CorrelationChartInfo.DateType
  795. if req.CorrelationChartInfo.StartDate != `` {
  796. startDateTime, tmpErr := time.ParseInLocation(utils.FormatDate, req.CorrelationChartInfo.StartDate, time.Local)
  797. if tmpErr != nil {
  798. err = tmpErr
  799. return
  800. }
  801. correlationChart.StartDate = startDateTime
  802. }
  803. if req.CorrelationChartInfo.EndDate != `` {
  804. endDateTime, tmpErr := time.ParseInLocation(utils.FormatDate, req.CorrelationChartInfo.EndDate, time.Local)
  805. if tmpErr != nil {
  806. err = tmpErr
  807. return
  808. }
  809. correlationChart.EndDate = endDateTime
  810. }
  811. }
  812. correlationChart.EdbInfoIdFirst = req.CorrelationChartInfo.EdbInfoIdList[0].EdbInfoId
  813. correlationChart.EdbInfoIdSecond = req.CorrelationChartInfo.EdbInfoIdList[1].EdbInfoId
  814. correlationChart.CreateTime = time.Now().Local()
  815. correlationChart.ModifyTime = time.Now().Local()
  816. //// 生成图表x轴y轴数据
  817. //edbInfoMappingA, e := data_manage.GetChartEdbMappingByEdbInfoId(req.CorrelationChartInfo.EdbInfoIdList[0].EdbInfoId)
  818. //if e != nil {
  819. // errMsg = "获取失败"
  820. // err = errors.New("获取相关性图表, A指标mapping信息失败, Err:" + e.Error())
  821. // return
  822. //}
  823. //edbInfoMappingB, e := data_manage.GetChartEdbMappingByEdbInfoId(req.CorrelationChartInfo.EdbInfoIdList[1].EdbInfoId)
  824. //if e != nil {
  825. // errMsg = "获取失败"
  826. // err = errors.New("获取相关性图表, B指标mapping信息失败, Err:" + e.Error())
  827. // return
  828. //}
  829. //periodData, correlationData, e := GetChartDataByEdbInfo(edbInfoMappingA, edbInfoMappingB, req.CorrelationChartInfo.LeadValue, req.CorrelationChartInfo.LeadUnit, req.CorrelationChartInfo.StartDate, req.CorrelationChartInfo.EndDate)
  830. //if e != nil {
  831. // errMsg = "获取失败"
  832. // err = errors.New("获取相关性图表, 图表计算值失败, Err:" + e.Error())
  833. // return
  834. //}
  835. //periodDataByte, e := json.Marshal(periodData)
  836. //if e != nil {
  837. // errMsg = "获取失败"
  838. // err = errors.New("相关性图表, X轴信息有误, Err:" + e.Error())
  839. // return
  840. //}
  841. //correlationDataByte, e := json.Marshal(correlationData[0].Value)
  842. //if e != nil {
  843. // errMsg = "获取失败"
  844. // err = errors.New("相关性图表, Y轴信息有误, Err:" + e.Error())
  845. // return
  846. //}
  847. //correlationChart.PeriodData = string(periodDataByte)
  848. //correlationChart.CorrelationData = string(correlationDataByte)
  849. // 新增图表和指标mapping
  850. chartInfoId, e := data_manage.CreateCorrelationChartAndEdb(chartInfo, mapList, correlationChart)
  851. if e != nil {
  852. errMsg = "操作失败"
  853. err = errors.New("新增相关性图表失败, Err: " + e.Error())
  854. return
  855. }
  856. //添加es数据
  857. go data.EsAddOrEditChartInfo(chartInfoId)
  858. return
  859. }
  860. // EditChartInfo 编辑图表
  861. func EditChartInfo(req data_manage.EditChartInfoReq, sysUser *system.Admin, lang string) (chartItem *data_manage.ChartInfo, err error, errMsg string, isSendEmail bool) {
  862. isSendEmail = true
  863. chartItem, err = data_manage.GetChartInfoById(req.ChartInfoId)
  864. if err != nil {
  865. if err.Error() == utils.ErrNoRow() {
  866. errMsg = "图表已被删除,请刷新页面"
  867. err = errors.New(errMsg)
  868. isSendEmail = false
  869. return
  870. }
  871. errMsg = "获取图表信息失败"
  872. err = errors.New("获取图表信息失败,Err:" + err.Error())
  873. return
  874. }
  875. if chartItem.Source != utils.CHART_SOURCE_CORRELATION && chartItem.Source != utils.CHART_SOURCE_ROLLING_CORRELATION {
  876. errMsg = "该图不是相关性图表!"
  877. err = errors.New(errMsg)
  878. isSendEmail = false
  879. return
  880. }
  881. req.ChartName = strings.Trim(req.ChartName, " ")
  882. if req.ChartClassifyId <= 0 {
  883. errMsg = "分类参数错误!"
  884. err = errors.New(errMsg)
  885. isSendEmail = false
  886. return
  887. }
  888. // 相关性图表配置
  889. if req.CorrelationChartInfo.LeadValue == 0 && chartItem.Source == utils.CHART_SOURCE_CORRELATION {
  890. errMsg = "请输入领先期数"
  891. err = errors.New(errMsg)
  892. isSendEmail = false
  893. return
  894. }
  895. if req.CorrelationChartInfo.LeadUnit == "" {
  896. errMsg = "请填写领先单位"
  897. err = errors.New(errMsg)
  898. isSendEmail = false
  899. return
  900. }
  901. //if req.CorrelationChartInfo.StartDate == "" || req.CorrelationChartInfo.EndDate == "" {
  902. // errMsg = "请填写开始结束日期"
  903. // err = errors.New(errMsg)
  904. // isSendEmail = false
  905. // return
  906. //}
  907. startDate, e := time.Parse(utils.FormatDate, req.CorrelationChartInfo.StartDate)
  908. if e != nil {
  909. errMsg = "开始日期格式有误"
  910. err = errors.New(errMsg)
  911. isSendEmail = false
  912. return
  913. }
  914. var endDate time.Time
  915. if req.CorrelationChartInfo.EndDate != `` {
  916. endDate, e = time.Parse(utils.FormatDate, req.CorrelationChartInfo.EndDate)
  917. if e != nil {
  918. errMsg = "结束日期格式有误"
  919. err = errors.New(errMsg)
  920. isSendEmail = false
  921. return
  922. }
  923. }
  924. if len(req.CorrelationChartInfo.EdbInfoIdList) != 2 {
  925. errMsg = "请选择AB指标"
  926. err = errors.New(errMsg)
  927. isSendEmail = false
  928. return
  929. }
  930. chartClassify, err := data_manage.GetChartClassifyById(req.ChartClassifyId)
  931. if err != nil {
  932. if err.Error() == utils.ErrNoRow() {
  933. errMsg = "分类不存在"
  934. err = errors.New(errMsg)
  935. isSendEmail = false
  936. return
  937. }
  938. errMsg = "获取分类信息失败"
  939. err = errors.New("获取分类信息失败,Err:" + err.Error())
  940. return
  941. }
  942. if chartClassify == nil {
  943. errMsg = "分类不存在"
  944. err = errors.New(errMsg)
  945. isSendEmail = false
  946. return
  947. }
  948. // 图表操作权限
  949. ok := data.CheckOpChartPermission(sysUser, chartItem.SysUserId, true)
  950. if !ok {
  951. errMsg = "没有该图表的操作权限"
  952. err = errors.New(errMsg)
  953. isSendEmail = false
  954. return
  955. }
  956. var edbInfoIdArr []int
  957. for _, v := range req.CorrelationChartInfo.EdbInfoIdList {
  958. edbInfoId := v.EdbInfoId
  959. edbInfo, tmpErr := data_manage.GetEdbInfoById(edbInfoId)
  960. if tmpErr != nil {
  961. if tmpErr.Error() == utils.ErrNoRow() {
  962. errMsg = "图表不存在!"
  963. err = errors.New("图表指标不存在,ChartInfoId:" + strconv.Itoa(edbInfoId))
  964. return
  965. } else {
  966. errMsg = "获取图表信息失败!"
  967. err = errors.New("获取图表的指标信息失败,Err:" + tmpErr.Error())
  968. return
  969. }
  970. }
  971. if edbInfo == nil {
  972. errMsg = "指标不存在!"
  973. err = errors.New("指标不存在,ChartInfoId:" + strconv.Itoa(edbInfoId))
  974. return
  975. }
  976. edbInfoIdArr = append(edbInfoIdArr, edbInfoId)
  977. }
  978. sort.Ints(edbInfoIdArr)
  979. var edbInfoIdArrStr []string
  980. for _, v := range edbInfoIdArr {
  981. edbInfoIdArrStr = append(edbInfoIdArrStr, strconv.Itoa(v))
  982. }
  983. edbInfoIdStr := strings.Join(edbInfoIdArrStr, ",")
  984. //判断图表是否存在
  985. {
  986. var condition string
  987. var pars []interface{}
  988. condition += " AND chart_info_id <> ? "
  989. pars = append(pars, req.ChartInfoId)
  990. switch lang {
  991. case utils.EnLangVersion:
  992. condition += " AND chart_name_en = ? AND source = ? "
  993. default:
  994. condition += " AND chart_name=? AND source = ? "
  995. }
  996. pars = append(pars, req.ChartName, chartItem.Source)
  997. count, tmpErr := data_manage.GetChartInfoCountByCondition(condition, pars)
  998. if tmpErr != nil {
  999. errMsg = "判断图表名称是否存在失败"
  1000. err = errors.New("判断图表名称是否存在失败,Err:" + tmpErr.Error())
  1001. return
  1002. }
  1003. if count > 0 {
  1004. errMsg = "图表已存在,请重新填写"
  1005. err = errors.New(errMsg)
  1006. isSendEmail = false
  1007. return
  1008. }
  1009. }
  1010. correlationChart := new(data_manage.ChartInfoCorrelation)
  1011. if e := correlationChart.GetItemById(chartItem.ChartInfoId); e != nil {
  1012. errMsg = "操作失败"
  1013. err = errors.New("图表相关性信息不存在, Err: " + e.Error())
  1014. return
  1015. }
  1016. // 图表启用与否
  1017. disableVal := data.CheckIsDisableChart(edbInfoIdArr)
  1018. // 重新生成图表值, 并修改相关性图表扩展信息
  1019. //edbInfoMappingA, e := data_manage.GetChartEdbMappingByEdbInfoId(req.CorrelationChartInfo.EdbInfoIdList[0].EdbInfoId)
  1020. //if e != nil {
  1021. // errMsg = "获取失败"
  1022. // err = errors.New("获取相关性图表, A指标mapping信息失败, Err:" + e.Error())
  1023. // return
  1024. //}
  1025. //edbInfoMappingB, e := data_manage.GetChartEdbMappingByEdbInfoId(req.CorrelationChartInfo.EdbInfoIdList[1].EdbInfoId)
  1026. //if e != nil {
  1027. // errMsg = "获取失败"
  1028. // err = errors.New("获取相关性图表, B指标mapping信息失败, Err:" + e.Error())
  1029. // return
  1030. //}
  1031. //periodData, correlationData, e := GetChartDataByEdbInfo(edbInfoMappingA, edbInfoMappingB, req.CorrelationChartInfo.LeadValue, req.CorrelationChartInfo.LeadUnit, req.CorrelationChartInfo.StartDate, req.CorrelationChartInfo.EndDate)
  1032. //if e != nil {
  1033. // errMsg = "获取失败"
  1034. // err = errors.New("获取相关性图表, 图表计算值失败, Err:" + e.Error())
  1035. // return
  1036. //}
  1037. //periodDataByte, e := json.Marshal(periodData)
  1038. //if e != nil {
  1039. // errMsg = "获取失败"
  1040. // err = errors.New("相关性图表, X轴信息有误, Err:" + e.Error())
  1041. // return
  1042. //}
  1043. //correlationDataByte, e := json.Marshal(correlationData[0].Value)
  1044. //if e != nil {
  1045. // errMsg = "获取失败"
  1046. // err = errors.New("相关性图表, Y轴信息有误, Err:" + e.Error())
  1047. // return
  1048. //}
  1049. correlationChart.LeadValue = req.CorrelationChartInfo.LeadValue
  1050. correlationChart.LeadUnit = req.CorrelationChartInfo.LeadUnit
  1051. correlationChart.CalculateValue = req.CorrelationChartInfo.CalculateValue
  1052. correlationChart.CalculateUnit = req.CorrelationChartInfo.CalculateUnit
  1053. correlationChart.StartDate = startDate
  1054. correlationChart.EndDate = endDate
  1055. correlationChart.EdbInfoIdFirst = req.CorrelationChartInfo.EdbInfoIdList[0].EdbInfoId
  1056. correlationChart.EdbInfoIdSecond = req.CorrelationChartInfo.EdbInfoIdList[1].EdbInfoId
  1057. // 滚动相关性会有日期等信息
  1058. if chartItem.Source == utils.CHART_SOURCE_ROLLING_CORRELATION {
  1059. correlationChart.DateType = req.CorrelationChartInfo.DateType
  1060. }
  1061. //correlationChart.PeriodData = string(periodDataByte)
  1062. //correlationChart.CorrelationData = string(correlationDataByte)
  1063. correlationChart.ModifyTime = time.Now().Local()
  1064. //correlationUpdateCols := []string{"LeadValue", "LeadUnit", "StartDate", "EndDate", "EdbInfoIdFirst", "EdbInfoIdSecond", "PeriodData","CorrelationData", "ModifyTime"}
  1065. correlationUpdateCols := []string{"LeadValue", "LeadUnit", "CalculateValue", "CalculateUnit", "DateType", "StartDate", "EndDate", "EdbInfoIdFirst", "EdbInfoIdSecond", "ModifyTime"}
  1066. // 修改图表与指标mapping
  1067. req.ChartType = 9
  1068. err = data_manage.EditCorrelationChartInfoAndMapping(&req, edbInfoIdStr, "公历", 6, disableVal, ``,
  1069. correlationChart, correlationUpdateCols)
  1070. if err != nil {
  1071. errMsg = "保存失败"
  1072. err = errors.New("保存失败,Err:" + err.Error())
  1073. return
  1074. }
  1075. resp := new(data_manage.AddChartInfoResp)
  1076. resp.ChartInfoId = chartItem.ChartInfoId
  1077. resp.UniqueCode = chartItem.UniqueCode
  1078. resp.ChartType = req.ChartType
  1079. //添加es数据
  1080. go data.EsAddOrEditChartInfo(chartItem.ChartInfoId)
  1081. //修改my eta es数据
  1082. go data.EsAddOrEditMyChartInfoByChartInfoId(chartItem.ChartInfoId)
  1083. return
  1084. }
  1085. // CopyChartInfo 复制图表
  1086. func CopyChartInfo(configId, classifyId int, chartName string, correlationChartInfoReq data_manage.CorrelationChartInfoReq, oldChartInfo *data_manage.ChartInfo, sysUser *system.Admin, lang string) (chartInfo *data_manage.ChartInfo, err error, errMsg string, isSendEmail bool) {
  1087. configSource := 2
  1088. isSendEmail = true
  1089. // 获取相关性图的配置
  1090. multipleGraphConfigChartMapping, err := data_manage.GetMultipleGraphConfigChartMappingByIdAndSource(configId, configSource)
  1091. if err != nil {
  1092. return
  1093. }
  1094. multipleGraphConfig, err := data_manage.GetMultipleGraphConfigById(configId)
  1095. if err != nil {
  1096. return
  1097. }
  1098. multipleGraphConfig.MultipleGraphConfigId = 0
  1099. err = data_manage.AddMultipleGraphConfig(multipleGraphConfig)
  1100. if err != nil {
  1101. return
  1102. }
  1103. // 添加图
  1104. addChartReq := data_manage.AddChartInfoReq{
  1105. ChartClassifyId: classifyId,
  1106. ChartName: chartName,
  1107. ChartType: utils.CHART_TYPE_CURVE,
  1108. Calendar: "公历",
  1109. CorrelationChartInfo: correlationChartInfoReq,
  1110. ChartThemeId: oldChartInfo.ChartThemeId,
  1111. SourcesFrom: oldChartInfo.SourcesFrom,
  1112. Instructions: oldChartInfo.Instructions,
  1113. MarkersLines: oldChartInfo.MarkersLines,
  1114. MarkersAreas: oldChartInfo.MarkersAreas,
  1115. }
  1116. chartSource := utils.CHART_SOURCE_CORRELATION // 默认是相关性图
  1117. chartInfo, err, errMsg, isSendEmail = AddChartInfo(addChartReq, chartSource, sysUser, lang)
  1118. if err != nil {
  1119. return
  1120. }
  1121. // 添加关系
  1122. multipleGraphConfigChartMapping = &data_manage.MultipleGraphConfigChartMapping{
  1123. //Id: 0,
  1124. MultipleGraphConfigId: multipleGraphConfig.MultipleGraphConfigId,
  1125. ChartInfoId: chartInfo.ChartInfoId,
  1126. Source: configSource,
  1127. ModifyTime: time.Now(),
  1128. CreateTime: time.Now(),
  1129. }
  1130. err = data_manage.AddMultipleGraphConfigChartMapping(multipleGraphConfigChartMapping)
  1131. if err != nil {
  1132. return
  1133. }
  1134. //添加es数据
  1135. go data.EsAddOrEditChartInfo(chartInfo.ChartInfoId)
  1136. return
  1137. }
  1138. // CalculateCorrelation 计算相关性-获取x轴和y轴
  1139. func CalculateCorrelation(leadValue int, leadUnit, frequencyA, frequencyB string, dataListA, dataListB []*data_manage.EdbDataList) (xEdbIdValue []int, yDataList []data_manage.YData, err error) {
  1140. xData := make([]int, 0)
  1141. yData := make([]float64, 0)
  1142. if leadValue == 0 {
  1143. xData = append(xData, 0)
  1144. }
  1145. if leadValue > 0 {
  1146. leadMin := 0 - leadValue
  1147. xLen := 2*leadValue + 1
  1148. for i := 0; i < xLen; i++ {
  1149. n := leadMin + i
  1150. xData = append(xData, n)
  1151. }
  1152. }
  1153. // 计算窗口,不包含第一天
  1154. //startDateTime, _ := time.ParseInLocation(utils.FormatDate, startDate, time.Local)
  1155. //startDate = startDateTime.AddDate(0, 0, 1).Format(utils.FormatDate)
  1156. //// 2023-03-02 时间序列始终以指标B为基准, 始终是A进行平移
  1157. //baseEdbInfo := edbInfoMappingB
  1158. //changeEdbInfo := edbInfoMappingA
  1159. // 2023-03-17 时间序列始终以指标A为基准, 始终是B进行平移
  1160. //baseEdbInfo := edbInfoMappingA
  1161. //changeEdbInfo := edbInfoMappingB
  1162. // 获取时间基准指标在时间区间内的值
  1163. //aDataList := make([]*data_manage.EdbDataList, 0)
  1164. //switch baseEdbInfo.EdbInfoCategoryType {
  1165. //case 0:
  1166. // aDataList, err = data_manage.GetEdbDataList(baseEdbInfo.Source, baseEdbInfo.SubSource, baseEdbInfo.EdbInfoId, startDate, endDate)
  1167. //case 1:
  1168. // _, aDataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(baseEdbInfo.EdbInfoId, startDate, endDate, false)
  1169. //default:
  1170. // err = errors.New("指标base类型异常")
  1171. // return
  1172. //}
  1173. //
  1174. //// 获取变频指标所有日期的值, 插值法完善数据
  1175. //bDataList := make([]*data_manage.EdbDataList, 0)
  1176. //switch changeEdbInfo.EdbInfoCategoryType {
  1177. //case 0:
  1178. // bDataList, err = data_manage.GetEdbDataList(changeEdbInfo.Source, changeEdbInfo.SubSource, changeEdbInfo.EdbInfoId, "", "")
  1179. //case 1:
  1180. // _, bDataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(changeEdbInfo.EdbInfoId, "", "", false)
  1181. //default:
  1182. // err = errors.New("指标change类型异常")
  1183. // return
  1184. //}
  1185. //changeDataMap := make(map[string]float64)
  1186. //newChangeDataList, e := HandleDataByLinearRegression(bDataList, changeDataMap)
  1187. //if e != nil {
  1188. // err = fmt.Errorf("获取变频指标插值法Map失败, Err: %s", e.Error())
  1189. // return
  1190. //}
  1191. // 2023-03-17 时间序列始终以指标A为基准, 始终是B进行平移
  1192. baseDataList := make([]*data_manage.EdbDataList, 0)
  1193. baseDataMap := make(map[string]float64)
  1194. changeDataList := make([]*data_manage.EdbDataList, 0)
  1195. changeDataMap := make(map[string]float64)
  1196. // 先把低频指标升频为高频
  1197. {
  1198. frequencyIntMap := map[string]int{
  1199. "日度": 1,
  1200. "周度": 2,
  1201. "旬度": 3,
  1202. "月度": 4,
  1203. "季度": 5,
  1204. "年度": 6,
  1205. }
  1206. // 如果A指标是高频,那么就需要对B指标进行升频
  1207. if frequencyIntMap[frequencyA] < frequencyIntMap[frequencyB] {
  1208. tmpNewChangeDataList, e := HandleDataByLinearRegression(dataListB, changeDataMap)
  1209. if e != nil {
  1210. err = fmt.Errorf("获取变频指标插值法Map失败, Err: %s", e.Error())
  1211. return
  1212. }
  1213. changeDataList = tmpNewChangeDataList
  1214. baseDataList = dataListA
  1215. for _, v := range baseDataList {
  1216. baseDataMap[v.DataTime] = v.Value
  1217. }
  1218. } else if frequencyIntMap[frequencyA] > frequencyIntMap[frequencyB] {
  1219. // 如果B指标是高频,那么就需要对A指标进行升频
  1220. tmpNewChangeDataList, e := HandleDataByLinearRegression(dataListA, baseDataMap)
  1221. if e != nil {
  1222. err = fmt.Errorf("获取变频指标插值法Map失败, Err: %s", e.Error())
  1223. return
  1224. }
  1225. baseDataList = tmpNewChangeDataList
  1226. changeDataList = dataListB
  1227. for _, v := range changeDataList {
  1228. changeDataMap[v.DataTime] = v.Value
  1229. }
  1230. } else {
  1231. baseDataList = dataListA
  1232. for _, v := range baseDataList {
  1233. baseDataMap[v.DataTime] = v.Value
  1234. }
  1235. changeDataList = dataListB
  1236. for _, v := range changeDataList {
  1237. changeDataMap[v.DataTime] = v.Value
  1238. }
  1239. }
  1240. }
  1241. // 计算不领先也不滞后时的相关系数
  1242. baseCalculateData := make([]float64, 0)
  1243. baseDataTimeArr := make([]string, 0)
  1244. for i := range baseDataList {
  1245. baseDataTimeArr = append(baseDataTimeArr, baseDataList[i].DataTime)
  1246. baseCalculateData = append(baseCalculateData, baseDataList[i].Value)
  1247. }
  1248. //zeroBaseData := make([]float64, 0)
  1249. //zeroCalculateData := make([]float64, 0)
  1250. //for i := range baseDataTimeArr {
  1251. // tmpBaseVal, ok1 := baseDataMap[baseDataTimeArr[i]]
  1252. // tmpCalculateVal, ok2 := changeDataMap[baseDataTimeArr[i]]
  1253. // if ok1 && ok2 {
  1254. // zeroBaseData = append(zeroBaseData, tmpBaseVal)
  1255. // zeroCalculateData = append(zeroCalculateData, tmpCalculateVal)
  1256. // }
  1257. //}
  1258. //if len(zeroBaseData) != len(zeroCalculateData) {
  1259. // err = fmt.Errorf("相关系数两组序列元素数不一致, %d-%d", len(baseCalculateData), len(zeroCalculateData))
  1260. // return
  1261. //}
  1262. //zeroRatio := utils.CalculateCorrelationByIntArr(zeroBaseData, zeroCalculateData)
  1263. //if leadValue == 0 {
  1264. // yData = append(yData, zeroRatio)
  1265. //}
  1266. // 计算领先/滞后N期
  1267. if leadValue > 0 {
  1268. // 平移变频指标领先/滞后的日期(单位天)
  1269. moveUnitDays := utils.FrequencyDaysMap[leadUnit]
  1270. for i := range xData {
  1271. //if xData[i] == 0 {
  1272. // yData = append(yData, zeroRatio)
  1273. // continue
  1274. //}
  1275. xCalculateData := make([]float64, 0)
  1276. yCalculateData := make([]float64, 0)
  1277. // 平移指定天数
  1278. mDays := int(moveUnitDays) * xData[i]
  1279. _, dMap := MoveDataDaysToNewDataList(changeDataList, mDays)
  1280. // 取出对应的基准日期的值
  1281. for i2 := range baseDataTimeArr {
  1282. tmpDate := baseDataTimeArr[i2]
  1283. if yVal, ok := dMap[tmpDate]; ok {
  1284. xCalculateData = append(xCalculateData, baseCalculateData[i2])
  1285. yCalculateData = append(yCalculateData, yVal)
  1286. }
  1287. }
  1288. if len(yCalculateData) <= 0 {
  1289. //err = fmt.Errorf("领先滞后相关系数两组序列元素数不一致, %d-%d", len(baseCalculateData), len(yCalculateData))
  1290. //return
  1291. // 领先滞后后,没有可以计算的数据了
  1292. continue
  1293. }
  1294. // 公式计算出领先/滞后频度对应点的相关性系数
  1295. ratio := utils.CalculateCorrelationByIntArr(xCalculateData, yCalculateData)
  1296. yData = append(yData, ratio)
  1297. }
  1298. }
  1299. xEdbIdValue = xData
  1300. yDataList = make([]data_manage.YData, 0)
  1301. yDate := "0000-00-00"
  1302. yDataList = append(yDataList, data_manage.YData{
  1303. Date: yDate,
  1304. Value: yData,
  1305. })
  1306. return
  1307. }
  1308. // GetFactorChartDataByChartId 获取多因子相关性图表数据
  1309. func GetFactorChartDataByChartId(chartInfoId int, extraConfig string) (xEdbIdValue []int, yDataList []data_manage.YData, err error) {
  1310. if chartInfoId <= 0 {
  1311. return
  1312. }
  1313. // 指标对应的图例
  1314. extra := new(data_manage.CorrelationChartInfoExtraConfig)
  1315. if extraConfig != "" {
  1316. if e := json.Unmarshal([]byte(extraConfig), extra); e != nil {
  1317. err = fmt.Errorf("解析图表额外配置失败, err: %v", e)
  1318. return
  1319. }
  1320. }
  1321. legends := make(map[string]*data_manage.CorrelationChartLegend)
  1322. if extra != nil {
  1323. for _, v := range extra.LegendConfig {
  1324. s := fmt.Sprintf("%d-%d", v.SeriesId, v.EdbInfoId)
  1325. legends[s] = v
  1326. }
  1327. }
  1328. // 获取图表引用到的系列指标
  1329. edbMappings, e := data_manage.GetChartUsedFactorSeriesEdb(chartInfoId)
  1330. if e != nil {
  1331. err = fmt.Errorf("获取图表引用系列指标失败, err: %v", e)
  1332. return
  1333. }
  1334. // 取出计算结果
  1335. yDataList = make([]data_manage.YData, 0)
  1336. yDate := "0000-00-00"
  1337. for k, m := range edbMappings {
  1338. var values []data_manage.FactorEdbSeriesCorrelationMatrixValues
  1339. if m.CalculateData != "" {
  1340. e = json.Unmarshal([]byte(m.CalculateData), &values)
  1341. if e != nil {
  1342. err = fmt.Errorf("系列指标计算数据有误, err: %v", e)
  1343. return
  1344. }
  1345. }
  1346. var y []float64
  1347. for _, v := range values {
  1348. if k == 0 {
  1349. xEdbIdValue = append(xEdbIdValue, v.XData)
  1350. }
  1351. y = append(y, v.YData)
  1352. }
  1353. var yData data_manage.YData
  1354. yData.Date = yDate
  1355. yData.Value = y
  1356. yData.SeriesEdb.SeriesId = m.FactorEdbSeriesId
  1357. yData.SeriesEdb.EdbInfoId = m.EdbInfoId
  1358. // 图例
  1359. s := fmt.Sprintf("%d-%d", m.FactorEdbSeriesId, m.EdbInfoId)
  1360. legend := legends[s]
  1361. if legend != nil {
  1362. yData.Name = legend.LegendName
  1363. yData.Color = legend.Color
  1364. }
  1365. yDataList = append(yDataList, yData)
  1366. }
  1367. return
  1368. }
  1369. // FormatChartEdbInfoMappings 补充指标信息
  1370. func FormatChartEdbInfoMappings(chartInfoId int, mappings []*data_manage.ChartEdbInfoMapping) (edbList []*data_manage.ChartEdbInfoMapping, err error) {
  1371. edbList = make([]*data_manage.ChartEdbInfoMapping, 0)
  1372. if len(mappings) == 0 {
  1373. return
  1374. }
  1375. for _, v := range mappings {
  1376. if chartInfoId <= 0 {
  1377. v.IsAxis = 1
  1378. v.LeadValue = 0
  1379. v.LeadUnit = ""
  1380. v.ChartEdbMappingId = 0
  1381. v.ChartInfoId = 0
  1382. v.IsOrder = false
  1383. v.EdbInfoType = 1
  1384. v.ChartStyle = ""
  1385. v.ChartColor = ""
  1386. v.ChartWidth = 0
  1387. } else {
  1388. v.LeadUnitEn = data.GetLeadUnitEn(v.LeadUnit)
  1389. v.LeadUnitEn = data.GetLeadUnitEn(v.LeadUnit)
  1390. }
  1391. v.FrequencyEn = data.GetFrequencyEn(v.Frequency)
  1392. if v.Unit == `无` {
  1393. v.Unit = ``
  1394. }
  1395. edbList = append(edbList, v)
  1396. }
  1397. return
  1398. }