mirror of
https://github.com/laoxong/nofx.git
synced 2026-06-04 09:58:22 +08:00
7e96c5d0f2
* feat: add AI grid trading and market regime classification - Add GridTrader interface with PlaceLimitOrder, CancelOrder, GetOrderBook - Implement GridTrader for all exchanges (Binance, Bybit, OKX, Bitget, Hyperliquid, Aster, Lighter) - Add grid engine with ATR-based boundary calculation and fund distribution - Add market regime classification documents (Chinese/English) - Add GridConfigEditor component for frontend configuration * fix: implement GetOpenOrders for Lighter exchange * debug: add logging for Lighter GetActiveOrders API call * fix: correct Lighter API response parsing for GetOpenOrders - Changed response field from 'data' to 'orders' to match Lighter API - Updated OrderResponse struct to match Lighter's actual field names - Fixed field types: price/quantity as strings, is_ask for side * feat: implement GetOpenOrders for Aster, OKX, Bitget exchanges - Aster: uses /fapi/v3/openOrders endpoint - OKX: uses /api/v5/trade/orders-pending and orders-algo-pending - Bitget: uses /api/v2/mix/order/orders-pending and orders-plan-pending * fix: address code review issues for GetOpenOrders - Add error logging for OKX/Bitget API failures (was silently swallowed) - Fix Lighter position side logic to handle reduce-only orders - Change verbose debug logs from Infof to Debugf level * fix: provide FromAccountIndex and ApiKeyIndex for Lighter nonce auto-fetch Root cause: SDK requires these fields to fetch nonce from API, otherwise nonce gets cached/stuck * fix: use auth query parameter instead of Authorization header for Lighter API * test: add Lighter API authentication tests and diagnostic tools * fix(grid): add leverage setting before order placement CRITICAL BUG FIX: - Call SetLeverage() in GridTraderAdapter.PlaceLimitOrder() - Set leverage during grid initialization - Log leverage setting results * fix(grid): prevent CancelOrder from canceling all orders CRITICAL BUG FIX: - CancelOrder no longer calls CancelAllOrders - Try exchange-specific CancelOrder if available - Return error if individual cancellation not supported * fix(grid): add total position value limit check CRITICAL: Prevent excessive position accumulation - New checkTotalPositionLimit() function - Checks current + pending + new order value - Rejects orders that would exceed TotalInvestment x Leverage - Logs clear error messages when limit exceeded * feat(grid): implement stop loss execution CRITICAL: Add code-level stop loss protection - New checkAndExecuteStopLoss() function - Checks each filled level against StopLossPct - Automatically closes positions exceeding stop loss - Called during every grid state sync * feat(grid): add breakout detection and auto-pause CRITICAL: Detect price breakout from grid range - New checkBreakout() function to detect upper/lower breakouts - Auto-pause grid on significant breakout (>2%) - Cancel all orders when breakout detected - Prevent continued losses in trending market - Minor breakouts (1-2%) logged for AI consideration * feat(grid): enforce max drawdown limit with emergency exit CRITICAL: Add drawdown protection - New checkMaxDrawdown() function tracks peak equity - emergencyExit() closes all positions and cancels orders - Auto-pause grid when MaxDrawdownPct exceeded - Protect capital from excessive losses * feat(grid): enforce daily loss limit - Add checkDailyLossLimit() function to check if daily loss exceeds limit - Track daily PnL with auto-reset at midnight - Pause grid when DailyLossLimitPct exceeded - Add updateDailyPnL() helper for realized PnL tracking - Prevent excessive single-day losses * fix(grid): update daily PnL when stop loss is executed The updateDailyPnL() function was added but never called, leaving DailyPnL always at 0 and preventing daily loss limit checks from triggering. This fix updates DailyPnL and TotalProfit directly in checkAndExecuteStopLoss() when a stop loss is executed. We update directly rather than calling updateDailyPnL() because the mutex is already held in that function. * feat(grid): add automatic grid adjustment - New checkGridSkew() detects imbalanced grid - autoAdjustGrid() reinitializes around current price - Prevents grid from becoming ineffective after drift - Triggers when one side is 3x more filled than other * fix(grid): recalculate bounds in autoAdjustGrid before reinitializing levels Critical fix for grid auto-adjustment: - Recalculate grid bounds (UpperPrice, LowerPrice, GridSpacing) centered on current price before reinitializing grid levels - Preserve filled positions during adjustment by saving and restoring them to the closest new level after reinitialization - Hold mutex lock for the entire adjustment operation to ensure atomicity - Add locked variants of calculateDefaultBounds, calculateATRBounds, and initializeGridLevels to use during adjustment Without this fix, autoAdjustGrid was using old boundaries when creating new grid levels, defeating the purpose of auto-adjustment when price moved significantly. * fix(grid): improve order state sync logic - Don't assume missing orders are filled - Compare position size to determine fill vs cancel - Properly reset cancelled orders to empty state - More accurate grid state tracking * fix(grid): use actual PositionSize sum instead of count in syncGridState heuristic The position-based heuristic was using `float64(previousFilledCount) * level.OrderQuantity` which incorrectly assumed uniform order quantities. Since the grid uses weighted distribution (gaussian, pyramid, uniform) where orders have different quantities, this could lead to incorrect fill detection. Now sums the actual PositionSize from filled levels for accurate comparison. Also adds warning log when GetPositions() fails. * docs: add grid market regime detection design Design for enhanced market state recognition with: - Multi-dimensional indicators (ATR, Bollinger, EMA, MACD, RSI) - Multi-period box indicators (72/240/500 1h candles) - 4-level ranging classification - Breakout detection and handling - Frontend risk control panel * docs: add grid market regime implementation plan 20 tasks covering: - Donchian channel calculation - Box data types and API - Regime classification (4 levels) - Breakout detection and handling - False breakout recovery - Frontend risk panel - AI prompt updates * feat(market): add Donchian channel calculation Add calculateDonchian function to compute highest high and lowest low over a specified period. This is the foundation for box (range) detection in the multi-period box indicator system for grid trading. * fix(market): handle invalid period in calculateDonchian * feat(market): add BoxData and RegimeLevel types * feat(market): add GetBoxData for multi-period box calculation Adds calculateBoxData internal function and GetBoxData public API that fetches 1h klines and computes three Donchian box levels (short/mid/long). This will be used by the grid trading system to detect market regime. * feat(store): add box and regime fields to grid models * feat(trader): add regime classification and breakout detection Implements Tasks 6-9 for grid market regime awareness: - Task 6: classifyRegimeLevel with Bollinger/ATR thresholds - Task 7: detectBoxBreakout for multi-period box breakouts - Task 8: confirmBreakout with 3-candle confirmation logic - Task 9: getBreakoutAction mapping breakout levels to actions * feat(trader): integrate box breakout detection into grid cycle - Task 10: Add checkBoxBreakout with 3-candle confirmation - Task 11: Add checkFalseBreakoutRecovery for 50% position recovery - Task 12: Add box/breakout/regime fields to GridState * feat: add grid risk panel with API endpoint - Task 13: Add GridRiskInfo type to frontend - Task 14: Add /traders/:id/grid-risk API endpoint - Task 15: Add GetGridRiskInfo method to AutoTrader - Task 16: Create GridRiskPanel component with i18n * feat(kernel): add box indicators to AI prompt - Add BoxData field to GridContext - Add box indicator table to both zh/en prompts - Show breakout/warning alerts based on price position * feat(web): integrate GridRiskPanel into TraderDashboardPage * feat(lighter): improve API key validation and market caching - Add API key validation status tracking - Add market list caching to reduce API calls - Improve logging (debug vs info levels) - Add comprehensive integration tests - Update trader manager and store for lighter support * fix: remove hardcoded test wallet address * fix(grid): improve GridRiskPanel layout and fix liquidation data - Make panel collapsible with summary badges when collapsed - Use compact 2-column grid layout for detailed info - Fix auth token key (token -> auth_token) - Only calculate liquidation distance when position exists * fix(grid): add isRunning checks to prevent trades after Stop() is called
231 lines
9.1 KiB
Go
231 lines
9.1 KiB
Go
package trader
|
|
|
|
import (
|
|
"fmt"
|
|
"nofx/logger"
|
|
"time"
|
|
)
|
|
|
|
// ClosedPnLRecord represents a single closed position record from exchange
|
|
type ClosedPnLRecord struct {
|
|
Symbol string // Trading pair (e.g., "BTCUSDT")
|
|
Side string // "long" or "short"
|
|
EntryPrice float64 // Entry price
|
|
ExitPrice float64 // Exit/close price
|
|
Quantity float64 // Position size
|
|
RealizedPnL float64 // Realized profit/loss
|
|
Fee float64 // Trading fee/commission
|
|
Leverage int // Leverage used
|
|
EntryTime time.Time // Position open time
|
|
ExitTime time.Time // Position close time
|
|
OrderID string // Close order ID
|
|
CloseType string // "manual", "stop_loss", "take_profit", "liquidation", "unknown"
|
|
ExchangeID string // Exchange-specific position ID
|
|
}
|
|
|
|
// TradeRecord represents a single trade/fill from exchange
|
|
// Used for reconstructing position history with unified algorithm
|
|
type TradeRecord struct {
|
|
TradeID string // Unique trade ID from exchange
|
|
Symbol string // Trading pair (e.g., "BTCUSDT")
|
|
Side string // "BUY" or "SELL"
|
|
PositionSide string // "LONG", "SHORT", or "BOTH" (for one-way mode)
|
|
OrderAction string // "open_long", "open_short", "close_long", "close_short" (from exchange Dir field)
|
|
Price float64 // Execution price
|
|
Quantity float64 // Executed quantity
|
|
RealizedPnL float64 // Realized PnL (non-zero for closing trades)
|
|
Fee float64 // Trading fee/commission
|
|
Time time.Time // Trade execution time
|
|
}
|
|
|
|
// Trader Unified trader interface
|
|
// Supports multiple trading platforms (Binance, Hyperliquid, etc.)
|
|
type Trader interface {
|
|
// GetBalance Get account balance
|
|
GetBalance() (map[string]interface{}, error)
|
|
|
|
// GetPositions Get all positions
|
|
GetPositions() ([]map[string]interface{}, error)
|
|
|
|
// OpenLong Open long position
|
|
OpenLong(symbol string, quantity float64, leverage int) (map[string]interface{}, error)
|
|
|
|
// OpenShort Open short position
|
|
OpenShort(symbol string, quantity float64, leverage int) (map[string]interface{}, error)
|
|
|
|
// CloseLong Close long position (quantity=0 means close all)
|
|
CloseLong(symbol string, quantity float64) (map[string]interface{}, error)
|
|
|
|
// CloseShort Close short position (quantity=0 means close all)
|
|
CloseShort(symbol string, quantity float64) (map[string]interface{}, error)
|
|
|
|
// SetLeverage Set leverage
|
|
SetLeverage(symbol string, leverage int) error
|
|
|
|
// SetMarginMode Set position mode (true=cross margin, false=isolated margin)
|
|
SetMarginMode(symbol string, isCrossMargin bool) error
|
|
|
|
// GetMarketPrice Get market price
|
|
GetMarketPrice(symbol string) (float64, error)
|
|
|
|
// SetStopLoss Set stop-loss order
|
|
SetStopLoss(symbol string, positionSide string, quantity, stopPrice float64) error
|
|
|
|
// SetTakeProfit Set take-profit order
|
|
SetTakeProfit(symbol string, positionSide string, quantity, takeProfitPrice float64) error
|
|
|
|
// CancelStopLossOrders Cancel only stop-loss orders (BUG fix: don't delete take-profit when adjusting stop-loss)
|
|
CancelStopLossOrders(symbol string) error
|
|
|
|
// CancelTakeProfitOrders Cancel only take-profit orders (BUG fix: don't delete stop-loss when adjusting take-profit)
|
|
CancelTakeProfitOrders(symbol string) error
|
|
|
|
// CancelAllOrders Cancel all pending orders for this symbol
|
|
CancelAllOrders(symbol string) error
|
|
|
|
// CancelStopOrders Cancel stop-loss/take-profit orders for this symbol (for adjusting stop-loss/take-profit positions)
|
|
CancelStopOrders(symbol string) error
|
|
|
|
// FormatQuantity Format quantity to correct precision
|
|
FormatQuantity(symbol string, quantity float64) (string, error)
|
|
|
|
// GetOrderStatus Get order status
|
|
// Returns: status(FILLED/NEW/CANCELED), avgPrice, executedQty, commission
|
|
GetOrderStatus(symbol string, orderID string) (map[string]interface{}, error)
|
|
|
|
// GetClosedPnL Get closed position PnL records from exchange
|
|
// startTime: start time for query (usually last sync time)
|
|
// limit: max number of records to return
|
|
// Returns accurate exit price, fees, and close reason for positions closed externally
|
|
GetClosedPnL(startTime time.Time, limit int) ([]ClosedPnLRecord, error)
|
|
|
|
// GetOpenOrders Get open/pending orders from exchange
|
|
// Returns stop-loss, take-profit, and limit orders that haven't been filled
|
|
GetOpenOrders(symbol string) ([]OpenOrder, error)
|
|
}
|
|
|
|
// OpenOrder represents a pending order on the exchange
|
|
type OpenOrder struct {
|
|
OrderID string `json:"order_id"`
|
|
Symbol string `json:"symbol"`
|
|
Side string `json:"side"` // BUY/SELL
|
|
PositionSide string `json:"position_side"` // LONG/SHORT
|
|
Type string `json:"type"` // LIMIT/STOP_MARKET/TAKE_PROFIT_MARKET
|
|
Price float64 `json:"price"` // Order price (for limit orders)
|
|
StopPrice float64 `json:"stop_price"` // Trigger price (for stop orders)
|
|
Quantity float64 `json:"quantity"`
|
|
Status string `json:"status"` // NEW
|
|
}
|
|
|
|
// LimitOrderRequest represents a limit order request for grid trading
|
|
type LimitOrderRequest struct {
|
|
Symbol string `json:"symbol"`
|
|
Side string `json:"side"` // BUY/SELL
|
|
PositionSide string `json:"position_side"` // LONG/SHORT (for hedge mode)
|
|
Price float64 `json:"price"` // Limit price
|
|
Quantity float64 `json:"quantity"`
|
|
Leverage int `json:"leverage"`
|
|
PostOnly bool `json:"post_only"` // Maker only order
|
|
ReduceOnly bool `json:"reduce_only"` // Reduce position only
|
|
ClientID string `json:"client_id"` // Client order ID for tracking
|
|
}
|
|
|
|
// LimitOrderResult represents the result of placing a limit order
|
|
type LimitOrderResult struct {
|
|
OrderID string `json:"order_id"`
|
|
ClientID string `json:"client_id"`
|
|
Symbol string `json:"symbol"`
|
|
Side string `json:"side"`
|
|
PositionSide string `json:"position_side"`
|
|
Price float64 `json:"price"`
|
|
Quantity float64 `json:"quantity"`
|
|
Status string `json:"status"` // NEW, PARTIALLY_FILLED, FILLED, CANCELED
|
|
}
|
|
|
|
// GridTrader extends Trader interface with limit order support for grid trading
|
|
// Exchanges that support grid trading should implement this interface
|
|
type GridTrader interface {
|
|
Trader
|
|
|
|
// PlaceLimitOrder places a limit order at specified price
|
|
// Returns order ID and status
|
|
PlaceLimitOrder(req *LimitOrderRequest) (*LimitOrderResult, error)
|
|
|
|
// CancelOrder cancels a specific order by ID
|
|
CancelOrder(symbol, orderID string) error
|
|
|
|
// GetOrderBook gets current order book (for price validation)
|
|
// Returns best bid/ask prices
|
|
GetOrderBook(symbol string, depth int) (bids, asks [][]float64, err error)
|
|
}
|
|
|
|
// GridTraderAdapter wraps a basic Trader to provide GridTrader interface
|
|
// Uses stop orders as a fallback when limit orders aren't directly available
|
|
type GridTraderAdapter struct {
|
|
Trader
|
|
}
|
|
|
|
// NewGridTraderAdapter creates an adapter for basic Trader
|
|
func NewGridTraderAdapter(t Trader) *GridTraderAdapter {
|
|
return &GridTraderAdapter{Trader: t}
|
|
}
|
|
|
|
// PlaceLimitOrder implements limit order using available methods
|
|
// For exchanges without native limit order support, this uses conditional orders
|
|
func (a *GridTraderAdapter) PlaceLimitOrder(req *LimitOrderRequest) (*LimitOrderResult, error) {
|
|
// CRITICAL FIX: Set leverage before placing order
|
|
if req.Leverage > 0 {
|
|
if err := a.Trader.SetLeverage(req.Symbol, req.Leverage); err != nil {
|
|
logger.Warnf("[Grid] Failed to set leverage %dx: %v", req.Leverage, err)
|
|
// Continue anyway - some exchanges don't require explicit leverage setting
|
|
}
|
|
}
|
|
|
|
// Use SetStopLoss/SetTakeProfit as conditional limit orders
|
|
// For buy orders below current price, use stop-loss mechanism
|
|
// For sell orders above current price, use take-profit mechanism
|
|
var err error
|
|
if req.Side == "BUY" {
|
|
err = a.Trader.SetStopLoss(req.Symbol, "SHORT", req.Quantity, req.Price)
|
|
} else {
|
|
err = a.Trader.SetTakeProfit(req.Symbol, "LONG", req.Quantity, req.Price)
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &LimitOrderResult{
|
|
OrderID: req.ClientID,
|
|
ClientID: req.ClientID,
|
|
Symbol: req.Symbol,
|
|
Side: req.Side,
|
|
PositionSide: req.PositionSide,
|
|
Price: req.Price,
|
|
Quantity: req.Quantity,
|
|
Status: "NEW",
|
|
}, nil
|
|
}
|
|
|
|
// CancelOrder cancels a specific order
|
|
func (a *GridTraderAdapter) CancelOrder(symbol, orderID string) error {
|
|
// Try to use CancelOrder if trader supports it directly
|
|
if canceler, ok := a.Trader.(interface {
|
|
CancelOrder(symbol, orderID string) error
|
|
}); ok {
|
|
return canceler.CancelOrder(symbol, orderID)
|
|
}
|
|
|
|
// For traders that only support CancelAllOrders, log a warning
|
|
// This is a limitation - we cannot cancel individual orders
|
|
logger.Warnf("[Grid] Trader does not support individual order cancellation, "+
|
|
"cannot cancel order %s. Consider using exchange-specific GridTrader implementation.", orderID)
|
|
|
|
// Return error instead of canceling all orders
|
|
return fmt.Errorf("individual order cancellation not supported for this exchange")
|
|
}
|
|
|
|
// GetOrderBook returns empty order book (not supported in basic Trader)
|
|
func (a *GridTraderAdapter) GetOrderBook(symbol string, depth int) (bids, asks [][]float64, err error) {
|
|
// Not supported, return empty
|
|
return nil, nil, nil
|
|
}
|