#!/usr/bin/env python3
import os
import sys
import time
import json
import uvicorn

from fastapi import FastAPI, Request
from pydantic import BaseModel

from src.agent import Agent
from src.logger import setup_logger

hf_model = os.getenv("AGENT_LLM_MODEL", "default_model")
run_id = os.getenv("RUN_ID", "default_run")

logger = setup_logger(
    __name__,
    level="INFO",
    log_subdir=hf_model,
    log_filename=f"run_{run_id}.log"
)

LLM_CONFIG_PATH = "./llm_config.json"

app = FastAPI()
agent = None
cfg = None

@app.on_event("startup")
def startup_event():
    global agent, cfg
    
    logger.info("=== Environment Variables Debug ===")
    logger.info(f"AGENT_LLM_TYPE: {os.getenv('AGENT_LLM_TYPE', 'NOT_SET')}")
    logger.info(f"AGENT_LLM_MODEL: {os.getenv('AGENT_LLM_MODEL', 'NOT_SET')}")
    logger.info(f"AGENT_LLM_OLLAMA_URL: {os.getenv('AGENT_LLM_OLLAMA_URL', 'NOT_SET')}")
    logger.info("===================================")
    
    cfg = get_llm_config_from_env()
    logger.info(f"Loaded config: {cfg}")
    
    logger.info("Starting agent initialization...")
    time.sleep(5) 
    
    max_retries = 3
    for attempt in range(max_retries):
        try:
            logger.info(f"Attempting to initialize agent (attempt {attempt + 1}/{max_retries})")
            agent = llm_deploy_worker(cfg)
            if agent is not None:
                agent.start()
                logger.info("Agent initialized successfully")
                break
            else:
                logger.error("Agent initialization returned None")
                if attempt < max_retries - 1:
                    logger.info("Retrying in 10 seconds...")
                    time.sleep(10)
        except Exception as e:
            logger.error(f"Agent initialization failed (attempt {attempt + 1}): {str(e)}")
            if attempt < max_retries - 1:
                logger.info("Retrying in 10 seconds...")
                time.sleep(10)
            else:
                logger.error("All initialization attempts failed")
                raise
def start_goal_oriented_mode(agent, goal):
    logger.info(f"Starting goal-oriented mode with goal: {goal}")
    print("\nAI Agent Goal-Oriented Mode")
    print(f"Goal: {goal}")
    agent.set_goal(goal)
    step_idx = 1
    results = []
    while agent.running:
        try:
            result = agent.step(step_idx)
            results.append(result)
            if result["status"] == "finished":
                logger.info("\n✅ Goal completed successfully!")
                break
            elif result["status"] == "in_progress":
                if "action" in result:
                    print(f"> {result['action']}")
                step_idx += 1
            else:
                logger.info(f"\n⚠️ Unexpected status: {result.get('status')}")
            import time
            time.sleep(15) 
        except Exception as e:
            logger.error(f"Error during goal processing: {str(e)}")
            print(f"\nSystem error: {str(e)}")
            results.append({"status": "error", "error": str(e)})
            break
    agent.stop()
    print("Session ended.")
    return results


class GoalRequest(BaseModel):
    goal: str

def llm_deploy_worker(cfg):
    agent = None
    logger.info(f"Initializing agent with config: {cfg}")
    
    try:
        if cfg.get("type") == "hf_model" and cfg.get("hf_model"):
            logger.info("Creating agent with hf_model configuration")
            agent = Agent(hf_model=cfg["hf_model"])
        elif cfg.get("type") == "close_name" and cfg.get("close_name"):
            logger.info("Creating agent with close_name configuration")
            agent = Agent(close_name=cfg["close_name"])
        else:
            logger.error(f"Invalid configuration: {cfg}")
            raise ValueError(f"Invalid configuration type: {cfg.get('type')}")
        
        if agent is None:
            logger.error("Agent creation returned None")
            raise RuntimeError("Agent creation failed")
        
        logger.info("Agent created successfully")
        return agent
        
    except Exception as e:
        logger.error(f"Failed to create agent: {str(e)}")
        import traceback
        logger.error(f"Full traceback: {traceback.format_exc()}")
        raise

