fix: try to fix error on update on computer switch off

This commit is contained in:
2026-05-19 08:11:28 +02:00
parent 4ba70cefc6
commit 8d0d9835be
7 changed files with 161 additions and 20 deletions

View File

@ -4,7 +4,7 @@ Name[en]=System Updates
Comment=Gérer les mises à jour du système CachyOS
Comment[en]=Manage CachyOS system updates
Exec=cachyos-updater-ui
Icon=software-update-available
Icon=cachyos-updater
Terminal=false
Type=Application
Categories=System;PackageManager;

30
data/cachyos-updater.svg Normal file
View File

@ -0,0 +1,30 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
<defs>
<linearGradient id="a" x1="12" y1="2" x2="20" y2="13" gradientUnits="userSpaceOnUse">
<stop offset="0" stop-color="#00ffcc"/>
<stop offset="1" stop-color="#00ccff"/>
</linearGradient>
<linearGradient id="b" x1="12" y1="22" x2="4" y2="11" gradientUnits="userSpaceOnUse">
<stop offset="0" stop-color="#00aa88"/>
<stop offset="1" stop-color="#00ccff" stop-opacity=".55"/>
</linearGradient>
<linearGradient id="c" x1="8" y1="8" x2="16" y2="16" gradientUnits="userSpaceOnUse">
<stop offset="0" stop-color="#00ffcc"/>
<stop offset="1" stop-color="#020202" stop-opacity="0"/>
</linearGradient>
</defs>
<!-- Flèche du haut (gradient cyan clair → bleu) -->
<path fill="url(#a)"
d="M12 6v3l4-4-4-4v3C7.58 4 4 7.58 4 12c0 1.57.46 3.03 1.24 4.26L6.7 14.8
C6.25 13.97 6 13.01 6 12c0-3.31 2.69-6 6-6z"/>
<!-- Flèche du bas (gradient teal → cyan transparent) -->
<path fill="url(#b)"
d="M18.76 7.74 17.3 9.2c.44.84.7 1.79.7 2.8 0 3.31-2.69 6-6 6v-3l-4 4 4 4v-3
c4.42 0 8-3.58 8-8 0-1.57-.46-3.03-1.24-4.26z"/>
<!-- Reflet interne (style CachyOS) -->
<path fill="url(#c)"
d="M12 6v3l2.12-2.12A5.99 5.99 0 0 0 12 6z"/>
</svg>

After

Width:  |  Height:  |  Size: 1.3 KiB

View File

@ -55,6 +55,12 @@ install -m 644 "$SCRIPT_DIR"/systemd/cachyos-updater.timer \
install -m 644 "$SCRIPT_DIR"/systemd/cachyos-updater-shutdown.service \
/usr/lib/systemd/system/cachyos-updater-shutdown.service
# Icône
install -d /usr/share/icons/hicolor/scalable/apps
install -m 644 "$SCRIPT_DIR"/data/cachyos-updater.svg \
/usr/share/icons/hicolor/scalable/apps/cachyos-updater.svg
gtk-update-icon-cache -f -t /usr/share/icons/hicolor &>/dev/null || true
# Fichier .desktop
install -m 644 "$SCRIPT_DIR"/data/cachyos-updater.desktop \
/usr/share/applications/cachyos-updater.desktop

View File

