from __future__ import annotations

from typing import Any, Dict, Iterable, List, Optional

from agent.event_log import EventLog
from agent.state import AgentState
from python_src.io_utils import tail_lines


def read_recent_events(
    exp_id: str,
    limit: int,
    *,
    event_log: Optional[EventLog] = None,
) -> List[Dict[str, Any]]:
    logger = event_log or EventLog()
    rows = logger.read_recent_events(exp_id, limit)
    return [row for row in rows if isinstance(row, dict)]


def last_assistant_message(state: AgentState) -> Optional[str]:
    for msg in reversed(state.conversation_history):
        if msg.get("role") == "assistant":
            return msg.get("content")
    return None


def collect_task_tags(state: AgentState) -> List[str]:
    tags: List[str] = []
    for tag in state.pending_callbacks:
        if tag:
            tags.append(tag)
    for run in state.runs:
        if run.task_tag:
            tags.append(run.task_tag)
    for piece in state.piece_statuses.values():
        if piece.task_tag:
            tags.append(piece.task_tag)
    return list(dict.fromkeys(tags))


def extract_log_file_names(entries: Iterable[Dict[str, Any]]) -> List[str]:
    names: List[str] = []
    for item in entries:
        if item.get("is_dir"):
            continue
        name = str(item.get("name", ""))
        if name.endswith(".log"):
            names.append(name)
    return sorted(names)


def fetch_log_tail(
    tool_client: Any,
    task_tag: str,
    *,
    lines: int,
    max_files: int,
) -> Dict[str, str]:
    logs: Dict[str, str] = {}
    listing = tool_client.call("anum.artifacts.list", {"artifact_id": f"{task_tag}:logs"})
    if listing.get("status") != "ok":
        return logs
    entries = listing.get("data", {}).get("entries", [])
    names = extract_log_file_names(entries)[: max(1, max_files)]
    for name in names:
        artifact_id = f"{task_tag}:logs/{name}"
        content_resp = tool_client.call(
            "anum.artifacts.get",
            {"artifact_id": artifact_id, "format": "text"},
        )
        if content_resp.get("status") != "ok":
            continue
        content = content_resp.get("data", {}).get("content", "")
        logs[name] = tail_lines(str(content), lines)
    return logs
