dashboard.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440
  1. package backend
  2. import (
  3. "context"
  4. "kpt-pasture/model"
  5. "kpt-pasture/util"
  6. "net/http"
  7. "strconv"
  8. "strings"
  9. "time"
  10. "gorm.io/gorm"
  11. "gitee.com/xuyiping_admin/pkg/logger/zaplog"
  12. "gitee.com/xuyiping_admin/pkg/xerr"
  13. "go.uber.org/zap"
  14. pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
  15. )
  16. func (s *StoreEntry) NeckRingWarning(ctx context.Context) (*pasturePb.IndexNeckRingResponse, error) {
  17. userModel, err := s.GetUserModel(ctx)
  18. if err != nil {
  19. return nil, xerr.WithStack(err)
  20. }
  21. estrusWarningCowList := make([]*model.DashboardNeckRingEstrusWarning, 0)
  22. estrusWarningLevelItems := map[int32]int32{
  23. int32(pasturePb.EstrusLevel_Low): 0,
  24. int32(pasturePb.EstrusLevel_Middle): 0,
  25. int32(pasturePb.EstrusLevel_High): 0,
  26. }
  27. pref, err := s.EstrusWarningQuery(ctx, userModel.AppPasture.Id)
  28. if err != nil {
  29. return nil, xerr.Customf("系统错误!")
  30. }
  31. if err = pref.Order("a.level DESC").
  32. Select("a.level, count(a.level) as count").
  33. Group("a.level").
  34. Find(&estrusWarningCowList).Error; err != nil {
  35. return nil, xerr.WithStack(err)
  36. }
  37. /*if err = s.DB.Table(fmt.Sprintf("%s as a", new(model.NeckRingEstrusWarning).TableName())).
  38. Select("a.level, count(a.level) as count").
  39. Where("a.pasture_id = ?", userModel.AppPasture.Id).
  40. Where("a.is_show = ?", pasturePb.IsShow_Ok).
  41. Group("a.level").
  42. Find(&estrusWarningCowList).Error; err != nil {
  43. zaplog.Error("NeckRingWarning", zap.Any("estrusWarningNumber", err))
  44. }*/
  45. countEstrusWarning := 0
  46. for _, v := range estrusWarningCowList {
  47. estrusWarningLevelItems[int32(v.Level)] = estrusWarningLevelItems[v.Count]
  48. countEstrusWarning += int(v.Count)
  49. }
  50. abortionCount := int64(0)
  51. pref, err = s.AbortionWarningQuery(ctx, userModel.AppPasture.Id)
  52. if err != nil {
  53. return nil, xerr.Customf("系统错误!")
  54. }
  55. if err = pref.Group("cow_id").Count(&abortionCount).Error; err != nil {
  56. return nil, xerr.WithStack(err)
  57. }
  58. healthWarningNumber := int64(0)
  59. if err = s.DB.Model(new(model.NeckRingHealth)).
  60. Where("pasture_id = ?", userModel.AppPasture.Id).
  61. Group("cow_id").
  62. Count(&healthWarningNumber).Error; err != nil {
  63. zaplog.Error("NeckRingWarning", zap.Any("estrusWarningNumber", err))
  64. }
  65. return &pasturePb.IndexNeckRingResponse{
  66. Code: http.StatusOK,
  67. Msg: "ok",
  68. Data: &pasturePb.NeckRingData{
  69. EstrusWarningNumber: int32(countEstrusWarning),
  70. HealthWarningNumber: int32(healthWarningNumber),
  71. AbortionWarningNumber: int32(abortionCount),
  72. StressWarningNumber: 0,
  73. EstrusWarningLevelItems: estrusWarningLevelItems,
  74. },
  75. }, nil
  76. }
  77. func (s *StoreEntry) FocusIndicatorsList(ctx context.Context, dimension string) (*pasturePb.IndexFocusIndicatorsResponse, error) {
  78. userModel, err := s.GetUserModel(ctx)
  79. if err != nil {
  80. return nil, xerr.WithStack(err)
  81. }
  82. userFocusIndicators := userModel.SystemUser.IndicatorsKinds
  83. if len(userFocusIndicators) <= 0 {
  84. userFocusIndicators = model.DefaultFocusIndicators
  85. }
  86. userFocusIndicatorsList := strings.Split(userFocusIndicators, ",")
  87. indicatorsDataList := make([]*model.IndicatorsData, 0)
  88. pref := s.DB.Model(new(model.IndicatorsData)).
  89. Where("pasture_id = ?", userModel.AppPasture.Id).
  90. Where("kind in (?)", userFocusIndicatorsList)
  91. /*if dimension == "Year" {
  92. pref.Where("date = ?", time.Now().Format(model.LayoutMonth))
  93. }*/
  94. nowTime := time.Now()
  95. if dimension == "Month" {
  96. pref.Where("date = ?", nowTime.Format(model.LayoutMonth))
  97. }
  98. if err = pref.Find(&indicatorsDataList).Error; err != nil {
  99. zaplog.Error("FocusIndicators", zap.Any("err", err))
  100. }
  101. indicatorsDetailsMap, _, err := s.GetIndicatorsDetailsMap(ctx)
  102. if err != nil {
  103. return nil, xerr.WithStack(err)
  104. }
  105. focusIndicatorsList := make([]*pasturePb.FocusIndicators, 0)
  106. for _, v := range indicatorsDataList {
  107. indicatorsDetails, ok := indicatorsDetailsMap[v.Kind]
  108. if !ok {
  109. continue
  110. }
  111. onYear, onMonth := "", ""
  112. isUp := pasturePb.IsShow_Ok
  113. if dimension == "Year" {
  114. return nil, xerr.Custom("暂不支持该维度")
  115. }
  116. if dimension == "Month" {
  117. lastMonth := nowTime.AddDate(0, -1, 0).Format(model.LayoutMonth)
  118. oldIndicators, _ := s.GetIndicatorsDataByDate(userModel.AppPasture.Id, v.Kind, lastMonth)
  119. if oldIndicators != nil {
  120. if oldIndicators.Value != "" && oldIndicators.Value != "0" {
  121. oldValue, _ := strconv.ParseFloat(oldIndicators.Value, 64)
  122. currValue, _ := strconv.ParseFloat(v.Value, 64)
  123. onMonthValue := (oldValue - currValue) / oldValue * 100
  124. omv := util.RoundToTwoDecimals(onMonthValue)
  125. if omv < 0 {
  126. isUp = pasturePb.IsShow_No
  127. }
  128. onMonth = strconv.FormatFloat(omv, 'f', 2, 64) + "%"
  129. }
  130. }
  131. }
  132. focusIndicatorsList = append(focusIndicatorsList, &pasturePb.FocusIndicators{
  133. Kind: indicatorsDetails.Kind,
  134. Name: indicatorsDetails.Name,
  135. Value: v.Value,
  136. Describe: indicatorsDetails.Zh,
  137. UnitName: indicatorsDetails.Unit,
  138. OnMonth: onMonth,
  139. OnYear: onYear,
  140. IsUp: isUp,
  141. })
  142. }
  143. indicatorsDetailsList, _ := s.FindIndicatorsDetailsList(ctx)
  144. return &pasturePb.IndexFocusIndicatorsResponse{
  145. Code: http.StatusOK,
  146. Msg: "ok",
  147. Data: &pasturePb.FocusData{
  148. FocusIndicators: focusIndicatorsList,
  149. IndicatorsSet: model.IndicatorsDetailsSlice(indicatorsDetailsList).ToPB(userFocusIndicatorsList),
  150. },
  151. }, err
  152. }
  153. func (s *StoreEntry) FocusIndicatorsSet(ctx context.Context, req *pasturePb.IndexFocusIndicatorsSetRequest) error {
  154. userModel, err := s.GetUserModel(ctx)
  155. if err != nil {
  156. return xerr.WithStack(err)
  157. }
  158. if len(req.IndicatorsKind) <= 0 {
  159. return nil
  160. }
  161. userFocusIndicators := strings.Join(req.IndicatorsKind, ",")
  162. if err = s.DB.Model(new(model.SystemUser)).
  163. Where("id = ?", userModel.SystemUser.Id).
  164. Update("indicators_kinds", userFocusIndicators).Error; err != nil {
  165. return xerr.WithStack(err)
  166. }
  167. return nil
  168. }
  169. func (s *StoreEntry) DataWarningSet(ctx context.Context, req *pasturePb.IndexDataWarningSetRequest) error {
  170. userModel, err := s.GetUserModel(ctx)
  171. if err != nil {
  172. return xerr.WithStack(err)
  173. }
  174. if len(req.WarningDataSet) <= 0 {
  175. return xerr.Custom("请选择预警数据")
  176. }
  177. defaultDataWarning, _ := s.FindDataWarning(ctx, model.DefaultUserId)
  178. if len(defaultDataWarning) <= 0 {
  179. return xerr.Custom("默认预警数据不存在,请联系管理员!")
  180. }
  181. userDataWarningList, err := s.FindDataWarning(ctx, userModel.SystemUser.Id)
  182. if err != nil {
  183. return xerr.WithStack(err)
  184. }
  185. if len(userDataWarningList) <= 0 { // 新增
  186. return s.addUserDataWarning(ctx, userModel.SystemUser.Id, defaultDataWarning, req.WarningDataSet)
  187. }
  188. return s.updateUserDataWarning(ctx, userModel.SystemUser.Id, userDataWarningList, req.WarningDataSet)
  189. }
  190. func (s *StoreEntry) DataWarningList(ctx context.Context) (*pasturePb.IndexDataWarningResponse, error) {
  191. userModel, err := s.GetUserModel(ctx)
  192. if err != nil {
  193. return nil, xerr.WithStack(err)
  194. }
  195. defaultDataWarning, _ := s.FindDataWarning(ctx, model.DefaultUserId)
  196. if len(defaultDataWarning) <= 0 {
  197. return nil, xerr.Custom("默认预警数据有误,请联系管理员!")
  198. }
  199. var isExist bool // 判断是否存在自己的设置的数据
  200. userDataWarning, _ := s.FindDataWarning(ctx, userModel.SystemUser.Id)
  201. if len(userDataWarning) == 0 {
  202. // 如果用户没有配置自己的预警数据,则使用默认数据
  203. isExist = true
  204. userDataWarning = defaultDataWarning
  205. }
  206. newTime := time.Now().Unix()
  207. needUpdateWarningIds := make([]int64, 0)
  208. for _, warningData := range userDataWarning {
  209. // 如果预警数据更新时间大于预警条件更新时间,并且更新时间距离当前时间小于2小时,则跳过
  210. if warningData.DataUpdateAt > warningData.ConditionUpdateAt && newTime-warningData.DataUpdateAt < int64(2*time.Hour) {
  211. continue
  212. }
  213. needUpdateWarningIds = append(needUpdateWarningIds, warningData.Id)
  214. }
  215. // 需要重新计算更新的warningId
  216. if len(needUpdateWarningIds) > 0 {
  217. s.UpdateWarningData(ctx, needUpdateWarningIds)
  218. }
  219. userDataWarningItems := make([]*model.DataWarningItems, 0)
  220. // 计算过后重新获取数据
  221. if isExist {
  222. userDataWarning, _ = s.FindDataWarning(ctx, model.DefaultUserId)
  223. userDataWarningItems, _ = s.FindDataWarningItems(ctx, model.DefaultUserId)
  224. } else {
  225. userDataWarning, _ = s.FindDataWarning(ctx, userModel.SystemUser.Id)
  226. userDataWarningItems, _ = s.FindDataWarningItems(ctx, userModel.SystemUser.Id)
  227. }
  228. return &pasturePb.IndexDataWarningResponse{
  229. Code: http.StatusOK,
  230. Msg: "ok",
  231. Data: &pasturePb.DataWarning{
  232. DataSet: model.DataWarningItemsSlice(userDataWarningItems).ToPB(userDataWarning),
  233. DataShow: model.DataWarningSlice(userDataWarning).ToPB(),
  234. },
  235. }, nil
  236. }
  237. // 新增用户预警数据
  238. func (s *StoreEntry) addUserDataWarning(ctx context.Context, userId int64, defaultDataWarning []*model.DataWarning, warningDataSet []*pasturePb.WarningDataSet) error {
  239. // 将默认预警数据按 Kind 映射
  240. defaultDataWarningMap := make(map[string]*model.DataWarning)
  241. for _, v := range defaultDataWarning {
  242. defaultDataWarningMap[v.Kind] = v
  243. }
  244. // 在事务中执行新增操作
  245. return s.DB.Transaction(func(tx *gorm.DB) error {
  246. addedKinds := make(map[string]bool) // 记录已添加的 Kind
  247. for _, set := range warningDataSet {
  248. dataWarning := model.NewDataWarning(userId, set.Kind, pasturePb.IsShow_Ok, defaultDataWarningMap[set.Kind])
  249. // 如果该 Kind 已添加,跳过
  250. if !addedKinds[set.Kind] {
  251. // 创建新的预警数据
  252. if err := tx.Create(dataWarning).Error; err != nil {
  253. return xerr.WithStack(err)
  254. }
  255. } else {
  256. oldDataWarning := &model.DataWarning{}
  257. if err := tx.Model(new(model.DataWarning)).
  258. Where("user_id = ?", userId).
  259. Where("kind = ?", set.Kind).
  260. First(oldDataWarning).Error; err != nil {
  261. return xerr.WithStack(err)
  262. }
  263. dataWarning.Id = oldDataWarning.Id
  264. }
  265. // 创建预警项数据
  266. if err := tx.Create(model.NewDataWarningItems(userId, dataWarning, set)).Error; err != nil {
  267. return xerr.WithStack(err)
  268. }
  269. addedKinds[set.Kind] = true
  270. }
  271. return nil
  272. })
  273. }
  274. // 更新用户预警数据
  275. func (s *StoreEntry) updateUserDataWarning(ctx context.Context, userId int64, userDataWarningList []*model.DataWarning, warningDataSet []*pasturePb.WarningDataSet) error {
  276. // 将请求数据按 WarningId 和 Id 映射
  277. warningIsShowMap := make(map[int32]pasturePb.IsShow_Kind)
  278. warningItemDataMap := make(map[int32]*pasturePb.WarningDataSet)
  279. for _, set := range warningDataSet {
  280. warningIsShowMap[set.WarningId] = set.IsShow
  281. warningItemDataMap[set.Id] = set
  282. }
  283. // 获取用户预警项数据
  284. userDataWarningItems, err := s.FindDataWarningItems(ctx, userId)
  285. if err != nil {
  286. return xerr.WithStack(err)
  287. }
  288. if len(userDataWarningItems) == 0 {
  289. return xerr.Custom("预警数据有误,请联系管理员!")
  290. }
  291. // 在事务中执行更新操作
  292. return s.DB.Transaction(func(tx *gorm.DB) error {
  293. // 更新预警数据的 IsShow 字段
  294. for _, warning := range userDataWarningList {
  295. if isShow, ok := warningIsShowMap[int32(warning.Id)]; ok {
  296. if err = tx.Model(&model.DataWarning{}).
  297. Where("id = ?", warning.Id).
  298. Update("is_show", isShow).Error; err != nil {
  299. return xerr.WithStack(err)
  300. }
  301. }
  302. }
  303. // 更新预警项数据的 IsShow 和 Value 字段
  304. for _, item := range userDataWarningItems {
  305. if set, ok := warningItemDataMap[int32(item.Id)]; ok {
  306. if err = tx.Model(&model.DataWarningItems{}).
  307. Where("id = ?", item.Id).
  308. Updates(map[string]interface{}{
  309. "is_show": set.IsShow,
  310. "value": set.Value,
  311. }).Error; err != nil {
  312. return xerr.WithStack(err)
  313. }
  314. }
  315. }
  316. return nil
  317. })
  318. }
  319. func (s *StoreEntry) FindDataWarning(ctx context.Context, userId int64) ([]*model.DataWarning, error) {
  320. dataWarningList := make([]*model.DataWarning, 0)
  321. if err := s.DB.Model(new(model.DataWarning)).
  322. Where("user_id = ?", userId).
  323. Find(&dataWarningList).Error; err != nil {
  324. return nil, xerr.WithStack(err)
  325. }
  326. return dataWarningList, nil
  327. }
  328. func (s *StoreEntry) FindDataWarningItems(ctx context.Context, userId int64) ([]*model.DataWarningItems, error) {
  329. dataWarningItemsList := make([]*model.DataWarningItems, 0)
  330. if err := s.DB.Model(new(model.DataWarningItems)).
  331. Where("user_id = ?", userId).
  332. Find(&dataWarningItemsList).Error; err != nil {
  333. return nil, xerr.WithStack(err)
  334. }
  335. return dataWarningItemsList, nil
  336. }
  337. func (s *StoreEntry) FindDataWarningMap(ctx context.Context, userId int64) (map[int64]*model.DataWarning, error) {
  338. dataWarning, err := s.FindDataWarning(ctx, userId)
  339. if err != nil {
  340. return nil, xerr.Custom("默认预警数据有误,请联系管理员!")
  341. }
  342. dataWarningMap := make(map[int64]*model.DataWarning)
  343. for _, v := range dataWarning {
  344. dataWarningMap[v.Id] = v
  345. }
  346. return dataWarningMap, nil
  347. }
  348. func (s *StoreEntry) FindDataWarningItemsMap(ctx context.Context, userId int64) (map[int64]*model.DataWarningItems, error) {
  349. dataWarningItemsList := make([]*model.DataWarningItems, 0)
  350. if err := s.DB.Model(new(model.DataWarningItems)).
  351. Where("user_id = ?", userId).
  352. Find(&dataWarningItemsList).Error; err != nil {
  353. return nil, xerr.WithStack(err)
  354. }
  355. dataWarningItemsMap := make(map[int64]*model.DataWarningItems)
  356. for _, v := range dataWarningItemsList {
  357. dataWarningItemsMap[v.Id] = v
  358. }
  359. return dataWarningItemsMap, nil
  360. }
  361. // UpdateWarningData 更新计算数据
  362. func (s *StoreEntry) UpdateWarningData(ctx context.Context, needUpdateWarningIds []int64) {
  363. if len(needUpdateWarningIds) <= 0 {
  364. return
  365. }
  366. for _, warningId := range needUpdateWarningIds {
  367. query, params, err := s.BuildQuery(warningId)
  368. if err != nil {
  369. zaplog.Error("UpdateWarningData", zap.Any("BuildQuery", err), zap.Any("warningId", warningId))
  370. }
  371. if len(query) == 0 || len(params) == 0 {
  372. continue
  373. }
  374. var count int64
  375. if err = s.DB.Model(new(model.Cow)).Where(query, params...).Count(&count).Error; err != nil {
  376. zaplog.Error("UpdateWarningData", zap.Any("err", err), zap.Any("query", query), zap.Any("params", params))
  377. }
  378. if err = s.DB.Model(new(model.DataWarning)).
  379. Where("id = ?", warningId).
  380. Updates(map[string]interface{}{
  381. "data_value": count,
  382. "data_update_at": time.Now().Unix(),
  383. }).Error; err != nil {
  384. zaplog.Error("UpdateWarningData", zap.Any("update", err))
  385. }
  386. }
  387. }