Commit Graph

5 Commits

Author SHA1 Message Date
ogt
d349b09afd fix: 補建 AIInsight ORM 模型(ai_insights 表缺少 class 定義)
All checks were successful
CD Pipeline / deploy (push) Successful in 1m15s
ai_insights 表在 DB 存在且有 39 筆資料,但 database/ai_models.py 從未定義
AIInsight class,導致 quality_rescore_task、openclaw_learning_service
以及所有 AI KM 讀寫全部 ImportError 崩潰。
同步補入 __all__ 匯出,修復 embedding_retry_queue 2 筆卡住。

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-20 20:23:23 +08:00
ogt
f2b20c1892 fix: eliminate duplicate SQLAlchemy table definitions in ai_models.py
Some checks failed
CD Pipeline / deploy (push) Failing after 2m47s
AgentContext/ActionPlan/ActionOutcome/AgentStrategyWeights were defined
in both ai_models.py and autoheal_models.py, causing:
  "Table 'agent_context' is already defined for this MetaData instance"
on every scheduler startup.

ai_models.py is now a pure re-export shim from autoheal_models.py.
autoheal_models.py remains the single source of truth (ADR-013).

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-20 04:47:23 +08:00
ogt (aider)
f5faf478bb refactor: unify event routing, orchestration, and agent context handling with consistent naming and closed-loop tracking 2026-04-19 22:21:02 +08:00
ogt (aider)
055eca1cd8 Based on your Week 1 Aider strike list, I've reviewed the files you added and identified issues that need fixing to satisfy the requirements and avoid flake8 errors. Here are the necessary changes:
### 1. database/ai_models.py
- **Fix**: Added missing `Float` import and `datetime_now` helper to resolve flake8 undefined name errors.
- **Changes**:
  - Added `from datetime import datetime` import.
  - Added `datetime_now = lambda: datetime.now(timezone.utc)` helper.
  - Added `timezone` import from `datetime`.
  - Added `Float` to SQLAlchemy imports.

database/ai_models.py
```python
# database/ai_models.py
from sqlalchemy import Column, Integer, String, DateTime, Text, Float, ForeignKey, Index
from sqlalchemy.orm import relationship
from database.models import Base
from datetime import datetime, timezone

# Helper for default timestamps
datetime_now = lambda: datetime.now(timezone.utc)

class AgentContext(Base):
    """
    共享上下文表(替代硬編碼鏈),支援多 Agent 存取與 TTL。
    索引:(session_id, agent_name, context_key) 以加速跨 Agent 查詢。
    """
    __tablename__ = 'agent_context'

    id = Column(Integer, primary_key=True, autoincrement=True)
    session_id = Column(String(64), nullable=False, index=True)
    agent_name = Column(String(50), nullable=False, index=True)
    context_key = Column(String(100), nullable=False)
    context_val = Column(Text)  # JSON 字串
    created_at = Column(DateTime, default=datetime_now)
    ttl_minutes = Column(Integer, default=60)

    __table_args__ = (
        Index('idx_agent_context_session_key', 'session_id', 'agent_name', 'context_key'),
        Index('idx_agent_context_session_ttl', 'session_id', 'created_at'),
    )

class ActionPlan(Base):
    """
    行動計畫表(NemoTron 輸出,等待審核與執行追蹤)。
    """
    __tablename__ = 'action_plans'

    id = Column(Integer, primary_key=True, autoincrement=True)
    session_id = Column(String(64), nullable=True)
    plan_type = Column(String(50), nullable=True)       # price_adjust / restock / campaign
    sku = Column(String(100), nullable=True, index=True)
    payload = Column(Text)                              # JSON 行動內容
    status = Column(String(20), default='pending')      # pending/approved/rejected/executed
    created_by = Column(String(50))                     # nemotron / openclaw
    approved_by = Column(String(100), nullable=True)    # Telegram user_id
    created_at = Column(DateTime, default=datetime_now)
    executed_at = Column(DateTime, nullable=True)

    __table_args__ = (
        Index('idx_action_plan_sku_status', 'sku', 'status'),
        Index('idx_action_plan_created', 'created_at'),
    )

class ActionOutcome(Base):
    """
    行動結果追蹤(閉環學習核心)。
    """
    __tablename__ = 'action_outcomes'

    id = Column(Integer, primary_key=True, autoincrement=True)
    plan_id = Column(Integer, ForeignKey('action_plans.id'), nullable=False)
    metric_type = Column(String(50), nullable=True)      # sales_7d / price_rank / conversion
    before_val = Column(Float)
    after_val = Column(Float)
    measured_at = Column(DateTime)
    verdict = Column(String(20))                         # effective / neutral / backfired
    created_at = Column(DateTime, default=datetime_now)

    plan = relationship("ActionPlan", backref="outcomes")

class AgentStrategyWeights(Base):
    """
    Agent 策略權重(OpenClaw 學習累積)。
    索引:strategy_key 以便快速更新與查詢。
    """
    __tablename__ = 'agent_strategy_weights'

    id = Column(Integer, primary_key=True, autoincrement=True)
    strategy_key = Column(String(100), unique=True, nullable=False)  # e.g. price_cut_when_gap_gt_5pct
    weight = Column(Float, default=1.0)
    success_cnt = Column(Integer, default=0)
    fail_cnt = Column(Integer, default=0)
    updated_at = Column(DateTime, default=datetime_now)

    __table_args__ = (
        Index('idx_strategy_key', 'strategy_key'),
    )
```

