neck_active_habit.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. package model
  2. import (
  3. "fmt"
  4. "kpt-pasture/util"
  5. "math"
  6. "strconv"
  7. "strings"
  8. pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
  9. )
  10. const (
  11. InitChangeFilter = -10000
  12. DefaultChangeFilter = -99
  13. DefaultRuminaFilter = -99
  14. DefaultChewFilter = -99
  15. DefaultFilterCorrect = 100
  16. DefaultWeeklyActive = 1500
  17. DefaultRecordCount = 6
  18. LowActivity = 62
  19. MiddleActivity = 80
  20. )
  21. type NeckActiveHabit struct {
  22. Id int64 `json:"id"`
  23. PastureId int64 `json:"pastureId"`
  24. NeckRingNumber string `json:"neckRingNumber"`
  25. CowId int64 `json:"cowId"`
  26. EarNumber string `json:"earNumber"`
  27. Lact int32 `json:"lact"`
  28. CalvingAge int32 `json:"calvingAge"`
  29. ActiveTime string `json:"activeTime"`
  30. Frameid int32 `json:"frameid"`
  31. HeatDate string `json:"heatDate"`
  32. Rumina int32 `json:"rumina"`
  33. Intake int32 `json:"intake"`
  34. Inactive int32 `json:"inactive"`
  35. Gasp int32 `json:"gasp"`
  36. Other int32 `json:"other"`
  37. High int32 `json:"high"`
  38. Active int32 `json:"active"`
  39. FilterHigh int32 `json:"filterHigh"`
  40. FilterRumina int32 `json:"filterRumina"`
  41. FilterChew int32 `json:"filterChew"`
  42. WeekHigh int32 `json:"weekHigh"`
  43. HighHabit int32 `json:"highHabit"`
  44. RuminaHabit int32 `json:"ruminaHabit"`
  45. IntakeHabit int32 `json:"intakeHabit"`
  46. ChewHabit int32 `json:"chewHabit"`
  47. InactiveHabit int32 `json:"inactiveHabit"`
  48. OtherHabit int32 `json:"otherHabit"`
  49. ChangeHigh int32 `json:"changeHigh"`
  50. ChangeRumina int32 `json:"changeRumina"`
  51. ChangeChew int32 `json:"changeChew"`
  52. ChangeAdjust int32 `json:"changeAdjust"`
  53. ChangeFilter int32 `json:"changeFilter"`
  54. RuminaFilter int32 `json:"ruminaFilter"`
  55. ChewFilter int32 `json:"chewFilter"`
  56. FilterCorrect int32 `json:"filterCorrect"`
  57. SumRumina int32 `json:"sumRumina"`
  58. SumIntake int32 `json:"sumIntake"`
  59. SumInactive int32 `json:"sumInactive"`
  60. SumActive int32 `json:"sumActive"`
  61. SumMinHigh int32 `json:"sumMinHigh"`
  62. SumMaxHigh int32 `json:"sumMaxHigh"`
  63. SumMinChew int32 `json:"SumMinChew"`
  64. BeforeThreeSumRumina int32 `json:"beforeThreeSumRumina"`
  65. BeforeThreeSumIntake int32 `json:"beforeThreeSumIntake"`
  66. Score int32 `json:"score"`
  67. IsShow pasturePb.IsShow_Kind `json:"isShow"`
  68. Cft float32 `json:"cft"`
  69. Voltage int32 `json:"voltage"`
  70. RecordCount int32 `json:"recordCount"`
  71. FirmwareVersion int32 `json:"firmwareVersion"`
  72. CreatedAt int64 `json:"createdAt"`
  73. UpdatedAt int64 `json:"updatedAt"`
  74. }
  75. func (n *NeckActiveHabit) UnShardTableName() string {
  76. return "neck_active_habit"
  77. }
  78. /*func (n *NeckActiveHabit) TableName() string {
  79. return fmt.Sprintf("%s_%06d", n.UnShardTableName(), n.PastureId)
  80. }*/
  81. func (n *NeckActiveHabit) TableName() string {
  82. return "neck_active_habit"
  83. }
  84. func (n *NeckActiveHabit) SumAvg() {
  85. n.Rumina = n.Rumina / n.RecordCount * n.RecordCount
  86. n.Inactive = n.Inactive / n.RecordCount * n.RecordCount
  87. n.Active = n.Active / n.RecordCount * n.RecordCount
  88. n.Intake = n.Intake / n.RecordCount * n.RecordCount
  89. n.Other = n.Other / n.RecordCount * n.RecordCount
  90. n.Gasp = n.Gasp / n.RecordCount * n.RecordCount
  91. n.High = n.High / n.RecordCount * n.RecordCount
  92. }
  93. func (n *NeckActiveHabit) UpdateIsShowOk() {
  94. n.IsShow = pasturePb.IsShow_Ok
  95. }
  96. func NewNeckActiveHabit(data *NeckRingOriginalMerge) *NeckActiveHabit {
  97. return &NeckActiveHabit{
  98. PastureId: data.PastureId,
  99. Frameid: data.XframeId,
  100. HeatDate: data.ActiveDate,
  101. NeckRingNumber: data.NeckRingNumber,
  102. Active: data.Active,
  103. Gasp: data.Gasp,
  104. High: data.High,
  105. Inactive: data.Inactive,
  106. Intake: data.Intake,
  107. Other: data.Other,
  108. Rumina: data.Rumina,
  109. IsShow: data.IsShow,
  110. WeekHigh: DefaultWeeklyActive,
  111. ChangeFilter: InitChangeFilter,
  112. FilterCorrect: InitChangeFilter,
  113. RuminaFilter: InitChangeFilter,
  114. ChewFilter: InitChangeFilter,
  115. ActiveTime: fmt.Sprintf("%s %02d:00:00", data.ActiveDate, data.XframeId*2+1),
  116. RecordCount: data.RecordCount,
  117. FirmwareVersion: data.FirmwareVersion,
  118. Voltage: data.Voltage,
  119. }
  120. }
  121. type NeckActiveHabitSlice []*NeckActiveHabit
  122. func (n NeckActiveHabitSlice) Len() int { return len(n) }
  123. func (n NeckActiveHabitSlice) Swap(i, j int) { n[i], n[j] = n[j], n[i] }
  124. func (n NeckActiveHabitSlice) Less(i, j int) bool {
  125. if n[i].HeatDate != n[j].HeatDate {
  126. return n[i].HeatDate < n[j].HeatDate
  127. }
  128. return n[i].Frameid < n[j].Frameid
  129. }
  130. func (n NeckActiveHabitSlice) ToPB(curveName string) *CowBehaviorCurveData {
  131. res := &CowBehaviorCurveData{
  132. OriginalDateList: make([]int32, 0),
  133. ChangeDateList: make([]int32, 0),
  134. SumDateList: make([]int32, 0),
  135. DateTimeList: make([]string, 0),
  136. EstrusList: make(map[pasturePb.EstrusLevel_Kind][]string),
  137. EventList: make([]*pasturePb.CowEvent, 0),
  138. EventMap: make(map[pasturePb.EventType_Kind]string),
  139. RuminaChange: make([]int32, 0),
  140. IQR1: make([]int32, 0),
  141. IQR3: make([]int32, 0),
  142. }
  143. initFrameId := int32(0)
  144. dateFrameMap := make(map[string][]int32)
  145. for _, v := range n {
  146. if dateFrameMap[v.HeatDate] == nil {
  147. initFrameId = 0
  148. dateFrameMap[v.HeatDate] = make([]int32, 0)
  149. }
  150. // 补全结尾不够的数据
  151. if initFrameId == 0 && len(res.DateTimeList) > 0 {
  152. lastDateTime := res.DateTimeList[len(res.DateTimeList)-1]
  153. lastDatePare := strings.Split(lastDateTime, " ")
  154. if len(lastDatePare) == 2 {
  155. lastDay := lastDatePare[0]
  156. lastHourStr := lastDatePare[1]
  157. lastHour, _ := strconv.ParseInt(lastHourStr, 10, 64)
  158. maxHour := util.ExpectedFrameIDs[len(util.ExpectedFrameIDs)-1]
  159. xframeId := int32(lastHour-1)/2 + 1
  160. if xframeId != maxHour {
  161. for ; xframeId <= maxHour; xframeId++ {
  162. res.DateTimeList = append(res.DateTimeList, fmt.Sprintf("%s %02d", lastDay, util.ExpectedFrameIDs[xframeId]*2+1))
  163. res.OriginalDateList = append(res.OriginalDateList, 0)
  164. res.ChangeDateList = append(res.ChangeDateList, 0)
  165. res.SumDateList = append(res.SumDateList, 0)
  166. res.RuminaChange = append(res.RuminaChange, 0)
  167. res.SumChewList = append(res.SumChewList, 0)
  168. }
  169. }
  170. }
  171. }
  172. expectedFrameId := util.ExpectedFrameIDs[initFrameId]
  173. if expectedFrameId != v.Frameid {
  174. maxFrameId := int32(math.Abs(float64(expectedFrameId - v.Frameid)))
  175. for ; expectedFrameId < maxFrameId; expectedFrameId++ {
  176. res.DateTimeList = append(res.DateTimeList, fmt.Sprintf("%s %02d", v.HeatDate, util.ExpectedFrameIDs[expectedFrameId]*2+1))
  177. res.OriginalDateList = append(res.OriginalDateList, 0)
  178. res.ChangeDateList = append(res.ChangeDateList, 0)
  179. res.SumDateList = append(res.SumDateList, 0)
  180. res.RuminaChange = append(res.RuminaChange, 0)
  181. res.SumChewList = append(res.SumChewList, 0)
  182. }
  183. initFrameId = expectedFrameId
  184. }
  185. // 格式化为到小时的字符串
  186. parsedTime, _ := util.TimeParseLocal(LayoutTime, v.ActiveTime)
  187. hourStr := parsedTime.Format(LayoutHour)
  188. res.DateTimeList = append(res.DateTimeList, hourStr)
  189. switch curveName {
  190. case "active": // 活动量
  191. changeDateList := v.ChangeFilter * v.FilterCorrect / 100
  192. if changeDateList == DefaultChangeFilter {
  193. changeDateList = 0
  194. }
  195. res.OriginalDateList = append(res.OriginalDateList, v.High)
  196. res.ChangeDateList = append(res.ChangeDateList, changeDateList)
  197. res.RuminaChange = append(res.RuminaChange, v.RuminaFilter)
  198. case "rumina": // 反刍
  199. res.OriginalDateList = append(res.OriginalDateList, v.Rumina)
  200. res.ChangeDateList = append(res.ChangeDateList, v.RuminaFilter)
  201. res.SumDateList = append(res.SumDateList, v.SumRumina)
  202. res.SumChewList = append(res.SumChewList, v.SumRumina+v.SumIntake)
  203. case "intake": // 采食
  204. res.OriginalDateList = append(res.OriginalDateList, v.Intake)
  205. res.SumDateList = append(res.SumDateList, v.SumIntake)
  206. res.SumChewList = append(res.SumChewList, v.SumRumina+v.SumIntake)
  207. case "inactive": // 休息
  208. res.OriginalDateList = append(res.OriginalDateList, v.Inactive)
  209. res.SumDateList = append(res.SumDateList, v.SumInactive)
  210. case "chew": // 咀嚼
  211. res.OriginalDateList = append(res.OriginalDateList, v.Rumina+v.Intake)
  212. res.ChangeDateList = append(res.ChangeDateList, v.ChewFilter)
  213. res.SumDateList = append(res.SumDateList, v.SumRumina+v.SumIntake)
  214. case "immobility": // 静止
  215. res.OriginalDateList = append(res.OriginalDateList, 120-v.Active)
  216. res.SumDateList = append(res.SumDateList, 60*24-v.SumActive)
  217. }
  218. initFrameId++
  219. }
  220. return res
  221. }
  222. func (n NeckActiveHabitSlice) ToPB2(dataBetween []string, groupNeckActiveHabitList []*NeckActiveHabit) *pasturePb.CowBehaviorRateData {
  223. data := &pasturePb.CowBehaviorRateData{
  224. DateTime: dataBetween,
  225. RuminaRate: make([]float32, 0),
  226. IntakeRate: make([]float32, 0),
  227. InactiveRate: make([]float32, 0),
  228. GaspRate: make([]float32, 0),
  229. }
  230. neckActiveHabitMap := make(map[string]*NeckActiveHabit)
  231. for _, v := range n {
  232. if v.HeatDate == "" {
  233. continue
  234. }
  235. if _, ok := neckActiveHabitMap[v.HeatDate]; !ok {
  236. neckActiveHabitMap[v.HeatDate] = v
  237. }
  238. }
  239. groupNeckActiveHabitMap := make(map[string]*NeckActiveHabit)
  240. for _, v := range groupNeckActiveHabitList {
  241. if v.HeatDate == "" {
  242. continue
  243. }
  244. if _, ok := groupNeckActiveHabitMap[v.HeatDate]; !ok {
  245. groupNeckActiveHabitMap[v.HeatDate] = v
  246. }
  247. }
  248. for _, t := range dataBetween {
  249. ruminaRate, intakeRate, inactiveRate, gaspRate := float32(0), float32(0), float32(0), float32(0)
  250. if v, ok := neckActiveHabitMap[t]; ok {
  251. sum := v.Rumina + v.Intake + v.Inactive + v.Gasp
  252. if sum > 0 {
  253. // 保留小数点后两位
  254. ruminaRate = float32(math.Round(float64(v.Rumina)/float64(sum)*100) / 100)
  255. intakeRate = float32(math.Round(float64(v.Intake)/float64(sum)*100) / 100)
  256. inactiveRate = float32(math.Round(float64(v.Inactive)/float64(sum)*100) / 100)
  257. gaspRate = float32(math.Round(float64(v.Gasp)/float64(sum)*100) / 100)
  258. }
  259. }
  260. data.RuminaRate = append(data.RuminaRate, ruminaRate)
  261. data.IntakeRate = append(data.IntakeRate, intakeRate)
  262. data.InactiveRate = append(data.InactiveRate, inactiveRate)
  263. data.GaspRate = append(data.GaspRate, gaspRate)
  264. groupRuminaRate, groupIntakeRate, groupInactiveRate, groupGaspRate := float32(0), float32(0), float32(0), float32(0)
  265. if v, ok := groupNeckActiveHabitMap[t]; ok {
  266. groupSum := v.Rumina + v.Intake + v.Inactive + v.Gasp
  267. if groupSum > 0 {
  268. // 保留小数点后两位
  269. groupRuminaRate = float32(math.Round(float64(v.Rumina)/float64(groupSum)*100) / 100)
  270. groupIntakeRate = float32(math.Round(float64(v.Intake)/float64(groupSum)*100) / 100)
  271. groupInactiveRate = float32(math.Round(float64(v.Inactive)/float64(groupSum)*100) / 100)
  272. groupGaspRate = float32(math.Round(float64(v.Gasp)/float64(groupSum)*100) / 100)
  273. }
  274. }
  275. data.GroupRuminaRate = append(data.GroupRuminaRate, groupRuminaRate)
  276. data.GroupIntakeRate = append(data.GroupIntakeRate, groupIntakeRate)
  277. data.GroupInactiveRate = append(data.GroupInactiveRate, groupInactiveRate)
  278. data.GroupGaspRate = append(data.GroupGaspRate, groupGaspRate)
  279. }
  280. return data
  281. }
  282. type MaxHabitIdModel struct {
  283. Id int64 `json:"id"`
  284. }
  285. type WeeklyActiveModel struct {
  286. CowId int64 `json:"cow_id"`
  287. HeatDate string `json:"heat_date"`
  288. Nb int32 `json:"nb"`
  289. High int32 `json:"high"`
  290. }
  291. type NeckRingErrorModel struct {
  292. CowId int64
  293. Nb int32
  294. Nba int32
  295. Voltage int32
  296. }