From 01da1f754426a1285fc20b9cf1b80672b16d8814 Mon Sep 17 00:00:00 2001
From: Administrator <15274802129@163.com>
Date: Fri, 08 May 2026 15:04:28 +0800
Subject: [PATCH] refactor(gateApi): 重构网格交易服务添加补仓重试机制

---
 src/main/java/com/xcong/excoin/modules/gateApi/gateApi-logic.md |  129 +++++++++++++++++++++++++++++++------------
 1 files changed, 93 insertions(+), 36 deletions(-)

diff --git a/src/main/java/com/xcong/excoin/modules/gateApi/gateApi-logic.md b/src/main/java/com/xcong/excoin/modules/gateApi/gateApi-logic.md
index c58d0d0..a4dd080 100644
--- a/src/main/java/com/xcong/excoin/modules/gateApi/gateApi-logic.md
+++ b/src/main/java/com/xcong/excoin/modules/gateApi/gateApi-logic.md
@@ -7,8 +7,8 @@
 | [GateWebSocketClientManager](#gatewebsocketclientmanager) | `@Component` | Spring 启动入口,组装组件 + 生命周期 |
 | [GateConfig](#gateconfig) | 配置 | Builder 模式:API 密钥、合约、策略参数、环境切换 |
 | [GateKlineWebSocketClient](#gateklinewebsocketclient) | WS 连接管理 | 连接/心跳/重连/消息路由 |
-| [GateGridTradeService](#gategridtradeservice) | 交易服务 | 网格策略状态机 + 盈亏管理 |
-| [GateTradeExecutor](#gatetradeexecutor) | 异步执行器 | 独立线程池执行 REST 下单,不阻塞 WS 回调线程 |
+| [GateGridTradeService](#gategridtradeservice) | 交易服务 | 网格策略状态机 + 盈亏管理 + 补仓重试 |
+| [GateTradeExecutor](#gatetradeexecutor) | 异步执行器 | 独立线程池执行 REST 下单,成功/失败双回调 |
 | [GateWebSocketClientMain](#gatewebsocketclientmain) | main 入口 | 独立测试启动 |
 | [Example.java](#examplejava) | 示例 | Gate SDK 用法参考 |
 
@@ -19,7 +19,7 @@
 | `wsHandler/GateChannelHandler.java` | **接口** | subscribe / unsubscribe / handleMessage / getChannelName |
 | `wsHandler/AbstractPrivateChannelHandler.java` | **抽象类** | 私有频道基类:HMAC-SHA512 签名 + 认证请求 |
 | `wsHandler/handler/CandlestickChannelHandler.java` | 公开频道 | K 线解析 → `onKline()` |
-| `wsHandler/handler/PositionsChannelHandler.java` | 私有频道 | 仓位推送 → `onPositionUpdate()` |
+| `wsHandler/handler/PositionsChannelHandler.java` | 私有频道 | 仓位推送 → `onPositionUpdate()`(传 `Position.ModeEnum`) |
 | `wsHandler/handler/PositionClosesChannelHandler.java` | 私有频道 | 平仓推送 → `onPositionClose()` |
 
 ---
@@ -36,7 +36,7 @@
 │     │ REST BasePath           │ GateTradeExecutor    │ WS URL    │
 │     ▼                         ▼                      ▼           │
 │  Gate API (REST)        独立线程池 (async)     Gate WebSocket     │
-│                                               ┌──────────────┐  │
+│                      onSuccess/onFailure双回调  ┌──────────────┐  │
 │                                               │Candlestick H  │  │
 │                                               │Positions H    │  │
 │                                               │PosCloses H    │  │
@@ -62,9 +62,10 @@
 │
 ├─ futures.positions (私有, HMAC-SHA512)
 │   └─ PositionsChannelHandler
+│       ├─ 解析 mode → Position.ModeEnum(DUAL_LONG / DUAL_SHORT)
 │       └─ gridTradeService.onPositionUpdate(mode, size, entryPrice)
-│           ├─ size=0 && longActive → tryReopenLong()
-│           └─ size=0 && shortActive → tryReopenShort()
+│           ├─ DUAL_LONG, size=0 && longActive → tryReopenLong()
+│           └─ DUAL_SHORT, size=0 && shortActive → tryReopenShort()
 │
 ├─ futures.position_closes (私有, HMAC-SHA512)
 │   └─ PositionClosesChannelHandler
@@ -73,7 +74,8 @@
 │
 └─ 所有下单操作
     └─ GateTradeExecutor (单线程 + 64队列 + CallerRunsPolicy)
-        ├─ openLong/openShort → 市价单 (REST)
+        ├─ openLong/Short(qty, onSuccess, onFailure)
+        │   └─ 失败回调 → tryReopenXxx() 递归重试
         └─ placeTakeProfit → 条件单 (REST)
 ```
 
@@ -85,7 +87,7 @@
 GateChannelHandler (接口)
   ├── CandlestickChannelHandler          (公开频道)
   └── AbstractPrivateChannelHandler      (私有频道基类: HMAC-SHA512)
-        ├── PositionsChannelHandler
+        ├── PositionsChannelHandler       (解析 mode → Position.ModeEnum)
         └── PositionClosesChannelHandler
 ```
 
@@ -93,6 +95,7 @@
 - **unsubscribe**: 发送取消订阅请求(私有频道也带签名认证)
 - **handleMessage**: 解析推送数据并回调GateGridTradeService,返回true表示已处理
 - 消息路由: update/all事件 → 遍历channelHandlers → handler内部二次匹配channel名 → 匹配成功回调并停止遍历
+- **PositionsChannelHandler 特殊处理**: 推送的 mode 字符串("dual_long")通过 `Position.ModeEnum.fromValue()` 转为枚举,避免调用方用字符串匹配
 
 ---
 
@@ -101,9 +104,11 @@
 ```
 WAITING_KLINE ──onKline──→ OPENING ──双开成功──→ ACTIVE
      │                        │                     │
-     │                    双开失败             ├─ size=0 → REOPENING_L/S → ACTIVE
-     │                                         ├─ 补仓失败 retry → 仍失败 → STOPPED
-     │                                         └─ cumulativePnl≥TP 或 ≤-maxLoss → STOPPED
+     │                    双开失败             ├─ size=0 → REOPENING_L/S
+     │                                         │   ├─ 补仓成功 → ACTIVE
+     │                                         │   └─ 补仓失败 → 递归重试
+     │                                         │       └─ 超 reopenMaxRetries → STOPPED
+     │                                         └─ cumPnl≥TP 或 ≤-maxLoss → STOPPED
      ▼
    STOPPED ←─────────────────────────────────────────┘
 ```
@@ -115,19 +120,26 @@
 | `ACTIVE` | 网格运行中,等待止盈触发 |
 | `REOPENING_LONG` | 正在补开多头 |
 | `REOPENING_SHORT` | 正在补开空头 |
-| `STOPPED` | 停止(盈利达标 / 亏损超限 / 异常退出) |
+| `STOPPED` | 停止(盈利达标 / 亏损超限 / 补仓重试耗尽 / 异常退出) |
 
 ---
 
 ## 策略时序
 
-### 阶段 1:启动
+### 阶段 1:启动与初始化
 
 ```
 Spring @PostConstruct
   → GateConfig.builder()...build()
   → GateGridTradeService(config)
-    → init(): 查ID → 查账户切持仓 → 清旧条件单 → 设杠杆
+    → init():
+      1. 查用户ID
+      2. 查账户 → 如需要切持仓模式
+      3. 清除旧止盈止损条件单
+      4. 查当前合约所有仓位 → 逐个市价平仓(reduce_only, IOC)
+         - 单向持仓: size=相反数平仓
+         - 双向持仓: size=0, close=false, autoSize=LONG/SHORT
+      5. 设杠杆
   → GateKlineWebSocketClient(config.getWsUrl())
     → addChannelHandler x3 → init() → connect()
       → onOpen: handlers依次subscribe → sendPing
@@ -138,28 +150,35 @@
 
 ```
 K线推送 → onKline(closePrice) → state=OPENING
-  → GateTradeExecutor.openLong → 市价开多 → onSuccess: longActive=true, 下TP单
-  → GateTradeExecutor.openShort → 市价开空 → onSuccess: shortActive=true, 下TP单
+  → GateTradeExecutor.openLong(qty, onSuccess, null)
+    → 市价开多 → onSuccess: longActive=true, placeTakeProfit(多头TP)
+  → GateTradeExecutor.openShort(-qty, onSuccess, null)
+    → 市价开空 → onSuccess: shortActive=true, placeTakeProfit(空头TP)
   → 双开均完成 → state=ACTIVE
 ```
 
-### 阶段 3:止盈触发 → 补仓
+### 阶段 3:止盈触发 → 补仓(含重试)
 
 ```
-仓位推送: dual_long, size=0 → longActive且无仓位 → tryReopenLong
-  → GateTradeExecutor.openLong → 市价补多 → onSuccess: 下新TP单
+仓位推送: mode=DUAL_LONG, size=0 → longActive且无仓位 → tryReopenLong()
+  ┌─ longReopenFails++  → 超 reopenMaxRetries → STOPPED
+  └─ GateTradeExecutor.openLong(qty,
+        onSuccess: failCount=0, ACTIVE, 下新TP单,
+        onFailure: → 递归调用 tryReopenLong() 再试
+     )
 
-仓位推送: dual_short, size=0 → shortActive且无仓位 → tryReopenShort
-  → GateTradeExecutor.openShort → 市价补空 → onSuccess: 下新TP单
+仓位推送: mode=DUAL_SHORT, size=0 → 同理 tryReopenShort()
 ```
 
 > 止盈由 Gate 服务端条件单自动执行。只补被平掉的单方向,另一方不受影响。
+> 补仓失败通过 onFailure 回调递归重试,连续失败超过 `reopenMaxRetries`(默认3次)则停止策略。
 
 ### 阶段 4:停止
 
 ```
 平仓推送: pnl=+0.6 → cumulativePnl=0.6 ≥ overallTp → state=STOPPED
 平仓推送: pnl=-8.0 → cumulativePnl=-8.0 ≤ -maxLoss → state=STOPPED
+补仓连续失败 4 次 → 超过 reopenMaxRetries=3 → state=STOPPED
 ```
 
 ---
@@ -184,22 +203,27 @@
 | overallTp | 0.5 USDT | 整体止盈 |
 | maxLoss | 7.5 USDT | 最大亏损 |
 | quantity | 1 | 下单张数 |
-| reopenMaxRetries | 3 | 补仓重试次数 |
+| reopenMaxRetries | 3 | 补仓最大重试次数 |
 
 ---
 
 ## GateTradeExecutor
 
-**角色**: 独立线程池执行 REST API 下单,解决 WebSocket 回调线程阻塞问题。
+**角色**: 独立线程池执行 REST API 下单。采用成功/失败双回调模式支持补仓重试。
 
 **线程模型**:
 - `ThreadPoolExecutor(1, 1, 60s, LinkedBlockingQueue(64), CallerRunsPolicy)`
 - 单线程保序 + 有界队列防堆积 + CallerRuns背压
 
+**回调设计**:
+- 每个下单方法接受 `onSuccess` 和 `onFailure` 两个 `Runnable`
+- REST 调用成功 → 执行 `onSuccess`(更新状态、下止盈单、重置失败计数)
+- REST 调用失败 → 执行 `onFailure`(递归重试入口)
+
 | 方法 | 说明 |
 |------|------|
-| `openLong(qty, onSuccess)` | 异步 IOC 市价开多,成功回调 |
-| `openShort(qty, onSuccess)` | 异步 IOC 市价开空 |
+| `openLong(qty, onSuccess, onFailure)` | 异步 IOC 市价开多,双回调 |
+| `openShort(qty, onSuccess, onFailure)` | 异步 IOC 市价开空,双回调 |
 | `placeTakeProfit(trigger, rule, type, auto)` | 异步条件单。已存在则清除旧单重试 |
 | `cancelAllPriceTriggeredOrders()` | 清除所有条件单 |
 | `shutdown()` | 等待10秒,超时强制关闭 |
@@ -210,12 +234,31 @@
 
 **角色**: 策略核心,使用 Gate SDK 管理状态和执行下单。
 
-**状态**: StrategyState enum + longActive/shortActive boolean
+**状态**: `StrategyState` enum + `longActive`/`shortActive` boolean
+
+**关键常量**(替代字面字符串):
+```java
+private static final String AUTO_SIZE_LONG = "close_long";
+private static final String AUTO_SIZE_SHORT = "close_short";
+private static final String ORDER_TYPE_CLOSE_LONG = "close-long-position";
+private static final String ORDER_TYPE_CLOSE_SHORT = "close-short-position";
+```
 
 **回调方法**:
 - `onKline(closePrice)`: 缓存价格,WAITING_KLINE状态下首次触发双开
-- `onPositionUpdate(mode, size, entryPrice)`: size=0且方向活跃 → 补仓
+- `onPositionUpdate(Position.ModeEnum mode, size, entryPrice)`: `== DUAL_LONG/DUAL_SHORT` 枚举比较,size=0且方向活跃 → 补仓重试
 - `onPositionClose(side, pnl)`: 累加盈亏,检查停止条件
+
+**补仓重试逻辑**:
+```
+tryReopenLong():
+  1. longReopenFails++(失败计数递增)
+  2. 超过 reopenMaxRetries → STOPPED
+  3. openLong(qty,
+       onSuccess → failCount=0, ACTIVE, 下新TP单,
+       onFailure → 递归 tryReopenLong() 重试
+     )
+```
 
 **止盈计算**:
 
@@ -226,15 +269,29 @@
 
 **REST API 调用**:
 
-| 操作 | API | 方法 |
-|------|-----|------|
-| 获取用户ID | `GET /account/detail` | `AccountApi.getAccountDetail()` |
-| 切持仓模式 | `POST /futures/usdt/set_position_mode` | `FuturesApi.setPositionMode()` |
-| 设杠杆 | `POST /futures/usdt/positions/{contract}/leverage` | `FuturesApi.updateContractPositionLeverageCall()` |
-| 查账户 | `GET /futures/usdt/accounts` | `FuturesApi.listFuturesAccounts()` |
-| 清除条件单 | `DELETE /futures/usdt/price_orders` | `FuturesApi.cancelPriceTriggeredOrderList()` |
-| 市价单 | `POST /futures/usdt/orders` (price=0, IOC) | `FuturesApi.createFuturesOrder()` |
-| 条件单 | `POST /futures/usdt/price_orders` | `FuturesApi.createPriceTriggeredOrder()` |
+| 操作 | API | 方法 | 说明 |
+|------|-----|------|------|
+| 获取用户ID | `GET /account/detail` | `AccountApi.getAccountDetail()` | |
+| 切持仓模式 | `POST /futures/usdt/set_position_mode` | `FuturesApi.setPositionMode()` | |
+| 查仓位 | `GET /futures/usdt/positions` | `FuturesApi.listPositions()` | 遍历所有仓位,按合约过滤 |
+| 市价平仓 | `POST /futures/usdt/orders` | `FuturesApi.createFuturesOrder()` | reduce_only, IOC。双向: size=0+close=false+autoSize |
+| 设杠杆 | `POST /futures/usdt/positions/{contract}/leverage` | `FuturesApi.updateContractPositionLeverageCall()` | |
+| 查账户 | `GET /futures/usdt/accounts` | `FuturesApi.listFuturesAccounts()` | |
+| 清除条件单 | `DELETE /futures/usdt/price_orders` | `FuturesApi.cancelPriceTriggeredOrderList()` | |
+| 市价单 | `POST /futures/usdt/orders` | `FuturesApi.createFuturesOrder()` | price=0, tif=IOC |
+| 条件单 | `POST /futures/usdt/price_orders` | `FuturesApi.createPriceTriggeredOrder()` | strategy=0, price_type=0, expiration=0 |
+
+**初始化顺序** (`init()`):
+```
+1. 获取用户 ID
+2. 查账户 → 如需要切持仓模式
+3. 清除旧的止盈止损条件单
+4. 查当前合约所有仓位 → 逐个市价平仓
+   - 单向持仓(single): size=相反数, reduce_only=true
+   - 双向持仓(dual): size=0, close=false, autoSize=LONG/SHORT, reduce_only=true
+5. 设杠杆
+6. 打印账户余额
+```
 
 ---
 

--
Gitblit v1.9.1