#!/bin/bash set -e APP_NAME="rss" DB_NAME="rss" DB_USER="rss" APP_USER="x" APP_DIR=$(pwd) PYTHON_ENV="$APP_DIR/venv" WSGI_APP_ENTRY="app:app" WEB_PORT=8000 echo "🟢 Paso 0: Verificaciones y confirmación de seguridad" if [[ $EUID -ne 0 ]]; then echo "❌ Este script debe ser ejecutado como root (usa sudo)." exit 1 fi echo "------------------------------------------------------------------" echo "⚠️ ADVERTENCIA: Este script realizará las siguientes acciones DESTRUCTIVAS:" echo " - Eliminará TODOS los servicios systemd que empiecen por '$APP_NAME'." echo " - Eliminará PERMANENTEMENTE la base de datos '$DB_NAME'." echo " - Eliminará PERMANENTEMENTE el usuario de base de datos '$DB_USER'." echo "------------------------------------------------------------------" read -p "Estás seguro de que quieres continuar? (escribe 'si' para confirmar): " CONFIRM if [ "$CONFIRM" != "si" ]; then echo "Operación cancelada por el usuario." exit 0 fi read -sp "🔑 Introduce la contraseña para el usuario de la base de datos '$DB_USER' (se creará de nuevo): " DB_PASS echo if [ -z "$DB_PASS" ]; then echo "❌ La contraseña no puede estar vacía. Abortando." exit 1 fi echo "🧹 Paso 0.5: Limpiando instalación anterior..." echo " -> Buscando y eliminando servicios systemd antiguos..." for service in $(systemctl list-unit-files | grep "^$APP_NAME" | cut -d' ' -f1); do echo " -> Deteniendo y deshabilitando $service" systemctl stop "$service" || true systemctl disable "$service" || true done rm -f /etc/systemd/system/$APP_NAME* systemctl daemon-reload echo " -> Servicios systemd limpiados." echo "🟢 Paso 1: Instalando dependencias del sistema..." apt-get update apt-get install -y wget ca-certificates postgresql postgresql-contrib python3-venv python3-pip python3-dev libpq-dev gunicorn echo "🔥 Paso 2: Eliminando y recreando la base de datos y el usuario..." ## CORRECCIÓN: Se exporta PGPASSWORD para evitar exponer la contraseña en la línea de comandos ## y se agrupa con la del paso 4 para mayor eficiencia. export PGPASSWORD="$DB_PASS" sudo -u postgres psql -c "DROP DATABASE IF EXISTS $DB_NAME;" sudo -u postgres psql -c "DROP USER IF EXISTS $DB_USER;" echo " -> Entidades de BD anteriores eliminadas." sudo -u postgres psql -c "CREATE USER $DB_USER WITH PASSWORD '$DB_PASS';" sudo -u postgres psql -c "CREATE DATABASE $DB_NAME OWNER $DB_USER;" echo "✅ Base de datos y usuario recreados con éxito." echo "🐍 Paso 3: Configurando el entorno de la aplicación..." if ! id "$APP_USER" &>/dev/null; then echo "👤 Creando usuario del sistema '$APP_USER'..." useradd -m -s /bin/bash "$APP_USER" else echo "✅ Usuario del sistema '$APP_USER' ya existe." fi chown -R "$APP_USER":"$APP_USER" "$APP_DIR" sudo -u "$APP_USER" bash < Creando entorno virtual en $PYTHON_ENV" rm -rf "$PYTHON_ENV" python3 -m venv "$PYTHON_ENV" echo " -> Instalando dependencias desde requirements.txt..." "$PYTHON_ENV/bin/python" -m pip install --upgrade pip if [ -f "requirements.txt" ]; then "$PYTHON_ENV/bin/python" -m pip install -r "requirements.txt" else echo "⚠️ ADVERTENCIA: No se encontró requirements.txt." fi EOF echo "🧠 Paso 3.5: Descargando modelos de lenguaje para Newspaper3k..." if [ -f "download_models.py" ]; then sudo -u "$APP_USER" "$PYTHON_ENV/bin/python" "$APP_DIR/download_models.py" echo "✅ Modelos NLP verificados/descargados." else echo "⚠️ ADVERTENCIA: No se encontró download_models.py. El scraping de URLs puede fallar." fi echo "📐 Paso 4: Creando esquema de BD y sembrando datos..." # La variable PGPASSWORD ya está exportada desde el paso 2 psql -U "$DB_USER" -h localhost -d "$DB_NAME" < Buscando archivos .sql para sembrar datos..." if [ -f "continentes.sql" ]; then echo " -> Cargando continentes.sql..." psql -U "$DB_USER" -h localhost -d "$DB_NAME" -f "continentes.sql" fi if [ -f "categorias.sql" ]; then echo " -> Cargando categorias.sql..." psql -U "$DB_USER" -h localhost -d "$DB_NAME" -f "categorias.sql" fi if [ -f "paises.sql" ]; then echo " -> Cargando paises.sql..." psql -U "$DB_USER" -h localhost -d "$DB_NAME" -f "paises.sql" fi echo " -> Actualizando contadores de secuencias de la base de datos..." psql -U "$DB_USER" -h localhost -d "$DB_NAME" < /etc/systemd/system/$APP_NAME.service [Unit] Description=Gunicorn instance to serve $APP_NAME After=network.target [Service] User=$APP_USER Group=$APP_USER WorkingDirectory=$APP_DIR Environment="PATH=$PYTHON_ENV/bin" Environment="SECRET_KEY=$(python3 -c 'import os; print(os.urandom(24).hex())')" Environment="DB_HOST=localhost" Environment="DB_PORT=5432" Environment="DB_NAME=$DB_NAME" Environment="DB_USER=$DB_USER" Environment="DB_PASS=$DB_PASS" ExecStart=$PYTHON_ENV/bin/gunicorn --workers 3 --bind 0.0.0.0:$WEB_PORT --timeout 120 $WSGI_APP_ENTRY Restart=always [Install] WantedBy=multi-user.target EOF ## CORRECCIÓN: Se elimina la creación de rss-worker.service y rss-worker.timer. ## Se añade el nuevo servicio para el planificador persistente scheduler.py. cat < /etc/systemd/system/$APP_NAME-scheduler.service [Unit] Description=$APP_NAME Scheduler Worker After=postgresql.service [Service] Type=simple User=$APP_USER Group=$APP_USER WorkingDirectory=$APP_DIR Environment="PATH=$PYTHON_ENV/bin" Environment="SECRET_KEY=$(python3 -c 'import os; print(os.urandom(24).hex())')" Environment="DB_HOST=localhost" Environment="DB_PORT=5432" Environment="DB_NAME=$DB_NAME" Environment="DB_USER=$DB_USER" Environment="DB_PASS=$DB_PASS" ExecStart=$PYTHON_ENV/bin/python $APP_DIR/scheduler.py Restart=always [Install] WantedBy=multi-user.target EOF echo "✅ Archivos de servicio creados." echo "🚀 Paso 7: Recargando, habilitando, arrancando servicios y configurando firewall..." systemctl daemon-reload systemctl enable $APP_NAME.service systemctl start $APP_NAME.service ## CORRECCIÓN: Se habilitan e inician el nuevo servicio del planificador. systemctl enable $APP_NAME-scheduler.service systemctl start $APP_NAME-scheduler.service if command -v ufw &> /dev/null && ufw status | grep -q 'Status: active'; then echo " -> Firewall UFW detectado. Abriendo puerto $WEB_PORT..." ufw allow $WEB_PORT/tcp ufw reload fi echo "" echo "🎉 ¡REINSTALACIÓN COMPLETADA!" echo "--------------------------------" echo "" echo "✅ La aplicación web está ahora accesible en:" echo " http://:$WEB_PORT" echo "" echo "Puedes verificar el estado de los servicios con:" echo "sudo systemctl status $APP_NAME.service" ## CORRECCIÓN: Se actualiza el mensaje para reflejar el nuevo nombre del servicio del worker. echo "sudo systemctl status $APP_NAME-scheduler.service" echo "" echo "Para ver los logs de la aplicación web:" echo "sudo journalctl -u $APP_NAME.service -f" echo "Para ver los logs del planificador de noticias:" echo "sudo journalctl -u $APP_NAME-scheduler.service -f"