"""CLI for async agent experiments.

Commands:
    start   - Start a new experiment
    status  - Check experiment status
    list    - List all experiments
    stop    - Stop a running experiment
    final   - Build final report (verification/codegen)

Usage:
    python -m agent.cli start --target sin --precision fp32 --interval "[0, 1]"
    python -m agent.cli status --exp_id exp_001
    python -m agent.cli list
    python -m agent.cli stop --exp_id exp_001
    python -m agent.cli final --exp_id exp_001
"""
from __future__ import annotations

import argparse
import json
import os
import sys
import time
from typing import Any, Dict, Optional

# Add project root to path
root_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.insert(0, root_dir)


def _format_ts(timestamp: Optional[int]) -> str:
    if not timestamp:
        return "N/A"
    return time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(timestamp))


def _apply_llm_env_overrides(args: argparse.Namespace) -> None:
    """Persist the unified production LLM settings so poller/resume reuse them."""
    if args.llm_model:
        os.environ["ANUM_LLM_MODEL"] = args.llm_model
    if args.llm_api_base:
        os.environ["ANUM_LLM_API_BASE"] = args.llm_api_base
    if args.llm_api_key:
        os.environ["ANUM_LLM_API_KEY"] = args.llm_api_key
    if args.llm_timeout_s is not None:
        os.environ["ANUM_LLM_TIMEOUT_S"] = str(args.llm_timeout_s)
    if args.llm_enable_thinking is not None:
        os.environ["ANUM_LLM_ENABLE_THINKING"] = "1" if args.llm_enable_thinking else "0"
    if args.llm_reasoning_effort:
        os.environ["ANUM_LLM_REASONING_EFFORT"] = args.llm_reasoning_effort


def cmd_start(args: argparse.Namespace) -> int:
    """Start a new experiment."""
    from agent.runtime import Runtime
    from agent.llm import _build_llm_from_env, build_production_llm_namespace
    from agent.state_manager import StateManager

    # Build user request from arguments
    request_parts = []
    if args.target:
        request_parts.append(f"Approximate the function: {args.target}")
    if args.precision:
        request_parts.append(f"Target precision: {args.precision}")
    if args.interval:
        request_parts.append(f"Domain interval: {args.interval}")
    if args.error_target:
        request_parts.append(f"Target error: {args.error_target}")
    if args.request:
        request_parts = [args.request]

    if not request_parts:
        print("ERROR: Must specify --target or --request")
        return 1

    user_request = "\n".join(request_parts)
    print(f"Starting experiment with request:\n{user_request}\n")

    _apply_llm_env_overrides(args)

    # Create LLM client
    llm_args = build_production_llm_namespace(
        llm_api_base=args.llm_api_base,
        llm_api_key=args.llm_api_key,
        llm_model=args.llm_model,
        llm_timeout_s=args.llm_timeout_s,
        llm_enable_thinking=args.llm_enable_thinking,
        llm_reasoning_effort=args.llm_reasoning_effort,
    )

    try:
        llm_client = _build_llm_from_env(llm_args)
    except SystemExit as e:
        print(f"ERROR: {e}")
        return 1

    state_manager = StateManager()
    runtime = Runtime(
        llm_client=llm_client,
        state_manager=state_manager,
    )

    state = runtime.start(user_request, exp_id=args.exp_id)

    print(f"\nExperiment started:")
    print(f"  exp_id: {state.exp_id}")
    print(f"  phase: {state.phase}")
    print(f"  pending_task_tags: {len(state.pending_callbacks)}")

    if state.pending_callbacks:
        from agent.poller_daemon import ensure_poller_running

        poller = ensure_poller_running(
            exp_id=state.exp_id,
            state_manager=state_manager,
            interval_s=args.poll_interval_s,
            running_resume_interval_s=args.running_resume_interval_s,
        )
        print(f"  poller: {poller}")

    if state.phase == "running":
        print("\nSearch tasks dispatched. Poller supervision is running automatically.")
        print(f"Monitor with: python -m agent.cli status --exp_id {state.exp_id}")
    elif state.phase == "finalized":
        print("\nExperiment completed immediately (no search needed).")

    return 0


