edb_info_calculate.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  1. package data
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "eta/eta_api/models/data_manage"
  6. "eta/eta_api/utils"
  7. "fmt"
  8. "github.com/shopspring/decimal"
  9. "math"
  10. "regexp"
  11. "strings"
  12. "time"
  13. )
  14. func CheckFormula(formula string) map[string]string {
  15. mathFormula := []string{"MAX", "MIN", "ABS", "ACOS", "ASIN", "CEIL", "MOD", "POW", "ROUND", "SIGN", "SIN", "TAN", "LOG10", "LOG2", "LOG", "LN", "EXP"}
  16. str := strings.ToUpper(formula)
  17. for _, v := range mathFormula {
  18. str = strings.Replace(str, v, "", -1)
  19. }
  20. str = strings.Replace(str, "(", "", -1)
  21. str = strings.Replace(str, ")", "", -1)
  22. byteMap := make(map[string]string)
  23. for i := 0; i < len(str); i++ {
  24. byteInt := str[i]
  25. if byteInt >= 65 && byteInt <= 90 {
  26. byteStr := string(byteInt)
  27. if _, ok := byteMap[byteStr]; !ok {
  28. byteMap[byteStr] = byteStr
  29. }
  30. }
  31. }
  32. return byteMap
  33. }
  34. type FormulaListItem struct {
  35. Formula string `json:"f"`
  36. Date string `json:"d"`
  37. }
  38. // CheckFormulaJson 检测计算公式json串是否异常
  39. func CheckFormulaJson(formula string) (formulaSlice []string, err error) {
  40. list := make([]FormulaListItem, 0)
  41. err = json.Unmarshal([]byte(formula), &list)
  42. if err != nil {
  43. err = fmt.Errorf("公式串解析失败: json.Unmarshal Err: %v", err)
  44. return
  45. }
  46. formulaSlice = make([]string, 0)
  47. // 日期排序
  48. for _, v := range list {
  49. formulaSlice = append(formulaSlice, v.Formula)
  50. }
  51. return
  52. }
  53. type CalculateItems struct {
  54. EdbInfoId int
  55. DataMap map[string]float64
  56. }
  57. func GetFormulaMap() map[string]string {
  58. funMap := make(map[string]string)
  59. funMap["MAX"] = "[@@]"
  60. funMap["MIN"] = "[@!]"
  61. funMap["ABS"] = "[@#]"
  62. funMap["CEIL"] = "[@$]"
  63. funMap["COS"] = "[@%]"
  64. funMap["FLOOR"] = "[@^]"
  65. funMap["MOD"] = "[@&]"
  66. funMap["POW"] = "[@*]"
  67. funMap["ROUND"] = "[@(]"
  68. return funMap
  69. }
  70. // 处理整个数据
  71. func handleDateSaveDataMap(dateList []string, realSaveDataMap, saveDataMap map[string]map[int]float64, edbInfoIdArr []*data_manage.EdbInfo, emptyType int) {
  72. var startDate, endDate string
  73. var startDateT, endDateT time.Time
  74. if emptyType == 2 || emptyType == 3 {
  75. for k, _ := range realSaveDataMap {
  76. if k > endDate {
  77. endDate = k
  78. }
  79. if k < startDate || startDate == "" {
  80. startDate = k
  81. }
  82. }
  83. startDateT, _ = time.ParseInLocation(utils.FormatDate, startDate, time.Local)
  84. endDateT, _ = time.ParseInLocation(utils.FormatDate, endDate, time.Local)
  85. }
  86. for _, date := range dateList {
  87. tmpDataMap := realSaveDataMap[date]
  88. for _, edbInfo := range edbInfoIdArr {
  89. tmpEdbInfoId := edbInfo.EdbInfoId // 当前指标id
  90. // 如果该日期不存在该指标数据,那么需要找寻前后日期的数据,进行填补
  91. if _, ok := tmpDataMap[tmpEdbInfoId]; !ok {
  92. //day := 0
  93. //switch edbInfo.Frequency {
  94. //case "周度":
  95. // day = 7
  96. //case "旬度":
  97. // day = 15
  98. //case "月度":
  99. // day = 30
  100. //case "季度":
  101. // day = 90
  102. //case "年度":
  103. // day = 365
  104. //}
  105. // 需求池 255 指标运算文案修改,补数据遍历区间修改(2023-3-7 09:37:23修改)
  106. switch emptyType {
  107. case 0:
  108. handleDateDataMap(realSaveDataMap, saveDataMap, date, tmpEdbInfoId, 35)
  109. case 2:
  110. handleDateDataMapBefore(realSaveDataMap, saveDataMap, date, tmpEdbInfoId, startDateT, endDateT)
  111. case 3:
  112. handleDateDataMapAfter(realSaveDataMap, saveDataMap, date, tmpEdbInfoId, startDateT, endDateT)
  113. case 4:
  114. handleDateDataMapZero(saveDataMap, date, tmpEdbInfoId)
  115. }
  116. }
  117. }
  118. }
  119. }
  120. // handleDataMap 处理单个日期的数据
  121. func handleDateDataMap(realSaveDataMap, saveDataMap map[string]map[int]float64, date string, edbInfoId, day int) {
  122. currDate, _ := time.ParseInLocation(utils.FormatDate, date, time.Local)
  123. // 后一天
  124. nextDateDayStr := currDate.AddDate(0, 0, 1).Format(utils.FormatDate)
  125. // 前一天
  126. preDateDayStr := currDate.AddDate(0, 0, -1).Format(utils.FormatDate)
  127. for i := 1; i <= day; i++ {
  128. // 下个日期的数据
  129. {
  130. if i >= 1 {
  131. nextDateDayStr = currDate.AddDate(0, 0, i).Format(utils.FormatDate)
  132. }
  133. if findDataMap, hasFindDataMap := realSaveDataMap[nextDateDayStr]; hasFindDataMap { // 下一个日期有数据
  134. if val, hasFindItem := findDataMap[edbInfoId]; hasFindItem {
  135. saveDataMap[date][edbInfoId] = val
  136. return
  137. }
  138. }
  139. }
  140. // 上个日期的数据
  141. {
  142. if i >= 1 {
  143. preDateDayStr = currDate.AddDate(0, 0, -i).Format(utils.FormatDate)
  144. }
  145. if findDataMap, hasFindDataMap := realSaveDataMap[preDateDayStr]; hasFindDataMap { // 下一个日期有数据
  146. if val, hasFindItem := findDataMap[edbInfoId]; hasFindItem {
  147. saveDataMap[date][edbInfoId] = val
  148. return
  149. }
  150. }
  151. }
  152. }
  153. }
  154. // handleDataByLinearRegression 插值法补充数据(线性方程式)
  155. func handleDataByLinearRegression(edbInfoDataList []*data_manage.EdbDataList, handleDataMap map[string]float64) (err error) {
  156. if len(edbInfoDataList) < 2 {
  157. return
  158. }
  159. var startEdbInfoData *data_manage.EdbDataList
  160. for _, v := range edbInfoDataList {
  161. handleDataMap[v.DataTime] = v.Value
  162. // 第一个数据就给过滤了,给后面的试用
  163. if startEdbInfoData == nil {
  164. startEdbInfoData = v
  165. continue
  166. }
  167. // 获取两条数据之间相差的天数
  168. startDataTime, _ := time.ParseInLocation(utils.FormatDate, startEdbInfoData.DataTime, time.Local)
  169. currDataTime, _ := time.ParseInLocation(utils.FormatDate, v.DataTime, time.Local)
  170. betweenHour := int(currDataTime.Sub(startDataTime).Hours())
  171. betweenDay := betweenHour / 24
  172. // 如果相差一天,那么过滤
  173. if betweenDay <= 1 {
  174. startEdbInfoData = v
  175. continue
  176. }
  177. // 生成线性方程式
  178. var a, b float64
  179. {
  180. coordinateData := make([]utils.Coordinate, 0)
  181. tmpCoordinate1 := utils.Coordinate{
  182. X: 1,
  183. Y: startEdbInfoData.Value,
  184. }
  185. coordinateData = append(coordinateData, tmpCoordinate1)
  186. tmpCoordinate2 := utils.Coordinate{
  187. X: float64(betweenDay) + 1,
  188. Y: v.Value,
  189. }
  190. coordinateData = append(coordinateData, tmpCoordinate2)
  191. a, b = utils.GetLinearResult(coordinateData)
  192. if math.IsNaN(a) || math.IsNaN(b) {
  193. err = errors.New("线性方程公式生成失败")
  194. return
  195. }
  196. }
  197. // 生成对应的值
  198. {
  199. for i := 1; i < betweenDay; i++ {
  200. tmpDataTime := startDataTime.AddDate(0, 0, i)
  201. aDecimal := decimal.NewFromFloat(a)
  202. xDecimal := decimal.NewFromInt(int64(i) + 1)
  203. bDecimal := decimal.NewFromFloat(b)
  204. val, _ := aDecimal.Mul(xDecimal).Add(bDecimal).Round(4).Float64()
  205. handleDataMap[tmpDataTime.Format(utils.FormatDate)] = val
  206. }
  207. }
  208. startEdbInfoData = v
  209. }
  210. return
  211. }
  212. // HandleDataByLinearRegression 插值法补充数据(线性方程式)
  213. func HandleDataByLinearRegression(edbInfoDataList []*data_manage.EdbDataList, handleDataMap map[string]float64) (err error) {
  214. return handleDataByLinearRegression(edbInfoDataList, handleDataMap)
  215. }
  216. // CallCalculateComputeCorrelation 调用计算拟合残差的相关系数
  217. func CallCalculateComputeCorrelation(data *data_manage.EdbInfoCalculateBatchSaveReqByEdbLib, lang string) (val string, err error, errMsg string) {
  218. errMsg = "计算失败"
  219. // 调用指标库去更新
  220. reqJson, err := json.Marshal(data)
  221. if err != nil {
  222. errMsg = "计算相关系数参数解析异常!"
  223. err = errors.New("参数解析失败,Err:" + err.Error())
  224. return
  225. }
  226. respItem, err := CalculateComputeCorrelation(string(reqJson), lang)
  227. if err != nil {
  228. return
  229. }
  230. if respItem.Ret == 200 {
  231. val = respItem.Data
  232. }
  233. return
  234. }
  235. // handleDateDataMapBefore 前值填充:空值优先以最近的前值填充,没有前值时,用后值填充
  236. func handleDateDataMapBefore(realSaveDataMap, saveDataMap map[string]map[int]float64, date string, edbInfoId int, startDateT, endDateT time.Time) {
  237. currDate, _ := time.ParseInLocation(utils.FormatDate, date, time.Local)
  238. // 后一天
  239. nextDateDay := currDate
  240. // 前一天
  241. preDateDay := currDate
  242. for i := 1; preDateDay.After(startDateT) || preDateDay == startDateT; i++ {
  243. // 上个日期的数据
  244. {
  245. preDateDay = currDate.AddDate(0, 0, -i)
  246. preDateDayStr := preDateDay.Format(utils.FormatDate)
  247. if findDataMap, hasFindDataMap := realSaveDataMap[preDateDayStr]; hasFindDataMap { // 下一个日期有数据
  248. if val, hasFindItem := findDataMap[edbInfoId]; hasFindItem {
  249. fmt.Println(fmt.Sprintf("date:%s, 无值,取%s的值%.4f", date, preDateDayStr, val))
  250. saveDataMap[date][edbInfoId] = val
  251. return
  252. }
  253. }
  254. }
  255. }
  256. for i := 1; nextDateDay.Before(endDateT) || nextDateDay == endDateT; i++ {
  257. // 下个日期的数据
  258. {
  259. nextDateDay = currDate.AddDate(0, 0, i)
  260. nextDateDayStr := nextDateDay.Format(utils.FormatDate)
  261. if findDataMap, hasFindDataMap := realSaveDataMap[nextDateDayStr]; hasFindDataMap { // 下一个日期有数据
  262. if val, hasFindItem := findDataMap[edbInfoId]; hasFindItem {
  263. fmt.Println(fmt.Sprintf("date:%s, 无值,取%s的值%.4f", date, nextDateDayStr, val))
  264. saveDataMap[date][edbInfoId] = val
  265. return
  266. }
  267. }
  268. }
  269. }
  270. return
  271. }
  272. // handleDateDataMapAfter 后值填充:空值优先以最近的后值填充,没有后值时,用前值填充
  273. func handleDateDataMapAfter(realSaveDataMap, saveDataMap map[string]map[int]float64, date string, edbInfoId int, startDateT, endDateT time.Time) {
  274. currDate, _ := time.ParseInLocation(utils.FormatDate, date, time.Local)
  275. // 后一天
  276. nextDateDay := currDate
  277. // 前一天
  278. preDateDay := currDate
  279. for i := 1; nextDateDay.Before(endDateT) || nextDateDay == endDateT; i++ {
  280. // 下个日期的数据
  281. {
  282. nextDateDay = currDate.AddDate(0, 0, i)
  283. nextDateDayStr := nextDateDay.Format(utils.FormatDate)
  284. if findDataMap, hasFindDataMap := realSaveDataMap[nextDateDayStr]; hasFindDataMap { // 下一个日期有数据
  285. if val, hasFindItem := findDataMap[edbInfoId]; hasFindItem {
  286. fmt.Println(fmt.Sprintf("date:%s, 无值,取%s的值%.4f", date, nextDateDayStr, val))
  287. saveDataMap[date][edbInfoId] = val
  288. return
  289. }
  290. }
  291. }
  292. }
  293. for i := 1; preDateDay.After(startDateT) || preDateDay == startDateT; i++ {
  294. // 上个日期的数据
  295. {
  296. preDateDay = currDate.AddDate(0, 0, -i)
  297. preDateDayStr := preDateDay.Format(utils.FormatDate)
  298. if findDataMap, hasFindDataMap := realSaveDataMap[preDateDayStr]; hasFindDataMap { // 下一个日期有数据
  299. if val, hasFindItem := findDataMap[edbInfoId]; hasFindItem {
  300. fmt.Println(fmt.Sprintf("date:%s, 无值,取%s的值%.4f", date, preDateDayStr, val))
  301. saveDataMap[date][edbInfoId] = val
  302. return
  303. }
  304. }
  305. }
  306. }
  307. return
  308. }
  309. // handleDateDataMapZero 等于0
  310. func handleDateDataMapZero(saveDataMap map[string]map[int]float64, date string, edbInfoId int) {
  311. saveDataMap[date][edbInfoId] = 0
  312. return
  313. }
  314. func GetMaxMinEdbInfo(formula string) string {
  315. //formula := "A+min(A,B,max(A,C))"
  316. // todo 无法处理max里嵌套max或者min的情况
  317. // 使用正则表达式匹配MAX和MIN函数及其参数
  318. regex := regexp.MustCompile(`(?i)(MAX|MIN)\((.*?)\)`)
  319. matches := regex.FindAllStringSubmatch(formula, -1)
  320. // 遍历匹配结果,输出MAX和MIN函数及其参数
  321. for _, match := range matches {
  322. if len(match) == 3 {
  323. parameter := strings.ToLower(match[0]) // 参数
  324. formula = strings.ReplaceAll(formula, match[0], parameter)
  325. fmt.Printf("formula: %s\n", formula)
  326. }
  327. }
  328. formula = strings.ReplaceAll(formula, "max", "MAX")
  329. formula = strings.ReplaceAll(formula, "min", "MIN")
  330. return formula
  331. }