Parcourir la source

neck_ring_habit: update

Yi il y a 2 mois
Parent
commit
57c446e3bf

+ 15 - 0
cmd/neckring.go

@@ -0,0 +1,15 @@
+package cmd
+
+import (
+	"fmt"
+
+	"github.com/spf13/cobra"
+)
+
+var NeckRingCmd = &cobra.Command{
+	Use:   "neckRing",
+	Short: "merge neck ring data",
+	Run: func(cmd *cobra.Command, args []string) {
+		fmt.Println("neckRing called")
+	},
+}

+ 1 - 0
cmd/root.go

@@ -24,4 +24,5 @@ func init() {
 	RootCmd.AddCommand(CrontabCmd)
 	RootCmd.AddCommand(ConsumerCmd)
 	RootCmd.AddCommand(MqttCmd)
+	RootCmd.AddCommand(NeckRingCmd)
 }

+ 1 - 1
go.mod

@@ -3,7 +3,7 @@ module kpt-pasture
 go 1.17
 
 require (
-	gitee.com/xuyiping_admin/go_proto v0.0.0-20250107064743-4115ed6f2a73
+	gitee.com/xuyiping_admin/go_proto v0.0.0-20250115062928-2823e210486d
 	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

+ 2 - 0
go.sum

@@ -118,6 +118,8 @@ gitee.com/xuyiping_admin/go_proto v0.0.0-20250107024845-37a577c13d5e h1:P3cy/uJu
 gitee.com/xuyiping_admin/go_proto v0.0.0-20250107024845-37a577c13d5e/go.mod h1:BKrFW6YLDectlQcQk3FYKBeXvjEiodAKJ5rq7O/QiPE=
 gitee.com/xuyiping_admin/go_proto v0.0.0-20250107064743-4115ed6f2a73 h1:M53fn2/vuz8+jAi/sdKMzsubz34k5R5vdeNi4ULmp3c=
 gitee.com/xuyiping_admin/go_proto v0.0.0-20250107064743-4115ed6f2a73/go.mod h1:BKrFW6YLDectlQcQk3FYKBeXvjEiodAKJ5rq7O/QiPE=
+gitee.com/xuyiping_admin/go_proto v0.0.0-20250115062928-2823e210486d h1:qcqBgZFyNXyrg5090MnkbJY5CM5ps8ANzIWNNlwkxEs=
+gitee.com/xuyiping_admin/go_proto v0.0.0-20250115062928-2823e210486d/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=

+ 23 - 0
http/handler/cow/cow.go

@@ -60,3 +60,26 @@ func EventList(c *gin.Context) {
 	}
 	ginutil.JSONResp(c, res)
 }
+
+func BehaviorCurve(c *gin.Context) {
+	var req pasturePb.CowBehaviorCurveRequest
+	if err := ginutil.BindProto(c, &req); err != nil {
+		apierr.AbortBadRequest(c, http.StatusBadRequest, err)
+		return
+	}
+
+	if err := valid.ValidateStruct(&req,
+		valid.Field(&req.CowId, valid.Required),
+		valid.Field(&req.CurveName, valid.Required),
+	); err != nil {
+		apierr.AbortBadRequest(c, http.StatusBadRequest, err)
+		return
+	}
+
+	res, err := middleware.Dependency(c).StoreEventHub.OpsService.BehaviorCurve(c, &req)
+	if err != nil {
+		apierr.ClassifiedAbort(c, err)
+		return
+	}
+	ginutil.JSONResp(c, res)
+}

+ 1 - 0
http/route/cow_api.go

@@ -15,5 +15,6 @@ func CowAPI(opts ...func(engine *gin.Engine)) func(s *gin.Engine) {
 		eventRoute := authRouteGroup(s, "/api/v1/cow/")
 		eventRoute.POST("/list", cow.List)
 		eventRoute.POST("/event/list", cow.EventList)
+		eventRoute.POST("/behavior/curve", cow.BehaviorCurve)
 	}
 }

+ 3 - 3
model/cow.go

