import os import psutil import subprocess import time import json import asyncio from fastapi import FastAPI, Request, Header, Depends, HTTPException, status from fastapi.responses import HTMLResponse, JSONResponse, FileResponse from fastapi.templating import Jinja2Templates from dotenv import load_dotenv from starlette.concurrency import run_in_threadpool from ai_agent import query_agent_async from config import get_config, save_config from credential_manager import fetch_from_gitea_repo_async from orchestrator import ( orchestrate_async, handle_message_async, get_orchestrator_status, get_llm_config, set_llm_config, format_confirmation_message, format_completion_message ) from llm_providers import get_available_models load_dotenv() app = FastAPI(title="BotVPS API") templates = Jinja2Templates(directory="templates") # ============================================================ # STARTUP # ============================================================ @app.on_event("startup") async def startup_event(): print("[INIT] Sincronizando credenciais...") await fetch_from_gitea_repo_async(force=True) # --- SEGURANÇA --- async def verify_password(x_web_password: str = Header(None)): # Autenticação desativada conforme solicitado return True @app.get("/api/login") async def login_bypass(): return {"status": "ok", "message": "Autenticação desativada"} # --- WEB UI --- @app.get("/", response_class=FileResponse) async def read_root(request: Request): return FileResponse("templates/index.html") @app.get("/api/status") async def get_system_status(is_auth: bool = Depends(verify_password)): vm = psutil.virtual_memory() return { "cpu": psutil.cpu_percent(), "ram": { "percent": vm.percent, "used": round(vm.used / (1024**3), 2), "total": round(vm.total / (1024**3), 2) }, "disk": {"percent": psutil.disk_usage('/').percent} } # --- CONFIGURAÇÃO GERAL --- @app.get("/api/config") async def read_config(is_auth: bool = Depends(verify_password)): return get_config() @app.post("/api/config") async def update_config(cfg: dict, is_auth: bool = Depends(verify_password)): save_config(cfg) return {"status": "success"} # --- CONFIGURAÇÃO LLM (ORQUESTRADOR) --- @app.get("/api/llm-config") async def read_llm_config(is_auth: bool = Depends(verify_password)): return get_llm_config() @app.post("/api/llm-config") async def update_llm_config(cfg: dict, is_auth: bool = Depends(verify_password)): set_llm_config( planner_provider=cfg.get("planner_provider"), planner_model=cfg.get("planner_model"), executor_provider=cfg.get("executor_provider"), executor_model=cfg.get("executor_model") ) return {"status": "success"} @app.get("/api/llm-models") async def list_models(is_auth: bool = Depends(verify_password)): return {"models": get_available_models()} # --- SYNC & ACTIONS --- @app.post("/api/sync-credentials") async def sync_creds(is_auth: bool = Depends(verify_password)): from credential_manager import sync_credentials return sync_credentials() @app.post("/api/sync-from-repo") async def sync_from_repo(is_auth: bool = Depends(verify_password)): await fetch_from_gitea_repo_async(force=True) return {"status": "synced"} @app.post("/api/action") async def run_action(data: dict, is_auth: bool = Depends(verify_password)): action_type = data.get("type") if action_type == "ping": return {"status": "success", "message": "Pong! Servidor respondendo."} if action_type == "restart_bot": # Simula reinício disparando sinal de término - o docker restart cuidará do resto os.system("pkill -9 -f bot_logic.py") return {"status": "success", "message": "Bot reiniciado."} if action_type == "reboot_vps": return {"status": "error", "message": "Reboot bloqueado via Web por segurança."} return {"status": "error", "message": f"Ação {action_type} desconhecida."} @app.get("/api/test_llm") async def test_llm_latency(is_auth: bool = Depends(verify_password)): t0 = time.time() try: reply = await query_agent_async("responda apenas 'pong'") latency = round(time.time() - t0, 2) return {"status": "success", "latency": latency, "reply": reply} except Exception as e: return {"status": "error", "message": str(e)} # --- CHAT & ORCHESTRATION --- @app.post("/api/chat") async def web_chat(message: dict, is_auth: bool = Depends(verify_password)): user_text = message.get("text", "") if not user_text: return {"reply": "Vazio."} reply = await query_agent_async(user_text) return {"reply": reply} @app.post("/api/orchestrate") async def orchestrate_task(task_data: dict, is_auth: bool = Depends(verify_password)): task = task_data.get("task", "") confirmed = task_data.get("confirmed", False) result = await orchestrate_async(task, user_confirmed=confirmed) if result["status"] == "needs_confirmation": return { "status": "needs_confirmation", "plan": result["plan"], "message": format_confirmation_message(result) } return { "status": "completed", "results": result.get("results", []), "message": format_completion_message(result) } @app.get("/api/orchestrator-status") async def get_orch_status(is_auth: bool = Depends(verify_password)): return get_orchestrator_status() # --- SERVER --- if __name__ == "__main__": import uvicorn uvicorn.run("main:app", host="0.0.0.0", port=8000, reload=True)