cow.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. package backend
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "kpt-pasture/model"
  7. "kpt-pasture/util"
  8. "net/http"
  9. "sort"
  10. "strings"
  11. "time"
  12. "github.com/nicksnyder/go-i18n/v2/i18n"
  13. "gorm.io/gorm"
  14. "gitee.com/xuyiping_admin/pkg/xerr"
  15. pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
  16. )
  17. func (s *StoreEntry) Detail(ctx context.Context, req *pasturePb.SearchEventRequest) (*pasturePb.CowInfoResponse, error) {
  18. userModel, err := s.GetUserModel(ctx)
  19. if err != nil {
  20. return nil, xerr.WithStack(err)
  21. }
  22. if req.EarNumber == "" && req.NeckRingNumber == "" {
  23. messageId, _ := userModel.LanguageContent.Localize(&i18n.LocalizeConfig{
  24. MessageID: "cow.inputCow",
  25. })
  26. return nil, xerr.Custom(messageId)
  27. }
  28. cowInfo := &model.Cow{}
  29. pref := s.DB.Model(new(model.Cow)).
  30. Where("pasture_id = ?", userModel.AppPasture.Id)
  31. if req.EarNumber != "" {
  32. pref.Where("ear_number = ?", strings.TrimSpace(req.EarNumber))
  33. }
  34. if req.NeckRingNumber != "" {
  35. pref.Where("neck_ring_number = ?", strings.TrimSpace(req.NeckRingNumber))
  36. }
  37. if err = pref.First(cowInfo).Error; err != nil {
  38. if errors.Is(err, gorm.ErrRecordNotFound) {
  39. messageId, _ := userModel.LanguageContent.Localize(&i18n.LocalizeConfig{
  40. MessageID: "cow.noCow",
  41. })
  42. return nil, xerr.Custom(messageId)
  43. } else {
  44. return nil, xerr.WithStack(err)
  45. }
  46. }
  47. cowTypeMap := s.CowTypeMap(userModel)
  48. breedStatusMap := s.CowBreedStatusMap(userModel)
  49. cowKindMap := s.CowKindMap(userModel)
  50. cowSourceMap := s.CowSourceMap(userModel)
  51. admissionStatusMap := s.AdmissionStatusMap(userModel)
  52. healthStatusMap := s.HealthStatusMap(userModel)
  53. purposeMap := s.PurposeMap(userModel)
  54. systemBasic, err := s.GetSystemBasicByName(ctx, userModel.AppPasture.Id, model.PregnancyAge)
  55. if err != nil {
  56. messageId, _ := userModel.LanguageContent.Localize(&i18n.LocalizeConfig{
  57. MessageID: "auth.pregnancyDays",
  58. })
  59. return nil, xerr.Custom(messageId)
  60. }
  61. cowDetails := model.CowSlice([]*model.Cow{cowInfo}).ToPB(
  62. cowTypeMap, breedStatusMap, cowKindMap, cowSourceMap,
  63. admissionStatusMap, healthStatusMap, purposeMap, systemBasic.MinValue,
  64. )
  65. if len(cowDetails) != 1 {
  66. messageId, _ := userModel.LanguageContent.Localize(&i18n.LocalizeConfig{
  67. MessageID: "auth.noCow",
  68. })
  69. return nil, xerr.Custom(messageId)
  70. }
  71. data := cowDetails[0]
  72. return &pasturePb.CowInfoResponse{
  73. Code: http.StatusOK,
  74. Msg: "ok",
  75. Data: data,
  76. }, nil
  77. }
  78. func (s *StoreEntry) List(ctx context.Context, req *pasturePb.SearchEventRequest, pagination *pasturePb.PaginationModel) (*pasturePb.SearchCowListResponse, error) {
  79. userModel, err := s.GetUserModel(ctx)
  80. if err != nil {
  81. return nil, xerr.WithStack(err)
  82. }
  83. systemBasic, err := s.GetSystemBasicByName(ctx, userModel.AppPasture.Id, model.PregnancyAge)
  84. if err != nil {
  85. messageId, _ := userModel.LanguageContent.Localize(&i18n.LocalizeConfig{
  86. MessageID: "auth.pregnancyDays",
  87. })
  88. return nil, xerr.Custom(messageId)
  89. }
  90. cowList := make([]*model.Cow, 0)
  91. var count int64 = 0
  92. pref := s.DB.Model(new(model.Cow)).
  93. Where("pasture_id = ?", userModel.AppPasture.Id)
  94. if len(req.CowId) > 0 {
  95. cowIds := strings.Split(req.CowId, ",")
  96. pref.Where("id IN ?", cowIds)
  97. }
  98. if req.EarNumber != "" {
  99. pref.Where("ear_number like ?", fmt.Sprintf("%s%s%s", "%", req.EarNumber, "%"))
  100. }
  101. if req.Id > 0 {
  102. pref.Where("id = ?", req.Id)
  103. }
  104. if req.PenId > 0 {
  105. pref.Where("pen_id = ?", req.PenId)
  106. }
  107. if req.CowType > 0 {
  108. pref.Where("cow_type = ?", req.CowType)
  109. }
  110. if req.BreedStatus > 0 {
  111. pref.Where("breed_status = ?", req.BreedStatus)
  112. }
  113. if req.CowKind > 0 {
  114. pref.Where("cow_kind = ?", req.CowKind)
  115. }
  116. if req.Sex > 0 {
  117. pref.Where("sex = ?", req.Sex)
  118. }
  119. if req.Lact > 0 {
  120. pref.Where("lact = ?", req.Lact)
  121. }
  122. if req.CowSource > 0 {
  123. pref.Where("source_id = ?", req.CowSource)
  124. }
  125. if err = pref.Order("id desc").
  126. Count(&count).
  127. Limit(int(pagination.PageSize)).
  128. Offset(int(pagination.PageOffset)).
  129. Find(&cowList).Error; err != nil {
  130. return nil, xerr.WithStack(err)
  131. }
  132. cowTypeMap := s.CowTypeMap(userModel)
  133. breedStatusMap := s.CowBreedStatusMap(userModel)
  134. cowKindMap := s.CowKindMap(userModel)
  135. cowSourceMap := s.CowSourceMap(userModel)
  136. admissionStatusMap := s.AdmissionStatusMap(userModel)
  137. healthStatusMap := s.HealthStatusMap(userModel)
  138. purposeMap := s.PurposeMap(userModel)
  139. return &pasturePb.SearchCowListResponse{
  140. Code: http.StatusOK,
  141. Msg: "ok",
  142. Data: &pasturePb.SearchCowData{
  143. List: model.CowSlice(cowList).ToPB(
  144. cowTypeMap, breedStatusMap, cowKindMap, cowSourceMap,
  145. admissionStatusMap, healthStatusMap, purposeMap, systemBasic.MinValue,
  146. ),
  147. Total: int32(count),
  148. PageSize: pagination.PageSize,
  149. Page: pagination.Page,
  150. },
  151. }, nil
  152. }
  153. func (s *StoreEntry) EventList(ctx context.Context, req *pasturePb.SearchCowEventListRequest, pagination *pasturePb.PaginationModel) (*pasturePb.CowEventListResponse, error) {
  154. userModel, err := s.GetUserModel(ctx)
  155. if err != nil {
  156. return nil, xerr.WithStack(err)
  157. }
  158. eventCowLogList := make([]*model.EventCowLog, 0)
  159. cowInfo, err := s.GetCowEventByEarNumber(ctx, userModel.AppPasture.Id, req.EarNumber)
  160. if err != nil {
  161. messageId, _ := userModel.LanguageContent.Localize(&i18n.LocalizeConfig{
  162. MessageID: "auth.errorCow",
  163. TemplateData: map[string]interface{}{"earNumber": req.EarNumber},
  164. })
  165. return nil, xerr.Customf(messageId)
  166. }
  167. eventCowLog := &model.EventCowLog{CowId: cowInfo.Id}
  168. pref := s.DB.Table(eventCowLog.TableName()).
  169. Where("pasture_id = ?", userModel.AppPasture.Id).
  170. Where("cow_id = ?", cowInfo.Id)
  171. if req.Lact >= 0 {
  172. pref.Where("lact = ?", req.Lact)
  173. }
  174. if req.EventCategoryKind > 0 {
  175. pref.Where("event_type = ?", req.EventCategoryKind)
  176. }
  177. if err = pref.Order("event_at DESC").
  178. Limit(int(pagination.PageSize)).
  179. Offset(int(pagination.PageOffset)).
  180. Find(&eventCowLogList).Error; err != nil {
  181. return nil, xerr.WithStack(err)
  182. }
  183. eventCategoryMap := s.EventCategoryMap(userModel)
  184. return &pasturePb.CowEventListResponse{
  185. Code: http.StatusOK,
  186. Msg: "ok",
  187. Data: &pasturePb.CowEventData{
  188. List: model.EventCowLogSlice(eventCowLogList).ToPB(eventCategoryMap),
  189. Total: int32(len(eventCowLogList)),
  190. PageSize: pagination.PageSize,
  191. Page: pagination.Page,
  192. },
  193. }, nil
  194. }
  195. func (s *StoreEntry) BehaviorCurve(ctx context.Context, req *pasturePb.CowBehaviorCurveRequest) (*model.CowBehaviorCurveResponse, error) {
  196. userModel, err := s.GetUserModel(ctx)
  197. if err != nil {
  198. return nil, xerr.WithStack(err)
  199. }
  200. cowInfo, err := s.GetCowEventByEarNumber(ctx, userModel.AppPasture.Id, req.EarNumber)
  201. if err != nil {
  202. messageId, _ := userModel.LanguageContent.Localize(&i18n.LocalizeConfig{
  203. MessageID: "auth.errorCow",
  204. TemplateData: map[string]interface{}{"earNumber": req.EarNumber},
  205. })
  206. return nil, xerr.Customf(messageId)
  207. }
  208. nowTime := time.Now().Local()
  209. nowDayZero := util.TimeParseLocalUnix(nowTime.Format(model.LayoutDate2))
  210. endDataTime := nowTime.Format(model.LayoutDate2)
  211. startDataTime := nowTime.AddDate(0, 0, -model.NeckRingDataDays).Format(model.LayoutDate2)
  212. dayRange, err := util.GetDaysBetween(startDataTime, endDataTime)
  213. if err != nil {
  214. return nil, xerr.WithStack(err)
  215. }
  216. if len(dayRange) <= 0 {
  217. messageId, _ := userModel.LanguageContent.Localize(&i18n.LocalizeConfig{
  218. MessageID: "auth.errorDateRange",
  219. })
  220. return nil, xerr.Customf(messageId)
  221. }
  222. // 行为曲线数据
  223. neckActiveHabitList := make([]*model.NeckActiveHabit, 0)
  224. if err = s.DB.Table(new(model.NeckActiveHabit).TableName()).
  225. Where("neck_ring_number = ?", cowInfo.NeckRingNumber).
  226. Where("pasture_id = ?", userModel.AppPasture.Id).
  227. Where("is_show = ?", pasturePb.IsShow_Ok).
  228. Where("cow_id > ?", 0).
  229. Where("heat_date IN (?)", dayRange).
  230. Order("heat_date, frameid").
  231. Find(&neckActiveHabitList).Error; err != nil {
  232. return nil, xerr.WithStack(err)
  233. }
  234. data := model.NeckActiveHabitSlice(neckActiveHabitList).ToPB(req.CurveName)
  235. q1, q3 := s.CowIQR(cowInfo, req.CurveName, dayRange, data.DateTimeList)
  236. data.IQR3 = q3
  237. data.IQR1 = q1
  238. eventMapList := s.EventTypeMap(userModel)
  239. for k, v := range eventMapList {
  240. if k == pasturePb.EventType_Enter || k == pasturePb.EventType_Body_Score || k == pasturePb.EventType_Birth ||
  241. k == pasturePb.EventType_Weaning || k == pasturePb.EventType_Sale || k == pasturePb.EventType_Weight ||
  242. k == pasturePb.EventType_Castrated {
  243. continue
  244. }
  245. data.EventMap[k] = v
  246. }
  247. data.LowActivity = model.LowActivity
  248. data.MiddleActivity = model.MiddleActivity
  249. // 牛只事件列表
  250. eventLogList := make([]*model.EventCowLog, 0)
  251. eventLog := &model.EventCowLog{CowId: cowInfo.Id}
  252. if err = s.DB.Table(eventLog.TableName()).
  253. Where("cow_id = ?", cowInfo.Id).
  254. Where("pasture_id = ?", userModel.AppPasture.Id).
  255. Where("event_at BETWEEN ? AND ?", nowDayZero-(model.NeckRingDataDays*86400), nowDayZero+86400).
  256. Order("event_at").
  257. Find(&eventLogList).Error; err != nil {
  258. return nil, xerr.WithStack(err)
  259. }
  260. for _, v := range eventLogList {
  261. if v.EventType == pasturePb.EventType_Enter || v.EventType == pasturePb.EventType_Body_Score ||
  262. v.EventType == pasturePb.EventType_Birth || v.EventType == pasturePb.EventType_Weaning ||
  263. v.EventType == pasturePb.EventType_Sale || v.EventType == pasturePb.EventType_Weight ||
  264. v.EventType == pasturePb.EventType_Castrated {
  265. continue
  266. }
  267. if v.EventAt <= 0 {
  268. continue
  269. }
  270. eventAt := time.Unix(v.EventAt, 0).Local()
  271. data.EventList = append(data.EventList, &pasturePb.CowEvent{
  272. EventTypeKind: v.EventType,
  273. EventTypeName: v.EventTypeName,
  274. EventDescription: v.EventDescription,
  275. Remarks: v.Remarks,
  276. EventAtFormat: fmt.Sprintf("%s 09", eventAt.Format(model.LayoutDate2)),
  277. })
  278. }
  279. // 发情数据
  280. estrusList := make([]*model.NeckRingEstrus, 0)
  281. if err = s.DB.Model(new(model.NeckRingEstrus)).
  282. Select("id,active_level,MAX(active_time) AS active_time,is_peak").
  283. Where("cow_id = ?", cowInfo.Id).
  284. Where("pasture_id = ?", userModel.AppPasture.Id).
  285. Where("active_time BETWEEN ? AND ?", fmt.Sprintf("%s 00:00:00", startDataTime), fmt.Sprintf("%s 23:59:59", endDataTime)).
  286. Where("is_peak = ?", pasturePb.IsShow_Ok).
  287. Group("first_time").
  288. Find(&estrusList).Error; err != nil {
  289. return nil, xerr.WithStack(err)
  290. }
  291. for _, v := range estrusList {
  292. if data.EstrusList[v.ActiveLevel] == nil {
  293. data.EstrusList[v.ActiveLevel] = make([]string, 0)
  294. }
  295. data.EstrusList[v.ActiveLevel] = append(data.EstrusList[v.ActiveLevel], strings.TrimSuffix(v.ActiveTime, ":00:00"))
  296. }
  297. return &model.CowBehaviorCurveResponse{
  298. Code: http.StatusOK,
  299. Msg: "ok",
  300. Data: data,
  301. }, nil
  302. }
  303. func (s *StoreEntry) CowIQR(cowInfo *model.Cow, curveName string, dayRange []string, dateTimeList []string) ([]int32, []int32) {
  304. q1, q3 := make([]int32, 0), make([]int32, 0)
  305. if curveName == "" || curveName == "active" || curveName == "behavior" || len(dateTimeList) <= 0 {
  306. return q1, q3
  307. }
  308. penId := cowInfo.PenId
  309. cowList := make([]*model.Cow, 0)
  310. if err := s.DB.Table(new(model.Cow).TableName()).
  311. Where("pen_id = ?", penId).
  312. Where("pasture_id = ?", cowInfo.PastureId).
  313. Where("neck_ring_number != ?", "").
  314. Find(&cowList).Error; err != nil {
  315. return q1, q3
  316. }
  317. cowIds := make([]int64, 0)
  318. for _, cow := range cowList {
  319. cowIds = append(cowIds, cow.Id)
  320. }
  321. neckActiveHabitList, err := s.GetNeckActiveHabitsConcurrent(cowInfo, cowIds, dayRange)
  322. if err != nil {
  323. return q1, q3
  324. }
  325. neckActiveHabitMap := make(map[string][]*model.NeckActiveHabit)
  326. for _, habit := range neckActiveHabitList {
  327. activeTime := habit.ActiveTime[:13]
  328. neckActiveHabitMap[activeTime] = append(neckActiveHabitMap[activeTime], habit)
  329. }
  330. penOriginal := make(map[string][]int32)
  331. for _, dt := range dateTimeList {
  332. habits, ok := neckActiveHabitMap[dt]
  333. if !ok {
  334. penOriginal[dt] = append(penOriginal[dt], int32(0))
  335. continue
  336. }
  337. switch curveName {
  338. case "rumina":
  339. for _, habit := range habits {
  340. penOriginal[dt] = append(penOriginal[dt], habit.Rumina)
  341. }
  342. case "intake":
  343. for _, habit := range habits {
  344. penOriginal[dt] = append(penOriginal[dt], habit.Inactive)
  345. }
  346. case "inactive":
  347. for _, habit := range habits {
  348. penOriginal[dt] = append(penOriginal[dt], habit.Inactive)
  349. }
  350. case "chew":
  351. for _, habit := range habits {
  352. penOriginal[dt] = append(penOriginal[dt], habit.Rumina+habit.Intake)
  353. }
  354. case "immobility":
  355. for _, habit := range habits {
  356. penOriginal[dt] = append(penOriginal[dt], 120-habit.Active)
  357. }
  358. }
  359. }
  360. if len(penOriginal) > 0 {
  361. for _, dataList := range penOriginal {
  362. dl := len(dataList)
  363. if dl < 4 {
  364. q1 = append(q1, int32(0))
  365. q3 = append(q3, int32(0))
  366. continue
  367. }
  368. sort.Slice(dataList, func(i, j int) bool {
  369. return dataList[i] < dataList[j]
  370. })
  371. v1 := (len(dataList) + 1) / 4
  372. // 防止下标溢出panic
  373. s1 := float64(dataList[v1-1]) + 0.25*float64(dataList[v1]-dataList[v1-1])
  374. v3 := v1 * 3
  375. s3 := float64(dataList[v3-1]) + 0.75*float64(dataList[v3]-dataList[v3-1])
  376. q1 = append(q1, int32(s1))
  377. q3 = append(q3, int32(s3))
  378. }
  379. }
  380. return q1, q3
  381. }