@@ -75,12 +75,12 @@ func (c *Cow) EventInfoUpdate() {
 // EventCalvingUpdate 产犊更新
 func (c *Cow) EventCalvingUpdate(calvingAt int64) {
 	c.Lact += 1
-	c.CalvingAge = int64(time.Now().Sub(time.Unix(calvingAt, 0)).Hours() / 24)
 	c.MatingTimes = 0
 	c.PregnancyAge = 0
 	c.BreedStatus = pasturePb.BreedStatus_Calving
 	c.IsPregnant = pasturePb.IsShow_No
 	c.LastCalvingAt = calvingAt
+	c.CalvingAge = c.GetCalvingAge()
 }
 
 // EventWeaningUpdate 断奶更新
@@ -382,10 +382,10 @@ func (c *Cow) GetDayAge() int32 {
 
 // GetCalvingAge 产后天数
 func (c *Cow) GetCalvingAge() int64 {
-	if c.LastMatingAt <= 0 {
+	if c.LastCalvingAt <= 0 {
 		return 0
 	}
-	return int64(math.Floor(float64(time.Now().Unix()-c.LastMatingAt) / 86400))
+	return int64(math.Floor(float64(time.Now().Unix()-c.LastCalvingAt) / 86400))
 }
 
 // GetDaysPregnant 怀孕天数

+ 16 - 7
model/event_estrus.go

@@ -14,13 +14,13 @@ type EventEstrus struct {
 	CowId            int64                           `json:"cowId"`
 	Lact             int32                           `json:"lact"`
 	ExposeEstrusType pasturePb.ExposeEstrusType_Kind `json:"exposeEstrusType"`
-	FilterHigh       int32                           `json:"filter_high"`
-	EstrusDate       string                          `json:"estrusDate"`
+	EstrusStartDate  string                          `json:"estrusStartDate"`
 	ActiveDate       string                          `json:"activeDate"`
 	LastEstrusDate   string                          `json:"lastEstrusDate"`
 	Level            pasturePb.EstrusLevel_Kind      `json:"level"`
 	IsPeak           pasturePb.IsShow_Kind           `json:"isPeak"`
-	DaylongHigh      int32                           `json:"daylongHigh"`
+	DayHigh          int32                           `json:"dayHigh"`
+	MaxHigh          int32                           `json:"maxHigh"`
 	Result           pasturePb.EstrusResult_Kind     `json:"result"`
 	Remarks          string                          `json:"remarks"`
 	IsShow           pasturePb.IsShow_Kind           `json:"isShow"`
@@ -37,20 +37,29 @@ func (e *EventEstrus) TableName() string {
 }
 
 func NewEventEstrus(
+	pastureId int64,
 	exposeEstrusType pasturePb.ExposeEstrusType_Kind,
 	level pasturePb.EstrusLevel_Kind,
-	isShow pasturePb.IsShow_Kind,
+	result pasturePb.EstrusResult_Kind,
+	isShow, isPerk pasturePb.IsShow_Kind,
+	lastEstrusDate, activeDate string,
+	dayHigh, maxHigh int32,
 	cow *Cow,
-	estrusDate, remarks string,
 ) *EventEstrus {
 	return &EventEstrus{
+		PastureId:        pastureId,
+		NeckRingNumber:   cow.NeckRingNumber,
 		CowId:            cow.Id,
 		Lact:             cow.Lact,
 		ExposeEstrusType: exposeEstrusType,
-		EstrusDate:       estrusDate,
+		LastEstrusDate:   lastEstrusDate,
+		ActiveDate:       activeDate,
 		Level:            level,
 		IsShow:           isShow,
-		Remarks:          remarks,
+		Result:           result,
+		DayHigh:          dayHigh,
+		MaxHigh:          maxHigh,
+		IsPeak:           isPerk,
 	}
 }
 

+ 47 - 7
model/neck_active_habit.go

@@ -36,12 +36,12 @@ type NeckActiveHabit struct {
 	FilterRumina            int32                 `json:"filterRumina"`
 	FilterChew              int32                 `json:"filterChew"`
 	WeekHigh                int32                 `json:"weekHigh"`
-	AvgHighHabit            int32                 `json:"avgHighHabit"`
-	AvgRuminaHabit          int32                 `json:"avgRuminaHabit"`
-	AvgIntakeHabit          int32                 `json:"avgIntakeHabit"`
-	AvgChewHabit            int32                 `json:"avgChewHabit"`
-	AvgInactiveHabit        int32                 `json:"avgInactiveHabit"`
-	AvgOtherHabit           int32                 `json:"avgOtherHabit"`
+	WeekHighHabit           int32                 `json:"weekHighHabit"`
+	WeekRuminaHabit         int32                 `json:"weekRuminaHabit"`
+	WeekIntakeHabit         int32                 `json:"weekIntakeHabit"`
+	WeekChewHabit           int32                 `json:"weekChewHabit"`
+	WeekInactiveHabit       int32                 `json:"weekInactiveHabit"`
+	WeekOtherHabit          int32                 `json:"weekOtherHabit"`
 	ChangeHigh              int32                 `json:"changeHigh"`
 	ChangeRumina            int32                 `json:"changeRumina"`
 	ChangeChew              int32                 `json:"changeChew"`
@@ -85,8 +85,8 @@ func NewNeckActiveHabit(data *NeckRingOriginalMerge) *NeckActiveHabit {
 		Intake:          data.Intake,
 		Other:           data.Other,
 		Rumina:          data.Rumina,
+		IsShow:          data.IsShow,
 		WeekHigh:        DefaultWeeklyActive,
-		IsShow:          pasturePb.IsShow_No,
 		IsMaxTime:       pasturePb.IsShow_No,
 		ChangeFilter:    InitChangeFilter,
 		FilterCorrect:   InitChangeFilter,
@@ -98,6 +98,46 @@ func NewNeckActiveHabit(data *NeckRingOriginalMerge) *NeckActiveHabit {
 	}
 }
 
+type NeckActiveHabitSlice []*NeckActiveHabit
+
+func (n NeckActiveHabitSlice) ToPB(curveName string) *pasturePb.CowBehaviorCurveData {
+	res := &pasturePb.CowBehaviorCurveData{
+		OriginalDataList: make([]int32, 0),
+		ChangeDataList:   make([]int32, 0),
+		SumDataList:      make([]int32, 0),
+		DataTimeList:     make([]string, 0),
+		EstrusList:       make([]string, 0),
+		EventList:        make(map[string]string),
+	}
+
+	for _, v := range n {
+		res.DataTimeList = append(res.DataTimeList, fmt.Sprintf("%s %02d", v.HeatDate, v.Frameid*2+1))
+		switch curveName {
+		case "active": // 活动量
+			res.OriginalDataList = append(res.OriginalDataList, v.High)
+			res.ChangeDataList = append(res.ChangeDataList, v.ChangeFilter)
+		case "rumina": // 反刍
+			res.OriginalDataList = append(res.OriginalDataList, v.Rumina)
+			res.ChangeDataList = append(res.ChangeDataList, v.ChangeRumina)
+			res.SumDataList = append(res.SumDataList, v.SumRumina)
+		case "intake": // 采食
+			res.OriginalDataList = append(res.OriginalDataList, v.Intake)
+			res.SumDataList = append(res.SumDataList, v.SumIntake)
+		case "inactive": // 休息
+			res.OriginalDataList = append(res.OriginalDataList, v.Inactive)
+			res.SumDataList = append(res.SumDataList, v.SumInactive)
+		case "chew": // 咀嚼
+			res.OriginalDataList = append(res.OriginalDataList, 0)
+			res.ChangeDataList = append(res.ChangeDataList, v.ChangeChew)
+			res.SumDataList = append(res.SumDataList, 0)
+		case "immobility": // 静止
+			res.OriginalDataList = append(res.OriginalDataList, 0)
+			res.SumDataList = append(res.SumDataList, 0)
+		}
+	}
+	return res
+}
+
 func (n *NeckActiveHabit) SumAvg() {
 	n.Rumina = n.Rumina / n.RecordCount * n.RecordCount
 	n.Inactive = n.Inactive / n.RecordCount * n.RecordCount

+ 2 - 0
model/neck_ring_original.go

@@ -94,6 +94,7 @@ type NeckRingOriginalMerge struct {
 	RecordCount     int32
 	PastureId       int64
 	FirmwareVersion int32
+	IsShow          pasturePb.IsShow_Kind
 }
 
 func (n *NeckRingOriginalMerge) IsMageData(data *NeckRingOriginal, xframeId int32) {
@@ -120,6 +121,7 @@ func (n *NeckRingOriginalMerge) IsMageData(data *NeckRingOriginal, xframeId int3
 	n.XframeId = xframeId
 	n.PastureId = data.PastureId
 	n.FirmwareVersion = data.FirmwareVersion
+	n.IsShow = pasturePb.IsShow_Ok
 }
 
 func (n *NeckRingOriginalMerge) SumAvg() {

+ 71 - 0
module/backend/cow.go

@@ -3,8 +3,10 @@ package backend
 import (
 	"context"
 	"kpt-pasture/model"
+	"kpt-pasture/util"
 	"net/http"
 	"strings"
+	"time"
 
 	"gitee.com/xuyiping_admin/pkg/logger/zaplog"
 	"go.uber.org/zap"
@@ -143,3 +145,72 @@ func (s *StoreEntry) EventList(ctx context.Context, req *pasturePb.SearchCowEven
 		},
 	}, nil
 }
+
+func (s *StoreEntry) BehaviorCurve(ctx context.Context, req *pasturePb.CowBehaviorCurveRequest) (*pasturePb.CowBehaviorCurveResponse, error) {
+	currentUser, err := s.GetCurrentSystemUser(ctx)
+	if err != nil {
+		return nil, xerr.Custom("当前用户信息错误,请退出重新登录")
+	}
+	cowInfo, err := s.GetCowInfoByCowId(ctx, currentUser.PastureId, int64(req.CowId))
+	if err != nil {
+		return nil, xerr.Customf("错误的牛只信息: %d", req.CowId)
+	}
+	nowTime := time.Now()
+
+	nowDayZero := util.TimeParseLocalUnix(nowTime.Format(model.LayoutDate2))
+	endDataTime := nowTime.Format(model.LayoutDate2)
+	startDataTime := nowTime.AddDate(0, 0, -30).Format(model.LayoutDate2)
+
+	// 行为曲线数据
+	neckActiveHabitList := make([]*model.NeckActiveHabit, 0)
+	if err = s.DB.Table(new(model.NeckActiveHabit).TableName()).
+		Where("neck_ring_number = ?", cowInfo.NeckRingNumber).
+		Where("pasture_id = ?", currentUser.PastureId).
+		Where("heat_date >= ?", startDataTime).
+		Where("heat_date <= ?", endDataTime).
+		Order("heat_date, frameid").
+		Find(&neckActiveHabitList).Error; err != nil {
+		return nil, xerr.WithStack(err)
+	}
+
+	// 牛只事件列表
+	eventLogList := make([]*model.EventCowLog, 0)
+	eventLog := &model.EventCowLog{CowId: cowInfo.Id}
+	if err = s.DB.Table(eventLog.TableName()).
+		Where("cow_id = ?", cowInfo.Id).
+		Where("pasture_id = ?", currentUser.PastureId).
+		Where("event_at >= ?", nowDayZero-(30*86400)).
+		Where("event_at <= ?", nowDayZero+86400).
+		Order("event_at").
+		Find(&eventLogList).Error; err != nil {
+		return nil, xerr.WithStack(err)
+	}
+
+	data := model.NeckActiveHabitSlice(neckActiveHabitList).ToPB(req.CurveName)
+	for _, v := range eventLogList {
+		eventAt := time.Unix(v.EventAt, 0)
+		data.EventList[eventAt.Format(model.LayoutDate2)] = v.EventTypeName
+	}
+
+	// 发情数据
+	estrusList := make([]*model.EventEstrus, 0)
+	if err = s.DB.Table(new(model.EventEstrus).TableName()).
+		Where("cow_id = ?", cowInfo.Id).
+		Where("pasture_id = ?", currentUser.PastureId).
+		Where("estrus_start_date >= ?", nowTime.Format(model.LayoutTime)).
+		Where("estrus_start_date <= ?", nowTime.AddDate(0, 0, -30).Format(model.LayoutTime)).
+		Order("estrus_start_date").
+		Find(&estrusList).Error; err != nil {
+		return nil, xerr.WithStack(err)
+	}
+
+	for _, v := range estrusList {
+		data.EstrusList = append(data.EstrusList, v.EstrusStartDate)
+	}
+
+	return &pasturePb.CowBehaviorCurveResponse{
+		Code:    http.StatusOK,
+		Message: "ok",
+		Data:    data,
+	}, nil
+}

+ 1 - 0
module/backend/interface.go

@@ -211,6 +211,7 @@ type EventService interface {
 type CowService interface {
 	List(ctx context.Context, req *pasturePb.SearchEventRequest, pagination *pasturePb.PaginationModel) (*pasturePb.SearchCowListResponse, error)
 	EventList(ctx context.Context, req *pasturePb.SearchCowEventListRequest, pagination *pasturePb.PaginationModel) (*pasturePb.CowEventListResponse, error)
+	BehaviorCurve(ctx context.Context, req *pasturePb.CowBehaviorCurveRequest) (*pasturePb.CowBehaviorCurveResponse, error)
 }
 
 //go:generate mockgen -destination mock/GoodsService.go -package kptservicemock kpt-pasture/module/backend GoodsService

+ 12 - 13
module/crontab/model.go

@@ -14,7 +14,7 @@ type XToday struct {
 	ActiveHigh     int64
 }
 
-type AvgHabit struct {
+type WeekHabit struct {
 	NeckRingNumber   string
 	AvgHighHabit     int32
 	AvgRuminaHabit   int32
@@ -35,18 +35,6 @@ type SumHabit struct {
 	SumMinChew     int32
 }
 
-type ChangeFilterData struct {
-	Id             int64
-	NeckRingNumber string
-	HighChange     int32
-	ChangeFilter   int32
-	RuminaFilter   int32
-	ChangeRumina   int32
-	ChewFilter     int32
-	ChangeChew     int32
-	XlcDisCount    float64
-}
-
 type ActivityVolume struct {
 	NeckRingNumber string
 	AvgFilter      int32
@@ -68,3 +56,14 @@ type CowEstrus struct {
 	ActiveDate string
 	HadJust    int32
 }
+
+type EstrusStartData struct {
+	CowId           int64
+	EstrusStartDate string
+}
+
+type EstrusIsPeakData struct {
+	CowId           int64
+	EstrusStartDate string
+	ActiveDate      string
+}

+ 172 - 147
module/crontab/neck_ring_estrus.go

@@ -32,7 +32,7 @@ func (e *Entry) PastureUpdateCowEstrus() (err error) {
 		if err = e.EntryCowEstrus(pasture.Id); err != nil {
 			zaplog.Error("EntryCrontab", zap.Any("PastureUpdateCowEstrus", err), zap.Any("pasture", pasture))
 		}
-		zaplog.Error("PastureUpdateCowEstrus-success", zap.Any("pasture", pasture.Id))
+		zaplog.Info("PastureUpdateCowEstrus-success", zap.Any("pasture", pasture.Id))
 	}
 	return nil
 }
@@ -41,189 +41,214 @@ func (e *Entry) EntryCowEstrus(pastureId int64) (err error) {
 	activeLowValue := e.GetSystemConfigure(pastureId, model.ActiveLow).Value
 	activeMiddleValue := e.GetSystemConfigure(pastureId, model.ActiveMiddle).Value
 	activeHighValue := e.GetSystemConfigure(pastureId, model.ActiveHigh).Value
-	lastMaxEstrusId := e.GetSystemConfigure(pastureId, model.MaxEstrus).Value
-
-	currentMaxHabit := &model.NeckActiveHabit{}
-	if err = e.DB.Model(new(model.NeckActiveHabit)).
-		Where("id > ?", lastMaxEstrusId).
-		Order("id desc").
-		First(currentMaxHabit).Error; err != nil {
-		return xerr.WithStack(err)
-	}
-	// 当前Id<=上次执行的id,则不执行
-	if currentMaxHabit.Id <= lastMaxEstrusId {
-		return nil
+	xToday := &XToday{
+		ActiveLow:    activeLowValue,
+		ActiveMiddle: activeMiddleValue,
+		ActiveHigh:   activeHighValue,
 	}
-
-	defer func() {
-		if err == nil {
-			e.DB.Model(new(model.SystemConfigure)).
-				Where("name = ?", model.MaxEstrus).
-				Update("value", currentMaxHabit.Id)
-		}
-	}()
-
-	xToday := &XToday{}
-	if err = e.DB.Model(new(model.NeckActiveHabit)).
-		Select(`MIN(heat_date) as x_beg_date, MAX(heat_date) as x_end_date`).
-		Where("id BETWEEN ? AND ?", lastMaxEstrusId, currentMaxHabit.Id).
-		First(xToday).Error; err != nil {
-		return xerr.WithStack(err)
-	}
-
-	xToday.LastMaxHabitId = lastMaxEstrusId
-	xToday.CurrMaxHabitId = currentMaxHabit.Id
-	xToday.ActiveLow = activeLowValue
-	xToday.ActiveMiddle = activeMiddleValue
-	xToday.ActiveHigh = activeHighValue
-	if err = e.CowEstrusWarning(xToday); err != nil {
+	if err = e.CowEstrusWarning(pastureId, xToday); err != nil {
 		zaplog.Error("EntryCowEstrus", zap.Any("CowEstrusWarning", err), zap.Any("xToday", xToday))
 	}
 	return nil
 }
 
 // CowEstrusWarning 发情预警
-func (e *Entry) CowEstrusWarning(xToday *XToday) (err error) {
-	startDate, _ := time.Parse(model.LayoutDate2, xToday.XBegDate)
-	endDate, _ := time.Parse(model.LayoutDate2, xToday.XEndDate)
-
-	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("neck_ring_number").
-			Find(&neckActiveHabitList).
-			Error; err != nil {
-			zaplog.Error("CowEstrusWarning", zap.Any("FindNeckActiveHabit", err), zap.Any("xToday", xToday))
+func (e *Entry) CowEstrusWarning(pastureId int64, xToday *XToday) (err error) {
+	nowTime := time.Now()
+	neckActiveHabitList := make([]*model.NeckActiveHabit, 0) // todo 需要考虑到数据量太大的情况
+	if err = e.DB.Model(new(model.NeckActiveHabit)).
+		Where("heat_date = ?", nowTime.Format(model.LayoutDate2)).
+		Where("pasture_id = ?", pastureId).
+		Where("filter_high > 0 AND change_filter > ?", model.DefaultChangeFilter).
+		Where("cow_id > ?", 0).
+		Where(e.DB.Where("calving_age > ?", MinCalvingAge).Or("lact = ?", MinLact)). // 排除产后20天内的发情牛
+		Find(&neckActiveHabitList).Error; err != nil {
+		return xerr.WithStack(err)
+	}
+
+	neckActiveHabitMap := make(map[int64][]*model.NeckActiveHabit)
+	for _, habit := range neckActiveHabitList {
+		cft := calculateCFT(habit)
+		if cft < float32(xToday.ActiveLow-XAdjust21) {
 			continue
 		}
+		if _, ok := neckActiveHabitMap[habit.CowId]; !ok {
+			neckActiveHabitMap[habit.CowId] = make([]*model.NeckActiveHabit, 0)
+		}
+		neckActiveHabitMap[habit.CowId] = append(neckActiveHabitMap[habit.CowId], habit)
+	}
 
-		eventEstrusList := make([]*model.EventEstrus, 0)
-		for _, v := range neckActiveHabitList {
-			if ok := e.IsAdJustLow(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 int32(cft) <= eventEstrus.DaylongHigh {
-				continue
-			}
+	eventEstrusList := make([]*model.EventEstrus, 0)
+	for cowId, cowHabitList := range neckActiveHabitMap {
+		// 最近3天最大发情记录,小于该变化趋势的不再插入
+		before3Data := e.GetBeforeThreeDaysCowEstrus(cowId, nowTime.AddDate(0, 0, -2).Format(model.LayoutTime))
+		// 判断最近50天内是否存在发情记录(发情等级>=2),如果18~25天@xadjust21,如果36~50天@xadjust42
+		cowEstrus := e.GetTwoEstrus(pastureId, cowId, nowTime.AddDate(0, 0, -100).Format(model.LayoutTime), nowTime.AddDate(0, 0, -2).Format(model.LayoutTime))
+		activeDateTime, _ := time.Parse(model.LayoutTime, cowEstrus.ActiveDate)
+		if activeDateTime.Unix() >= nowTime.AddDate(0, 0, -25).Unix() && activeDateTime.Unix() <= nowTime.AddDate(0, 0, -18).Unix() {
+			cowEstrus.HadJust = XAdjust21
+		}
+		if activeDateTime.Unix() >= nowTime.AddDate(0, 0, -50).Unix() && activeDateTime.Unix() <= nowTime.AddDate(0, 0, -36).Unix() {
+			cowEstrus.HadJust = XAdjust42
+		}
 
-			// 判断最近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))
-			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
+		maxCft := float32(0)
+		maxHigh := int32(0)
+		for _, habit := range cowHabitList {
+			cft := calculateCFT(habit)
+			if cft > maxCft {
+				maxCft = cft
 			}
-			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
+			if habit.FilterHigh > maxHigh {
+				maxHigh = habit.FilterHigh
 			}
+		}
 
-			cowInfo := e.FindCowInfoByNeckRingNumber(v.NeckRingNumber)
-			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
-			}
+		activeDate := ""
+		if len(cowHabitList) > 0 {
+			sortHabits := sortHabitsByChangeFilter(cowHabitList)
+			activeDate = sortHabits[0].ActiveTime
+		}
 
-			result := pasturePb.EstrusResult_Invalid
-			if eventEstrus.Result == pasturePb.EstrusResult_Fail && eventEstrus.DaylongHigh > int32(cft)+cowEstrus.HadJust {
-				result = pasturePb.EstrusResult_Fail
-			}
-			// todo 待定
-			if result == pasturePb.EstrusResult_Invalid {
-				result = pasturePb.EstrusResult_Correct
-			}
+		b48 := float64(0)
+		t1, _ := time.Parse(model.LayoutTime, activeDate)
+		t3, err := time.Parse(model.LayoutTime, before3Data.ActiveDate)
+		if err == nil {
+			b48 = t3.Sub(t1).Hours()
+		}
 
-			// 更新 已孕 + 不显示
+		if (int32(maxCft) > before3Data.DayHigh || b48 > 48) && int32(maxCft)+cowEstrus.HadJust > int32(xToday.ActiveLow) {
+			level := calculateLevel(maxCft, cowEstrus, xToday)
+			cowInfo := e.FindCowInfoByNeckRingNumber(cowHabitList[0].NeckRingNumber)
 			isShow := pasturePb.IsShow_Ok
 			if cowInfo.IsPregnant == pasturePb.IsShow_Ok && level == pasturePb.EstrusLevel_Low {
 				isShow = pasturePb.IsShow_No
 			}
-
-			eventEstrusList = append(eventEstrusList, &model.EventEstrus{
-				CowId:            cowInfo.Id,
-				Lact:             cowInfo.Lact,
-				ExposeEstrusType: pasturePb.ExposeEstrusType_Natural_Estrus,
-				FilterHigh:       v.FilterHigh,
-				EstrusDate:       v.ActiveTime,
-				ActiveDate:       v.ActiveTime,
-				LastEstrusDate:   cowEstrus.ActiveDate,
-				Level:            level,
-				IsPeak:           pasturePb.IsShow_No,
-				DaylongHigh:      int32(cft) + cowEstrus.HadJust,
-				Result:           result,
-				IsShow:           isShow,
-			})
+			dayHigh := int32(maxCft) + cowEstrus.HadJust
+			lastEstrusDate := cowEstrus.ActiveDate
+			result := getResult(before3Data, maxCft, cowEstrus)
+			isPeak := pasturePb.IsShow_Ok
+			zaplog.Info("CowEstrusWarning",
+				zap.Any("level", level),
+				zap.Any("result", result),
+				zap.Any("isShow", isShow),
+				zap.Any("isPeak", isPeak),
+				zap.Any("lastEstrusDate", lastEstrusDate),
+				zap.Any("activeDate", activeDate),
+				zap.Any("dayHigh", dayHigh),
+				zap.Any("cft", maxCft),
+				zap.Any("before3Data", before3Data),
+				zap.Any("cowEstrus", cowEstrus),
+				zap.Any("cowInfo", cowInfo),
+				zap.Any("cowHabitList", cowHabitList),
+			)
+			newEstrus := model.NewEventEstrus(
+				pastureId, pasturePb.ExposeEstrusType_Neck_Ring, level, result, isShow,
+				isPeak, lastEstrusDate, activeDate, dayHigh, maxHigh, cowInfo,
+			)
+			eventEstrusList = append(eventEstrusList, newEstrus)
+		}
+	}
+	if len(eventEstrusList) > 0 {
+		if err = e.DB.Model(new(model.EventEstrus)).Create(eventEstrusList).Error; err != nil {
+			zaplog.Error("CowEstrusWarningNew", zap.Any("eventEstrusList", eventEstrusList), zap.Any("err", err))
 		}
-		if len(eventEstrusList) > 0 {
-			if err = e.DB.Create(eventEstrusList).Error; err != nil {
-				zaplog.Error("CowEstrusWarning", zap.Any("eventEstrusList", eventEstrusList), zap.Any("err", err))
+	}
+
+	if err = e.UpdateEstrusStartDate(pastureId, nowTime); err != nil {
+		zaplog.Error("UpdateEstrusStartDate", zap.Any("err", err))
+	}
+	return err
+}
+
+// UpdateEstrusStartDate 更新发情开始时间数据
+func (e *Entry) UpdateEstrusStartDate(pastureId int64, xToday time.Time) (err error) {
+	beforeEventEstrus := make([]*EstrusStartData, 0)
+	if err = e.DB.Model(new(model.EventEstrus)).
+		Select("cow_id,MIN(estrus_start_date) as estrus_start_date").
+		Where("active_date BETWEEN  ? AND  ?", xToday.Add(-24*time.Hour).Format(model.LayoutTime), xToday.Format(model.LayoutTime)).
+		Where("estrus_start_date != ?", "").
+		Where("expose_estrus_type = ?", pasturePb.ExposeEstrusType_Neck_Ring).
+		Where("pasture_id = ?", pastureId).
+		Group("cow_id").Find(&beforeEventEstrus).Error; err != nil {
+		return xerr.WithStack(err)
+	}
+	if len(beforeEventEstrus) > 0 {
+		for _, v := range beforeEventEstrus {
+			if err = e.DB.Model(new(model.EventEstrus)).
+				Where("cow_id = ?", v.CowId).
+				Where("active_date >= ? AND <= ?", xToday.Add(-1*time.Hour).Format(model.LayoutTime), xToday.Add(24*time.Hour).Format(model.LayoutTime)).
+				Update("estrus_start_date", v.EstrusStartDate).Error; err != nil {
+				zaplog.Error("UpdateEstrusStartDate", zap.Any("err", err))
 			}
 		}
-		startDate = startDate.AddDate(0, 0, 1)
 	}
 	return nil
 }
 
-// UpdateNewEstrus 更新新发情数据
-func (e *Entry) UpdateNewEstrus(xToday *XToday) (err error) {
-
+// UpdateIsPeak 更新IsPeak是否是高峰字段 1 是 0 否
+func (e *Entry) UpdateIsPeak(pastureId int64, xToday time.Time) (err error) {
 	return nil
 }
 
-// IsAdJustLow 是否低于最低活动量
-func (e *Entry) IsAdJustLow(xToday *XToday, habit *model.NeckActiveHabit) bool {
-	ruminaAdJust := float64(0)
+// calculateCFT 计算cft值
+func calculateCFT(habit *model.NeckActiveHabit) (cft float32) {
+	cft = float32(habit.ChangeFilter) - float32(habit.ChangeAdjust) + 3
+	if habit.ChangeAdjust < 10 {
+		cft = float32(habit.ChangeFilter)
+	}
+	cft = cft * float32(habit.FilterCorrect) / 100
+	ruminaAdjust := float32(0)
 	switch {
 	case habit.RuminaFilter > MaxRuminaAdJust:
-		ruminaAdJust = 5
+		ruminaAdjust = float32(5)
 	case habit.RuminaFilter > 0:
-		ruminaAdJust = float64(habit.RuminaFilter) * 0.25
+		ruminaAdjust = float32(habit.RuminaFilter) * 0.25
 	case habit.RuminaFilter < -MaxRuminaAdJust:
-		ruminaAdJust = -MaxRuminaAdJust * RumtoHeat
+		ruminaAdjust = -MaxRuminaAdJust * RumtoHeat
 	default:
-		ruminaAdJust = float64(habit.RuminaFilter) * RumtoHeat
+		ruminaAdjust = float32(habit.RuminaFilter) * RumtoHeat
+	}
+	cft -= ruminaAdjust
+	return cft
+}
+
+// calculateLevel 计算发情等级
+func calculateLevel(cft float32, cowEstrus *CowEstrus, xToday *XToday) pasturePb.EstrusLevel_Kind {
+	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
 	}
-	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
+	return level
+}
+
+// getResult 根据b3数据计算结果 0 1 2 3 -1 -2
+func getResult(b3 *model.EventEstrus, cft float32, cowEstrus *CowEstrus) pasturePb.EstrusResult_Kind {
+	result := pasturePb.EstrusResult_Invalid
+	if b3.Result == pasturePb.EstrusResult_Fail && b3.DayHigh > int32(cft)+cowEstrus.HadJust {
+		result = pasturePb.EstrusResult_Fail
 	}
 
-	if isContinue < int32(activeLow) {
-		return true
+	if b3.Result == pasturePb.EstrusResult_Overdue {
+		result = pasturePb.EstrusResult_Correct
+	}
+	return result
+}
+
+// sortHabitsByChangeFilter 根据change_filter排序
+func sortHabitsByChangeFilter(habits []*model.NeckActiveHabit) []*model.NeckActiveHabit {
+	sorted := make([]*model.NeckActiveHabit, len(habits))
+	copy(sorted, habits)
+	for i := range sorted {
+		for j := i + 1; j < len(sorted); j++ {
+			if sorted[i].ChangeFilter < sorted[j].ChangeFilter {
+				sorted[i], sorted[j] = sorted[j], sorted[i]
+			}
+		}
 	}
-	return false
+	return sorted
 }

+ 158 - 0
module/crontab/neck_ring_estrus_test.go

@@ -0,0 +1,158 @@
+package crontab
+
+import (
+	"database/sql"
+	"fmt"
+	"math"
+
+	"gorm.io/driver/mysql"
+	"gorm.io/gorm"
+)
+
+type HActiveHabit struct {
+	ID            uint
+	IntPastureID  uint
+	IntCowID      uint
+	VarCowCode    string
+	IntCurFetal   uint
+	DIM           uint
+	Createtime    sql.NullString
+	Changeadjust  uint
+	Changefilter  uint
+	Filtercorrect float64
+	Ruminafilter  int
+	Filterhigh    uint
+	Heatdate      sql.NullString
+}
+
+func main() {
+	dsn := "username:password@tcp(127.0.0.1:3306)/dbname?charset=utf8mb4&parseTime=True&loc=Local"
+	db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{})
+	if err != nil {
+		panic("failed to connect database")
+	}
+
+	// 假设这些变量是从外部传入的
+	xbegactid, xendactid := uint(1), uint(100)
+	xPastuId := uint(123)
+	xthatdate := "2023-10-01"
+	xactlevellow := float64(10)
+	xadjust21 := float64(2)
+	maxruminaadjust := int(10)
+	rumtoheat := float64(0.5) // 假设值
+
+	var habits []HActiveHabit
+	db.Where("id BETWEEN ? AND ? AND int_pasture_id = ? AND heatdate = ?", xbegactid, xendactid, xPastuId, xthatdate).
+		Find(&habits)
+
+	var results []struct {
+		IntPastureID uint
+		IntCowID     uint
+		VarCowCode   string
+		IntCurFetal  uint
+		DIM          uint
+		Createtime   string
+		CFT          float64
+		High         uint
+	}
+
+	cowMap := make(map[uint][]HActiveHabit)
+	for _, habit := range habits {
+		cowMap[habit.IntCowID] = append(cowMap[habit.IntCowID], habit)
+	}
+
+	for cowID, habits := range cowMap {
+		var createtime string
+		var cft float64
+		var high uint
+		if len(habits) > 0 {
+			sortedHabits := sortHabitsByChangefilterDesc(habits)
+			createtime = sortedHabits[0].Createtime.String
+
+			maxCft := float64(-math.MaxFloat64)
+			for _, habit := range sortedHabits {
+				adjustedValue := calculateAdjustedValue(habit, maxruminaadjust, rumtoheat)
+				if adjustedValue > maxCft {
+					maxCft = adjustedValue
+					high = habit.Filterhigh
+				}
+			}
+			cft = round(maxCft)
+		}
+
+		if cft >= (xactlevellow - xadjust21) {
+			anyDimValid := false
+			for _, habit := range habits {
+				if habit.DIM >= 20 || habit.IntCurFetal == 0 {
+					anyDimValid = true
+					break
+				}
+			}
+			if anyDimValid {
+				results = append(results, struct {
+					IntPastureID uint
+					IntCowID     uint
+					VarCowCode   string
+					IntCurFetal  uint
+					DIM          uint
+					Createtime   string
+					CFT          float64
+					High         uint
+				}{
+					IntPastureID: habits[0].IntPastureID,
+					IntCowID:     cowID,
+					VarCowCode:   habits[0].VarCowCode,
+					IntCurFetal:  habits[0].IntCurFetal,
+					DIM:          habits[0].DIM,
+					Createtime:   createtime,
+					CFT:          cft,
+					High:         high,
+				})
+			}
+		}
+	}
+
+	for _, result := range results {
+		fmt.Printf("IntPastureID: %d, IntCowID: %d, VarCowCode: %s, IntCurFetal: %d, DIM: %d, Createtime: %s, CFT: %.2f, High: %d\n",
+			result.IntPastureID, result.IntCowID, result.VarCowCode, result.IntCurFetal, result.DIM, result.Createtime, result.CFT, result.High)
+	}
+}
+
+func sortHabitsByChangefilterDesc(habits []HActiveHabit) []HActiveHabit {
+	sorted := make([]HActiveHabit, len(habits))
+	copy(sorted, habits)
+	for i := range sorted {
+		for j := i + 1; j < len(sorted); j++ {
+			if sorted[i].Changefilter < sorted[j].Changefilter {
+				sorted[i], sorted[j] = sorted[j], sorted[i]
+			}
+		}
+	}
+	return sorted
+}
+
+func calculateAdjustedValue(habit HActiveHabit, maxruminaadjust int, rumtoheat float64) float64 {
+	baseValue := float64(0)
+	if habit.Changeadjust >= 10 {
+		baseValue = float64(habit.Changefilter-habit.Changeadjust+3) * habit.Filtercorrect / 100
+	} else {
+		baseValue = float64(habit.Changefilter) * habit.Filtercorrect / 100
+	}
+
+	ruminaAdjustment := float64(0)
+	if habit.Ruminafilter > maxruminaadjust {
+		ruminaAdjustment = 5
+	} else if habit.Ruminafilter > 0 {
+		ruminaAdjustment = float64(habit.Ruminafilter) * 0.25
+	} else if habit.Ruminafilter < -maxruminaadjust {
+		ruminaAdjustment = float64(-maxruminaadjust) * rumtoheat
+	} else {
+		ruminaAdjustment = float64(habit.Ruminafilter) * rumtoheat
+	}
+
+	return baseValue - ruminaAdjustment
+}
+
+func round(value float64) float64 {
+	return math.Round(value*100) / 100
+}

+ 20 - 3
module/crontab/sql.go

@@ -61,7 +61,7 @@ func (e *Entry) GetSystemConfigure(pastureId int64, name string) *model.SystemCo
 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(daylong_high) as daylong_high").
+		Select("MAX(max_high) as max_high,cow_id,MAX(day_high) as day_high").
 		Select("MAX(IF(result=1,3,result)) AS result").
 		Where("cow_id = ?", cowId).
 		Where("active_date >= ?", activeTime).
@@ -72,11 +72,12 @@ func (e *Entry) GetBeforeThreeDaysCowEstrus(cowId int64, activeTime string) *mod
 }
 
 // GetTwoEstrus 判断最近50天内是否存在发情记录(发情等级>=2),如果18~25天@xadjust21,如果36~50天@xadjust42
-func (e *Entry) GetTwoEstrus(cowId int64, startActiveTime, endActiveTime string) *CowEstrus {
+func (e *Entry) GetTwoEstrus(pastureId, 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("pasture_id = ?", pastureId).
 		Where("active_date BETWEEN ? AND ?", startActiveTime, endActiveTime).
 		Where("level >= ?", pasturePb.EstrusLevel_Middle).
 		First(newCowEstrus).Error; err != nil {
@@ -88,7 +89,23 @@ func (e *Entry) GetTwoEstrus(cowId int64, startActiveTime, endActiveTime string)
 func (e *Entry) FindCowInfoByNeckRingNumber(neckRingNumber string) *model.Cow {
 	res := &model.Cow{}
 	if err := e.DB.Model(new(model.Cow)).
-		Where("neck_ring_number = ?", neckRingNumber).First(res).Error; err != nil {
+		Where("neck_ring_number = ?", neckRingNumber).
+		Where("admission_status = ?", pasturePb.AdmissionStatus_Admission).
+		First(res).Error; err != nil {
+		return nil
+	}
+	return res
+}
+
+func (e *Entry) IsExistEventEstrus(pastureId, cowId int64) *model.EventEstrus {
+	res := &model.EventEstrus{}
+	if err := e.DB.Model(new(model.EventEstrus)).
+		Where("cow_id = ?", cowId).
+		Where("pasture_id = ?", pastureId).
+		Where("expose_estrus_type = ?", pasturePb.ExposeEstrusType_Neck_Ring).
+		Where("is_show = ?", pasturePb.IsShow_Ok).
+		Order("id DESC").
+		First(res).Error; err != nil {
 		return nil
 	}
 	return res

+ 68 - 22
module/mqtt/merge_handle.go

@@ -5,12 +5,12 @@ import (
 	"fmt"
 	"kpt-pasture/model"
 	"kpt-pasture/util"
-	"math"
-	"sort"
 	"strconv"
 	"strings"
 	"time"
 
+	pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
+
 	"github.com/jinzhu/copier"
 
 	"gitee.com/xuyiping_admin/pkg/logger/zaplog"
@@ -61,15 +61,16 @@ func (e *Entry) NeckRingOriginalMergeData() {
 	mergeDataMaxId := e.GetSystemConfigure(model.UpdateOriginalMax).Value
 
 	newTime := time.Now()
+	createdAt := newTime.Add(-1 * time.Hour)
+
 	neckRingList := make([]*model.NeckRingOriginal, 0)
 	if err = e.DB.Model(new(model.NeckRingOriginal)).
-		Where("created_at <= ?", newTime.Add(-1*time.Hour).Unix()).
-		Where("id > ?", mergeDataMaxId).
-		Order("id asc").Limit(int(limit)).
-		Find(&neckRingList).Error; err != nil {
+		Where("is_show <= ?", pasturePb.IsShow_No).
+		Where("created_at <= ?", createdAt.Unix()).
+		Order("neck_ring_number,active_date,frameid").
+		Limit(int(limit)).Find(&neckRingList).Error; err != nil {
 		return
 	}
-
 	if len(neckRingList) <= 0 {
 		return
 	}
@@ -84,6 +85,9 @@ func (e *Entry) NeckRingOriginalMergeData() {
 			e.DB.Model(new(model.SystemConfigure)).
 				Where("name = ?", model.UpdateOriginalMax).
 				Update("value", newMergeDataMaxId)
+			e.DB.Model(new(model.NeckRingOriginal)).
+				Where("id BETWEEN ? AND ?", mergeDataMaxId, newMergeDataMaxId).
+				Update("is_show", pasturePb.IsShow_Ok)
 		}
 
 		if newTime.Day()%15 == 0 && !isDelete {
@@ -109,12 +113,24 @@ func (e *Entry) NeckRingOriginalMergeData() {
 			//更新脖环牛只相关信息 新数据直接插入
 			historyNeckActiveHabit, ct := e.IsExistNeckActiveHabit(neckActiveHabit.NeckRingNumber, neckActiveHabit.HeatDate, neckActiveHabit.Frameid)
 			if ct <= 0 {
+				// 过滤牛只未绑定的脖环的数据
+				cowInfo := e.GetCowInfoByNeckRingNumber(neckActiveHabit.PastureId, neckActiveHabit.NeckRingNumber)
+				if cowInfo == nil || cowInfo.Id <= 0 {
+					continue
+				}
+				neckActiveHabit.CowId = cowInfo.Id
+				neckActiveHabit.Lact = cowInfo.Lact
+				neckActiveHabit.CalvingAge = int32(cowInfo.CalvingAge)
 				if err = tx.Create(neckActiveHabit).Error; err != nil {
 					return xerr.WithStack(err)
 				}
-				if err = tx.Create(model.NewNeckRingProcess(neckActiveHabit)).Error; err != nil {
+				newNeckRingProcess := model.NewNeckRingProcess(neckActiveHabit)
+				if err = tx.Create(newNeckRingProcess).Error; err != nil {
 					return xerr.WithStack(err)
 				}
+				if err = e.UpdateNeckRingOriginalIsShow(neckActiveHabit); err != nil {
+					zaplog.Error("UpdateNeckRingOriginalIsShow", zap.Any("err", err), zap.Any("neckActiveHabit", neckActiveHabit))
+				}
 				continue
 			}
 
@@ -220,6 +236,9 @@ func (e *Entry) MsgDataFormat2(msg []byte) *DataInsertNeckRingLog {
 		zaplog.Error("MsgDataFormat", zap.Any("err", err), zap.Any("msg", string(msg)))
 		return nil
 	}
+	if neckLogList.Type == "heartbeat" {
+		return nil
+	}
 
 	normalOriginal := make([]*model.NeckRingOriginal, 0)
 	errorOriginal := make([]*model.NeckRingError, 0)
@@ -252,33 +271,32 @@ func (e *Entry) recalculate(neckRingList []*model.NeckRingOriginal) []*model.Nec
 	originalMapData := make(map[string]*model.NeckRingOriginalMerge)
 	// 合并成2个小时的
 	for _, v := range neckRingList {
-		xframeId := int32(math.Floor(float64(v.Frameid) / 10))
+		xframeId := util.XFrameId(v.Frameid)
 		mapKey := fmt.Sprintf("%s%s%s%s%d", v.NeckRingNumber, model.JoinKey, v.ActiveDate, model.JoinKey, xframeId) // 0001/2023-12-04/0 0001/2023-12-03/4
-		if _, ok := originalMapData[mapKey]; !ok {
+		if originalMapData[mapKey] == nil {
 			originalMapData[mapKey] = new(model.NeckRingOriginalMerge)
 		}
 		originalMapData[mapKey].IsMageData(v, xframeId)
 	}
 
 	// 算平均值
-	for _, v := range originalMapData {
+	for k, v := range originalMapData {
+		// 过滤掉合并后不满6条数据
+		if v.RecordCount != 6 {
+			maxFrameId := e.CurrentMaxXFrameId(v.NeckRingNumber, v.ActiveDate)
+			currXframeId := util.XFrameId(maxFrameId)
+			if currXframeId-v.XframeId <= 1 {
+				delete(originalMapData, k)
+			}
+		}
 		v.SumAvg()
 	}
-
-	dataList := model.NeckRingOriginalMap(originalMapData).ForMatData()
-	sort.Slice(dataList, func(i, j int) bool {
-		return dataList[i].ActiveTime < dataList[j].ActiveTime
-	})
-	return dataList
+	return model.NeckRingOriginalMap(originalMapData).ForMatData()
 }
 
 func (e *Entry) againRecalculate(data *model.NeckActiveHabit) *model.NeckActiveHabit {
 	originalList := make([]*model.NeckRingOriginal, 0)
-	originalFrameId := data.Frameid
-	frameIds := make([]int32, 0)
-	for i := 1; i <= 6; i++ {
-		frameIds = append(frameIds, originalFrameId*10+int32(i))
-	}
+	frameIds := util.FrameIds(data.Frameid)
 	if err := e.DB.Model(new(model.NeckRingOriginal)).
 		Where("neck_ring_number = ?", data.NeckRingNumber).
 		Where("active_date = ?", data.HeatDate).
@@ -294,6 +312,34 @@ func (e *Entry) againRecalculate(data *model.NeckActiveHabit) *model.NeckActiveH
 	return newDataList[0]
 }
 
+// CurrentMaxXFrameId 当前最大frameid
+func (e *Entry) CurrentMaxXFrameId(neckRingNumber, activeDate string) (frameid int32) {
+	type Fm struct {
+		Frameid int32
+	}
+	maxFx := &Fm{}
+	if err := e.DB.Model(new(model.NeckRingOriginal)).
+		Select("frameid").
+		Where("neck_ring_number = ?", neckRingNumber).
+		Where("active_date = ?", activeDate).
+		Order("frameid DESC").First(maxFx).Error; err != nil {
+		zaplog.Error("CurrentMaxXFrameId", zap.Any("err", err))
+		return 0
+	}
+	return maxFx.Frameid
+}
+
+func (e *Entry) UpdateNeckRingOriginalIsShow(neckRingList *model.NeckActiveHabit) error {
+	if err := e.DB.Model(new(model.NeckRingOriginal)).
+		Where("neck_ring_number = ?", neckRingList.NeckRingNumber).
+		Where("active_date = ?", neckRingList.HeatDate).
+		Where("frameid IN (?)", util.FrameIds(neckRingList.Frameid)).
+		Update("is_show", pasturePb.IsShow_Ok).Error; err != nil {
+		return xerr.WithStack(err)
+	}
+	return nil
+}
+
 func (e *Entry) MsgDataFormat(msg []byte) []*model.NeckRingOriginal {
 	msgData := make(map[string]interface{})
 	pairs := strings.Split(util.MsgFormat(string(msg)), " ")

+ 30 - 12
module/mqtt/model.go

@@ -1,17 +1,19 @@
 package mqtt
 
 type XToday struct {
-	XBegDate       string
-	XEndDate       string
-	LastMaxHabitId int64
-	CurrMaxHabitId int64
-	XMin2Id        int64
-	XMin7Id        int64
-	ActiveLowest   int64
-	RuminaLowest   int64
-	ActiveLow      int64
-	ActiveMiddle   int64
-	ActiveHigh     int64
+	XBegDate        string
+	XEndDate        string
+	LastMaxHabitId  int64
+	CurrMaxHabitId  int64
+	XMin2Id         int64
+	XMin7Id         int64
+	ActiveLowest    int32
+	RuminaLowest    int32
+	ActiveLow       int32
+	ActiveMiddle    int32
+	ActiveHigh      int32
+	XRuminaDisc     int32
+	XChangeDiscount int32
 }
 
 type AvgHabit struct {
@@ -38,13 +40,15 @@ type SumHabit struct {
 type ChangeFilterData struct {
 	Id             int64
 	NeckRingNumber string
-	HighChange     int32
+	ChangeHigh     int32
 	ChangeFilter   int32
 	RuminaFilter   int32
 	ChangeRumina   int32
 	ChewFilter     int32
 	ChangeChew     int32
 	XlcDisCount    float64
+	HeatDate       string
+	FrameId        int32
 }
 
 type ActivityVolume struct {
@@ -53,3 +57,17 @@ type ActivityVolume struct {
 	StdFilter      int32
 	Nb             int32
 }
+
+type FirstFilterData struct {
+	NeckRingNumber string
+	FilterHigh     int32
+	FilterRumina   int32
+	FilterChew     int32
+}
+
+type SecondFilterData struct {
+	NeckRingNumber string
+	ChangeFilter   float64
+	RuminaFilter   float64
+	ChewFilter     float64
+}

+ 182 - 215
module/mqtt/neck_ring_habit.go

@@ -7,18 +7,20 @@ import (
 	"math"
 	"time"
 
+	pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
+
 	"gorm.io/gorm"
 
 	"gitee.com/xuyiping_admin/pkg/logger/zaplog"
 	"go.uber.org/zap"
 
-	pasturePb "gitee.com/xuyiping_admin/go_proto/proto/go/backend/cow"
 	"gitee.com/xuyiping_admin/pkg/xerr"
 )
 
 const (
 	MinChangeFilter = -99
 	MinRuminaFilter = -99
+	MinChewFilter   = -99
 	MinChangeHigh   = -99
 	DefaultNb       = 30
 	DefaultScore    = 100
@@ -45,88 +47,29 @@ func (e *Entry) EntryUpdateActiveHabit(pastureId int64) (err error) {
 	}
 
 	lastMaxHabitId := lastMaxHabitData.Value
-	currentMaxHabit := &model.NeckActiveHabit{}
-	if err = e.DB.Model(currentMaxHabit).
-		Where("id > ?", lastMaxHabitId).
-		Order("id desc").First(currentMaxHabit).Error; err != nil {
-		return xerr.WithStack(err)
-	}
-	// 本次执行<=上次执行的id,则不执行
-	if currentMaxHabit.Id < lastMaxHabitId {
-		return nil
-	}
-
-	// 统一更新is_max_time为0
-	if err = e.DB.Model(new(model.NeckActiveHabit)).
-		Where("is_max_time = ?", pasturePb.IsShow_Ok).
-		Where("heat_date > ?", time.Now().AddDate(0, 0, -10).Format(model.LayoutDate2)).
-		Update("is_max_time", pasturePb.IsShow_No).Error; err != nil {
-		return xerr.WithStack(err)
-	}
-
-	// 获取这段执行数据内最大日期和最小日期
-	xToday := &XToday{}
+	currMaxHabit := &model.NeckActiveHabit{}
 	if err = e.DB.Model(new(model.NeckActiveHabit)).
-		Select(`MIN(heat_date) as x_beg_date, MAX(heat_date) as x_end_date`).
-		Where("id BETWEEN ? AND ?", lastMaxHabitId, currentMaxHabit.Id).
-		First(xToday).Error; err != nil {
-		return xerr.WithStack(err)
-	}
-
-	xToday.LastMaxHabitId = lastMaxHabitId     // 上次执行的id
-	xToday.CurrMaxHabitId = currentMaxHabit.Id // 本次执行的id
-
-	minHeatDateParse, err := time.Parse(model.LayoutDate2, xToday.XBegDate) // 开始日期
-	if err != nil {
-		return xerr.WithStack(err)
-	}
-	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 {
+		Where("id > ?", lastMaxHabitId).
+		Where("pasture_id = ?", pastureId).
+		Where(e.DB.Where("record_count = ?", model.AvgHours).Or("is_show = ?", pasturePb.IsShow_Ok)).
+		Order("id desc").First(currMaxHabit).Error; err != nil {
 		return xerr.WithStack(err)
 	}
 
-	xMin7Id, err := e.GetMinIdByHeatDate(xBefore7Day, xToday.LastMaxHabitId)
-	if err != nil {
-		return xerr.WithStack(err)
+	if currMaxHabit.Id <= 0 || currMaxHabit.Id <= lastMaxHabitId {
+		return nil
 	}
-
 	defer func() {
 		// 更新最后一次执行的id值
 		if err == nil {
 			e.DB.Model(new(model.SystemConfigure)).
 				Where("name = ?", model.MaxHabit).
 				Where("pasture_id = ?", pastureId).
-				Update("value", currentMaxHabit.Id)
+				Update("value", currMaxHabit.Id)
 		}
 	}()
-	xToday.XMin2Id = xMin2Id
-	xToday.XMin7Id = xMin7Id
-
-	//  id到上一次执行结果并且heat_date > 7天之前的最大牛只id置为is_max_time=1
-	maxHabitIdArray := make([]*model.MaxHabitIdModel, 0)
-	if err = e.DB.Model(new(model.NeckActiveHabit)).
-		Select("Max(id) as id").
-		Where("change_filter > ?", model.DefaultChangeFilter).
-		Where("heat_date > ?", xBefore7Day).
-		Group("neck_ring_number").
-		Find(&maxHabitIdArray).Error; err != nil {
-		return xerr.WithStack(err)
-	}
-
-	if len(maxHabitIdArray) > 0 {
-		maxHabitIds := make([]int64, 0)
-		for _, v := range maxHabitIdArray {
-			maxHabitIds = append(maxHabitIds, v.Id)
-		}
-		if err = e.DB.Model(new(model.NeckActiveHabit).TableName()).
-			Where("id IN (?)", maxHabitIds).
-			Update("is_max_time", pasturePb.IsShow_Ok).Error; err != nil {
-			return xerr.WithStack(err)
-		}
-	}
-
+	// 获取这段执行数据内最大日期和最小日期
+	xToday := &XToday{}
 	activeLowest, err := e.GetSystemConfigure2(pastureId, model.ActiveLowest)
 	if err != nil {
 		return xerr.WithStack(err)
@@ -135,8 +78,21 @@ func (e *Entry) EntryUpdateActiveHabit(pastureId int64) (err error) {
 	if err != nil {
 		return xerr.WithStack(err)
 	}
-	xToday.ActiveLowest = activeLowest.Value
-	xToday.RuminaLowest = ruminaLowest.Value
+	xRuminaDisc, err := e.GetSystemConfigure2(pastureId, model.XRuminaDisc)
+	if err != nil {
+		return err
+	}
+	xChangeDiscount, err := e.GetSystemConfigure2(pastureId, model.XChangeDiscount)
+	if err != nil {
+		return err
+	}
+
+	xToday.ActiveLowest = int32(activeLowest.Value)
+	xToday.RuminaLowest = int32(ruminaLowest.Value)
+	xToday.XRuminaDisc = int32(xRuminaDisc.Value)
+	xToday.XChangeDiscount = int32(xChangeDiscount.Value)
+	xToday.LastMaxHabitId = lastMaxHabitId  // 上次执行的id
+	xToday.CurrMaxHabitId = currMaxHabit.Id // 本次执行的id
 	// 更新活动滤波
 	if err = e.FirstFilterUpdate(pastureId, xToday); err != nil {
 		zaplog.Error("EntryUpdateActiveHabit", zap.Any("FirstFilterUpdate", err), zap.Any("xToday", xToday))
@@ -145,12 +101,13 @@ func (e *Entry) EntryUpdateActiveHabit(pastureId int64) (err error) {
 	if err = e.WeeklyActiveAvgUpdate(pastureId, xToday); err != nil {
 		zaplog.Error("EntryUpdateActiveHabit", zap.Any("WeeklyActiveAvgUpdate", err), zap.Any("xToday", xToday))
 	}
-
+	// 二次更新滤波
 	if err = e.SecondUpdateChangeFilter(pastureId, xToday); err != nil {
 		zaplog.Error("EntryUpdateActiveHabit", zap.Any("SecondUpdateChangeFilter", err), zap.Any("xToday", xToday))
 	}
 
-	if err = e.ActivityVolumeChanges(pastureId, xToday); err != nil {
+	// 活动量校正系数和健康评分
+	if err = e.FilterCorrectAndScoreUpdate(pastureId, xToday); err != nil {
 		zaplog.Error("EntryUpdateActiveHabit", zap.Any("ActivityVolumeChanges", err), zap.Any("xToday", xToday))
 	}
 
@@ -188,49 +145,70 @@ func (e *Entry) EntryUpdateActiveHabit(pastureId int64) (err error) {
 func (e *Entry) FirstFilterUpdate(pastureId int64, xToDay *XToday) error {
 	newNeckActiveHabitList := make([]*model.NeckActiveHabit, 0)
 	if err := e.DB.Model(new(model.NeckActiveHabit)).
+		Where("id BETWEEN ? AND ?", xToDay.LastMaxHabitId, xToDay.CurrMaxHabitId).
 		Where("pasture_id = ?", pastureId).
-		Where(e.DB.Where("change_filter = ?", model.InitChangeFilter).Or("is_max_time = ?", pasturePb.IsShow_Ok)).
-		Order("neck_ring_number,id").
-		Find(&newNeckActiveHabitList).Error; err != nil {
+		Where("change_filter = ?", model.InitChangeFilter).
+		Where(e.DB.Where("high >= ?", xToDay.ActiveLowest).Or("rumina >= ?", xToDay.RuminaLowest)). // 活动量过低牛只不参与计算
+		Order("neck_ring_number,heat_date,frameid").
+		Limit(int(defaultLimit)).Find(&newNeckActiveHabitList).Error; err != nil {
 		return xerr.WithStack(err)
 	}
-
-	var filterValues = make(map[string]*model.NeckActiveHabit)
 	// 活动量滤波
 	for _, v := range newNeckActiveHabitList {
-		// 活动量过低牛只不参与计算
-		if v.High < int32(xToDay.ActiveLowest) || v.Rumina < int32(xToDay.RuminaLowest) {
-			continue
+		firstFilterData := &FirstFilterData{}
+		frameId := v.Frameid
+		heatDate := v.HeatDate
+		if v.Frameid == 0 {
+			frameId = 11
+			heatDateParse, _ := time.Parse(model.LayoutDate2, heatDate)
+			heatDate = heatDateParse.AddDate(0, 0, -1).Format(model.LayoutDate2)
 		}
-		prev, ok := filterValues[v.NeckRingNumber]
-		if !ok {
-			if v.FilterHigh <= 0 {
-				v.FilterHigh = v.High
-			}
-			if v.FilterRumina <= 0 {
-				v.FilterRumina = v.Rumina
-			}
-			if v.FilterChew <= 0 {
-				v.FilterChew = v.Rumina + v.Intake
-			}
-			filterValues[v.NeckRingNumber] = v
-			continue
+		if err := e.DB.Model(new(model.NeckActiveHabit)).
+			Select("neck_ring_number", "filter_high", "filter_rumina", "filter_chew").
+			Where("neck_ring_number = ?", v.NeckRingNumber).
+			Where("heat_date = ?", heatDate).
+			Where("frameid = ?", frameId).
+			First(&firstFilterData).Error; err != nil {
+			zaplog.Error("EntryUpdateActiveHabit", zap.Any("FirstFilterUpdate", err))
 		}
-		if v.FilterHigh <= 0 {
-			v.FilterHigh = int32(computeIfPositiveElse(float64(v.High), float64(prev.FilterHigh), 0.23, 0.77))
+
+		if v.FilterHigh > 0 {
+			firstFilterData.FilterHigh = v.FilterHigh
+		} else {
+			if v.NeckRingNumber == firstFilterData.NeckRingNumber {
+				firstFilterData.FilterHigh = int32(computeIfPositiveElse(float64(v.High), float64(firstFilterData.FilterHigh), 0.23, 0.77))
+			} else {
+				firstFilterData.FilterHigh = v.High
+			}
 		}
-		if v.FilterRumina <= 0 {
-			v.FilterRumina = int32(computeIfPositiveElse(float64(v.Rumina), float64(prev.FilterRumina), 0.33, 0.67))
+
+		if v.FilterRumina > 0 {
+			firstFilterData.FilterRumina = v.FilterRumina
+		} else {
+			if v.NeckRingNumber == firstFilterData.NeckRingNumber {
+				firstFilterData.FilterRumina = int32(computeIfPositiveElse(float64(v.Rumina), float64(firstFilterData.FilterRumina), 0.33, 0.67))
+			} else {
+				firstFilterData.FilterRumina = v.Rumina
+			}
 		}
-		if v.FilterChew <= 0 {
-			v.FilterChew = int32(computeIfPositiveElse(float64(v.Rumina+v.Intake), float64(prev.FilterChew), 0.33, 0.67))
+		if v.FilterChew > 0 {
+			firstFilterData.FilterChew = v.FilterChew
+		} else {
+			if v.NeckRingNumber == firstFilterData.NeckRingNumber {
+				firstFilterData.FilterChew = int32(computeIfPositiveElse(float64(v.Rumina+v.Intake), float64(firstFilterData.FilterChew), 0.33, 0.67))
+			} else {
+				firstFilterData.FilterChew = v.Rumina + v.Intake
+			}
 		}
 		// 更新过滤值
-		filterValues[v.NeckRingNumber] = v
 		if err := e.DB.Model(new(model.NeckActiveHabit)).
 			Select("filter_high", "filter_rumina", "filter_chew").
 			Where("id = ?", v.Id).
-			Updates(v).Error; err != nil {
+			Updates(map[string]interface{}{
+				"filter_high":   firstFilterData.FilterHigh,
+				"filter_rumina": firstFilterData.FilterRumina,
+				"filter_chew":   firstFilterData.FilterChew,
+			}).Error; err != nil {
 			return xerr.WithStack(err)
 		}
 	}
@@ -248,21 +226,22 @@ func (e *Entry) WeeklyActiveAvgUpdate(pastureId int64, xToday *XToday) (err erro
 	if err = e.DB.Transaction(func(tx *gorm.DB) error {
 		neckActiveHabitList := make([]*model.NeckActiveHabit, 0)
 		if err = tx.Model(new(model.NeckActiveHabit)).
-			Where("id BETWEEN ? AND ?", xToday.XMin2Id, xToday.CurrMaxHabitId).
+			Where("id BETWEEN ? AND ?", xToday.LastMaxHabitId, xToday.CurrMaxHabitId).
 			Where("pasture_id = ?", pastureId).
 			Where("change_filter = ?", model.InitChangeFilter).
 			Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina > ?", xToday.RuminaLowest)).
-			Group("neck_ring_number,frameid").
-			Find(&neckActiveHabitList).Error; err != nil {
-			zaplog.Error("WeeklyActiveAvgUpdate-0", zap.Any("error", err), zap.Any("xToday", xToday))
+			Order("neck_ring_number,frameid").
+			Limit(int(defaultLimit)).Find(&neckActiveHabitList).Error; err != nil {
 			return xerr.WithStack(err)
 		}
+
+		zaplog.Info("WeeklyActiveAvgUpdate-0", zap.Any("neckActiveHabitList", neckActiveHabitList))
 		for _, v := range neckActiveHabitList {
 			beginDayDate, _ := time.Parse(model.LayoutDate2, v.HeatDate)
 			before7DayDate := beginDayDate.AddDate(0, 0, -7).Format(model.LayoutDate2)
 			before1DayDate := beginDayDate.AddDate(0, 0, -1).Format(model.LayoutDate2)
 
-			avgHabitData := &crontab.AvgHabit{}
+			weekHabitData := &crontab.WeekHabit{}
 			if err = tx.Model(new(model.NeckActiveHabit)).
 				Select("neck_ring_number").
 				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").
@@ -270,12 +249,11 @@ func (e *Entry) WeeklyActiveAvgUpdate(pastureId int64, xToday *XToday) (err erro
 				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("pasture_id = ?", pastureId).
 				Where("neck_ring_number = ? AND frameid = ?", v.NeckRingNumber, v.Frameid).
 				Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina > ?", xToday.RuminaLowest)).
-				First(avgHabitData).Error; err != nil {
+				Group("neck_ring_number").First(weekHabitData).Error; err != nil {
 				zaplog.Error("WeeklyActiveAvgUpdate-1", zap.Any("error", err), zap.Any("xToday", xToday))
 			}
 
@@ -290,26 +268,26 @@ func (e *Entry) WeeklyActiveAvgUpdate(pastureId int64, xToday *XToday) (err erro
 				Select("MAX(h2.change_filter) as sum_max_high").
 				Select("MIN(IF(change_filter > ?, change_filter, ?)) as sum_min_high", model.DefaultChangeFilter, model.InitChangeFilter).
 				Select("MIN( CASE WHEN filter_chew > ? THEN filter_chew WHEN filter_rumina >= ? THEN filter_rumina ELSE 0 END) as sum_min_chew", model.DefaultChangeFilter, model.DefaultRuminaFilter).
-				Where("id BETWEEN ? AND ?", xToday.XMin2Id, xToday.CurrMaxHabitId).
 				Where("pasture_id = ?", pastureId).
 				Where("heat_date BETWEEN ? AND ?", before1DayDate, beginDayDate).
 				Where("active_time BETWEEN ? AND ?", beginDayDate.Add(-23*time.Hour).Format(model.LayoutTime), beginDayDate.Format(model.LayoutTime)).
 				Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina >= ?", xToday.RuminaLowest)).
 				Where("neck_ring_number = ? AND frameid = ?", v.NeckRingNumber, v.Frameid).
+				Group("neck_ring_number").
 				First(sumHabitData).Error; err != nil {
 				zaplog.Error("WeeklyActiveAvgUpdate-2", zap.Any("error", err), zap.Any("xToday", xToday))
 			}
 
 			// 变化百分比
-			if v.AvgHighHabit > 0 {
-				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)
+			if v.WeekHighHabit > 0 {
+				if v.FilterHigh-v.WeekHighHabit > 0 {
+					v.ChangeHigh = (v.FilterHigh - v.WeekHighHabit) / int32(float64(v.WeekHigh)*0.6+float64(v.WeekHighHabit)*0.2+float64(weeklyActive.Value)*0.2)
 				} else {
-					v.ChangeHigh = v.FilterHigh - v.AvgHighHabit/v.AvgHighHabit*100
+					v.ChangeHigh = v.FilterHigh - v.WeekHighHabit/v.WeekHighHabit*100
 				}
 
-				v.ChangeRumina = v.RuminaFilter - v.AvgRuminaHabit/v.AvgHighHabit*100
-				v.ChangeChew = v.FilterChew - v.AvgChewHabit/v.AvgHighHabit*100
+				v.ChangeRumina = v.RuminaFilter - v.WeekRuminaHabit/v.WeekHighHabit*100
+				v.ChangeChew = v.FilterChew - v.WeekChewHabit/v.WeekHighHabit*100
 				if err = e.DB.Model(new(model.NeckActiveHabit)).
 					Select("change_high", "change_rumina", "change_chew").
 					Where("id = ?", v.Id).
@@ -326,37 +304,22 @@ func (e *Entry) WeeklyActiveAvgUpdate(pastureId int64, xToday *XToday) (err erro
 				Where("pasture_id = ?", pastureId).
 				Where("neck_ring_number = ?", v.NeckRingNumber).
 				Where("heat_date = ?", before3DayDate).
-				Where("frameid = ? ", v.Frameid).First(before3DaysNeckActiveHabit).Error; err != nil {
+				Where("frameid = ? ", v.Frameid).
+				First(before3DaysNeckActiveHabit).Error; err != nil {
 				zaplog.Error("WeeklyActiveAvgUpdate-3", zap.Any("error", err), zap.Any("xToday", xToday))
 			}
 
-			cowId := int64(0)
-			Lact := int32(0)
-			calvingAge := int64(0)
-			cow := e.GetCowInfoByNeckRingNumber(pastureId, v.NeckRingNumber)
-			if cow != nil {
-				cowId = cow.Id
-				Lact = cow.Lact
-				calvingAge = cow.CalvingAge
-			} else {
-				zaplog.Error("WeeklyActiveAvgUpdate-4", zap.Any("error", err), zap.Any("xToday", xToday))
-			}
-
 			if err = tx.Model(new(model.NeckActiveHabit)).
 				Select("avg_high_habit", "avg_rumina_habit", "avg_chew_habit", "avg_intake_habit", "avg_inactive_habit").
 				Select("sum_rumina", "sum_intake", "sum_inactive", "sum_active", "sum_max_high", "sum_min_high", "sum_min_chew").
-				Select("change_high", "change_rumina", "change_chew", "cow_id", "lact", "calving_age").
-				Select("sum_rumina_before_three_day,sum_intake_before_three_day").
+				Select("change_high", "change_rumina", "change_chew", "sum_rumina_before_three_day", "sum_intake_before_three_day").
 				Where("id = ?", v.Id).
 				Updates(map[string]interface{}{
-					"cow_id":                      cowId,
-					"lact":                        Lact,
-					"calving_age":                 calvingAge,
-					"avg_high_habit":              avgHabitData.AvgHighHabit,
-					"avg_rumina_habit":            avgHabitData.AvgRuminaHabit,
-					"avg_chew_habit":              avgHabitData.AvgChewHabit,
-					"avg_intake_habit":            avgHabitData.AvgIntakeHabit,
-					"avg_inactive_habit":          avgHabitData.AvgIntakeHabit,
+					"week_high_habit":             weekHabitData.AvgHighHabit,
+					"week_rumina_habit":           weekHabitData.AvgRuminaHabit,
+					"week_chew_habit":             weekHabitData.AvgChewHabit,
+					"week_intake_habit":           weekHabitData.AvgIntakeHabit,
+					"week_inactive_habit":         weekHabitData.AvgIntakeHabit,
 					"sum_rumina":                  sumHabitData.SumRumina,
 					"sum_intake":                  sumHabitData.SumIntake,
 					"sum_inactive":                sumHabitData.SumInactive,
@@ -371,121 +334,133 @@ func (e *Entry) WeeklyActiveAvgUpdate(pastureId int64, xToday *XToday) (err erro
 					"sum_intake_before_three_day": before3DaysNeckActiveHabit.SumIntake,
 				}).Error; err != nil {
 				zaplog.Error("WeeklyActiveAvgUpdate-6", zap.Any("error", err), zap.Any("xToday", xToday))
-				continue
 			}
+
+			zaplog.Info("WeeklyActiveAvgUpdate-7",
+				zap.Any("v", v),
+				zap.Any("xToday", xToday),
+				zap.Any("before3DaysNeckActiveHabit", before3DaysNeckActiveHabit),
+				zap.Any("sumHabitData", sumHabitData),
+				zap.Any("weekHabitData", weekHabitData),
+			)
 		}
 		return nil
 	}); err != nil {
 		return xerr.WithStack(err)
 	}
-
+	zaplog.Info("WeeklyActiveAvgUpdate-Success", zap.Any("xToday", xToday))
 	return nil
 }
 
 // SecondUpdateChangeFilter 第二次更新变化趋势滤波
 func (e *Entry) SecondUpdateChangeFilter(pastureId int64, xToday *XToday) (err error) {
-	xRuminaDisc, err := e.GetSystemConfigure2(pastureId, model.XRuminaDisc)
-	if err != nil {
-		return err
-	}
-	xChangeDiscount, err := e.GetSystemConfigure2(pastureId, model.XChangeDiscount)
-	if err != nil {
-		return err
-	}
-	newChangeFilterList := make([]*crontab.ChangeFilterData, 0)
+	newChangeFilterList := make([]*ChangeFilterData, 0)
 	if err = e.DB.Model(new(model.NeckActiveHabit)).
-		Select("id,neck_ring_number,change_high,change_filter,rumina_filter,change_rumina,chew_filter,change_chew").
+		Select("id,neck_ring_number,change_high,change_filter,rumina_filter,change_rumina,chew_filter,change_chew,heat_date,frameid").
 		Select("IF(lact = 0, 0.8, 1) as xlc_dis_count").
 		Where("id BETWEEN ? AND ?", xToday.XMin2Id, xToday.CurrMaxHabitId).
 		Where("pasture_id = ?", pastureId).
-		Where(e.DB.Where("change_filter = ?", model.InitChangeFilter).Or("is_max_time = ?", pasturePb.IsShow_Ok)).
+		Where("change_filter = ?", model.InitChangeFilter).
 		Where("change_high > ?", MinChangeHigh).
 		Order("neck_ring_number,heat_date,frameid").
 		Find(&newChangeFilterList).Error; err != nil {
 		return xerr.WithStack(err)
 	}
-	var filterValues = make(map[string]*crontab.ChangeFilterData)
+
 	for _, v := range newChangeFilterList {
-		prev, ok := filterValues[v.NeckRingNumber]
-		if v.ChangeFilter <= MinChangeFilter {
-			prefChangeFilter := int32(0)
-			if ok {
-				prefChangeFilter = prev.ChangeFilter
-			}
-			leastValue := v.HighChange
-			if prefChangeFilter < v.HighChange {
-				leastValue = prefChangeFilter
-			}
-			v.ChangeFilter = int32(float64(prefChangeFilter)*(1-(float64(xChangeDiscount.Value)/10)*v.XlcDisCount) +
-				float64(leastValue)*(float64(xChangeDiscount.Value)/10)*v.XlcDisCount)
+		secondFilterData := &SecondFilterData{}
+		frameId := v.FrameId
+		heatDate := v.HeatDate
+		if v.FrameId == 0 {
+			frameId = 11
+			heatDateParse, _ := time.Parse(model.LayoutDate2, heatDate)
+			heatDate = heatDateParse.AddDate(0, 0, -1).Format(model.LayoutDate2)
+		}
+		if err = e.DB.Model(new(model.NeckActiveHabit)).
+			Select("neck_ring_number", "filter_high", "filter_rumina", "filter_chew").
+			Where("neck_ring_number = ?", v.NeckRingNumber).
+			Where("heat_date = ?", heatDate).
+			Where("frameid = ?", frameId).
+			First(&secondFilterData).Error; err != nil {
+			zaplog.Error("EntryUpdateActiveHabit", zap.Any("FirstFilterUpdate", err))
 		}
 
-		if v.RuminaFilter <= MinChangeFilter {
-			prefRuminaFilter := int32(0)
-			if ok {
-				prefRuminaFilter = prev.RuminaFilter
+		if v.ChangeFilter > MinChangeFilter {
+			secondFilterData.ChangeFilter = float64(v.ChangeFilter)
+		} else {
+			if v.NeckRingNumber == secondFilterData.NeckRingNumber {
+				secondFilterData.ChangeFilter = secondFilterData.ChangeFilter*(1-(float64(xToday.XChangeDiscount)/10)*v.XlcDisCount) +
+					math.Min(float64(v.ChangeHigh), secondFilterData.ChangeFilter+135)*(float64(xToday.XChangeDiscount)/10)*v.XlcDisCount
+			} else {
+				secondFilterData.ChangeFilter = 0
 			}
+		}
 
-			factor := float64(1)
-			if math.Abs(float64(v.ChangeRumina)) > 60 {
-				factor = 0.5
+		if v.RuminaFilter > MinRuminaFilter {
+			secondFilterData.RuminaFilter = float64(v.ChangeFilter)
+		} else {
+			if v.NeckRingNumber == secondFilterData.NeckRingNumber {
+				discount := float64(xToday.XRuminaDisc) / 10 * v.XlcDisCount
+				if math.Abs(float64(v.ChangeRumina)) > 60 {
+					discount *= 0.5
+				}
+				secondFilterData.RuminaFilter = secondFilterData.RuminaFilter*(1-discount) + float64(v.ChangeRumina)*discount
+			} else {
+				secondFilterData.RuminaFilter = 0
 			}
-
-			v.RuminaFilter = int32(float64(prefRuminaFilter)*(1-float64(xRuminaDisc.Value/10)*v.XlcDisCount*factor) +
-				float64(v.ChangeRumina)*float64(xRuminaDisc.Value)/10*v.XlcDisCount*factor)
-
-		}
-		if v.RuminaFilter > 50 {
-			v.RuminaFilter = 50
 		}
 
-		if v.ChewFilter <= MinChangeFilter {
-			prefChewFilter := int32(0)
-			if ok {
-				prefChewFilter = prev.ChewFilter
-			}
-			factor := float64(1)
-			if math.Abs(float64(v.ChangeChew)) > 60 {
-				factor = 0.5
-			}
+		secondFilterData.RuminaFilter = math.Min(50, secondFilterData.RuminaFilter)
 
-			v.ChewFilter = int32(float64(prefChewFilter)*(1-float64(xRuminaDisc.Value)/10*factor) +
-				float64(v.ChangeChew)*float64(xRuminaDisc.Value)/10*factor)
+		if v.ChewFilter > MinChewFilter {
+			secondFilterData.ChewFilter = float64(v.ChangeChew)
+		} else {
+			if v.NeckRingNumber == secondFilterData.NeckRingNumber {
+				discount := float64(xToday.XRuminaDisc) / 10
+				if math.Abs(float64(v.ChangeChew)) > 60 {
+					discount *= 0.5
+				}
+				secondFilterData.ChewFilter = secondFilterData.ChewFilter*(1-discount) + float64(v.ChangeChew)*discount
+			} else {
+				secondFilterData.ChewFilter = 0
+			}
 		}
 
-		if v.ChewFilter > 50 {
-			v.ChangeChew = 50
-		}
+		secondFilterData.ChewFilter = math.Min(50, secondFilterData.ChewFilter)
 		if err = e.DB.Model(new(model.NeckActiveHabit)).
 			Select("change_filter", "rumina_filter", "chew_filter").
 			Where("id = ?", v.Id).
 			Where("neck_ring_number = ?", v.NeckRingNumber).
 			Where("change_filter = ?", model.InitChangeFilter).
-			Updates(v).Error; err != nil {
+			Updates(map[string]interface{}{
+				"change_filter": secondFilterData.ChangeFilter,
+				"rumina_filter": secondFilterData.RuminaFilter,
+				"chew_filter":   secondFilterData.ChewFilter,
+			}).Error; err != nil {
 			zaplog.Error("SecondUpdateChangeFilter-1", zap.Any("error", err), zap.Any("xToday", xToday))
 		}
-		filterValues[v.NeckRingNumber] = v
+		zaplog.Info("SecondUpdateChangeFilter",
+			zap.Any("secondFilterData", secondFilterData),
+			zap.Any("xToday", xToday),
+			zap.Any("v", v),
+		)
 	}
 	return nil
 }
 
-// ActivityVolumeChanges 计算活动量变化趋势校正值(活跃度校正)
-func (e *Entry) ActivityVolumeChanges(pastureId int64, xToday *XToday) error {
-	beginDayDate, err := time.Parse(model.LayoutDate2, xToday.XBegDate)
-	if err != nil {
-		return xerr.WithStack(err)
-	}
+// FilterCorrectAndScoreUpdate 计算活动量变化趋势校正值(活跃度校正)和健康评分
+func (e *Entry) FilterCorrectAndScoreUpdate(pastureId int64, xToday *XToday) error {
+	beginDayDate := time.Now()
 	before7DayDate := beginDayDate.AddDate(0, 0, -7).Format(model.LayoutDate2)
 	before1DayDate := beginDayDate.AddDate(0, 0, -1).Format(model.LayoutDate2)
 
 	activityVolumeList := make([]*crontab.ActivityVolume, 0)
 	activityVolumeMap := make(map[string]*crontab.ActivityVolume)
-	if err = e.DB.Model(new(model.NeckActiveHabit)).
+	if err := e.DB.Model(new(model.NeckActiveHabit)).
 		Select("neck_ring_number").
 		Select("AVG(IF(change_filter>=60, 60, change_filter)) as avg_filter").
 		Select("ROUND(STD(IF(change_filter>=60, 60, change_filter))) as std_filter").
 		Select("COUNT(1) as nb").
-		Where("id BETWEEN ? AND ?", xToday.XMin7Id, xToday.CurrMaxHabitId).
 		Where("heat_date BETWEEN ? AND ?", before7DayDate, before1DayDate).
 		Where("pasture_id = ?", pastureId).
 		Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina >= ?", xToday.RuminaLowest)).
@@ -504,7 +479,7 @@ func (e *Entry) ActivityVolumeChanges(pastureId int64, xToday *XToday) error {
 	}
 
 	neckActiveHabitList := make([]*model.NeckActiveHabit, 0)
-	if err = e.DB.Model(new(model.NeckActiveHabit)).
+	if err := e.DB.Model(new(model.NeckActiveHabit)).
 		Where("id BETWEEN ? AND ?", xToday.XMin2Id, xToday.CurrMaxHabitId).
 		Where("pasture_id = ?", pastureId).
 		Where(e.DB.Where("high > ?", xToday.ActiveLowest).Or("rumina > ?", xToday.RuminaLowest)).
@@ -516,10 +491,8 @@ func (e *Entry) ActivityVolumeChanges(pastureId int64, xToday *XToday) error {
 	for _, v := range neckActiveHabitList {
 		if filterCorrectMap, ok := activityVolumeMap[v.NeckRingNumber]; ok {
 			filterCorrect := model.DefaultFilterCorrect - int(math.Floor(float64(filterCorrectMap.AvgFilter)/3+float64(filterCorrectMap.StdFilter)/2))
-			v.FilterCorrect = int32(filterCorrect)
-
 			// 活动量校正系数
-			if err = e.DB.Model(new(model.NeckActiveHabit)).
+			if err := e.DB.Model(new(model.NeckActiveHabit)).
 				Where("id = ?", v.Id).
 				Where("neck_ring_number = ?", v.NeckRingNumber).
 				Update("filter_correct", filterCorrect).Error; err != nil {
@@ -527,19 +500,13 @@ func (e *Entry) ActivityVolumeChanges(pastureId int64, xToday *XToday) error {
 				continue
 			}
 		}
-
 		// 健康评分
 		if v.Score != 0 {
 			continue
 		}
 
-		cow := e.GetCowInfoByNeckRingNumber(v.PastureId, v.NeckRingNumber)
-		if cow == nil {
-			continue
-		}
-
 		cowScore := calculateScore(v)
-		if err = e.DB.Model(new(model.NeckActiveHabit)).
+		if err := e.DB.Model(new(model.NeckActiveHabit)).
 			Where("id = ?", v.Id).
 			Where("neck_ring_number = ?", v.NeckRingNumber).
 			Update("score", cowScore).Error; err != nil {

+ 103 - 0
module/mqtt/neck_ring_habit_test.go

@@ -0,0 +1,103 @@
+package mqtt
+
+import (
+	"encoding/json"
+	"fmt"
+	"kpt-pasture/model"
+	"testing"
+)
+
+func TestFirstFilterUpdate(t *testing.T) {
+	newNeckActiveHabitList := []*model.NeckActiveHabit{
+		{
+			Id:             1,
+			NeckRingNumber: "123",
+			High:           731,
+			Rumina:         20,
+			Intake:         0,
+			Inactive:       65,
+			Gasp:           1,
+			Other:          0,
+			Active:         27,
+			FilterHigh:     0,
+			FilterRumina:   0,
+			FilterChew:     0,
+		},
+		{
+			Id:             2,
+			NeckRingNumber: "123",
+			High:           318,
+			Rumina:         55,
+			Intake:         2,
+			Inactive:       99,
+			Gasp:           1,
+			Other:          0,
+			Active:         18,
+			FilterHigh:     0,
+			FilterRumina:   0,
+			FilterChew:     0,
+		},
+		{
+			Id:             2,
+			NeckRingNumber: "123",
+			High:           254,
+			Rumina:         68,
+			Intake:         0,
+			Inactive:       114,
+			Gasp:           22,
+			Other:          0,
+			Active:         6,
+			FilterHigh:     0,
+			FilterRumina:   0,
+			FilterChew:     0,
+		},
+	}
+
+	test(newNeckActiveHabitList)
+}
+
+func test(newNeckActiveHabitList []*model.NeckActiveHabit) {
+
+	var (
+		neckRingNumber string
+		filterHigh     float64
+		filterRumina   float64
+		filterChew     float64
+	)
+	// 活动量滤波
+	for _, v := range newNeckActiveHabitList {
+		if v.FilterHigh > 0 {
+			filterHigh = float64(v.FilterHigh)
+		} else {
+			if v.NeckRingNumber == neckRingNumber {
+				filterHigh = computeIfPositiveElse(float64(v.High), filterHigh, 0.23, 0.77)
+			} else {
+				filterHigh = float64(v.High)
+			}
+		}
+
+		if v.FilterRumina > 0 {
+			filterRumina = float64(v.FilterRumina)
+		} else {
+			if v.NeckRingNumber == neckRingNumber {
+				filterRumina = computeIfPositiveElse(float64(v.Rumina), filterRumina, 0.33, 0.67)
+			} else {
+				filterRumina = float64(v.Rumina)
+			}
+		}
+		if v.FilterChew > 0 {
+			filterChew = float64(v.FilterChew)
+		} else {
+			if v.NeckRingNumber == neckRingNumber {
+				filterChew = computeIfPositiveElse(float64(v.Rumina+v.Intake), filterChew, 0.33, 0.67)
+			} else {
+				filterChew = float64(v.Rumina + v.Intake)
+			}
+		}
+
+		// 	Select("filter_high", "filter_rumina", "filter_chew")
+		b, _ := json.Marshal(v)
+		fmt.Println("======v=====", string(b))
+	}
+
+}

+ 25 - 6
module/mqtt/sql.go

@@ -57,15 +57,34 @@ func (e *Entry) IsExistNeckActiveHabit(neckRingNumber, heatDate string, frameId
 		Where("neck_ring_number = ?", neckRingNumber).
 		Where("active_date = ?", heatDate).
 		Where("frameid = ?", frameId).
-		Count(&count).First(neckRingProcess).Error; err != nil {
+		Count(&count).
+		First(neckRingProcess).Error; err != nil {
 		return nil, 0
 	}
+
 	res := &model.NeckActiveHabit{}
-	if neckRingProcess != nil && neckRingProcess.HabitId > 0 {
-		if err := e.DB.Model(new(model.NeckActiveHabit)).
-			Where("id = ?", neckRingProcess.HabitId).
-			First(res).Error; err != nil {
-			return nil, 0
+	if neckRingProcess != nil {
+		if neckRingProcess.HabitId > 0 {
+			if err := e.DB.Model(new(model.NeckActiveHabit)).
+				Where("id = ?", neckRingProcess.HabitId).
+				First(res).Error; err != nil {
+				return nil, 0
+			}
+		} else {
+			if err := e.DB.Model(new(model.NeckActiveHabit)).
+				Where("heat_date = ?", heatDate).
+				Where("neck_ring_number = ?", neckRingNumber).
+				Where("frameid = ?", frameId).
+				First(res).Error; err != nil {
+				return nil, 0
+			}
+
+			if err := e.DB.Model(new(model.NeckRingProcess)).
+				Where("id = ?", neckRingProcess.Id).
+				Where("frameid = ?", frameId).
+				Update("habit_id", res.Id).Error; err != nil {
+				return nil, 0
+			}
 		}
 	}
 	return res, count

+ 23 - 0
util/util.go

@@ -374,9 +374,13 @@ func GetNeckRingActiveTimer(frameId int32) (dateTime string, hours int) {
 	hours, ok := SpecialHours[int(frameId)]
 	day := 0
 	if ok {
+		if hours == 0 {
+			hours = 24
+		}
 		if hours > currHour {
 			day = -1
 		}
+		hours = 0
 		dateTime = nowTime.AddDate(0, 0, day).Format(Layout)
 		return
 	}
@@ -401,6 +405,25 @@ func GetNeckRingActiveTimer(frameId int32) (dateTime string, hours int) {
 	return
 }
 
+// XFrameId 获取XFrameId
+func XFrameId(frameid int32) int32 {
+	return int32(math.Floor(float64(frameid / 10)))
+}
+
+// FrameIds 获取FrameIds
+func FrameIds(xFrameId int32) []int32 {
+	frameIds := make([]int32, 0)
+	for i := 1; i <= 6; i++ {
+		frameIds = append(frameIds, xFrameId*10+int32(i))
+	}
+	return frameIds
+}
+
+func CurrentMaxFrameId() int32 {
+	currentHour := time.Now().Hour()
+	return int32(math.Floor(float64(currentHour/2))) * 10
+}
+
 func ArrayInt32ToStrings(cowIds []int32, cutset string) string {
 	cows := ""
 	if len(cowIds) <= 0 {

+ 2 - 20
util/util_test.go

@@ -2,7 +2,6 @@ package util
 
 import (
 	"fmt"
-	"math"
 	"testing"
 	"time"
 
@@ -522,24 +521,7 @@ type XToday struct {
 }
 
 func Test_demo(t *testing.T) {
-	frameId := []int32{
-		1, 2, 3, 4, 5, 6, 8,
-		11, 12, 13, 14, 15, 16, 18,
-		21, 22, 23, 24, 25, 26, 28,
-		31, 32, 33, 34, 35, 36, 38,
-		41, 42, 43, 44, 45, 46, 48,
-		51, 52, 53, 54, 55, 56, 58,
-		61, 62, 63, 64, 65, 66, 68,
-		71, 72, 73, 74, 75, 76, 78,
-		81, 82, 83, 84, 85, 86, 88,
-		91, 92, 93, 94, 95, 96, 98,
-		101, 102, 103, 104, 105, 106, 108,
-		111, 112, 113, 114, 115, 116, 118,
-		8, 18, 28, 38, 48, 58, 68, 78, 88, 98, 108, 118,
-	}
-	for _, v := range frameId {
-		activeDate, hour := GetNeckRingActiveTimer(v)
-		xframeId := int32(math.Floor(float64(v) / 10))
-		fmt.Println("v", v, "activeDate", activeDate, "hour", hour, "xframeId", xframeId)
+	for i := 0; i < 23; i++ {
+		fmt.Println(CurrentMaxFrameId())
 	}
 }