@ -72,30 +72,26 @@ class Daemon:
existing.add(pkg.name)
self._save()
# Pré-télécharger tous les paquets reportés pendant que le réseau est disponible
safe_names = [p.name for p in safe]
pending_names = [p["name"] for p in self._state["pending_restart"]]
if pending_names:
logger.info(f"Pré-téléchargement des paquets reportés : {', '.join(pending_names)}")
ok = await asyncio.to_thread(download_packages, pending_names)
if not ok:
logger.warning("Pré-téléchargement partiel — une nouvelle tentative aura lieu au prochain cycle")
# Télécharger safe + reportés en une seule transaction pour que pacman
# puisse résoudre toutes les dépendances inter-paquets correctement.
all_names = list(dict.fromkeys(safe_names + pending_names))
self._set_status("downloading")
logger.info(f"Téléchargement ({len(all_names)} paquets) : {', '.join(all_names)}")
ok = await asyncio.to_thread(download_packages, all_names)
if not ok:
self._add_error("Échec du téléchargement des paquets")
self._set_status("error")
return
if not safe:
logger.info("Aucun paquet à installer immédiatement")
self._set_status("idle")
return
# Synchroniser la DB et télécharger
self._set_status("downloading")
safe_names = [p.name for p in safe]
logger.info(f"Téléchargement : {', '.join(safe_names)}")
ok = await asyncio.to_thread(download_packages, safe_names)
if not ok:
self._add_error("Échec du téléchargement des paquets")
self._set_status("error")
return
# Installer les paquets sûrs
self._set_status("installing")
logger.info(f"Installation : {', '.join(safe_names)}")
@ -129,6 +125,51 @@ class Daemon:
self._add_error(str(e))
self._set_status("error")
async def install_pending_cycle(self):
if self._state["status"] not in ("idle", "error"):
logger.info("Une opération est déjà en cours, installation ignorée")
return
pending = self._state.get("pending_restart", [])
if not pending:
logger.info("Aucun paquet en attente à installer")
return
if is_pacman_locked():
logger.info("pacman est verrouillé, impossible d'installer maintenant")
self._add_error("pacman est verrouillé, réessayez dans quelques instants")
return
names = [p["name"] for p in pending]
try:
self._set_status("installing")
logger.info(f"Installation manuelle : {', '.join(names)}")
result = await asyncio.to_thread(install_packages, names)
if result.success:
logger.info("Installation manuelle réussie")
now = datetime.now().isoformat()
for pkg in pending:
self._state["installed_history"].append({**pkg, "installed_at": now})
self._state["installed_history"] = self._state["installed_history"][-200:]
self._state["pending_restart"] = []
installed_set = set(names)
self._state["available_updates"] = [
u for u in self._state["available_updates"]
if u["name"] not in installed_set
]
else:
logger.error(f"Installation manuelle échouée : {result.error}")
self._add_error(f"Échec de l'installation : {result.error}")
self._set_status("idle")
except Exception as e:
logger.exception(f"Erreur lors de l'installation manuelle : {e}")
self._add_error(str(e))
self._set_status("error")
def _add_error(self, message: str):
self._state["errors"].append({
"time": datetime.now().isoformat(),
@ -146,6 +187,9 @@ class Daemon:
if action == "check_now":
asyncio.create_task(self.update_cycle())
response = {"status": "ok", "message": "Vérification lancée"}
elif action == "install_pending":
asyncio.create_task(self.install_pending_cycle())
response = {"status": "ok", "message": "Installation lancée"}
elif action == "get_state":
response = {"status": "ok", "state": self._state}
else:

View File

@ -88,3 +88,28 @@ def install_packages(names: list[str]) -> InstallResult:
failed=names,
error=result.stderr[-800:],
)
def upgrade_cached_packages(expected_names: list[str]) -> InstallResult:
"""Installe toutes les mises à jour présentes dans le cache pacman.
Utilisé à l'extinction pour éviter les échecs de dépendances liés aux
mises à jour partielles : pacman -Su installe l'ensemble cohérent des
paquets déjà téléchargés plutôt qu'une liste isolée.
"""
if not expected_names:
return InstallResult(success=True)
result = subprocess.run(
["pacman", "-Su", "--noconfirm", "--noprogressbar"],
capture_output=True, text=True, timeout=600
)
if result.returncode == 0:
return InstallResult(success=True, installed=expected_names)
else:
return InstallResult(
success=False,
failed=expected_names,
error=result.stderr[-800:],
)

View File

@ -13,7 +13,7 @@ import time
from datetime import datetime
import state as state_mod
from package_manager import install_packages, is_pacman_locked
from package_manager import upgrade_cached_packages, is_pacman_locked
logging.basicConfig(
level=logging.INFO,
@ -209,7 +209,7 @@ def run():
try:
display.update(f"Installation de {len(names)} paquet(s)…")
result = install_packages(names)
result = upgrade_cached_packages(names)
except Exception as e:
logger.exception(f"Erreur inattendue : {e}")

View File

@ -59,6 +59,13 @@ class UpdaterWindow(Adw.ApplicationWindow):
self.pending_group.set_description(
"Ces paquets seront installés automatiquement à la prochaine extinction du PC"
)
self.install_now_btn = Gtk.Button.new_with_label("Installer maintenant")
self.install_now_btn.add_css_class("suggested-action")
self.install_now_btn.add_css_class("pill")
self.install_now_btn.connect("clicked", self._on_install_pending_clicked)
self.pending_group.set_header_suffix(self.install_now_btn)
main_box.append(self.pending_group)
# Groupe "Récemment installées"
@ -233,6 +240,10 @@ class UpdaterWindow(Adw.ApplicationWindow):
pass
self.status_sub_lbl.set_text(sub)
# ── Bouton "Installer maintenant" ────────────────────────────────
can_install = status in ("idle", "error") and bool(pending)
self.install_now_btn.set_sensitive(can_install)
# ── Liste des paquets en attente de redémarrage ──────────────────
for row in self._pending_rows:
self.pending_group.remove(row)
@ -306,6 +317,31 @@ class UpdaterWindow(Adw.ApplicationWindow):
self.refresh_btn.set_sensitive(False)
threading.Thread(target=self._send_check_now, daemon=True).start()
def _on_install_pending_clicked(self, _btn):
self.install_now_btn.set_sensitive(False)
threading.Thread(target=self._send_install_pending, daemon=True).start()
def _send_install_pending(self):
try:
s = sock_module.socket(sock_module.AF_UNIX, sock_module.SOCK_STREAM)
s.settimeout(5)
s.connect(str(SOCKET_PATH))
s.sendall(json.dumps({"action": "install_pending"}).encode())
s.recv(1024)
s.close()
except PermissionError:
GLib.idle_add(
self._show_toast,
"Permission refusée — êtes-vous dans le groupe wheel ?",
)
GLib.idle_add(self.install_now_btn.set_sensitive, True)
except Exception as e:
GLib.idle_add(
self._show_toast,
f"Impossible de contacter le service : {e}",
)
GLib.idle_add(self.install_now_btn.set_sensitive, True)
def _send_check_now(self):
try:
s = sock_module.socket(sock_module.AF_UNIX, sock_module.SOCK_STREAM)