mirror of
https://github.com/laoxong/nofx.git
synced 2026-06-04 01:48:22 +08:00
23dbbf6bdd
- Add kucoin to validTypes in api/server.go - Add KuCoin trader creation in trader_manager.go - Fix PostgreSQL duplicate key in equity.go (Omit ID) - Start KuCoin order sync in auto_trader.go - Update FooterSection UI
182 lines
6.0 KiB
Go
182 lines
6.0 KiB
Go
package store
|
|
|
|
import (
|
|
"fmt"
|
|
"time"
|
|
|
|
"gorm.io/gorm"
|
|
)
|
|
|
|
// EquityStore account equity storage (for plotting return curves)
|
|
type EquityStore struct {
|
|
db *gorm.DB
|
|
}
|
|
|
|
// EquitySnapshot equity snapshot
|
|
type EquitySnapshot struct {
|
|
ID int64 `gorm:"primaryKey;autoIncrement" json:"id"`
|
|
TraderID string `gorm:"column:trader_id;not null;index:idx_equity_trader_time" json:"trader_id"`
|
|
Timestamp time.Time `gorm:"not null;index:idx_equity_trader_time,sort:desc;index:idx_equity_timestamp,sort:desc" json:"timestamp"`
|
|
TotalEquity float64 `gorm:"column:total_equity;not null;default:0" json:"total_equity"`
|
|
Balance float64 `gorm:"not null;default:0" json:"balance"`
|
|
UnrealizedPnL float64 `gorm:"column:unrealized_pnl;not null;default:0" json:"unrealized_pnl"`
|
|
PositionCount int `gorm:"column:position_count;default:0" json:"position_count"`
|
|
MarginUsedPct float64 `gorm:"column:margin_used_pct;default:0" json:"margin_used_pct"`
|
|
CreatedAt time.Time `json:"created_at"`
|
|
}
|
|
|
|
func (EquitySnapshot) TableName() string { return "trader_equity_snapshots" }
|
|
|
|
// NewEquityStore creates a new EquityStore
|
|
func NewEquityStore(db *gorm.DB) *EquityStore {
|
|
return &EquityStore{db: db}
|
|
}
|
|
|
|
// initTables initializes equity tables
|
|
func (s *EquityStore) initTables() error {
|
|
// For PostgreSQL with existing table, skip AutoMigrate
|
|
if s.db.Dialector.Name() == "postgres" {
|
|
var tableExists int64
|
|
s.db.Raw(`SELECT COUNT(*) FROM information_schema.tables WHERE table_name = 'trader_equity_snapshots'`).Scan(&tableExists)
|
|
if tableExists > 0 {
|
|
return nil
|
|
}
|
|
}
|
|
return s.db.AutoMigrate(&EquitySnapshot{})
|
|
}
|
|
|
|
// Save saves equity snapshot
|
|
func (s *EquityStore) Save(snapshot *EquitySnapshot) error {
|
|
if snapshot.Timestamp.IsZero() {
|
|
snapshot.Timestamp = time.Now().UTC()
|
|
} else {
|
|
snapshot.Timestamp = snapshot.Timestamp.UTC()
|
|
}
|
|
|
|
// Omit ID to let PostgreSQL sequence auto-generate it
|
|
// Without this, GORM inserts ID=0 which causes duplicate key errors
|
|
if err := s.db.Omit("ID").Create(snapshot).Error; err != nil {
|
|
return fmt.Errorf("failed to save equity snapshot: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// GetLatest gets the latest N equity records for specified trader (sorted in ascending chronological order: old to new)
|
|
func (s *EquityStore) GetLatest(traderID string, limit int) ([]*EquitySnapshot, error) {
|
|
var snapshots []*EquitySnapshot
|
|
err := s.db.Where("trader_id = ?", traderID).
|
|
Order("timestamp DESC").
|
|
Limit(limit).
|
|
Find(&snapshots).Error
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to query equity records: %w", err)
|
|
}
|
|
|
|
// Reverse the array to sort time from old to new (suitable for plotting curves)
|
|
for i, j := 0, len(snapshots)-1; i < j; i, j = i+1, j-1 {
|
|
snapshots[i], snapshots[j] = snapshots[j], snapshots[i]
|
|
}
|
|
|
|
return snapshots, nil
|
|
}
|
|
|
|
// GetByTimeRange gets equity records within specified time range
|
|
func (s *EquityStore) GetByTimeRange(traderID string, start, end time.Time) ([]*EquitySnapshot, error) {
|
|
var snapshots []*EquitySnapshot
|
|
err := s.db.Where("trader_id = ? AND timestamp >= ? AND timestamp <= ?", traderID, start, end).
|
|
Order("timestamp ASC").
|
|
Find(&snapshots).Error
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to query equity records: %w", err)
|
|
}
|
|
return snapshots, nil
|
|
}
|
|
|
|
// GetAllTradersLatest gets latest equity for all traders (for leaderboards)
|
|
func (s *EquityStore) GetAllTradersLatest() (map[string]*EquitySnapshot, error) {
|
|
// Use raw SQL for this complex query with subquery
|
|
var snapshots []*EquitySnapshot
|
|
err := s.db.Raw(`
|
|
SELECT e.id, e.trader_id, e.timestamp, e.total_equity, e.balance,
|
|
e.unrealized_pnl, e.position_count, e.margin_used_pct, e.created_at
|
|
FROM trader_equity_snapshots e
|
|
INNER JOIN (
|
|
SELECT trader_id, MAX(timestamp) as max_ts
|
|
FROM trader_equity_snapshots
|
|
GROUP BY trader_id
|
|
) latest ON e.trader_id = latest.trader_id AND e.timestamp = latest.max_ts
|
|
`).Scan(&snapshots).Error
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to query latest equity: %w", err)
|
|
}
|
|
|
|
result := make(map[string]*EquitySnapshot)
|
|
for _, snap := range snapshots {
|
|
result[snap.TraderID] = snap
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
// CleanOldRecords cleans old records from N days ago
|
|
func (s *EquityStore) CleanOldRecords(traderID string, days int) (int64, error) {
|
|
cutoffTime := time.Now().AddDate(0, 0, -days)
|
|
|
|
result := s.db.Where("trader_id = ? AND timestamp < ?", traderID, cutoffTime).
|
|
Delete(&EquitySnapshot{})
|
|
if result.Error != nil {
|
|
return 0, fmt.Errorf("failed to clean old records: %w", result.Error)
|
|
}
|
|
return result.RowsAffected, nil
|
|
}
|
|
|
|
// GetCount gets record count for specified trader
|
|
func (s *EquityStore) GetCount(traderID string) (int, error) {
|
|
var count int64
|
|
err := s.db.Model(&EquitySnapshot{}).Where("trader_id = ?", traderID).Count(&count).Error
|
|
return int(count), err
|
|
}
|
|
|
|
// MigrateFromDecision migrates data from old decision_account_snapshots table
|
|
func (s *EquityStore) MigrateFromDecision() (int64, error) {
|
|
// Check if migration is needed (whether new table is empty)
|
|
var count int64
|
|
s.db.Model(&EquitySnapshot{}).Count(&count)
|
|
if count > 0 {
|
|
return 0, nil // Already has data, skip migration
|
|
}
|
|
|
|
// Check if old table exists (SQLite specific check, but works for migration)
|
|
var tableName string
|
|
err := s.db.Raw(`
|
|
SELECT name FROM sqlite_master
|
|
WHERE type='table' AND name='decision_account_snapshots'
|
|
`).Scan(&tableName).Error
|
|
if err != nil || tableName == "" {
|
|
return 0, nil // Old table doesn't exist, skip
|
|
}
|
|
|
|
// Migrate data: join query from decision_records + decision_account_snapshots
|
|
result := s.db.Exec(`
|
|
INSERT INTO trader_equity_snapshots (
|
|
trader_id, timestamp, total_equity, balance,
|
|
unrealized_pnl, position_count, margin_used_pct
|
|
)
|
|
SELECT
|
|
dr.trader_id,
|
|
dr.timestamp,
|
|
das.total_balance,
|
|
das.available_balance,
|
|
das.total_unrealized_profit,
|
|
das.position_count,
|
|
das.margin_used_pct
|
|
FROM decision_records dr
|
|
JOIN decision_account_snapshots das ON dr.id = das.decision_id
|
|
ORDER BY dr.timestamp ASC
|
|
`)
|
|
if result.Error != nil {
|
|
return 0, fmt.Errorf("failed to migrate data: %w", result.Error)
|
|
}
|
|
|
|
return result.RowsAffected, nil
|
|
}
|