cow.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436
  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. PastureId int64 `json:"pastureId"`
  12. Sex pasturePb.Genders_Kind `json:"sex"`
  13. NeckRingNumber string `json:"neckRingNumber"`
  14. EarNumber string `json:"earNumber"`
  15. EarOldNumber string `json:"earOldNumber"`
  16. PenId int32 `json:"penId"`
  17. PenName string `json:"penName"`
  18. Lact int32 `json:"lact"`
  19. DayAge int32 `json:"dayAge"`
  20. CalvingAge int64 `json:"calvingAge"`
  21. PregnancyAge int32 `json:"pregnancyAge"` // 怀孕天数 孕检结果有阳性更新,产犊后至0
  22. AdmissionAge int32 `json:"admissionAge"`
  23. AbortionAge int64 `json:"abortionAge"` // 流产天数
  24. CowType pasturePb.CowType_Kind `json:"cowType"`
  25. BreedStatus pasturePb.BreedStatus_Kind `json:"breedStatus"`
  26. CowKind pasturePb.CowKind_Kind `json:"cowKind"`
  27. BirthWeight int64 `json:"birthWeight"`
  28. CurrentWeight int64 `json:"currentWeight"`
  29. AdmissionWeight int64 `json:"admissionWeight"`
  30. SourceId pasturePb.CowSource_Kind `json:"sourceId"`
  31. FatherNumber string `json:"fatherNumber"`
  32. MotherNumber string `json:"motherNumber"`
  33. AdmissionStatus pasturePb.AdmissionStatus_Kind `json:"admissionStatus"`
  34. IsPregnant pasturePb.IsShow_Kind `json:"isPregnant"`
  35. HealthStatus pasturePb.HealthStatus_Kind `json:"healthStatus"`
  36. WeaningAt int64 `json:"weaningAt"`
  37. BirthAt int64 `json:"birthAt"`
  38. AdmissionAt int64 `json:"admissionAt"`
  39. FirstMatingAt int64 `json:"firstMatingAt"`
  40. MatingTimes int32 `json:"matingTimes"`
  41. WeeklyActive int32 `json:"weeklyActive"`
  42. LastEstrusAt int64 `json:"lastEstrusAt"`
  43. LastCalvingAt int64 `json:"lastCalvingAt"`
  44. LastMatingAt int64 `json:"lastMatingAt"`
  45. LastBullNumber string `json:"lastBullNumber"`
  46. LastPregnantCheckAt int64 `json:"lastPregnantCheckAt"`
  47. LastDryMilkAt int64 `json:"lastDryMilkAt"`
  48. LastSecondWeight int64 `json:"lastSecondWeight"`
  49. LastSecondWeightAt int64 `json:"lastSecondWeightAt"`
  50. LastAbortionAt int64 `json:"lastAbortionAt"`
  51. LastWeightAt int64 `json:"lastWeightAt"`
  52. CreatedAt int64 `json:"createdAt"`
  53. UpdatedAt int64 `json:"updatedAt"`
  54. }
  55. func (c *Cow) TableName() string {
  56. return "cow"
  57. }
  58. // EventCalvingUpdate 产犊更新
  59. func (c *Cow) EventCalvingUpdate(calvingAt int64) {
  60. c.Lact += 1
  61. c.CalvingAge = int64(time.Now().Sub(time.Unix(calvingAt, 0)).Hours() / 24)
  62. c.MatingTimes = 0
  63. c.PregnancyAge = 0
  64. c.BreedStatus = pasturePb.BreedStatus_Calving
  65. c.IsPregnant = pasturePb.IsShow_No
  66. c.LastCalvingAt = calvingAt
  67. }
  68. // EventWeaningUpdate 断奶更新
  69. func (c *Cow) EventWeaningUpdate(weaningAt int64, penId int32, currentWeight int64) {
  70. c.PenId = penId
  71. c.WeaningAt = weaningAt
  72. c.CurrentWeight = currentWeight
  73. c.LastWeightAt = weaningAt
  74. }
  75. // EventPregnantCheckUpdate 孕检更新
  76. func (c *Cow) EventPregnantCheckUpdate(breedStatus pasturePb.BreedStatus_Kind, pregnantCheckAt int64, isPregnant pasturePb.IsShow_Kind) {
  77. c.BreedStatus = breedStatus
  78. c.LastPregnantCheckAt = pregnantCheckAt
  79. c.IsPregnant = isPregnant
  80. }
  81. // EventAbortionUpdate 流产更新
  82. func (c *Cow) EventAbortionUpdate(abortionAt int64) {
  83. c.IsPregnant = pasturePb.IsShow_No
  84. c.LastAbortionAt = abortionAt
  85. c.BreedStatus = pasturePb.BreedStatus_Abort
  86. }
  87. // EventWeightUpdate 称重更新
  88. func (c *Cow) EventWeightUpdate(weight int64, weightAt int64) {
  89. c.LastSecondWeight = c.CurrentWeight
  90. c.LastSecondWeightAt = c.LastWeightAt
  91. c.LastWeightAt = weightAt
  92. c.CurrentWeight = weight
  93. }
  94. // EventHealthStatusUpdate 健康状态更新
  95. func (c *Cow) EventHealthStatusUpdate(healthStatus pasturePb.HealthStatus_Kind) {
  96. c.HealthStatus = healthStatus
  97. }
  98. // EventPenUpdate 更新栏舍
  99. func (c *Cow) EventPenUpdate(pen *Pen) {
  100. c.PenId = pen.Id
  101. c.PenName = pen.Name
  102. }
  103. // EventEarNumberUpdate 更新脖环
  104. func (c *Cow) EventEarNumberUpdate(newEarNumber string) {
  105. c.EarOldNumber = c.EarNumber
  106. c.EarNumber = newEarNumber
  107. }
  108. type CowSlice []*Cow
  109. func (c CowSlice) ToPB(
  110. penMap map[int32]*Pen,
  111. cowTypeMap map[pasturePb.CowType_Kind]string,
  112. breedStatusMap map[pasturePb.BreedStatus_Kind]string,
  113. cowKindMap map[pasturePb.CowKind_Kind]string,
  114. cowSourceMap map[pasturePb.CowSource_Kind]string,
  115. admissionStatusMap map[pasturePb.AdmissionStatus_Kind]string,
  116. healthStatusMap map[pasturePb.HealthStatus_Kind]string,
  117. ) []*pasturePb.CowDetails {
  118. res := make([]*pasturePb.CowDetails, len(c))
  119. for i, v := range c {
  120. penName := ""
  121. if pen, ok := penMap[v.PenId]; ok {
  122. penName = pen.Name
  123. }
  124. sex := "公"
  125. if v.Sex == pasturePb.Genders_Female {
  126. sex = "母"
  127. }
  128. lastWeightAtFormat := ""
  129. if v.LastWeightAt > 0 {
  130. lastWeightAtFormat = time.Unix(v.LastWeightAt, 0).Format(LayoutDate2)
  131. }
  132. isPregnantName := ""
  133. if v.IsPregnant == pasturePb.IsShow_Ok {
  134. isPregnantName = "已孕"
  135. } else {
  136. isPregnantName = "未孕"
  137. }
  138. admissionAtFormat := ""
  139. if v.AdmissionAt > 0 {
  140. admissionAtFormat = time.Unix(v.AdmissionAt, 0).Format(LayoutDate2)
  141. }
  142. birthAtFormat := ""
  143. if v.BirthAt > 0 {
  144. birthAtFormat = time.Unix(v.BirthAt, 0).Format(LayoutDate2)
  145. }
  146. weaningAtFormat := ""
  147. if v.WeaningAt > 0 {
  148. weaningAtFormat = time.Unix(v.WeaningAt, 0).Format(LayoutDate2)
  149. }
  150. firstMatingAtFormat := ""
  151. if v.FirstMatingAt > 0 {
  152. firstMatingAtFormat = time.Unix(v.FirstMatingAt, 0).Format(LayoutDate2)
  153. }
  154. lastMatingAtFormat := ""
  155. if v.LastMatingAt > 0 {
  156. lastMatingAtFormat = time.Unix(v.LastMatingAt, 0).Format(LayoutDate2)
  157. }
  158. lastPregnantCheckAtFormat := ""
  159. if v.LastPregnantCheckAt > 0 {
  160. lastPregnantCheckAtFormat = time.Unix(v.LastPregnantCheckAt, 0).Format(LayoutDate2)
  161. }
  162. lastCalvingAtFormat := ""
  163. if v.LastCalvingAt > 0 {
  164. lastCalvingAtFormat = time.Unix(v.LastCalvingAt, 0).Format(LayoutDate2)
  165. }
  166. res[i] = &pasturePb.CowDetails{
  167. CowId: int32(v.Id),
  168. Sex: sex,
  169. NeckRingNumber: v.NeckRingNumber,
  170. PenName: penName,
  171. Lact: v.Lact,
  172. CowTypeName: cowTypeMap[v.CowType],
  173. BreedStatusName: breedStatusMap[v.BreedStatus],
  174. CowKindName: cowKindMap[v.CowKind],
  175. EarNumber: v.EarNumber,
  176. BirthWeight: float32(v.BirthWeight) / 1000,
  177. CurrentWeight: float32(v.CurrentWeight) / 1000,
  178. DayAge: v.DayAge,
  179. SourceName: cowSourceMap[v.SourceId],
  180. MontherNumber: v.MotherNumber,
  181. FatherNumber: v.FatherNumber,
  182. AdmissionStatusName: admissionStatusMap[v.AdmissionStatus],
  183. HealthStatusName: healthStatusMap[v.HealthStatus],
  184. IsPregnantName: isPregnantName,
  185. AdmissionAtFormat: admissionAtFormat,
  186. BirthAtFormat: birthAtFormat,
  187. WeaningAtFormat: weaningAtFormat,
  188. CalvingAge: int32(v.GetCalvingAge()),
  189. AbortionAge: int32(v.AbortionAge),
  190. MatingTimes: v.MatingTimes,
  191. FirstMatingAtFormat: firstMatingAtFormat,
  192. LastMatingAtFormat: lastMatingAtFormat,
  193. LastBullNumber: v.LastBullNumber,
  194. LastPregnantCheckAtFormat: lastPregnantCheckAtFormat,
  195. LastWeightAtFormat: lastWeightAtFormat,
  196. LastCalvingAtFormat: lastCalvingAtFormat,
  197. //PregnancyAge: v.PregnancyAge,
  198. }
  199. }
  200. return res
  201. }
  202. func (c CowSlice) ToPB2(penMap map[int32]*Pen, penWeightSlice PenWeightSlice) []*pasturePb.CowList {
  203. res := make([]*pasturePb.CowList, len(c))
  204. for i, v := range c {
  205. penName := ""
  206. if pen, ok := penMap[v.PenId]; ok {
  207. penName = pen.Name
  208. }
  209. penWeight := penWeightSlice.GetPenWeight(v.PenId)
  210. lastWeightDay := util.Ceil(float64(v.LastWeightAt-v.LastSecondWeightAt) / 86400)
  211. penAvgWeight := float32(0)
  212. previousStageDailyWeight := float32(0)
  213. cowPenAvgWeightDiffValue := float32(0)
  214. if penWeight != nil {
  215. penAvgWeight = float32(penWeight.AvgWeight) / 1000
  216. cowPenAvgWeightDiffValue = float32(v.CurrentWeight-int64(penWeight.AvgWeight)) / 1000
  217. if lastWeightDay > 0 {
  218. previousStageDailyWeight = float32(v.CurrentWeight-v.LastSecondWeight) / 1000 / float32(lastWeightDay)
  219. }
  220. }
  221. res[i] = &pasturePb.CowList{
  222. CowId: int32(v.Id),
  223. DayAge: v.DayAge,
  224. DailyWeightGain: float32(v.GetDayWeight()),
  225. AverageDailyWeightGain: float32(v.GetAverageDailyWeight()),
  226. EarNumber: v.EarNumber,
  227. PenName: penName,
  228. BirthAt: int32(v.BirthAt),
  229. BirthWeight: float32(v.BirthWeight) / 1000,
  230. CurrentWeight: float32(v.CurrentWeight) / 1000,
  231. LastWeightAt: int32(v.LastWeightAt),
  232. AdmissionAge: int32(v.AdmissionAge),
  233. AdmissionWeight: float32(v.AbortionAge) / 1000,
  234. PreviousStageDailyWeight: previousStageDailyWeight,
  235. PenAvgWeight: penAvgWeight,
  236. CowPenAvgWeightDiffValue: cowPenAvgWeightDiffValue,
  237. }
  238. }
  239. return res
  240. }
  241. func NewCow(req *pasturePb.EventEnterRequest) *Cow {
  242. var isPregnant = pasturePb.IsShow_No
  243. if req.BreedStatus == pasturePb.BreedStatus_Pregnant {
  244. isPregnant = pasturePb.IsShow_Ok
  245. }
  246. return &Cow{
  247. Sex: req.Sex,
  248. EarNumber: req.EarNumber,
  249. PenId: req.PenId,
  250. Lact: req.Lact,
  251. CowType: req.CowType,
  252. BreedStatus: req.BreedStatus,
  253. CowKind: req.CowKind,
  254. SourceId: req.CowSource,
  255. FatherNumber: req.FatherNumber,
  256. MotherNumber: req.MotherNumber,
  257. AdmissionStatus: pasturePb.AdmissionStatus_Admission,
  258. HealthStatus: pasturePb.HealthStatus_Health,
  259. IsPregnant: isPregnant,
  260. WeaningAt: int64(req.WeaningAt),
  261. BirthAt: int64(req.BirthAt),
  262. AdmissionWeight: int64(req.Weight * 1000),
  263. FirstMatingAt: int64(req.MatingAt),
  264. LastMatingAt: int64(req.MatingAt),
  265. LastPregnantCheckAt: int64(req.PregnancyCheckAt),
  266. AdmissionAt: time.Now().Unix(),
  267. }
  268. }
  269. func NewCalfCow(motherId int64, fatherNumber string, calf *CalvingCalf) *Cow {
  270. return &Cow{
  271. Sex: calf.Sex,
  272. EarNumber: calf.EarNumber,
  273. PenId: calf.PenId,
  274. CowType: pasturePb.CowType_Lactating_Calf, // 哺乳犊牛
  275. BreedStatus: pasturePb.BreedStatus_UnBreed, // 未配
  276. CowKind: calf.CowKind, // 牛只品种
  277. BirthWeight: calf.BirthWeight,
  278. BirthAt: calf.BirthAt,
  279. SourceId: pasturePb.CowSource_Calving, // 产犊方式
  280. FatherNumber: fatherNumber,
  281. MotherNumber: fmt.Sprintf("%d", motherId),
  282. AdmissionStatus: pasturePb.AdmissionStatus_Admission,
  283. IsPregnant: pasturePb.IsShow_No,
  284. AdmissionAt: calf.BirthAt,
  285. }
  286. }
  287. type BarCowStruct struct {
  288. Number int32 `json:"number"`
  289. TypeId pasturePb.CowType_Kind `json:"type_id"`
  290. }
  291. // BarCowStructSlice 首页牛群结构
  292. type BarCowStructSlice []*BarCowStruct
  293. func (b BarCowStructSlice) ToPB(cowTypeMap map[pasturePb.CowType_Kind]string, count int32) []*pasturePb.BarCowStruct {
  294. var pb []*pasturePb.BarCowStruct
  295. for _, v := range b {
  296. name := fmt.Sprintf("%s", cowTypeMap[v.TypeId])
  297. pb = append(pb, &pasturePb.BarCowStruct{Name: name, Value: v.Number})
  298. }
  299. return pb
  300. }
  301. // GetDayAge 日龄
  302. func (c *Cow) GetDayAge() int32 {
  303. if c.BirthAt <= 0 {
  304. return 0
  305. }
  306. return int32(math.Floor(float64(time.Now().Unix()-c.BirthAt) / 86400))
  307. }
  308. // GetCalvingAge 产后天数
  309. func (c *Cow) GetCalvingAge() int64 {
  310. if c.LastMatingAt <= 0 {
  311. return 0
  312. }
  313. return int64(math.Floor(float64(time.Now().Unix()-c.LastMatingAt) / 86400))
  314. }
  315. // GetDaysPregnant 怀孕天数
  316. func (c *Cow) GetDaysPregnant() int32 {
  317. if c.BreedStatus == pasturePb.BreedStatus_Pregnant &&
  318. c.AdmissionStatus == pasturePb.AdmissionStatus_Admission &&
  319. c.IsPregnant == pasturePb.IsShow_Ok {
  320. return int32(math.Floor(float64(time.Now().Unix()-c.LastMatingAt) / 86400))
  321. }
  322. return 0
  323. }
  324. // GetLactationDays 泌乳天数
  325. func (c *Cow) GetLactationDays() int32 {
  326. if c.BreedStatus == pasturePb.BreedStatus_Calving && c.AdmissionStatus == pasturePb.AdmissionStatus_Admission {
  327. return int32(math.Floor(float64(time.Now().Unix()-c.LastCalvingAt) / 86400))
  328. }
  329. return 0
  330. }
  331. // GetAdmissionAge 入场天数
  332. func (c *Cow) GetAdmissionAge() int32 {
  333. if c.AdmissionAt > 0 && c.AdmissionStatus == pasturePb.AdmissionStatus_Admission {
  334. return int32(math.Floor(float64(time.Now().Unix()-c.AdmissionAt) / 86400))
  335. }
  336. return 0
  337. }
  338. // GetDayWeight 日增重
  339. func (c *Cow) GetDayWeight() float64 {
  340. if c.CurrentWeight-c.LastSecondWeight > 0 && c.LastWeightAt > c.LastSecondWeightAt {
  341. days := int32(math.Floor(float64(c.LastWeightAt-c.LastSecondWeightAt) / 86400))
  342. if days <= 0 {
  343. return 0
  344. }
  345. dayWeight := math.Round(1.0 * float64(c.CurrentWeight-c.LastSecondWeight) / float64(days))
  346. return dayWeight / 1000
  347. }
  348. return 0
  349. }
  350. // GetAverageDailyWeight 平均日增重
  351. func (c *Cow) GetAverageDailyWeight() float64 {
  352. if c.CurrentWeight-c.BirthWeight > 0 && c.LastWeightAt > c.BirthAt {
  353. days := int32(math.Floor(float64(c.LastWeightAt-c.BirthAt) / 86400))
  354. if days <= 0 {
  355. return 0
  356. }
  357. dailyWeight := math.Round(1.0 * float64(c.CurrentWeight-c.BirthWeight) / float64(days))
  358. return dailyWeight / 1000
  359. }
  360. return 0
  361. }
  362. func (c *Cow) GetAbortionAge() int32 {
  363. if c.LastAbortionAt > 0 && c.AdmissionStatus == pasturePb.AdmissionStatus_Admission {
  364. return int32(math.Floor(float64(time.Now().Unix()-c.LastAbortionAt) / 86400))
  365. }
  366. return 0
  367. }
  368. type CowWeightRange struct {
  369. WeightRange string `json:"weight_range"`
  370. Count int32 `json:"count"`
  371. }
  372. func (c CowSlice) WeightRangeToPB(penMap map[int32]*Pen) []*pasturePb.CowList {
  373. res := make([]*pasturePb.CowList, len(c))
  374. for i, v := range c {
  375. penName := ""
  376. if pen, ok := penMap[v.PenId]; ok {
  377. penName = pen.Name
  378. }
  379. res[i] = &pasturePb.CowList{
  380. CowId: int32(v.Id),
  381. DayAge: v.DayAge,
  382. DailyWeightGain: float32(v.GetDayWeight()),
  383. AverageDailyWeightGain: float32(v.GetAverageDailyWeight()),
  384. EarNumber: v.EarNumber,
  385. PenName: penName,
  386. BirthAt: int32(v.BirthAt),
  387. BirthWeight: float32(v.BirthWeight) / 1000,
  388. CurrentWeight: float32(v.CurrentWeight) / 1000,
  389. LastWeightAt: int32(v.LastWeightAt),
  390. }
  391. }
  392. return res
  393. }