cow.go 13 KB

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