package com.xcong.excoin.modules.gateApi.wsHandler.handler;
import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject;
import com.xcong.excoin.modules.gateApi.GateGridTradeService;
import com.xcong.excoin.modules.gateApi.wsHandler.AbstractPrivateChannelHandler;
import lombok.extern.slf4j.Slf4j;
import java.math.BigDecimal;
/**
* 平仓频道处理器。
*
*
数据用途
* 每笔平仓发生时推送 pnl(盈亏金额),累加到 {@code cumulativePnl} 用于判断策略停止条件:
* cumulativePnl ≥ overallTp(达到止盈目标)或 ≤ -maxLoss(超过亏损上限)。
* 止盈由 Gate 服务端条件单自动触发,平仓后仓位变为 0,盈亏通过本频道推送。
*
* 推送字段
* contract, side(long / short), pnl(该次平仓的盈亏,如 "+0.2" 或 "-0.1")
*
* 注意
* 平仓盈亏来自服务器端,不受本地计算误差影响。这是策略停止的唯一盈亏判断来源。
*
* @author Administrator
*/
@Slf4j
public class PositionClosesChannelHandler extends AbstractPrivateChannelHandler {
private static final String CHANNEL_NAME = "futures.position_closes";
/**
* @param apiKey Gate API v4 密钥,用于签名认证
* @param apiSecret Gate API v4 签名密钥
* @param contract 合约名称(如 ETH_USDT)
* @param gridTradeService 网格交易策略服务实例
*/
public PositionClosesChannelHandler(String apiKey, String apiSecret,
String contract,
GateGridTradeService gridTradeService) {
super(CHANNEL_NAME, apiKey, apiSecret, contract, gridTradeService);
}
/**
* 处理平仓推送消息。
*
* 数据提取
* result 数组中每个元素包含:
*
* - contract:合约名称
* - side:平仓方向("long" / "short")
* - pnl:本次平仓的盈亏金额(字符串格式,如 "+0.2" / "-0.1")
*
*
* 数据处理
* 按合约名称过滤 → 提取 pnl 和 side → 调用 gridTradeService.onPositionClose() 累加盈亏。
* pnl 来自服务端,不受本地计算误差影响。
*
* @param response WebSocket 推送的完整 JSON
* @return true 表示已处理(匹配成功)
*/
@Override
public boolean handleMessage(JSONObject response) {
if (!CHANNEL_NAME.equals(response.getString("channel"))) {
return false;
}
try {
JSONArray resultArray = response.getJSONArray("result");
if (resultArray == null || resultArray.isEmpty()) {
return true;
}
for (int i = 0; i < resultArray.size(); i++) {
JSONObject item = resultArray.getJSONObject(i);
if (!getContract().equals(item.getString("contract"))) {
continue;
}
BigDecimal pnl = new BigDecimal(item.getString("pnl"));
String side = item.getString("side");
log.info("[{}] 平仓更新, 方向:{}, 盈亏:{}", CHANNEL_NAME, side, pnl);
if (getGridTradeService() != null) {
getGridTradeService().onPositionClose(getContract(), side, pnl);
}
}
} catch (Exception e) { log.error("[{}] 处理数据失败", CHANNEL_NAME, e); }
return true;
}
}