index.go 33 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132
  1. package ai_predict_model
  2. import (
  3. "encoding/json"
  4. "eta/eta_api/controllers"
  5. "eta/eta_api/models"
  6. aiPredictModel "eta/eta_api/models/ai_predict_model"
  7. "eta/eta_api/models/data_manage"
  8. dataSourceModel "eta/eta_api/models/data_source"
  9. "eta/eta_api/models/system"
  10. "eta/eta_api/services"
  11. "eta/eta_api/services/data"
  12. "eta/eta_api/services/elastic"
  13. "eta/eta_api/utils"
  14. "fmt"
  15. "github.com/rdlucklib/rdluck_tools/paging"
  16. "github.com/tealeg/xlsx"
  17. "os"
  18. "strconv"
  19. "strings"
  20. "time"
  21. )
  22. // AiPredictModelIndexController AI预测模型标的
  23. type AiPredictModelIndexController struct {
  24. controllers.BaseAuthController
  25. }
  26. // List
  27. // @Title 标的列表
  28. // @Description 标的列表
  29. // @Param PageSize query int true "每页数据条数"
  30. // @Param CurrentIndex query int true "当前页页码,从1开始"
  31. // @Param ClassifyId query int false "分类id"
  32. // @Param IndexId query int false "模型标的ID"
  33. // @Param Keyword query string false "搜索关键词"
  34. // @Success 200 {object} data_manage.ChartListResp
  35. // @router /index/list [get]
  36. func (this *AiPredictModelIndexController) List() {
  37. br := new(models.BaseResponse).Init()
  38. defer func() {
  39. this.Data["json"] = br
  40. this.ServeJSON()
  41. }()
  42. sysUser := this.SysUser
  43. if sysUser == nil {
  44. br.Msg = "请登录"
  45. br.ErrMsg = "请登录,SysUser Is Empty"
  46. br.Ret = 408
  47. return
  48. }
  49. pageSize, _ := this.GetInt("PageSize")
  50. currentIndex, _ := this.GetInt("CurrentIndex")
  51. classifyId, _ := this.GetInt("ClassifyId")
  52. indexId, _ := this.GetInt("IndexId")
  53. keyword := this.GetString("KeyWord")
  54. if keyword == "" {
  55. keyword = this.GetString("Keyword")
  56. }
  57. keyword = strings.TrimSpace(keyword)
  58. resp := new(aiPredictModel.AiPredictModelIndexPageListResp)
  59. // 分页
  60. var startSize int
  61. if pageSize <= 0 {
  62. pageSize = utils.PageSize20
  63. }
  64. if currentIndex <= 0 {
  65. currentIndex = 1
  66. }
  67. startSize = paging.StartIndex(currentIndex, pageSize)
  68. // 分类
  69. classifyIdName := make(map[int]string)
  70. {
  71. classifyOb := new(aiPredictModel.AiPredictModelClassify)
  72. list, e := classifyOb.GetItemsByCondition("", make([]interface{}, 0), []string{}, "")
  73. if e != nil {
  74. br.Msg = "获取失败"
  75. br.ErrMsg = fmt.Sprintf("获取分类失败, %v", e)
  76. return
  77. }
  78. for _, v := range list {
  79. classifyIdName[v.AiPredictModelClassifyId] = v.ClassifyName
  80. }
  81. }
  82. // 筛选条件
  83. highlightMap := make(map[int]string)
  84. indexOb := new(aiPredictModel.AiPredictModelIndex)
  85. var cond string
  86. var pars []interface{}
  87. {
  88. if indexId > 0 {
  89. cond += fmt.Sprintf(" AND %s = ?", indexOb.Cols().PrimaryId)
  90. pars = append(pars, indexId)
  91. }
  92. if classifyId > 0 {
  93. cond += fmt.Sprintf(" AND %s = ?", indexOb.Cols().ClassifyId)
  94. pars = append(pars, classifyId)
  95. }
  96. //if keyword != "" {
  97. // cond += fmt.Sprintf(" AND %s LIKE ?", indexOb.Cols().IndexName)
  98. // pars = append(pars, fmt.Sprint("%", keyword, "%"))
  99. //}
  100. // 有关键词从es中搜索
  101. if keyword != "" {
  102. _, list, e := elastic.SearchDataSourceIndex(utils.EsDataSourceIndexName, keyword, utils.DATA_SOURCE_AI_PREDICT_MODEL, 0, []int{}, []int{}, []string{}, startSize, pageSize)
  103. if e != nil {
  104. br.Msg = "获取失败"
  105. br.ErrMsg = fmt.Sprintf("ES-搜索手工指标列表失败, %v", e)
  106. return
  107. }
  108. if len(list) == 0 {
  109. resp.List = make([]*aiPredictModel.AiPredictModelIndexItem, 0)
  110. br.Ret = 200
  111. br.Success = true
  112. br.Msg = "获取成功"
  113. br.Data = resp
  114. return
  115. }
  116. var ids []int
  117. for _, v := range list {
  118. ids = append(ids, v.PrimaryId)
  119. highlightMap[v.PrimaryId] = v.SearchText
  120. }
  121. cond += fmt.Sprintf(` AND %s IN (%s)`, indexOb.Cols().PrimaryId, utils.GetOrmInReplace(len(ids)))
  122. pars = append(pars, ids)
  123. }
  124. }
  125. // 获取列表
  126. total, e := indexOb.GetCountByCondition(cond, pars)
  127. if e != nil {
  128. br.Msg = "获取失败"
  129. br.ErrMsg = fmt.Sprintf("获取标的总数失败, %v", e)
  130. return
  131. }
  132. list, e := indexOb.GetPageItemsByCondition(cond, pars, []string{}, "", startSize, pageSize)
  133. if e != nil {
  134. br.Msg = "获取失败"
  135. br.ErrMsg = fmt.Sprintf("获取分页列表失败, %v", e)
  136. return
  137. }
  138. pageList := make([]*aiPredictModel.AiPredictModelIndexItem, 0)
  139. for _, v := range list {
  140. t := v.Format2Item()
  141. t.ClassifyName = classifyIdName[v.ClassifyId]
  142. // 搜索高亮
  143. t.SearchText = v.IndexName
  144. s := highlightMap[v.AiPredictModelIndexId]
  145. if s != "" {
  146. t.SearchText = s
  147. }
  148. pageList = append(pageList, t)
  149. }
  150. page := paging.GetPaging(currentIndex, pageSize, total)
  151. resp.Paging = page
  152. resp.List = pageList
  153. br.Data = resp
  154. br.Ret = 200
  155. br.Success = true
  156. br.Msg = "获取成功"
  157. }
  158. // Import
  159. // @Title 导入标的和数据
  160. // @Description 导入标的和数据
  161. // @Param IndexFile query file true "标的文件"
  162. // @Success 200 Ret=200 录入成功
  163. // @router /index/import [post]
  164. func (this *AiPredictModelIndexController) Import() {
  165. br := new(models.BaseResponse).Init()
  166. defer func() {
  167. if br.ErrMsg == "" {
  168. br.IsSendEmail = false
  169. }
  170. this.Data["json"] = br
  171. this.ServeJSON()
  172. }()
  173. sysUser := this.SysUser
  174. if sysUser == nil {
  175. br.Msg = "请登录"
  176. br.ErrMsg = "请登录,SysUser Is Empty"
  177. br.Ret = 408
  178. return
  179. }
  180. file, _, e := this.GetFile("IndexFile")
  181. if e != nil {
  182. br.Msg = "导入失败"
  183. br.ErrMsg = fmt.Sprintf("获取文件失败, %v", e)
  184. return
  185. }
  186. path := "./static/ai_predict_model_temp_" + time.Now().Format(utils.FormatDateTimeUnSpace) + ".xlsx"
  187. defer func() {
  188. _ = file.Close()
  189. _ = os.Remove(path)
  190. }()
  191. if e = this.SaveToFile("IndexFile", path); e != nil {
  192. br.Msg = "导入失败"
  193. br.ErrMsg = fmt.Sprintf("保存文件失败, %v", e)
  194. return
  195. }
  196. xlFile, e := xlsx.OpenFile(path)
  197. if e != nil {
  198. br.Msg = "导入失败"
  199. br.ErrMsg = fmt.Sprintf("打开excel文件失败, %v", e)
  200. return
  201. }
  202. // 获取分类和用户,遍历时校验
  203. classifyNameId := make(map[string]int)
  204. adminNameId := make(map[string]int)
  205. {
  206. classifyOb := new(aiPredictModel.AiPredictModelClassify)
  207. classifyCond := fmt.Sprintf(` AND %s = ?`, classifyOb.Cols().ParentId)
  208. classifyPars := make([]interface{}, 0)
  209. classifyPars = append(classifyPars, 0) // 只取一级分类(临时过渡方案,业务端只会加一级)
  210. classifies, e := classifyOb.GetItemsByCondition(classifyCond, classifyPars, []string{}, "")
  211. if e != nil {
  212. br.Msg = "导入失败"
  213. br.ErrMsg = fmt.Sprintf("获取分类失败, %v", e)
  214. return
  215. }
  216. for _, v := range classifies {
  217. classifyNameId[v.ClassifyName] = v.AiPredictModelClassifyId
  218. }
  219. admins, e := system.GetSysAdminList(``, make([]interface{}, 0), []string{}, "")
  220. if e != nil {
  221. br.Msg = "导入失败"
  222. br.ErrMsg = fmt.Sprintf("获取用户失败, %v", e)
  223. return
  224. }
  225. for _, v := range admins {
  226. adminNameId[v.RealName] = v.AdminId
  227. }
  228. }
  229. // 遍历sheet页
  230. // 列表页:预测标的|分类|模型框架|创建人|预测日期|预测值|预测频度|方向准确率|绝对偏差
  231. type ImportDataColKey struct {
  232. IndexName string
  233. ColKey int
  234. DataDate time.Time
  235. }
  236. imports := make(map[string]*aiPredictModel.AiPredictModelImportData)
  237. importsData := make(map[string]map[time.Time]*aiPredictModel.AiPredictModelData)
  238. importsDailyData := make(map[string]map[time.Time]*aiPredictModel.AiPredictModelData)
  239. for sheetKey, sheet := range xlFile.Sheets {
  240. maxRow := sheet.MaxRow
  241. // 列表页
  242. if sheetKey == 0 {
  243. for i := 0; i < maxRow; i++ {
  244. // 忽略首行标题
  245. if i < 1 {
  246. continue
  247. }
  248. row := sheet.Row(i)
  249. cells := row.Cells
  250. if len(cells) < 9 {
  251. continue
  252. }
  253. // 标的名称
  254. indexName := strings.TrimSpace(cells[0].String())
  255. if indexName == "" {
  256. continue
  257. }
  258. if imports[indexName] == nil {
  259. imports[indexName] = new(aiPredictModel.AiPredictModelImportData)
  260. imports[indexName].Index = new(aiPredictModel.AiPredictModelIndex)
  261. imports[indexName].Data = make([]*aiPredictModel.AiPredictModelData, 0)
  262. }
  263. imports[indexName].Index.IndexName = indexName
  264. imports[indexName].Index.CreateTime = time.Now()
  265. imports[indexName].Index.ModifyTime = time.Now()
  266. // 分类
  267. classifyName := strings.TrimSpace(cells[1].String())
  268. if classifyNameId[classifyName] <= 0 {
  269. br.Msg = fmt.Sprintf("分类:%s不存在", classifyName)
  270. return
  271. }
  272. imports[indexName].Index.ClassifyId = classifyNameId[classifyName]
  273. // 创建人
  274. adminName := strings.TrimSpace(cells[3].String())
  275. if adminNameId[adminName] <= 0 {
  276. br.Msg = fmt.Sprintf("创建人:%s不存在", adminName)
  277. return
  278. }
  279. imports[indexName].Index.SysUserId = adminNameId[adminName]
  280. imports[indexName].Index.SysUserRealName = adminName
  281. // 其余信息
  282. imports[indexName].Index.ModelFramework = strings.TrimSpace(cells[2].String())
  283. strDate := strings.TrimSpace(cells[4].String())
  284. predictDate, _ := utils.GetExcelDate(strDate)
  285. imports[indexName].Index.PredictDate = predictDate
  286. strVal := strings.TrimSpace(cells[5].String())
  287. if strVal == "" {
  288. continue
  289. }
  290. predictVal, _ := strconv.ParseFloat(strVal, 64)
  291. imports[indexName].Index.PredictValue = predictVal
  292. imports[indexName].Index.PredictFrequency = strings.TrimSpace(cells[6].String())
  293. imports[indexName].Index.DirectionAccuracy = strings.TrimSpace(cells[7].String())
  294. imports[indexName].Index.AbsoluteDeviation = strings.TrimSpace(cells[8].String())
  295. }
  296. }
  297. // 月度数据页
  298. if sheetKey == 1 {
  299. // 每五列为一个指标的数据
  300. colKeys := make(map[int]*ImportDataColKey) // 每一列对应的指标名称以及对应的字段序号
  301. for i := 0; i < maxRow; i++ {
  302. // 首行为指标名称
  303. if i == 0 {
  304. nameCol := 0
  305. row := sheet.Row(i)
  306. for ck, cell := range row.Cells {
  307. nameCol += 1
  308. if nameCol > 5 {
  309. nameCol = 1
  310. }
  311. if nameCol == 1 {
  312. // nameCol=1时为指标/数据行则为日期
  313. indexName := strings.TrimSpace(cell.String())
  314. if indexName == "" {
  315. continue
  316. }
  317. importsData[indexName] = make(map[time.Time]*aiPredictModel.AiPredictModelData)
  318. colKeys[ck] = &ImportDataColKey{
  319. ColKey: 1,
  320. IndexName: indexName,
  321. }
  322. // 后面四列分别对应: 实际值|预测值|方向|偏差率, 这里直接加无须考虑是否会越界
  323. colKeys[ck+1] = &ImportDataColKey{
  324. ColKey: 2,
  325. IndexName: indexName,
  326. }
  327. colKeys[ck+2] = &ImportDataColKey{
  328. ColKey: 3,
  329. IndexName: indexName,
  330. }
  331. colKeys[ck+3] = &ImportDataColKey{
  332. ColKey: 4,
  333. IndexName: indexName,
  334. }
  335. colKeys[ck+4] = &ImportDataColKey{
  336. ColKey: 5,
  337. IndexName: indexName,
  338. }
  339. continue
  340. }
  341. }
  342. continue
  343. }
  344. // 第二行为标题,跳过
  345. if i == 1 {
  346. continue
  347. }
  348. // 剩余为数据行
  349. row := sheet.Row(i)
  350. for ck, cell := range row.Cells {
  351. if colKeys[ck] == nil {
  352. continue
  353. }
  354. if colKeys[ck].IndexName == "" {
  355. continue
  356. }
  357. switch colKeys[ck].ColKey {
  358. case 1:
  359. // 日期列
  360. strDate := strings.TrimSpace(cell.String())
  361. dataDate, _ := utils.GetExcelDate(strDate)
  362. if dataDate.IsZero() {
  363. continue
  364. }
  365. colKeys[ck].DataDate = dataDate
  366. colKeys[ck+1].DataDate = dataDate
  367. colKeys[ck+2].DataDate = dataDate
  368. colKeys[ck+3].DataDate = dataDate
  369. colKeys[ck+4].DataDate = dataDate
  370. importRow := imports[colKeys[ck].IndexName]
  371. if importRow == nil {
  372. continue
  373. }
  374. // 新增当前日期数据
  375. importsData[colKeys[ck].IndexName][dataDate] = new(aiPredictModel.AiPredictModelData)
  376. importsData[colKeys[ck].IndexName][dataDate].DataTime = dataDate
  377. importsData[colKeys[ck].IndexName][dataDate].CreateTime = time.Now()
  378. importsData[colKeys[ck].IndexName][dataDate].ModifyTime = time.Now()
  379. importsData[colKeys[ck].IndexName][dataDate].Source = aiPredictModel.ModelDataSourceMonthly
  380. case 2, 3:
  381. // 实际值和预测值, 可能为空
  382. dataDate := colKeys[ck].DataDate
  383. if importsData[colKeys[ck].IndexName][dataDate] == nil {
  384. continue
  385. }
  386. strVal := strings.TrimSpace(cell.String())
  387. if strVal == "" {
  388. continue
  389. }
  390. val, _ := strconv.ParseFloat(strVal, 64)
  391. if colKeys[ck].ColKey == 2 {
  392. importsData[colKeys[ck].IndexName][dataDate].Value.Valid = true
  393. importsData[colKeys[ck].IndexName][dataDate].Value.Float64 = val
  394. } else {
  395. importsData[colKeys[ck].IndexName][dataDate].PredictValue.Valid = true
  396. importsData[colKeys[ck].IndexName][dataDate].PredictValue.Float64 = val
  397. }
  398. case 4, 5:
  399. // 方向/偏差率
  400. dataDate := colKeys[ck].DataDate
  401. if importsData[colKeys[ck].IndexName][dataDate] == nil {
  402. continue
  403. }
  404. str := strings.TrimSpace(cell.String())
  405. if str == "" {
  406. continue
  407. }
  408. if colKeys[ck].ColKey == 4 {
  409. importsData[colKeys[ck].IndexName][dataDate].Direction = str
  410. } else {
  411. importsData[colKeys[ck].IndexName][dataDate].DeviationRate = str
  412. }
  413. default:
  414. continue
  415. }
  416. }
  417. }
  418. }
  419. // 日度数据页
  420. if sheetKey == 2 {
  421. // 每3列为一个指标的数据
  422. colKeys := make(map[int]*ImportDataColKey) // 每一列对应的指标名称以及对应的字段序号
  423. for i := 0; i < maxRow; i++ {
  424. // 首行为指标名称
  425. if i == 0 {
  426. nameCol := 0
  427. row := sheet.Row(i)
  428. for ck, cell := range row.Cells {
  429. nameCol += 1
  430. if nameCol > 3 {
  431. nameCol = 1
  432. }
  433. if nameCol == 1 {
  434. // nameCol=1时为指标/数据行则为日期
  435. indexName := strings.TrimSpace(cell.String())
  436. if indexName == "" {
  437. continue
  438. }
  439. importsDailyData[indexName] = make(map[time.Time]*aiPredictModel.AiPredictModelData)
  440. colKeys[ck] = &ImportDataColKey{
  441. ColKey: 1,
  442. IndexName: indexName,
  443. }
  444. // 后面两列分别对应: 实际值|预测值, 这里直接加无须考虑是否会越界
  445. colKeys[ck+1] = &ImportDataColKey{
  446. ColKey: 2,
  447. IndexName: indexName,
  448. }
  449. colKeys[ck+2] = &ImportDataColKey{
  450. ColKey: 3,
  451. IndexName: indexName,
  452. }
  453. continue
  454. }
  455. }
  456. continue
  457. }
  458. // 第二行为标题,遇到"预测值"单元格,需要取出其中的值作为预测图例名称
  459. if i == 1 {
  460. row := sheet.Row(i)
  461. for ck, cell := range row.Cells {
  462. if colKeys[ck] == nil {
  463. continue
  464. }
  465. if colKeys[ck].IndexName == "" {
  466. continue
  467. }
  468. if colKeys[ck].ColKey != 3 {
  469. continue
  470. }
  471. if imports[colKeys[ck].IndexName] != nil && imports[colKeys[ck].IndexName].Index != nil {
  472. var extraConfig aiPredictModel.AiPredictModelIndexExtraConfig
  473. extraConfig.DailyChart.PredictLegendName = strings.TrimSpace(cell.String())
  474. b, _ := json.Marshal(extraConfig)
  475. imports[colKeys[ck].IndexName].Index.ExtraConfig = string(b)
  476. }
  477. }
  478. continue
  479. }
  480. // 剩余为数据行
  481. row := sheet.Row(i)
  482. for ck, cell := range row.Cells {
  483. if colKeys[ck] == nil {
  484. continue
  485. }
  486. if colKeys[ck].IndexName == "" {
  487. continue
  488. }
  489. switch colKeys[ck].ColKey {
  490. case 1:
  491. // 日期列
  492. strDate := strings.TrimSpace(cell.String())
  493. dataDate, _ := utils.GetExcelDate(strDate)
  494. if dataDate.IsZero() {
  495. continue
  496. }
  497. colKeys[ck].DataDate = dataDate
  498. colKeys[ck+1].DataDate = dataDate
  499. colKeys[ck+2].DataDate = dataDate
  500. importRow := imports[colKeys[ck].IndexName]
  501. if importRow == nil {
  502. continue
  503. }
  504. // 新增当前日期数据
  505. importsDailyData[colKeys[ck].IndexName][dataDate] = new(aiPredictModel.AiPredictModelData)
  506. importsDailyData[colKeys[ck].IndexName][dataDate].DataTime = dataDate
  507. importsDailyData[colKeys[ck].IndexName][dataDate].CreateTime = time.Now()
  508. importsDailyData[colKeys[ck].IndexName][dataDate].ModifyTime = time.Now()
  509. importsDailyData[colKeys[ck].IndexName][dataDate].Source = aiPredictModel.ModelDataSourceDaily
  510. case 2, 3:
  511. // 实际值和预测值, 可能为空
  512. dataDate := colKeys[ck].DataDate
  513. if importsDailyData[colKeys[ck].IndexName][dataDate] == nil {
  514. continue
  515. }
  516. strVal := strings.TrimSpace(cell.String())
  517. if strVal == "" {
  518. continue
  519. }
  520. val, _ := strconv.ParseFloat(strVal, 64)
  521. if colKeys[ck].ColKey == 2 {
  522. importsDailyData[colKeys[ck].IndexName][dataDate].Value.Valid = true
  523. importsDailyData[colKeys[ck].IndexName][dataDate].Value.Float64 = val
  524. } else {
  525. importsDailyData[colKeys[ck].IndexName][dataDate].PredictValue.Valid = true
  526. importsDailyData[colKeys[ck].IndexName][dataDate].PredictValue.Float64 = val
  527. }
  528. default:
  529. continue
  530. }
  531. }
  532. }
  533. }
  534. }
  535. for indexName, v := range importsData {
  536. if imports[indexName] == nil {
  537. continue
  538. }
  539. for _, dateData := range v {
  540. imports[indexName].Data = append(imports[indexName].Data, dateData)
  541. }
  542. }
  543. for indexName, v := range importsDailyData {
  544. if imports[indexName] == nil {
  545. continue
  546. }
  547. for _, dateData := range v {
  548. imports[indexName].Data = append(imports[indexName].Data, dateData)
  549. }
  550. }
  551. importIndexes := make([]*aiPredictModel.AiPredictModelImportData, 0)
  552. for _, v := range imports {
  553. importIndexes = append(importIndexes, v)
  554. }
  555. // 导入指标
  556. if e = services.ImportAiPredictModelIndexAndData(importIndexes, sysUser.AdminId, sysUser.RealName); e != nil {
  557. br.Msg = "操作失败"
  558. br.ErrMsg = fmt.Sprintf("导入指标数据失败, %v", e)
  559. return
  560. }
  561. // 写入es
  562. go func() {
  563. for _, v := range importIndexes {
  564. indexItem := new(dataSourceModel.SearchDataSource)
  565. indexItem.PrimaryId = v.Index.AiPredictModelIndexId
  566. indexItem.IndexName = v.Index.IndexName
  567. indexItem.IndexCode = v.Index.IndexCode
  568. indexItem.ClassifyId = v.Index.ClassifyId
  569. indexItem.Source = utils.DATA_SOURCE_AI_PREDICT_MODEL
  570. indexItem.SourceName = "AI预测模型"
  571. indexItem.CreateTime = utils.TimeTransferString(utils.FormatDateTime, v.Index.CreateTime)
  572. indexItem.ModifyTime = utils.TimeTransferString(utils.FormatDateTime, v.Index.ModifyTime)
  573. docId := fmt.Sprintf("%d-%d", indexItem.Source, indexItem.PrimaryId)
  574. if e := elastic.EsAddOrEditDataSourceIndex(utils.EsDataSourceIndexName, docId, indexItem); e != nil {
  575. utils.FileLog.Info("AI预测模型-写入es失败, %v", e)
  576. continue
  577. }
  578. }
  579. }()
  580. br.Ret = 200
  581. br.Success = true
  582. br.Msg = "操作成功"
  583. }
  584. // Detail
  585. // @Title 标的详情
  586. // @Description 标的详情
  587. // @Param IndexId query int true "标的ID"
  588. // @Success 200 {object} data_manage.ChartListResp
  589. // @router /index/detail [get]
  590. func (this *AiPredictModelIndexController) Detail() {
  591. br := new(models.BaseResponse).Init()
  592. defer func() {
  593. if br.ErrMsg == "" {
  594. br.IsSendEmail = false
  595. }
  596. this.Data["json"] = br
  597. this.ServeJSON()
  598. }()
  599. sysUser := this.SysUser
  600. if sysUser == nil {
  601. br.Msg = "请登录"
  602. br.ErrMsg = "请登录,SysUser Is Empty"
  603. br.Ret = 408
  604. return
  605. }
  606. indexId, _ := this.GetInt("IndexId")
  607. if indexId <= 0 {
  608. br.Msg = "参数有误"
  609. br.ErrMsg = fmt.Sprintf("参数有误, IndexId: %d", indexId)
  610. return
  611. }
  612. resp := new(aiPredictModel.AiPredictModelDetailResp)
  613. indexOb := new(aiPredictModel.AiPredictModelIndex)
  614. indexItem, e := indexOb.GetItemById(indexId)
  615. if e != nil {
  616. if utils.IsErrNoRow(e) {
  617. br.Msg = "标的已被删除,请刷新页面"
  618. return
  619. }
  620. br.Msg = "获取失败"
  621. br.ErrMsg = fmt.Sprintf("获取标的失败, %v", e)
  622. return
  623. }
  624. // 获取标的数据
  625. monthData, dailyData := make([]*aiPredictModel.AiPredictModelData, 0), make([]*aiPredictModel.AiPredictModelData, 0)
  626. {
  627. tableData := make([]*aiPredictModel.AiPredictModelDataItem, 0)
  628. dataOb := new(aiPredictModel.AiPredictModelData)
  629. dataCond := fmt.Sprintf(` AND %s = ?`, dataOb.Cols().IndexCode)
  630. dataPars := make([]interface{}, 0)
  631. dataPars = append(dataPars, indexItem.IndexCode)
  632. list, e := dataOb.GetItemsByCondition(dataCond, dataPars, []string{}, fmt.Sprintf("%s DESC", dataOb.Cols().DataTime))
  633. if e != nil {
  634. br.Msg = "获取失败"
  635. br.ErrMsg = fmt.Sprintf("获取标的数据失败, %v", e)
  636. return
  637. }
  638. // tableData取月度数据,最多显示10条
  639. count, limit := 0, 10
  640. for _, v := range list {
  641. // 日度数据
  642. if v.Source == aiPredictModel.ModelDataSourceDaily {
  643. dailyData = append(dailyData, v)
  644. continue
  645. }
  646. // 月度数据
  647. if count < limit {
  648. tableData = append(tableData, v.Format2Item())
  649. count += 1
  650. }
  651. monthData = append(monthData, v)
  652. }
  653. resp.TableData = tableData
  654. }
  655. // 月度图表
  656. if len(monthData) > 0 {
  657. chartDetail, e := services.GetAiPredictChartDetailByData(indexItem, monthData, aiPredictModel.ModelDataSourceMonthly)
  658. if e != nil {
  659. br.Msg = "获取失败"
  660. br.ErrMsg = fmt.Sprintf("获取月度图表失败, %v", e)
  661. return
  662. }
  663. resp.ChartView = chartDetail
  664. }
  665. // 日度图表
  666. if len(dailyData) > 0 {
  667. dailyChartDetail, e := services.GetAiPredictChartDetailByData(indexItem, dailyData, aiPredictModel.ModelDataSourceDaily)
  668. if e != nil {
  669. br.Msg = "获取失败"
  670. br.ErrMsg = fmt.Sprintf("获取日度图表失败, %v", e)
  671. return
  672. }
  673. resp.DailyChartView = dailyChartDetail
  674. }
  675. br.Data = resp
  676. br.Ret = 200
  677. br.Success = true
  678. br.Msg = "获取成功"
  679. }
  680. // Save
  681. // @Title 保存标的
  682. // @Description 保存标的
  683. // @Param request body aiPredictModel.AiPredictModelIndexSaveReq true "type json string"
  684. // @Success 200 Ret=200 保存成功
  685. // @router /index/save [post]
  686. func (this *AiPredictModelIndexController) Save() {
  687. br := new(models.BaseResponse).Init()
  688. defer func() {
  689. if br.ErrMsg == "" {
  690. br.IsSendEmail = false
  691. }
  692. this.Data["json"] = br
  693. this.ServeJSON()
  694. }()
  695. sysUser := this.SysUser
  696. if sysUser == nil {
  697. br.Msg = "请登录"
  698. br.ErrMsg = "请登录,SysUser Is Empty"
  699. br.Ret = 408
  700. return
  701. }
  702. var req aiPredictModel.AiPredictModelIndexSaveReq
  703. if e := json.Unmarshal(this.Ctx.Input.RequestBody, &req); e != nil {
  704. br.Msg = "参数解析异常"
  705. br.ErrMsg = fmt.Sprintf("参数解析异常, %v", e)
  706. return
  707. }
  708. if req.IndexId < 0 {
  709. br.Msg = "参数有误"
  710. br.ErrMsg = fmt.Sprintf("标的ID有误, IndexId: %d", req.IndexId)
  711. return
  712. }
  713. indexOb := new(aiPredictModel.AiPredictModelIndex)
  714. indexItem, e := indexOb.GetItemById(req.IndexId)
  715. if e != nil {
  716. if utils.IsErrNoRow(e) {
  717. br.Msg = "标的已被删除,请刷新页面"
  718. return
  719. }
  720. br.Msg = "操作失败"
  721. br.ErrMsg = fmt.Sprintf("获取标的失败, %v", e)
  722. return
  723. }
  724. var extraConfig aiPredictModel.AiPredictModelIndexExtraConfig
  725. if indexItem.ExtraConfig != "" {
  726. if e = json.Unmarshal([]byte(indexItem.ExtraConfig), &extraConfig); e != nil {
  727. br.Msg = "操作失败"
  728. br.ErrMsg = fmt.Sprintf("标的配置解析失败, %v", e)
  729. return
  730. }
  731. }
  732. if req.MonthlyChart != nil {
  733. extraConfig.MonthlyChart.LeftMin = req.MonthlyChart.LeftMin
  734. extraConfig.MonthlyChart.LeftMax = req.MonthlyChart.LeftMax
  735. extraConfig.MonthlyChart.Unit = req.MonthlyChart.Unit
  736. }
  737. if req.DailyChart != nil {
  738. extraConfig.DailyChart.LeftMin = req.DailyChart.LeftMin
  739. extraConfig.DailyChart.LeftMax = req.DailyChart.LeftMax
  740. extraConfig.DailyChart.Unit = req.DailyChart.Unit
  741. }
  742. configByte, _ := json.Marshal(extraConfig)
  743. indexItem.ExtraConfig = string(configByte)
  744. indexItem.ModifyTime = time.Now()
  745. updateCols := []string{indexOb.Cols().ExtraConfig, indexOb.Cols().ModifyTime}
  746. if e = indexItem.Update(updateCols); e != nil {
  747. br.Msg = "操作失败"
  748. br.ErrMsg = fmt.Sprintf("保存标的失败, %v", e)
  749. return
  750. }
  751. br.Ret = 200
  752. br.Msg = "操作成功"
  753. br.Success = true
  754. }
  755. // DashboardSave
  756. // @Title 保存看板
  757. // @Description 保存看板
  758. // @Param request body aiPredictModel.AiPredictModelDashboardSaveReq true "type json string"
  759. // @Success 200 Ret=200 新增成功
  760. // @router /index/dashboard/save [post]
  761. func (this *AiPredictModelIndexController) DashboardSave() {
  762. br := new(models.BaseResponse).Init()
  763. defer func() {
  764. if br.ErrMsg == "" {
  765. br.IsSendEmail = false
  766. }
  767. this.Data["json"] = br
  768. this.ServeJSON()
  769. }()
  770. sysUser := this.SysUser
  771. if sysUser == nil {
  772. br.Msg = "请登录"
  773. br.ErrMsg = "请登录,SysUser Is Empty"
  774. br.Ret = 408
  775. return
  776. }
  777. var req aiPredictModel.AiPredictModelDashboardSaveReq
  778. if e := json.Unmarshal(this.Ctx.Input.RequestBody, &req); e != nil {
  779. br.Msg = "参数解析异常"
  780. br.ErrMsg = fmt.Sprintf("参数解析异常, %v", e)
  781. return
  782. }
  783. if req.IndexId <= 0 {
  784. br.Msg = "参数有误"
  785. br.ErrMsg = fmt.Sprintf("参数有误, %d", req.IndexId)
  786. return
  787. }
  788. req.DashboardName = strings.TrimSpace(req.DashboardName)
  789. indexOb := new(aiPredictModel.AiPredictModelIndex)
  790. _, e := indexOb.GetItemById(req.IndexId)
  791. if e != nil {
  792. if utils.IsErrNoRow(e) {
  793. br.Msg = "标的已被删除,请刷新页面"
  794. return
  795. }
  796. br.Msg = "操作失败"
  797. br.ErrMsg = fmt.Sprintf("获取标的失败, %v", e)
  798. return
  799. }
  800. // 获取看板
  801. var isUpdate bool
  802. var updateCols []string
  803. dashboardItem := new(aiPredictModel.AiPredictModelDashboard)
  804. if req.IndexId > 0 {
  805. cond := fmt.Sprintf(" AND %s = ?", dashboardItem.Cols().AiPredictModelIndexId)
  806. pars := make([]interface{}, 0)
  807. pars = append(pars, req.IndexId)
  808. item, e := dashboardItem.GetItemByCondition(cond, pars, "")
  809. if e != nil && !utils.IsErrNoRow(e) {
  810. br.Msg = "操作失败"
  811. br.ErrMsg = fmt.Sprintf("获取标的看板失败, %v", e)
  812. return
  813. }
  814. if item != nil && item.AiPredictModelIndexId > 0 {
  815. isUpdate = true
  816. dashboardItem = item
  817. dashboardItem.DashboardName = req.DashboardName
  818. dashboardItem.ModifyTime = time.Now()
  819. updateCols = append(updateCols, dashboardItem.Cols().DashboardName, dashboardItem.Cols().ModifyTime)
  820. }
  821. }
  822. if !isUpdate {
  823. dashboardItem.AiPredictModelIndexId = req.IndexId
  824. dashboardItem.DashboardName = req.DashboardName
  825. dashboardItem.SysUserId = sysUser.AdminId
  826. dashboardItem.SysUserRealName = sysUser.RealName
  827. dashboardItem.CreateTime = time.Now()
  828. dashboardItem.ModifyTime = time.Now()
  829. }
  830. // 详情
  831. dashboardDetails := make([]*aiPredictModel.AiPredictModelDashboardDetail, 0)
  832. for i, v := range req.List {
  833. t := &aiPredictModel.AiPredictModelDashboardDetail{
  834. Type: v.Type,
  835. UniqueCode: v.UniqueCode,
  836. Sort: i + 1,
  837. CreateTime: time.Now(),
  838. ModifyTime: time.Now(),
  839. }
  840. dashboardDetails = append(dashboardDetails, t)
  841. }
  842. // 保存
  843. if e := dashboardItem.SaveIndexDashboard(dashboardItem, dashboardDetails, isUpdate, updateCols); e != nil {
  844. br.Msg = "操作失败"
  845. br.ErrMsg = fmt.Sprintf("保存标的看板失败, %v", e)
  846. return
  847. }
  848. br.Ret = 200
  849. br.Success = true
  850. br.Msg = "操作成功"
  851. }
  852. // DashboardDetail
  853. // @Title 看板详情
  854. // @Description 看板详情
  855. // @Param IndexId query int true "标的ID"
  856. // @Success 200 {object} aiPredictModel.AiPredictModelDashboardDetailResp
  857. // @router /index/dashboard/detail [get]
  858. func (this *AiPredictModelIndexController) DashboardDetail() {
  859. br := new(models.BaseResponse).Init()
  860. defer func() {
  861. if br.ErrMsg == "" {
  862. br.IsSendEmail = false
  863. }
  864. this.Data["json"] = br
  865. this.ServeJSON()
  866. }()
  867. sysUser := this.SysUser
  868. if sysUser == nil {
  869. br.Msg = "请登录"
  870. br.ErrMsg = "请登录,SysUser Is Empty"
  871. br.Ret = 408
  872. return
  873. }
  874. indexId, _ := this.GetInt("IndexId")
  875. resp := new(aiPredictModel.AiPredictModelDashboardDetailResp)
  876. indexOb := new(aiPredictModel.AiPredictModelIndex)
  877. indexItem, e := indexOb.GetItemById(indexId)
  878. if e != nil {
  879. if utils.IsErrNoRow(e) {
  880. br.Msg = "标的已被删除,请刷新页面"
  881. return
  882. }
  883. br.Msg = "获取失败"
  884. br.ErrMsg = fmt.Sprintf("获取标的失败, %v", e)
  885. return
  886. }
  887. resp.CreateUserId = indexItem.SysUserId
  888. resp.CreateUserRealName = indexItem.SysUserRealName
  889. // 获取标的看板
  890. dashboardOb := new(aiPredictModel.AiPredictModelDashboard)
  891. dashboardItem := new(aiPredictModel.AiPredictModelDashboardItem)
  892. {
  893. cond := fmt.Sprintf(" AND %s = ?", dashboardOb.Cols().AiPredictModelIndexId)
  894. pars := make([]interface{}, 0)
  895. pars = append(pars, indexId)
  896. item, e := dashboardOb.GetItemByCondition(cond, pars, "")
  897. if e != nil && !utils.IsErrNoRow(e) {
  898. br.Msg = "操作失败"
  899. br.ErrMsg = fmt.Sprintf("获取标的看板失败, %v", e)
  900. return
  901. }
  902. if item != nil {
  903. dashboardItem = item.Format2Item()
  904. }
  905. }
  906. // 获取看板详情
  907. dashboardDetails := make([]*aiPredictModel.AiPredictModelDashboardDetailItem, 0)
  908. if dashboardItem.DashboardId > 0 {
  909. detailOb := new(aiPredictModel.AiPredictModelDashboardDetail)
  910. cond := fmt.Sprintf(" AND %s = ?", detailOb.Cols().AiPredictModelDashboardId)
  911. pars := make([]interface{}, 0)
  912. pars = append(pars, dashboardItem.DashboardId)
  913. list, e := detailOb.GetItemsByCondition(cond, pars, []string{}, fmt.Sprintf("%s ASC", detailOb.Cols().Sort))
  914. if e != nil {
  915. br.Msg = "获取失败"
  916. br.ErrMsg = fmt.Sprintf("获取看板详情失败, %v", e)
  917. return
  918. }
  919. for _, v := range list {
  920. dashboardDetails = append(dashboardDetails, v.Format2Item())
  921. }
  922. }
  923. resp.AiPredictModelDashboardItem = dashboardItem
  924. resp.List = dashboardDetails
  925. br.Data = resp
  926. br.Ret = 200
  927. br.Success = true
  928. br.Msg = "获取成功"
  929. }
  930. // SearchByEs
  931. // @Title 图表模糊搜索(从es获取)
  932. // @Description 图表模糊搜索(从es获取)
  933. // @Param Keyword query string true "图表名称"
  934. // @Param IsShowMe query bool true "是否只看我的,true、false"
  935. // @Param Source query int true "来源,14:日度预测,15:月度预测,默认0:全部14+15"
  936. // @Success 200 {object} data_manage.ChartInfo
  937. // @router /chart/search_by_es [get]
  938. func (this *AiPredictModelIndexController) SearchByEs() {
  939. br := new(models.BaseResponse).Init()
  940. defer func() {
  941. if br.ErrMsg == "" {
  942. br.IsSendEmail = false
  943. }
  944. this.Data["json"] = br
  945. this.ServeJSON()
  946. }()
  947. sysUser := this.SysUser
  948. if sysUser == nil {
  949. br.Msg = "请登录"
  950. br.ErrMsg = "请登录,SysUser Is Empty"
  951. br.Ret = 408
  952. return
  953. }
  954. pageSize, _ := this.GetInt("PageSize")
  955. currentIndex, _ := this.GetInt("CurrentIndex")
  956. var startSize int
  957. if pageSize <= 0 {
  958. pageSize = utils.PageSize20
  959. }
  960. if currentIndex <= 0 {
  961. currentIndex = 1
  962. }
  963. startSize = paging.StartIndex(currentIndex, pageSize)
  964. keyword := this.GetString("Keyword")
  965. keyword = strings.TrimSpace(keyword)
  966. if keyword == "" {
  967. keyword = this.GetString("KeyWord")
  968. keyword = strings.TrimSpace(keyword)
  969. }
  970. //只看我的
  971. isShowMe, _ := this.GetBool("IsShowMe")
  972. showSysId := 0
  973. if isShowMe {
  974. showSysId = sysUser.AdminId
  975. }
  976. source, _ := this.GetInt("Source")
  977. sourceList := make([]int, 0)
  978. if source <= 0 {
  979. sourceList = append(sourceList, utils.CHART_SOURCE_AI_PREDICT_MODEL_MONTHLY, utils.CHART_SOURCE_AI_PREDICT_MODEL_DAILY)
  980. } else {
  981. sourceList = append(sourceList, source)
  982. }
  983. var searchList []*data_manage.ChartInfoMore
  984. var total int64
  985. var err error
  986. // 获取当前账号的不可见指标(AI预测的指标为标的均可见)
  987. noPermissionChartIdList := make([]int, 0)
  988. //{
  989. // obj := data_manage.EdbInfoNoPermissionAdmin{}
  990. // confList, err := obj.GetAllChartListByAdminId(this.SysUser.AdminId)
  991. // if err != nil && !utils.IsErrNoRow(err) {
  992. // br.Msg = "获取失败"
  993. // br.ErrMsg = "获取不可见指标配置数据失败,Err:" + err.Error()
  994. // return
  995. // }
  996. // for _, v := range confList {
  997. // noPermissionChartIdList = append(noPermissionChartIdList, v.ChartInfoId)
  998. // }
  999. //}
  1000. if keyword != "" {
  1001. searchList, total, err = data.EsSearchChartInfo(keyword, showSysId, sourceList, noPermissionChartIdList, startSize, pageSize)
  1002. } else {
  1003. total, searchList, err = data_manage.ChartInfoSearchByEmptyKeyWord(showSysId, sourceList, noPermissionChartIdList, startSize, pageSize)
  1004. if err != nil && !utils.IsErrNoRow(err) {
  1005. br.Msg = "获取失败"
  1006. br.ErrMsg = "获取图表信息失败,Err:" + err.Error()
  1007. return
  1008. }
  1009. }
  1010. finalList := make([]*data_manage.ChartInfoMore, 0)
  1011. if len(searchList) > 0 {
  1012. chartInfoIds := ""
  1013. chartEdbMap := make(map[int][]*data_manage.ChartEdbInfoMapping)
  1014. for _, v := range searchList {
  1015. chartInfoIds += strconv.Itoa(v.ChartInfoId) + ","
  1016. }
  1017. if chartInfoIds != "" {
  1018. chartInfoIds = strings.Trim(chartInfoIds, ",")
  1019. //判断是否需要展示英文标识
  1020. edbList, e := data_manage.GetChartEdbMappingListByChartInfoIds(chartInfoIds)
  1021. if e != nil {
  1022. br.Msg = "获取失败"
  1023. br.ErrMsg = "获取图表,指标信息失败,Err:" + e.Error()
  1024. return
  1025. }
  1026. for _, v := range edbList {
  1027. chartEdbMap[v.ChartInfoId] = append(chartEdbMap[v.ChartInfoId], v)
  1028. }
  1029. }
  1030. for _, v := range searchList {
  1031. tmp := new(data_manage.ChartInfoMore)
  1032. tmp.ChartInfo = v.ChartInfo
  1033. // 图表数据权限
  1034. tmp.HaveOperaAuth = true
  1035. //判断是否需要展示英文标识
  1036. if edbTmpList, ok := chartEdbMap[v.ChartInfoId]; ok {
  1037. tmp.IsEnChart = data.CheckIsEnChart(v.ChartNameEn, edbTmpList, v.Source, v.ChartType)
  1038. }
  1039. tmp.SearchText = v.SearchText
  1040. if tmp.SearchText == "" {
  1041. tmp.SearchText = v.ChartName
  1042. }
  1043. finalList = append(finalList, tmp)
  1044. }
  1045. }
  1046. //新增搜索词记录
  1047. {
  1048. searchKeyword := new(data_manage.SearchKeyword)
  1049. searchKeyword.KeyWord = keyword
  1050. searchKeyword.CreateTime = time.Now()
  1051. go data_manage.AddSearchKeyword(searchKeyword)
  1052. }
  1053. page := paging.GetPaging(currentIndex, pageSize, int(total))
  1054. resp := data_manage.ChartInfoListByEsResp{
  1055. Paging: page,
  1056. List: finalList,
  1057. }
  1058. br.Ret = 200
  1059. br.Success = true
  1060. br.Msg = "获取成功"
  1061. br.Data = resp
  1062. }