🚀 Auto-deploy: BotVPS atualizado em 21/04/2026 20:42:48

This commit is contained in:
2026-04-21 20:42:48 +00:00
parent 0ab0fb826c
commit c538fdb9c2
4 changed files with 186 additions and 0 deletions

63
deploy_manager.py Normal file
View File

@@ -0,0 +1,63 @@
import os
import subprocess
import shutil
from typing import Dict
class DeployManager:
BASE_APPS_DIR = "/root/Apps"
@staticmethod
def run_command(command: str, cwd: str = None):
result = subprocess.run(command, shell=True, capture_output=True, text=True, cwd=cwd)
return {
"success": result.returncode == 0,
"output": (result.stdout or result.stderr).strip()
}
def magic_deploy(self, git_url: str) -> str:
if not git_url.endswith(".git"):
return "Erro: URL do repositório inválida (deve terminar em .git)."
repo_name = git_url.split("/")[-1].replace(".git", "")
target_dir = os.path.join(self.BASE_APPS_DIR, repo_name)
if os.path.exists(target_dir):
return f"Erro: O diretório {target_dir} já existe. Use /bash git pull se quiser atualizar."
# 1. Clone
print(f"[DEPLOY] Clonando {git_url}...")
clone_res = self.run_command(f"git clone {git_url} {target_dir}")
if not clone_res["success"]:
return f"Falha no clone: {clone_res['output']}"
# 2. Detecção de Stack
files = os.listdir(target_dir)
strategy = ""
if "docker-compose.yml" in files or "docker-compose.yaml" in files:
strategy = "Docker Compose"
deploy_res = self.run_command("docker compose up -d", cwd=target_dir)
elif "package.json" in files:
strategy = "Node.js (PM2)"
self.run_command("npm install", cwd=target_dir)
deploy_res = self.run_command(f"pm2 start index.js --name {repo_name}", cwd=target_dir)
elif "requirements.txt" in files:
strategy = "Python (PM2/Gunicorn)"
self.run_command("pip install -r requirements.txt", cwd=target_dir)
# Tenta encontrar o arquivo principal
main_file = "main.py" if "main.py" in files else ("app.py" if "app.py" in files else None)
if main_file:
deploy_res = self.run_command(f"pm2 start {main_file} --name {repo_name}", cwd=target_dir)
else:
return f"Repositório clonado ({strategy}), mas não encontrei main.py ou app.py para iniciar."
else:
return f"Repositório clonado em {target_dir}, mas não detectei uma stack automática (Docker, Node ou Python)."
if deploy_res["success"]:
return f"✅ **Deploy Mágico Concluído!**\n- **Repo:** {repo_name}\n- **Estratégia:** {strategy}\n- **Status:** Sucesso"
else:
return f"❌ **Falha no Deploy ({strategy}):**\n{deploy_res['output']}"
if __name__ == "__main__":
# Teste rápido
pass

View File

@@ -17,6 +17,13 @@ module.exports = {
interpreter: "python3",
cwd: "/root/Apps/BotVPS",
restart_delay: 3000
},
{
name: "watchdog-vps",
script: "watchdog.py",
interpreter: "python3",
cwd: "/root/Apps/BotVPS",
restart_delay: 5000
}
]
};

View File

