core/plate.go

144 lines
4.0 KiB
Go
Raw Normal View History

2024-12-15 17:40:14 +08:00
package core
import (
"encoding/json"
2025-02-12 03:56:15 +08:00
"fmt"
2024-12-15 17:40:14 +08:00
"time"
)
type Plate struct {
InstID string `json:"instId,string"`
Scale float64 `json:"scale,number"`
Count int `json:"count,number"`
CoasterMap map[string]Coaster `json:"coasterMap"`
}
func (pl *Plate) Init(instId string) {
pl.InstID = instId
pl.Count = 24
pl.Scale = float64(0.005)
pl.CoasterMap = make(map[string]Coaster)
}
// TODO 从redis里读出来已经存储的plate如果不存在就创建一个新的
2025-02-12 03:56:15 +08:00
// LoadPlate 加载或创建指定Instrument的Plate
// 1. 尝试从Redis加载已存储的Plate
// 2. 如果Redis中不存在则初始化一个新的Plate
// 3. 根据配置创建所有需要的Coaster
// 参数:
// - cr: 核心上下文对象
// - instId: Instrument ID
//
// 返回值:
// - *Plate: 加载或新建的Plate对象
// - error: 操作过程中发生的错误
2024-12-15 17:40:14 +08:00
func LoadPlate(cr *Core, instId string) (*Plate, error) {
2025-02-12 03:56:15 +08:00
// 初始化一个空的Plate对象
2024-12-15 17:40:14 +08:00
pl := Plate{}
2025-02-12 03:56:15 +08:00
// 构造Redis中存储Plate数据的key格式为"instId|plate"
2024-12-15 17:40:14 +08:00
plateName := instId + "|plate"
2025-02-12 03:56:15 +08:00
// 尝试从Redis获取Plate数据
str, err := cr.RedisLocalCli.Get(plateName).Result()
// 如果Redis中存在数据且没有错误
if err == nil && str != "" {
// 将JSON字符串反序列化为Plate对象
if err := json.Unmarshal([]byte(str), &pl); err != nil {
// 反序列化失败时初始化一个新的Plate
// logrus.Warnf("failed to unmarshal plate data from redis, init new plate: %v", err)
pl.Init(instId)
}
// 返回从Redis加载的Plate对象
return &pl, nil
}
// Redis不可用或数据不存在时初始化一个新的Plate
pl.Init(instId)
// 从配置中获取candleDimentions配置项
prs := cr.Cfg.Config.Get("candleDimentions")
// 检查配置项是否存在
if prs == nil || prs.Interface() == nil {
return nil, fmt.Errorf("candleDimentions config not found")
}
// 将配置项转换为数组
periods := prs.MustArray()
// 遍历所有周期配置
for _, v := range periods {
// 将interface{}类型转换为string
period, ok := v.(string)
// 检查周期字符串是否有效
if !ok || period == "" {
continue // 跳过无效的周期配置
}
// 为每个周期创建Coaster
if err := pl.MakeCoaster(cr, period); err != nil {
// 如果创建失败,返回错误
return nil, fmt.Errorf("failed to create coaster for period %s: %v", period, err)
2024-12-15 17:40:14 +08:00
}
}
2025-02-12 03:56:15 +08:00
// 将新创建的Plate保存到Redis可选操作
if err := savePlateToRedis(cr, plateName, &pl); err != nil {
// 保存失败时记录警告日志
// logrus.Warnf("failed to save plate to redis: %v", err)
}
// 返回创建好的Plate对象
2024-12-15 17:40:14 +08:00
return &pl, nil
}
2025-02-12 03:56:15 +08:00
func savePlateToRedis(cr *Core, key string, pl *Plate) error {
data, err := json.Marshal(pl)
if err != nil {
return err
}
return cr.RedisLocalCli.Set(key, data, 0).Err()
}
2024-12-15 17:40:14 +08:00
func (pl *Plate) SetToKey(cr *Core) error {
js, _ := json.Marshal(*pl)
plateName := pl.InstID + "|plate"
_, err := cr.RedisLocalCli.Set(plateName, string(js), 0).Result()
return err
}
2025-01-15 00:24:56 +08:00
func (pl *Plate) MakeCoaster(cr *Core, period string) error {
2024-12-15 17:40:14 +08:00
lastTime := time.Now()
setName := "candle" + period + "|" + pl.InstID + "|sortedSet"
cdl, err := cr.GetRangeCandleSortedSet(setName, pl.Count, lastTime)
if err != nil {
2025-01-15 00:24:56 +08:00
return err
2024-12-15 17:40:14 +08:00
}
err = cdl.RecursiveBubbleS(len(cdl.List), "asc")
2024-12-15 17:40:14 +08:00
setName7 := "ma7|" + setName
setName30 := "ma30|" + setName
mxl7, err := cr.GetRangeMaXSortedSet(setName7, pl.Count, lastTime)
if err != nil {
2025-01-15 00:24:56 +08:00
return err
2024-12-15 17:40:14 +08:00
}
err = mxl7.RecursiveBubbleS(len(mxl7.List), "asc")
if err != nil {
return err
}
2024-12-15 17:40:14 +08:00
mxl30, err := cr.GetRangeMaXSortedSet(setName30, pl.Count, lastTime)
if err != nil {
2025-01-15 00:24:56 +08:00
return err
2024-12-15 17:40:14 +08:00
}
err = mxl30.RecursiveBubbleS(len(mxl30.List), "asc")
2024-12-15 17:40:14 +08:00
coaster := Coaster{
InstID: pl.InstID,
Period: period,
Count: pl.Count,
Scale: pl.Scale,
CandleList: *cdl,
Ma7List: *mxl7,
Ma30List: *mxl30,
}
pl.CoasterMap["period"+period] = coaster
2025-01-15 00:24:56 +08:00
return err
2024-12-15 17:40:14 +08:00
}