持仓导入完成:holding.xls→SQLite+portfolio.json全同步
1. 从 ~/stocks/holding.xls 导入25只持仓(14A/11H) 2. 同时写入 portfolio.json + SQLite holdings 表 3. stale_push_wlin 现金来源从 stale_report 改为 portfolio.json 4. portfolio.json 增加 total_assets 字段兼容 stale_detector 5. 导入脚本已规范化为 MoFin/scripts/import_holding_xls.py 用法:python3 import_holding_xls.py [--cash 金额] 6. 全量策略重评+决策树重建立即执行 Dad下次更新holding.xls后跑: cd MoFin && python3 scripts/import_holding_xls.py
This commit is contained in:
+62
-62
@@ -315,18 +315,18 @@
|
|||||||
{
|
{
|
||||||
"code": "000711",
|
"code": "000711",
|
||||||
"name": "ST京蓝",
|
"name": "ST京蓝",
|
||||||
"price": 5.16,
|
"price": 5.0,
|
||||||
"cost": 0,
|
"cost": 0,
|
||||||
"shares": 0,
|
"shares": 0,
|
||||||
"avg_price": 0,
|
"avg_price": 0,
|
||||||
"action": "盈利持有 | ⚠️盈亏比不足1:1.5,不建议买入 | 目标5.38 | 止损4.4 | 买入区5.06~5.11 | 信号:观望",
|
"action": "盈利持有 | 目标5.38 | 止损4.82 | 买入区4.9~5.04 | 信号:观望",
|
||||||
"stop_loss": 4.4,
|
"stop_loss": 4.82,
|
||||||
"entry_low": 5.06,
|
"entry_low": 4.9,
|
||||||
"entry_high": 5.11,
|
"entry_high": 5.04,
|
||||||
"tech_snapshot": "形态:带下影阳线/neutral 量价:主动卖盘占优 强撑:4.86 弱撑:4.93 弱压:5.32 强压:5.38",
|
"tech_snapshot": "形态:倒T线/射击之星/bearish 量价:主动卖盘占优 强撑:4.86 弱撑:4.82 弱压:5.21 强压:5.38",
|
||||||
"timing_signal": "观望",
|
"timing_signal": "观望",
|
||||||
"rr_ratio": 0.96,
|
"rr_ratio": 2.11,
|
||||||
"status": "review",
|
"status": "updated",
|
||||||
"note": "⚠️盈亏比不足1:1.5,不建议买入",
|
"note": "⚠️盈亏比不足1:1.5,不建议买入",
|
||||||
"timestamp": "2026-06-24 11:16",
|
"timestamp": "2026-06-24 11:16",
|
||||||
"updated_at": "2026-06-24 11:16",
|
"updated_at": "2026-06-24 11:16",
|
||||||
@@ -430,7 +430,7 @@
|
|||||||
{
|
{
|
||||||
"id": "000711_stop_loss",
|
"id": "000711_stop_loss",
|
||||||
"condition": {
|
"condition": {
|
||||||
"price": "<4.4"
|
"price": "<4.82"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "sell",
|
"type": "sell",
|
||||||
@@ -447,13 +447,13 @@
|
|||||||
"id": "000711_buy_dip",
|
"id": "000711_buy_dip",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "weak_consolidation",
|
"scenario": "weak_consolidation",
|
||||||
"price": "<=5.11",
|
"price": "<=5.04",
|
||||||
"price_lower": ">=5.06"
|
"price_lower": ">=4.9"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "buy",
|
"type": "buy",
|
||||||
"amount": "normal",
|
"amount": "normal",
|
||||||
"limit": 5.06,
|
"limit": 4.9,
|
||||||
"reason": "回调支撑买入"
|
"reason": "回调支撑买入"
|
||||||
},
|
},
|
||||||
"priority": 1,
|
"priority": 1,
|
||||||
@@ -743,17 +743,17 @@
|
|||||||
{
|
{
|
||||||
"code": "002594",
|
"code": "002594",
|
||||||
"name": "比亚迪",
|
"name": "比亚迪",
|
||||||
"price": 82.82,
|
"price": 82.79,
|
||||||
"cost": 0,
|
"cost": 0,
|
||||||
"shares": 0,
|
"shares": 0,
|
||||||
"avg_price": 0,
|
"avg_price": 0,
|
||||||
"action": "盈利持有 | 目标92.55 | 止损80.34 | 买入区81.16~84.48 | 信号:观望",
|
"action": "盈利持有 | 目标92.54 | 止损80.31 | 买入区81.13~84.45 | 信号:观望",
|
||||||
"stop_loss": 80.34,
|
"stop_loss": 80.31,
|
||||||
"entry_low": 81.16,
|
"entry_low": 81.13,
|
||||||
"entry_high": 84.48,
|
"entry_high": 84.45,
|
||||||
"tech_snapshot": "形态:光头光脚阴线/bearish 量价:主动卖盘占优 强撑:76.5 弱撑:82.17 弱压:85.0 强压:92.55",
|
"tech_snapshot": "形态:光头光脚阴线/bearish 量价:主动卖盘占优 强撑:76.5 弱撑:82.15 弱压:85.0 强压:92.54",
|
||||||
"timing_signal": "观望",
|
"timing_signal": "观望",
|
||||||
"rr_ratio": 3.92,
|
"rr_ratio": 3.93,
|
||||||
"status": "updated",
|
"status": "updated",
|
||||||
"note": "",
|
"note": "",
|
||||||
"timestamp": "2026-06-24 11:16",
|
"timestamp": "2026-06-24 11:16",
|
||||||
@@ -765,7 +765,7 @@
|
|||||||
"position_advice": "减仓或观望",
|
"position_advice": "减仓或观望",
|
||||||
"time_horizon": "观望",
|
"time_horizon": "观望",
|
||||||
"created_at": "2026-06-18 17:15",
|
"created_at": "2026-06-18 17:15",
|
||||||
"take_profit": 92.55,
|
"take_profit": 92.54,
|
||||||
"updated_reason": "技术面重评: 止损77.92→77.93 | 形态:光头光脚阴线/bearish 量价:主动卖盘占优 强撑:76.5 弱撑:82.17 弱压:85.0 强压:92.5",
|
"updated_reason": "技术面重评: 止损77.92→77.93 | 形态:光头光脚阴线/bearish 量价:主动卖盘占优 强撑:76.5 弱撑:82.17 弱压:85.0 强压:92.5",
|
||||||
"changelog": [
|
"changelog": [
|
||||||
{
|
{
|
||||||
@@ -865,7 +865,7 @@
|
|||||||
{
|
{
|
||||||
"id": "002594_stop_loss",
|
"id": "002594_stop_loss",
|
||||||
"condition": {
|
"condition": {
|
||||||
"price": "<80.34"
|
"price": "<80.31"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "sell",
|
"type": "sell",
|
||||||
@@ -882,13 +882,13 @@
|
|||||||
"id": "002594_buy_dip",
|
"id": "002594_buy_dip",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "weak_consolidation",
|
"scenario": "weak_consolidation",
|
||||||
"price": "<=84.48",
|
"price": "<=84.45",
|
||||||
"price_lower": ">=81.16"
|
"price_lower": ">=81.13"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "buy",
|
"type": "buy",
|
||||||
"amount": "normal",
|
"amount": "normal",
|
||||||
"limit": 81.16,
|
"limit": 81.13,
|
||||||
"reason": "回调支撑买入"
|
"reason": "回调支撑买入"
|
||||||
},
|
},
|
||||||
"priority": 1,
|
"priority": 1,
|
||||||
@@ -901,7 +901,7 @@
|
|||||||
"id": "002594_breakout_chase",
|
"id": "002594_breakout_chase",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "bullish_recovery",
|
"scenario": "bullish_recovery",
|
||||||
"price": ">=92.55"
|
"price": ">=92.54"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "buy",
|
"type": "buy",
|
||||||
@@ -935,7 +935,7 @@
|
|||||||
{
|
{
|
||||||
"id": "002594_take_profit",
|
"id": "002594_take_profit",
|
||||||
"condition": {
|
"condition": {
|
||||||
"price": ">=92.55"
|
"price": ">=92.54"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "sell",
|
"type": "sell",
|
||||||
@@ -5192,17 +5192,17 @@
|
|||||||
{
|
{
|
||||||
"code": "300124",
|
"code": "300124",
|
||||||
"name": "汇川技术",
|
"name": "汇川技术",
|
||||||
"price": 65.87,
|
"price": 66.01,
|
||||||
"cost": 0,
|
"cost": 0,
|
||||||
"shares": 0,
|
"shares": 0,
|
||||||
"avg_price": 0,
|
"avg_price": 0,
|
||||||
"action": "盈利持有 | 目标72.99 | 止损63.89 | 买入区64.55~67.19 | 信号:观望",
|
"action": "盈利持有 | 目标73.03 | 止损64.03 | 买入区64.69~67.33 | 信号:观望",
|
||||||
"stop_loss": 63.89,
|
"stop_loss": 64.03,
|
||||||
"entry_low": 64.55,
|
"entry_low": 64.69,
|
||||||
"entry_high": 67.19,
|
"entry_high": 67.33,
|
||||||
"tech_snapshot": "形态:带上影阴线/bearish 量价:主动卖盘占优 强撑:59.75 弱撑:65.24 弱压:66.99 强压:72.99",
|
"tech_snapshot": "形态:带上影阴线/bearish 量价:主动卖盘占优 强撑:59.79 弱撑:65.34 弱压:67.09 强压:73.03",
|
||||||
"timing_signal": "观望",
|
"timing_signal": "观望",
|
||||||
"rr_ratio": 3.6,
|
"rr_ratio": 3.55,
|
||||||
"status": "updated",
|
"status": "updated",
|
||||||
"note": "",
|
"note": "",
|
||||||
"timestamp": "2026-06-24 11:17",
|
"timestamp": "2026-06-24 11:17",
|
||||||
@@ -5214,7 +5214,7 @@
|
|||||||
"position_advice": "减仓或观望",
|
"position_advice": "减仓或观望",
|
||||||
"time_horizon": "观望",
|
"time_horizon": "观望",
|
||||||
"created_at": "2026-06-18 17:15",
|
"created_at": "2026-06-18 17:15",
|
||||||
"take_profit": 72.99,
|
"take_profit": 73.03,
|
||||||
"updated_reason": "技术面重评: 止损61.97→62.13 | 形态:带上影阴线/bearish 量价:主动卖盘占优 强撑:59.78 弱撑:65.32 弱压:67.07 强压:73.",
|
"updated_reason": "技术面重评: 止损61.97→62.13 | 形态:带上影阴线/bearish 量价:主动卖盘占优 强撑:59.78 弱撑:65.32 弱压:67.07 强压:73.",
|
||||||
"changelog": [
|
"changelog": [
|
||||||
{
|
{
|
||||||
@@ -5307,7 +5307,7 @@
|
|||||||
{
|
{
|
||||||
"id": "300124_stop_loss",
|
"id": "300124_stop_loss",
|
||||||
"condition": {
|
"condition": {
|
||||||
"price": "<63.89"
|
"price": "<64.03"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "sell",
|
"type": "sell",
|
||||||
@@ -5324,13 +5324,13 @@
|
|||||||
"id": "300124_buy_dip",
|
"id": "300124_buy_dip",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "weak_consolidation",
|
"scenario": "weak_consolidation",
|
||||||
"price": "<=67.19",
|
"price": "<=67.33",
|
||||||
"price_lower": ">=64.55"
|
"price_lower": ">=64.69"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "buy",
|
"type": "buy",
|
||||||
"amount": "normal",
|
"amount": "normal",
|
||||||
"limit": 64.55,
|
"limit": 64.69,
|
||||||
"reason": "回调支撑买入"
|
"reason": "回调支撑买入"
|
||||||
},
|
},
|
||||||
"priority": 1,
|
"priority": 1,
|
||||||
@@ -5343,7 +5343,7 @@
|
|||||||
"id": "300124_breakout_chase",
|
"id": "300124_breakout_chase",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "bullish_recovery",
|
"scenario": "bullish_recovery",
|
||||||
"price": ">=72.99"
|
"price": ">=73.03"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "buy",
|
"type": "buy",
|
||||||
@@ -5377,7 +5377,7 @@
|
|||||||
{
|
{
|
||||||
"id": "300124_take_profit",
|
"id": "300124_take_profit",
|
||||||
"condition": {
|
"condition": {
|
||||||
"price": ">=72.99"
|
"price": ">=73.03"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "sell",
|
"type": "sell",
|
||||||
@@ -7264,11 +7264,11 @@
|
|||||||
"cost": 0,
|
"cost": 0,
|
||||||
"shares": 0,
|
"shares": 0,
|
||||||
"avg_price": 0,
|
"avg_price": 0,
|
||||||
"action": "盈利持有 | 目标53.71 | 止损47.9 | 买入区48.39~50.22 | 信号:观望",
|
"action": "盈利持有 | 目标53.72 | 止损47.9 | 买入区48.39~50.23 | 信号:观望",
|
||||||
"stop_loss": 47.9,
|
"stop_loss": 47.9,
|
||||||
"entry_low": 48.39,
|
"entry_low": 48.39,
|
||||||
"entry_high": 50.22,
|
"entry_high": 50.23,
|
||||||
"tech_snapshot": "形态:光头光脚阴线/bearish 量价:主动卖盘占优 强撑:45.81 弱撑:48.65 弱压:50.5 强压:53.71",
|
"tech_snapshot": "形态:光头光脚阴线/bearish 量价:主动卖盘占优 强撑:45.82 弱撑:48.65 弱压:50.5 强压:53.72",
|
||||||
"timing_signal": "观望",
|
"timing_signal": "观望",
|
||||||
"rr_ratio": 2.93,
|
"rr_ratio": 2.93,
|
||||||
"status": "updated",
|
"status": "updated",
|
||||||
@@ -7282,7 +7282,7 @@
|
|||||||
"position_advice": "减仓或观望",
|
"position_advice": "减仓或观望",
|
||||||
"time_horizon": "观望",
|
"time_horizon": "观望",
|
||||||
"created_at": "2026-06-22 11:50",
|
"created_at": "2026-06-22 11:50",
|
||||||
"take_profit": 53.71,
|
"take_profit": 53.72,
|
||||||
"changelog": [
|
"changelog": [
|
||||||
{
|
{
|
||||||
"date": "2026-06-22 12:01",
|
"date": "2026-06-22 12:01",
|
||||||
@@ -7335,7 +7335,7 @@
|
|||||||
"id": "601318_buy_dip",
|
"id": "601318_buy_dip",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "weak_consolidation",
|
"scenario": "weak_consolidation",
|
||||||
"price": "<=50.22",
|
"price": "<=50.23",
|
||||||
"price_lower": ">=48.39"
|
"price_lower": ">=48.39"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
@@ -7354,7 +7354,7 @@
|
|||||||
"id": "601318_breakout_chase",
|
"id": "601318_breakout_chase",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "bullish_recovery",
|
"scenario": "bullish_recovery",
|
||||||
"price": ">=53.71"
|
"price": ">=53.72"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "buy",
|
"type": "buy",
|
||||||
@@ -7388,7 +7388,7 @@
|
|||||||
{
|
{
|
||||||
"id": "601318_take_profit",
|
"id": "601318_take_profit",
|
||||||
"condition": {
|
"condition": {
|
||||||
"price": ">=53.71"
|
"price": ">=53.72"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "sell",
|
"type": "sell",
|
||||||
@@ -8426,17 +8426,17 @@
|
|||||||
{
|
{
|
||||||
"code": "688795",
|
"code": "688795",
|
||||||
"name": "摩尔线程-U",
|
"name": "摩尔线程-U",
|
||||||
"price": 678.5,
|
"price": 679.95,
|
||||||
"cost": 0,
|
"cost": 0,
|
||||||
"shares": 0,
|
"shares": 0,
|
||||||
"avg_price": 0,
|
"avg_price": 0,
|
||||||
"action": "盈利持有 | 目标816.82 | 止损658.14 | 买入区664.93~692.07 | 信号:观望",
|
"action": "盈利持有 | 目标817.3 | 止损659.55 | 买入区666.35~693.55 | 信号:观望",
|
||||||
"stop_loss": 658.14,
|
"stop_loss": 659.55,
|
||||||
"entry_low": 664.93,
|
"entry_low": 666.35,
|
||||||
"entry_high": 692.07,
|
"entry_high": 693.55,
|
||||||
"tech_snapshot": "形态:带下影阴线/neutral 量价:主动卖盘占优 强撑:563.22 弱撑:665.33 弱压:704.03 强压:816.82",
|
"tech_snapshot": "形态:带下影阴线/neutral 量价:主动卖盘占优 强撑:563.22 弱撑:666.3 弱压:704.03 强压:817.3",
|
||||||
"timing_signal": "观望",
|
"timing_signal": "观望",
|
||||||
"rr_ratio": 6.79,
|
"rr_ratio": 6.73,
|
||||||
"status": "updated",
|
"status": "updated",
|
||||||
"note": "",
|
"note": "",
|
||||||
"timestamp": "2026-06-24 11:17",
|
"timestamp": "2026-06-24 11:17",
|
||||||
@@ -8448,7 +8448,7 @@
|
|||||||
"position_advice": "正常配置",
|
"position_advice": "正常配置",
|
||||||
"time_horizon": "数月~1年",
|
"time_horizon": "数月~1年",
|
||||||
"created_at": "2026-06-18 17:15",
|
"created_at": "2026-06-18 17:15",
|
||||||
"take_profit": 816.82,
|
"take_profit": 817.3,
|
||||||
"updated_reason": "技术面重评: 止损635.13→637.94 | 形态:光头光脚阴线/bearish 量价:主动卖盘占优 强撑:563.22 弱撑:665.0 弱压:704.03 强压:",
|
"updated_reason": "技术面重评: 止损635.13→637.94 | 形态:光头光脚阴线/bearish 量价:主动卖盘占优 强撑:563.22 弱撑:665.0 弱压:704.03 强压:",
|
||||||
"changelog": [
|
"changelog": [
|
||||||
{
|
{
|
||||||
@@ -8541,7 +8541,7 @@
|
|||||||
{
|
{
|
||||||
"id": "688795_stop_loss",
|
"id": "688795_stop_loss",
|
||||||
"condition": {
|
"condition": {
|
||||||
"price": "<658.14"
|
"price": "<659.55"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "sell",
|
"type": "sell",
|
||||||
@@ -8558,13 +8558,13 @@
|
|||||||
"id": "688795_buy_dip",
|
"id": "688795_buy_dip",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "weak_consolidation",
|
"scenario": "weak_consolidation",
|
||||||
"price": "<=692.07",
|
"price": "<=693.55",
|
||||||
"price_lower": ">=664.93"
|
"price_lower": ">=666.35"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "buy",
|
"type": "buy",
|
||||||
"amount": "normal",
|
"amount": "normal",
|
||||||
"limit": 664.93,
|
"limit": 666.35,
|
||||||
"reason": "回调支撑买入"
|
"reason": "回调支撑买入"
|
||||||
},
|
},
|
||||||
"priority": 1,
|
"priority": 1,
|
||||||
@@ -8577,7 +8577,7 @@
|
|||||||
"id": "688795_breakout_chase",
|
"id": "688795_breakout_chase",
|
||||||
"condition": {
|
"condition": {
|
||||||
"scenario": "bullish_recovery",
|
"scenario": "bullish_recovery",
|
||||||
"price": ">=816.82"
|
"price": ">=817.3"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "buy",
|
"type": "buy",
|
||||||
@@ -8611,7 +8611,7 @@
|
|||||||
{
|
{
|
||||||
"id": "688795_take_profit",
|
"id": "688795_take_profit",
|
||||||
"condition": {
|
"condition": {
|
||||||
"price": ">=816.82"
|
"price": ">=817.3"
|
||||||
},
|
},
|
||||||
"action": {
|
"action": {
|
||||||
"type": "sell",
|
"type": "sell",
|
||||||
@@ -9065,5 +9065,5 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"total": 42,
|
"total": 42,
|
||||||
"regenerated_at": "2026-06-24 11:19"
|
"regenerated_at": "2026-06-24 11:21"
|
||||||
}
|
}
|
||||||
+47
-39
@@ -5,21 +5,22 @@
|
|||||||
"name": "德明利",
|
"name": "德明利",
|
||||||
"shares": 100,
|
"shares": 100,
|
||||||
"avail_shares": 100,
|
"avail_shares": 100,
|
||||||
"price": 790.0,
|
"price": 790.9,
|
||||||
"cost_price": 737.0374,
|
"cost_price": 737.0374,
|
||||||
"pl": 5096.26,
|
"pl": 5096.26,
|
||||||
"pl_pct": 6.91,
|
"pl_pct": 6.91,
|
||||||
"currency": "CNY",
|
"currency": "CNY",
|
||||||
"market_val": 78800.0,
|
"market_val": 78800.0,
|
||||||
"cost_amount": 73703.74,
|
"cost_amount": 73703.74,
|
||||||
"exchange_rate": 0.866
|
"exchange_rate": 0.866,
|
||||||
|
"change_pct": 4.75
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "01478",
|
"code": "01478",
|
||||||
"name": "丘钛科技",
|
"name": "丘钛科技",
|
||||||
"shares": 11000,
|
"shares": 11000,
|
||||||
"avail_shares": 11000,
|
"avail_shares": 11000,
|
||||||
"price": 7.57,
|
"price": 7.54,
|
||||||
"cost_price": 13.8089,
|
"cost_price": 13.8089,
|
||||||
"pl": -59438.6,
|
"pl": -59438.6,
|
||||||
"pl_pct": -45.18,
|
"pl_pct": -45.18,
|
||||||
@@ -27,14 +28,14 @@
|
|||||||
"market_val": 72120.15,
|
"market_val": 72120.15,
|
||||||
"cost_amount": 131558.75,
|
"cost_amount": 131558.75,
|
||||||
"exchange_rate": 0.8661,
|
"exchange_rate": 0.8661,
|
||||||
"change_pct": -3.93
|
"change_pct": -4.19
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "600739",
|
"code": "600739",
|
||||||
"name": "辽宁成大",
|
"name": "辽宁成大",
|
||||||
"shares": 6600,
|
"shares": 6600,
|
||||||
"avail_shares": 6600,
|
"avail_shares": 6600,
|
||||||
"price": 10.44,
|
"price": 10.45,
|
||||||
"cost_price": 12.2862,
|
"cost_price": 12.2862,
|
||||||
"pl": -11789.02,
|
"pl": -11789.02,
|
||||||
"pl_pct": -14.54,
|
"pl_pct": -14.54,
|
||||||
@@ -42,14 +43,14 @@
|
|||||||
"market_val": 69300.0,
|
"market_val": 69300.0,
|
||||||
"cost_amount": 81089.02,
|
"cost_amount": 81089.02,
|
||||||
"exchange_rate": 0.866,
|
"exchange_rate": 0.866,
|
||||||
"change_pct": -1.97
|
"change_pct": -1.88
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "601899",
|
"code": "601899",
|
||||||
"name": "紫金矿业",
|
"name": "紫金矿业",
|
||||||
"shares": 2400,
|
"shares": 2400,
|
||||||
"avail_shares": 2400,
|
"avail_shares": 2400,
|
||||||
"price": 27.38,
|
"price": 27.36,
|
||||||
"cost_price": 40.2685,
|
"cost_price": 40.2685,
|
||||||
"pl": -30572.29,
|
"pl": -30572.29,
|
||||||
"pl_pct": -31.63,
|
"pl_pct": -31.63,
|
||||||
@@ -63,21 +64,22 @@
|
|||||||
"name": "华恒生物",
|
"name": "华恒生物",
|
||||||
"shares": 2800,
|
"shares": 2800,
|
||||||
"avail_shares": 0,
|
"avail_shares": 0,
|
||||||
"price": 21.31,
|
"price": 21.35,
|
||||||
"cost_price": 21.5085,
|
"cost_price": 21.5085,
|
||||||
"pl": -135.75,
|
"pl": -135.75,
|
||||||
"pl_pct": -0.23,
|
"pl_pct": -0.23,
|
||||||
"currency": "CNY",
|
"currency": "CNY",
|
||||||
"market_val": 60088.0,
|
"market_val": 60088.0,
|
||||||
"cost_amount": 60223.75,
|
"cost_amount": 60223.75,
|
||||||
"exchange_rate": 0.866
|
"exchange_rate": 0.866,
|
||||||
|
"change_pct": -2.2
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "09988",
|
"code": "09988",
|
||||||
"name": "阿里巴巴-W",
|
"name": "阿里巴巴-W",
|
||||||
"shares": 700,
|
"shares": 700,
|
||||||
"avail_shares": 700,
|
"avail_shares": 700,
|
||||||
"price": 98.55,
|
"price": 98.6,
|
||||||
"cost_price": 126.1516,
|
"cost_price": 126.1516,
|
||||||
"pl": -16582.48,
|
"pl": -16582.48,
|
||||||
"pl_pct": -21.68,
|
"pl_pct": -21.68,
|
||||||
@@ -85,7 +87,7 @@
|
|||||||
"market_val": 59899.48,
|
"market_val": 59899.48,
|
||||||
"cost_amount": 76481.96,
|
"cost_amount": 76481.96,
|
||||||
"exchange_rate": 0.8661,
|
"exchange_rate": 0.8661,
|
||||||
"change_pct": -0.4
|
"change_pct": -0.35
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "603259",
|
"code": "603259",
|
||||||
@@ -106,7 +108,7 @@
|
|||||||
"name": "中芯国际",
|
"name": "中芯国际",
|
||||||
"shares": 300,
|
"shares": 300,
|
||||||
"avail_shares": 300,
|
"avail_shares": 300,
|
||||||
"price": 149.89,
|
"price": 150.64,
|
||||||
"cost_price": 126.0681,
|
"cost_price": 126.0681,
|
||||||
"pl": 7173.58,
|
"pl": 7173.58,
|
||||||
"pl_pct": 18.97,
|
"pl_pct": 18.97,
|
||||||
@@ -114,7 +116,7 @@
|
|||||||
"market_val": 44994.0,
|
"market_val": 44994.0,
|
||||||
"cost_amount": 37820.42,
|
"cost_amount": 37820.42,
|
||||||
"exchange_rate": 0.866,
|
"exchange_rate": 0.866,
|
||||||
"change_pct": 5.78
|
"change_pct": 6.31
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "01888",
|
"code": "01888",
|
||||||
@@ -150,7 +152,7 @@
|
|||||||
"name": "中银香港",
|
"name": "中银香港",
|
||||||
"shares": 1000,
|
"shares": 1000,
|
||||||
"avail_shares": 1000,
|
"avail_shares": 1000,
|
||||||
"price": 46.22,
|
"price": 46.26,
|
||||||
"cost_price": 43.7892,
|
"cost_price": 43.7892,
|
||||||
"pl": 2053.39,
|
"pl": 2053.39,
|
||||||
"pl_pct": 5.41,
|
"pl_pct": 5.41,
|
||||||
@@ -158,14 +160,14 @@
|
|||||||
"market_val": 39979.18,
|
"market_val": 39979.18,
|
||||||
"cost_amount": 37925.79,
|
"cost_amount": 37925.79,
|
||||||
"exchange_rate": 0.8661,
|
"exchange_rate": 0.8661,
|
||||||
"change_pct": -1.66
|
"change_pct": -1.62
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "300750",
|
"code": "300750",
|
||||||
"name": "宁德时代",
|
"name": "宁德时代",
|
||||||
"shares": 100,
|
"shares": 100,
|
||||||
"avail_shares": 100,
|
"avail_shares": 100,
|
||||||
"price": 391.09,
|
"price": 391.19,
|
||||||
"cost_price": 401.7803,
|
"cost_price": 401.7803,
|
||||||
"pl": -1063.03,
|
"pl": -1063.03,
|
||||||
"pl_pct": -2.65,
|
"pl_pct": -2.65,
|
||||||
@@ -173,21 +175,22 @@
|
|||||||
"market_val": 39115.0,
|
"market_val": 39115.0,
|
||||||
"cost_amount": 40178.03,
|
"cost_amount": 40178.03,
|
||||||
"exchange_rate": 0.866,
|
"exchange_rate": 0.866,
|
||||||
"change_pct": -0.36
|
"change_pct": -0.34
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "01211",
|
"code": "01211",
|
||||||
"name": "比亚迪股份",
|
"name": "比亚迪股份",
|
||||||
"shares": 600,
|
"shares": 600,
|
||||||
"avail_shares": 600,
|
"avail_shares": 600,
|
||||||
"price": 75.0,
|
"price": 74.85,
|
||||||
"cost_price": 105.0542,
|
"cost_price": 105.0542,
|
||||||
"pl": -15539.97,
|
"pl": -15539.97,
|
||||||
"pl_pct": -28.47,
|
"pl_pct": -28.47,
|
||||||
"currency": "HKD",
|
"currency": "HKD",
|
||||||
"market_val": 39052.45,
|
"market_val": 39052.45,
|
||||||
"cost_amount": 54592.42,
|
"cost_amount": 54592.42,
|
||||||
"exchange_rate": 0.8661
|
"exchange_rate": 0.8661,
|
||||||
|
"change_pct": -1.25
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "00700",
|
"code": "00700",
|
||||||
@@ -209,7 +212,7 @@
|
|||||||
"name": "中芯国际",
|
"name": "中芯国际",
|
||||||
"shares": 500,
|
"shares": 500,
|
||||||
"avail_shares": 500,
|
"avail_shares": 500,
|
||||||
"price": 83.15,
|
"price": 83.25,
|
||||||
"cost_price": 76.0724,
|
"cost_price": 76.0724,
|
||||||
"pl": 3151.55,
|
"pl": 3151.55,
|
||||||
"pl_pct": 9.57,
|
"pl_pct": 9.57,
|
||||||
@@ -217,28 +220,29 @@
|
|||||||
"market_val": 36094.72,
|
"market_val": 36094.72,
|
||||||
"cost_amount": 32943.17,
|
"cost_amount": 32943.17,
|
||||||
"exchange_rate": 0.8661,
|
"exchange_rate": 0.8661,
|
||||||
"change_pct": 6.81
|
"change_pct": 6.62
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "09868",
|
"code": "09868",
|
||||||
"name": "小鹏集团-W",
|
"name": "小鹏集团-W",
|
||||||
"shares": 700,
|
"shares": 700,
|
||||||
"avail_shares": 700,
|
"avail_shares": 700,
|
||||||
"price": 50.2,
|
"price": 50.1,
|
||||||
"cost_price": 51.3644,
|
"cost_price": 51.3644,
|
||||||
"pl": -675.62,
|
"pl": -675.62,
|
||||||
"pl_pct": -2.17,
|
"pl_pct": -2.17,
|
||||||
"currency": "HKD",
|
"currency": "HKD",
|
||||||
"market_val": 30465.07,
|
"market_val": 30465.07,
|
||||||
"cost_amount": 31140.69,
|
"cost_amount": 31140.69,
|
||||||
"exchange_rate": 0.8661
|
"exchange_rate": 0.8661,
|
||||||
|
"change_pct": 1.5
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "600036",
|
"code": "600036",
|
||||||
"name": "招商银行",
|
"name": "招商银行",
|
||||||
"shares": 800,
|
"shares": 800,
|
||||||
"avail_shares": 800,
|
"avail_shares": 800,
|
||||||
"price": 37.06,
|
"price": 37.05,
|
||||||
"cost_price": 38.1582,
|
"cost_price": 38.1582,
|
||||||
"pl": -846.53,
|
"pl": -846.53,
|
||||||
"pl_pct": -2.77,
|
"pl_pct": -2.77,
|
||||||
@@ -246,14 +250,14 @@
|
|||||||
"market_val": 29680.0,
|
"market_val": 29680.0,
|
||||||
"cost_amount": 30526.53,
|
"cost_amount": 30526.53,
|
||||||
"exchange_rate": 0.866,
|
"exchange_rate": 0.866,
|
||||||
"change_pct": -0.91
|
"change_pct": -0.94
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "300548",
|
"code": "300548",
|
||||||
"name": "长芯博创",
|
"name": "长芯博创",
|
||||||
"shares": 100,
|
"shares": 100,
|
||||||
"avail_shares": 100,
|
"avail_shares": 100,
|
||||||
"price": 276.12,
|
"price": 276.79,
|
||||||
"cost_price": 231.46,
|
"cost_price": 231.46,
|
||||||
"pl": 4766.0,
|
"pl": 4766.0,
|
||||||
"pl_pct": 20.59,
|
"pl_pct": 20.59,
|
||||||
@@ -261,14 +265,14 @@
|
|||||||
"market_val": 27912.0,
|
"market_val": 27912.0,
|
||||||
"cost_amount": 23146.0,
|
"cost_amount": 23146.0,
|
||||||
"exchange_rate": 0.866,
|
"exchange_rate": 0.866,
|
||||||
"change_pct": -3.46
|
"change_pct": -3.23
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "02318",
|
"code": "02318",
|
||||||
"name": "中国平安",
|
"name": "中国平安",
|
||||||
"shares": 500,
|
"shares": 500,
|
||||||
"avail_shares": 500,
|
"avail_shares": 500,
|
||||||
"price": 52.7,
|
"price": 52.8,
|
||||||
"cost_price": 54.8199,
|
"cost_price": 54.8199,
|
||||||
"pl": -896.35,
|
"pl": -896.35,
|
||||||
"pl_pct": -3.78,
|
"pl_pct": -3.78,
|
||||||
@@ -276,7 +280,7 @@
|
|||||||
"market_val": 22843.39,
|
"market_val": 22843.39,
|
||||||
"cost_amount": 23739.74,
|
"cost_amount": 23739.74,
|
||||||
"exchange_rate": 0.8661,
|
"exchange_rate": 0.8661,
|
||||||
"change_pct": -1.68
|
"change_pct": -1.49
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "300035",
|
"code": "300035",
|
||||||
@@ -290,14 +294,15 @@
|
|||||||
"currency": "CNY",
|
"currency": "CNY",
|
||||||
"market_val": 22442.0,
|
"market_val": 22442.0,
|
||||||
"cost_amount": 31207.91,
|
"cost_amount": 31207.91,
|
||||||
"exchange_rate": 0.866
|
"exchange_rate": 0.866,
|
||||||
|
"change_pct": -2.08
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "000700",
|
"code": "000700",
|
||||||
"name": "模塑科技",
|
"name": "模塑科技",
|
||||||
"shares": 1400,
|
"shares": 1400,
|
||||||
"avail_shares": 1400,
|
"avail_shares": 1400,
|
||||||
"price": 14.53,
|
"price": 14.52,
|
||||||
"cost_price": 14.8336,
|
"cost_price": 14.8336,
|
||||||
"pl": -327.0,
|
"pl": -327.0,
|
||||||
"pl_pct": -1.57,
|
"pl_pct": -1.57,
|
||||||
@@ -305,14 +310,14 @@
|
|||||||
"market_val": 20440.0,
|
"market_val": 20440.0,
|
||||||
"cost_amount": 20767.0,
|
"cost_amount": 20767.0,
|
||||||
"exchange_rate": 0.866,
|
"exchange_rate": 0.866,
|
||||||
"change_pct": -3.0
|
"change_pct": -3.07
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "518880",
|
"code": "518880",
|
||||||
"name": "黄金ETF华安",
|
"name": "黄金ETF华安",
|
||||||
"shares": 2400,
|
"shares": 2400,
|
||||||
"avail_shares": 2400,
|
"avail_shares": 2400,
|
||||||
"price": 8.43,
|
"price": 8.44,
|
||||||
"cost_price": 12.1915,
|
"cost_price": 12.1915,
|
||||||
"pl": -8982.04,
|
"pl": -8982.04,
|
||||||
"pl_pct": -30.7,
|
"pl_pct": -30.7,
|
||||||
@@ -320,21 +325,22 @@
|
|||||||
"market_val": 20277.6,
|
"market_val": 20277.6,
|
||||||
"cost_amount": 29259.64,
|
"cost_amount": 29259.64,
|
||||||
"exchange_rate": 0.866,
|
"exchange_rate": 0.866,
|
||||||
"change_pct": -1.21
|
"change_pct": -1.16
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "01088",
|
"code": "01088",
|
||||||
"name": "中国神华",
|
"name": "中国神华",
|
||||||
"shares": 500,
|
"shares": 500,
|
||||||
"avail_shares": 500,
|
"avail_shares": 500,
|
||||||
"price": 41.92,
|
"price": 41.84,
|
||||||
"cost_price": 45.974,
|
"cost_price": 45.974,
|
||||||
"pl": -1764.27,
|
"pl": -1764.27,
|
||||||
"pl_pct": -8.86,
|
"pl_pct": -8.86,
|
||||||
"currency": "HKD",
|
"currency": "HKD",
|
||||||
"market_val": 18144.8,
|
"market_val": 18144.8,
|
||||||
"cost_amount": 19909.06,
|
"cost_amount": 19909.06,
|
||||||
"exchange_rate": 0.8661
|
"exchange_rate": 0.8661,
|
||||||
|
"change_pct": -0.52
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"code": "600563",
|
"code": "600563",
|
||||||
@@ -355,14 +361,15 @@
|
|||||||
"name": "双一科技",
|
"name": "双一科技",
|
||||||
"shares": 400,
|
"shares": 400,
|
||||||
"avail_shares": 400,
|
"avail_shares": 400,
|
||||||
"price": 22.7,
|
"price": 22.68,
|
||||||
"cost_price": 27.178,
|
"cost_price": 27.178,
|
||||||
"pl": -1759.2,
|
"pl": -1759.2,
|
||||||
"pl_pct": -16.18,
|
"pl_pct": -16.18,
|
||||||
"currency": "CNY",
|
"currency": "CNY",
|
||||||
"market_val": 9112.0,
|
"market_val": 9112.0,
|
||||||
"cost_amount": 10871.2,
|
"cost_amount": 10871.2,
|
||||||
"exchange_rate": 0.866
|
"exchange_rate": 0.866,
|
||||||
|
"change_pct": -3.98
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"cash": 80476,
|
"cash": 80476,
|
||||||
@@ -370,5 +377,6 @@
|
|||||||
"total_pl": -164810.3,
|
"total_pl": -164810.3,
|
||||||
"position_pct": 92.0,
|
"position_pct": 92.0,
|
||||||
"updated_at": "2026-06-24 11:47",
|
"updated_at": "2026-06-24 11:47",
|
||||||
"source": "/home/hmo/stocks/holding.xls"
|
"source": "/home/hmo/stocks/holding.xls",
|
||||||
|
"total_assets": 1010675.93
|
||||||
}
|
}
|
||||||
@@ -108,7 +108,7 @@
|
|||||||
"date": "2026-06-24",
|
"date": "2026-06-24",
|
||||||
"high": 88.32,
|
"high": 88.32,
|
||||||
"low": 82.78,
|
"low": 82.78,
|
||||||
"close": 82.82
|
"close": 82.79
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"00700": [
|
"00700": [
|
||||||
@@ -1284,7 +1284,7 @@
|
|||||||
"date": "2026-06-24",
|
"date": "2026-06-24",
|
||||||
"high": 68.8,
|
"high": 68.8,
|
||||||
"low": 65.74,
|
"low": 65.74,
|
||||||
"close": 65.87
|
"close": 66.01
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"300548": [
|
"300548": [
|
||||||
@@ -2316,7 +2316,7 @@
|
|||||||
"date": "2026-06-24",
|
"date": "2026-06-24",
|
||||||
"high": 745.2,
|
"high": 745.2,
|
||||||
"low": 660.01,
|
"low": 660.01,
|
||||||
"close": 678.5
|
"close": 679.95
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"688802": [
|
"688802": [
|
||||||
@@ -2466,7 +2466,7 @@
|
|||||||
"date": "2026-06-24",
|
"date": "2026-06-24",
|
||||||
"high": 5.38,
|
"high": 5.38,
|
||||||
"low": 4.86,
|
"low": 4.86,
|
||||||
"close": 5.16
|
"close": 5.0
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"688630": [
|
"688630": [
|
||||||
@@ -2610,7 +2610,7 @@
|
|||||||
"date": "2026-06-24",
|
"date": "2026-06-24",
|
||||||
"high": 52.43,
|
"high": 52.43,
|
||||||
"low": 49.03,
|
"low": 49.03,
|
||||||
"close": 49.38
|
"close": 49.39
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"002171": [
|
"002171": [
|
||||||
|
|||||||
@@ -1,19 +1,16 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
"""
|
"""
|
||||||
import_holding_xls.py — 从券商导出文件 holding.xls 导入持仓数据
|
import_holding_xls.py — 从holding.xls导入持仓到SQLite + portfolio.json
|
||||||
|
|
||||||
读取 /home/hmo/stocks/holding.xls(TSV格式,GBK编码)
|
用法:python3 import_holding_xls.py [--cash 现金金额]
|
||||||
写入 /home/hmo/web-dashboard/data/portfolio.json
|
|
||||||
触发 per_stock_reassess 全量重评(更新 decisions.json + 决策树)
|
|
||||||
|
|
||||||
用法:python3 import_holding_xls.py
|
|
||||||
"""
|
"""
|
||||||
import csv, json, sys, subprocess
|
import csv, json, sys, subprocess, sqlite3
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
|
||||||
STOCKS_FILE = "/home/hmo/stocks/holding.xls"
|
STOCKS_FILE = "/home/hmo/stocks/holding.xls"
|
||||||
PORTFOLIO_PATH = "/home/hmo/web-dashboard/data/portfolio.json"
|
PORTFOLIO_PATH = "/home/hmo/web-dashboard/data/portfolio.json"
|
||||||
CASH = 80476 # 现金余额(手动更新或从其他源读取)
|
DB_PATH = "/home/hmo/web-dashboard/data/mofin.db"
|
||||||
|
CASH = 80476 # default cash
|
||||||
|
|
||||||
|
|
||||||
def clean_cell(v):
|
def clean_cell(v):
|
||||||
@@ -26,6 +23,12 @@ def clean_cell(v):
|
|||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
# Parse args
|
||||||
|
global CASH
|
||||||
|
for i, a in enumerate(sys.argv[1:]):
|
||||||
|
if a == '--cash' and i + 2 < len(sys.argv):
|
||||||
|
CASH = float(sys.argv[i + 2])
|
||||||
|
|
||||||
with open(STOCKS_FILE, 'r', encoding='gbk') as f:
|
with open(STOCKS_FILE, 'r', encoding='gbk') as f:
|
||||||
reader = csv.reader(f, delimiter='\t')
|
reader = csv.reader(f, delimiter='\t')
|
||||||
rows = list(reader)
|
rows = list(reader)
|
||||||
@@ -34,73 +37,81 @@ def main():
|
|||||||
|
|
||||||
holdings = []
|
holdings = []
|
||||||
total_mv_cny = 0
|
total_mv_cny = 0
|
||||||
total_pl = 0
|
|
||||||
|
|
||||||
for r in rows[1:]:
|
for r in rows[1:]:
|
||||||
code = clean_cell(r[0])
|
code = clean_cell(r[0])
|
||||||
name = r[1].strip()
|
name = r[1].strip()
|
||||||
shares = int(clean_cell(r[2]))
|
shares = int(clean_cell(r[2]))
|
||||||
avail = int(clean_cell(r[3]))
|
|
||||||
|
|
||||||
price_raw = r[4].strip()
|
price_raw = r[4].strip()
|
||||||
currency = 'HKD' if '港币' in price_raw or '港' in r[10] else 'CNY'
|
currency = 'HKD' if '港币' in price_raw or '港' in r[10] else 'CNY'
|
||||||
price_str = price_raw.replace('港币', '').replace('港元', '').replace('港', '').strip()
|
price_str = price_raw.replace('港币', '').replace('港元', '').replace('港', '').strip()
|
||||||
price = float(price_str)
|
price = float(price_str)
|
||||||
|
|
||||||
cost_price = float(clean_cell(r[5]))
|
cost_price = float(clean_cell(r[5]))
|
||||||
pl = float(clean_cell(r[6]))
|
pl = float(clean_cell(r[6]))
|
||||||
pl_pct = float(clean_cell(r[7])) if r[7].strip() else 0.0
|
pl_pct = float(clean_cell(r[7])) if r[7].strip() else 0.0
|
||||||
mkt_val = float(clean_cell(r[11]))
|
mkt_val = float(clean_cell(r[11]))
|
||||||
cost_amount = float(clean_cell(r[15])) if r[15].strip() and r[15].strip() != '--' else 0
|
cost_amount = float(clean_cell(r[15])) if r[15].strip() and r[15].strip() != '--' else 0
|
||||||
|
|
||||||
rate_str = clean_cell(r[16])
|
rate_str = clean_cell(r[16])
|
||||||
rate = float(rate_str) if rate_str and rate_str != '--' else 0.866
|
rate = float(rate_str) if rate_str and rate_str != '--' else 0.866
|
||||||
|
|
||||||
holdings.append({
|
|
||||||
'code': code,
|
|
||||||
'name': name,
|
|
||||||
'shares': shares,
|
|
||||||
'avail_shares': avail,
|
|
||||||
'price': price,
|
|
||||||
'cost_price': cost_price,
|
|
||||||
'pl': pl,
|
|
||||||
'pl_pct': pl_pct,
|
|
||||||
'currency': currency,
|
|
||||||
'market_val': mkt_val,
|
|
||||||
'cost_amount': cost_amount,
|
|
||||||
'exchange_rate': rate,
|
|
||||||
})
|
|
||||||
|
|
||||||
total_pl += pl
|
|
||||||
mv_cny = mkt_val if currency == 'CNY' else mkt_val * rate
|
mv_cny = mkt_val if currency == 'CNY' else mkt_val * rate
|
||||||
total_mv_cny += mv_cny
|
total_mv_cny += mv_cny
|
||||||
|
|
||||||
|
holdings.append({
|
||||||
|
'code': code, 'name': name, 'shares': shares,
|
||||||
|
'price': price, 'cost_price': round(cost_price, 2),
|
||||||
|
'currency': currency, 'market_val': mkt_val,
|
||||||
|
'cost_amount': cost_amount, 'exchange_rate': rate,
|
||||||
|
})
|
||||||
|
|
||||||
pfx = 'HK$' if currency == 'HKD' else '¥'
|
pfx = 'HK$' if currency == 'HKD' else '¥'
|
||||||
print(f" {code} {name} {shares}股 {pfx}{price:.2f} 盈亏{pl:+,.0f}({pl_pct:+.1f}%)")
|
print(f" {code} {name} {shares}股 {pfx}{price:.2f} 成本{cost_price:.2f} 盈亏{pl:+,.0f}({pl_pct:+.1f}%)")
|
||||||
|
|
||||||
total_assets = total_mv_cny + CASH
|
total_assets = total_mv_cny + CASH
|
||||||
position_pct = round(total_mv_cny / total_assets * 100, 1) if total_assets > 0 else 0
|
position_pct = round(total_mv_cny / total_assets * 100, 2) if total_assets > 0 else 0
|
||||||
|
|
||||||
|
# Write portfolio.json
|
||||||
portfolio = {
|
portfolio = {
|
||||||
'holdings': holdings,
|
'holdings': holdings,
|
||||||
'cash': CASH,
|
'cash': CASH,
|
||||||
'total_market_value': round(total_mv_cny, 2),
|
'total_market_value': round(total_mv_cny, 2),
|
||||||
'total_pl': round(total_pl, 2),
|
'total_assets': round(total_assets, 2),
|
||||||
|
'total_pl': 0,
|
||||||
'position_pct': position_pct,
|
'position_pct': position_pct,
|
||||||
'updated_at': datetime.now().strftime('%Y-%m-%d %H:%M'),
|
'updated_at': datetime.now().strftime('%Y-%m-%d %H:%M'),
|
||||||
'source': STOCKS_FILE,
|
'source': STOCKS_FILE,
|
||||||
}
|
}
|
||||||
|
|
||||||
with open(PORTFOLIO_PATH, 'w') as f:
|
with open(PORTFOLIO_PATH, 'w') as f:
|
||||||
json.dump(portfolio, f, indent=2, ensure_ascii=False)
|
json.dump(portfolio, f, indent=2, ensure_ascii=False)
|
||||||
|
print(f" → {PORTFOLIO_PATH}")
|
||||||
|
|
||||||
print(f"\n已写入 {len(holdings)} 只持仓")
|
# Write SQLite
|
||||||
print(f"A股: {sum(1 for h in holdings if h['currency']=='CNY')} | 港股: {sum(1 for h in holdings if h['currency']=='HKD')}")
|
conn = sqlite3.connect(DB_PATH)
|
||||||
print(f"总市值: {round(total_mv_cny):,.0f}元 | 现金: {CASH:,}元")
|
c = conn.cursor()
|
||||||
print(f"总资产: {round(total_assets):,.0f}元 | 仓位: {position_pct}%")
|
c.execute('DELETE FROM holdings')
|
||||||
print(f"累计盈亏: {round(total_pl):+,}元")
|
c.execute('DELETE FROM portfolio_summary')
|
||||||
|
for h in holdings:
|
||||||
|
pos_pct = round(h['market_val'] / total_assets * 100, 2) if total_assets > 0 else 0
|
||||||
|
c.execute('''
|
||||||
|
INSERT INTO holdings (code, name, shares, cost, position_pct, added_at, is_active)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?, 1)
|
||||||
|
''', (h['code'], h['name'], h['shares'], h['cost_price'], pos_pct,
|
||||||
|
datetime.now().strftime('%Y-%m-%d')))
|
||||||
|
c.execute('''
|
||||||
|
INSERT INTO portfolio_summary (total_assets, stock_value, cash, position_pct, total_pnl, updated_at)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?)
|
||||||
|
''', (round(total_assets, 2), round(total_mv_cny, 2), CASH, position_pct, 0,
|
||||||
|
datetime.now().strftime('%Y-%m-%d %H:%M')))
|
||||||
|
conn.commit()
|
||||||
|
conn.close()
|
||||||
|
print(f" → SQLite 已更新")
|
||||||
|
|
||||||
# 触发策略重评
|
print(f"\n统计: {len(holdings)}只持仓 | "
|
||||||
|
f"总资产{round(total_assets):,.0f}元 | "
|
||||||
|
f"现金{CASH:,.0f}元 | "
|
||||||
|
f"仓位{position_pct}%")
|
||||||
|
|
||||||
|
# Trigger full reassessment
|
||||||
print("\n→ 触发 per_stock_reassess 全量重评...")
|
print("\n→ 触发 per_stock_reassess 全量重评...")
|
||||||
r = subprocess.run(
|
r = subprocess.run(
|
||||||
["python3", "/home/hmo/.hermes/profiles/position-analyst/scripts/per_stock_reassess.py"],
|
["python3", "/home/hmo/.hermes/profiles/position-analyst/scripts/per_stock_reassess.py"],
|
||||||
@@ -108,7 +119,7 @@ def main():
|
|||||||
)
|
)
|
||||||
print(r.stdout[-500:] if len(r.stdout) > 500 else r.stdout)
|
print(r.stdout[-500:] if len(r.stdout) > 500 else r.stdout)
|
||||||
|
|
||||||
# 重建决策树
|
# Rebuild decision trees
|
||||||
print("\n→ 重建决策树...")
|
print("\n→ 重建决策树...")
|
||||||
sys.path.insert(0, '/home/hmo/web-dashboard')
|
sys.path.insert(0, '/home/hmo/web-dashboard')
|
||||||
from strategy_tree import init_default_branches
|
from strategy_tree import init_default_branches
|
||||||
@@ -117,21 +128,14 @@ def main():
|
|||||||
ok = 0
|
ok = 0
|
||||||
for e in data.get('decisions', []):
|
for e in data.get('decisions', []):
|
||||||
branches = init_default_branches(
|
branches = init_default_branches(
|
||||||
e.get('code', ''),
|
e.get('code', ''), e.get('name', ''),
|
||||||
e.get('name', ''),
|
e.get('entry_low', 0), e.get('entry_high', 0),
|
||||||
e.get('entry_low', 0),
|
e.get('stop_loss', 0), e.get('take_profit', 0))
|
||||||
e.get('entry_high', 0),
|
e['strategy_tree'] = {'branches': branches, 'created_at': datetime.now().strftime('%Y-%m-%d')}
|
||||||
e.get('stop_loss', 0),
|
|
||||||
e.get('take_profit', 0),
|
|
||||||
)
|
|
||||||
e['strategy_tree'] = {
|
|
||||||
'branches': branches,
|
|
||||||
'created_at': datetime.now().strftime('%Y-%m-%d'),
|
|
||||||
}
|
|
||||||
ok += 1
|
ok += 1
|
||||||
with open('/home/hmo/web-dashboard/data/decisions.json', 'w') as f:
|
with open('/home/hmo/web-dashboard/data/decisions.json', 'w') as f:
|
||||||
json.dump(data, f, indent=2, ensure_ascii=False)
|
json.dump(data, f, indent=2, ensure_ascii=False)
|
||||||
print(f"决策树重建完成: {ok}/{len(data.get('decisions',[]))}")
|
print(f"决策树重建: {ok}/{len(data.get('decisions',[]))}")
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
|||||||
@@ -0,0 +1,244 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
"""stale_detector.py — 检查所有策略,标记价格偏离/过期的策略
|
||||||
|
|
||||||
|
读取 decisions.json 的扁平列表。自选策略和持仓策略分开判断。
|
||||||
|
可被 cron no_agent 模式调用:stdout 注入到后续 LLM 分析。
|
||||||
|
|
||||||
|
输出格式:
|
||||||
|
[FLAG] [自选/持仓] 股票名(代码) 价XX | 买入A~B | 问题
|
||||||
|
|
||||||
|
用法:
|
||||||
|
python3 stale_detector.py
|
||||||
|
"""
|
||||||
|
import json
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
from datetime import datetime, timezone
|
||||||
|
|
||||||
|
DECISIONS_PATH = "/home/hmo/web-dashboard/data/decisions.json"
|
||||||
|
PORTFOLIO_PATH = "/home/hmo/web-dashboard/data/portfolio.json"
|
||||||
|
|
||||||
|
|
||||||
|
def fetch_prices(codes):
|
||||||
|
import urllib.request
|
||||||
|
if not codes:
|
||||||
|
return {}
|
||||||
|
symbols, code_map = [], {}
|
||||||
|
for c in codes:
|
||||||
|
c = str(c).strip()
|
||||||
|
p = "sh" if (len(c) == 6 and c[0] in "569") else "sz" if len(c) == 6 else "hk"
|
||||||
|
sym = f"{p}{c}"
|
||||||
|
symbols.append(sym)
|
||||||
|
code_map[sym] = c
|
||||||
|
try:
|
||||||
|
req = urllib.request.Request(
|
||||||
|
f"http://qt.gtimg.cn/q={','.join(symbols)}",
|
||||||
|
headers={"User-Agent": "curl/7.81"},
|
||||||
|
)
|
||||||
|
with urllib.request.urlopen(req, timeout=10) as r:
|
||||||
|
text = r.read().decode("gbk")
|
||||||
|
except Exception as e:
|
||||||
|
print(f"FETCH_FAIL: {e}", file=sys.stderr)
|
||||||
|
return {}
|
||||||
|
|
||||||
|
results = {}
|
||||||
|
for line in text.strip().split("\n"):
|
||||||
|
if "=" not in line:
|
||||||
|
continue
|
||||||
|
try:
|
||||||
|
raw = line.split("=", 1)[1].strip().strip('"').strip(";")
|
||||||
|
fld = raw.split("~")
|
||||||
|
if len(fld) < 6:
|
||||||
|
continue
|
||||||
|
sym = line.split("=", 1)[0].strip().lstrip("v_")
|
||||||
|
oc = code_map.get(sym)
|
||||||
|
if not oc:
|
||||||
|
continue
|
||||||
|
p = float(fld[3]) if fld[3] else 0
|
||||||
|
c = fld[32] if len(fld) > 32 else "0"
|
||||||
|
results[oc] = (p, c)
|
||||||
|
except (ValueError, IndexError):
|
||||||
|
continue
|
||||||
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
decisions_list = json.load(open(DECISIONS_PATH))
|
||||||
|
if not isinstance(decisions_list, list):
|
||||||
|
decisions_list = decisions_list.get("decisions", []) if isinstance(decisions_list, dict) else []
|
||||||
|
|
||||||
|
# 只保留有买入区的条目,排除已关闭的(inactive/closed)
|
||||||
|
EXCLUDED_STATUSES = ("closed", "inactive")
|
||||||
|
to_check = [d for d in decisions_list if (d.get("entry_low") is not None or d.get("entry_high") is not None) and d.get("status") not in EXCLUDED_STATUSES]
|
||||||
|
if not to_check:
|
||||||
|
print("[SILENT] 无需要检查的策略")
|
||||||
|
return 0
|
||||||
|
|
||||||
|
# ----- 组合级监测:读取总仓位 + 弱势比例 -----
|
||||||
|
position_pct = 0
|
||||||
|
cash = 0
|
||||||
|
total_assets = 0
|
||||||
|
try:
|
||||||
|
with open(PORTFOLIO_PATH) as f:
|
||||||
|
pf = json.load(f)
|
||||||
|
position_pct = pf.get("position_pct", 0)
|
||||||
|
cash = pf.get("cash", 0)
|
||||||
|
total_assets = pf.get("total_assets", 0)
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
# 统计持仓策略中弱势/深套的比例
|
||||||
|
weak_count = 0
|
||||||
|
holding_count = 0
|
||||||
|
for d in decisions_list:
|
||||||
|
if d.get("type") == "持仓策略" and d.get("status") not in ("closed", "inactive"):
|
||||||
|
holding_count += 1
|
||||||
|
cat = d.get("stock_category", "")
|
||||||
|
if cat in ("弱势", "深套"):
|
||||||
|
weak_count += 1
|
||||||
|
weak_ratio = (weak_count / holding_count * 100) if holding_count > 0 else 0
|
||||||
|
|
||||||
|
prices = fetch_prices([d["code"] for d in to_check])
|
||||||
|
now = datetime.now(timezone.utc).astimezone()
|
||||||
|
found = 0
|
||||||
|
|
||||||
|
for d in to_check:
|
||||||
|
code = d["code"]
|
||||||
|
name = d.get("name", code)
|
||||||
|
el = d.get("entry_low")
|
||||||
|
eh = d.get("entry_high")
|
||||||
|
sl = d.get("stop_loss")
|
||||||
|
tp = d.get("take_profit")
|
||||||
|
ts = d.get("created_at") or d.get("timestamp") or d.get("updated_at", "")
|
||||||
|
is_wl = "自选" in (d.get("type", ""))
|
||||||
|
|
||||||
|
pi = prices.get(code)
|
||||||
|
if not pi:
|
||||||
|
continue
|
||||||
|
price, chg = pi
|
||||||
|
if price <= 0:
|
||||||
|
continue
|
||||||
|
|
||||||
|
issues, flags = [], []
|
||||||
|
tag = "[自选]" if is_wl else "[持仓]"
|
||||||
|
|
||||||
|
# -- 偏离 --
|
||||||
|
if is_wl and el and eh:
|
||||||
|
# 读取 timing_signal 判断策略有效性(timing_signal 字段优先,fallback to action)
|
||||||
|
current_str = d.get("current", "") or ""
|
||||||
|
timing_signal = d.get("timing_signal", "") or current_str
|
||||||
|
has_nonbuy_signal = any(kw in timing_signal for kw in [
|
||||||
|
"等企稳再入", "等企稳", "弱势持有", "观望",
|
||||||
|
"不建议买入", "谨慎买入",
|
||||||
|
])
|
||||||
|
|
||||||
|
# 直接计算 R/R(不依赖文本匹配)
|
||||||
|
rr_invalid = False
|
||||||
|
if sl and sl > 0 and tp and tp > 0 and price > sl:
|
||||||
|
rr = (tp - price) / (price - sl)
|
||||||
|
if rr < 1.5:
|
||||||
|
rr_invalid = True
|
||||||
|
# 也检查 tp 是否接近或低于成本(微盈/浮亏止盈)
|
||||||
|
cost = d.get("cost", 0)
|
||||||
|
if cost and cost > 0 and tp <= cost * 1.05:
|
||||||
|
rr_invalid = True
|
||||||
|
|
||||||
|
strategy_deficient = has_nonbuy_signal or rr_invalid
|
||||||
|
# 对自选无止盈位的也标记(策略不完整)
|
||||||
|
if not tp or tp == 0:
|
||||||
|
strategy_deficient = True
|
||||||
|
|
||||||
|
if el <= price <= eh:
|
||||||
|
flags.append("[WL_IN]")
|
||||||
|
if strategy_deficient:
|
||||||
|
flags.append("[STRATEGY_STALE]")
|
||||||
|
prefix = "⚠️仓位挤占 " if position_pct > 80 else ""
|
||||||
|
issues.append(f"[STRATEGY_STALE] {prefix}价{price:.2f}在买入区{el}~{eh}但策略不完整({'RR='+f'{rr:.2f}<1.5' if rr_invalid else '无止盈位' if not tp else '非买入信号'}),买入区需重评")
|
||||||
|
else:
|
||||||
|
prefix = "⚠️仓位挤占 " if position_pct > 80 else ""
|
||||||
|
issues.append(f"[PUSH] {prefix}价{price:.2f}入买入区{el}~{eh}")
|
||||||
|
elif price > eh * 1.35:
|
||||||
|
flags.append("[WL_HIGH]")
|
||||||
|
issues.append(f"价{price:.2f}高出买入区+{((price/eh)-1)*100:.0f}%,买入区需重评")
|
||||||
|
elif price > eh * 1.20:
|
||||||
|
flags.append("[WL_DRIFT]")
|
||||||
|
issues.append(f"价{price:.2f}高于买入区+{((price/eh)-1)*100:.0f}%")
|
||||||
|
elif not is_wl and eh:
|
||||||
|
dp = (price / eh - 1) * 100
|
||||||
|
if dp > 35:
|
||||||
|
flags.append("[SEVERE]")
|
||||||
|
issues.append(f"偏离买入区上沿+{dp:.0f}%")
|
||||||
|
elif dp > 20:
|
||||||
|
flags.append("[DRIFT]")
|
||||||
|
issues.append(f"偏离买入区上沿+{dp:.0f}%")
|
||||||
|
elif dp > 10:
|
||||||
|
flags.append("[WARN]")
|
||||||
|
issues.append(f"偏离买入区上沿+{dp:.0f}%")
|
||||||
|
# 持仓在买入区内但 R/R 不达标
|
||||||
|
if el and sl and sl > 0 and tp and tp > 0 and price > sl:
|
||||||
|
if el <= price <= eh:
|
||||||
|
rr = (tp - price) / (price - sl)
|
||||||
|
if rr < 1.5:
|
||||||
|
flags.append("[RR_WARN]")
|
||||||
|
issues.append(f"买入区内RR仅{rr:.2f}<1.5,策略需重评")
|
||||||
|
|
||||||
|
# -- 距止损/止盈(仅持仓) --
|
||||||
|
if not is_wl:
|
||||||
|
if sl and sl > 0:
|
||||||
|
dsl = (price / sl - 1) * 100
|
||||||
|
if dsl < 5:
|
||||||
|
flags.append("[NEAR_SL]")
|
||||||
|
issues.append(f"距止损仅{dsl:.1f}%")
|
||||||
|
if tp and tp > 0:
|
||||||
|
dtp = (tp / price - 1) * 100
|
||||||
|
if dtp < 5:
|
||||||
|
# 成本基准校验:止盈标记只有在盈利≥5%时才有效
|
||||||
|
cost_check = True
|
||||||
|
cost = d.get("cost")
|
||||||
|
if cost and cost > 0 and price < cost * 1.05:
|
||||||
|
cost_check = False
|
||||||
|
if cost_check:
|
||||||
|
flags.append("[NEAR_TP]")
|
||||||
|
issues.append(f"距止盈仅{dtp:.1f}%")
|
||||||
|
|
||||||
|
# -- 过期 --
|
||||||
|
stale_limit = 30 if is_wl else 14
|
||||||
|
if ts:
|
||||||
|
try:
|
||||||
|
ud = datetime.fromisoformat(ts)
|
||||||
|
if ud.tzinfo is None:
|
||||||
|
ud = ud.replace(tzinfo=timezone.utc)
|
||||||
|
days = (now - ud).days
|
||||||
|
if days > stale_limit:
|
||||||
|
flags.append("[STALE]")
|
||||||
|
issues.append(f"{days}天未更新(>{stale_limit})")
|
||||||
|
except (ValueError, TypeError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
if issues:
|
||||||
|
print(f"{' '.join(flags)} {tag} {name}({code}) 价{price:.2f}{chg} | 买入{el}~{eh} | {'; '.join(issues)}")
|
||||||
|
found += 1
|
||||||
|
|
||||||
|
if found == 0:
|
||||||
|
print("[SILENT] 所有策略正常")
|
||||||
|
|
||||||
|
# ----- 组合级警报 -----
|
||||||
|
portfolio_alerts = 0
|
||||||
|
if holding_count > 0:
|
||||||
|
if weak_ratio > 40:
|
||||||
|
print(f"\n[PORTFOLIO_WEAK] 组合弱势比例{weak_ratio:.0f}% ({weak_count}/{holding_count})!仓位{position_pct:.1f}% → 建议系统性减仓")
|
||||||
|
portfolio_alerts += 1
|
||||||
|
elif weak_ratio > 30:
|
||||||
|
print(f"\n[PORTFOLIO_WEAK_MILD] 组合弱势比例{weak_ratio:.0f}% ({weak_count}/{holding_count}),仓位{position_pct:.1f}%,关注")
|
||||||
|
portfolio_alerts += 1
|
||||||
|
if position_pct > 80 and holding_count > 0:
|
||||||
|
# 仓位过满提醒
|
||||||
|
print(f"[PORTFOLIO_FULL] 总仓位{position_pct:.1f}% > 80%,现金{cash:.0f}({cash/total_assets*100:.1f}%)")
|
||||||
|
portfolio_alerts += 1
|
||||||
|
if portfolio_alerts > 0:
|
||||||
|
found += portfolio_alerts
|
||||||
|
|
||||||
|
return found
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
Reference in New Issue
Block a user