mirror of
https://github.com/laoxong/nofx.git
synced 2026-06-04 01:48:22 +08:00
7ae5bf8247
* feat(store): prevent deletion of active strategies and update translations (#1461) Co-authored-by: Dean <afei.wuhao@gmail.com> * fix: allow model switching without re-entering wallet key Users with existing wallets could not switch AI models because the "Start Trading" button required a valid private key even when one was already configured. Now the button is enabled when hasExistingWallet is true, and handleSubmit passes an empty key so the backend preserves the existing key. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * refactor: replace window.location with useNavigate for routing in auth components (#1470) Co-authored-by: Dean <afei.wuhao@gmail.com> * feat(trader): implement margin mode handling for order and leverage settings * refactor(trader): update SetMarginMode to avoid legacy endpoint and improve logging * feat(api): enhance strategy handling by integrating claw402 wallet key validation Added validation for the claw402 model's wallet key during strategy test runs. If the selected AI model is claw402, the server now checks for a valid wallet key and returns appropriate error messages if it's missing or if the model fails to load. This ensures better error handling and user feedback when working with AI models. * refactor(api): streamline claw402 wallet key retrieval and error handling Refactored the strategy handling logic to encapsulate claw402 wallet key retrieval in a new method, `resolveStrategyDataWalletKey`. This improves code readability and maintains consistent error handling for missing or invalid wallet keys during strategy test runs. The changes enhance the overall robustness of the AI model integration. * feat(trader): add claw402 wallet key resolution for trader configuration Implemented a new method, `resolveTraderDataWalletKey`, to retrieve the claw402 wallet key based on the selected AI model and user ID. This enhancement allows for better integration of the claw402 model within the trader configuration, ensuring that the correct wallet key is used for trading operations. The `AutoTraderConfig` struct has been updated to include the new `Claw402WalletKey` field, improving the overall handling of wallet keys in the trading process. * feat(claw402): preflight USDC balance before AI calls (#1479) * chore: ignore nofx-server build artifact Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * feat(claw402): preflight USDC balance before AI calls Short-circuit claw402 Call/CallWithRequestFull when the wallet balance can't cover the estimated cost of the call, surfacing ErrInsufficientFunds instead of letting x402 fail mid-flight after the sign step. - wallet: cached balance lookup (30s TTL, per-address mutex) to avoid hammering the Base RPC; separate error-returning and display-only APIs so callers can distinguish zero balance from an unreachable RPC. - claw402: 1.5× safety multiplier on the flat per-call estimate, 4.0× for reasoner models whose chain-of-thought cost can blow past the flat rate. Fail-open on RPC errors — x402 still gates actually-empty wallets, and we prefer availability over extra strictness. - shortAddr redacts the wallet in error strings to avoid leaking the full address into telemetry bundles. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com> * fix(telemetry): report token usage for SSE streaming paths (#1475) * fix(telemetry): report token usage for SSE streaming paths ParseSSEStream already parsed the usage block from SSE chunks but only printed it, so claw402 streaming calls (and native streaming) never fired TokenUsageCallback. GA4 therefore undercounted AI usage on the streaming path. Return the parsed usage from ParseSSEStream and have both callers fire the callback with their own Provider/Model. * chore: drop leftover debug Printf in ParseSSEStream Telemetry is now wired via TokenUsageCallback, so the Printf is redundant noise in the stream path. * fix(gemini): update default model to gemini-3.1-pro Google discontinued gemini-3-pro-preview on 2026-03-26 and directs all callers to gemini-3.1-pro / gemini-3.1-pro-preview. Users on their own API key were getting errors from the native Gemini endpoint because the provider default pointed at the retired ID. Claw402 was unaffected because its route map already used gemini-3.1-pro. Align both the native provider default and the handler's preset list with gemini-3.1-pro so every code path sends a live model ID. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * refactor: extract ResolveClaw402WalletKey to store layer and expand OKX margin mode tests - Move duplicated claw402 wallet resolution logic into store.AIModelStore.ResolveClaw402WalletKey - api/strategy.go and manager/trader_manager.go now delegate to the shared method - Add detailed doc comment on OKX SetMarginMode explaining the local-state-only approach and why the legacy /api/v5/account/set-isolated-mode endpoint is not called - Add 3 new test cases: cross mode leverage, OpenShort tdMode, SetTakeProfit tdMode * fix(auth): prevent SetupPage remount from wiping freshly-set auth token (#1481) After #1470 moved routing into react-router, SetupPage is rendered at two different tree positions (top-level guard + /setup Route). When register success flushSync-sets `user`, the top-level guard stops matching and the Route-level SetupPage mounts as a new instance, re-running its cleanup useEffect and removing the auth_token that handlePostAuthSuccess just wrote. Subsequent requests 401 and bounce the user back to /login. Redirect /setup to /welcome when user is already set so SetupPage is never re-mounted during the auth transition. * fix(wallet): handle JSON-RPC null error field in balance query Some RPC implementations return explicit "error": null on success. json.RawMessage deserializes this as the 4-byte literal "null", so len() > 0 was true, causing every balance query to fail with "rpc error: null". Skip the null literal to avoid false positives. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> --------- Co-authored-by: deanokk <wuhao@vergex.trade> Co-authored-by: Dean <afei.wuhao@gmail.com> Co-authored-by: Claude Opus 4.6 (1M context) <noreply@anthropic.com> Co-authored-by: root <root@localhost.localdomain>
310 lines
8.6 KiB
Go
310 lines
8.6 KiB
Go
package okx
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/hmac"
|
|
"crypto/rand"
|
|
"crypto/sha256"
|
|
"encoding/base64"
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"nofx/logger"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
// OKX API endpoints
|
|
const (
|
|
okxBaseURL = "https://www.okx.com"
|
|
okxAccountPath = "/api/v5/account/balance"
|
|
okxPositionPath = "/api/v5/account/positions"
|
|
okxOrderPath = "/api/v5/trade/order"
|
|
okxLeveragePath = "/api/v5/account/set-leverage"
|
|
okxTickerPath = "/api/v5/market/ticker"
|
|
okxInstrumentsPath = "/api/v5/public/instruments"
|
|
okxCancelOrderPath = "/api/v5/trade/cancel-order"
|
|
okxPendingOrdersPath = "/api/v5/trade/orders-pending"
|
|
okxAlgoOrderPath = "/api/v5/trade/order-algo"
|
|
okxCancelAlgoPath = "/api/v5/trade/cancel-algos"
|
|
okxAlgoPendingPath = "/api/v5/trade/orders-algo-pending"
|
|
okxPositionModePath = "/api/v5/account/set-position-mode"
|
|
okxAccountConfigPath = "/api/v5/account/config"
|
|
)
|
|
|
|
// OKXTrader OKX futures trader
|
|
type OKXTrader struct {
|
|
apiKey string
|
|
secretKey string
|
|
passphrase string
|
|
|
|
// Margin mode setting used for new orders and leverage changes.
|
|
isCrossMargin bool
|
|
|
|
// Position mode: "long_short_mode" (hedge) or "net_mode" (one-way)
|
|
positionMode string
|
|
|
|
// HTTP client (proxy disabled)
|
|
httpClient *http.Client
|
|
|
|
// Balance cache
|
|
cachedBalance map[string]interface{}
|
|
balanceCacheTime time.Time
|
|
balanceCacheMutex sync.RWMutex
|
|
|
|
// Positions cache
|
|
cachedPositions []map[string]interface{}
|
|
positionsCacheTime time.Time
|
|
positionsCacheMutex sync.RWMutex
|
|
|
|
// Instrument info cache
|
|
instrumentsCache map[string]*OKXInstrument
|
|
instrumentsCacheTime time.Time
|
|
instrumentsCacheMutex sync.RWMutex
|
|
|
|
// Cache duration
|
|
cacheDuration time.Duration
|
|
}
|
|
|
|
// OKXInstrument OKX instrument info
|
|
type OKXInstrument struct {
|
|
InstID string // Instrument ID
|
|
CtVal float64 // Contract value
|
|
CtMult float64 // Contract multiplier
|
|
LotSz float64 // Minimum order size
|
|
MinSz float64 // Minimum order size
|
|
MaxMktSz float64 // Maximum market order size
|
|
TickSz float64 // Minimum price increment
|
|
CtType string // Contract type
|
|
}
|
|
|
|
// OKXResponse OKX API response
|
|
type OKXResponse struct {
|
|
Code string `json:"code"`
|
|
Msg string `json:"msg"`
|
|
Data json.RawMessage `json:"data"`
|
|
}
|
|
|
|
// OKX order tag
|
|
var okxTag = func() string {
|
|
b, _ := base64.StdEncoding.DecodeString("NGMzNjNjODFlZGM1QkNERQ==")
|
|
return string(b)
|
|
}()
|
|
|
|
// genOkxClOrdID generates OKX order ID
|
|
func genOkxClOrdID() string {
|
|
timestamp := time.Now().UnixNano() % 10000000000000
|
|
randomBytes := make([]byte, 4)
|
|
rand.Read(randomBytes)
|
|
randomHex := hex.EncodeToString(randomBytes)
|
|
// OKX clOrdId max 32 characters
|
|
orderID := fmt.Sprintf("%s%d%s", okxTag, timestamp, randomHex)
|
|
if len(orderID) > 32 {
|
|
orderID = orderID[:32]
|
|
}
|
|
return orderID
|
|
}
|
|
|
|
// NewOKXTrader creates OKX trader
|
|
func NewOKXTrader(apiKey, secretKey, passphrase string) *OKXTrader {
|
|
// Use default transport which respects system proxy settings
|
|
// OKX requires proxy in China due to DNS pollution
|
|
httpClient := &http.Client{
|
|
Timeout: 30 * time.Second,
|
|
Transport: http.DefaultTransport,
|
|
}
|
|
|
|
trader := &OKXTrader{
|
|
apiKey: apiKey,
|
|
secretKey: secretKey,
|
|
passphrase: passphrase,
|
|
isCrossMargin: true,
|
|
httpClient: httpClient,
|
|
cacheDuration: 15 * time.Second,
|
|
instrumentsCache: make(map[string]*OKXInstrument),
|
|
}
|
|
|
|
// Get current position mode first
|
|
if err := trader.detectPositionMode(); err != nil {
|
|
logger.Infof("⚠️ Failed to detect OKX position mode: %v, assuming dual mode", err)
|
|
trader.positionMode = "long_short_mode"
|
|
}
|
|
|
|
// Try to set dual position mode (only if not already)
|
|
if trader.positionMode != "long_short_mode" {
|
|
if err := trader.setPositionMode(); err != nil {
|
|
logger.Infof("⚠️ Failed to set OKX position mode: %v (current mode: %s)", err, trader.positionMode)
|
|
}
|
|
}
|
|
|
|
logger.Infof("✓ OKX trader initialized with position mode: %s, default margin mode: %s",
|
|
trader.positionMode, trader.marginMode())
|
|
return trader
|
|
}
|
|
|
|
func (t *OKXTrader) marginMode() string {
|
|
if t.isCrossMargin {
|
|
return "cross"
|
|
}
|
|
return "isolated"
|
|
}
|
|
|
|
// detectPositionMode gets current position mode from account config
|
|
func (t *OKXTrader) detectPositionMode() error {
|
|
data, err := t.doRequest("GET", okxAccountConfigPath, nil)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get account config: %w", err)
|
|
}
|
|
|
|
var configs []struct {
|
|
PosMode string `json:"posMode"`
|
|
}
|
|
|
|
if err := json.Unmarshal(data, &configs); err != nil {
|
|
return fmt.Errorf("failed to parse account config: %w", err)
|
|
}
|
|
|
|
if len(configs) > 0 {
|
|
t.positionMode = configs[0].PosMode
|
|
logger.Infof("✓ Detected OKX position mode: %s", t.positionMode)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// setPositionMode sets dual position mode
|
|
func (t *OKXTrader) setPositionMode() error {
|
|
body := map[string]string{
|
|
"posMode": "long_short_mode", // Dual position mode
|
|
}
|
|
|
|
_, err := t.doRequest("POST", okxPositionModePath, body)
|
|
if err != nil {
|
|
// Ignore error if already in dual position mode
|
|
if strings.Contains(err.Error(), "already") || strings.Contains(err.Error(), "Position mode is not modified") {
|
|
logger.Infof(" ✓ OKX account is already in dual position mode")
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
logger.Infof(" ✓ OKX account switched to dual position mode")
|
|
return nil
|
|
}
|
|
|
|
// sign generates OKX API signature
|
|
func (t *OKXTrader) sign(timestamp, method, requestPath, body string) string {
|
|
preHash := timestamp + method + requestPath + body
|
|
h := hmac.New(sha256.New, []byte(t.secretKey))
|
|
h.Write([]byte(preHash))
|
|
return base64.StdEncoding.EncodeToString(h.Sum(nil))
|
|
}
|
|
|
|
// doRequest executes HTTP request
|
|
func (t *OKXTrader) doRequest(method, path string, body interface{}) ([]byte, error) {
|
|
var bodyBytes []byte
|
|
var err error
|
|
|
|
if body != nil {
|
|
bodyBytes, err = json.Marshal(body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to serialize request body: %w", err)
|
|
}
|
|
}
|
|
|
|
timestamp := time.Now().UTC().Format("2006-01-02T15:04:05.000Z")
|
|
signature := t.sign(timestamp, method, path, string(bodyBytes))
|
|
|
|
req, err := http.NewRequest(method, okxBaseURL+path, bytes.NewReader(bodyBytes))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to create request: %w", err)
|
|
}
|
|
|
|
req.Header.Set("OK-ACCESS-KEY", t.apiKey)
|
|
req.Header.Set("OK-ACCESS-SIGN", signature)
|
|
req.Header.Set("OK-ACCESS-TIMESTAMP", timestamp)
|
|
req.Header.Set("OK-ACCESS-PASSPHRASE", t.passphrase)
|
|
req.Header.Set("Content-Type", "application/json")
|
|
// Set request header
|
|
req.Header.Set("x-simulated-trading", "0")
|
|
|
|
resp, err := t.httpClient.Do(req)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("request failed: %w", err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
respBody, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to read response: %w", err)
|
|
}
|
|
|
|
var okxResp OKXResponse
|
|
if err := json.Unmarshal(respBody, &okxResp); err != nil {
|
|
return nil, fmt.Errorf("failed to parse response: %w", err)
|
|
}
|
|
|
|
// code=1 indicates partial success, need to check specific results in data
|
|
// code=2 indicates complete failure
|
|
if okxResp.Code != "0" && okxResp.Code != "1" {
|
|
return nil, fmt.Errorf("OKX API error: code=%s, msg=%s", okxResp.Code, okxResp.Msg)
|
|
}
|
|
|
|
return okxResp.Data, nil
|
|
}
|
|
|
|
// convertSymbol converts generic symbol to OKX format
|
|
// e.g. BTCUSDT -> BTC-USDT-SWAP
|
|
func (t *OKXTrader) convertSymbol(symbol string) string {
|
|
// Remove USDT suffix and build OKX format
|
|
base := strings.TrimSuffix(symbol, "USDT")
|
|
return fmt.Sprintf("%s-USDT-SWAP", base)
|
|
}
|
|
|
|
// convertSymbolBack converts OKX format back to generic symbol
|
|
// e.g. BTC-USDT-SWAP -> BTCUSDT
|
|
func (t *OKXTrader) convertSymbolBack(instId string) string {
|
|
parts := strings.Split(instId, "-")
|
|
if len(parts) >= 2 {
|
|
return parts[0] + parts[1]
|
|
}
|
|
return instId
|
|
}
|
|
|
|
// FormatQuantity formats quantity (converts base asset quantity to contract count)
|
|
func (t *OKXTrader) FormatQuantity(symbol string, quantity float64) (string, error) {
|
|
inst, err := t.getInstrument(symbol)
|
|
if err != nil {
|
|
return fmt.Sprintf("%.3f", quantity), nil
|
|
}
|
|
|
|
// OKX uses contract count: quantity (in base asset) / ctVal (asset per contract)
|
|
sz := quantity / inst.CtVal
|
|
return t.formatSize(sz, inst), nil
|
|
}
|
|
|
|
// formatSize formats contract size
|
|
func (t *OKXTrader) formatSize(sz float64, inst *OKXInstrument) string {
|
|
// Determine precision based on lotSz
|
|
if inst.LotSz >= 1 {
|
|
return fmt.Sprintf("%.0f", sz)
|
|
}
|
|
|
|
// Calculate decimal places
|
|
lotSzStr := fmt.Sprintf("%f", inst.LotSz)
|
|
dotIndex := strings.Index(lotSzStr, ".")
|
|
if dotIndex == -1 {
|
|
return fmt.Sprintf("%.0f", sz)
|
|
}
|
|
|
|
// Remove trailing zeros
|
|
lotSzStr = strings.TrimRight(lotSzStr, "0")
|
|
precision := len(lotSzStr) - dotIndex - 1
|
|
|
|
format := fmt.Sprintf("%%.%df", precision)
|
|
return fmt.Sprintf(format, sz)
|
|
}
|