excel_info.go 40 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372
  1. package excel
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "eta/eta_api/models/data_manage"
  6. "eta/eta_api/models/data_manage/excel"
  7. "eta/eta_api/models/data_manage/excel/request"
  8. "eta/eta_api/models/data_manage/excel/response"
  9. "eta/eta_api/models/system"
  10. "eta/eta_api/services/data"
  11. "eta/eta_api/services/data/data_manage_permission"
  12. "eta/eta_api/utils"
  13. "fmt"
  14. "github.com/shopspring/decimal"
  15. "github.com/yidane/formula"
  16. "sort"
  17. "strconv"
  18. "strings"
  19. "time"
  20. )
  21. // GetExcelDetailInfoByExcelInfoId 根据表格id获取表格详情
  22. func GetExcelDetailInfoByExcelInfoId(excelInfoId, sysUserId int) (excelDetail response.ExcelInfoDetail, errMsg string, err error) {
  23. errMsg = `获取失败`
  24. //获取eta表格信息
  25. excelInfo, err := excel.GetExcelInfoById(excelInfoId)
  26. if err != nil {
  27. err = errors.New("获取ETA表格信息失败,Err:" + err.Error())
  28. if err.Error() == utils.ErrNoRow() {
  29. errMsg = "ETA表格被删除,请刷新页面"
  30. err = errors.New("ETA表格被删除,请刷新页面,Err:" + err.Error())
  31. }
  32. return
  33. }
  34. return formatExcelInfo2Detail(excelInfo, sysUserId)
  35. }
  36. // GetExcelDetailInfoByUnicode 根据表格编码获取表格详情
  37. func GetExcelDetailInfoByUnicode(unicode string, sysUserId int) (excelDetail response.ExcelInfoDetail, errMsg string, err error) {
  38. errMsg = `获取失败`
  39. // 获取eta表格信息
  40. excelInfo, err := excel.GetExcelInfoByUnicode(unicode)
  41. if err != nil {
  42. err = errors.New("获取ETA表格信息失败,Err:" + err.Error())
  43. if err.Error() == utils.ErrNoRow() {
  44. errMsg = "ETA表格被删除,请刷新页面"
  45. err = errors.New("ETA表格被删除,请刷新页面,Err:" + err.Error())
  46. }
  47. return
  48. }
  49. return formatExcelInfo2Detail(excelInfo, sysUserId)
  50. }
  51. func formatExcelInfo2Detail(excelInfo *excel.ExcelInfo, sysUserId int) (excelDetail response.ExcelInfoDetail, errMsg string, err error) {
  52. // 数据权限
  53. haveOperaAuth, err := data_manage_permission.CheckExcelPermissionByExcelInfoId(excelInfo.ExcelInfoId, excelInfo.ExcelClassifyId, excelInfo.IsJoinPermission, sysUserId)
  54. if err != nil {
  55. err = errors.New("获取表格权限信息失败,Err" + err.Error())
  56. return
  57. }
  58. excelDetail = response.ExcelInfoDetail{
  59. ExcelInfoId: excelInfo.ExcelInfoId,
  60. Source: excelInfo.Source,
  61. ExcelType: excelInfo.ExcelType,
  62. ExcelName: excelInfo.ExcelName,
  63. UniqueCode: excelInfo.UniqueCode,
  64. ExcelClassifyId: excelInfo.ExcelClassifyId,
  65. SysUserId: excelInfo.SysUserId,
  66. SysUserRealName: excelInfo.SysUserRealName,
  67. Content: excelInfo.Content,
  68. ExcelImage: excelInfo.ExcelImage,
  69. FileUrl: excelInfo.FileUrl,
  70. Sort: excelInfo.Sort,
  71. IsDelete: excelInfo.IsDelete,
  72. ModifyTime: excelInfo.ModifyTime,
  73. CreateTime: excelInfo.CreateTime,
  74. TableData: nil,
  75. HaveOperaAuth: haveOperaAuth,
  76. }
  77. // 无权限,不需要返回数据
  78. if !haveOperaAuth {
  79. return
  80. }
  81. switch excelInfo.Source {
  82. case utils.TIME_TABLE: // 自定义表格
  83. var tableDataConfig TableDataConfig
  84. err = json.Unmarshal([]byte(excelDetail.Content), &tableDataConfig)
  85. if err != nil {
  86. err = errors.New("表格json转结构体失败,Err:" + err.Error())
  87. return
  88. }
  89. result, tmpErr := GetDataByTableDataConfig(tableDataConfig)
  90. if tmpErr != nil {
  91. err = errors.New("获取最新的表格数据失败,Err:" + tmpErr.Error())
  92. return
  93. }
  94. excelDetail.TableData = result
  95. case utils.MIXED_TABLE: // 混合表格
  96. var result request.MixedTableReq
  97. err = json.Unmarshal([]byte(excelDetail.Content), &result)
  98. if err != nil {
  99. err = errors.New("表格json转结构体失败,Err:" + err.Error())
  100. return
  101. }
  102. newData, tmpErr, tmpErrMsg := GetMixedTableCellData(result)
  103. if tmpErr != nil {
  104. errMsg = "获取失败"
  105. if tmpErrMsg != `` {
  106. errMsg = tmpErrMsg
  107. }
  108. err = errors.New("获取最新的数据失败,Err:" + tmpErr.Error())
  109. return
  110. }
  111. result.Data = newData
  112. excelDetail.TableData = result
  113. }
  114. return
  115. }
  116. // GetExcelInfoOpButton 获取ETA表格的操作权限
  117. func GetExcelInfoOpButton(sysUser *system.Admin, belongUserId, source int, haveOperaAuth bool) (button response.ExcelInfoDetailButton) {
  118. // 如果没有数据权限,那么直接返回
  119. if !haveOperaAuth {
  120. return
  121. }
  122. //非管理员角色查看其他用户创建的表格,可刷新、另存为、下载表格;
  123. button.RefreshButton = true
  124. button.CopyButton = true
  125. button.DownloadButton = true
  126. // 1、本用户创建的表格,可编辑、刷新、另存为、下载、删除,删除需二次确认;
  127. // 2、管理员角色对所有表格有如上权限;
  128. // 3、在线excel所有人都能编辑
  129. if sysUser.RoleTypeCode == utils.ROLE_TYPE_CODE_ADMIN || sysUser.RoleTypeCode == utils.ROLE_TYPE_CODE_FICC_ADMIN || sysUser.AdminId == belongUserId || source == utils.EXCEL_DEFAULT {
  130. button.OpButton = true
  131. button.DeleteButton = true
  132. }
  133. // 自定义分析
  134. if source == utils.CUSTOM_ANALYSIS_TABLE {
  135. if sysUser.RoleTypeCode == utils.ROLE_TYPE_CODE_ADMIN || sysUser.RoleTypeCode == utils.ROLE_TYPE_CODE_FICC_ADMIN || sysUser.AdminId == belongUserId {
  136. button.OpEdbButton = true // 生成、查看指标按钮
  137. button.RefreshEdbButton = true // 刷新指标按钮
  138. }
  139. }
  140. return
  141. }
  142. // GetFirstEdbDataList 获取第一列的数据
  143. func GetFirstEdbDataList(edbInfo *data_manage.EdbInfo, num int, manualDateList []string) (resultDataList []request.ManualDataReq, err error) {
  144. var dataList []*data_manage.EdbDataList
  145. switch edbInfo.EdbInfoType {
  146. case 0:
  147. dataList, err = data_manage.GetEdbDataList(edbInfo.Source, edbInfo.SubSource, edbInfo.EdbInfoId, ``, ``)
  148. case 1:
  149. _, dataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(edbInfo.EdbInfoId, ``, ``, false)
  150. default:
  151. err = errors.New(fmt.Sprint("获取失败,指标类型异常", edbInfo.EdbInfoType))
  152. }
  153. if err != nil {
  154. return
  155. }
  156. // 获取需要的期数
  157. lenData := len(dataList)
  158. if lenData <= 0 {
  159. return
  160. }
  161. tmpManualDateNum := 0 // 手工数据的期数
  162. lenManualDate := len(manualDateList)
  163. if lenManualDate > 0 {
  164. sortDateList := manualDateList
  165. baseDateList := utils.StrArr{}
  166. baseDateList = append(baseDateList, sortDateList...)
  167. sort.Sort(baseDateList)
  168. sortDateList = append([]string{}, baseDateList...)
  169. lastData := dataList[lenData-1]
  170. lastDataDate, tmpErr := time.ParseInLocation(utils.FormatDate, lastData.DataTime, time.Local)
  171. if tmpErr != nil {
  172. err = tmpErr
  173. return
  174. }
  175. // 遍历倒序后的日期,然后匹配在实际数据之后日期的个数
  176. for _, tmpDateStr := range sortDateList {
  177. tmpDate, tmpErr := time.ParseInLocation(utils.FormatDate, tmpDateStr, time.Local)
  178. if tmpErr != nil {
  179. err = tmpErr
  180. return
  181. }
  182. if tmpDate.After(lastDataDate) {
  183. tmpManualDateNum++
  184. continue
  185. }
  186. break
  187. }
  188. }
  189. // 需要的期数减去手工数据的期数,这才是A列指标需要的数据
  190. num = num - tmpManualDateNum
  191. if num > lenData {
  192. num = lenData
  193. }
  194. latestDateTime, _ := time.ParseInLocation(utils.FormatDate, edbInfo.LatestDate, time.Local)
  195. for i := 1; i <= num; i++ {
  196. dataTime, _ := time.ParseInLocation(utils.FormatDate, dataList[lenData-i].DataTime, time.Local)
  197. dataType := 1
  198. // 如果是预测指标,且当前值的日期,晚于实际日期,那么是预测值
  199. if edbInfo.EdbInfoType == 1 && dataTime.After(latestDateTime) {
  200. dataType = 5
  201. }
  202. resultDataList = append(resultDataList, request.ManualDataReq{
  203. DataType: dataType,
  204. DataTime: dataList[lenData-i].DataTime,
  205. ShowValue: fmt.Sprint(dataList[lenData-i].Value),
  206. Value: fmt.Sprint(dataList[lenData-i].Value),
  207. DataTimeType: 1,
  208. })
  209. }
  210. return
  211. }
  212. // GetOtherEdbDataList 获取其他列的数据
  213. func GetOtherEdbDataList(edbInfo *data_manage.EdbInfo, dateList []string) (resultDataList []request.ManualDataReq, err error) {
  214. lenDate := len(dateList)
  215. if lenDate <= 0 {
  216. return
  217. }
  218. sortDateList := dateList
  219. baseDateList := utils.StrArr{}
  220. baseDateList = append(baseDateList, sortDateList...)
  221. sort.Sort(baseDateList)
  222. sortDateList = append([]string{}, baseDateList...)
  223. endDateTime, err := time.ParseInLocation(utils.FormatDate, sortDateList[0], time.Local)
  224. if err != nil {
  225. return
  226. }
  227. firstDateTime, err := time.ParseInLocation(utils.FormatDate, sortDateList[lenDate-1], time.Local)
  228. if err != nil {
  229. return
  230. }
  231. var dataList []*data_manage.EdbDataList
  232. switch edbInfo.EdbInfoType {
  233. case 0:
  234. dataList, err = data_manage.GetEdbDataList(edbInfo.Source, edbInfo.SubSource, edbInfo.EdbInfoId, ``, ``)
  235. case 1:
  236. _, dataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(edbInfo.EdbInfoId, ``, ``, false)
  237. default:
  238. err = errors.New(fmt.Sprint("获取失败,指标类型异常", edbInfo.EdbInfoType))
  239. }
  240. if err != nil {
  241. return
  242. }
  243. // 获取日期内的数据(包含开始日期前一个日期,以及 结束日期后一个日期,目的为了做空日期时的 插值法兼容)
  244. baseDataList := make([]*data_manage.EdbDataList, 0)
  245. var lastData *data_manage.EdbDataList
  246. var isInsert bool
  247. for _, data := range dataList {
  248. tmpDate := data.DataTime
  249. tmpDateTime, tmpErr := time.ParseInLocation(utils.FormatDate, tmpDate, time.Local)
  250. if tmpErr != nil {
  251. err = tmpErr
  252. return
  253. }
  254. if tmpDateTime.Before(firstDateTime) {
  255. lastData = data
  256. continue
  257. }
  258. // 如果是第一次写入数据
  259. if !isInsert && lastData != nil {
  260. baseDataList = append(baseDataList, lastData)
  261. }
  262. if tmpDateTime.After(endDateTime) {
  263. baseDataList = append(baseDataList, data)
  264. break
  265. }
  266. baseDataList = append(baseDataList, data)
  267. isInsert = true
  268. }
  269. // 实际数据的日期map
  270. realValMap := make(map[string]string)
  271. for _, v := range baseDataList {
  272. realValMap[v.DataTime] = v.DataTime
  273. }
  274. // 插值法处理
  275. handleDataMap := make(map[string]float64)
  276. err = data.HandleDataByLinearRegression(baseDataList, handleDataMap)
  277. if err != nil {
  278. return
  279. }
  280. latestDateTime, _ := time.ParseInLocation(utils.FormatDate, edbInfo.LatestDate, time.Local)
  281. // 对于不存在的数据做补充
  282. for _, date := range sortDateList {
  283. dataType := 1
  284. if _, ok := realValMap[date]; !ok {
  285. dataType = 2
  286. } else {
  287. dataTime, _ := time.ParseInLocation(utils.FormatDate, date, time.Local)
  288. // 如果是预测指标,且当前值的日期,晚于实际日期,那么是预测值
  289. if edbInfo.EdbInfoType == 1 && dataTime.After(latestDateTime) {
  290. dataType = 5
  291. }
  292. }
  293. var value, showValue string
  294. if tmpVal, ok := handleDataMap[date]; ok {
  295. value = fmt.Sprint(tmpVal)
  296. showValue = value
  297. } else {
  298. dataType = 3
  299. }
  300. resultDataList = append(resultDataList, request.ManualDataReq{
  301. DataType: dataType,
  302. DataTime: date,
  303. ShowValue: showValue,
  304. Value: value,
  305. })
  306. }
  307. return
  308. }
  309. // GetFirstHistoryEdbDataList 获取指标的历史的数据
  310. func GetFirstHistoryEdbDataList(edbInfo *data_manage.EdbInfo, num int, endDate string) (resultDataList []request.ManualDataReq, err error) {
  311. endDateTime, err := time.ParseInLocation(utils.FormatDate, endDate, time.Local)
  312. if err != nil {
  313. return
  314. }
  315. var dataList []*data_manage.EdbDataList
  316. switch edbInfo.EdbInfoType {
  317. case 0:
  318. dataList, err = data_manage.GetEdbDataList(edbInfo.Source, edbInfo.SubSource, edbInfo.EdbInfoId, ``, endDate)
  319. case 1:
  320. _, dataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(edbInfo.EdbInfoId, ``, endDate, true)
  321. default:
  322. err = errors.New(fmt.Sprint("获取失败,指标类型异常", edbInfo.EdbInfoType))
  323. }
  324. if err != nil {
  325. return
  326. }
  327. // 获取需要的期数
  328. lenData := len(dataList)
  329. if lenData <= 0 {
  330. return
  331. }
  332. lastData := dataList[lenData-1]
  333. lastDataDateTime, err := time.ParseInLocation(utils.FormatDate, lastData.DataTime, time.Local)
  334. if err != nil {
  335. return
  336. }
  337. if endDateTime.Equal(lastDataDateTime) || lastDataDateTime.After(endDateTime) {
  338. dataList = dataList[:lenData-1]
  339. lenData = len(dataList)
  340. }
  341. if num > lenData {
  342. num = lenData
  343. }
  344. latestDateTime, _ := time.ParseInLocation(utils.FormatDate, edbInfo.LatestDate, time.Local)
  345. for i := 1; i <= num; i++ {
  346. dataTime, _ := time.ParseInLocation(utils.FormatDate, dataList[lenData-i].DataTime, time.Local)
  347. dataType := 1
  348. // 如果是预测指标,且当前值的日期,晚于实际日期,那么是预测值
  349. if edbInfo.EdbInfoType == 1 && dataTime.After(latestDateTime) {
  350. dataType = 5
  351. }
  352. resultDataList = append(resultDataList, request.ManualDataReq{
  353. DataType: dataType,
  354. DataTime: dataList[lenData-i].DataTime,
  355. ShowValue: fmt.Sprint(dataList[lenData-i].Value),
  356. Value: fmt.Sprint(dataList[lenData-i].Value),
  357. })
  358. }
  359. return
  360. }
  361. type TableDataConfig struct {
  362. EdbInfoIdList []int `description:"指标id列表,从左至右,从上到下的顺序"`
  363. Sort int `description:"日期排序,0:倒序,1:正序"`
  364. Data []ManualData `description:"数据列表"`
  365. Num int `description:"实际数据需要列出来的期数"`
  366. RemoveDate []string `description:"不展示的日期"`
  367. ManualDate []string `description:"手动配置的日期(未来的日期)"`
  368. TableEdbInfoList []TableEdbInfo `description:"表格内指标信息"`
  369. TextRowData [][]request.ManualDataReq `description:"文本列表"`
  370. }
  371. type TableEdbInfo struct {
  372. EdbInfoId int `description:"指标ID"`
  373. Tag string `description:"标签"`
  374. EdbName string `description:"指标名称"`
  375. EdbAliasName string `description:"指标别名"`
  376. Frequency string `description:"频度"`
  377. Unit string `description:"单位"`
  378. }
  379. type ManualData struct {
  380. DataType int `description:"数据类型,1:普通的,2:插值法,3:手动输入,4:公式计算"`
  381. DataTime string `description:"所属日期"`
  382. DataTimeType int `description:"日期类型,1:实际日期;2:未来日期"`
  383. ShowValue string `description:"展示值"`
  384. Value string `description:"实际值(计算公式)"`
  385. EdbInfoId int `description:"指标id"`
  386. Tag string `description:"下标"`
  387. RelationEdbInfoList []request.RelationEdbInfo `description:"关联指标(计算公式中关联的指标,用于计算的时候去匹配)"`
  388. }
  389. // GetTableDataConfig 根据TableDataReq配置获取相关数据配置
  390. func GetTableDataConfig(reqData request.TableDataReq) (tableDataConfig TableDataConfig, err error) {
  391. // 指标数据
  392. tableDataConfig.EdbInfoIdList = reqData.EdbInfoIdList
  393. tableDataConfig.Sort = reqData.Sort
  394. if len(reqData.Data) <= 0 {
  395. err = errors.New("数据不能为空")
  396. return
  397. }
  398. // 开始日期
  399. var startDate string
  400. // A列的指标id
  401. var firstEdbInfoId int
  402. // 手工操作的数据列
  403. manualDataList := make([]ManualData, 0)
  404. // 指标配置列表
  405. tableEdbInfoList := make([]TableEdbInfo, 0)
  406. // 第一列的日期map
  407. firstDateMap := make(map[string]string)
  408. manualDateMap := make(map[string]string)
  409. for _, v := range reqData.Data {
  410. // 指标信息
  411. tmpTableEdbInfo := TableEdbInfo{
  412. EdbInfoId: v.EdbInfoId,
  413. Tag: v.Tag,
  414. EdbName: v.EdbName,
  415. EdbAliasName: v.EdbAliasName,
  416. Frequency: v.Frequency,
  417. Unit: v.Unit,
  418. }
  419. tableEdbInfoList = append(tableEdbInfoList, tmpTableEdbInfo)
  420. // 确定数据A列
  421. if v.Tag == "A" {
  422. firstEdbInfoId = v.EdbInfoId
  423. lenData := len(v.Data)
  424. if lenData <= 0 {
  425. err = errors.New("A列不能为空")
  426. return
  427. }
  428. index := 0
  429. if reqData.Sort == 1 {
  430. // 倒序
  431. index = lenData - 1
  432. }
  433. startDate = v.Data[index].DataTime
  434. // 存在的日期列表
  435. for _, data := range v.Data {
  436. firstDateMap[data.DataTime] = data.DataTime
  437. if data.DataTimeType == 2 {
  438. manualDateMap[data.DataTime] = data.DataTime
  439. }
  440. }
  441. }
  442. for _, data := range v.Data {
  443. if data.DataType == 3 || data.DataType == 4 {
  444. tmpManualData := ManualData{
  445. DataType: data.DataType,
  446. DataTime: data.DataTime,
  447. DataTimeType: data.DataTimeType,
  448. ShowValue: data.ShowValue,
  449. Value: data.Value,
  450. EdbInfoId: v.EdbInfoId,
  451. Tag: v.Tag,
  452. RelationEdbInfoList: data.RelationEdbInfoList,
  453. }
  454. if data.DataType == 4 {
  455. tmpManualData.ShowValue = ``
  456. }
  457. manualDataList = append(manualDataList, tmpManualData)
  458. }
  459. }
  460. }
  461. // 总共需要的期数
  462. num := len(reqData.Data[0].Data)
  463. removeDate := make([]string, 0)
  464. // 获取期数
  465. {
  466. firstDateTime, tmpErr := time.ParseInLocation(utils.FormatDate, startDate, time.Local)
  467. if tmpErr != nil {
  468. err = tmpErr
  469. return
  470. }
  471. edbInfo, tmpErr := data_manage.GetEdbInfoById(firstEdbInfoId)
  472. if tmpErr != nil {
  473. err = tmpErr
  474. return
  475. }
  476. var firstDataList []*data_manage.EdbDataList
  477. switch edbInfo.EdbInfoType {
  478. case 0:
  479. firstDataList, err = data_manage.GetEdbDataList(edbInfo.Source, edbInfo.SubSource, edbInfo.EdbInfoId, ``, ``)
  480. case 1:
  481. _, firstDataList, _, _, err, _ = data.GetPredictDataListByPredictEdbInfoId(edbInfo.EdbInfoId, ``, ``, false)
  482. default:
  483. err = errors.New(fmt.Sprint("获取失败,指标类型异常", edbInfo.EdbInfoType))
  484. }
  485. if err != nil {
  486. return
  487. }
  488. // 获取日期内的数据(包含开始日期前一个日期,以及 结束日期后一个日期,目的为了做空日期时的 插值法兼容)
  489. baseDataList := make([]*data_manage.EdbDataList, 0)
  490. for _, data := range firstDataList {
  491. tmpDate := data.DataTime
  492. tmpDateTime, tmpErr := time.ParseInLocation(utils.FormatDate, tmpDate, time.Local)
  493. if tmpErr != nil {
  494. err = tmpErr
  495. return
  496. }
  497. if tmpDateTime.Before(firstDateTime) {
  498. continue
  499. }
  500. baseDataList = append(baseDataList, data)
  501. }
  502. // 筛选出需要删除的日期
  503. for _, tmpData := range baseDataList {
  504. //firstDateMap{}
  505. if _, ok := firstDateMap[tmpData.DataTime]; !ok {
  506. removeDate = append(removeDate, tmpData.DataTime)
  507. }
  508. }
  509. }
  510. tableDataConfig.Num = num
  511. tableDataConfig.RemoveDate = removeDate
  512. tableDataConfig.Data = manualDataList
  513. tableDataConfig.TableEdbInfoList = tableEdbInfoList
  514. tableDataConfig.TextRowData = reqData.TextRowData
  515. return
  516. }
  517. // GetDataByTableDataConfig 根据数据配置获取表格数据
  518. func GetDataByTableDataConfig(tableDataConfig TableDataConfig) (resultResp request.TableDataReq, err error) {
  519. // 没有选择指标的情况下,直接返回吧
  520. if len(tableDataConfig.EdbInfoIdList) <= 0 {
  521. return
  522. }
  523. // 实际期数没有的情况下,直接返回吧
  524. if tableDataConfig.Num <= 0 {
  525. return
  526. }
  527. // 获取所有的指标信息
  528. edbInfoMap := make(map[int]*data_manage.EdbInfo)
  529. edbInfoIdList := make([]int, 0)
  530. // 标签与指标id的map
  531. tagEdbInfoIdMap := make(map[string]int)
  532. {
  533. for _, tableEdbInfo := range tableDataConfig.TableEdbInfoList {
  534. edbInfoIdList = append(edbInfoIdList, tableEdbInfo.EdbInfoId)
  535. tagEdbInfoIdMap[tableEdbInfo.Tag] = tableEdbInfo.EdbInfoId
  536. }
  537. edbInfoList, tmpErr := data_manage.GetEdbInfoByIdList(edbInfoIdList)
  538. if tmpErr != nil {
  539. err = tmpErr
  540. return
  541. }
  542. for _, v := range edbInfoList {
  543. edbInfoMap[v.EdbInfoId] = v
  544. }
  545. }
  546. manualDateMap := make(map[string]string, 0)
  547. manualDateList := make([]string, 0)
  548. for _, v := range tableDataConfig.Data {
  549. if _, ok := manualDateMap[v.DataTime]; !ok {
  550. manualDateMap[v.DataTime] = v.DataTime
  551. manualDateList = append(manualDateList, v.DataTime)
  552. }
  553. }
  554. // 寻找A列的数据列表
  555. firstEdbInfo, ok := edbInfoMap[tableDataConfig.TableEdbInfoList[0].EdbInfoId]
  556. if !ok {
  557. err = errors.New("找不到A列指标")
  558. return
  559. }
  560. baseFirstEdbInfoDataList, err := GetFirstEdbDataList(firstEdbInfo, tableDataConfig.Num, manualDateList)
  561. if err != nil {
  562. return
  563. }
  564. // A列找不到数据,那么就直接返回吧
  565. if len(baseFirstEdbInfoDataList) <= 0 {
  566. return
  567. }
  568. firstEdbInfoDataList := make([]request.ManualDataReq, 0)
  569. if tableDataConfig.RemoveDate != nil && len(tableDataConfig.RemoveDate) > 0 {
  570. for _, v := range baseFirstEdbInfoDataList {
  571. if utils.InArrayByStr(tableDataConfig.RemoveDate, v.DataTime) {
  572. continue
  573. }
  574. firstEdbInfoDataList = append(firstEdbInfoDataList, v)
  575. }
  576. } else {
  577. firstEdbInfoDataList = baseFirstEdbInfoDataList
  578. }
  579. if len(firstEdbInfoDataList) <= 0 {
  580. return
  581. }
  582. // 实际数据的最后一天
  583. lastRealDateTime, err := time.ParseInLocation(utils.FormatDate, firstEdbInfoDataList[0].DataTime, time.Local)
  584. if err != nil {
  585. return
  586. }
  587. dateMap := make(map[string]string)
  588. dateList := make([]string, 0)
  589. edbInfoIdDateDataMap := make(map[int]map[string]request.ManualDataReq)
  590. firstDateDataMap := make(map[string]request.ManualDataReq)
  591. for _, v := range firstEdbInfoDataList {
  592. dateList = append(dateList, v.DataTime)
  593. dateMap[v.DataTime] = v.DataTime
  594. firstDateDataMap[v.DataTime] = v
  595. }
  596. // 将手工数据的日期填补进去(未来的日期,过去的就不管了)
  597. for _, manualData := range tableDataConfig.Data {
  598. if !utils.InArrayByStr(dateList, manualData.DataTime) {
  599. tmpDateTime, tmpErr := time.ParseInLocation(utils.FormatDate, manualData.DataTime, time.Local)
  600. if tmpErr != nil {
  601. err = tmpErr
  602. return
  603. }
  604. if tmpDateTime.After(lastRealDateTime) {
  605. dateList = append(dateList, manualData.DataTime)
  606. }
  607. }
  608. }
  609. edbInfoIdDateDataMap[firstEdbInfo.EdbInfoId] = firstDateDataMap
  610. for k, edbInfoId := range tableDataConfig.EdbInfoIdList {
  611. if k == 0 {
  612. continue
  613. }
  614. tmpEdbInfo, ok := edbInfoMap[edbInfoId]
  615. if !ok {
  616. err = errors.New("找不到A列指标")
  617. return
  618. }
  619. otherDataList, tmpErr := GetOtherEdbDataList(tmpEdbInfo, dateList)
  620. if tmpErr != nil {
  621. err = tmpErr
  622. return
  623. }
  624. tmpDateDataMap := make(map[string]request.ManualDataReq)
  625. for _, v := range otherDataList {
  626. tmpDateDataMap[v.DataTime] = v
  627. }
  628. edbInfoIdDateDataMap[tmpEdbInfo.EdbInfoId] = tmpDateDataMap
  629. }
  630. for _, v := range tableDataConfig.Data {
  631. tmpDate := v.DataTime
  632. if _, ok := dateMap[tmpDate]; !ok {
  633. dateMap[v.DataTime] = tmpDate
  634. }
  635. edbInfoIdDateData, ok := edbInfoIdDateDataMap[v.EdbInfoId]
  636. if !ok {
  637. edbInfoIdDateData = make(map[string]request.ManualDataReq)
  638. }
  639. // 判断是否存在该日期的数据(不存在,那么插入数据吧,存在就不管了)
  640. tmpManualData, ok := edbInfoIdDateData[tmpDate]
  641. if !ok {
  642. edbInfoIdDateData[tmpDate] = request.ManualDataReq{
  643. DataType: v.DataType,
  644. DataTime: v.DataTime,
  645. ShowValue: v.ShowValue,
  646. Value: v.Value,
  647. }
  648. } else {
  649. if (tmpManualData.DataType == 3 || tmpManualData.DataType == 4) && tmpManualData.ShowValue == `` {
  650. tmpManualData.DataType = v.DataType
  651. tmpManualData.ShowValue = v.ShowValue
  652. tmpManualData.Value = v.Value
  653. tmpManualData.RelationEdbInfoList = v.RelationEdbInfoList
  654. edbInfoIdDateData[tmpDate] = tmpManualData
  655. }
  656. }
  657. edbInfoIdDateDataMap[v.EdbInfoId] = edbInfoIdDateData
  658. }
  659. // 获取数据的日期排序
  660. sortDateTimeList := make([]time.Time, 0)
  661. {
  662. sortDateList := dateList
  663. if tableDataConfig.Sort == 1 {
  664. baseDateList := utils.StrArr{}
  665. baseDateList = append(baseDateList, sortDateList...)
  666. sort.Sort(baseDateList)
  667. sortDateList = append([]string{}, baseDateList...)
  668. } else {
  669. sort.Strings(sortDateList)
  670. }
  671. for _, v := range sortDateList {
  672. tmpDateTime, tmpErr := time.ParseInLocation(utils.FormatDate, v, time.Local)
  673. if tmpErr != nil {
  674. err = tmpErr
  675. return
  676. }
  677. sortDateTimeList = append(sortDateTimeList, tmpDateTime)
  678. }
  679. }
  680. // 数据处理,处理成表格的数据格式
  681. tableDataMap, textRowListDataResp := handleTable(tagEdbInfoIdMap, lastRealDateTime, sortDateTimeList, edbInfoIdDateDataMap, tableDataConfig.Data, tableDataConfig.TextRowData)
  682. data := make([]request.EdbInfoData, 0)
  683. for _, tableEdbInfo := range tableDataConfig.TableEdbInfoList {
  684. tagEdbInfoIdMap[tableEdbInfo.Tag] = tableEdbInfo.EdbInfoId
  685. manualDataReqList := make([]request.ManualDataReq, 0)
  686. tmpEdbInfoData := request.EdbInfoData{
  687. EdbInfoId: tableEdbInfo.EdbInfoId,
  688. Tag: tableEdbInfo.Tag,
  689. EdbName: tableEdbInfo.EdbName,
  690. EdbAliasName: tableEdbInfo.EdbAliasName,
  691. Frequency: tableEdbInfo.Frequency,
  692. Unit: tableEdbInfo.Unit,
  693. Data: manualDataReqList,
  694. }
  695. edbInfo, ok := edbInfoMap[tableEdbInfo.EdbInfoId]
  696. if ok {
  697. tmpEdbInfoData.EdbName = edbInfo.EdbName
  698. tmpEdbInfoData.Frequency = edbInfo.Frequency
  699. tmpEdbInfoData.Unit = edbInfo.Unit
  700. }
  701. for index, dateTime := range sortDateTimeList {
  702. dataTimeType := 1
  703. if dateTime.After(lastRealDateTime) {
  704. dataTimeType = 2
  705. }
  706. tmpDateTimeStr := dateTime.Format(utils.FormatDate)
  707. rowData, ok := tableDataMap[index+1]
  708. if !ok {
  709. manualDataReqList = append(manualDataReqList, request.ManualDataReq{
  710. DataType: 3,
  711. DataTime: tmpDateTimeStr,
  712. DataTimeType: dataTimeType,
  713. ShowValue: "",
  714. Value: "",
  715. RelationEdbInfoList: nil,
  716. })
  717. continue
  718. }
  719. tmpData, ok := rowData[tableEdbInfo.Tag]
  720. if !ok {
  721. manualDataReqList = append(manualDataReqList, request.ManualDataReq{
  722. DataType: 3,
  723. DataTime: tmpDateTimeStr,
  724. DataTimeType: dataTimeType,
  725. ShowValue: "",
  726. Value: "",
  727. RelationEdbInfoList: nil,
  728. })
  729. continue
  730. }
  731. tmpData.DataTimeType = dataTimeType
  732. manualDataReqList = append(manualDataReqList, tmpData)
  733. }
  734. tmpEdbInfoData.Data = manualDataReqList
  735. data = append(data, tmpEdbInfoData)
  736. }
  737. // 处理一下数据格式
  738. for _, d := range data {
  739. for k2, d2 := range d.Data {
  740. // 可能有ShowValue非数值, 转换一下报错则continue
  741. vf, e := strconv.ParseFloat(d2.ShowValue, 64)
  742. if e != nil {
  743. continue
  744. }
  745. d.Data[k2].ShowValue = utils.FormatTableDataShowValue(vf)
  746. }
  747. }
  748. for _, d := range textRowListDataResp {
  749. for k2, d2 := range d {
  750. // 可能有ShowValue非数值, 转换一下报错则continue
  751. vf, e := strconv.ParseFloat(d2.ShowValue, 64)
  752. if e != nil {
  753. continue
  754. }
  755. d[k2].ShowValue = utils.FormatTableDataShowValue(vf)
  756. }
  757. }
  758. resultResp = request.TableDataReq{
  759. EdbInfoIdList: edbInfoIdList,
  760. Sort: tableDataConfig.Sort,
  761. TextRowData: textRowListDataResp,
  762. Data: data,
  763. }
  764. return
  765. }
  766. // handleTable 表格数据处理
  767. func handleTable(tagEdbInfoIdMap map[string]int, lastRealDateTime time.Time, sortDateTimeList []time.Time, edbInfoIdDateDataMap map[int]map[string]request.ManualDataReq, manualDataList []ManualData, textRowData [][]request.ManualDataReq) (tableDataMap map[int]map[string]request.ManualDataReq, textRowListDataResp [][]request.ManualDataReq) {
  768. tagList := make([]string, 0)
  769. for tag, _ := range tagEdbInfoIdMap {
  770. tagList = append(tagList, tag)
  771. }
  772. sort.Strings(tagList)
  773. tableDataMap = make(map[int]map[string]request.ManualDataReq) //行、列数据
  774. // 日期与行的关系
  775. dateIndexMap := make(map[string]int)
  776. for k, dateTime := range sortDateTimeList {
  777. rowDataMap := make(map[string]request.ManualDataReq)
  778. dataTimeType := 1
  779. if dateTime.After(lastRealDateTime) {
  780. dataTimeType = 2
  781. }
  782. tmpDateTimeStr := dateTime.Format(utils.FormatDate)
  783. dateIndexMap[tmpDateTimeStr] = k + 1
  784. for _, tag := range tagList {
  785. edbInfoId, ok := tagEdbInfoIdMap[tag]
  786. if !ok { // 没有找到该指标的映射关系,那么就用空串填补
  787. rowDataMap[tag] = request.ManualDataReq{
  788. DataType: 3,
  789. DataTime: tmpDateTimeStr,
  790. DataTimeType: dataTimeType,
  791. ShowValue: "",
  792. Value: "",
  793. RelationEdbInfoList: nil,
  794. }
  795. continue
  796. }
  797. // 获取指标的数据map
  798. dateDataMap, ok := edbInfoIdDateDataMap[edbInfoId]
  799. if !ok { // 没有找到该指标的数据,那么就用空串填补
  800. rowDataMap[tag] = request.ManualDataReq{
  801. DataType: 3,
  802. DataTime: tmpDateTimeStr,
  803. DataTimeType: dataTimeType,
  804. ShowValue: "",
  805. Value: "",
  806. RelationEdbInfoList: nil,
  807. }
  808. continue
  809. }
  810. // 获取指标该日期的数据
  811. tmpData, ok := dateDataMap[tmpDateTimeStr]
  812. if !ok { // 该指标没有找到对应日期的数据,那么就用空串填补
  813. rowDataMap[tag] = request.ManualDataReq{
  814. DataType: 3,
  815. DataTime: tmpDateTimeStr,
  816. DataTimeType: dataTimeType,
  817. ShowValue: "",
  818. Value: "",
  819. RelationEdbInfoList: nil,
  820. }
  821. continue
  822. }
  823. tmpData.DataTimeType = dataTimeType
  824. rowDataMap[tag] = tmpData
  825. }
  826. tableDataMap[k+1] = rowDataMap
  827. }
  828. // 替换手工设置的数据
  829. for _, manualData := range manualDataList {
  830. // 找不到该日期,说明这日期过期了,不处理
  831. index, ok := dateIndexMap[manualData.DataTime]
  832. if !ok {
  833. continue
  834. }
  835. // 获取对应行的数据
  836. rowDataMap, ok := tableDataMap[index]
  837. if !ok {
  838. continue
  839. }
  840. // 找到对应的单元格
  841. tmpData, ok := rowDataMap[manualData.Tag]
  842. if !ok {
  843. continue
  844. }
  845. // 如果该单元格实际有数据(包含预测值),或者插值法补充了数据的话,那么就不用手动填入的数据
  846. if utils.InArrayByInt([]int{1, 2, 5}, tmpData.DataType) {
  847. continue
  848. }
  849. // 手工填写的数字
  850. if tmpData.DataType == 3 {
  851. tmpData.ShowValue = manualData.ShowValue
  852. tmpData.Value = manualData.Value
  853. tableDataMap[index][manualData.Tag] = tmpData
  854. //edbInfoIdDateDataMap[manualData.EdbInfoId][manualData.DataTime] = tmpData
  855. continue
  856. }
  857. // 公式
  858. tmpData.DataType = manualData.DataType
  859. tmpData.ShowValue = ``
  860. tmpData.Value = manualData.Value
  861. tmpData.RelationEdbInfoList = manualData.RelationEdbInfoList
  862. tableDataMap[index][manualData.Tag] = tmpData
  863. }
  864. // 文本行的列表插入
  865. lenTableData := len(tableDataMap)
  866. // 文本行第一列的数据列表(可能多行)
  867. firstColTextRowList := make([]request.ManualDataReq, 0)
  868. // 参与计算的文本行列表数据
  869. tmpTextRowList := make([][]request.ManualDataReq, 0)
  870. for k, textRowList := range textRowData {
  871. // 判断列数是否匹配,不匹配的话那么过滤
  872. if len(tagList)+1 != len(textRowList) {
  873. continue
  874. }
  875. rowDataMap := make(map[string]request.ManualDataReq)
  876. tmpTextRow := make([]request.ManualDataReq, 0)
  877. for index, textRow := range textRowList {
  878. // 移除第一列,因为第一列是日期列
  879. if index == 0 {
  880. firstColTextRowList = append(firstColTextRowList, textRow)
  881. continue
  882. }
  883. rowDataMap[tagList[index-1]] = textRow
  884. tmpTextRow = append(tmpTextRow, textRow)
  885. }
  886. tableDataMap[lenTableData+k+1] = rowDataMap
  887. tmpTextRowList = append(tmpTextRowList, tmpTextRow)
  888. }
  889. // 参与计算的单元格
  890. calculateCellMap := make(map[string]string)
  891. // 计算手工填写的单元格
  892. for _, manualData := range manualDataList {
  893. // 找不到该日期,说明这日期过期了,不处理
  894. index, ok := dateIndexMap[manualData.DataTime]
  895. if !ok {
  896. continue
  897. }
  898. // 获取对应行的数据
  899. rowDataMap, ok := tableDataMap[index]
  900. if !ok {
  901. continue
  902. }
  903. // 找到对应的单元格
  904. colData, ok := rowDataMap[manualData.Tag]
  905. if !ok {
  906. continue
  907. }
  908. // 如果该单元格不是计算公式的单元格,那么直接退出当前循环即可
  909. if colData.DataType != 4 {
  910. continue
  911. }
  912. tagMap := make(map[string]float64)
  913. lenRelation := len(colData.RelationEdbInfoList)
  914. replaceNum := 0
  915. for _, relation := range colData.RelationEdbInfoList {
  916. relationCellTagName := strings.ToUpper(relation.Tag) + relation.Row
  917. valStr, tmpErr := getCalculateValue(tableDataMap, relation.Tag, relation.Row, calculateCellMap)
  918. if tmpErr != nil {
  919. continue
  920. }
  921. tmpValDecimal, tmpErr := decimal.NewFromString(valStr)
  922. if tmpErr != nil {
  923. continue
  924. }
  925. tagMap[relationCellTagName], _ = tmpValDecimal.Float64()
  926. replaceNum++
  927. }
  928. // 如果替换的数据与关联的不一致,那么就退出当前循环
  929. if lenRelation != replaceNum {
  930. continue
  931. }
  932. // 计算
  933. val, _, err := calculate(strings.ToUpper(colData.Value), tagMap)
  934. // 计算失败,退出循环
  935. if err != nil {
  936. continue
  937. }
  938. // 重新赋值
  939. colData.ShowValue = val
  940. tableDataMap[index][manualData.Tag] = colData
  941. }
  942. // 计算文本行的单元格
  943. for k, textRow := range tmpTextRowList {
  944. // 获取对应行的数据
  945. index := lenTableData + k + 1
  946. rowDataMap, ok := tableDataMap[index]
  947. if !ok {
  948. continue
  949. }
  950. for colIndex, _ := range textRow {
  951. currTag := tagList[colIndex]
  952. // 找到对应的单元格
  953. colData, ok := rowDataMap[currTag]
  954. if !ok {
  955. continue
  956. }
  957. // 如果该单元格不是计算公式的单元格,那么直接退出当前循环即可
  958. if colData.DataType != 4 {
  959. continue
  960. }
  961. tagMap := make(map[string]float64)
  962. lenRelation := len(colData.RelationEdbInfoList)
  963. replaceNum := 0
  964. for _, relation := range colData.RelationEdbInfoList {
  965. relationCellTagName := strings.ToUpper(relation.Tag) + relation.Row
  966. valStr, tmpErr := getCalculateValue(tableDataMap, relation.Tag, relation.Row, calculateCellMap)
  967. if tmpErr != nil {
  968. continue
  969. }
  970. tmpValDecimal, tmpErr := decimal.NewFromString(valStr)
  971. if tmpErr != nil {
  972. continue
  973. }
  974. tagMap[relationCellTagName], _ = tmpValDecimal.Float64()
  975. replaceNum++
  976. }
  977. // 如果替换的数据与关联的不一致,那么就退出当前循环
  978. if lenRelation != replaceNum {
  979. continue
  980. }
  981. // 计算
  982. val, _, err := calculate(strings.ToUpper(colData.Value), tagMap)
  983. // 计算失败,退出循环
  984. if err != nil {
  985. continue
  986. }
  987. // 重新赋值
  988. colData.ShowValue = val
  989. tableDataMap[index][currTag] = colData
  990. }
  991. }
  992. // 计算文本行第一列的数据值(多行)
  993. for k, colData := range firstColTextRowList {
  994. // 如果该单元格不是计算公式的单元格,那么直接退出当前循环即可
  995. if colData.DataType != 4 {
  996. continue
  997. }
  998. tagMap := make(map[string]float64)
  999. lenRelation := len(colData.RelationEdbInfoList)
  1000. replaceNum := 0
  1001. for _, relation := range colData.RelationEdbInfoList {
  1002. relationCellTagName := strings.ToUpper(relation.Tag) + relation.Row
  1003. valStr, tmpErr := getCalculateValue(tableDataMap, relation.Tag, relation.Row, calculateCellMap)
  1004. if tmpErr != nil {
  1005. continue
  1006. }
  1007. tmpValDecimal, tmpErr := decimal.NewFromString(valStr)
  1008. if tmpErr != nil {
  1009. continue
  1010. }
  1011. tagMap[relationCellTagName], _ = tmpValDecimal.Float64()
  1012. replaceNum++
  1013. }
  1014. // 如果替换的数据与关联的不一致,那么就退出当前循环
  1015. if lenRelation != replaceNum {
  1016. continue
  1017. }
  1018. // 计算
  1019. val, _, err := calculate(strings.ToUpper(colData.Value), tagMap)
  1020. // 计算失败,退出循环
  1021. if err != nil {
  1022. continue
  1023. }
  1024. // 重新赋值
  1025. colData.ShowValue = val
  1026. firstColTextRowList[k] = colData
  1027. }
  1028. {
  1029. // 文本行的数据处理返回
  1030. textRowListDataResp = make([][]request.ManualDataReq, 0)
  1031. newLenTableDataMap := len(tableDataMap)
  1032. // 文本行的第一行所在的位置
  1033. firstTextRow := lenTableData + 1
  1034. for i := firstTextRow; i <= newLenTableDataMap; i++ {
  1035. textRowDataResp := make([]request.ManualDataReq, 0)
  1036. textRowDataResp = append(textRowDataResp, firstColTextRowList[i-firstTextRow])
  1037. for _, tmpTag := range tagList {
  1038. textRowDataResp = append(textRowDataResp, tableDataMap[i][tmpTag])
  1039. }
  1040. textRowListDataResp = append(textRowListDataResp, textRowDataResp)
  1041. }
  1042. }
  1043. return
  1044. }
  1045. // getCalculateValue 获取公式计算的结果
  1046. func getCalculateValue(tableDataMap map[int]map[string]request.ManualDataReq, tag, row string, calculateCellMap map[string]string) (val string, err error) {
  1047. rowInt, err := strconv.Atoi(row)
  1048. if err != nil {
  1049. return
  1050. }
  1051. // 单元格的标签名
  1052. cellTagName := strings.ToUpper(tag) + row
  1053. val, ok := calculateCellMap[cellTagName]
  1054. if ok {
  1055. return
  1056. }
  1057. // 查找行数据
  1058. rowData, ok := tableDataMap[rowInt]
  1059. if !ok {
  1060. err = errors.New("查找" + row + "行的数据失败")
  1061. return
  1062. }
  1063. // 查找单元格数据
  1064. colData, ok := rowData[tag]
  1065. if !ok {
  1066. err = errors.New("查找单元格" + tag + row + "的数据失败")
  1067. return
  1068. }
  1069. // 如果不是计算单元格
  1070. if colData.DataType != 4 {
  1071. val = colData.ShowValue
  1072. return
  1073. }
  1074. // 如果是计算单元格
  1075. calculateCellMap[cellTagName] = ``
  1076. tagMap := make(map[string]float64)
  1077. for _, relation := range colData.RelationEdbInfoList {
  1078. relationCellTagName := strings.ToUpper(relation.Tag) + relation.Row
  1079. valStr, tmpErr := getCalculateValue(tableDataMap, relation.Tag, relation.Row, calculateCellMap)
  1080. if tmpErr != nil {
  1081. err = tmpErr
  1082. return
  1083. }
  1084. tmpValDecimal, tmpErr := decimal.NewFromString(valStr)
  1085. if tmpErr != nil {
  1086. err = tmpErr
  1087. return
  1088. }
  1089. tagMap[relationCellTagName], _ = tmpValDecimal.Float64()
  1090. }
  1091. // 计算
  1092. val, _, err = calculate(strings.ToUpper(colData.Value), tagMap)
  1093. if err != nil {
  1094. return
  1095. }
  1096. // 重新赋值
  1097. colData.ShowValue = val
  1098. tableDataMap[rowInt][tag] = colData
  1099. calculateCellMap[cellTagName] = val
  1100. return
  1101. }
  1102. // calculate 公式计算
  1103. func calculate(calculateFormula string, TagMap map[string]float64) (calVal, errMsg string, err error) {
  1104. if calculateFormula == "" {
  1105. errMsg = "公式异常"
  1106. err = errors.New(errMsg)
  1107. return
  1108. }
  1109. calculateFormula = strings.TrimPrefix(calculateFormula, "=")
  1110. calculateFormula = strings.Replace(calculateFormula, "(", "(", -1)
  1111. calculateFormula = strings.Replace(calculateFormula, ")", ")", -1)
  1112. calculateFormula = strings.Replace(calculateFormula, ",", ",", -1)
  1113. calculateFormula = strings.Replace(calculateFormula, "。", ".", -1)
  1114. calculateFormula = strings.Replace(calculateFormula, "%", "*0.01", -1)
  1115. formulaFormStr := utils.ReplaceFormula(TagMap, calculateFormula)
  1116. //计算公式异常,那么就移除该指标
  1117. if formulaFormStr == `` {
  1118. errMsg = "公式异常"
  1119. err = errors.New(errMsg)
  1120. return
  1121. }
  1122. expression := formula.NewExpression(formulaFormStr)
  1123. calResult, err := expression.Evaluate()
  1124. if err != nil {
  1125. errMsg = "计算失败"
  1126. err = errors.New("计算失败:Err:" + err.Error() + ";formulaStr:" + formulaFormStr)
  1127. // 分母为0的报错
  1128. if strings.Contains(err.Error(), "divide by zero") {
  1129. errMsg = "分母不能为0"
  1130. err = errors.New("分母不能为空,计算公式:" + formulaFormStr)
  1131. }
  1132. return
  1133. }
  1134. // 如果计算结果是NAN,那么就提示报错
  1135. if calResult.IsNan() {
  1136. errMsg = "计算失败"
  1137. err = errors.New("计算失败:计算结果是:NAN;formulaStr:" + formulaFormStr)
  1138. return
  1139. }
  1140. calVal = calResult.String()
  1141. // 转Decimal然后四舍五入
  1142. valDecimal, err := decimal.NewFromString(calVal)
  1143. if err != nil {
  1144. errMsg = "计算失败"
  1145. err = errors.New("计算失败,结果转 Decimal 失败:Err:" + err.Error() + ";formulaStr:" + formulaFormStr)
  1146. return
  1147. }
  1148. calVal = valDecimal.Round(4).String()
  1149. return
  1150. }
  1151. // GetEdbIdsFromExcelCodes 获取表格中的指标IDs
  1152. func GetEdbIdsFromExcelCodes(excelCodes []string, sysUserId int) (edbIds []int, err error) {
  1153. edbIds = make([]int, 0)
  1154. edbIdExist := make(map[int]bool)
  1155. for _, v := range excelCodes {
  1156. // 表格详情
  1157. detail, msg, e := GetExcelDetailInfoByUnicode(v, sysUserId)
  1158. if e != nil {
  1159. err = fmt.Errorf("GetExcelDetailInfoByExcelInfoId err: %s, errMsg: %s", e.Error(), msg)
  1160. return
  1161. }
  1162. // 自定义表格
  1163. if detail.Source == utils.TIME_TABLE {
  1164. jsonByte, e := json.Marshal(detail.TableData)
  1165. if e != nil {
  1166. err = fmt.Errorf("JSON格式化自定义表格数据失败, Err: %s", e.Error())
  1167. return
  1168. }
  1169. var tableData request.TableDataReq
  1170. if e = json.Unmarshal(jsonByte, &tableData); e != nil {
  1171. err = fmt.Errorf("解析自定义表格数据失败, Err: %s", e.Error())
  1172. return
  1173. }
  1174. for _, tv := range tableData.EdbInfoIdList {
  1175. if edbIdExist[tv] {
  1176. continue
  1177. }
  1178. edbIdExist[tv] = true
  1179. edbIds = append(edbIds, tv)
  1180. }
  1181. }
  1182. // 混合表格
  1183. if detail.Source == utils.MIXED_TABLE {
  1184. jsonByte, e := json.Marshal(detail.TableData)
  1185. if e != nil {
  1186. err = fmt.Errorf("JSON格式化混合表格数据失败, Err: %s", e.Error())
  1187. return
  1188. }
  1189. var tableData request.MixedTableReq
  1190. if e = json.Unmarshal(jsonByte, &tableData); e != nil {
  1191. err = fmt.Errorf("解析混合表格数据失败, Err: %s", e.Error())
  1192. return
  1193. }
  1194. if len(tableData.Data) > 0 {
  1195. for _, td := range tableData.Data {
  1196. for _, tv := range td {
  1197. if tv.EdbInfoId > 0 && !edbIdExist[tv.EdbInfoId] {
  1198. edbIdExist[tv.EdbInfoId] = true
  1199. edbIds = append(edbIds, tv.EdbInfoId)
  1200. }
  1201. }
  1202. }
  1203. }
  1204. }
  1205. }
  1206. return
  1207. }
  1208. // GetExcelEdbBatchRefreshKey 获取批量刷新表格指标缓存key
  1209. func GetExcelEdbBatchRefreshKey(source string, reportId, chapterId int) string {
  1210. if source == `` {
  1211. return ``
  1212. }
  1213. return fmt.Sprint("batch_refresh_excel_edb:", source, ":", reportId, ":", chapterId)
  1214. }