dangerzone/tests/isolation_provider/test_qubes.py
Alexis Métaireau cbbd6afcc1
chore: remove unused code
This commit removes code that's not being used, it can be exceptions
with the `as e` where the exception itself is not used, the same with
`with` statements, and some other parts where there were duplicated
code.
2024-06-05 14:19:31 +02:00

102 lines
3.4 KiB
Python

import os
import pathlib
import subprocess
import time
import pytest
from pytest import MonkeyPatch
from pytest_mock import MockerFixture
from dangerzone.conversion import errors
from dangerzone.document import Document
from dangerzone.isolation_provider.qubes import (
Qubes,
is_qubes_native_conversion,
running_on_qubes,
)
# XXX Fixtures used in abstract Test class need to be imported regardless
from .. import (
pdf_11k_pages,
sample_bad_height,
sample_bad_width,
sample_doc,
sanitized_text,
uncommon_text,
)
from .base import IsolationProviderTermination, IsolationProviderTest
@pytest.fixture
def provider() -> Qubes:
return Qubes()
class QubesWait(Qubes):
"""Qubes isolation provider that blocks until the disposable qube has started."""
def start_doc_to_pixels_proc(self, document: Document) -> subprocess.Popen:
# Check every 100ms if the disposable qube has started. Qubes gives us no
# way to figure this out, but `qrexec-client-vm` has an interesting
# property. It will start a vchan server **only** once the disposable qube
# has started (i.e., on MSG_SERVICE_CONNECT) [1]
#
# While `qrexec-client-vm` does not report this either, we can snoop on its
# open file descriptors, and see when it opens the `/dev/xen` character
# devices. This is super flaky and probably subject to race conditions, but
# since it's test code, we can live with it.
#
# [1]: https://www.qubes-os.org/doc/qrexec-internals/#domx-invoke-execution-of-qubes-service-qubesservice-in-domy
proc = super().start_doc_to_pixels_proc(document)
for i in range(300):
for p in pathlib.Path(f"/proc/{proc.pid}/fd").iterdir():
if str(p.resolve()).startswith("/dev/xen"):
# The `qrexec-client-vm` process has opened a `/dev/xen` character
# device. We can now yield control back to the caller.
return proc
time.sleep(0.1)
raise RuntimeError("Disposable qube did not start within 30 seconds")
@pytest.fixture
def provider_wait() -> QubesWait:
return QubesWait()
@pytest.mark.skipif(not running_on_qubes(), reason="Not on a Qubes system")
class TestQubes(IsolationProviderTest):
def test_out_of_ram(
self,
provider: Qubes,
mocker: MockerFixture,
monkeypatch: MonkeyPatch,
sample_doc: str,
tmpdir: str,
) -> None:
provider.progress_callback = mocker.MagicMock()
proc = subprocess.Popen(
# XXX error 126 simulates a qrexec-policy failure. Source:
# https://github.com/QubesOS/qubes-core-qrexec/blob/fdcbfd7/daemon/qrexec-daemon.c#L1022
["exit 126"],
stdin=subprocess.PIPE,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
shell=True,
)
with pytest.raises(errors.ConverterProcException):
doc = Document(sample_doc)
provider.doc_to_pixels(doc, tmpdir, proc)
assert provider.get_proc_exception(proc) == errors.QubesQrexecFailed
@pytest.mark.skipif(
os.environ.get("DUMMY_CONVERSION", False),
reason="cannot run for dummy conversions",
)
@pytest.mark.skipif(
not is_qubes_native_conversion(), reason="Qubes native conversion is not enabled"
)
class TestQubesTermination(IsolationProviderTermination):
pass