From c6e2a842a2b6c59f0d1a524f81ffebb6eec9dfe2 Mon Sep 17 00:00:00 2001 From: gerblesh <101901964+gerblesh@users.noreply.github.com> Date: Sat, 26 Oct 2024 18:47:49 -0700 Subject: [PATCH] fix: use login1 dbus API, fix systemd-run --- src/ublue_update/cli.py | 20 +++++++----- src/ublue_update/session.py | 24 +++++++++----- tests/unit/test_cli.py | 1 - tests/unit/test_session.py | 62 +++++++++++++------------------------ ublue-update.spec | 2 +- 5 files changed, 50 insertions(+), 59 deletions(-) diff --git a/src/ublue_update/cli.py b/src/ublue_update/cli.py index e692146..6c5695b 100644 --- a/src/ublue_update/cli.py +++ b/src/ublue_update/cli.py @@ -11,7 +11,7 @@ from ublue_update.update_inhibitors.hardware import check_hardware_inhibitors from ublue_update.update_inhibitors.custom import check_custom_inhibitors from ublue_update.config import cfg -from ublue_update.session import get_active_sessions +from ublue_update.session import get_active_users from ublue_update.filelock import acquire_lock, release_lock @@ -33,7 +33,7 @@ def notify(title: str, body: str, actions: list = [], urgency: str = "normal"): if process_uid == 0: users = [] try: - users = get_active_sessions() + users = get_active_users() except KeyError as e: log.error("failed to get active logind session info", e) for user in users: @@ -41,8 +41,9 @@ def notify(title: str, body: str, actions: list = [], urgency: str = "normal"): "/usr/bin/systemd-run", "--user", "--machine", - f"{user['user']}@", - "--wait", + f"{user[1]}@", # magic number, corresponds to user name in ListUsers (see session.py) + "--pipe", + "--quiet", ] user_args += args out = subprocess.run(user_args, capture_output=True) @@ -63,8 +64,10 @@ def ask_for_updates(system): "critical", ) if out is None: + log.info("balls") return # if the user has confirmed + print(out.stdout.decode("utf-8")) if "universal-blue-update-confirm" in out.stdout.decode("utf-8"): run_updates(system, True) @@ -106,7 +109,7 @@ def run_updates(system, system_update_available): ) users = [] try: - users = get_active_sessions() + users = get_active_users() except KeyError as e: log.error("failed to get active logind session info", e) @@ -133,15 +136,16 @@ def run_updates(system, system_update_available): """Users""" for user in users: - log.info(f"""Running update for user: '{user['user']}'""") + log.info(f"""Running update for user: '{user[1]}'""") # magic number, corresponds to username (see session.py) out = subprocess.run( [ "/usr/bin/systemd-run", "--setenv=TOPGRADE_SKIP_BRKC_NOTIFY=true", "--user", "--machine", - f"{user['user']}@", - "--wait", + f"{user[1]}@", + "--pipe", + "--quiet", "/usr/bin/topgrade", "--config", "/usr/share/ublue-update/topgrade-user.toml", diff --git a/src/ublue_update/session.py b/src/ublue_update/session.py index 2371452..a54ce23 100644 --- a/src/ublue_update/session.py +++ b/src/ublue_update/session.py @@ -2,14 +2,22 @@ import json -def get_active_sessions(): +def get_active_users(): out = subprocess.run( - ["/usr/bin/loginctl", "list-sessions", "--output=json"], + [ + "/usr/bin/busctl", + "--system", + "-j", + "call", + "org.freedesktop.login1", + "/org/freedesktop/login1", + "org.freedesktop.login1.Manager", + "ListUsers", + ], capture_output=True, ) - sessions = json.loads(out.stdout.decode("utf-8")) - active_sessions = [] - for session in sessions: - if session.get("state") == "active": - active_sessions.append(session) - return active_sessions + # https://www.freedesktop.org/software/systemd/man/latest/org.freedesktop.login1.html + # ListUsers() returns an array of all currently logged in users. The structures in the array consist of the following fields: user id, user name, user object path. + users = json.loads(out.stdout.decode("utf-8")) + # sample output: {'type': 'a(uso)', 'data': [[[1000, 'user', '/org/freedesktop/login1/user/_1000']]] + return users["data"][0] diff --git a/tests/unit/test_cli.py b/tests/unit/test_cli.py index 913644d..e003d71 100644 --- a/tests/unit/test_cli.py +++ b/tests/unit/test_cli.py @@ -47,7 +47,6 @@ def test_notify_uid_user(mock_run, mock_log, mock_os, mock_cfg): capture_output=True, ) - @patch("ublue_update.cli.cfg") def test_ask_for_updates_no_dbus_notify(mock_cfg): mock_cfg.dbus_notify = False diff --git a/tests/unit/test_session.py b/tests/unit/test_session.py index 551e42d..c7c8d6b 100644 --- a/tests/unit/test_session.py +++ b/tests/unit/test_session.py @@ -7,35 +7,10 @@ 0, os.path.abspath(os.path.join(os.path.dirname(__file__), "../../src")) ) -from ublue_update.session import get_active_sessions +from ublue_update.session import get_active_users -loginctl_json_output = b""" -[ - { - "session" : "3", - "uid" : 1001, - "user" : "test", - "seat" : null, - "leader" : 6205, - "class" : "manager", - "tty" : null, - "state": "active", - "idle" : false, - "since" : null - }, - { - "session" : "c1", - "uid" : 1001, - "user" : "test", - "seat" : null, - "leader" : 6230, - "class" : "manager", - "tty" : null, - "state": "inactive", - "idle" : false, - "since" : null - } -] +busctl_json_output = b""" +{'type': 'a(uso)', 'data': [[[1000, 'user', '/org/freedesktop/login1/user/_1000']]] """ @@ -45,19 +20,24 @@ def test_get_active_sessions(mock_run): MagicMock(stdout=loginctl_json_output), ] assert get_active_sessions() == [ - { - "session": "3", - "uid": 1001, - "user": "test", - "seat": None, - "leader": 6205, - "class": "manager", - "tty": None, - "state": "active", - "idle": False, - "since": None, - } + [ + [ + "1000", + "user", + "/org/freedesktop/login1/user/_1000", + ] + ] ] mock_run.assert_any_call( - ["/usr/bin/loginctl", "list-sessions", "--output=json"], capture_output=True + [ + "/usr/bin/busctl", + "--system", + "-j", + "call", + "org.freedesktop.login1", + "/org/freedesktop/login1", + "org.freedesktop.login1.Manager", + "ListUsers", + ], + capture_output=True, ) diff --git a/ublue-update.spec b/ublue-update.spec index 6a2e722..c9b97dc 100644 --- a/ublue-update.spec +++ b/ublue-update.spec @@ -29,7 +29,7 @@ BuildRequires: python-setuptools_scm BuildRequires: python-wheel Requires: skopeo Requires: libnotify -Requires: sudo +Requires: systemd %global sub_name %{lua:t=string.gsub(rpm.expand("%{NAME}"), "^ublue%-", ""); print(t)}