package trader import ( "bytes" "encoding/json" "fmt" "io" "mime/multipart" "net/http" "nofx/logger" "strconv" "github.com/elliottech/lighter-go/types" ) // SetStopLoss Set stop-loss order (implements Trader interface) // IMPORTANT: Uses StopLossOrder type (type=2) with TriggerPrice, NOT regular limit order func (t *LighterTraderV2) SetStopLoss(symbol string, positionSide string, quantity, stopPrice float64) error { if t.txClient == nil { return fmt.Errorf("TxClient not initialized") } logger.Infof("🛑 LIGHTER Setting stop-loss: %s %s qty=%.4f, trigger=%.2f", symbol, positionSide, quantity, stopPrice) // Determine order direction (long position uses sell order, short position uses buy order) isAsk := (positionSide == "LONG" || positionSide == "long") // Create stop-loss order with TriggerPrice (type=2: StopLossOrder) _, err := t.CreateStopOrder(symbol, isAsk, quantity, stopPrice, "stop_loss") if err != nil { return fmt.Errorf("failed to set stop-loss: %w", err) } logger.Infof("✓ LIGHTER stop-loss set: trigger=%.2f", stopPrice) return nil } // SetTakeProfit Set take-profit order (implements Trader interface) // IMPORTANT: Uses TakeProfitOrder type (type=4) with TriggerPrice, NOT regular limit order func (t *LighterTraderV2) SetTakeProfit(symbol string, positionSide string, quantity, takeProfitPrice float64) error { if t.txClient == nil { return fmt.Errorf("TxClient not initialized") } logger.Infof("🎯 LIGHTER Setting take-profit: %s %s qty=%.4f, trigger=%.2f", symbol, positionSide, quantity, takeProfitPrice) // Determine order direction (long position uses sell order, short position uses buy order) isAsk := (positionSide == "LONG" || positionSide == "long") // Create take-profit order with TriggerPrice (type=4: TakeProfitOrder) _, err := t.CreateStopOrder(symbol, isAsk, quantity, takeProfitPrice, "take_profit") if err != nil { return fmt.Errorf("failed to set take-profit: %w", err) } logger.Infof("✓ LIGHTER take-profit set: trigger=%.2f", takeProfitPrice) return nil } // CancelAllOrders Cancel all orders (implements Trader interface) func (t *LighterTraderV2) CancelAllOrders(symbol string) error { if t.txClient == nil { return fmt.Errorf("TxClient not initialized") } if err := t.ensureAuthToken(); err != nil { return fmt.Errorf("invalid auth token: %w", err) } // Get all active orders orders, err := t.GetActiveOrders(symbol) if err != nil { return fmt.Errorf("failed to get active orders: %w", err) } if len(orders) == 0 { logger.Infof("✓ LIGHTER - No orders to cancel (no active orders)") return nil } // Batch cancel canceledCount := 0 for _, order := range orders { if err := t.CancelOrder(symbol, order.OrderID); err != nil { logger.Infof("⚠️ Failed to cancel order (ID: %s): %v", order.OrderID, err) } else { canceledCount++ } } logger.Infof("✓ LIGHTER - Canceled %d orders", canceledCount) return nil } // GetOrderStatus Get order status (implements Trader interface) func (t *LighterTraderV2) GetOrderStatus(symbol string, orderID string) (map[string]interface{}, error) { // LIGHTER market orders are usually filled immediately // Try to query order status if err := t.ensureAuthToken(); err != nil { return nil, fmt.Errorf("invalid auth token: %w", err) } // Build request URL endpoint := fmt.Sprintf("%s/api/v1/order/%s", t.baseURL, orderID) req, err := http.NewRequest("GET", endpoint, nil) if err != nil { return nil, err } req.Header.Set("Authorization", t.authToken) req.Header.Set("Content-Type", "application/json") resp, err := t.client.Do(req) if err != nil { // ✅ 正确做法:查询失败返回错误,而不是假设成交 return nil, fmt.Errorf("failed to query order status: %w", err) } defer resp.Body.Close() body, err := io.ReadAll(resp.Body) if err != nil { return nil, fmt.Errorf("failed to read response body: %w", err) } // Check HTTP status code if resp.StatusCode != 200 { return nil, fmt.Errorf("API returned status %d: %s", resp.StatusCode, string(body)) } var order OrderResponse if err := json.Unmarshal(body, &order); err != nil { return nil, fmt.Errorf("failed to parse order response: %w, body: %s", err, string(body)) } // Convert status to unified format unifiedStatus := order.Status switch order.Status { case "filled": unifiedStatus = "FILLED" case "open": unifiedStatus = "NEW" case "cancelled": unifiedStatus = "CANCELED" } return map[string]interface{}{ "orderId": order.OrderID, "status": unifiedStatus, "avgPrice": order.Price, "executedQty": order.FilledQty, "commission": 0.0, }, nil } // CancelStopLossOrders Cancel only stop-loss orders (implements Trader interface) func (t *LighterTraderV2) CancelStopLossOrders(symbol string) error { // LIGHTER cannot distinguish between stop-loss and take-profit orders yet, will cancel all stop orders logger.Infof("⚠️ LIGHTER cannot distinguish stop-loss/take-profit orders, will cancel all stop orders") return t.CancelStopOrders(symbol) } // CancelTakeProfitOrders Cancel only take-profit orders (implements Trader interface) func (t *LighterTraderV2) CancelTakeProfitOrders(symbol string) error { // LIGHTER cannot distinguish between stop-loss and take-profit orders yet, will cancel all stop orders logger.Infof("⚠️ LIGHTER cannot distinguish stop-loss/take-profit orders, will cancel all stop orders") return t.CancelStopOrders(symbol) } // CancelStopOrders Cancel stop-loss/take-profit orders for this symbol (implements Trader interface) func (t *LighterTraderV2) CancelStopOrders(symbol string) error { if t.txClient == nil { return fmt.Errorf("TxClient not initialized") } if err := t.ensureAuthToken(); err != nil { return fmt.Errorf("invalid auth token: %w", err) } // Get active orders orders, err := t.GetActiveOrders(symbol) if err != nil { return fmt.Errorf("failed to get active orders: %w", err) } canceledCount := 0 for _, order := range orders { // TODO: Check order type, only cancel stop orders // For now, cancel all orders if err := t.CancelOrder(symbol, order.OrderID); err != nil { logger.Infof("⚠️ Failed to cancel order (ID: %s): %v", order.OrderID, err) } else { canceledCount++ } } logger.Infof("✓ LIGHTER - Canceled %d stop orders", canceledCount) return nil } // GetActiveOrders Get active orders func (t *LighterTraderV2) GetActiveOrders(symbol string) ([]OrderResponse, error) { if err := t.ensureAuthToken(); err != nil { return nil, fmt.Errorf("invalid auth token: %w", err) } // Get market index marketIndex, err := t.getMarketIndex(symbol) if err != nil { return nil, fmt.Errorf("failed to get market index: %w", err) } // Build request URL endpoint := fmt.Sprintf("%s/api/v1/accountActiveOrders?account_index=%d&market_id=%d", t.baseURL, t.accountIndex, marketIndex) // Send GET request req, err := http.NewRequest("GET", endpoint, nil) if err != nil { return nil, fmt.Errorf("failed to create request: %w", err) } // Add authentication header req.Header.Set("Authorization", t.authToken) req.Header.Set("Content-Type", "application/json") resp, err := t.client.Do(req) if err != nil { return nil, fmt.Errorf("request failed: %w", err) } defer resp.Body.Close() body, err := io.ReadAll(resp.Body) if err != nil { return nil, fmt.Errorf("failed to read response: %w", err) } // Parse response var apiResp struct { Code int `json:"code"` Message string `json:"message"` Data []OrderResponse `json:"data"` } if err := json.Unmarshal(body, &apiResp); err != nil { return nil, fmt.Errorf("failed to parse response: %w, body: %s", err, string(body)) } if apiResp.Code != 200 { return nil, fmt.Errorf("failed to get active orders (code %d): %s", apiResp.Code, apiResp.Message) } logger.Infof("✓ LIGHTER - Retrieved %d active orders", len(apiResp.Data)) return apiResp.Data, nil } // CancelOrder Cancel a single order func (t *LighterTraderV2) CancelOrder(symbol, orderID string) error { if t.txClient == nil { return fmt.Errorf("TxClient not initialized") } // Get market index marketIndexU16, err := t.getMarketIndex(symbol) if err != nil { return fmt.Errorf("failed to get market index: %w", err) } marketIndex := uint8(marketIndexU16) // SDK expects uint8 // Convert orderID to int64 orderIndex, err := strconv.ParseInt(orderID, 10, 64) if err != nil { return fmt.Errorf("invalid order ID: %w", err) } // Build cancel order request txReq := &types.CancelOrderTxReq{ MarketIndex: marketIndex, Index: orderIndex, } // Sign transaction using SDK nonce := int64(-1) // -1 means auto-fetch tx, err := t.txClient.GetCancelOrderTransaction(txReq, &types.TransactOpts{ Nonce: &nonce, }) if err != nil { return fmt.Errorf("failed to sign cancel order: %w", err) } // Serialize transaction txBytes, err := json.Marshal(tx) if err != nil { return fmt.Errorf("failed to serialize transaction: %w", err) } // Submit cancel order to LIGHTER API _, err = t.submitCancelOrder(txBytes) if err != nil { return fmt.Errorf("failed to submit cancel order: %w", err) } logger.Infof("✓ LIGHTER order canceled - ID: %s", orderID) return nil } // submitCancelOrder Submit signed cancel order to LIGHTER API using multipart/form-data func (t *LighterTraderV2) submitCancelOrder(signedTx []byte) (map[string]interface{}, error) { const TX_TYPE_CANCEL_ORDER = 15 // Build multipart form data (Lighter API requires form-data, not JSON) var body bytes.Buffer writer := multipart.NewWriter(&body) // Add tx_type field if err := writer.WriteField("tx_type", strconv.Itoa(TX_TYPE_CANCEL_ORDER)); err != nil { return nil, fmt.Errorf("failed to write tx_type: %w", err) } // Add tx_info field if err := writer.WriteField("tx_info", string(signedTx)); err != nil { return nil, fmt.Errorf("failed to write tx_info: %w", err) } // Close multipart writer if err := writer.Close(); err != nil { return nil, fmt.Errorf("failed to close multipart writer: %w", err) } // Send POST request to /api/v1/sendTx endpoint := fmt.Sprintf("%s/api/v1/sendTx", t.baseURL) httpReq, err := http.NewRequest("POST", endpoint, &body) if err != nil { return nil, err } httpReq.Header.Set("Content-Type", writer.FormDataContentType()) resp, err := t.client.Do(httpReq) if err != nil { return nil, err } defer resp.Body.Close() respBody, err := io.ReadAll(resp.Body) if err != nil { return nil, err } // Parse response var sendResp SendTxResponse if err := json.Unmarshal(respBody, &sendResp); err != nil { return nil, fmt.Errorf("failed to parse response: %w, body: %s", err, string(respBody)) } // Check response code if sendResp.Code != 200 { return nil, fmt.Errorf("failed to submit cancel order (code %d): %s", sendResp.Code, sendResp.Message) } result := map[string]interface{}{ "tx_hash": sendResp.Data["tx_hash"], "status": "cancelled", } logger.Infof("✓ Cancel order submitted to LIGHTER - tx_hash: %v", sendResp.Data["tx_hash"]) return result, nil }