fix: price_monitor HKD→CNY + Eastmoney 2s timeout + None-safe + summary fix

This commit is contained in:
知微
2026-07-03 13:24:10 +08:00
parent 9124a7ad56
commit c1c4ba4a81
2 changed files with 51 additions and 9 deletions
+20 -8
View File
@@ -131,21 +131,26 @@ def fetch_hk_eastmoney(codes):
results = {}
# 主通道:东方财富实时行情(逐股查询,港股仅~10只,<1秒完成。该API不支持批量
# 主通道:东方财富实时行情(逐股查询,2秒超时。连续失败则立即切腾讯兜底
consecutive_fail = 0
for code in hk_codes:
try:
if consecutive_fail >= 3:
break # 前3只都失败,东财不可用,直接切腾讯
url = (f"https://push2.eastmoney.com/api/qt/stock/get"
f"?secid=116.{code}"
f"&fields=f43,f170,f60,f57,f58"
f"&fltt=2")
req = urllib.request.Request(url, headers={"User-Agent": UA})
with urllib.request.urlopen(req, timeout=5) as r:
with urllib.request.urlopen(req, timeout=2) as r:
resp = json.loads(r.read().decode("utf-8"))
if resp.get("rc") != 0:
consecutive_fail += 1
continue
item = resp.get("data", {})
if not item:
consecutive_fail += 1
continue
price = float(item.get("f43", 0)) if item.get("f43") else 0
prev_close = float(item.get("f60", 0)) if item.get("f60") else 0
@@ -153,10 +158,12 @@ def fetch_hk_eastmoney(codes):
change_pct = str(item.get("f170", "0"))
if price > 0:
results[code] = (price, change, change_pct)
time.sleep(0.1) # 防止触发东财反爬(逐股查询,不支持批量)
consecutive_fail = 0 # 成功后重置计数
time.sleep(0.1)
except Exception as e:
consecutive_fail += 1
if consecutive_fail <= 2:
print(f"⚠️ 东方财富 {code} 拉取失败: {e}", file=sys.stderr)
continue
# Fallback: 腾讯 qt.gtimg.cn15分钟延迟)
missing = [c for c in hk_codes if c not in results]
@@ -248,12 +255,14 @@ def refresh_data_prices():
if s['code'] in prices:
price, _, change_pct = prices[s['code']]
if price > 0:
# 港股API返回HKD直接存HKD原值。calc_total_mv统一CNY折算
old = s.get('price', 0)
# 港股API返回HKD需转CNY。系统统一CNY标价
if is_hk_stock(s['code']):
price = round(price * HK_RATE, 2)
old = s.get('price') or 0
if abs(old - price) > 0.001:
s['price'] = round(price, 2)
s['change_pct'] = float(change_pct) if change_pct else 0
s['currency'] = 'HKD' if is_hk_stock(s['code']) else 'CNY'
s['currency'] = 'CNY'
updated += 1
changed = True
if changed:
@@ -297,7 +306,10 @@ def refresh_data_prices():
if s['code'] in prices:
price, _, change_pct = prices[s['code']]
if price > 0:
old = s.get('price', 0)
# 港股:API返回HKD,需转CNY
if is_hk_stock(s['code']):
price = round(price * HK_RATE, 2)
old = s.get('price') or 0
if abs(old - price) > 0.001:
s['price'] = round(price, 2)
s['change_pct'] = float(change_pct) if change_pct else 0
+30
View File
@@ -0,0 +1,30 @@
"""Fix portfolio_summary directly from holdings table"""
import sqlite3, sys
sys.path.insert(0, '/home/hmo/MoFin')
from mo_models import calc_total_mv, calc_total_assets, calc_position_pct
from datetime import datetime
db = sqlite3.connect('/home/hmo/web-dashboard/data/mofin.db')
# Get holdings
rows = db.execute("SELECT code, name, shares, cost, price, market_value, change_pct, currency, position_pct FROM holdings WHERE is_active=1").fetchall()
holdings = [dict(zip(['code','name','shares','cost','price','market_value','change_pct','currency','position_pct'], r)) for r in rows]
# Get cash/summary
sr = db.execute("SELECT cash, frozen_cash FROM portfolio_summary WHERE id=1").fetchone()
cash = sr[0] or 0
frozen = sr[1] or 0
pf = {'holdings': holdings, 'cash': cash, 'frozen_cash': frozen}
mv = calc_total_mv(holdings)
ta = calc_total_assets(pf)
pp = calc_position_pct(pf)
pnl = sum((h['price'] or 0) * (h['shares'] or 0) - (h['cost'] or 0) * (h['shares'] or 0) for h in holdings)
print(f"mv={mv} ta={ta} pnl={pnl} pp={pp}%")
db.execute("UPDATE portfolio_summary SET total_mv=?, total_assets=?, total_pnl=?, position_pct=?, updated_at=? WHERE id=1",
(mv, ta, pnl, pp, datetime.now().strftime('%Y-%m-%d %H:%M:%S')))
db.commit()
print("portfolio_summary updated")
db.close()