From e9a397babbbfa9cff8a5ed026447d585e739c37f Mon Sep 17 00:00:00 2001
From: Administrator <15274802129@163.com>
Date: Tue, 12 May 2026 21:47:21 +0800
Subject: [PATCH] refactor(gateApi): 将网格交易策略从限价单改为条件单
---
src/main/java/com/xcong/excoin/modules/gateApi/GateWebSocketClientManager.java | 192 ++++++++++++++++++++++--------------------------
1 files changed, 88 insertions(+), 104 deletions(-)
diff --git a/src/main/java/com/xcong/excoin/modules/gateApi/GateWebSocketClientManager.java b/src/main/java/com/xcong/excoin/modules/gateApi/GateWebSocketClientManager.java
index f0cd365..f9c2c35 100644
--- a/src/main/java/com/xcong/excoin/modules/gateApi/GateWebSocketClientManager.java
+++ b/src/main/java/com/xcong/excoin/modules/gateApi/GateWebSocketClientManager.java
@@ -9,28 +9,26 @@
import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import java.math.BigDecimal;
-import java.util.ArrayList;
-import java.util.List;
/**
- * Gate 模块 Spring 入口,管理多个账号实例的生命周期。
+ * Gate 模块 Spring 入口,组装所有组件并管理生命周期。
*
* <h3>启动流程 ({@code @PostConstruct})</h3>
* <ol>
- * <li>遍历 {@link #buildAccountConfigs()} 构建每个账号的 {@link GateConfig}</li>
- * <li>每个账号创建独立的 {@link AccountInstance}(含 service + wsClient)</li>
- * <li>逐个 init → startGrid</li>
+ * <li>构建 {@link GateConfig}(Builder 模式,含 API 密钥、合约、策略参数)</li>
+ * <li>创建 {@link GateGridTradeService} → init():切持仓模式、清旧条件单、设杠杆</li>
+ * <li>创建 {@link GateKlineWebSocketClient} → 注册 3 个 Handler → init():建立 WS 连接</li>
+ * <li>gridTradeService.startGrid():激活策略,等待 K 线触发首次双开</li>
* </ol>
*
* <h3>销毁流程 ({@code @PreDestroy})</h3>
* <ol>
- * <li>遍历 instances:stopGrid + wsClient.destroy()</li>
+ * <li>gridTradeService.stopGrid():取消条件单 → 关闭交易线程池</li>
+ * <li>wsClient.destroy():取消订阅 → 断开 WS → 关闭线程池</li>
* </ol>
*
- * <h3>添加新账号</h3>
- * 在 {@link #buildAccountConfigs()} 方法中添加新的 {@link GateConfig} 即可。
- * 通过 {@link GateConfig#getLabel()} 区分日志输出,
- * 通过 {@link GateConfig#isProduction()} 控制模拟盘/实盘环境。
+ * <h3>配置</h3>
+ * 当前在代码中硬编码测试网参数。切换到生产网只需改为 {@code .isProduction(true)}。
*
* @author Administrator
*/
@@ -38,109 +36,95 @@
@Component
public class GateWebSocketClientManager {
- /** 所有账号实例列表 */
- private final List<AccountInstance> instances = new ArrayList<>();
-
- /**
- * 封装一个账号的完整运行时组件。
- */
- private static class AccountInstance {
- final String label;
- final GateConfig config;
- final GateGridTradeService service;
- final GateKlineWebSocketClient wsClient;
-
- AccountInstance(GateConfig config) {
- this.label = config.getLabel();
- this.config = config;
- this.service = new GateGridTradeService(config);
- this.wsClient = new GateKlineWebSocketClient(config.getWsUrl());
- }
-
- void init() {
- service.init();
- wsClient.addChannelHandler(new CandlestickChannelHandler(config.getContract(), service));
- wsClient.addChannelHandler(new PositionsChannelHandler(
- config.getApiKey(), config.getApiSecret(), config.getContract(), service));
- wsClient.addChannelHandler(new PositionClosesChannelHandler(
- config.getApiKey(), config.getApiSecret(), config.getContract(), service));
- wsClient.init();
- log.info("[{}] WS已连接, 已注册 3 个频道处理器", label);
- service.startGrid();
- }
-
- void destroy() {
- log.info("[{}] 开始销毁...", label);
- service.stopGrid();
- wsClient.destroy();
- log.info("[{}] 销毁完成", label);
- }
- }
-
- /**
- * 在此方法中配置所有账号。
- * <p>每个 {@link GateConfig} 对应一个账号,label 用于日志区分,
- * isProduction 控制连接测试网(true=实盘)还是测试网(false=模拟盘)。
- */
- private static GateConfig[] buildAccountConfigs() {
- return new GateConfig[]{
- GateConfig.builder()
- .label("模拟盘1")
- .apiKey("d90ca272391992b8e74f8f92cedb21ec")
- .apiSecret("1861e4f52de4bb53369ea3208d9ede38ece4777368030f96c77d27934c46c274")
- .contract("ETH_USDT")
- .leverage("100")
- .marginMode("CROSS")
- .positionMode("dual")
- .gridRate(new BigDecimal("0.0015"))
- .overallTp(new BigDecimal("5"))
- .maxLoss(new BigDecimal("15"))
- .quantity("1")
- .contractMultiplier(new BigDecimal("0.01"))
- .unrealizedPnlPriceMode(GateConfig.PnLPriceMode.LAST_PRICE)
- .isProduction(false)
- .reopenMaxRetries(3)
- .build(),
- };
- }
+ /** WebSocket 连接管理器 */
+ private GateKlineWebSocketClient wsClient;
+ /** 网格交易策略服务 */
+ private GateGridTradeService gridTradeService;
+ /** 统一配置 */
+ private GateConfig config;
@PostConstruct
public void init() {
- GateConfig[] configs = buildAccountConfigs();
- log.info("[管理器] 开始初始化 {} 个账号...", configs.length);
- for (GateConfig config : configs) {
- try {
- AccountInstance instance = new AccountInstance(config);
- instance.init();
- instances.add(instance);
- } catch (Exception e) {
- log.error("[管理器] 账号 {} 初始化失败", config.getLabel(), e);
- }
- }
+ log.info("[管理器] 开始初始化...");
- log.info("[管理器] 初始化完成, 成功启动 {}/{} 个账号", instances.size(), configs.length);
+ try {
+ //测试盘
+// config = GateConfig.builder()
+// .apiKey("d90ca272391992b8e74f8f92cedb21ec")
+// .apiSecret("1861e4f52de4bb53369ea3208d9ede38ece4777368030f96c77d27934c46c274")
+// .contract("ETH_USDT")
+// .leverage("100")
+// .marginMode("CROSS")
+// .positionMode("dual")
+// .gridRate(new BigDecimal("0.003"))
+// .overallTp(new BigDecimal("5"))
+// .maxLoss(new BigDecimal("15"))
+// .quantity("1")
+// .contractMultiplier(new BigDecimal("0.01"))
+// .unrealizedPnlPriceMode(GateConfig.PnLPriceMode.LAST_PRICE)
+// .isProduction(false)
+// .reopenMaxRetries(3)
+// .build();
+ //实盘
+ config = GateConfig.builder()
+ .apiKey("865371cdaccd5d238aceb06a55f0143a")
+ .apiSecret("49589c30dfdc3acba007eed445a94990c4b0aa5faac9843e32defdd7371f5a50")
+ .contract("ETH_USDT")
+ .leverage("100")
+ .marginMode("CROSS")
+ .positionMode("dual")
+ .gridRate(new BigDecimal("0.005"))
+ .overallTp(new BigDecimal("5"))
+ .maxLoss(new BigDecimal("15"))
+ .quantity("1")
+ .contractMultiplier(new BigDecimal("0.01"))
+ .unrealizedPnlPriceMode(GateConfig.PnLPriceMode.LAST_PRICE)
+ .isProduction(true)
+ .reopenMaxRetries(3)
+ .build();
+
+ // 1. 初始化交易服务:查用户ID → 切持仓模式 → 清条件单 → 平已有仓位 → 设杠杆
+ gridTradeService = new GateGridTradeService(config);
+ gridTradeService.init();
+
+ // 2. 创建 WS 客户端并注册 3 个频道处理器
+ wsClient = new GateKlineWebSocketClient(config.getWsUrl());
+ wsClient.addChannelHandler(new CandlestickChannelHandler(config.getContract(), gridTradeService));
+ wsClient.addChannelHandler(new PositionsChannelHandler(
+ config.getApiKey(), config.getApiSecret(), config.getContract(), gridTradeService));
+ wsClient.addChannelHandler(new PositionClosesChannelHandler(
+ config.getApiKey(), config.getApiSecret(), config.getContract(), gridTradeService));
+ wsClient.init();
+ log.info("[管理器] WS已连接, 已注册 3 个频道处理器");
+
+ // 3. 激活策略,等待首根 K 线触发基底双开
+ gridTradeService.startGrid();
+ } catch (Exception e) {
+ log.error("[管理器] 初始化失败", e);
+ }
}
+ /**
+ * 销毁:停止策略 → 关闭交易线程池 → 取消 WS 订阅 → 断开连接 → 关闭 WS 线程池。
+ */
@PreDestroy
public void destroy() {
- log.info("[管理器] 开始销毁 {} 个账号...", instances.size());
- for (AccountInstance instance : instances) {
- try {
- instance.destroy();
- } catch (Exception e) {
- log.error("[管理器] 账号 {} 销毁失败", instance.label, e);
- }
+ log.info("[管理器] 开始销毁...");
+ if (gridTradeService != null) {
+ gridTradeService.stopGrid();
+ }
+ if (wsClient != null) {
+ wsClient.destroy();
}
log.info("[管理器] 销毁完成");
}
- /** 获取账号数量 */
- public int getInstanceCount() {
- return instances.size();
- }
-
- /** 获取指定索引的 GridTradeService(用于外部监控/查询) */
- public GateGridTradeService getGridTradeService(int index) {
- return instances.get(index).service;
- }
+ /**
+ * @return WebSocket 连接管理器实例
+ */
+ public GateKlineWebSocketClient getKlineWebSocketClient() { return wsClient; }
+ /**
+ * @return 网格交易策略服务实例
+ */
+ public GateGridTradeService getGridTradeService() { return gridTradeService; }
}
--
Gitblit v1.9.1