alphax/app/db/strategy_insights.py
2026-05-20 00:57:46 +08:00

235 lines
9.7 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

"""Strategy attribution read model based on opportunity and paper-trading conversion."""
import json
import re
from app.db.schema import get_conn
def safe_list_json(value):
try:
if isinstance(value, list):
return value
if isinstance(value, str) and value.strip():
parsed = json.loads(value)
return parsed if isinstance(parsed, list) else []
except Exception:
pass
return []
def safe_dict_json(value):
try:
if isinstance(value, dict):
return value
if isinstance(value, str) and value.strip():
parsed = json.loads(value)
return parsed if isinstance(parsed, dict) else {}
except Exception:
pass
return {}
def get_strategy_insights():
"""Strategy attribution based on opportunity and paper-trading conversion.
Recommendation rows are opportunities/signals, not an execution ledger.
Therefore this read model does not use recommendation.pnl_pct as strategy
PnL. Paper-trading PnL is exposed only as an execution-conversion metric.
"""
conn = get_conn()
rows = conn.execute(
"""
SELECT
r.*,
pt.id AS paper_trade_id,
pt.status AS paper_status,
pt.realized_pnl_pct AS paper_realized_pnl_pct,
pt.realized_pnl_usdt AS paper_realized_pnl_usdt,
pt.pnl_pct AS paper_pnl_pct,
pt.exit_reason AS paper_exit_reason
FROM recommendation r
LEFT JOIN paper_trades pt ON pt.recommendation_id = r.id
ORDER BY r.rec_time DESC, r.id DESC
"""
).fetchall()
conn.close()
items = [dict(r) for r in rows]
actionable_statuses = {"buy_now", "wait_pullback"}
total = len(items)
actionable = [x for x in items if (x.get("execution_status") or "") in actionable_statuses]
buy_now = [x for x in items if (x.get("execution_status") or "") == "buy_now"]
paper_items = [x for x in items if x.get("paper_trade_id")]
closed_paper = [x for x in paper_items if x.get("paper_status") == "closed"]
paper_wins = [x for x in closed_paper if float(x.get("paper_realized_pnl_pct") or 0) > 0]
paper_realized_usdt = round(sum(float(x.get("paper_realized_pnl_usdt") or 0) for x in closed_paper), 4)
overview = {
"total_opportunities": total,
"actionable_count": len(actionable),
"buy_now_count": len(buy_now),
"paper_trade_count": len(paper_items),
"closed_paper_trade_count": len(closed_paper),
"paper_win_count": len(paper_wins),
"paper_win_rate_pct": round(len(paper_wins) / len(closed_paper) * 100, 1) if closed_paper else 0,
"paper_realized_pnl_usdt": paper_realized_usdt,
"actionable_conversion_pct": round(len(actionable) / total * 100, 1) if total else 0,
"paper_conversion_pct": round(len(paper_items) / len(buy_now) * 100, 1) if buy_now else 0,
"definition": "策略归因只看机会转化和模拟交易转化;收益只来自 paper_trades不读取 recommendation.pnl_pct。",
}
def add_bucket(bucket_map, key, item):
if not key:
return
b = bucket_map.setdefault(key, {
"opportunity_count": 0,
"actionable_count": 0,
"buy_now_count": 0,
"paper_trade_count": 0,
"closed_paper_trade_count": 0,
"paper_win_count": 0,
"paper_realized_pnl_usdt": 0.0,
})
execution_status = item.get("execution_status") or ""
paper_status = item.get("paper_status") or ""
b["opportunity_count"] += 1
if execution_status in actionable_statuses:
b["actionable_count"] += 1
if execution_status == "buy_now":
b["buy_now_count"] += 1
if item.get("paper_trade_id"):
b["paper_trade_count"] += 1
if paper_status == "closed":
b["closed_paper_trade_count"] += 1
pnl_pct = float(item.get("paper_realized_pnl_pct") or 0)
if pnl_pct > 0:
b["paper_win_count"] += 1
b["paper_realized_pnl_usdt"] += float(item.get("paper_realized_pnl_usdt") or 0)
factor_map = {}
env_map = {}
version_map = {}
evidence_map = {}
for item in items:
labels = safe_list_json(item.get("signal_labels_json")) or safe_list_json(item.get("signals"))
codes = safe_list_json(item.get("signal_codes_json"))
for factor in labels:
add_bucket(factor_map, str(factor).strip(), item)
for code in codes:
text = str(code or "").strip()
if text.startswith(("sentiment_", "listing_", "ecosystem_")):
add_bucket(evidence_map, "舆情:" + text, item)
elif text.startswith(("dex_", "liquidity_", "exchange_", "whale_", "smart_money", "holder_")):
add_bucket(evidence_map, "链上:" + text, item)
mc = safe_dict_json(item.get("market_context_json"))
for key in ("btc_trend", "market_regime", "altcoin_regime", "sentiment"):
if mc.get(key):
add_bucket(env_map, f"{key}:{mc.get(key)}", item)
for bucket in env_buckets_from_market_context(mc):
add_bucket(env_map, bucket, item)
if item.get("strategy_version"):
add_bucket(version_map, str(item.get("strategy_version")).strip(), item)
return {
"overview": overview,
"metric_definition": {
"opportunity_count": "进入 opportunity/recommendation 表的机会样本数,不代表交易。",
"actionable_count": "确认层输出 buy_now 或 wait_pullback 的样本数。",
"paper_trade_count": "已经被模拟交易账本执行的样本数。",
"paper_realized_pnl_usdt": "仅来自 paper_trades 的已平仓模拟收益。",
},
"factor_attribution": serialize_buckets("factor", factor_map)[:30],
"market_environment": serialize_buckets("environment", env_map)[:20],
"evidence_attribution": serialize_buckets("evidence", evidence_map)[:20],
"version_performance": serialize_buckets("strategy_version", version_map, sort_by_version=True)[:20],
}
def env_buckets_from_market_context(mc):
"""Convert market_context_json numeric fields into attribution buckets."""
buckets = []
try:
change_24h = float(mc.get("change_24h", 0) or 0)
turn_1h = float(mc.get("turnover_acceleration_1h", 0) or 0)
turn_4h = float(mc.get("turnover_acceleration_4h", 0) or 0)
volume_24h = float(mc.get("volume_24h") or mc.get("quote_volume_24h") or 0)
funding = float(mc.get("funding_rate", 0) or 0)
except Exception:
change_24h = turn_1h = turn_4h = volume_24h = funding = 0
if change_24h >= 8:
buckets.append("24h涨幅:强势拉升≥8%")
elif change_24h >= 3:
buckets.append("24h涨幅:温和上涨3-8%")
elif change_24h <= -3:
buckets.append("24h涨幅:回撤≤-3%")
else:
buckets.append("24h涨幅:震荡-3~3%")
if turn_1h >= 3:
buckets.append("1h成交加速:爆量≥3x")
elif turn_1h >= 1.5:
buckets.append("1h成交加速:放量1.5-3x")
elif turn_1h > 0:
buckets.append("1h成交加速:平量<1.5x")
if turn_4h >= 3:
buckets.append("4h成交加速:爆量≥3x")
elif turn_4h >= 1.5:
buckets.append("4h成交加速:放量1.5-3x")
elif turn_4h > 0:
buckets.append("4h成交加速:平量<1.5x")
if volume_24h >= 100_000_000:
buckets.append("24h成交额:高流动性≥1亿")
elif volume_24h >= 10_000_000:
buckets.append("24h成交额:中等流动性1千万-1亿")
elif volume_24h > 0:
buckets.append("24h成交额:低流动性<1千万")
if funding >= 0.0005:
buckets.append("资金费率:多头拥挤")
elif funding <= -0.0005:
buckets.append("资金费率:空头拥挤")
return buckets
def serialize_buckets(name_key, bucket_map, sort_by_version=False):
rows = []
for key, bucket in bucket_map.items():
rows.append({
name_key: key,
"opportunity_count": bucket["opportunity_count"],
"actionable_count": bucket["actionable_count"],
"buy_now_count": bucket["buy_now_count"],
"paper_trade_count": bucket["paper_trade_count"],
"closed_paper_trade_count": bucket["closed_paper_trade_count"],
"paper_win_count": bucket["paper_win_count"],
"actionable_conversion_pct": round(bucket["actionable_count"] / bucket["opportunity_count"] * 100, 1) if bucket["opportunity_count"] else 0,
"paper_conversion_pct": round(bucket["paper_trade_count"] / bucket["buy_now_count"] * 100, 1) if bucket["buy_now_count"] else 0,
"paper_win_rate_pct": round(bucket["paper_win_count"] / bucket["closed_paper_trade_count"] * 100, 1) if bucket["closed_paper_trade_count"] else 0,
"paper_realized_pnl_usdt": round(bucket["paper_realized_pnl_usdt"], 4),
})
if sort_by_version:
rows.sort(key=lambda x: (version_sort_key(x[name_key]), x["opportunity_count"], x["actionable_conversion_pct"]), reverse=True)
else:
rows.sort(key=lambda x: (-x["opportunity_count"], -x["actionable_conversion_pct"], x[name_key]))
return rows
def version_sort_key(version: str):
text = str(version or '').strip()
if text.startswith('v') or text.startswith('V'):
text = text[1:]
parts = []
for chunk in text.replace('-', '.').split('.'):
if chunk.isdigit():
parts.append(int(chunk))
else:
match = re.match(r'^(\d+)', chunk)
if match:
parts.append(int(match.group(1)))
else:
parts.append(chunk)
return tuple(parts)