cow.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  1. package model
  2. import (
  3. "fmt"
  4. "kpt-pasture/util"
  5. "math"
  6. "time"
  7. pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
  8. )
  9. type Cow struct {
  10. Id int64 `json:"id"`
  11. Sex pasturePb.Genders_Kind `json:"sex"`
  12. NeckRingNumber string `json:"neckRingNumber"`
  13. EarNumber string `json:"earNumber"`
  14. EarOldNumber string `json:"earOldNumber"`
  15. PenId int32 `json:"penId"`
  16. Lact int32 `json:"lact"`
  17. DayAge int32 `json:"dayAge"`
  18. CalvingAge int64 `json:"calvingAge"`
  19. PregnancyAge int32 `json:"pregnancyAge"` // 怀孕天数 孕检结果有阳性更新,产犊后至0
  20. AdmissionAge int32 `json:"admissionAge"`
  21. AbortionAge int64 `json:"abortionAge"` // 流产天数
  22. CowType pasturePb.CowType_Kind `json:"cowType"`
  23. BreedStatus pasturePb.BreedStatus_Kind `json:"breedStatus"`
  24. CowKind pasturePb.CowKind_Kind `json:"cowKind"`
  25. BirthWeight int64 `json:"birthWeight"`
  26. CurrentWeight int64 `json:"currentWeight"`
  27. AdmissionWeight int64 `json:"admissionWeight"`
  28. SourceId pasturePb.CowSource_Kind `json:"sourceId"`
  29. FatherNumber string `json:"fatherNumber"`
  30. MotherNumber string `json:"motherNumber"`
  31. AdmissionStatus pasturePb.AdmissionStatus_Kind `json:"admissionStatus"`
  32. IsPregnant pasturePb.IsShow_Kind `json:"isPregnant"`
  33. HealthStatus pasturePb.HealthStatus_Kind `json:"healthStatus"`
  34. WeaningAt int64 `json:"weaningAt"`
  35. BirthAt int64 `json:"birthAti"`
  36. AdmissionAt int64 `json:"admissionAt"`
  37. FirstMatingAt int64 `json:"firstMatingAt"`
  38. LastEstrusAt int64 `json:"lastEstrusAt"`
  39. LastCalvingAt int64 `json:"lastCalvingAt"`
  40. LastMatingAt int64 `json:"lastMatingAt"`
  41. LastBullNumber string `json:"lastBullNumber"`
  42. LastPregnantCheckAt int64 `json:"lastPregnantCheckAt"`
  43. LastDryMilkAt int64 `json:"lastDryMilkAt"`
  44. LastSecondWeight int64 `json:"lastSecondWeight"`
  45. LastSecondWeightAt int64 `json:"lastSecondWeightAt"`
  46. LastAbortionAt int64 `json:"lastAbortionAt"`
  47. LastWeightAt int64 `json:"lastWeightAt"`
  48. CreatedAt int64 `json:"createdAt"`
  49. UpdatedAt int64 `json:"updatedAt"`
  50. }
  51. func (c *Cow) TableName() string {
  52. return "cow"
  53. }
  54. type CowSlice []*Cow
  55. func (c CowSlice) ToPB(
  56. penMap map[int32]*Pen,
  57. cowTypeMap map[pasturePb.CowType_Kind]string,
  58. breedStatusMap map[pasturePb.BreedStatus_Kind]string,
  59. cowKindMap map[pasturePb.CowKind_Kind]string,
  60. ) []*pasturePb.SearchCowList {
  61. res := make([]*pasturePb.SearchCowList, len(c))
  62. for i, v := range c {
  63. penName := ""
  64. if pen, ok := penMap[v.PenId]; ok {
  65. penName = pen.Name
  66. }
  67. res[i] = &pasturePb.SearchCowList{
  68. CowId: int32(v.Id),
  69. Sex: v.Sex,
  70. NeckRingNumber: v.NeckRingNumber,
  71. EarNumber: v.EarNumber,
  72. PenId: v.PenId,
  73. PenName: penName,
  74. CowType: int32(v.CowType),
  75. Lact: v.Lact,
  76. CowTypeName: cowTypeMap[v.CowType],
  77. BreedStatus: v.BreedStatus,
  78. BreedStatusName: breedStatusMap[v.BreedStatus],
  79. CowKind: v.CowKind,
  80. CowKindName: cowKindMap[v.CowKind],
  81. BirthAt: int32(v.BirthAt),
  82. BirthWeight: int32(v.BirthWeight) / 1000,
  83. CurrentWeight: int32(v.CurrentWeight) / 1000,
  84. LastWeightAt: int32(v.LastWeightAt),
  85. }
  86. }
  87. return res
  88. }
  89. func (c CowSlice) ToPB2(penMap map[int32]*Pen, penWeightSlice PenWeightSlice) []*pasturePb.CowList {
  90. res := make([]*pasturePb.CowList, len(c))
  91. for i, v := range c {
  92. penName := ""
  93. if pen, ok := penMap[v.PenId]; ok {
  94. penName = pen.Name
  95. }
  96. penWeight := penWeightSlice.GetPenWeight(v.PenId)
  97. lastWeightDay := util.Ceil(float64(v.LastWeightAt-v.LastSecondWeightAt) / 86400)
  98. penAvgWeight := float32(0)
  99. previousStageDailyWeight := float32(0)
  100. cowPenAvgWeightDiffValue := float32(0)
  101. if penWeight != nil {
  102. penAvgWeight = float32(penWeight.AvgWeight) / 1000
  103. cowPenAvgWeightDiffValue = float32(v.CurrentWeight-int64(penWeight.AvgWeight)) / 1000
  104. if lastWeightDay > 0 {
  105. previousStageDailyWeight = float32(v.CurrentWeight-v.LastSecondWeight) / 1000 / float32(lastWeightDay)
  106. }
  107. }
  108. res[i] = &pasturePb.CowList{
  109. CowId: int32(v.Id),
  110. DayAge: v.DayAge,
  111. DailyWeightGain: float32(v.GetDayWeight()),
  112. AverageDailyWeightGain: float32(v.GetAverageDailyWeight()),
  113. EarNumber: v.EarNumber,
  114. PenName: penName,
  115. BirthAt: int32(v.BirthAt),
  116. BirthWeight: float32(v.BirthWeight) / 1000,
  117. CurrentWeight: float32(v.CurrentWeight) / 1000,
  118. LastWeightAt: int32(v.LastWeightAt),
  119. AdmissionAge: int32(v.AdmissionAge),
  120. AdmissionWeight: float32(v.AbortionAge) / 1000,
  121. PreviousStageDailyWeight: previousStageDailyWeight,
  122. PenAvgWeight: penAvgWeight,
  123. CowPenAvgWeightDiffValue: cowPenAvgWeightDiffValue,
  124. }
  125. }
  126. return res
  127. }
  128. func NewCow(req *pasturePb.EventEnterRequest) *Cow {
  129. var isPregnant = pasturePb.IsShow_No
  130. if req.BreedStatus == pasturePb.BreedStatus_Pregnant {
  131. isPregnant = pasturePb.IsShow_Ok
  132. }
  133. return &Cow{
  134. Sex: req.Sex,
  135. EarNumber: req.EarNumber,
  136. PenId: req.PenId,
  137. Lact: req.Lact,
  138. CowType: req.CowType,
  139. BreedStatus: req.BreedStatus,
  140. CowKind: req.CowKind,
  141. SourceId: req.CowSource,
  142. FatherNumber: req.FatherNumber,
  143. MotherNumber: req.MotherNumber,
  144. AdmissionStatus: pasturePb.AdmissionStatus_Admission,
  145. HealthStatus: pasturePb.HealthStatus_Health,
  146. IsPregnant: isPregnant,
  147. WeaningAt: int64(req.WeaningAt),
  148. BirthAt: int64(req.BirthAt),
  149. AdmissionWeight: int64(req.Weight * 1000),
  150. FirstMatingAt: int64(req.MatingAt),
  151. LastMatingAt: int64(req.MatingAt),
  152. LastPregnantCheckAt: int64(req.PregnancyCheckAt),
  153. AdmissionAt: time.Now().Unix(),
  154. }
  155. }
  156. func NewCalfCow(motherId int64, fatherNumber string, calf *CalvingCalf) *Cow {
  157. return &Cow{
  158. Sex: calf.Sex,
  159. EarNumber: calf.EarNumber,
  160. PenId: calf.PenId,
  161. CowType: pasturePb.CowType_Lactating_Calf, // 哺乳犊牛
  162. BreedStatus: pasturePb.BreedStatus_UnBreed, // 未配
  163. CowKind: calf.CowKind, // 牛只品种
  164. BirthWeight: calf.BirthWeight,
  165. BirthAt: calf.BirthAt,
  166. SourceId: pasturePb.CowSource_Calving, // 产犊方式
  167. FatherNumber: fatherNumber,
  168. MotherNumber: fmt.Sprintf("%d", motherId),
  169. AdmissionStatus: pasturePb.AdmissionStatus_Admission,
  170. IsPregnant: pasturePb.IsShow_No,
  171. AdmissionAt: calf.BirthAt,
  172. }
  173. }
  174. type BarCowStruct struct {
  175. Number int32 `json:"number"`
  176. TypeId pasturePb.CowType_Kind `json:"type_id"`
  177. }
  178. // BarCowStructSlice 首页牛群结构
  179. type BarCowStructSlice []*BarCowStruct
  180. func (b BarCowStructSlice) ToPB(cowTypeMap map[pasturePb.CowType_Kind]string, count int32) []*pasturePb.BarCowStruct {
  181. var pb []*pasturePb.BarCowStruct
  182. for _, v := range b {
  183. name := fmt.Sprintf("%s", cowTypeMap[v.TypeId])
  184. pb = append(pb, &pasturePb.BarCowStruct{Name: name, Value: v.Number})
  185. }
  186. return pb
  187. }
  188. // GetDayAge 日龄
  189. func (c *Cow) GetDayAge() int32 {
  190. if c.BirthAt <= 0 {
  191. return 0
  192. }
  193. return int32(math.Floor(float64(time.Now().Unix()-c.BirthAt) / 86400))
  194. }
  195. // GetCalvingAge 产后天数
  196. func (c *Cow) GetCalvingAge() int64 {
  197. if c.LastMatingAt <= 0 {
  198. return 0
  199. }
  200. return int64(math.Floor(float64(time.Now().Unix()-c.LastMatingAt) / 86400))
  201. }
  202. // GetDaysPregnant 怀孕天数
  203. func (c *Cow) GetDaysPregnant() int32 {
  204. if c.BreedStatus == pasturePb.BreedStatus_Pregnant &&
  205. c.AdmissionStatus == pasturePb.AdmissionStatus_Admission &&
  206. c.IsPregnant == pasturePb.IsShow_Ok {
  207. return int32(math.Floor(float64(time.Now().Unix()-c.LastMatingAt) / 86400))
  208. }
  209. return 0
  210. }
  211. // GetLactationDays 泌乳天数
  212. func (c *Cow) GetLactationDays() int32 {
  213. if c.BreedStatus == pasturePb.BreedStatus_Calving && c.AdmissionStatus == pasturePb.AdmissionStatus_Admission {
  214. return int32(math.Floor(float64(time.Now().Unix()-c.LastCalvingAt) / 86400))
  215. }
  216. return 0
  217. }
  218. // GetAdmissionAge 入场天数
  219. func (c *Cow) GetAdmissionAge() int32 {
  220. if c.AdmissionAt > 0 && c.AdmissionStatus == pasturePb.AdmissionStatus_Admission {
  221. return int32(math.Floor(float64(time.Now().Unix()-c.AdmissionAt) / 86400))
  222. }
  223. return 0
  224. }
  225. // GetDayWeight 日增重
  226. func (c *Cow) GetDayWeight() float64 {
  227. if c.CurrentWeight-c.LastSecondWeight > 0 && c.LastWeightAt > c.LastSecondWeightAt {
  228. days := int32(math.Floor(float64(c.LastWeightAt-c.LastSecondWeightAt) / 86400))
  229. if days <= 0 {
  230. return 0
  231. }
  232. dayWeight := math.Round(1.0 * float64(c.CurrentWeight-c.LastSecondWeight) / float64(days))
  233. return dayWeight / 1000
  234. }
  235. return 0
  236. }
  237. // GetAverageDailyWeight 平均日增重
  238. func (c *Cow) GetAverageDailyWeight() float64 {
  239. if c.CurrentWeight-c.BirthWeight > 0 && c.LastWeightAt > c.BirthAt {
  240. days := int32(math.Floor(float64(c.LastWeightAt-c.BirthAt) / 86400))
  241. if days <= 0 {
  242. return 0
  243. }
  244. dailyWeight := math.Round(1.0 * float64(c.CurrentWeight-c.BirthWeight) / float64(days))
  245. return dailyWeight / 1000
  246. }
  247. return 0
  248. }
  249. func (c *Cow) GetAbortionAge() int32 {
  250. if c.LastAbortionAt > 0 && c.AdmissionStatus == pasturePb.AdmissionStatus_Admission {
  251. return int32(math.Floor(float64(time.Now().Unix()-c.LastAbortionAt) / 86400))
  252. }
  253. return 0
  254. }
  255. type CowWeightRange struct {
  256. WeightRange string `json:"weight_range"`
  257. Count int32 `json:"count"`
  258. }
  259. func (c CowSlice) WeightRangeToPB(penMap map[int32]*Pen) []*pasturePb.CowList {
  260. res := make([]*pasturePb.CowList, len(c))
  261. for i, v := range c {
  262. penName := ""
  263. if pen, ok := penMap[v.PenId]; ok {
  264. penName = pen.Name
  265. }
  266. res[i] = &pasturePb.CowList{
  267. CowId: int32(v.Id),
  268. DayAge: v.DayAge,
  269. DailyWeightGain: float32(v.GetDayWeight()),
  270. AverageDailyWeightGain: float32(v.GetAverageDailyWeight()),
  271. EarNumber: v.EarNumber,
  272. PenName: penName,
  273. BirthAt: int32(v.BirthAt),
  274. BirthWeight: float32(v.BirthWeight) / 1000,
  275. CurrentWeight: float32(v.CurrentWeight) / 1000,
  276. LastWeightAt: int32(v.LastWeightAt),
  277. }
  278. }
  279. return res
  280. }