Browse Source

estrus: 脖环发情数据

Yi 3 months ago
parent
commit
62c58c1a98

+ 1 - 1
dep/di_crontab.go

@@ -77,7 +77,7 @@ func EntryCrontab(dependency CrontabDependency) *cron.Crontab {
 		panic(err)
 	}
 
-	err = newCrontab.Bind("NeckRing", cs.NeckRing, dependency.CrontabHub.NeckRingMergeData)
+	err = newCrontab.Bind("NeckRing", cs.NeckRing, dependency.CrontabHub.NeckRingOriginalMergeData)
 	if err != nil {
 		panic(err)
 	}

+ 1 - 1
go.mod

@@ -3,7 +3,7 @@ module kpt-pasture
 go 1.17
 
 require (
-	gitee.com/xuyiping_admin/go_proto v0.0.0-20241206015638-6b15cf82851d
+	gitee.com/xuyiping_admin/go_proto v0.0.0-20241217065218-21d2be069b22
 	gitee.com/xuyiping_admin/pkg v0.0.0-20241108060137-caea58c59f5b
 	github.com/dgrijalva/jwt-go v3.2.0+incompatible
 	github.com/eclipse/paho.mqtt.golang v1.4.3

+ 4 - 0
go.sum

@@ -55,6 +55,10 @@ gitee.com/xuyiping_admin/go_proto v0.0.0-20241205102558-36e3ea825ca3 h1:fHcQ+Mms
 gitee.com/xuyiping_admin/go_proto v0.0.0-20241205102558-36e3ea825ca3/go.mod h1:BKrFW6YLDectlQcQk3FYKBeXvjEiodAKJ5rq7O/QiPE=
 gitee.com/xuyiping_admin/go_proto v0.0.0-20241206015638-6b15cf82851d h1:L/gXdFvjt966dFbM56fDSS+2vdkb/5hSZUxHvK+Vw4Q=
 gitee.com/xuyiping_admin/go_proto v0.0.0-20241206015638-6b15cf82851d/go.mod h1:BKrFW6YLDectlQcQk3FYKBeXvjEiodAKJ5rq7O/QiPE=
+gitee.com/xuyiping_admin/go_proto v0.0.0-20241217030755-242924b1d633 h1:tj3mbsCdJ5FDN6X9GqhzLZ4d3WjC6gGz6geOt5esRII=
+gitee.com/xuyiping_admin/go_proto v0.0.0-20241217030755-242924b1d633/go.mod h1:BKrFW6YLDectlQcQk3FYKBeXvjEiodAKJ5rq7O/QiPE=
+gitee.com/xuyiping_admin/go_proto v0.0.0-20241217065218-21d2be069b22 h1:4qNwmfsRxhbWGst94GHgibyfTt5LAtu0oBwzpQjWyQI=
+gitee.com/xuyiping_admin/go_proto v0.0.0-20241217065218-21d2be069b22/go.mod h1:BKrFW6YLDectlQcQk3FYKBeXvjEiodAKJ5rq7O/QiPE=
 gitee.com/xuyiping_admin/pkg v0.0.0-20241108060137-caea58c59f5b h1:w05MxH7yqveRlaRbxHhbif5YjPrJFodRPfOjYhXn7Zk=
 gitee.com/xuyiping_admin/pkg v0.0.0-20241108060137-caea58c59f5b/go.mod h1:8tF25X6pE9WkFCczlNAC0K2mrjwKvhhp02I7o0HtDxY=
 github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=

+ 34 - 14
model/event_estrus.go

@@ -1,19 +1,31 @@
 package model
 
-import pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
+import (
+	"time"
+
+	pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
+)
 
 type EventEstrus struct {
 	Id                  int64                           `json:"id"`
 	CowId               int64                           `json:"cowId"`
 	DayAge              int32                           `json:"dayAge"`
-	Lact                int8                            `json:"lact"`
-	LactationDays       int32                           `json:"lactationDays"`
+	Lact                int32                           `json:"lact"`
+	CalvingAge          int32                           `json:"calvingAge"`
 	ExposeEstrusType    pasturePb.ExposeEstrusType_Kind `json:"exposeEstrusType"`
-	EstrusAt            int64                           `json:"estrusAt"`
+	FilterHigh          int32                           `json:"filter_high"`
+	EstrusDate          string                          `json:"estrusDate"`
+	ActiveDate          string                          `json:"activeDate"`
+	LastEstrusDate      string                          `json:"lastEstrusDate"`
+	Level               pasturePb.EstrusLevel_Kind      `json:"level"`
+	IsPeak              pasturePb.IsShow_Kind           `json:"isPeak"`
+	PerTwentyFourHigh   int32                           `json:"perTwentyFourHigh"`
+	Result              pasturePb.EstrusResult_Kind     `json:"result"`
 	UnMatingReasons     pasturePb.UnMatingReasons_Kind  `json:"unMatingReasons"`
 	UnMatingReasonsName string                          `json:"unMatingReasonsName"`
 	IsMating            pasturePb.IsShow_Kind           `json:"isMating"`
 	Remarks             string                          `json:"remarks"`
+	IsShow              pasturePb.IsShow_Kind           `json:"isShow"`
 	OperationId         int64                           `json:"operationId"`
 	OperationName       string                          `json:"operationName"`
 	MessageId           int64                           `json:"messageId"`
@@ -26,13 +38,19 @@ func (e *EventEstrus) TableName() string {
 	return "event_estrus"
 }
 
-func NewEventEstrus(exposeEstrusType pasturePb.ExposeEstrusType_Kind, cow *Cow, currentUser *SystemUser, operation *SystemUser, req *pasturePb.EventEstrus) *EventEstrus {
+func NewEventEstrus(
+	exposeEstrusType pasturePb.ExposeEstrusType_Kind,
+	cow *Cow,
+	currentUser *SystemUser,
+	operation *SystemUser,
+	req *pasturePb.EventEstrus,
+) *EventEstrus {
 	return &EventEstrus{
 		CowId:            cow.Id,
 		DayAge:           cow.GetDayAge(),
-		Lact:             int8(cow.Lact),
-		LactationDays:    cow.GetLactationDays(),
-		EstrusAt:         int64(req.EstrusAt),
+		Lact:             cow.Lact,
+		CalvingAge:       int32(cow.CalvingAge),
+		EstrusDate:       time.Unix(int64(req.EstrusAt), 0).Format(LayoutTime),
 		Remarks:          req.Remarks,
 		IsMating:         req.IsMathing,
 		UnMatingReasons:  req.UnMatingReasons,
@@ -50,14 +68,16 @@ func (e EstrusSlice) ToPB() []*pasturePb.SearchEstrusList {
 	res := make([]*pasturePb.SearchEstrusList, len(e))
 	for i, v := range e {
 		res[i] = &pasturePb.SearchEstrusList{
-			Id:                  int32(v.Id),
-			CowId:               int32(v.CowId),
-			DayAge:              v.DayAge,
-			Lact:                int32(v.Lact),
-			EstrusAt:            int32(v.EstrusAt),
+			Id:     int32(v.Id),
+			CowId:  int32(v.CowId),
+			DayAge: v.DayAge,
+			Lact:   v.Lact,
+			//EstrusAt:            int32(v.EstrusAt),
+			Level:               v.Level,
+			Result:              v.Result,
 			IsMathing:           v.IsMating,
 			UnMatingReasonsName: v.UnMatingReasonsName,
-			LactationDays:       v.LactationDays,
+			CalvingAge:          v.CalvingAge,
 			MessengerId:         int32(v.MessageId),
 			MessengerName:       v.MessageName,
 			Remarks:             v.Remarks,

+ 19 - 6
model/neck_active_habit.go

@@ -6,11 +6,19 @@ import (
 	pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
 )
 
+const (
+	DefaultChangeFilter  = -10000
+	DefaultRuminaFilter  = -10000
+	DefaultChewFilter    = -10000
+	DefaultFilterCorrect = 100
+)
+
 type NeckActiveHabit struct {
 	Id                      int64                 `json:"id"`
 	CowId                   int64                 `json:"cowId"`
 	NeckRingNumber          string                `json:"neckRingNumber"`
 	Lact                    int32                 `json:"lact"`
+	CalvingAge              int64                 `json:"calvingAge"`
 	Frameid                 int32                 `json:"frameid"`
 	HeatDate                string                `json:"heatDate"`
 	Rumina                  int32                 `json:"rumina"`
@@ -26,12 +34,12 @@ type NeckActiveHabit struct {
 	FilterRumina            int32                 `json:"filterRumina"`
 	FilterChew              int32                 `json:"filterChew"`
 	WeekHigh                int32                 `json:"weekHigh"`
-	WeekAvgHighHabit        int32                 `json:"weekAvgHighHabit"`
-	WeekAvgRuminaHabit      int32                 `json:"WeekAvgRuminaHabit"`
-	WeekAvgIntakeHabit      int32                 `json:"weekAvgIntakeHabit"`
-	WeekAvgChewHabit        int32                 `json:"weekAvgChewHabit"`
-	WeekAvgInactiveHabit    int32                 `json:"weekAvgInactiveHabit"`
-	WeekAvgOtherHabit       int32                 `json:"weekAvgOtherHabit"`
+	AvgHighHabit            int32                 `json:"avgHighHabit"`
+	AvgRuminaHabit          int32                 `json:"avgRuminaHabit"`
+	AvgIntakeHabit          int32                 `json:"avgIntakeHabit"`
+	AvgChewHabit            int32                 `json:"avgChewHabit"`
+	AvgInactiveHabit        int32                 `json:"avgInactiveHabit"`
+	AvgOtherHabit           int32                 `json:"avgOtherHabit"`
 	ChangeHigh              int32                 `json:"changeHigh"`
 	ChangeRumina            int32                 `json:"changeRumina"`
 	ChangeChew              int32                 `json:"changeChew"`
@@ -79,6 +87,7 @@ func NewNeckActiveHabit(defaultWeeklyActive, frameId int32, heatDate, neckRingNu
 		HeatDate:       heatDate,
 		NeckRingNumber: neckRingNumber,
 		Lact:           lact,
+		CalvingAge:     cow.CalvingAge,
 		CowId:          cowId,
 		Active:         data.Active,
 		Gasp:           data.Gasp,
@@ -90,6 +99,10 @@ func NewNeckActiveHabit(defaultWeeklyActive, frameId int32, heatDate, neckRingNu
 		WeekHigh:       weekHigh,
 		IsShow:         pasturePb.IsShow_No,
 		IsMaxTime:      pasturePb.IsShow_No,
+		ChangeFilter:   DefaultChangeFilter,
+		FilterCorrect:  DefaultFilterCorrect,
+		RuminaFilter:   DefaultRuminaFilter,
+		ChewFilter:     DefaultChewFilter,
 		ActiveTime:     fmt.Sprintf("%s %02d:00:00", heatDate, frameId),
 	}
 }

+ 2 - 0
model/system_configure.go

@@ -11,6 +11,8 @@ const (
 	ActiveLow       = "active_low"
 	ActiveMiddle    = "active_middle"
 	ActiveHigh      = "active_high"
+	MaxEstrus       = "max_estrus"
+	MaxHabit        = "max_habit"
 )
 
 type SystemConfigure struct {

+ 13 - 5
module/backend/event_breed.go

@@ -75,7 +75,10 @@ func (s *StoreEntry) CalvingCreate(ctx context.Context, req *pasturePb.EventCalv
 
 	cow := cowList[0]
 	newEventCalving := &model.EventCalving{}
-	if err = s.DB.Model(new(model.EventCalving)).Where("cow_id = ?", cow.Id).Where("lact = ?", cow.Lact).First(newEventCalving).Error; err != nil {
+	if err = s.DB.Model(new(model.EventCalving)).
+		Where("cow_id = ?", cow.Id).
+		Where("lact = ?", cow.Lact).
+		First(newEventCalving).Error; err != nil {
 		return xerr.Custom("该母牛信息不存在")
 	}
 
@@ -133,7 +136,8 @@ func (s *StoreEntry) CalvingCreate(ctx context.Context, req *pasturePb.EventCalv
 func (s *StoreEntry) EstrusList(ctx context.Context, req *pasturePb.SearchEventRequest, pagination *pasturePb.PaginationModel) (*pasturePb.EstrusEventResponse, error) {
 	estrusList := make([]*model.EventEstrus, 0)
 	var count int64 = 0
-	pref := s.DB.Table(new(model.EventEstrus).TableName())
+	pref := s.DB.Table(new(model.EventEstrus).TableName()).
+		Where("is_show = ?", pasturePb.IsShow_Ok)
 	if len(req.CowId) > 0 {
 		cowIds := strings.Split(req.CowId, ",")
 		pref.Where("cow_id IN ?", cowIds)
@@ -164,18 +168,21 @@ func (s *StoreEntry) EstrusCreate(ctx context.Context, req *pasturePb.EventEstru
 		return xerr.Custom("获取当前登录用户失败")
 	}
 	if err = s.DB.Transaction(func(tx *gorm.DB) error {
+
+		// 发情信息
 		if err = tx.Create(estrusInfo).Error; err != nil {
 			return xerr.WithStack(err)
 		}
 
+		// 配种信息
 		if estrusInfo.IsMating == pasturePb.IsShow_Ok {
 			cow, _ := s.GetCowInfoByCowId(ctx, int64(req.CowId))
-			newEventMating := model.NewEventMating(cow, estrusInfo.EstrusAt, estrusInfo.ExposeEstrusType)
+			planAt, _ := time.Parse(model.LayoutTime, estrusInfo.EstrusDate)
+			newEventMating := model.NewEventMating(cow, planAt.Unix(), estrusInfo.ExposeEstrusType)
 			if err = tx.Create(newEventMating).Error; err != nil {
 				return xerr.WithStack(err)
 			}
 		}
-
 		return nil
 	}); err != nil {
 		return xerr.WithStack(err)
@@ -193,7 +200,8 @@ func (s *StoreEntry) EstrusBatch(ctx context.Context, req *pasturePb.EventEstrus
 		}
 		estrusList = append(estrusList, estrusInfo)
 		cow, _ := s.GetCowInfoByCowId(ctx, int64(v.CowId))
-		eventMatingList = append(eventMatingList, model.NewEventMating(cow, estrusInfo.EstrusAt, estrusInfo.ExposeEstrusType))
+		planAt, _ := time.Parse(model.LayoutTime, estrusInfo.EstrusDate)
+		eventMatingList = append(eventMatingList, model.NewEventMating(cow, planAt.Unix(), estrusInfo.ExposeEstrusType))
 	}
 	if err := s.DB.Transaction(func(tx *gorm.DB) error {
 		if err := tx.Create(estrusList).Error; err != nil {

+ 1 - 1
module/backend/goods.go

@@ -136,7 +136,7 @@ func (s *StoreEntry) NeckRingLogCreateOrUpdate(ctx context.Context, req *pasture
 	if err = s.DB.Transaction(func(tx *gorm.DB) error {
 		// 解绑脖环号
 		if err = tx.Model(new(model.NeckRingLog)).
-			Where("cowId IN ?", cowIds).
+			Where("cow_id IN ?", cowIds).
 			Updates(map[string]interface{}{
 				"unbind_at":      time.Now().Unix(),
 				"status":         pasturePb.NeckRingStatus_Unbind,

+ 205 - 4
module/crontab/estrus_warning.go

@@ -2,14 +2,215 @@ package crontab
 
 import (
 	"context"
+	"kpt-pasture/model"
+	"time"
+
+	pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
+
+	"gitee.com/xuyiping_admin/pkg/logger/zaplog"
+	"go.uber.org/zap"
+
+	"gitee.com/xuyiping_admin/pkg/xerr"
 
 	"github.com/hibiken/asynq"
 )
 
-func (e *Entry) CowEstrus(ctx context.Context, t *asynq.Task) error {
-	/*activeLowValue := e.GetSystemConfigure(model.ActiveLow)
-	activeMiddleValue := e.GetSystemConfigure(model.ActiveMiddle)
-	activeHighValue := e.GetSystemConfigure(model.ActiveHigh)*/
+const (
+	MaxRuminaAdJust = 20
+	XAdjust21       = 15
+	XAdjust42       = 10
+	RumtoHeat       = 0.5
+	MinCalvingAge   = 20
+	MinLact         = 0
+	NormalChangJust = 10
+)
+
+func (e *Entry) EntryCowEstrus(ctx context.Context, t *asynq.Task) error {
+	activeLowValue := e.GetSystemConfigure(model.ActiveLow).Value
+	activeMiddleValue := e.GetSystemConfigure(model.ActiveMiddle).Value
+	activeHighValue := e.GetSystemConfigure(model.ActiveHigh).Value
+	lastMaxEstrusId := e.GetSystemConfigure(model.MaxEstrus).Value
+
+	currentMaxHabit := &model.NeckActiveHabit{}
+	if err := e.DB.Model(new(model.NeckActiveHabit)).
+		Order("id desc").
+		First(currentMaxHabit).Error; err != nil {
+		return xerr.WithStack(err)
+	}
 
+	xToday := &XToday{}
+	if err := e.DB.Model(new(model.NeckActiveHabit)).
+		Select(`MIN(h.heat_date) as x_beg_date, MAX(h.heat_date) as x_end_date`).
+		Where("id BETWEEN ? AND ?", lastMaxEstrusId, currentMaxHabit).
+		First(xToday).Error; err != nil {
+		return xerr.WithStack(err)
+	}
+
+	// 当前Id<=上次执行的id,则不执行
+	if currentMaxHabit.Id <= int64(lastMaxEstrusId) {
+		return nil
+	}
+
+	xToday.LastMaxHabitId = int64(lastMaxEstrusId)
+	xToday.CurrMaxHabitId = currentMaxHabit.Id
+	xToday.ActiveLow = int64(activeLowValue)
+	xToday.ActiveMiddle = int64(activeMiddleValue)
+	xToday.ActiveHigh = int64(activeHighValue)
+
+	if err := e.CowEstrusWarning(ctx, xToday); err != nil {
+		return xerr.WithStack(err)
+	}
 	return nil
 }
+
+func (e *Entry) CowEstrusWarning(ctx context.Context, xToday *XToday) error {
+	startDate, err := time.Parse(model.LayoutDate2, xToday.XBegDate)
+	if err != nil {
+		return xerr.WithStack(err)
+	}
+	endDate, err := time.Parse(model.LayoutDate2, xToday.XEndDate)
+	if err != nil {
+		return xerr.WithStack(err)
+	}
+
+	for startDate.Format(model.LayoutDate2) <= endDate.Format(model.LayoutDate2) {
+		neckActiveHabitList := make([]*model.NeckActiveHabit, 0)
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
+			Select("*,MAX(filter_high) as filter_high").
+			Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
+			Where("heat_date = ?", startDate.Format(model.LayoutDate2)).
+			Where("cow_id > ?", 0).
+			Where(e.DB.Where("calving_age > ?", MinCalvingAge).Or("lact = ?", MinLact)).
+			Group("cow_id").
+			Find(&neckActiveHabitList).
+			Error; err != nil {
+			return xerr.WithStack(err)
+		}
+
+		eventEstrusList := make([]*model.EventEstrus, 0)
+		for _, v := range neckActiveHabitList {
+			if ok := e.IsAdJustLow(ctx, xToday, v); ok {
+				continue
+			}
+			cft := float32(0)
+			if v.ChangeAdjust > 10 {
+				cft = float32(v.ChangeFilter) - float32(v.ChangeAdjust) + 3
+			} else {
+				value := float32(0)
+				switch {
+				case v.RuminaFilter > MaxRuminaAdJust:
+					value = float32(5)
+				case v.RuminaFilter > 0:
+					value = float32(v.RuminaFilter) * 0.25
+				case v.RuminaFilter < -MaxRuminaAdJust:
+					value = -MaxRuminaAdJust * RumtoHeat
+				default:
+					value = float32(v.RuminaFilter) * RumtoHeat
+				}
+				cft = float32(v.ChangeFilter)*float32(v.FilterCorrect)/100 - value
+			}
+			// 最近3天最大发情记录,小于该变化趋势的不再插入
+			eventEstrus := e.GetBeforeThreeDaysCowEstrus(v.CowId, startDate.AddDate(0, 0, -2).Format(model.LayoutTime))
+			if eventEstrus.CowId != v.CowId {
+				if int32(cft) <= eventEstrus.PerTwentyFourHigh {
+					continue
+				}
+			}
+			// 判断最近50天内是否存在发情记录(发情等级>=2),如果18~25天@xadjust21,如果36~50天@xadjust42
+			cowEstrus := e.GetTwoEstrus(v.CowId, startDate.AddDate(0, 0, -100).Format(model.LayoutTime), startDate.AddDate(0, 0, -2).Format(model.LayoutTime))
+			if cowEstrus.CowId == v.CowId {
+				activeDateTime, _ := time.Parse(model.LayoutTime, cowEstrus.ActiveDate)
+				if activeDateTime.Unix() >= startDate.AddDate(0, 0, -25).Unix() && activeDateTime.Unix() <= startDate.AddDate(0, 0, -18).Unix() {
+					cowEstrus.HadJust = XAdjust21
+				}
+				if activeDateTime.Unix() >= startDate.AddDate(0, 0, -50).Unix() && activeDateTime.Unix() <= startDate.AddDate(0, 0, -36).Unix() {
+					cowEstrus.HadJust = XAdjust42
+				}
+			}
+			if int32(cft)+cowEstrus.HadJust <= int32(xToday.ActiveLow) {
+				continue
+			}
+
+			cowInfo, err := e.GetCowById(v.CowId)
+			if err != nil {
+				zaplog.Error("CowEstrusWarning", zap.Any("GetCowById", err), zap.Any("cowId", v.CowId))
+				continue
+			}
+
+			level := pasturePb.EstrusLevel_High
+			if int32(cft)+cowEstrus.HadJust < int32(xToday.ActiveMiddle) {
+				level = pasturePb.EstrusLevel_Low
+			}
+			if int32(cft)+cowEstrus.HadJust >= int32(xToday.ActiveHigh) {
+				level = pasturePb.EstrusLevel_Middle
+			}
+
+			result := pasturePb.EstrusResult_Invalid
+			if eventEstrus.Result == pasturePb.EstrusResult_Fail && eventEstrus.PerTwentyFourHigh > int32(cft)+cowEstrus.HadJust {
+				result = pasturePb.EstrusResult_Fail
+			}
+			// todo 待定
+			if result == pasturePb.EstrusResult_Invalid {
+				result = pasturePb.EstrusResult_Correct
+			}
+
+			isShow := pasturePb.IsShow_Ok
+			if cowInfo.IsPregnant == pasturePb.IsShow_Ok {
+				isShow = pasturePb.IsShow_No
+			}
+
+			eventEstrusList = append(eventEstrusList, &model.EventEstrus{
+				CowId:             v.CowId,
+				DayAge:            cowInfo.DayAge,
+				Lact:              cowInfo.Lact,
+				CalvingAge:        int32(cowInfo.CalvingAge),
+				ExposeEstrusType:  pasturePb.ExposeEstrusType_Natural_Estrus,
+				FilterHigh:        v.FilterHigh,
+				EstrusDate:        v.ActiveTime,
+				ActiveDate:        v.ActiveTime,
+				LastEstrusDate:    cowEstrus.ActiveDate,
+				Level:             level,
+				IsPeak:            pasturePb.IsShow_No,
+				PerTwentyFourHigh: int32(cft) + cowEstrus.HadJust,
+				Result:            result,
+				IsShow:            isShow,
+			})
+		}
+		if len(eventEstrusList) > 0 {
+			if err = e.DB.Create(eventEstrusList).Error; err != nil {
+				zaplog.Error("CowEstrusWarning", zap.Any("eventEstrusList", eventEstrusList), zap.Any("err", err))
+			}
+		}
+		startDate.AddDate(0, 0, 1)
+	}
+
+	return nil
+}
+
+// IsAdJustLow 是否低于最低活动量
+func (e *Entry) IsAdJustLow(ctx context.Context, xToday *XToday, habit *model.NeckActiveHabit) bool {
+	ruminaAdJust := float64(0)
+	switch {
+	case habit.RuminaFilter > MaxRuminaAdJust:
+		ruminaAdJust = 5
+	case habit.RuminaFilter > 0:
+		ruminaAdJust = float64(habit.RuminaFilter) * 0.25
+	case habit.RuminaFilter < -MaxRuminaAdJust:
+		ruminaAdJust = -MaxRuminaAdJust * RumtoHeat
+	default:
+		ruminaAdJust = float64(habit.RuminaFilter) * RumtoHeat
+	}
+	ruminaAdJustSum := int32(float64(habit.FilterCorrect)/100 - ruminaAdJust)
+	isContinue := int32(0)
+	activeLow := xToday.ActiveLow - XAdjust21
+	if habit.ChangeFilter >= NormalChangJust {
+		isContinue = habit.ChangeFilter - habit.ChangeAdjust + 3*ruminaAdJustSum
+	} else {
+		isContinue = habit.ChangeFilter * ruminaAdJustSum
+	}
+
+	if isContinue < int32(activeLow) {
+		return true
+	}
+	return false
+}

+ 1 - 1
module/crontab/interface.go

@@ -34,5 +34,5 @@ type Crontab interface {
 	UpdateSameTime() error
 	SystemBasicCrontab() error
 	CowPregnant() error
-	NeckRingMergeData() error
+	NeckRingOriginalMergeData() error
 }

+ 34 - 13
module/crontab/model.go

@@ -1,21 +1,27 @@
 package crontab
 
 type XToday struct {
-	XTodaySBegFrameid int64  `gorm:"xTodaySBegFrameid"`
-	XBegDate          string `gorm:"xBegDate"`
-	XTodaySMaxFrameid int64  `gorm:"xTodaySMaxFrameid"`
-	XEndDate          string `gorm:"xEndDate"`
-	xEndUpdateActId   int64  `gorm:"xEndUpdateActId"`
+	XBegDate       string
+	XEndDate       string
+	LastMaxHabitId int64
+	CurrMaxHabitId int64
+	XMin2Id        int64
+	XMin7Id        int64
+	ActiveLowest   int64
+	RuminaLowest   int64
+	ActiveLow      int64
+	ActiveMiddle   int64
+	ActiveHigh     int64
 }
 
-type WeeklyHabit struct {
-	CowId                int64
-	WeekAvgHighHabit     int32
-	WeekAvgRuminaHabit   int32
-	WeekAvgChewHabit     int32
-	WeekAvgInactiveHabit int32
-	WeekAvgIntakeHabit   int32
-	WeekAvgOtherHabit    int32
+type AvgHabit struct {
+	CowId            int64
+	AvgHighHabit     int32
+	AvgRuminaHabit   int32
+	AvgChewHabit     int32
+	AvgInactiveHabit int32
+	AvgIntakeHabit   int32
+	AvgOtherHabit    int32
 }
 
 type SumHabit struct {
@@ -47,3 +53,18 @@ type ActivityVolume struct {
 	StdFilter int32
 	Nb        int32
 }
+
+type CowEstrusOriginal struct {
+	CowId      int64
+	Lact       int32
+	CalvingAge int64
+	CreateTime string
+	Cft        float32
+	High       int32
+}
+
+type CowEstrus struct {
+	CowId      int64
+	ActiveDate string
+	HadJust    int32
+}

+ 168 - 148
module/crontab/neck_ring.go

@@ -13,14 +13,16 @@ import (
 )
 
 const (
-	DefaultLimit        = 10000
-	DefaultChangeFilter = 0
-	MinChangeFilter     = -99
-	MinChangeHigh       = -99
+	DefaultLimit    = 10000
+	MinChangeFilter = -99
+	MinRuminaFilter = -99
+	MinChewFilter   = -99
+	MinChangeHigh   = -99
+	DefaultNb       = 30
 )
 
-// NeckRingMergeData 把
-func (e *Entry) NeckRingMergeData() error {
+// NeckRingOriginalMergeData 把脖环数据合并成2个小时的
+func (e *Entry) NeckRingOriginalMergeData() error {
 	// 先看看上次任务有没有执行结束,结束在执行下面的任务
 	if ok := e.IsExistCrontabLog(NeckRingOriginal); ok {
 		return nil
@@ -115,68 +117,64 @@ func (e *Entry) NeckRingMergeData() error {
 }
 
 func (e *Entry) ActiveHabit() error {
-	// 上次任务未处理完的数据
-	lastNeckActiveHabit := &model.NeckActiveHabit{}
+	lastMaxHabitId := e.GetSystemConfigure(model.MaxHabit).Value
+	currentMaxHabit := &model.NeckActiveHabit{}
 	if err := e.DB.Model(new(model.NeckActiveHabit)).
-		Where("is_show = ?", pasturePb.IsShow_No).
-		Order("id").First(lastNeckActiveHabit).Error; err != nil {
+		Order("id desc").First(currentMaxHabit).Error; err != nil {
 		return xerr.WithStack(err)
 	}
 
-	xToday := &XToday{}
-	if err := e.DB.Model(new(model.NeckActiveHabit)).
-		Select(`MIN(TO_DAYS(h.heat_date)*1000 + h.frameid) as xTodaySBegFrameid, MIN(h.heat_date) as xBegDate, 
-		MAX(TO_DAYS(h.heat_date)*1000 + h.frameid) as xTodaySMaxFrameid, MAX(h.heat_date) as xEndDate, MAX(h.id) as xEndUpdateActId`).
-		Where("id >= ?", lastNeckActiveHabit.Id).First(xToday).Error; err != nil {
-		return xerr.WithStack(err)
+	// 本次执行<=上次执行的id,则不执行
+	if currentMaxHabit.Id < int64(lastMaxHabitId) {
+		return nil
 	}
 
-	minHeatDate := struct {
-		HeatDate string
-	}{}
-
+	// 统一更新is_max_time为0
 	if err := e.DB.Model(new(model.NeckActiveHabit)).
-		Select("MIN(heat_date) as heat_date").
-		Where("id >= ?", lastNeckActiveHabit.Id).
-		First(&minHeatDate).Error; err != nil {
+		Where("is_max_time = ?", pasturePb.IsShow_Ok).
+		Update("is_max_time", pasturePb.IsShow_No).Error; err != nil {
 		return xerr.WithStack(err)
 	}
-	minHeatDateParse, _ := time.Parse(model.LayoutDate2, minHeatDate.HeatDate)
-	xMin2Id := struct {
-		Id int64
-	}{}
+
+	// 获取这段执行数据内最大日期和最小日期
+	xToday := &XToday{}
 	if err := e.DB.Model(new(model.NeckActiveHabit)).
-		Select("IFNULL(MIN(id), ?) as id", lastNeckActiveHabit.Id).
-		Where("heat_date = ?", minHeatDateParse.AddDate(0, 0, -1).Format(model.LayoutDate2)).
-		First(&xMin2Id).Error; err != nil {
+		Select(`MIN(h.heat_date) as x_beg_date, MAX(h.heat_date) as x_end_date`).
+		Where("id BETWEEN ? AND ?", lastMaxHabitId, currentMaxHabit).
+		First(xToday).Error; err != nil {
 		return xerr.WithStack(err)
 	}
 
-	xMin7Id := struct {
-		Id int64
-	}{}
-	XBegDateTime, _ := time.Parse(model.LayoutDate2, xToday.XBegDate)
-	if err := e.DB.Model(new(model.NeckActiveHabit)).
-		Select("MIN(id) as id").
-		Where("heat_date >= ?", XBegDateTime.AddDate(0, 0, -7).Format(model.LayoutDate2)).
-		First(&xMin7Id).Error; err != nil {
+	xToday.LastMaxHabitId = int64(lastMaxHabitId)
+	xToday.CurrMaxHabitId = currentMaxHabit.Id
+
+	minHeatDateParse, err := time.Parse(model.LayoutDate2, xToday.XBegDate)
+	if err != nil {
 		return xerr.WithStack(err)
 	}
-	// 更新is_max_time
-	if err := e.DB.Model(new(model.NeckActiveHabit)).
-		Where("is_max_time = ?", pasturePb.IsShow_Ok).
-		Update("is_max_time", pasturePb.IsShow_No).Error; err != nil {
+	xBefore2Day := minHeatDateParse.AddDate(0, 0, -1).Format(model.LayoutDate2)
+	xBefore7Day := minHeatDateParse.AddDate(0, 0, -7).Format(model.LayoutDate2)
+	xMin2Id, err := e.GetMinIdByHeatDate(xBefore2Day, xToday.LastMaxHabitId)
+	if err != nil {
 		return xerr.WithStack(err)
 	}
-	xBegDate, _ := time.Parse(model.LayoutDate2, xToday.XBegDate)
-	before7xBegDate := xBegDate.AddDate(0, 0, 7).Format(model.LayoutDate2)
+
+	xMin7Id, err := e.GetMinIdByHeatDate(xBefore7Day, xToday.LastMaxHabitId)
+	if err != nil {
+		return xerr.WithStack(err)
+	}
+	xToday.XMin2Id = xMin2Id
+	xToday.XMin7Id = xMin7Id
+
+	//  id到上一次执行结果并且heat_date > 7天之前的最大牛只id置为is_max_time=1
 	sqlQuery := e.DB.Model(new(model.NeckActiveHabit)).
 		Select("MAX(id) as id").
-		Where("id BETWEEN ? AND ?", xMin2Id.Id, lastNeckActiveHabit.Id).
+		Where("id BETWEEN ? AND ?", xToday.XMin2Id, xToday.LastMaxHabitId).
 		Where("change_filter > ?", MinChangeFilter).
-		Where("heat_date >", before7xBegDate).Group("cow_id")
+		Where("heat_date >", xBefore7Day).
+		Group("cow_id")
 
-	if err := e.DB.Model(new(model.NeckActiveHabit)).
+	if err = e.DB.Model(new(model.NeckActiveHabit)).
 		Joins("JOIN (?) bb ON neck_active_habit.id = bb.id", sqlQuery).
 		Update("is_max_time", pasturePb.IsShow_Ok).Error; err != nil {
 		return xerr.WithStack(err)
@@ -184,24 +182,34 @@ func (e *Entry) ActiveHabit() error {
 
 	activeLowest := e.GetSystemConfigure(model.ActiveLowest)
 	ruminaLowest := e.GetSystemConfigure(model.RuminaLowest)
+	xToday.ActiveLowest = int64(activeLowest.Value)
+	xToday.RuminaLowest = int64(ruminaLowest.Value)
 
 	// 更新活动滤波
-	if err := e.FilterUpdate(activeLowest.Value, ruminaLowest.Value); err != nil {
+	if err = e.FilterUpdate(xToday); err != nil {
 		return xerr.WithStack(err)
 	}
 	// 更新周平均值
-	if err := e.WeeklyActiveAvgUpdate(xMin2Id.Id, activeLowest.Value, ruminaLowest.Value, xToday, lastNeckActiveHabit); err != nil {
+	if err = e.WeeklyActiveAvgUpdate(xToday); err != nil {
+		return xerr.WithStack(err)
+	}
+
+	// 更新最后一次执行的id值
+	if err = e.DB.Model(new(model.SystemConfigure)).
+		Where("name = ?", model.MaxHabit).
+		Update("value = ?", xToday.CurrMaxHabitId+1).
+		Error; err != nil {
 		return xerr.WithStack(err)
 	}
 	return nil
 }
 
 // FilterUpdate 更新活动滤波
-func (e *Entry) FilterUpdate(activeLowest, ruminaLowest int32) error {
+func (e *Entry) FilterUpdate(xToDay *XToday) error {
 	newNeckActiveHabitList := make([]*model.NeckActiveHabit, 0)
 	if err := e.DB.Model(new(model.NeckActiveHabit)).
-		Where(e.DB.Where("change_filter = ?", DefaultChangeFilter).Or("is_max_time = ?", pasturePb.IsShow_Ok)).
-		Where(e.DB.Where("high >= ?", activeLowest).Or("rumina >= ?", ruminaLowest)).
+		Where(e.DB.Where("change_filter = ?", model.DefaultChangeFilter).Or("is_max_time = ?", pasturePb.IsShow_Ok)).
+		Where(e.DB.Where("high >= ?", xToDay.ActiveLowest).Or("rumina >= ?", xToDay.RuminaLowest)).
 		Order("cow_id,id").
 		Find(&newNeckActiveHabitList).Error; err != nil {
 		return xerr.WithStack(err)
@@ -245,56 +253,46 @@ func (e *Entry) FilterUpdate(activeLowest, ruminaLowest int32) error {
 	return nil
 }
 
-func (e *Entry) WeeklyActiveAvgUpdate(xMin2Id int64, activeLowest, ruminaLowest int32, xToday *XToday, lastNeckActiveHabit *model.NeckActiveHabit) error {
-	before7DaysMinId := struct {
-		Id int64 `json:"id"`
-	}{
-		Id: 0,
-	}
-	before7DayDate, _ := time.Parse(model.LayoutDate2, xToday.XBegDate)
-	before7DayDateStr := before7DayDate.AddDate(0, 0, -7).Format(model.LayoutDate2)
-	before1DayDateStr := before7DayDate.AddDate(0, 0, -1).Format(model.LayoutDate2)
-
-	if err := e.DB.Model(new(model.NeckActiveHabit)).
-		Select("MIN(id) as id").
-		Where("heat_date >= ?", before7DayDateStr).
-		First(&before7DaysMinId).Error; err != nil {
+func (e *Entry) WeeklyActiveAvgUpdate(xToday *XToday) error {
+	beginDayDate, err := time.Parse(model.LayoutDate2, xToday.XBegDate)
+	if err != nil {
 		return xerr.WithStack(err)
 	}
+	before7DayDate := beginDayDate.AddDate(0, 0, -7).Format(model.LayoutDate2)
+	before1DayDate := beginDayDate.AddDate(0, 0, -1).Format(model.LayoutDate2)
 
 	weeklyActive := e.GetSystemConfigure(model.WeeklyActive)
 	xframeId := int64(0)
 	maxXframeId := int64(11)
-	currDate, _ := time.Parse(model.LayoutDate2, xToday.XBegDate)
-	XEndDateTime, _ := time.Parse(model.LayoutDate2, xToday.XEndDate)
-	for currDate.Format(model.LayoutDate2) < XEndDateTime.Format(model.LayoutDate2) || (currDate == XEndDateTime && xframeId <= maxXframeId) {
-
+	xStartDate, _ := time.Parse(model.LayoutDate2, xToday.XBegDate)
+	xEndDate, _ := time.Parse(model.LayoutDate2, xToday.XEndDate)
+	for xStartDate.Format(model.LayoutDate2) < xEndDate.Format(model.LayoutDate2) || (xStartDate == xEndDate && xframeId <= maxXframeId) {
 		//  时间点周平均
-		weeklyHabitList := make([]*WeeklyHabit, 0)
-		if err := e.DB.Model(new(model.NeckActiveHabit)).
+		AvgHabitList := make([]*AvgHabit, 0)
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
 			Select("cow_id").
-			Select("IF(COUNT(1)>=3, ROUND((SUM(filter_high) -MIN(filter_high) -MAX(filter_high))/ABS(COUNT(1) -2),0), -1) as week_avg_high_habit").
-			Select("IF(COUNT(1)>=3, ROUND((SUM(filter_rumina) -MIN(filter_rumina) -MAX(filter_rumina))/ABS(COUNT(1) -2),0), -1) as week_avg_rumina_habit").
-			Select("IF(COUNT(1)>=3, ROUND((SUM(filter_chew) -MIN(filter_chew) -MAX(filter_chew))/ABS(COUNT(1) -2),0), -1) as week_avg_chew_habit").
-			Select("ROUND(AVG(intake),0) as week_avg_intake_habit").
-			Select("ROUND(AVG(inactive),0) as week_avg_inactive_habit").
-			Where("id BETWEEN ? AND ?", xMin2Id, xToday.xEndUpdateActId).
-			Where("heat_date BETWEEN ? AND ?", before7DayDateStr, before1DayDateStr).
+			Select("IF(COUNT(1)>=3, ROUND((SUM(filter_high) -MIN(filter_high) -MAX(filter_high))/ABS(COUNT(1) -2),0), -1) as avg_high_habit").
+			Select("IF(COUNT(1)>=3, ROUND((SUM(filter_rumina) -MIN(filter_rumina) -MAX(filter_rumina))/ABS(COUNT(1) -2),0), -1) as avg_rumina_habit").
+			Select("IF(COUNT(1)>=3, ROUND((SUM(filter_chew) -MIN(filter_chew) -MAX(filter_chew))/ABS(COUNT(1) -2),0), -1) as avg_chew_habit").
+			Select("ROUND(AVG(intake),0) as avg_intake_habit").
+			Select("ROUND(AVG(inactive),0) as avg_inactive_habit").
+			Where("id BETWEEN ? AND ?", xToday.XMin7Id, xToday.CurrMaxHabitId).
+			Where("heat_date BETWEEN ? AND ?", before7DayDate, before1DayDate).
 			Where("frameid = ?", xframeId).
-			Where("change_filter = ?", DefaultChangeFilter).
-			Where(e.DB.Where("high > ?", activeLowest).Or("rumina > ?", ruminaLowest)).
+			Where("change_filter = ?", model.DefaultChangeFilter).
+			Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina > ?", xToday.RuminaLowest)).
 			Group("cow_id").
-			Find(&weeklyHabitList).Error; err != nil {
+			Find(&AvgHabitList).Error; err != nil {
 			return xerr.WithStack(err)
 		}
-		for _, v := range weeklyHabitList {
+		for _, v := range AvgHabitList {
 			if err := e.DB.Model(new(model.NeckActiveHabit)).
-				Select("week_avg_high_habit", "week_avg_rumina_habit", "week_avg_chew_habit", "week_avg_intake_habit", "week_avg_inactive_habit").
+				Select("week_avg_high_habit", "avg_rumina_habit", "avg_chew_habit", "avg_intake_habit", "avg_inactive_habit").
 				Where("cow_id = ?", v.CowId).
-				Where("id BETWEEN ? AND ?", lastNeckActiveHabit.Id, xToday.xEndUpdateActId).
+				Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
 				Where("frameid = ?", xframeId).
-				Where("change_filter = ?", DefaultChangeFilter).
-				Where("heat_date = ?", currDate).
+				Where("change_filter = ?", model.DefaultChangeFilter).
+				Where("heat_date = ?", xStartDate).
 				Updates(v).Error; err != nil {
 				return xerr.WithStack(err)
 			}
@@ -302,7 +300,7 @@ func (e *Entry) WeeklyActiveAvgUpdate(xMin2Id int64, activeLowest, ruminaLowest
 
 		// 累计24小时数值
 		sumHabitList := make([]*SumHabit, 0)
-		if err := e.DB.Model(new(model.NeckActiveHabit)).
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
 			Select("cow_id").
 			Select("IF(COUNT(1)>6, ROUND(AVG( h2.filter_rumina)*12,0), 0) as sum_rumina").
 			Select("IF(COUNT(1)>6, ROUND(AVG( h2.intake)*12,0), 0) as sum_intake").
@@ -310,50 +308,55 @@ func (e *Entry) WeeklyActiveAvgUpdate(xMin2Id int64, activeLowest, ruminaLowest
 			Select("IF(COUNT(1)>6, ROUND(AVG( h2.active)*12,0), 0) as sum_active").
 			Select("MAX(h2.change_filter) as sum_max_high").
 			Select("MIN(IF(change_filter > ?, change_filter, 0)) as sum_min_high", MinChangeFilter).
-			Select("MIN( CASE WHEN filter_chew > ? THEN filter_chew WHEN filter_rumina >= ? THEN filter_rumina ELSE 0 END) as sum_min_chew", MinChangeFilter, MinChangeFilter).
-			Where("id BETWEEN ? AND ?", before7DaysMinId.Id, xToday.xEndUpdateActId).
-			Where("heat_date BETWEEN ? AND ?", currDate.AddDate(0, 0, -1).Format(model.LayoutDate2), currDate.Format(model.LayoutDate2)).
-			Where("created_at BETWEEN ? AND ?", currDate.Add(-23*time.Hour), currDate.Unix()).
-			Where(e.DB.Where("high > ?", activeLowest).Or("rumina >= ?", ruminaLowest)).
+			Select("MIN( "+
+				"CASE WHEN filter_chew > ? "+
+				"THEN filter_chew "+
+				"WHEN filter_rumina >= ? "+
+				"THEN filter_rumina "+
+				"ELSE 0 END) as sum_min_chew", MinChangeFilter, MinRuminaFilter).
+			Where("id BETWEEN ? AND ?", xToday.XMin2Id, xToday.CurrMaxHabitId).
+			Where("heat_date BETWEEN ? AND ?", xStartDate.AddDate(0, 0, -1).Format(model.LayoutDate2), xStartDate.Format(model.LayoutDate2)).
+			Where("created_at BETWEEN ? AND ?", xStartDate.Add(-23*time.Hour).Unix(), xStartDate.Unix()).
+			Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina >= ?", xToday.RuminaLowest)).
 			Group("cow_id").
 			Find(&sumHabitList).Error; err != nil {
 			return xerr.WithStack(err)
 		}
 
 		for _, v := range sumHabitList {
-			if err := e.DB.Model(new(model.NeckActiveHabit)).
+			if err = e.DB.Model(new(model.NeckActiveHabit)).
 				Select("sum_rumina", "sum_intake", "sum_inactive", "sum_active", "sum_max_high", "sum_min_high", "sum_min_chew").
 				Where("cow_id = ?", v.CowId).
-				Where("id BETWEEN ? AND ?", lastNeckActiveHabit.Id, xToday.xEndUpdateActId).
-				Where("heat_date = ?", currDate.Format(model.LayoutDate2)).
+				Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
+				Where("heat_date = ?", xStartDate.Format(model.LayoutDate2)).
 				Where("frameid = ?", xframeId).
-				Where("change_filter = ?", DefaultChangeFilter).
+				Where("change_filter = ?", model.DefaultChangeFilter).
 				Updates(v).Error; err != nil {
 				return xerr.WithStack(err)
 			}
 		}
 		// 变化百分比
 		changeHabitList := make([]*model.NeckActiveHabit, 0)
-		if err := e.DB.Model(new(model.NeckActiveHabit)).
-			Where("id BETWEEN ? AND ?", lastNeckActiveHabit.Id, xToday.xEndUpdateActId).
-			Where("heat_date = ?", currDate.Format(model.LayoutDate2)).
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
+			Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
+			Where("heat_date = ?", xStartDate.Format(model.LayoutDate2)).
 			Where("frameid = ?", xframeId).
-			Where("change_filter = ?", DefaultChangeFilter).
+			Where("change_filter = ?", model.DefaultChangeFilter).
 			Where("week_avg_high_habit > ?", 0).
-			Where(e.DB.Where("high > ?", activeLowest).Or("rumina >= ?", ruminaLowest)).
+			Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina >= ?", xToday.RuminaLowest)).
 			Find(&changeHabitList).Error; err != nil {
 			return xerr.WithStack(err)
 		}
 
 		for _, v := range changeHabitList {
-			if v.FilterHigh-v.WeekAvgHighHabit > 0 {
-				v.ChangeHigh = (v.FilterHigh - v.WeekAvgHighHabit) / int32(float64(v.WeekHigh)*0.6+float64(v.WeekAvgHighHabit)*0.2+float64(weeklyActive.Value)*0.2)
+			if v.FilterHigh-v.AvgHighHabit > 0 {
+				v.ChangeHigh = (v.FilterHigh - v.AvgHighHabit) / int32(float64(v.WeekHigh)*0.6+float64(v.AvgHighHabit)*0.2+float64(weeklyActive.Value)*0.2)
 			} else {
-				v.ChangeHigh = v.FilterHigh - v.WeekAvgHighHabit/v.WeekAvgHighHabit*100
+				v.ChangeHigh = v.FilterHigh - v.AvgHighHabit/v.AvgHighHabit*100
 			}
 
-			v.ChangeRumina = v.RuminaFilter - v.WeekAvgRuminaHabit/v.WeekAvgHighHabit*100
-			v.ChangeChew = v.FilterChew - v.WeekAvgChewHabit/v.WeekAvgHighHabit*100
+			v.ChangeRumina = v.RuminaFilter - v.AvgRuminaHabit/v.AvgHighHabit*100
+			v.ChangeChew = v.FilterChew - v.AvgChewHabit/v.AvgHighHabit*100
 			if err := e.DB.Model(new(model.NeckActiveHabit)).
 				Select("change_high", "change_rumina", "change_chew").
 				Where("id = ?", v.Id).
@@ -364,7 +367,7 @@ func (e *Entry) WeeklyActiveAvgUpdate(xMin2Id int64, activeLowest, ruminaLowest
 
 		if xframeId == maxXframeId {
 			xframeId = 0
-			currDate = currDate.AddDate(0, 0, 1)
+			xStartDate = xStartDate.AddDate(0, 0, 1)
 		} else {
 			xframeId++
 		}
@@ -374,15 +377,15 @@ func (e *Entry) WeeklyActiveAvgUpdate(xMin2Id int64, activeLowest, ruminaLowest
 }
 
 // UpdateChangeFilter  变化趋势滤波
-func (e *Entry) UpdateChangeFilter(xMin2Id int64, xToday *XToday) error {
+func (e *Entry) UpdateChangeFilter(xToday *XToday) error {
 	xRuminaDisc := e.GetSystemConfigure(model.XRuminaDisc)
 	xChangeDiscount := e.GetSystemConfigure(model.XChangeDiscount)
 	newChangeFilterList := make([]*ChangeFilterData, 0)
 	if err := e.DB.Model(new(model.NeckActiveHabit)).
 		Select("id,cow_id,change_high,change_filter,rumina_filter,change_rumina,chew_filter,change_chew").
 		Select("IF(lact=0,0.8,1) as xlc_dis_count").
-		Where("id BETWEEN ? AND ?", xMin2Id, xToday.xEndUpdateActId).
-		Where(e.DB.Where("change_filter = ?", DefaultChangeFilter).Or("is_max_time = ?", pasturePb.IsShow_Ok)).
+		Where("id BETWEEN ? AND ?", xToday.XMin2Id, xToday.CurrMaxHabitId).
+		Where(e.DB.Where("change_filter = ?", model.DefaultChangeFilter).Or("is_max_time = ?", pasturePb.IsShow_Ok)).
 		Where("change_high > ?", MinChangeHigh).
 		Order("cow_id,heat_date,frameid").
 		Find(&newChangeFilterList).Error; err != nil {
@@ -442,7 +445,9 @@ func (e *Entry) UpdateChangeFilter(xMin2Id int64, xToday *XToday) error {
 		}
 		if err := e.DB.Model(new(model.NeckActiveHabit)).
 			Select("change_filter", "rumina_filter", "chew_filter").
-			Where("id = ?", v.Id).Where("cow_id = ?", v.CowId).
+			Where("id = ?", v.Id).
+			Where("cow_id = ?", v.CowId).
+			Where("change_filter = ?", model.DefaultChangeFilter).
 			Updates(v).Error; err != nil {
 			return xerr.WithStack(err)
 		}
@@ -453,89 +458,104 @@ func (e *Entry) UpdateChangeFilter(xMin2Id int64, xToday *XToday) error {
 }
 
 // ActivityVolumeChanges 计算活动量变化趋势校正值(活跃度校正)
-func (e *Entry) ActivityVolumeChanges(xMin7Id int64, activeLowest, ruminaLowest int32, xToday *XToday, lastNeckActiveHabitId int64) error {
+func (e *Entry) ActivityVolumeChanges(xToday *XToday) error {
 	currDate, _ := time.Parse(model.LayoutDate2, xToday.XBegDate)
 	XEndDateTime, _ := time.Parse(model.LayoutDate2, xToday.XEndDate)
 	xframeId := int64(0)
 	maxXframeId := int64(11)
 	dayTimes := int64(1)
 	for currDate.Format(model.LayoutDate2) < XEndDateTime.Format(model.LayoutDate2) || (currDate == XEndDateTime && xframeId <= maxXframeId) {
+		activityVolumeList := make([]*ActivityVolume, 0)
+		activeTime := fmt.Sprintf("%s %02d:00:00", currDate.Format(model.LayoutDate2), xframeId*2)
+		activeTimeParse, err := time.Parse(model.LayoutTime, activeTime)
+		if err != nil {
+			return xerr.WithStack(err)
+		}
 		if dayTimes == 1 {
-			ActivityVolumeList := make([]*ActivityVolume, 0)
-			if err := e.DB.Model(new(model.NeckActiveHabit)).
+			if err = e.DB.Model(new(model.NeckActiveHabit)).
 				Select("cow_id").
-				Where("id BETWEEN ? AND ?", xMin7Id, xToday.xEndUpdateActId).
+				Select("").
+				Where("id BETWEEN ? AND ?", xToday.XMin7Id, xToday.CurrMaxHabitId).
 				Where("heat_date BETWEEN ? AND ?", currDate.AddDate(0, 0, -7).Format(model.LayoutDate2), currDate.AddDate(0, 0, -1).Format(model.LayoutDate2)).
 				Where("frameid = ?", xframeId).
-				Where(e.DB.Where("high > ?", activeLowest).Or("rumina >= ?", ruminaLowest)).
-				Where("created_at <= ?", currDate.Add(-12*time.Hour).Unix()).
+				Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina >= ?", xToday.RuminaLowest)).
+				Where("active_time <= ?", activeTimeParse.Add(-12*time.Hour)).
+				Where("change_filter > ?", MinChangeFilter).
+				Having("nb > ?", DefaultNb).
 				Group("cow_id").
-				Having("nb > ?", 30).
-				Find(&ActivityVolumeList).Error; err != nil {
+				Find(&activityVolumeList).Error; err != nil {
 				return xerr.WithStack(err)
 			}
-			for _, v := range ActivityVolumeList {
-				filterCorrect := 100 - math.Floor(float64(v.AvgFilter)/3+float64(v.StdFilter)/2)
-				if err := e.DB.Model(new(model.NeckActiveHabit)).
-					Where("cow_id = ?", v.CowId).
-					Where("id BETWEEN ? AND ?", lastNeckActiveHabitId, xToday.xEndUpdateActId). // todo: 待完善
-					Where("frameid = ?", xframeId).
-					Where("head_date = ?", currDate.Format(model.LayoutDate2)).
-					Update("filter_correct", filterCorrect).Error; err != nil {
-					return xerr.WithStack(err)
-				}
+		}
+
+		for _, v := range activityVolumeList {
+			filterCorrect := model.DefaultFilterCorrect - int(math.Floor(float64(v.AvgFilter)/3+float64(v.StdFilter)/2))
+			if err = e.DB.Model(new(model.NeckActiveHabit)).
+				Where("cow_id = ?", v.CowId).
+				Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
+				Where("frameid = ?", xframeId).
+				Where("head_date = ?", currDate.Format(model.LayoutDate2)).
+				Update("filter_correct", filterCorrect).Error; err != nil {
+				return xerr.WithStack(err)
 			}
 		}
 
-		n := 0
+		/*n := 0
 		if n <= 10 {
 
 			// todo
 
 			n += 2
+		}*/
+
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
+			Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
+			Where("heat_date = ?", currDate.Format(model.LayoutDate2)).
+			Where("frameid = ?", xframeId).
+			Where("change_filter = ?", model.DefaultChangeFilter).
+			Update("change_filter", MinChangeFilter).Error; err != nil {
+			return xerr.WithStack(err)
 		}
 
-		if err := e.DB.Model(new(model.NeckActiveHabit)).
-			Where("id BETWEEN ? AND ?", lastNeckActiveHabitId, xToday.xEndUpdateActId).
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
+			Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
 			Where("heat_date = ?", currDate.Format(model.LayoutDate2)).
 			Where("frameid = ?", xframeId).
-			Where("change_filter = ?", 0).
-			Update("change_filter", MinChangeHigh).Error; err != nil {
+			Where("rumina_filter = ?", model.DefaultRuminaFilter).
+			Update("rumina_filter", MinRuminaFilter).Error; err != nil {
 			return xerr.WithStack(err)
 		}
 
-		if err := e.DB.Model(new(model.NeckActiveHabit)).
-			Where("id BETWEEN ? AND ?", lastNeckActiveHabitId, xToday.xEndUpdateActId).
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
+			Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
 			Where("heat_date = ?", currDate.Format(model.LayoutDate2)).
 			Where("frameid = ?", xframeId).
-			Where("rumina_filter = ?", 0).
-			Update("rumina_filter", MinChangeHigh).Error; err != nil {
+			Where("chew_filter = ?", model.DefaultChewFilter).
+			Update("chew_filter", MinChewFilter).Error; err != nil {
 			return xerr.WithStack(err)
 		}
 
-		if err := e.DB.Model(new(model.NeckActiveHabit)).
-			Where("id BETWEEN ? AND ?", lastNeckActiveHabitId, xToday.xEndUpdateActId).
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
+			Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
 			Where("heat_date = ?", currDate.Format(model.LayoutDate2)).
 			Where("frameid = ?", xframeId).
-			Where("filter_correct < ?", 100).
+			Where("filter_correct < ?", model.DefaultFilterCorrect).
 			Where("change_filter < ?", 0).
-			Update("filter_correct", 100).Error; err != nil {
+			Update("filter_correct", model.DefaultFilterCorrect).Error; err != nil {
 			return xerr.WithStack(err)
 		}
 
 		// 更新评分
 		newNeckActiveHabitList := make([]*model.NeckActiveHabit, 0)
-		if err := e.DB.Model(new(model.NeckActiveHabit)).
-			Where("id BETWEEN ? AND ?", lastNeckActiveHabitId, xToday.xEndUpdateActId).
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
+			Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
 			Where("heat_date = ?", currDate.Format(model.LayoutDate2)).
 			Where("frameid = ?", xframeId).
 			Where("score = ?", 0).
 			Find(&newNeckActiveHabitList).Error; err != nil {
 			return xerr.WithStack(err)
 		}
-		/*for _, v := range newNeckActiveHabitList {
-
-		}*/
+		// todo 待开发
 	}
 
 	return nil

+ 59 - 0
module/crontab/sql.go

@@ -1,12 +1,25 @@
 package crontab
 
 import (
+	"errors"
 	"kpt-pasture/model"
 
+	"gorm.io/gorm"
+
 	pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
 	"gitee.com/xuyiping_admin/pkg/xerr"
 )
 
+func (e *Entry) GetCowById(cowId int64) (*model.Cow, error) {
+	cowInfo := &model.Cow{}
+	if err := e.DB.Model(new(model.Cow)).
+		Where("id = ?", cowId).
+		Where("admission_status = ?", pasturePb.AdmissionStatus_Admission).
+		First(cowInfo).Error; err != nil {
+		return nil, err
+	}
+	return cowInfo, nil
+}
 func (e *Entry) GetPenMapList() (map[int32]*model.Pen, error) {
 	penList := make([]*model.Pen, 0)
 	if err := e.DB.Where("is_delete = ?", pasturePb.IsShow_Ok).Find(&penList).Error; err != nil {
@@ -57,3 +70,49 @@ func (e *Entry) GetSystemConfigure(name string) *model.SystemConfigure {
 	}
 	return res
 }
+
+func (e *Entry) GetMinIdByHeatDate(heatDate string, defaultId int64) (int64, error) {
+	xMinId := struct {
+		Id int64
+	}{}
+	if err := e.DB.Model(new(model.NeckActiveHabit)).
+		Select("MIN(id) as id").
+		//Where("heat_date = ?", minHeatDateParse.AddDate(0, 0, -1).Format(model.LayoutDate2)).
+		Where("heat_date >= ?", heatDate).
+		First(&xMinId).Error; err != nil {
+		if errors.Is(err, gorm.ErrRecordNotFound) {
+			xMinId.Id = defaultId
+		} else {
+			return 0, xerr.WithStack(err)
+		}
+	}
+	return xMinId.Id, nil
+}
+
+// GetBeforeThreeDaysCowEstrus 获取值得时间之前三天内最大发情记录
+func (e *Entry) GetBeforeThreeDaysCowEstrus(cowId int64, activeTime string) *model.EventEstrus {
+	eventEstrus := &model.EventEstrus{}
+	if err := e.DB.Model(new(model.EventEstrus)).
+		Select("MAX(filter_high) as filter_high,cow_id,MAX(per_twenty_four_high) as per_twenty_four_high").
+		Select("MAX(IF(e16.result=1,3,e16.result))  AS result").
+		Where("cow_id = ?", cowId).
+		Where("active_date >= ?", activeTime).
+		First(eventEstrus).Error; err != nil {
+		return eventEstrus
+	}
+	return eventEstrus
+}
+
+// GetTwoEstrus 判断最近50天内是否存在发情记录(发情等级>=2),如果18~25天@xadjust21,如果36~50天@xadjust42
+func (e *Entry) GetTwoEstrus(cowId int64, startActiveTime, endActiveTime string) *CowEstrus {
+	newCowEstrus := &CowEstrus{}
+	if err := e.DB.Model(new(model.EventEstrus)).
+		Select("cow_id,MAX(active_date) as active_date").
+		Where("cow_id = ?", cowId).
+		Where("active_date BETWEEN ? AND ?", startActiveTime, endActiveTime).
+		Where("level >= ?", pasturePb.EstrusLevel_Middle).
+		First(newCowEstrus).Error; err != nil {
+		return newCowEstrus
+	}
+	return newCowEstrus
+}