@@ -7,6 +7,7 @@ from credential_manager import (
gitea_api_url, gitea_token, supabase_url, supabase_anon_key,
supabase_service_role_key
)
from deploy_manager import DeployManager
# ============================================================
# UTILS
@@ -138,6 +139,16 @@ class SystemTools:
except Exception as e:
return f"Erro ao listar {path}: {e}"
@staticmethod
def pm2_status() -> str:
"""Lista todos os processos gerenciados pelo PM2."""
return run_bash("pm2 jlist")["output"]
@staticmethod
def pm2_restart(name: str) -> str:
"""Reinicia um processo no PM2 pelo nome ou ID."""
return run_bash(f"pm2 restart {name}")["output"]
# ============================================================
# WORKSPACE TOOLS (GWS CLI)
# ============================================================
@@ -148,6 +159,12 @@ class WorkspaceTools:
"""Executa um comando gws completo (ex: gws-mr gmail +send ...)."""
return run_bash(cmd)["output"]
@staticmethod
def magic_deploy(git_url: str) -> str:
"""Clona um repositório git e tenta realizar o deploy automático (Docker/Node/Python)."""
dm = DeployManager()
return dm.magic_deploy(git_url)
# ============================================================
# REGISTRY
# ============================================================
@@ -172,6 +189,9 @@ TOOLS_V2 = {
"read_file": {"desc": "Lê conteúdo de um arquivo", "func": SystemTools.read_file, "danger": "safe"},
"write_file": {"desc": "Cria ou edita arquivo (caminho|conteúdo)", "func": SystemTools.write_file, "danger": "dangerous"},
"ls": {"desc": "Lista arquivos num diretório", "func": SystemTools.list_dir, "danger": "safe"},
"pm2_status": {"desc": "Status dos processos PM2", "func": SystemTools.pm2_status, "danger": "safe"},
"pm2_restart": {"desc": "Reiniciar processo PM2", "func": SystemTools.pm2_restart, "danger": "medium"},
"magic_deploy": {"desc": "Deploy automático via URL Git", "func": WorkspaceTools.magic_deploy, "danger": "dangerous"},
# Google Workspace
"gws": {"desc": "Executa comando GWS CLI (ex: gws-mr drive files list)", "func": WorkspaceTools.gws_command, "danger": "medium"},

96
watchdog.py Normal file
View File

@@ -0,0 +1,96 @@
import os
import psutil
import time
import json
import httpx
import subprocess
from dotenv import load_dotenv
load_dotenv()
TOKEN = os.getenv("TELEGRAM_BOT_TOKEN")
CHAT_ID = os.getenv("TELEGRAM_CHAT_ID")
# Configurações do Watchdog
CPU_THRESHOLD = 90.0
CPU_STREAK_LIMIT = 6 # 6 * 10s = 60s
CHECK_INTERVAL = 10 # segundos
class Watchdog:
def __init__(self):
self.cpu_streak = 0
self.last_alert_time = 0
self.alert_cooldown = 300 # 5 minutos entre alertas do mesmo tipo
async def send_telegram_message(self, message: str):
if not TOKEN or not CHAT_ID:
print("[WATCHDOG] Erro: TOKEN ou CHAT_ID não configurados.")
return
url = f"https://api.telegram.org/bot{TOKEN}/sendMessage"
payload = {
"chat_id": CHAT_ID,
"text": f"🚨 **[WATCHDOG VPS]**\n\n{message}",
"parse_mode": "Markdown"
}
try:
async with httpx.AsyncClient() as client:
await client.post(url, json=payload)
except Exception as e:
print(f"[WATCHDOG] Erro ao enviar Telegram: {e}")
def get_pm2_status(self):
try:
result = subprocess.run(["pm2", "jlist"], capture_output=True, text=True)
if result.returncode == 0:
data = json.loads(result.stdout)
issues = []
for proc in data:
if proc['pm2_env']['status'] != 'online':
issues.append(f"🔴 App '{proc['name']}' está {proc['pm2_env']['status']}!")
return issues
except Exception as e:
print(f"[WATCHDOG] Erro PM2: {e}")
return []
async def run(self):
print("[WATCHDOG] Iniciado. Vigilância ativa...")
while True:
try:
# 1. Monitoramento de CPU
cpu_usage = psutil.cpu_percent(interval=1)
if cpu_usage > CPU_THRESHOLD:
self.cpu_streak += 1
else:
self.cpu_streak = 0
if self.cpu_streak >= CPU_STREAK_LIMIT:
if time.time() - self.last_alert_time > self.alert_cooldown:
await self.send_telegram_message(
f"CPU em nível crítico: {cpu_usage}% por mais de 1 minuto!"
)
self.last_alert_time = time.time()
# 2. Monitoramento de PM2
pm2_issues = self.get_pm2_status()
if pm2_issues:
await self.send_telegram_message("\n".join(pm2_issues))
# 3. Monitoramento de Espaço em Disco
disk = psutil.disk_usage('/')
if disk.percent > 95:
if time.time() - self.last_alert_time > self.alert_cooldown:
await self.send_telegram_message(f"Espaço em disco crítico: {disk.percent}% ocupado!")
self.last_alert_time = time.time()
await asyncio.sleep(CHECK_INTERVAL)
except Exception as e:
print(f"[WATCHDOG] Erro no loop: {e}")
await asyncio.sleep(CHECK_INTERVAL)
if __name__ == "__main__":
import asyncio
dog = Watchdog()
asyncio.run(dog.run())