neck_ring_estrus.go 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. package crontab
  2. import (
  3. "kpt-pasture/model"
  4. "time"
  5. pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
  6. "gitee.com/xuyiping_admin/pkg/logger/zaplog"
  7. "go.uber.org/zap"
  8. "gitee.com/xuyiping_admin/pkg/xerr"
  9. )
  10. const (
  11. MaxRuminaAdJust = 20
  12. XAdjust21 = 15
  13. XAdjust42 = 10
  14. RumtoHeat = 0.5
  15. MinCalvingAge = 20
  16. MinLact = 0
  17. NormalChangJust = 10
  18. )
  19. func (e *Entry) PastureUpdateCowEstrus() (err error) {
  20. pastureList := e.FindPastureList()
  21. if pastureList == nil || len(pastureList) == 0 {
  22. return nil
  23. }
  24. for _, pasture := range pastureList {
  25. if err = e.EntryCowEstrus(pasture.Id); err != nil {
  26. zaplog.Error("EntryCrontab", zap.Any("PastureUpdateCowEstrus", err), zap.Any("pasture", pasture))
  27. }
  28. zaplog.Error("PastureUpdateCowEstrus-success", zap.Any("pasture", pasture.Id))
  29. }
  30. return nil
  31. }
  32. func (e *Entry) EntryCowEstrus(pastureId int64) (err error) {
  33. activeLowValue := e.GetSystemConfigure(pastureId, model.ActiveLow).Value
  34. activeMiddleValue := e.GetSystemConfigure(pastureId, model.ActiveMiddle).Value
  35. activeHighValue := e.GetSystemConfigure(pastureId, model.ActiveHigh).Value
  36. lastMaxEstrusId := e.GetSystemConfigure(pastureId, model.MaxEstrus).Value
  37. currentMaxHabit := &model.NeckActiveHabit{}
  38. if err = e.DB.Model(new(model.NeckActiveHabit)).
  39. Where("id > ?", lastMaxEstrusId).
  40. Order("id desc").
  41. First(currentMaxHabit).Error; err != nil {
  42. return xerr.WithStack(err)
  43. }
  44. // 当前Id<=上次执行的id,则不执行
  45. if currentMaxHabit.Id <= lastMaxEstrusId {
  46. return nil
  47. }
  48. defer func() {
  49. if err == nil {
  50. e.DB.Model(new(model.SystemConfigure)).
  51. Where("name = ?", model.MaxEstrus).
  52. Update("value", currentMaxHabit.Id)
  53. }
  54. }()
  55. xToday := &XToday{}
  56. if err = e.DB.Model(new(model.NeckActiveHabit)).
  57. Select(`MIN(heat_date) as x_beg_date, MAX(heat_date) as x_end_date`).
  58. Where("id BETWEEN ? AND ?", lastMaxEstrusId, currentMaxHabit.Id).
  59. First(xToday).Error; err != nil {
  60. return xerr.WithStack(err)
  61. }
  62. xToday.LastMaxHabitId = lastMaxEstrusId
  63. xToday.CurrMaxHabitId = currentMaxHabit.Id
  64. xToday.ActiveLow = activeLowValue
  65. xToday.ActiveMiddle = activeMiddleValue
  66. xToday.ActiveHigh = activeHighValue
  67. if err = e.CowEstrusWarning(xToday); err != nil {
  68. zaplog.Error("EntryCowEstrus", zap.Any("CowEstrusWarning", err), zap.Any("xToday", xToday))
  69. }
  70. return nil
  71. }
  72. // CowEstrusWarning 发情预警
  73. func (e *Entry) CowEstrusWarning(xToday *XToday) (err error) {
  74. startDate, _ := time.Parse(model.LayoutDate2, xToday.XBegDate)
  75. endDate, _ := time.Parse(model.LayoutDate2, xToday.XEndDate)
  76. for startDate.Format(model.LayoutDate2) <= endDate.Format(model.LayoutDate2) {
  77. neckActiveHabitList := make([]*model.NeckActiveHabit, 0)
  78. if err = e.DB.Model(new(model.NeckActiveHabit)).
  79. Select("*,MAX(filter_high) as filter_high").
  80. Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
  81. Where("heat_date = ?", startDate.Format(model.LayoutDate2)).
  82. Where("cow_id > ?", 0).
  83. Where(e.DB.Where("calving_age > ?", MinCalvingAge).Or("lact = ?", MinLact)).
  84. Group("neck_ring_number").
  85. Find(&neckActiveHabitList).
  86. Error; err != nil {
  87. zaplog.Error("CowEstrusWarning", zap.Any("FindNeckActiveHabit", err), zap.Any("xToday", xToday))
  88. continue
  89. }
  90. eventEstrusList := make([]*model.EventEstrus, 0)
  91. for _, v := range neckActiveHabitList {
  92. if ok := e.IsAdJustLow(xToday, v); ok {
  93. continue
  94. }
  95. cft := float32(0)
  96. if v.ChangeAdjust > 10 {
  97. cft = float32(v.ChangeFilter) - float32(v.ChangeAdjust) + 3
  98. } else {
  99. value := float32(0)
  100. switch {
  101. case v.RuminaFilter > MaxRuminaAdJust:
  102. value = float32(5)
  103. case v.RuminaFilter > 0:
  104. value = float32(v.RuminaFilter) * 0.25
  105. case v.RuminaFilter < -MaxRuminaAdJust:
  106. value = -MaxRuminaAdJust * RumtoHeat
  107. default:
  108. value = float32(v.RuminaFilter) * RumtoHeat
  109. }
  110. cft = float32(v.ChangeFilter)*float32(v.FilterCorrect)/100 - value
  111. }
  112. // 最近3天最大发情记录,小于该变化趋势的不再插入
  113. eventEstrus := e.GetBeforeThreeDaysCowEstrus(v.CowId, startDate.AddDate(0, 0, -2).Format(model.LayoutTime))
  114. if int32(cft) <= eventEstrus.DaylongHigh {
  115. continue
  116. }
  117. // 判断最近50天内是否存在发情记录(发情等级>=2),如果18~25天@xadjust21,如果36~50天@xadjust42
  118. cowEstrus := e.GetTwoEstrus(v.CowId, startDate.AddDate(0, 0, -100).Format(model.LayoutTime), startDate.AddDate(0, 0, -2).Format(model.LayoutTime))
  119. activeDateTime, _ := time.Parse(model.LayoutTime, cowEstrus.ActiveDate)
  120. if activeDateTime.Unix() >= startDate.AddDate(0, 0, -25).Unix() && activeDateTime.Unix() <= startDate.AddDate(0, 0, -18).Unix() {
  121. cowEstrus.HadJust = XAdjust21
  122. }
  123. if activeDateTime.Unix() >= startDate.AddDate(0, 0, -50).Unix() && activeDateTime.Unix() <= startDate.AddDate(0, 0, -36).Unix() {
  124. cowEstrus.HadJust = XAdjust42
  125. }
  126. if int32(cft)+cowEstrus.HadJust <= int32(xToday.ActiveLow) {
  127. continue
  128. }
  129. cowInfo := e.FindCowInfoByNeckRingNumber(v.NeckRingNumber)
  130. level := pasturePb.EstrusLevel_High
  131. if int32(cft)+cowEstrus.HadJust < int32(xToday.ActiveMiddle) {
  132. level = pasturePb.EstrusLevel_Low
  133. }
  134. if int32(cft)+cowEstrus.HadJust >= int32(xToday.ActiveHigh) {
  135. level = pasturePb.EstrusLevel_Middle
  136. }
  137. result := pasturePb.EstrusResult_Invalid
  138. if eventEstrus.Result == pasturePb.EstrusResult_Fail && eventEstrus.DaylongHigh > int32(cft)+cowEstrus.HadJust {
  139. result = pasturePb.EstrusResult_Fail
  140. }
  141. // todo 待定
  142. if result == pasturePb.EstrusResult_Invalid {
  143. result = pasturePb.EstrusResult_Correct
  144. }
  145. // 更新 已孕 + 不显示
  146. isShow := pasturePb.IsShow_Ok
  147. if cowInfo.IsPregnant == pasturePb.IsShow_Ok && level == pasturePb.EstrusLevel_Low {
  148. isShow = pasturePb.IsShow_No
  149. }
  150. eventEstrusList = append(eventEstrusList, &model.EventEstrus{
  151. CowId: cowInfo.Id,
  152. Lact: cowInfo.Lact,
  153. ExposeEstrusType: pasturePb.ExposeEstrusType_Natural_Estrus,
  154. FilterHigh: v.FilterHigh,
  155. EstrusDate: v.ActiveTime,
  156. ActiveDate: v.ActiveTime,
  157. LastEstrusDate: cowEstrus.ActiveDate,
  158. Level: level,
  159. IsPeak: pasturePb.IsShow_No,
  160. DaylongHigh: int32(cft) + cowEstrus.HadJust,
  161. Result: result,
  162. IsShow: isShow,
  163. })
  164. }
  165. if len(eventEstrusList) > 0 {
  166. if err = e.DB.Create(eventEstrusList).Error; err != nil {
  167. zaplog.Error("CowEstrusWarning", zap.Any("eventEstrusList", eventEstrusList), zap.Any("err", err))
  168. }
  169. }
  170. startDate = startDate.AddDate(0, 0, 1)
  171. }
  172. return nil
  173. }
  174. // UpdateNewEstrus 更新新发情数据
  175. func (e *Entry) UpdateNewEstrus(xToday *XToday) (err error) {
  176. return nil
  177. }
  178. // IsAdJustLow 是否低于最低活动量
  179. func (e *Entry) IsAdJustLow(xToday *XToday, habit *model.NeckActiveHabit) bool {
  180. ruminaAdJust := float64(0)
  181. switch {
  182. case habit.RuminaFilter > MaxRuminaAdJust:
  183. ruminaAdJust = 5
  184. case habit.RuminaFilter > 0:
  185. ruminaAdJust = float64(habit.RuminaFilter) * 0.25
  186. case habit.RuminaFilter < -MaxRuminaAdJust:
  187. ruminaAdJust = -MaxRuminaAdJust * RumtoHeat
  188. default:
  189. ruminaAdJust = float64(habit.RuminaFilter) * RumtoHeat
  190. }
  191. ruminaAdJustSum := int32(float64(habit.FilterCorrect)/100 - ruminaAdJust)
  192. isContinue := int32(0)
  193. activeLow := xToday.ActiveLow - XAdjust21
  194. if habit.ChangeFilter >= NormalChangJust {
  195. isContinue = habit.ChangeFilter - habit.ChangeAdjust + 3*ruminaAdJustSum
  196. } else {
  197. isContinue = habit.ChangeFilter * ruminaAdJustSum
  198. }
  199. if isContinue < int32(activeLow) {
  200. return true
  201. }
  202. return false
  203. }