refactor: move systemd functions to utils

This commit is contained in:
Inex Code 2024-03-05 11:55:52 +03:00
parent 2443ae0144
commit 71433da424
10 changed files with 129 additions and 134 deletions

View file

@ -4,10 +4,14 @@ After starting, track the status of the systemd unit and update the Job
status accordingly. status accordingly.
""" """
import subprocess import subprocess
import time
from selfprivacy_api.utils.huey import huey from selfprivacy_api.utils.huey import huey
from selfprivacy_api.jobs import JobStatus, Jobs, Job from selfprivacy_api.jobs import JobStatus, Jobs, Job
from datetime import datetime from selfprivacy_api.utils.waitloop import wait_until_true
from selfprivacy_api.utils.systemd import (
get_service_status,
get_last_log_lines,
ServiceStatus,
)
START_TIMEOUT = 60 * 5 START_TIMEOUT = 60 * 5
START_INTERVAL = 1 START_INTERVAL = 1
@ -15,6 +19,49 @@ RUN_TIMEOUT = 60 * 60
RUN_INTERVAL = 5 RUN_INTERVAL = 5
def check_if_started(unit_name: str):
"""Check if the systemd unit has started"""
try:
status = get_service_status(unit_name)
if status == ServiceStatus.ACTIVE:
return True
return False
except subprocess.CalledProcessError:
return False
def check_running_status(job: Job, unit_name: str):
"""Check if the systemd unit is running"""
try:
status = get_service_status(unit_name)
if status == ServiceStatus.INACTIVE:
Jobs.update(
job=job,
status=JobStatus.FINISHED,
result="System rebuilt.",
progress=100,
)
return True
if status == ServiceStatus.FAILED:
Jobs.update(
job=job,
status=JobStatus.ERROR,
error="System rebuild failed.",
)
return True
if status == ServiceStatus.ACTIVE:
log_lines = get_last_log_lines(unit_name, 1)
Jobs.update(
job=job,
status=JobStatus.RUNNING,
status_text=log_lines[0] if len(log_lines) > 0 else "",
)
return False
return False
except subprocess.CalledProcessError:
return False
@huey.task() @huey.task()
def rebuild_system_task(job: Job, upgrade: bool = False): def rebuild_system_task(job: Job, upgrade: bool = False):
"""Rebuild the system""" """Rebuild the system"""
@ -32,88 +79,39 @@ def rebuild_system_task(job: Job, upgrade: bool = False):
status=JobStatus.RUNNING, status=JobStatus.RUNNING,
status_text="Starting the system rebuild...", status_text="Starting the system rebuild...",
) )
# Get current time to handle timeout
start_time = datetime.now()
# Wait for the systemd unit to start # Wait for the systemd unit to start
while True:
try: try:
status = subprocess.run( wait_until_true(
["systemctl", "is-active", unit_name], lambda: check_if_started(unit_name),
check=True, timeout_sec=START_TIMEOUT,
capture_output=True, interval=START_INTERVAL,
text=True,
) )
if status.stdout.strip() == "active": except TimeoutError:
break
if (datetime.now() - start_time).total_seconds() > START_TIMEOUT:
Jobs.update( Jobs.update(
job=job, job=job,
status=JobStatus.ERROR, status=JobStatus.ERROR,
error="System rebuild timed out.", error="System rebuild timed out.",
) )
return return
time.sleep(START_INTERVAL)
except subprocess.CalledProcessError:
pass
Jobs.update( Jobs.update(
job=job, job=job,
status=JobStatus.RUNNING, status=JobStatus.RUNNING,
status_text="Rebuilding the system...", status_text="Rebuilding the system...",
) )
# Wait for the systemd unit to finish # Wait for the systemd unit to finish
while True:
try: try:
status = subprocess.run( wait_until_true(
["systemctl", "is-active", unit_name], lambda: check_running_status(job, unit_name),
check=False, timeout_sec=RUN_TIMEOUT,
capture_output=True, interval=RUN_INTERVAL,
text=True,
) )
if status.stdout.strip() == "inactive": except TimeoutError:
Jobs.update(
job=job,
status=JobStatus.FINISHED,
result="System rebuilt.",
progress=100,
)
break
elif status.stdout.strip() == "failed":
Jobs.update(
job=job,
status=JobStatus.ERROR,
error="System rebuild failed.",
)
break
elif status.stdout.strip() == "active":
log_line = subprocess.run(
[
"journalctl",
"-u",
unit_name,
"-n",
"1",
"-o",
"cat",
],
check=False,
capture_output=True,
text=True,
).stdout.strip()
Jobs.update(
job=job,
status=JobStatus.RUNNING,
status_text=f"{log_line}",
)
except subprocess.CalledProcessError:
pass
if (datetime.now() - start_time).total_seconds() > RUN_TIMEOUT:
Jobs.update( Jobs.update(
job=job, job=job,
status=JobStatus.ERROR, status=JobStatus.ERROR,
error="System rebuild timed out.", error="System rebuild timed out.",
) )
break return
time.sleep(RUN_INTERVAL)
except subprocess.CalledProcessError as e: except subprocess.CalledProcessError as e:
Jobs.update( Jobs.update(

View file

@ -5,9 +5,9 @@ import typing
from selfprivacy_api.jobs import Job, Jobs from selfprivacy_api.jobs import Job, Jobs
from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service
from selfprivacy_api.services.generic_status_getter import get_service_status from selfprivacy_api.utils.systemd import get_service_status
from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus
from selfprivacy_api.utils import ReadUserData, WriteUserData, get_domain from selfprivacy_api.utils import get_domain
from selfprivacy_api.utils.block_devices import BlockDevice from selfprivacy_api.utils.block_devices import BlockDevice
import selfprivacy_api.utils.network as network_utils import selfprivacy_api.utils.network as network_utils
from selfprivacy_api.services.bitwarden.icon import BITWARDEN_ICON from selfprivacy_api.services.bitwarden.icon import BITWARDEN_ICON

View file

@ -5,9 +5,9 @@ import typing
from selfprivacy_api.jobs import Job, Jobs from selfprivacy_api.jobs import Job, Jobs
from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service
from selfprivacy_api.services.generic_status_getter import get_service_status from selfprivacy_api.utils.systemd import get_service_status
from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus
from selfprivacy_api.utils import ReadUserData, WriteUserData, get_domain from selfprivacy_api.utils import get_domain
from selfprivacy_api.utils.block_devices import BlockDevice from selfprivacy_api.utils.block_devices import BlockDevice
import selfprivacy_api.utils.network as network_utils import selfprivacy_api.utils.network as network_utils
from selfprivacy_api.services.gitea.icon import GITEA_ICON from selfprivacy_api.services.gitea.icon import GITEA_ICON

View file

@ -4,11 +4,11 @@ import subprocess
import typing import typing
from selfprivacy_api.jobs import Job from selfprivacy_api.jobs import Job
from selfprivacy_api.services.generic_status_getter import ( from selfprivacy_api.utils.systemd import (
get_service_status_from_several_units, get_service_status_from_several_units,
) )
from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus
from selfprivacy_api.utils import ReadUserData, WriteUserData, get_domain from selfprivacy_api.utils import get_domain
from selfprivacy_api.utils.block_devices import BlockDevice from selfprivacy_api.utils.block_devices import BlockDevice
import selfprivacy_api.utils.network as network_utils import selfprivacy_api.utils.network as network_utils
from selfprivacy_api.services.jitsimeet.icon import JITSI_ICON from selfprivacy_api.services.jitsimeet.icon import JITSI_ICON

View file

@ -6,7 +6,7 @@ import typing
from selfprivacy_api.jobs import Job, Jobs from selfprivacy_api.jobs import Job, Jobs
from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service
from selfprivacy_api.services.generic_status_getter import ( from selfprivacy_api.utils.systemd import (
get_service_status_from_several_units, get_service_status_from_several_units,
) )
from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus

View file

@ -4,9 +4,9 @@ import subprocess
import typing import typing
from selfprivacy_api.jobs import Job, Jobs from selfprivacy_api.jobs import Job, Jobs
from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service
from selfprivacy_api.services.generic_status_getter import get_service_status from selfprivacy_api.utils.systemd import get_service_status
from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus
from selfprivacy_api.utils import ReadUserData, WriteUserData, get_domain from selfprivacy_api.utils import get_domain
from selfprivacy_api.utils.block_devices import BlockDevice from selfprivacy_api.utils.block_devices import BlockDevice
import selfprivacy_api.utils.network as network_utils import selfprivacy_api.utils.network as network_utils
from selfprivacy_api.services.nextcloud.icon import NEXTCLOUD_ICON from selfprivacy_api.services.nextcloud.icon import NEXTCLOUD_ICON

View file

@ -3,9 +3,8 @@ import base64
import subprocess import subprocess
import typing import typing
from selfprivacy_api.jobs import Job from selfprivacy_api.jobs import Job
from selfprivacy_api.services.generic_status_getter import get_service_status from selfprivacy_api.utils.systemd import get_service_status
from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus
from selfprivacy_api.utils import ReadUserData, WriteUserData
from selfprivacy_api.utils.block_devices import BlockDevice from selfprivacy_api.utils.block_devices import BlockDevice
from selfprivacy_api.services.ocserv.icon import OCSERV_ICON from selfprivacy_api.services.ocserv.icon import OCSERV_ICON
import selfprivacy_api.utils.network as network_utils import selfprivacy_api.utils.network as network_utils

View file

@ -4,10 +4,10 @@ import subprocess
import typing import typing
from selfprivacy_api.jobs import Job, Jobs from selfprivacy_api.jobs import Job, Jobs
from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service from selfprivacy_api.services.generic_service_mover import FolderMoveNames, move_service
from selfprivacy_api.services.generic_status_getter import get_service_status from selfprivacy_api.utils.systemd import get_service_status
from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus from selfprivacy_api.services.service import Service, ServiceDnsRecord, ServiceStatus
from selfprivacy_api.services.owned_path import OwnedPath from selfprivacy_api.services.owned_path import OwnedPath
from selfprivacy_api.utils import ReadUserData, WriteUserData, get_domain from selfprivacy_api.utils import get_domain
from selfprivacy_api.utils.block_devices import BlockDevice from selfprivacy_api.utils.block_devices import BlockDevice
import selfprivacy_api.utils.network as network_utils import selfprivacy_api.utils.network as network_utils
from selfprivacy_api.services.pleroma.icon import PLEROMA_ICON from selfprivacy_api.services.pleroma.icon import PLEROMA_ICON

View file

@ -1,5 +1,6 @@
"""Generic service status fetcher using systemctl""" """Generic service status fetcher using systemctl"""
import subprocess import subprocess
from typing import List
from selfprivacy_api.services.service import ServiceStatus from selfprivacy_api.services.service import ServiceStatus
@ -58,3 +59,24 @@ def get_service_status_from_several_units(services: list[str]) -> ServiceStatus:
if ServiceStatus.ACTIVE in service_statuses: if ServiceStatus.ACTIVE in service_statuses:
return ServiceStatus.ACTIVE return ServiceStatus.ACTIVE
return ServiceStatus.OFF return ServiceStatus.OFF
def get_last_log_lines(service: str, lines_count: int) -> List[str]:
if lines_count < 1:
raise ValueError("lines_count must be greater than 0")
try:
logs = subprocess.check_output(
[
"journalctl",
"-u",
service,
"-n",
str(lines_count),
"-o",
"cat",
],
shell=False,
).decode("utf-8")
return logs.splitlines()
except subprocess.CalledProcessError:
return []

View file

@ -4,6 +4,7 @@
import pytest import pytest
from selfprivacy_api.jobs import JobStatus, Jobs from selfprivacy_api.jobs import JobStatus, Jobs
from tests.test_graphql.common import assert_empty, assert_ok, get_data
class ProcessMock: class ProcessMock:
@ -92,8 +93,7 @@ def test_graphql_system_rebuild_unauthorized(client, fp, action):
"query": query, "query": query,
}, },
) )
assert response.status_code == 200 assert_empty(response)
assert response.json().get("data") is None
assert fp.call_count([fp.any()]) == 0 assert fp.call_count([fp.any()]) == 0
@ -111,23 +111,23 @@ def test_graphql_system_rebuild(authorized_client, fp, action, mock_sleep_interv
fp.register(["systemctl", "start", unit_name]) fp.register(["systemctl", "start", unit_name])
# Wait for it to start # Wait for it to start
fp.register(["systemctl", "is-active", unit_name], stdout="inactive") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=inactive")
fp.register(["systemctl", "is-active", unit_name], stdout="inactive") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=inactive")
fp.register(["systemctl", "is-active", unit_name], stdout="active") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=active")
# Check its exectution # Check its exectution
fp.register(["systemctl", "is-active", unit_name], stdout="active") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=active")
fp.register( fp.register(
["journalctl", "-u", unit_name, "-n", "1", "-o", "cat"], ["journalctl", "-u", unit_name, "-n", "1", "-o", "cat"],
stdout="Starting rebuild...", stdout="Starting rebuild...",
) )
fp.register(["systemctl", "is-active", unit_name], stdout="active") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=active")
fp.register( fp.register(
["journalctl", "-u", unit_name, "-n", "1", "-o", "cat"], stdout="Rebuilding..." ["journalctl", "-u", unit_name, "-n", "1", "-o", "cat"], stdout="Rebuilding..."
) )
fp.register(["systemctl", "is-active", unit_name], stdout="inactive") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=inactive")
response = authorized_client.post( response = authorized_client.post(
"/graphql", "/graphql",
@ -135,23 +135,11 @@ def test_graphql_system_rebuild(authorized_client, fp, action, mock_sleep_interv
"query": query, "query": query,
}, },
) )
assert response.status_code == 200 data = get_data(response)["system"][f"runSystem{action.capitalize()}"]
assert response.json().get("data") is not None assert_ok(data)
assert (
response.json()["data"]["system"][f"runSystem{action.capitalize()}"]["success"]
is True
)
assert (
response.json()["data"]["system"][f"runSystem{action.capitalize()}"]["message"]
is not None
)
assert (
response.json()["data"]["system"][f"runSystem{action.capitalize()}"]["code"]
== 200
)
assert fp.call_count(["systemctl", "start", unit_name]) == 1 assert fp.call_count(["systemctl", "start", unit_name]) == 1
assert fp.call_count(["systemctl", "is-active", unit_name]) == 6 assert fp.call_count(["systemctl", "show", unit_name]) == 6
job_id = response.json()["data"]["system"][f"runSystem{action.capitalize()}"][ job_id = response.json()["data"]["system"][f"runSystem{action.capitalize()}"][
"job" "job"
@ -176,23 +164,23 @@ def test_graphql_system_rebuild_failed(
fp.register(["systemctl", "start", unit_name]) fp.register(["systemctl", "start", unit_name])
# Wait for it to start # Wait for it to start
fp.register(["systemctl", "is-active", unit_name], stdout="inactive") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=inactive")
fp.register(["systemctl", "is-active", unit_name], stdout="inactive") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=inactive")
fp.register(["systemctl", "is-active", unit_name], stdout="active") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=active")
# Check its exectution # Check its exectution
fp.register(["systemctl", "is-active", unit_name], stdout="active") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=active")
fp.register( fp.register(
["journalctl", "-u", unit_name, "-n", "1", "-o", "cat"], ["journalctl", "-u", unit_name, "-n", "1", "-o", "cat"],
stdout="Starting rebuild...", stdout="Starting rebuild...",
) )
fp.register(["systemctl", "is-active", unit_name], stdout="active") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=active")
fp.register( fp.register(
["journalctl", "-u", unit_name, "-n", "1", "-o", "cat"], stdout="Rebuilding..." ["journalctl", "-u", unit_name, "-n", "1", "-o", "cat"], stdout="Rebuilding..."
) )
fp.register(["systemctl", "is-active", unit_name], stdout="failed") fp.register(["systemctl", "show", unit_name], stdout="ActiveState=failed")
response = authorized_client.post( response = authorized_client.post(
"/graphql", "/graphql",
@ -200,23 +188,11 @@ def test_graphql_system_rebuild_failed(
"query": query, "query": query,
}, },
) )
assert response.status_code == 200 data = get_data(response)["system"][f"runSystem{action.capitalize()}"]
assert response.json().get("data") is not None assert_ok(data)
assert (
response.json()["data"]["system"][f"runSystem{action.capitalize()}"]["success"]
is True
)
assert (
response.json()["data"]["system"][f"runSystem{action.capitalize()}"]["message"]
is not None
)
assert (
response.json()["data"]["system"][f"runSystem{action.capitalize()}"]["code"]
== 200
)
assert fp.call_count(["systemctl", "start", unit_name]) == 1 assert fp.call_count(["systemctl", "start", unit_name]) == 1
assert fp.call_count(["systemctl", "is-active", unit_name]) == 6 assert fp.call_count(["systemctl", "show", unit_name]) == 6
job_id = response.json()["data"]["system"][f"runSystem{action.capitalize()}"][ job_id = response.json()["data"]["system"][f"runSystem{action.capitalize()}"][
"job" "job"