def llm_infer_worker(agent, prompt, mode="health_check", goal_args=None):
    try:
        if mode == "health_check":
            result = agent.llm_health_check(prompt=prompt)
        elif mode == "set_goal":
            results = start_goal_oriented_mode(agent, goal_args["goal"])
            result = {"status": results[-1]["status"] if results else "unknown", "results": results}
        else:
            result = {"llm_status": "error", "error": "Unknown mode"}
    except Exception as e:
        result = {"llm_status": "error", "error": str(e)}
    return result

@app.api_route("/healthz", methods=["GET", "POST"])
async def health_check(request: Request):
    prompt = "health_check"
    if request.method == "GET":
        prompt = request.query_params.get("prompt", "health_check")
    elif request.method == "POST":
        data = await request.json()
        prompt = data.get("prompt", "health_check")
    llm_result = llm_infer_worker(agent, prompt, mode="health_check")
    if llm_result.get("llm_status") == "ok":
        logger.info(f"[HEALTHZ] LLM response: {llm_result.get('llm_response')}")
    else:
        logger.error(f"[HEALTHZ] LLM error: {llm_result.get('error', 'Unknown error')}, response: {llm_result.get('llm_response')}")
    return {
        "status": "ok" if llm_result.get("llm_status") == "ok" else "error",
        "service": "ai-agent-system",
        "llm_status": llm_result.get("llm_status"),
        "llm_response": llm_result.get("llm_response"),
        "llm_config": cfg,
        "used_prompt": prompt,
        "timestamp": time.time()
    }

@app.post("/set_goal")
def set_goal(req: GoalRequest):
    goal_args = {"goal": req.goal}
    result = llm_infer_worker(agent, req.goal, mode="set_goal", goal_args=goal_args)
    return result

def get_llm_config_from_file():
    try:
        with open(LLM_CONFIG_PATH, 'r') as f:
            return json.load(f)
    except Exception:
        return {}

def get_llm_config_from_env():
    llm_type = os.getenv("AGENT_LLM_TYPE")
    logger.info(f"Environment variable AGENT_LLM_TYPE: {llm_type}")
    
    if llm_type == "close_name":
        model = os.getenv("AGENT_LLM_MODEL")
        logger.info(f"Using close_name mode with model: {model}")
        return {"type": "close_name", "close_name": model}
    elif llm_type == "ollama":
        model = os.getenv("AGENT_LLM_MODEL")
        ollama_url = os.getenv("AGENT_LLM_OLLAMA_URL")
        logger.info(f"Using ollama mode with model: {model}, url: {ollama_url}")
        return {
            "type": "ollama",
            "ollama_url": ollama_url,
            "model_name": model
        }
    elif llm_type == "api_url":
        api_url = os.getenv("AGENT_LLM_MODEL")
        logger.info(f"Using api_url mode with url: {api_url}")
        return {"type": "api_url", "api_url": api_url}
    elif llm_type == "hf_model":
        model = os.getenv("AGENT_LLM_MODEL")
        logger.info(f"Using hf_model mode with model: {model}")
        return {"type": "hf_model", "hf_model": model}
    elif llm_type == "model_path":
        model_path = os.getenv("AGENT_LLM_MODEL")
        logger.info(f"Using model_path mode with path: {model_path}")
        return {"type": "model_path", "model_path": model_path}
    else:
        # Default configuration
        logger.warning(f"No valid AGENT_LLM_TYPE found ({llm_type}), using default hf_model configuration")
        return {"type": "hf_model", "hf_model": "Qwen2.5-7B-Instruct"}

def main():
    """Main entry point"""
    global agent, cfg
    try:
        print("\n===================================")
        print("  AI Agent System")
        print("===================================\n")
        print("Starting API server on 0.0.0.0:5656 ...")
        uvicorn.run("start:app", host="0.0.0.0", port=5656, reload=False)
        return 0
    except Exception as e:
        logger.error(f"Error in main: {str(e)}")
        print(f"Fatal error: {str(e)}")
        return 1

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