neck_ring_estrus.go 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. package crontab
  2. import (
  3. "fmt"
  4. "kpt-pasture/model"
  5. "time"
  6. pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
  7. "gitee.com/xuyiping_admin/pkg/logger/zaplog"
  8. "go.uber.org/zap"
  9. "gitee.com/xuyiping_admin/pkg/xerr"
  10. )
  11. const (
  12. MaxRuminaAdJust = 20
  13. XAdjust21 = 15
  14. XAdjust42 = 10
  15. RumtoHeat = 0.5
  16. MinCalvingAge = 20
  17. MinLact = 0
  18. NormalChangJust = 10
  19. B48 = 48
  20. )
  21. func (e *Entry) UpdateCowEstrus() (err error) {
  22. pastureList := e.FindPastureList()
  23. if pastureList == nil || len(pastureList) == 0 {
  24. return nil
  25. }
  26. for _, pasture := range pastureList {
  27. if err = e.EntryCowEstrus(pasture.Id); err != nil {
  28. zaplog.Error("EntryCrontab", zap.Any("PastureUpdateCowEstrus", err), zap.Any("pasture", pasture))
  29. }
  30. zaplog.Info("PastureUpdateCowEstrus-success", zap.Any("pasture", pasture.Id))
  31. }
  32. return nil
  33. }
  34. func (e *Entry) EntryCowEstrus(pastureId int64) (err error) {
  35. xToday := &XToday{}
  36. systemConfigureList, err := e.GetSystemNeckRingConfigure(pastureId)
  37. for _, v := range systemConfigureList {
  38. switch v.Name {
  39. case model.ActiveLow:
  40. xToday.ActiveLow = int32(v.Value)
  41. case model.ActiveMiddle:
  42. xToday.ActiveMiddle = int32(v.Value)
  43. case model.ActiveHigh:
  44. xToday.ActiveHigh = int32(v.Value)
  45. }
  46. }
  47. if err = e.CowEstrusWarning(pastureId, xToday); err != nil {
  48. zaplog.Error("EntryCowEstrus", zap.Any("CowEstrusWarning", err), zap.Any("xToday", xToday))
  49. }
  50. // 将3天前历史发情预警数据更新为已过期
  51. if err = e.DB.Model(new(model.NeckRingEstrus)).
  52. Where("first_time != ?", "").
  53. Where("first_time <= ?", time.Now().AddDate(0, 0, -3).Format(model.LayoutTime)).
  54. Update("check_result", pasturePb.CheckResult_Overdue).Error; err != nil {
  55. zaplog.Error("EntryCowEstrus", zap.Any("UpdateEventEstrus", err))
  56. }
  57. return nil
  58. }
  59. // CowEstrusWarning 发情预警
  60. func (e *Entry) CowEstrusWarning(pastureId int64, xToday *XToday) (err error) {
  61. nowTime := time.Now()
  62. neckActiveHabitList := make([]*model.NeckActiveHabit, 0) // todo 需要考虑到数据量太大的情况
  63. if err = e.DB.Model(new(model.NeckActiveHabit)).
  64. Where("heat_date = ?", nowTime.Format(model.LayoutDate2)).
  65. Where("pasture_id = ?", pastureId).
  66. Where("filter_high > 0 AND change_filter > ?", model.DefaultChangeFilter).
  67. Where("cow_id > ?", 0).
  68. Where(e.DB.Where("calving_age >= ?", MinCalvingAge).Or("lact = ?", MinLact)). // 排除产后20天内的发情牛
  69. Order("cow_id").
  70. Find(&neckActiveHabitList).Error; err != nil {
  71. return xerr.WithStack(err)
  72. }
  73. zaplog.Info("CowEstrusWarning", zap.Any("neckActiveHabitList", neckActiveHabitList))
  74. neckActiveHabitMap := make(map[int64][]*model.NeckActiveHabit)
  75. for _, habit := range neckActiveHabitList {
  76. cft := calculateCFT(habit)
  77. if cft < float32(xToday.ActiveLow-XAdjust21) {
  78. continue
  79. }
  80. neckActiveHabitMap[habit.CowId] = append(neckActiveHabitMap[habit.CowId], habit)
  81. }
  82. zaplog.Info("CowEstrusWarning", zap.Any("neckActiveHabitMap", neckActiveHabitMap))
  83. neckRingEstrusList := make([]*model.NeckRingEstrus, 0)
  84. for cowId, cowHabitList := range neckActiveHabitMap {
  85. // 最近3天最大发情记录,小于该变化趋势的不再插入
  86. before3Data := e.GetBeforeThreeDaysCowEstrus(cowId, nowTime.AddDate(0, 0, -2).Format(model.LayoutTime))
  87. // 判断最近50天内是否存在发情记录(发情等级>=2),如果18~25天@xadjust21,如果36~50天@xadjust42
  88. cowEstrus := e.GetTwoEstrus(pastureId, cowId, nowTime.AddDate(0, 0, -100).Format(model.LayoutTime), nowTime.AddDate(0, 0, -2).Format(model.LayoutTime))
  89. activeDateTime, _ := time.Parse(model.LayoutTime, cowEstrus.ActiveDate)
  90. if activeDateTime.Unix() >= nowTime.AddDate(0, 0, -25).Unix() && activeDateTime.Unix() <= nowTime.AddDate(0, 0, -18).Unix() {
  91. cowEstrus.HadJust = XAdjust21
  92. }
  93. if activeDateTime.Unix() >= nowTime.AddDate(0, 0, -50).Unix() && activeDateTime.Unix() <= nowTime.AddDate(0, 0, -36).Unix() {
  94. cowEstrus.HadJust = XAdjust42
  95. }
  96. maxCft := float32(0)
  97. maxHigh := int32(0)
  98. lastActiveDate := time.Time{}
  99. for _, habit := range cowHabitList {
  100. cft := calculateCFT(habit)
  101. if cft > maxCft {
  102. maxCft = cft
  103. }
  104. if habit.FilterHigh > maxHigh {
  105. maxHigh = habit.FilterHigh
  106. }
  107. // 获取最新的 CreateTime
  108. activeTimeParse, _ := time.Parse(habit.ActiveTime, model.LayoutTime)
  109. if activeTimeParse.After(lastActiveDate) {
  110. lastActiveDate = activeTimeParse
  111. }
  112. }
  113. b48 := float64(0)
  114. t3, _ := time.Parse(model.LayoutTime, before3Data.ActiveTime)
  115. b48 = t3.Sub(lastActiveDate).Hours()
  116. if (int32(maxCft) > before3Data.DayHigh || before3Data.CowId == 0 || b48 > B48) && int32(maxCft)+cowEstrus.HadJust > xToday.ActiveLow {
  117. level := calculateActiveLevel(maxCft, cowEstrus, xToday)
  118. cowInfo := e.FindCowInfoByCowId(cowId)
  119. if cowInfo == nil {
  120. zaplog.Error("CowEstrusWarning", zap.Any("FindCowInfoByCowId", cowId))
  121. continue
  122. }
  123. isShow := pasturePb.IsShow_Ok
  124. if cowInfo != nil && cowInfo.IsPregnant == pasturePb.IsShow_Ok && level == pasturePb.EstrusLevel_Low {
  125. isShow = pasturePb.IsShow_No
  126. }
  127. dayHigh := int32(maxCft) + cowEstrus.HadJust
  128. lastEstrusDate := cowEstrus.ActiveDate
  129. checkResult := getResult(before3Data, maxCft, cowEstrus)
  130. isPeak := pasturePb.IsShow_Ok
  131. cowEstrusStartData := e.FindCowEstrusFirstTime(pastureId, cowId, nowTime)
  132. firstTime := ""
  133. if cowEstrusStartData != nil {
  134. firstTime = cowEstrusStartData.FirstTime
  135. }
  136. zaplog.Info("CowEstrusWarning",
  137. zap.Any("level", level),
  138. zap.Any("b48", b48),
  139. zap.Any("checkResult", checkResult),
  140. zap.Any("isShow", isShow),
  141. zap.Any("isPeak", isPeak),
  142. zap.Any("lastEstrusDate", lastEstrusDate),
  143. zap.Any("activeDate", lastActiveDate),
  144. zap.Any("dayHigh", dayHigh),
  145. zap.Any("cft", maxCft),
  146. zap.Any("before3Data", before3Data),
  147. zap.Any("cowEstrus", cowEstrus),
  148. zap.Any("cowInfo", cowInfo),
  149. zap.Any("firstTime", firstTime),
  150. zap.Any("cowHabitList", cowHabitList),
  151. )
  152. newNeckRingEstrus := model.NewNeckRingEstrus(pastureId, cowInfo, level, checkResult, isShow, firstTime)
  153. newNeckRingEstrus.LastTime = lastEstrusDate
  154. newNeckRingEstrus.ActiveTime = lastActiveDate.Format(model.LayoutTime)
  155. newNeckRingEstrus.DayHigh = dayHigh
  156. newNeckRingEstrus.MaxHigh = maxHigh
  157. newNeckRingEstrus.IsPeak = isPeak
  158. neckRingEstrusList = append(neckRingEstrusList, newNeckRingEstrus)
  159. }
  160. }
  161. zaplog.Info("CowEstrusWarning", zap.Any("neckRingEstrusList", neckRingEstrusList))
  162. if len(neckRingEstrusList) > 0 {
  163. if err = e.DB.Model(new(model.NeckRingEstrus)).Create(neckRingEstrusList).Error; err != nil {
  164. zaplog.Error("CowEstrusWarningNew", zap.Any("eventEstrusList", neckRingEstrusList), zap.Any("err", err))
  165. }
  166. }
  167. return err
  168. }
  169. // FindCowEstrusFirstTime 查找牛只昨天是否有发情数据
  170. func (e *Entry) FindCowEstrusFirstTime(pastureId, cowId int64, xToday time.Time) *EstrusStartData {
  171. firstTimeEventEstrus := &EstrusStartData{}
  172. if err := e.DB.Model(new(model.NeckRingEstrus)).
  173. Select("cow_id,MIN(STR_TO_DATE(first_time, '%Y-%m-%d %H:%i:%s')) as first_time").
  174. Where("active_time BETWEEN ? AND ?",
  175. fmt.Sprintf("%s 00:00:00", xToday.AddDate(0, 0, -1).Format(model.LayoutDate2)),
  176. fmt.Sprintf("%s 23:59:59", xToday.Format(model.LayoutDate2)),
  177. ).Where("pasture_id = ?", pastureId).
  178. Where("cow_id = ?", cowId).
  179. First(&firstTimeEventEstrus).Error; err != nil {
  180. return nil
  181. }
  182. return firstTimeEventEstrus
  183. }
  184. // calculateCFT 计算cft值
  185. func calculateCFT(habit *model.NeckActiveHabit) (cft float32) {
  186. if habit.ChangeAdjust >= 10 {
  187. cft = (float32(habit.ChangeFilter) - float32(habit.ChangeAdjust) + 3) * float32(habit.FilterCorrect) / 100
  188. } else {
  189. cft = float32(habit.ChangeFilter) * float32(habit.FilterCorrect) / 100
  190. }
  191. switch {
  192. case habit.RuminaFilter > MaxRuminaAdJust:
  193. cft -= float32(5)
  194. case habit.RuminaFilter > 0:
  195. cft -= float32(habit.RuminaFilter) * 0.25
  196. case habit.RuminaFilter < -MaxRuminaAdJust:
  197. cft -= -MaxRuminaAdJust * RumtoHeat
  198. default:
  199. cft -= float32(habit.RuminaFilter) * RumtoHeat
  200. }
  201. return cft
  202. }
  203. // calculateActiveLevel 计算活动量等级
  204. func calculateActiveLevel(cft float32, cowEstrus *CowEstrus, xToday *XToday) pasturePb.EstrusLevel_Kind {
  205. if int32(cft)+cowEstrus.HadJust < xToday.ActiveMiddle {
  206. return pasturePb.EstrusLevel_Low
  207. } else if int32(cft)+cowEstrus.HadJust >= xToday.ActiveHigh {
  208. return pasturePb.EstrusLevel_Middle
  209. } else {
  210. return pasturePb.EstrusLevel_High
  211. }
  212. }
  213. // getResult 根据b3数据计算结果
  214. func getResult(b3 *model.NeckRingEstrus, cft float32, cowEstrus *CowEstrus) pasturePb.CheckResult_Kind {
  215. result := pasturePb.CheckResult_Pending
  216. if b3.CheckResult == pasturePb.CheckResult_Correct {
  217. result = pasturePb.CheckResult_Correct
  218. }
  219. if b3.CheckResult == pasturePb.CheckResult_Fail && b3.DayHigh > int32(cft)+cowEstrus.HadJust {
  220. result = pasturePb.CheckResult_Fail
  221. }
  222. return result
  223. }