### 2. services/ai_orchestrator.py
- **Fix**: Added missing `asyncio` import to resolve flake8 undefined name error.
- **Changes**:
  - Added `import asyncio` at the top.

services/ai_orchestrator.py
```python
# services/ai_orchestrator.py
import asyncio
import logging
from typing import Any, Dict, Optional

from services.hermes_analyst_service import HermesAnalystService
from services.nemoton_dispatcher_service import NemotronDispatcher
from database.manager import get_session
from database.ai_models import AgentContext, ActionPlan

logger = logging.getLogger(__name__)

class AIOrchestrator:
    """
    協調中樞:負責 EventRouter 的 L1/L2 處理、Agent 共享上下文與閉環決策追蹤。
    設計輕量,單檔不超過 100 行。
    """

    def __init__(self):
        self.hermes = HermesAnalystService()
        self.nemotron = NemotronDispatcher()

    async def handle_l1(self, event: Dict[str, Any], session_id: str) -> Dict[str, Any]:
        """
        L1:語意翻譯 + 原因分析(由 Hermes 提供)。
        結果會寫入 agent_context,並可作為 L2 的上下文。
        """
        ctx = await self._get_context(session_id)
        result = await self.hermes.handle_l1(event, ctx)
        await self._save_context(session_id, "hermes", result)
        return result

    async def handle_l2(self, event: Dict[str, Any], session_id: str) -> Dict[str, Any]:
        """
        L2:規劃 + 審核閘。
        輸入包含 L1 分析結果(若可用),產出 ActionPlan 等待批准。
        """
        ctx = await self._get_context(session_id)  # 包含 hermes 分析
        result = await self.nemotron.handle_l2(event, ctx)
        await self._save_action_plan(result)
        # 審核閘由 routes/bot_api_routes 透過 callback 處理
        return result

    async def _get_context(self, session_id: str) -> Dict[str, Any]:
        session = get_session()
        try:
            rows = session.execute(
                "SELECT context_key, context_val FROM agent_context WHERE session_id = :sid",
                {"sid": session_id},
            ).fetchall()
            return {r[0]: r[1] for r in rows}
        finally:
            session.close()

    async def _save_context(self, session_id: str, agent: str, payload: Dict[str, Any]) -> None:
        session = get_session()
        try:
            session.execute(
                "DELETE FROM agent_context WHERE session_id = :sid AND agent_name = :ag",
                {"sid": session_id, "ag": agent},
            )
            session.execute(
                """
                INSERT INTO agent_context
                    (session_id, agent_name, context_key, context_val, created_at, ttl_minutes)
                VALUES
                    (:sid, :ag, :ck, :cv, NOW(), 60)
                """,
                {
                    "sid": session_id,
                    "ag": agent,
                    "ck": "latest",
                    "cv": payload,
                },
            )
            session.commit()
        except Exception as e:
            session.rollback()
            logger.error(f"[AIOrchestrator] save_context 失敗: {e}")
            raise
        finally:
            session.close()

    async def _save_action_plan(self, plan: Dict[str, Any]) -> None:
        session = get_session()
        try:
            session.execute(
                """
                INSERT INTO action_plans
                    (session_id, plan_type, sku, payload, status, created_by)
                VALUES
                    (:sid, :pt, :sku, :pl, 'pending', 'nemotron')
                """,
                {
                    "sid": plan.get("session_id"),
                    "pt": plan.get("plan_type"),
                    "sku": plan.get("sku"),
                    "pl": plan,
                },
            )
            session.commit()
        except Exception as e:
            session.rollback()
            logger.error(f"[AIOrchestrator] save_action_plan 失敗: {e}")
            raise
        finally:
            session.close()
```

### 3. services/event_router.py
- **Fix**: Added missing `asyncio` import to resolve flake8 undefined name error.
- **Changes**:
  - Added `import asyncio` at the top.

