Update oginstaller to create release file
oginstaller/pipeline/head There was a failure building this commit Details

main
Nicolas Arenas 2025-04-03 09:24:57 +02:00
parent 8110058303
commit 65a086064b
2 changed files with 201 additions and 43 deletions

View File

@ -474,6 +474,14 @@ class MyApp(npyscreen.NPSAppManaged):
# Silenciar errores
return
# Crear el archivo de versión instalada
try:
os.makedirs("/opt/opengnsys", exist_ok=True)
with open("/opt/opengnsys/release", "w") as release_file:
release_file.write(f"Versión instalada: {selected_tag}\n")
except Exception as e:
print(f"Error al crear el archivo de versión: {e}")
# Actualizar los repositorios
try:
subprocess.run('apt-get update', shell=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL, check=True)
@ -509,4 +517,10 @@ class MyApp(npyscreen.NPSAppManaged):
install_components_with_ui(form, self.selected_components, self.selected_tag)
if __name__ == "__main__":
MyApp().run()
try:
MyApp().run()
except Exception as e:
print(f"[ERROR] Ocurrió un error: {e}")
finally:
# Restaurar el terminal al estado normal
npyscreen.wrapper_basic(lambda: None)

View File

@ -4,24 +4,27 @@ import subprocess
import requests
import sys
import npyscreen
import re
import os
# Configuración general
REPO_BASE_URL = "http://ognproject.evlt.uma.es/debian-opengnsys/opengnsys-devel"
RELEASES_URL = "https://ognproject.evlt.uma.es/debian-opengnsys/versions-dev.json"
APT_LIST_PATH = "/etc/apt/sources.list.d/opengnsys.list"
PACKAGES = ["ogrepository", "ogcore", "oggui", "ogclient", "ogboot", "ogdhcp"]
RELEASE_FILE = "/opt/opengnsys/release"
# === Sección npyscreen ===
class ServerURLForm(npyscreen.Form):
def create(self):
self.server_url = self.add(npyscreen.TitleText, name="Servidor de validación (URL completa):")
self.server_url = self.add(npyscreen.TitleText, name="Servidor de validación (URL completa):", value="http://localhost:5000/validar")
def afterEditing(self):
self.parentApp.server_url = self.server_url.value
self.parentApp.setNextForm("RELEASE")
class ReleaseSelectorForm(npyscreen.FormBaseNew):
class ReleaseSelectorForm(npyscreen.ActionForm):
def create(self):
self.releases = self.parentApp.releases
self.listbox = self.add(npyscreen.TitleSelectOne,
@ -30,9 +33,16 @@ class ReleaseSelectorForm(npyscreen.FormBaseNew):
scroll_exit=True,
max_height=len(self.releases)+4)
def afterEditing(self):
def on_ok(self):
selected_index = self.listbox.value[0] if self.listbox.value else None
self.parentApp.selected = self.releases[selected_index] if selected_index is not None else None
if selected_index is None:
npyscreen.notify_confirm("Debes seleccionar una release antes de continuar.", title="Error")
else:
self.parentApp.selected = self.releases[selected_index]
self.parentApp.setNextForm(None)
def on_cancel(self):
npyscreen.notify_confirm("Operación cancelada. Saliendo del formulario.", title="Cancelado")
self.parentApp.setNextForm(None)
class ReleaseSelectorApp(npyscreen.NPSAppManaged):
@ -53,12 +63,32 @@ def choose_release_and_server(releases):
# === Funciones principales ===
def backup_file(filepath):
"""Crea una copia de seguridad del archivo especificado."""
backup_path = f"{filepath}.bak"
if os.path.exists(filepath):
try:
os.replace(filepath, backup_path)
print(f"[INFO] Copia de seguridad creada: {backup_path}")
except Exception as e:
print(f"[ERROR] No se pudo crear la copia de seguridad de {filepath}: {e}")
return backup_path
def restore_file(backup_path, original_path):
"""Restaura el archivo desde su copia de seguridad."""
if os.path.exists(backup_path):
try:
os.replace(backup_path, original_path)
print(f"[INFO] Archivo restaurado: {original_path}")
except Exception as e:
print(f"[ERROR] No se pudo restaurar el archivo {original_path}: {e}")
def get_installed_packages():
installed = []
for pkg in PACKAGES:
try:
subprocess.run(
["dpkg-query", "-W", "-f=${Status}", pkg],
["dpkg-query", "-W", pkg],
stdout=subprocess.PIPE,
stderr=subprocess.DEVNULL,
text=True,
@ -69,21 +99,18 @@ def get_installed_packages():
continue
return installed
def get_installed_versions(packages):
versions = {}
for pkg in packages:
try:
result = subprocess.run(
["dpkg-query", "-W", "-f=${Version}", pkg],
stdout=subprocess.PIPE,
stderr=subprocess.DEVNULL,
text=True,
check=True
)
versions[pkg] = result.stdout.strip()
except subprocess.CalledProcessError:
versions[pkg] = None
return versions
def get_installed_release():
try:
with open(RELEASE_FILE, "r") as release_file:
line = release_file.readline().strip()
match = re.search(r".*:\s*(.+)", line)
if match:
return match.group(1).strip()
except FileNotFoundError:
print("El archivo de release no existe.")
except Exception as e:
print(f"Error al leer el archivo de release: {e}")
return None
def fetch_available_releases():
try:
@ -96,6 +123,7 @@ def fetch_available_releases():
sys.exit(1)
def update_repo_file(selected_release):
backup_path = backup_file(APT_LIST_PATH)
line = f"deb {REPO_BASE_URL}/{selected_release} noble main\n"
print(f"[INFO] Escribiendo nueva línea en {APT_LIST_PATH}:\n{line.strip()}")
try:
@ -103,11 +131,21 @@ def update_repo_file(selected_release):
f.write(line)
except PermissionError:
print("[ERROR] No tienes permisos para escribir en el archivo del repositorio. Ejecuta el script como root.")
restore_file(backup_path, APT_LIST_PATH)
sys.exit(1)
def check_compatibility(server_url, installed_versions, selected_release):
# Ejecutar apt update para actualizar la información del repositorio
try:
print("[INFO] Actualizando la información del repositorio con 'apt update'...")
subprocess.run(["sudo", "apt", "update"], check=True)
except subprocess.CalledProcessError as e:
print(f"[ERROR] Error al ejecutar 'apt update': {e}")
restore_file(backup_path, APT_LIST_PATH)
sys.exit(1)
def check_compatibility(server_url, installed_release, selected_release):
payload = {
"installed_versions": installed_versions,
"installed_release": installed_release,
"target_release": selected_release
}
try:
@ -119,49 +157,155 @@ def check_compatibility(server_url, installed_versions, selected_release):
print(f"[ERROR] No se pudo contactar con el servidor de validación: {e}")
return False, str(e)
def update_and_install(installed_packages):
def summarize_updates(installed_packages, selected_release):
"""Genera un resumen de los paquetes que se van a actualizar y los que no."""
to_update = []
up_to_date = []
for pkg in installed_packages:
try:
result = subprocess.run(
["apt-cache", "policy", pkg],
stdout=subprocess.PIPE,
stderr=subprocess.DEVNULL,
text=True,
check=True,
env={"LANG": "C"} # Forzar el idioma a inglés
)
installed = None
candidate = None
for line in result.stdout.splitlines():
if "Installed:" in line: # Siempre estará en inglés
installed = line.split(":", 1)[1].strip()
elif "Candidate:" in line: # Siempre estará en inglés
candidate = line.split(":", 1)[1].strip()
if not installed or candidate == "(none)":
to_update.append(f"{pkg} (no instalado o sin versión candidata)")
elif installed != candidate:
to_update.append(f"{pkg} ({installed}{candidate})")
else:
up_to_date.append(f"{pkg} ({installed})")
except subprocess.CalledProcessError:
to_update.append(f"{pkg} (error obteniendo versión)")
summary = "\n--- Resumen de actualización ---\n"
summary += f"Release objetivo: {selected_release}\n\n"
summary += "Paquetes que se actualizarán:\n"
summary += "\n".join(f" - {line}" for line in to_update) if to_update else " - Ninguno\n"
summary += "\nPaquetes que ya están actualizados:\n"
summary += "\n".join(f" - {line}" for line in up_to_date) if up_to_date else " - Ninguno\n"
summary += "\n--------------------------------"
# Mostrar el resumen en una ventana emergente
npyscreen.notify_confirm(summary, title="Resumen de actualización", wide=True)
if not to_update:
npyscreen.notify_confirm("[INFO] Todos los paquetes están actualizados. No es necesario continuar.", title="Información")
sys.exit(0)
if not npyscreen.notify_yes_no("¿Deseas continuar con la actualización?", title="Confirmación"):
npyscreen.notify_confirm("[INFO] Actualización cancelada por el usuario.", title="Cancelado")
restore_file(f"{APT_LIST_PATH}.bak", APT_LIST_PATH)
restore_file(f"{RELEASE_FILE}.bak", RELEASE_FILE)
sys.exit(0)
return [line.split()[0] for line in to_update]
def show_final_versions(packages):
print("\n✅ Resumen final de versiones instaladas:")
for pkg in packages:
try:
result = subprocess.run(
["dpkg-query", "-W", "-f=${Version}", pkg],
stdout=subprocess.PIPE,
stderr=subprocess.DEVNULL,
text=True
)
version = result.stdout.strip()
print(f" - {pkg}: {version}")
except subprocess.CalledProcessError:
print(f" - {pkg}: no instalado")
def update_and_install(packages_to_update, selected_release):
backup_release = backup_file(RELEASE_FILE)
try:
subprocess.run(["sudo", "apt", "update"], check=True)
subprocess.run(["sudo", "apt", "install", "-y"] + installed_packages, check=True)
subprocess.run(["sudo", "apt", "install", "-y"] + packages_to_update, check=True)
print("[INFO] Paquetes actualizados correctamente.")
# Actualizar el archivo de release con la versión seleccionada
try:
os.makedirs(os.path.dirname(RELEASE_FILE), exist_ok=True)
with open(RELEASE_FILE, "w") as release_file:
release_file.write(f"Versión instalada: {selected_release}\n")
print(f"[INFO] Archivo de release actualizado: {selected_release}")
except Exception as e:
print(f"[ERROR] No se pudo actualizar el archivo de release: {e}")
show_final_versions(packages_to_update)
except subprocess.CalledProcessError as e:
print(f"[ERROR] Error al instalar paquetes: {e}")
restore_file(backup_release, RELEASE_FILE)
sys.exit(1)
# === Entrada principal ===
def main():
print("[INFO] Detectando paquetes instalados...")
installed = get_installed_packages()
if not installed:
print("[WARN] No hay paquetes opengnsys instalados.")
sys.exit(0)
print(f"[INFO] Paquetes detectados: {', '.join(installed)}")
releases = fetch_available_releases()
if not releases:
print("[ERROR] No se encontraron releases disponibles.")
print("[INFO] Iniciando actualización de paquetes de OpenGnSys...")
installed_release = get_installed_release()
if installed_release:
print(f"[INFO] Versión instalada: {installed_release}")
else:
print("[WARN] No se encontró la versión instalada.")
sys.exit(1)
installed = get_installed_packages()
if not installed:
print("[ERROR] No se detectaron paquetes OpenGnSys instalados.")
sys.exit(1)
releases = fetch_available_releases()
selected, server_url = choose_release_and_server(releases)
if not selected or not server_url:
print("[WARN] No se seleccionó release o URL del servidor. Cancelando.")
print("[WARN] No se seleccionó release o URL del servidor. Restaurando archivos.")
restore_file(f"{APT_LIST_PATH}.bak", APT_LIST_PATH)
restore_file(f"{RELEASE_FILE}.bak", RELEASE_FILE)
sys.exit(0)
print(f"[INFO] Validando compatibilidad con {server_url}...")
installed_versions = get_installed_versions(installed)
compatible, message = check_compatibility(server_url, installed_versions, selected)
compatible, message = check_compatibility(server_url, installed_release, selected)
if not compatible:
print(f"[ERROR] El servidor indica que la actualización no es compatible: {message}")
restore_file(f"{APT_LIST_PATH}.bak", APT_LIST_PATH)
restore_file(f"{RELEASE_FILE}.bak", RELEASE_FILE)
sys.exit(1)
else:
print(f"[INFO] Compatibilidad validada: {message}")
update_repo_file(selected)
update_and_install(installed)
try:
update_repo_file(selected)
to_update = summarize_updates(installed, selected)
update_and_install(to_update, selected)
except Exception as e:
print(f"[ERROR] Error durante la actualización: {e}")
restore_file(f"{APT_LIST_PATH}.bak", APT_LIST_PATH)
restore_file(f"{RELEASE_FILE}.bak", RELEASE_FILE)
sys.exit(1)
if __name__ == "__main__":
main()
try:
main()
except SystemExit as e:
# Manejar la excepción SystemExit para evitar interrupciones
if e.code != 0:
print(f"[INFO] El script terminó con código de salida: {e.code}")
except Exception as e:
print(f"[ERROR] Ocurrió un error inesperado: {e}")
finally:
# Restaurar el terminal al estado normal
npyscreen.wrapper_basic(lambda stdscr: None)