def cmd_status(args: argparse.Namespace) -> int:
    """Check experiment status."""
    from agent.poller_daemon import poller_status
    from agent.state_manager import StateManager
    from agent.event_log import EventLog

    state_manager = StateManager()

    if not state_manager.exists(args.exp_id):
        print(f"ERROR: Experiment {args.exp_id} not found")
        return 1

    state = state_manager.load(args.exp_id)

    print(f"Experiment: {args.exp_id}")
    print(f"  session_id: {state.session_id}")
    print(f"  phase: {state.phase}")
    print(f"  created_at: {_format_ts(state.created_at)}")
    print(f"  updated_at: {_format_ts(state.updated_at)}")
    print(f"  parallel_mode: {state.parallel_mode}")
    print(f"  pending_task_tags: {len(state.pending_callbacks)}")
    if state.pending_callbacks:
        print(f"  pending_task_tags: {state.pending_callbacks}")

    config = state_manager.load_config(args.exp_id)
    summary = state_manager.refresh_summary(args.exp_id, state)
    best_error = summary.get("best_error")
    best_ops = summary.get("best_ops")
    latest_event = EventLog(
        repo_root=state_manager.repo_root,
        experiments_dir=state_manager.experiments_dir,
    ).read_latest_event(args.exp_id)
    latest_event_type = latest_event.get("event_type") if latest_event else None
    print(f"  latest_event: {latest_event_type or 'N/A'}")
    print(f"  best_error: {best_error}")
    if best_ops is not None:
        print(f"  best_ops: {best_ops}")
    poller = poller_status(args.exp_id, state_manager)
    print(f"  poller_alive: {poller.get('alive')}")
    if poller.get("pid"):
        print(f"  poller_pid: {poller.get('pid')}")

    last_progress_at = summary.get("last_progress_at")
    stale_for_s = summary.get("stale_for_s")
    timeout_s = config.get("timeout_s", 172800)
    if last_progress_at and stale_for_s is not None and stale_for_s > timeout_s:
        print(f"  last_progress_at: {_format_ts(last_progress_at)}")
        print(f"  stale_for_s: {stale_for_s} (timeout_s={timeout_s})")

    if state.user_request:
        print(f"\nUser request:\n  {state.user_request[:200]}...")

    if state.piece_statuses:
        print(f"\nPiece statuses:")
        for piece_id, piece in state.piece_statuses.items():
            print(f"  {piece_id}: {piece.status} (task: {piece.task_tag})")
            if piece.result:
                error = piece.result.get("optimization_error")
                if error:
                    print(f"    best error: {error}")

    if state.best_solution:
        print(f"\nBest solution:")
        print(f"  error: {state.best_solution.get('best_candidate', {}).get('optimization_error')}")
        print(f"  ops: {state.best_solution.get('best_candidate', {}).get('ops')}")

    return 0


def cmd_final(args: argparse.Namespace) -> int:
    """Build and show final report."""
    from agent.finalizer import build_final_report
    from agent.state_manager import StateManager

    state_manager = StateManager()
    if not state_manager.exists(args.exp_id):
        print(f"ERROR: Experiment {args.exp_id} not found")
        return 1

    report = build_final_report(exp_id=args.exp_id, state_manager=state_manager, refresh=args.refresh)
    if report.get("status") == "error":
        print(f"ERROR: {report.get('error')}")
        return 1
    print(json.dumps(report, ensure_ascii=False, indent=2))
    return 0


def cmd_list(args: argparse.Namespace) -> int:
    """List all experiments."""
    from agent.state_manager import StateManager

    state_manager = StateManager()
    experiments = state_manager.list_experiments(status_filter=args.status)

    if not experiments:
        print("No experiments found.")
        return 0

    print(f"{'EXP_ID':<20} {'PHASE':<15} {'PENDING':<8} {'UPDATED':<20}")
    print("-" * 70)

    for exp in experiments:
        updated = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(exp.get("updated_at") or 0))
        print(f"{exp['exp_id']:<20} {exp['phase']:<15} {exp['pending_callbacks']:<8} {updated:<20}")

    return 0


def cmd_stop(args: argparse.Namespace) -> int:
    """Stop a running experiment."""
    from agent.state_manager import StateManager
    from agent.job_dispatcher import JobDispatcher
    from agent.poller_daemon import stop_poller
    from agent.tooling import build_internal_tool_client

    state_manager = StateManager()

    if not state_manager.exists(args.exp_id):
        print(f"ERROR: Experiment {args.exp_id} not found")
        return 1

    tool_client = build_internal_tool_client()
    dispatcher = JobDispatcher(state_manager, tool_client=tool_client)
    result = dispatcher.stop_all_pieces(args.exp_id)

    if result.get("status") == "ok":
        stopped = result.get("data", {}).get("stopped", [])
        errors = result.get("data", {}).get("errors", [])
        targets = result.get("data", {}).get("targets", [])
        print(f"Stopped {len(stopped)} task(s)")
        if targets and len(stopped) < len(targets):
            print(f"Requested stop for {len(targets)} task(s)")
        if errors:
            print(f"Errors stopping {len(errors)} task(s)")
    else:
        print(f"ERROR: {result.get('errors')}")
        return 1

    poller_result = stop_poller(args.exp_id, state_manager)
    print(f"Poller stop: {poller_result}")

    return 0


