#!/usr/bin/env bash # ============================================================================= # COCONEWS - POC local (sin Docker, sin ML workers) # Levanta backend + frontend con datos de prueba en ~2 minutos # # Requisitos mínimos: Go 1.25, Node.js 18+, PostgreSQL, Redis # Uso: bash poc/poc.sh # ============================================================================= set -euo pipefail REPO_ROOT="$(cd "$(dirname "${BASH_SOURCE[0]}")/.." && pwd)" POC_DIR="$REPO_ROOT/poc" TMP_DIR="/tmp/coconews-poc" PID_FILE="$TMP_DIR/pids" export PATH=$PATH:/usr/local/go/bin RED='\033[0;31m'; GREEN='\033[0;32m'; YELLOW='\033[1;33m'; BLUE='\033[0;34m'; BOLD='\033[1m'; NC='\033[0m' info() { echo -e "${GREEN}[✓]${NC} $*"; } step() { echo -e "${BLUE}[→]${NC} $*"; } warn() { echo -e "${YELLOW}[!]${NC} $*"; } error() { echo -e "${RED}[✗]${NC} $*"; exit 1; } # Manejar Ctrl+C: para todos los procesos del POC cleanup() { echo "" warn "Deteniendo POC..." if [[ -f "$PID_FILE" ]]; then while IFS= read -r pid; do kill "$pid" 2>/dev/null || true done < "$PID_FILE" rm -f "$PID_FILE" fi echo -e "${YELLOW}POC detenido.${NC}" exit 0 } trap cleanup INT TERM mkdir -p "$TMP_DIR" > "$PID_FILE" echo "" echo -e "${BOLD}=================================================${NC}" echo -e "${BOLD} COCONEWS · POC Local${NC}" echo -e "${BOLD}=================================================${NC}" echo "" # ============================================================================= # 1. VERIFICAR PREREQUISITOS # ============================================================================= step "Verificando prerequisitos..." command -v go &>/dev/null || error "Go no encontrado. Instala Go 1.25+." command -v psql &>/dev/null || error "PostgreSQL no encontrado. Instala postgresql." command -v redis-cli &>/dev/null || error "Redis no encontrado. Instala redis-server." command -v node &>/dev/null || error "Node.js no encontrado. Instala Node.js 18+." info "Prerequisitos OK (Go: $(go version | awk '{print $3}'), Node: $(node -v))" # ============================================================================= # 2. CONFIGURACION POC (sin contrasenas fuertes, solo local) # ============================================================================= POC_DB_NAME="coconews_poc" POC_DB_USER="coconews_poc" POC_DB_PASS="poc_password_local" POC_REDIS_PORT="6380" # Puerto alternativo para no interferir con Redis principal POC_API_PORT="18080" POC_FRONTEND_PORT="18001" export DATABASE_URL="postgres://${POC_DB_USER}:${POC_DB_PASS}@127.0.0.1:5432/${POC_DB_NAME}?sslmode=disable" export REDIS_URL="redis://127.0.0.1:${POC_REDIS_PORT}" export SECRET_KEY="poc_secret_key_solo_para_desarrollo_local" export SERVER_PORT="$POC_API_PORT" export GIN_MODE="release" # ============================================================================= # 3. POSTGRESQL - crear DB de prueba # ============================================================================= step "Preparando base de datos POC (${POC_DB_NAME})..." # Verificar que PostgreSQL está corriendo pg_isready -q || { warn "PostgreSQL no está corriendo. Intentando iniciar..." sudo systemctl start postgresql 2>/dev/null || \ sudo service postgresql start 2>/dev/null || \ error "No se puede iniciar PostgreSQL. Inícialo manualmente." } # Crear usuario y BD de prueba sudo -u postgres psql -q -tc "SELECT 1 FROM pg_roles WHERE rolname='${POC_DB_USER}'" \ | grep -q 1 || \ sudo -u postgres psql -q -c "CREATE USER ${POC_DB_USER} WITH PASSWORD '${POC_DB_PASS}';" 2>/dev/null sudo -u postgres psql -q -tc "SELECT 1 FROM pg_database WHERE datname='${POC_DB_NAME}'" \ | grep -q 1 || \ sudo -u postgres createdb -O "${POC_DB_USER}" "${POC_DB_NAME}" 2>/dev/null # Aplicar schema completo sudo -u postgres psql -q -d "${POC_DB_NAME}" \ -f "$REPO_ROOT/init-db/00-complete-schema.sql" 2>/dev/null || true # Otorgar permisos al usuario POC sudo -u postgres psql -q -d "${POC_DB_NAME}" \ -c "GRANT ALL PRIVILEGES ON ALL TABLES IN SCHEMA public TO ${POC_DB_USER}; GRANT ALL PRIVILEGES ON ALL SEQUENCES IN SCHEMA public TO ${POC_DB_USER};" \ 2>/dev/null || true # Cargar datos de muestra sudo -u postgres psql -q -d "${POC_DB_NAME}" -f "$POC_DIR/seed.sql" 2>/dev/null || true NEWS_COUNT=$(sudo -u postgres psql -tq -d "${POC_DB_NAME}" -c "SELECT COUNT(*) FROM noticias;" 2>/dev/null | tr -d ' ') info "BD lista: ${NEWS_COUNT} noticias de prueba cargadas" # ============================================================================= # 4. REDIS (instancia temporal en puerto alternativo) # ============================================================================= step "Iniciando Redis en puerto ${POC_REDIS_PORT}..." redis-server --port "$POC_REDIS_PORT" --daemonize yes \ --logfile "$TMP_DIR/redis-poc.log" \ --pidfile "$TMP_DIR/redis-poc.pid" \ --maxmemory 128mb --maxmemory-policy allkeys-lru \ 2>/dev/null || warn "Redis ya corriendo en ${POC_REDIS_PORT}" sleep 1 redis-cli -p "$POC_REDIS_PORT" ping &>/dev/null && info "Redis OK (puerto ${POC_REDIS_PORT})" cat "$TMP_DIR/redis-poc.pid" 2>/dev/null >> "$PID_FILE" || true # ============================================================================= # 5. COMPILAR BACKEND GO # ============================================================================= step "Compilando backend Go..." mkdir -p "$TMP_DIR/bin" (cd "$REPO_ROOT/backend" && \ CGO_ENABLED=0 go build -buildvcs=false -o "$TMP_DIR/bin/server" ./cmd/server \ 2>"$TMP_DIR/build-backend.log") || { cat "$TMP_DIR/build-backend.log" error "Fallo al compilar backend. Ver log arriba." } info "Backend compilado OK" # ============================================================================= # 6. ARRANCAR BACKEND API # ============================================================================= step "Arrancando API en puerto ${POC_API_PORT}..." "$TMP_DIR/bin/server" > "$TMP_DIR/backend.log" 2>&1 & BACKEND_PID=$! echo "$BACKEND_PID" >> "$PID_FILE" # Esperar a que el backend responda for i in {1..15}; do sleep 1 if curl -sf "http://127.0.0.1:${POC_API_PORT}/api/stats" &>/dev/null; then info "API respondiendo en http://127.0.0.1:${POC_API_PORT}" break fi if [[ $i -eq 15 ]]; then cat "$TMP_DIR/backend.log" error "El backend no responde. Ver log arriba." fi done # ============================================================================= # 7. FRONTEND REACT # ============================================================================= step "Preparando frontend..." cd "$REPO_ROOT/frontend" if [[ ! -d node_modules ]]; then step " Instalando dependencias npm (primera vez)..." npm install --silent fi # Compilar apuntando al API local del POC VITE_API_URL="http://127.0.0.1:${POC_API_PORT}" \ npm run build -- --outDir "$TMP_DIR/frontend-dist" 2>"$TMP_DIR/build-frontend.log" || { cat "$TMP_DIR/build-frontend.log" error "Fallo al compilar frontend. Ver log arriba." } info "Frontend compilado OK" # Servir frontend con npx serve (simple, sin nginx) step "Sirviendo frontend en puerto ${POC_FRONTEND_PORT}..." npx --yes serve "$TMP_DIR/frontend-dist" -l "$POC_FRONTEND_PORT" \ > "$TMP_DIR/frontend.log" 2>&1 & FRONTEND_PID=$! echo "$FRONTEND_PID" >> "$PID_FILE" sleep 2 cd "$REPO_ROOT" # ============================================================================= # LISTO # ============================================================================= echo "" echo -e "${BOLD}${GREEN}=================================================${NC}" echo -e "${BOLD}${GREEN} COCONEWS POC corriendo${NC}" echo -e "${BOLD}${GREEN}=================================================${NC}" echo "" echo -e " ${BOLD}Frontend:${NC} http://127.0.0.1:${POC_FRONTEND_PORT}" echo -e " ${BOLD}API:${NC} http://127.0.0.1:${POC_API_PORT}/api/stats" echo "" echo -e " ${BOLD}Login:${NC} Registra el primer usuario en la UI" echo -e " (será admin automáticamente)" echo "" echo -e " ${BOLD}Noticias:${NC} ${NEWS_COUNT} artículos de prueba en español" echo -e " ${YELLOW}Nota:${NC} Sin workers ML activos." echo -e " Noticias no tendrán traducción ni entidades." echo "" echo -e " ${BLUE}Logs:${NC} $TMP_DIR/*.log" echo "" echo -e " Pulsa ${BOLD}Ctrl+C${NC} para detener el POC." echo "" # Mantener el script corriendo wait