From 6a51f45e6a00b65a9e7b0b0707b453c11311f3ef Mon Sep 17 00:00:00 2001
From: Administrator <15274802129@163.com>
Date: Mon, 11 May 2026 22:38:13 +0800
Subject: [PATCH] feat(okxApi): 添加仓位模式配置和REST客户端功能

---
 src/main/java/com/xcong/excoin/modules/okxApi/wsHandler/handler/OkxOrderInfoChannelHandler.java |  121 ++++++++++++++++++++++++++++++++++++++++
 1 files changed, 121 insertions(+), 0 deletions(-)

diff --git a/src/main/java/com/xcong/excoin/modules/okxApi/wsHandler/handler/OkxOrderInfoChannelHandler.java b/src/main/java/com/xcong/excoin/modules/okxApi/wsHandler/handler/OkxOrderInfoChannelHandler.java
new file mode 100644
index 0000000..0a33aa4
--- /dev/null
+++ b/src/main/java/com/xcong/excoin/modules/okxApi/wsHandler/handler/OkxOrderInfoChannelHandler.java
@@ -0,0 +1,121 @@
+package com.xcong.excoin.modules.okxApi.wsHandler.handler;
+
+import com.alibaba.fastjson.JSONArray;
+import com.alibaba.fastjson.JSONObject;
+import com.xcong.excoin.modules.okxApi.OkxConfig;
+import com.xcong.excoin.modules.okxApi.OkxGridTradeService;
+import com.xcong.excoin.modules.okxApi.wsHandler.OkxChannelHandler;
+import lombok.extern.slf4j.Slf4j;
+import org.java_websocket.client.WebSocketClient;
+
+import java.math.BigDecimal;
+
+/**
+ * OKX 订单信息频道处理器(orders)。
+ *
+ * <h3>数据用途</h3>
+ * 监控订单成交(filled)状态,跟踪已实现盈亏(fillPnl)。
+ * 成交后通过 batch-orders 频道设置止盈止损限价单。
+ *
+ * <h3>推送字段</h3>
+ * instId, ordId, clOrdId, side, posSide, state, accFillSz, avgPx, fillPnl, fillFee
+ *
+ * <h3>数据处理</h3>
+ * state=filled 且 accFillSz>0 → 成交 → 调用 gridTradeService.onOrderFilled() 跟踪盈亏
+ *
+ * @author Administrator
+ */
+@Slf4j
+public class OkxOrderInfoChannelHandler implements OkxChannelHandler {
+
+    private static final String CHANNEL_NAME = "orders";
+    private static final String CHANNEL = "orders";
+
+    private final String instId;
+    private final OkxGridTradeService gridTradeService;
+    private final OkxConfig config;
+
+    public OkxOrderInfoChannelHandler(String instId, OkxGridTradeService gridTradeService, OkxConfig config) {
+        this.instId = instId;
+        this.gridTradeService = gridTradeService;
+        this.config = config;
+    }
+
+    @Override
+    public String getChannelName() {
+        return CHANNEL_NAME;
+    }
+
+    @Override
+    public void subscribe(WebSocketClient ws) {
+        JSONObject msg = new JSONObject();
+        msg.put("op", "subscribe");
+        JSONArray args = new JSONArray();
+        JSONObject arg = new JSONObject();
+        arg.put("channel", CHANNEL);
+        arg.put("instType", "SWAP");
+        arg.put("instId", instId);
+        args.add(arg);
+        msg.put("args", args);
+        ws.send(msg.toJSONString());
+        log.info("[{}] 订阅成功, 合约:{}", CHANNEL_NAME, instId);
+    }
+
+    @Override
+    public void unsubscribe(WebSocketClient ws) {
+        JSONObject msg = new JSONObject();
+        msg.put("op", "unsubscribe");
+        JSONArray args = new JSONArray();
+        JSONObject arg = new JSONObject();
+        arg.put("channel", CHANNEL);
+        arg.put("instType", "SWAP");
+        arg.put("instId", instId);
+        args.add(arg);
+        msg.put("args", args);
+        ws.send(msg.toJSONString());
+        log.info("[{}] 取消订阅成功", CHANNEL_NAME);
+    }
+
+    @Override
+    public boolean handleMessage(JSONObject response) {
+        JSONObject argObj = response.getJSONObject("arg");
+        if (argObj == null) {
+            return false;
+        }
+        String channel = argObj.getString("channel");
+        if (!CHANNEL.equals(channel)) {
+            return false;
+        }
+        try {
+            JSONArray dataArray = response.getJSONArray("data");
+            if (dataArray == null || dataArray.isEmpty()) {
+                return true;
+            }
+            for (int i = 0; i < dataArray.size(); i++) {
+                JSONObject detail = dataArray.getJSONObject(i);
+                if (!instId.equals(detail.getString("instId"))) {
+                    continue;
+                }
+                String state = detail.getString("state");
+                String accFillSz = detail.getString("accFillSz");
+                String pnl = detail.getString("pnl");
+                String posSide = detail.getString("posSide");
+                String avgPx = detail.getString("avgPx");
+                String clOrdId = detail.getString("clOrdId");
+
+                log.info("[{}] 订单, 方向:{}, 状态:{}, 成交量:{}, 均价:{}, 盈亏:{}, 编号:{}",
+                        CHANNEL_NAME, posSide, state, accFillSz, avgPx, pnl, clOrdId);
+
+                if ("filled".equals(state) && accFillSz != null && new BigDecimal(accFillSz).compareTo(BigDecimal.ZERO) > 0) {
+                    if (gridTradeService != null) {
+                        BigDecimal pnlVal = pnl != null ? new BigDecimal(pnl) : BigDecimal.ZERO;
+                        gridTradeService.onOrderFilled(posSide, new BigDecimal(accFillSz), pnlVal);
+                    }
+                }
+            }
+        } catch (Exception e) {
+            log.error("[{}] 处理数据失败", CHANNEL_NAME, e);
+        }
+        return true;
+    }
+}

--
Gitblit v1.9.1