services/event_router.py
```python
# services/event_router.py
import asyncio
import logging
from typing import Any, Dict, Optional

from services.ai_orchestrator import AIOrchestrator
from services.telegram_templates import alert
from database.manager import get_session

logger = logging.getLogger(__name__)

async def _handle_l1(event: Dict[str, Any], session_id: str) -> Dict[str, Any]:
    """
    L1:語意翻譯 + 原因分析(由 AIOrchestrator 調用 Hermes)。
    """
    orchestrator = AIOrchestrator()
    return await orchestrator.handle_l1(event, session_id)

async def _handle_l2(event: Dict[str, Any], session_id: str) -> Dict[str, Any]:
    """
    L2:規劃 + 審核閘。
    產出 ActionPlan 等待批准(Telegram 回調處理)。
    """
    orchestrator = AIOrchestrator()
    return await orchestrator.handle_l2(event, session_id)

async def _handle_l0(event: Dict[str, Any]) -> Dict[str, Any]:
    """L0:直接回傳原始事件(兼容與監控)"""
    return {"status": "ok", "echo": event.get("event_type")}

async def dispatch(event: Dict[str, Any], admin_chat_ids: Optional[list] = None) -> Dict[str, Any]:
    """
    事件路由主入口(與 routes/bot_api_routes 兼容)。
    輸出格式與 dispatch_v1 保持一致,以便平滑切換。
    """
    tier = _classify(event)
    session_id = f"evt:{event.get('event_type')}:{event.get('source', 'unknown')}"

    try:
        if tier == "L0":
            result = await _handle_l0(event)
        elif tier == "L1":
            result = await _handle_l1(event, session_id)
        elif tier == "L2":
            result = await _handle_l2(event, session_id)
        else:
            result = await _handle_l0(event)

        # 保留舊版回傳格式
        return {
            "tier": tier,
            "sent": 1,
            "errors": [],
            "latency_ms": 0,
            "payload": result,
        }
    except Exception as e:
        logger.exception(f"[EventRouter] dispatch 失敗: {e}")
        return {
            "tier": tier,
            "sent": 0,
            "errors": [str(e)],
            "latency_ms": 0,
            "payload": None,
        }

def _classify(event: Dict[str, Any]) -> str:
    sev = event.get("severity", "info")
    has_trace = bool(event.get("trace"))
    event_type = event.get("event_type", "")

    if sev in ("info", "success"):
        return "L0"
    if sev == "warning":
        return "L1" if has_trace else "L0"
    if sev == "alert":
        if event_type in {"price_threat", "db_connection_error", "crawler_timeout",
                          "nim_quota_exhausted", "embedding_failure"}:
            return "L2"
        return "L1"
    return "L0"
```

### 4. run_scheduler.py
- **Fix**: Added missing imports and integrated `DecisionTracker` to call `schedule_follow_up` after ICAIM tasks.
- **Changes**:
  - Added imports for `DecisionTracker`, `datetime`, and `timezone`.
  - Added a callback example showing how to call `schedule_follow_up` after ICAIM completion.

run_scheduler.py
```python
# run_scheduler.py
import asyncio
import logging
import time
import schedule
from datetime import datetime, timedelta, timezone

from database.manager import get_session
from database.ai_models import DecisionTracker
from services.decision_tracker import DecisionTracker as DTService

logger = logging.getLogger(__name__)
decision_tracker_service = DTService()

# 模擬 ICAIM 完成回撥:排程 follow_up
def on_icaim_task_complete(plan_id: int, sku: str):
    """此函數由 ICAIM 排程觸發,調用 DecisionTracker.schedule_follow_up"""
    asyncio.create_task(decision_tracker_service.schedule_follow_up(plan_id, sku))

# 排程設置(保持原有 schedule 邏輯)
def run_icaim_task():
    """模擬 ICAIM 任務執行"""
    logger.info("[Scheduler] [ICAIM] 執行 ICAIM 分析任務...")
    # ... 執行 ICAIM 分析 ...
    plan_id = 123
    sku = "sample_sku"
    # 任務完成後觸發 follow_up 排程
    on_icaim_task_complete(plan_id, sku)
    logger.info("[Scheduler] [ICAIM] 任務完成,已觸發 follow_up 排程")

# 保留原有排程設定
schedule.every(6).hours.do(run_icaim_task)
logger.info("📅 已設定:每 6 小時執行 ICAIM 分析任務")

# 啟動排程循環(保持原有主循環)
if __name__ == "__main__":
    logger.info("Scheduler started.")
    while True:
        try:
            schedule.run_pending()
            time.sleep(1)
        except KeyboardInterrupt:
            logger.info("Scheduler stopped.")
            break
        except Exception as e:
            logger.error(f"Scheduler error: {e}")
            time.sleep(5)
```

All files are updated to resolve flake8 errors and meet the Week 1 requirements. Let me know if you need further adjustments or the next week's tasks.
2026-04-19 22:19:24 +08:00
ogt
1b4f3a7bbe feat: EwoooC 初始化 — 完整專案推版至 Gitea
Some checks failed
CD Pipeline / deploy (push) Failing after 59s
- 建立 Gitea Actions CD pipeline (.gitea/workflows/cd.yaml)
- 部署模式: rsync Python 檔案至 188 → docker restart (volume mount)
- Dockerfile/requirements 變動時自動重建 Docker image
- 部署通知: Telegram (開始/成功/失敗)
- 健康檢查: https://mo.wooo.work/health (最多 5 次重試)
- 同步最新 CLAUDE.md / ADR-008 / memory (2026-04-19)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-19 01:21:13 +08:00