custom_analysis_edb.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628
  1. package excel
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "eta/eta_api/models/data_manage/excel"
  6. "eta/eta_api/services/data"
  7. excelServices "eta/eta_api/services/excel"
  8. "eta/eta_api/utils"
  9. "fmt"
  10. "github.com/araddon/dateparse"
  11. "github.com/shopspring/decimal"
  12. "github.com/tealeg/xlsx"
  13. "github.com/xuri/excelize/v2"
  14. "strings"
  15. )
  16. // GetCustomAnalysisExcelData 获取自定义分析的表格data数据
  17. func GetCustomAnalysisExcelData(excelInfo *excel.ExcelInfo) (luckySheet excelServices.LuckySheet, err error, errMsg string) {
  18. // 查找当前excel的sheet列表
  19. sheetList, err := excel.GetAllSheetList(excelInfo.ExcelInfoId)
  20. if err != nil {
  21. errMsg = "保存失败"
  22. err = errors.New("查找当前excel的sheet列表失败,Err:" + err.Error())
  23. return
  24. }
  25. currSheetMap := make(map[string]string)
  26. for _, sheet := range sheetList {
  27. currSheetMap[sheet.SheetName] = sheet.SheetName
  28. }
  29. sheetCellDataMapList := make(map[int][]excelServices.LuckySheetCellData)
  30. // 通过excel的id获取各个sheet的单元格数据map
  31. {
  32. dataList, tmpErr := excel.GetAllSheetDataListByExcelInfoId(excelInfo.ExcelInfoId)
  33. if tmpErr != nil {
  34. err = tmpErr
  35. return
  36. }
  37. for _, cellData := range dataList {
  38. sheetDataList, ok := sheetCellDataMapList[cellData.ExcelSheetId]
  39. if !ok {
  40. sheetDataList = make([]excelServices.LuckySheetCellData, 0)
  41. }
  42. tmpSheetDataList := make([]excelServices.LuckySheetCellData, 0)
  43. err = json.Unmarshal([]byte(cellData.Data), &tmpSheetDataList)
  44. if err != nil {
  45. err = errors.New(fmt.Sprintf("解析data的配置失败,sheetId:%d,Err:%s", cellData.ExcelDataId, err.Error()))
  46. return
  47. }
  48. sheetCellDataMapList[cellData.ExcelSheetId] = append(sheetDataList, tmpSheetDataList...)
  49. }
  50. }
  51. // 转成luckySheet的数据格式
  52. luckySheet = excelServices.LuckySheet{
  53. SheetList: make([]excelServices.LuckySheetData, 0),
  54. }
  55. for _, sheet := range sheetList {
  56. var luckySheetDataConfig excelServices.LuckySheetDataConfig
  57. err = json.Unmarshal([]byte(sheet.Config), &luckySheetDataConfig)
  58. if err != nil {
  59. err = errors.New(fmt.Sprintf("解析sheet的配置失败,sheetId:%d,Err:%s", sheet.ExcelSheetId, err.Error()))
  60. return
  61. }
  62. tmpLuckySheetDataInfo := excelServices.LuckySheetData{
  63. Name: sheet.SheetName,
  64. Index: sheet.Sort,
  65. CellData: sheetCellDataMapList[sheet.ExcelSheetId],
  66. Config: luckySheetDataConfig,
  67. }
  68. luckySheet.SheetList = append(luckySheet.SheetList, tmpLuckySheetDataInfo)
  69. }
  70. return
  71. }
  72. // GenerateExcelCustomAnalysisExcel 根据自定义分析的表格data数据生成excel
  73. func GenerateExcelCustomAnalysisExcel(excelInfo *excel.ExcelInfo) (downloadFilePath string, err error, errMsg string) {
  74. luckySheet, err, errMsg := GetCustomAnalysisExcelData(excelInfo)
  75. if err != nil {
  76. return
  77. }
  78. downloadFilePath, err = luckySheet.ToExcel(false)
  79. return
  80. }
  81. // 数据集
  82. type dataStruct struct {
  83. Value float64
  84. Ok bool
  85. }
  86. // HandleEdbSequenceVal 处理日期集和数据集(获取可用的日期、数据集)
  87. func HandleEdbSequenceVal(dateSequenceVal, dataSequenceVal []string) (newDateList []string, newDataList []float64, err error, errMsg string) {
  88. newDateList = make([]string, 0)
  89. newDataList = make([]float64, 0)
  90. dataList := make([]dataStruct, 0)
  91. {
  92. for _, v := range dataSequenceVal {
  93. // 如果没有数据集,那么就过滤
  94. if v == `` {
  95. // 如果开始插入数据了,那么就需要插入不存在值
  96. dataList = append(dataList, dataStruct{
  97. Value: 0,
  98. Ok: false,
  99. })
  100. continue
  101. }
  102. v = strings.Replace(v, ",", "", -1)
  103. v = strings.Replace(v, ",", "", -1)
  104. // 过滤空格
  105. v = strings.Replace(v, " ", "", -1)
  106. var tmpVal float64
  107. if strings.Contains(v, "%") {
  108. // 百分比的数
  109. isPercentage, percentageValue := utils.IsPercentage(v)
  110. if !isPercentage {
  111. dataList = append(dataList, dataStruct{
  112. Value: 0,
  113. Ok: false,
  114. })
  115. continue
  116. }
  117. tmpValDec, tmpErr := decimal.NewFromString(percentageValue)
  118. if tmpErr != nil {
  119. dataList = append(dataList, dataStruct{
  120. Value: 0,
  121. Ok: false,
  122. })
  123. continue
  124. }
  125. tmpVal, _ = tmpValDec.Div(decimal.NewFromFloat(100)).Float64()
  126. } else {
  127. tmpValDec, tmpErr := decimal.NewFromString(v)
  128. if tmpErr != nil {
  129. dataList = append(dataList, dataStruct{
  130. Value: 0,
  131. Ok: false,
  132. })
  133. continue
  134. }
  135. tmpVal, _ = tmpValDec.Float64()
  136. }
  137. dataList = append(dataList, dataStruct{
  138. Value: tmpVal,
  139. Ok: true,
  140. })
  141. }
  142. }
  143. // 日期集
  144. dateList := make([]string, 0)
  145. {
  146. for _, v := range dateSequenceVal {
  147. // 如果没有数据集,那么就过滤
  148. if v == `` {
  149. // 如果开始插入数据了,那么就需要插入不存在值
  150. dateList = append(dateList, "")
  151. continue
  152. }
  153. // 过滤空格
  154. v = strings.Replace(v, " ", "", -1)
  155. t1, tmpErr := dateparse.ParseAny(v)
  156. if tmpErr != nil {
  157. dateList = append(dateList, "")
  158. continue
  159. }
  160. dateList = append(dateList, t1.Format(utils.FormatDate))
  161. }
  162. }
  163. lenData := len(dataList)
  164. lenDate := len(dateList)
  165. // 最小个数
  166. num := lenDate
  167. if num > lenData {
  168. num = lenData
  169. }
  170. for i := 0; i < num; i++ {
  171. date := dateList[i]
  172. tmpData := dataList[i]
  173. // 日期为空、数据为空
  174. if !tmpData.Ok || date == `` {
  175. continue
  176. }
  177. newDateList = append(newDateList, date)
  178. newDataList = append(newDataList, tmpData.Value)
  179. }
  180. return
  181. }
  182. // ResetCustomAnalysisData 数据重置的结构体
  183. type ResetCustomAnalysisData struct {
  184. EdbInfoId int
  185. DateList []string
  186. DataList []float64
  187. }
  188. // Refresh 刷新表格关联的指标信息
  189. func Refresh(excelInfo *excel.ExcelInfo, lang string) (err error, errMsg string, isSendEmail bool) {
  190. isSendEmail = true
  191. list, err := excel.GetAllExcelEdbMappingItemByExcelInfoId(excelInfo.ExcelInfoId)
  192. if err != nil {
  193. errMsg = "获取失败"
  194. err = errors.New("查找所有的mapping失败" + err.Error())
  195. return
  196. }
  197. // 没有关联指标,那么就退出吧
  198. if len(list) <= 0 {
  199. return
  200. }
  201. for k, v := range list {
  202. var tmpCalculateFormula excel.CalculateFormula
  203. err = json.Unmarshal([]byte(v.CalculateFormula), &tmpCalculateFormula)
  204. if err != nil {
  205. errMsg = "获取失败"
  206. err = errors.New(fmt.Sprintf("指标id:%d,公式转换失败,Err:%s", v.EdbInfoId, err.Error()))
  207. return
  208. }
  209. v.DateSequenceStr = tmpCalculateFormula.DateSequenceStr
  210. v.DataSequenceStr = tmpCalculateFormula.DataSequenceStr
  211. list[k] = v
  212. }
  213. luckySheet, err, errMsg := GetCustomAnalysisExcelData(excelInfo)
  214. if err != nil {
  215. err = errors.New(fmt.Sprintf("获取自定义分析Excel数据失败,Err:%s", err.Error()))
  216. return
  217. }
  218. // 获取excel表格数据
  219. xlsxFile, err := luckySheet.GetExcelData(false)
  220. if err != nil {
  221. err = errors.New(fmt.Sprintf("获取excel表格数据,Err:%s", err.Error()))
  222. return
  223. }
  224. //fmt.Println(xlsxFile)
  225. edbInfoIdList := make([]int, 0)
  226. for _, v := range list {
  227. edbInfoIdList = append(edbInfoIdList, v.EdbInfoId)
  228. // 获取对应的日期和数据列表
  229. relDateList, relDataList, tmpErr, tmpErrMsg := getDateAndDataList(v, xlsxFile)
  230. if tmpErr != nil {
  231. err = errors.New(fmt.Sprintf("获取对应的日期和数据列表,Err:%s", tmpErr.Error()))
  232. errMsg = tmpErrMsg
  233. return
  234. }
  235. //fmt.Println(v)
  236. req2 := &ResetCustomAnalysisData{
  237. EdbInfoId: v.EdbInfoId,
  238. DateList: relDateList,
  239. DataList: relDataList,
  240. }
  241. // 调用指标库去更新
  242. reqJson, tmpErr := json.Marshal(req2)
  243. if tmpErr != nil {
  244. err = errors.New(fmt.Sprintf("结构体转对象失败,Err:%s", tmpErr.Error()))
  245. return
  246. }
  247. respItem, tmpErr := data.ResetCustomAnalysisData(string(reqJson), lang)
  248. if tmpErr != nil {
  249. err = errors.New(fmt.Sprintf("调用指标库去更新,Err:%s", tmpErr.Error()))
  250. return
  251. }
  252. if respItem.Ret != 200 {
  253. errMsg = respItem.Msg
  254. err = errors.New(respItem.ErrMsg)
  255. return
  256. }
  257. }
  258. if len(edbInfoIdList) > 0 {
  259. err, _ = data.EdbInfoRefreshAllFromBaseV3(edbInfoIdList, false, true, true)
  260. }
  261. //xlsxFile.Sheet[]
  262. return
  263. }
  264. // getDateAndDataList
  265. // @Description: 获取待刷新的日期和数据
  266. // @author: Roc
  267. // @datetime 2023-12-21 15:21:14
  268. // @param excelEdbMappingItem *excel.ExcelEdbMappingItem
  269. // @param xlsxFile *xlsx.File
  270. // @return newDateList []string
  271. // @return newDataList []float64
  272. // @return err error
  273. // @return errMsg string
  274. func getDateAndDataList(excelEdbMappingItem *excel.ExcelEdbMappingItem, xlsxFile *xlsx.File) (newDateList []string, newDataList []float64, err error, errMsg string) {
  275. var dateList, dataList []string
  276. // 日期序列
  277. {
  278. sheetName, startColumnName, endColumnName, startNum, endNum, isAll, isRow, isColumn, tmpErr := GetSheetStr(excelEdbMappingItem.DateSequenceStr)
  279. if tmpErr != nil {
  280. err = tmpErr
  281. return
  282. }
  283. // 查找sheet页
  284. sheetInfo, ok := xlsxFile.Sheet[sheetName]
  285. if !ok {
  286. errMsg = "找不到" + sheetName
  287. err = errors.New(errMsg)
  288. return
  289. }
  290. startNum = startNum - 1
  291. endNum = endNum - 1
  292. // 选择行的数据
  293. if isRow {
  294. // 因为是选择一行的数据,所以开始行和结束行时一样的
  295. //endNum = startNum - 1
  296. // 开始列名、结束列
  297. var startColumn, endColumn int
  298. if isAll {
  299. // 结束列(其实也就是整列的个数)
  300. endColumn = len(sheetInfo.Cols) - 1
  301. } else {
  302. tmpStartColumn, tmpErr := excelize.ColumnNameToNumber(startColumnName)
  303. if tmpErr != nil {
  304. errMsg = "列名异常:" + startColumnName
  305. err = errors.New(errMsg)
  306. return
  307. }
  308. tmpEndColumn, tmpErr := excelize.ColumnNameToNumber(endColumnName)
  309. if tmpErr != nil {
  310. errMsg = "列名异常:" + endColumnName
  311. err = errors.New(errMsg)
  312. return
  313. }
  314. startColumn = tmpStartColumn - 1
  315. endColumn = tmpEndColumn - 1
  316. }
  317. // 最大列数,如果设置的超过了最大列数,那么结束列就是最大列数
  318. maxCol := len(sheetInfo.Cols)
  319. if endColumn > maxCol {
  320. endColumn = maxCol - 1
  321. }
  322. // 长度固定,避免一直申请内存空间
  323. dateList = make([]string, endColumn-startColumn+1)
  324. i := 0
  325. for currColumn := startColumn; currColumn <= endColumn; currColumn++ {
  326. currCell := sheetInfo.Cell(startNum, currColumn)
  327. if currCell == nil {
  328. errMsg = fmt.Sprintf("第%d列,第%d行数据异常", startColumn, startNum)
  329. err = errors.New(errMsg)
  330. return
  331. }
  332. //dateList = append(dateList, currCell.Value)
  333. dateList[i] = currCell.Value
  334. i++
  335. }
  336. } else if isColumn { // 选择列的数据
  337. if isAll {
  338. // 选择一整列的话,结束行得根据实际情况调整(其实也就是整个sheet有多少行)
  339. endNum = len(sheetInfo.Rows) - 1
  340. }
  341. startColumn, tmpErr := excelize.ColumnNameToNumber(startColumnName)
  342. if tmpErr != nil {
  343. errMsg = "列名异常:" + startColumnName
  344. err = errors.New(errMsg)
  345. return
  346. }
  347. startColumn = startColumn - 1
  348. // 最大行数,如果设置的超过了最大行数,那么结束行就是最大行数
  349. maxRow := len(sheetInfo.Rows)
  350. if endNum > maxRow {
  351. endNum = maxRow - 1
  352. }
  353. // 长度固定,避免一直申请内存空间
  354. dateList = make([]string, endNum-startNum+1)
  355. i := 0
  356. for currRow := startNum; currRow <= endNum; currRow++ {
  357. currCell := sheetInfo.Cell(currRow, startColumn)
  358. if currCell == nil {
  359. errMsg = fmt.Sprintf("第%d列,第%d行数据异常", startColumn, startNum)
  360. err = errors.New(errMsg)
  361. return
  362. }
  363. //dateList = append(dateList, currCell.Value)
  364. dateList[i] = currCell.Value
  365. i++
  366. }
  367. }
  368. }
  369. // 数据序列
  370. {
  371. sheetName, startColumnName, endColumnName, startNum, endNum, isAll, isRow, isColumn, tmpErr := GetSheetStr(excelEdbMappingItem.DataSequenceStr)
  372. if tmpErr != nil {
  373. err = tmpErr
  374. return
  375. }
  376. // 查找sheet页
  377. sheetInfo, ok := xlsxFile.Sheet[sheetName]
  378. if !ok {
  379. errMsg = "找不到" + sheetName
  380. err = errors.New(errMsg)
  381. return
  382. }
  383. startNum = startNum - 1
  384. endNum = endNum - 1
  385. // 选择行的数据
  386. if isRow {
  387. // 开始列名、结束列
  388. var startColumn, endColumn int
  389. if isAll {
  390. // 结束列(其实也就是整列的个数)
  391. endColumn = len(sheetInfo.Cols) - 1
  392. } else {
  393. tmpStartColumn, tmpErr := excelize.ColumnNameToNumber(startColumnName)
  394. if tmpErr != nil {
  395. errMsg = "列名异常:" + startColumnName
  396. err = errors.New(errMsg)
  397. return
  398. }
  399. tmpEndColumn, tmpErr := excelize.ColumnNameToNumber(endColumnName)
  400. if tmpErr != nil {
  401. errMsg = "列名异常:" + endColumnName
  402. err = errors.New(errMsg)
  403. return
  404. }
  405. startColumn = tmpStartColumn - 1
  406. endColumn = tmpEndColumn - 1
  407. }
  408. // 最大列数,如果设置的超过了最大列数,那么结束列就是最大列数
  409. maxCol := len(sheetInfo.Cols)
  410. if endColumn > maxCol {
  411. endColumn = maxCol - 1
  412. }
  413. // 长度固定,避免一直申请内存空间
  414. dataList = make([]string, endColumn-startColumn+1)
  415. i := 0
  416. for currColumn := startColumn; currColumn <= endColumn; currColumn++ {
  417. currCell := sheetInfo.Cell(startNum, currColumn)
  418. if currCell == nil {
  419. errMsg = fmt.Sprintf("第%d列,第%d行数据异常", startColumn, startNum)
  420. err = errors.New(errMsg)
  421. return
  422. }
  423. //dataList = append(dataList, currCell.Value)
  424. dataList[i] = currCell.Value
  425. i++
  426. }
  427. } else if isColumn { // 选择列的数据
  428. if isAll {
  429. // 选择一整列的话,结束行得根据实际情况调整(其实也就是整个sheet有多少行)
  430. endNum = len(sheetInfo.Rows) - 1
  431. }
  432. startColumn, tmpErr := excelize.ColumnNameToNumber(startColumnName)
  433. if tmpErr != nil {
  434. errMsg = "列名异常:" + startColumnName
  435. err = errors.New(errMsg)
  436. return
  437. }
  438. startColumn = startColumn - 1
  439. // 最大行数,如果设置的超过了最大行数,那么结束行就是最大行数
  440. maxRow := len(sheetInfo.Rows)
  441. if endNum > maxRow {
  442. endNum = maxRow - 1
  443. }
  444. // 长度固定,避免一直申请内存空间
  445. dataList = make([]string, endNum-startNum+1)
  446. i := 0
  447. for currRow := startNum; currRow <= endNum; currRow++ {
  448. currCell := sheetInfo.Cell(currRow, startColumn)
  449. if currCell == nil {
  450. errMsg = fmt.Sprintf("第%d列,第%d行数据异常", startColumn, startNum)
  451. err = errors.New(errMsg)
  452. return
  453. }
  454. //dataList = append(dataList, currCell.Value)
  455. dataList[i] = currCell.Value
  456. i++
  457. }
  458. }
  459. }
  460. //fmt.Println(dateList, dataList)
  461. //fmt.Println("日期序列结束")
  462. // 将excel中的日期、数据系列处理
  463. newDateList, newDataList, err, errMsg = HandleEdbSequenceVal(dateList, dataList)
  464. return
  465. }
  466. // GetSheetStr
  467. // @return sheetName string 用户选择的sheet名称
  468. // @return startColumnName string 用户选择的开始列名称
  469. // @return endColumnName string 用户选择的结束列名称
  470. // @return startNum int 用户选择的开始列单元格位置
  471. // @return endNum int 用户选择的结束列单元格位置
  472. // @return isAll bool 是否选择整行/列数据
  473. // @return isRow bool 是否选择行数据
  474. // @return isColumn bool 是否选择列数据
  475. func GetSheetStr(sequenceStr string) (sheetName, startColumnName, endColumnName string, startNum, endNum int, isAll, isRow, isColumn bool, err error) {
  476. // 找出sheetName
  477. tmpList := strings.Split(sequenceStr, "!")
  478. if len(tmpList) != 2 {
  479. err = errors.New("错误的公式,查找sheet异常:" + sequenceStr)
  480. return
  481. }
  482. sheetName = tmpList[0]
  483. // 分离开始/结束单元格
  484. tmpList = strings.Split(tmpList[1], ":")
  485. if len(tmpList) != 2 {
  486. err = errors.New("错误的公式,查找开始/结束单元格异常:" + sequenceStr)
  487. return
  488. }
  489. startList := strings.Split(tmpList[0], "$")
  490. endList := strings.Split(tmpList[1], "$")
  491. lenList := len(startList)
  492. if lenList != len(endList) {
  493. err = errors.New("错误的公式,开始与结束单元格异常:" + sequenceStr)
  494. return
  495. }
  496. if lenList != 3 && lenList != 2 {
  497. err = errors.New("错误的公式:" + sequenceStr)
  498. return
  499. }
  500. startColumnName = startList[1]
  501. endColumnName = endList[1]
  502. // 长度为2的话,那说明是整行或整列
  503. if lenList == 2 {
  504. isAll = true
  505. startDeci, tmpErr1 := decimal.NewFromString(startList[1])
  506. endDeci, tmpErr2 := decimal.NewFromString(endList[1])
  507. if tmpErr1 == nil && tmpErr2 == nil {
  508. isRow = true // 正常转换的话,那么就是整行
  509. startNum = int(startDeci.IntPart())
  510. endNum = int(endDeci.IntPart())
  511. startColumnName = ``
  512. endColumnName = ``
  513. return
  514. }
  515. if tmpErr1 == nil || tmpErr2 == nil {
  516. err = errors.New("错误的公式2:" + sequenceStr)
  517. return
  518. }
  519. // 如果不能转成数字,那么就是整列
  520. isColumn = true
  521. return
  522. }
  523. // 确定行
  524. startDeci, tmpErr1 := decimal.NewFromString(startList[2])
  525. endDeci, tmpErr2 := decimal.NewFromString(endList[2])
  526. if tmpErr1 != nil && tmpErr1 != tmpErr2 {
  527. err = errors.New("错误的公式3:" + sequenceStr)
  528. return
  529. }
  530. startNum = int(startDeci.IntPart())
  531. endNum = int(endDeci.IntPart())
  532. if startColumnName != endColumnName && startNum != endNum {
  533. err = errors.New("选区不允许跨行或者跨列")
  534. }
  535. if startColumnName == endColumnName {
  536. isColumn = true // 列数据
  537. } else {
  538. isRow = true // 行数据
  539. }
  540. return
  541. }