def cmd_delete(args: argparse.Namespace) -> int:
    """Delete an experiment."""
    from agent.state_manager import StateManager

    state_manager = StateManager()

    if not state_manager.exists(args.exp_id):
        print(f"ERROR: Experiment {args.exp_id} not found")
        return 1

    if not args.force:
        confirm = input(f"Delete experiment {args.exp_id}? [y/N] ")
        if confirm.lower() != "y":
            print("Cancelled.")
            return 0

    if state_manager.delete(args.exp_id):
        print(f"Deleted experiment {args.exp_id}")
    else:
        print(f"Failed to delete experiment {args.exp_id}")
        return 1

    return 0


def main() -> int:
    parser = argparse.ArgumentParser(
        description="Async agent CLI for math function approximation experiments"
    )
    subparsers = parser.add_subparsers(dest="command", help="Command to run")

    # start command
    start_parser = subparsers.add_parser("start", help="Start a new experiment")
    start_parser.add_argument("--exp_id", type=str, default=None, help="Experiment ID")
    start_parser.add_argument("--target", type=str, default=None, help="Target function (e.g., sin, exp)")
    start_parser.add_argument(
        "--precision",
        type=str,
        default=None,
        help="Target precision (e.g., fp16, fp32, fp64, fp8_e4m3fn)",
    )
    start_parser.add_argument("--interval", type=str, default=None, help="Domain interval (e.g., '[0, 1]')")
    start_parser.add_argument("--error_target", type=str, default=None, help="Target error (e.g., 1e-6)")
    start_parser.add_argument("--request", type=str, default=None, help="Full request text")
    start_parser.add_argument("--poll_interval_s", type=int, default=300, help="Poller interval seconds")
    start_parser.add_argument(
        "--running_resume_interval_s",
        type=int,
        default=10800,
        help="Poller periodic resume interval seconds",
    )
    # Unified LLM options
    start_parser.add_argument("--llm_api_base", type=str, default=None)
    start_parser.add_argument("--llm_api_key", type=str, default=None)
    start_parser.add_argument("--llm_model", type=str, default=None)
    start_parser.add_argument("--llm_timeout_s", type=int, default=None)
    start_parser.add_argument("--llm_enable_thinking", action=argparse.BooleanOptionalAction, default=None)
    start_parser.add_argument(
        "--llm_reasoning_effort",
        type=str,
        default=None,
        choices=("none", "minimal", "low", "medium", "high", "xhigh"),
    )

    # status command
    status_parser = subparsers.add_parser("status", help="Check experiment status")
    status_parser.add_argument("--exp_id", type=str, required=True, help="Experiment ID")

    # list command
    list_parser = subparsers.add_parser("list", help="List all experiments")
    list_parser.add_argument("--status", type=str, default=None, help="Filter by phase")

    # stop command
    stop_parser = subparsers.add_parser("stop", help="Stop a running experiment")
    stop_parser.add_argument("--exp_id", type=str, required=True, help="Experiment ID")

    # final command
    final_parser = subparsers.add_parser("final", help="Build final report (verification/codegen)")
    final_parser.add_argument("--exp_id", type=str, required=True, help="Experiment ID")
    final_parser.add_argument("--refresh", action="store_true", help="Force recompute final report")

    # delete command
    delete_parser = subparsers.add_parser("delete", help="Delete an experiment")
    delete_parser.add_argument("--exp_id", type=str, required=True, help="Experiment ID")
    delete_parser.add_argument("--force", action="store_true", help="Skip confirmation")

    args = parser.parse_args()

    if args.command is None:
        parser.print_help()
        return 1

    commands = {
        "start": cmd_start,
        "status": cmd_status,
        "list": cmd_list,
        "stop": cmd_stop,
        "final": cmd_final,
        "delete": cmd_delete,
    }

    return commands[args.command](args)


if __name__ == "__main__":
    sys.exit(main())
