index.go 29 KB

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