Files
allhaileris afb81b8278
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Close stale issues and PRs / stale (push) Has been cancelled
init
2026-02-16 15:50:16 +03:00

337 lines
9.8 KiB
Python

# SPDX-License-Identifier: LGPL-2.1-or-later
#
# This file is formatted with Python Black
# mypy: disable-error-code="misc"
from tests.templates import Response, init_logger
from collections import namedtuple
from itertools import count
from gi.repository import GLib
from dataclasses import dataclass
import dbus
import dbus.service
import socket
BUS_NAME = "org.freedesktop.impl.portal.Test"
MAIN_OBJ = "/org/freedesktop/portal/desktop"
SYSTEM_BUS = False
MAIN_IFACE = "org.freedesktop.impl.portal.InputCapture"
VERSION = 1
logger = init_logger(__name__)
serials = count()
Barrier = namedtuple("Barrier", ["id", "position"])
@dataclass
class InputcaptureParameters:
delay: int
supported_capabilities: int
capabilities: int
default_zone: list
disable_delay: int
activated_delay: int
deactivated_delay: int
zones_changed_delay: int
def load(mock, parameters={}):
logger.debug(f"Loading parameters: {parameters}")
assert not hasattr(mock, "inputcapture_params")
mock.inputcapture_params = InputcaptureParameters(
delay=parameters.get("delay", 0),
supported_capabilities=parameters.get("supported_capabilities", 0xF),
capabilities=parameters.get("capabilities", None),
default_zone=parameters.get("default-zone", [(1920, 1080, 0, 0)]),
disable_delay=parameters.get("disable-delay", 0),
activated_delay=parameters.get("activated-delay", 0),
deactivated_delay=parameters.get("deactivated-delay", 0),
zones_changed_delay=parameters.get("zones-changed-delay", 0),
)
mock.current_zones = mock.inputcapture_params.default_zone
mock.current_zone_set = next(serials)
mock.AddProperties(
MAIN_IFACE,
dbus.Dictionary(
{
"version": dbus.UInt32(parameters.get("version", VERSION)),
"SupportedCapabilities": dbus.UInt32(
mock.inputcapture_params.supported_capabilities
),
}
),
)
mock.active_session_handles = []
@dbus.service.method(
MAIN_IFACE,
in_signature="oossa{sv}",
out_signature="ua{sv}",
)
def CreateSession(self, handle, session_handle, app_id, parent_window, options):
try:
logger.debug(f"CreateSession({parent_window}, {options})")
params = self.inputcapture_params
assert "capabilities" in options
# Filter to the subset of supported capabilities
if params.capabilities is None:
capabilities = options["capabilities"]
else:
capabilities = params.capabilities
capabilities &= params.supported_capabilities
response = Response(0, {})
response.results["capabilities"] = dbus.UInt32(capabilities)
self.active_session_handles.append(session_handle)
logger.debug(f"CreateSession with response {response}")
return response.response, response.results
except Exception as e:
logger.critical(e)
return (2, {})
@dbus.service.method(
MAIN_IFACE,
in_signature="oosa{sv}",
out_signature="ua{sv}",
)
def GetZones(self, handle, session_handle, app_id, options):
try:
logger.debug(f"GetZones({session_handle}, {options})")
params = self.inputcapture_params
assert session_handle in self.active_session_handles
response = Response(0, {})
response.results["zones"] = params.default_zone
response.results["zone_set"] = dbus.UInt32(
self.current_zone_set, variant_level=1
)
logger.debug(f"GetZones with response {response}")
if response.response == 0:
self.current_zones = response.results["zones"]
return response.response, response.results
except Exception as e:
logger.critical(e)
return (2, {})
@dbus.service.method(
MAIN_IFACE,
in_signature="oosa{sv}aa{sv}u",
out_signature="ua{sv}",
)
def SetPointerBarriers(
self, handle, session_handle, app_id, options, barriers, zone_set
):
try:
logger.debug(
f"SetPointerBarriers({session_handle}, {options}, {barriers}, {zone_set})"
)
assert session_handle in self.active_session_handles
assert zone_set == self.current_zone_set
self.current_barriers = []
failed_barriers = []
# Barrier sanity checks:
for b in barriers:
id = b["barrier_id"]
x1, y1, x2, y2 = b["position"]
if (x1 != x2 and y1 != y2) or (x1 == x2 and y1 == y2):
logger.debug(f"Barrier {id} is not horizontal or vertical")
failed_barriers.append(id)
continue
for z in self.current_zones:
w, h, x, y = z
if x1 < x or x1 > x + w:
continue
if y1 < y or y1 > y + h:
continue
# x1/y1 fit into our current zone
if x2 < x or x2 > x + w or y2 < y or y2 > y + h:
logger.debug(f"Barrier {id} spans multiple zones")
elif x1 == x2 and (x1 != x and x1 != x + w):
logger.debug(f"Barrier {id} is not on vertical edge")
elif y1 == y2 and (y1 != y and y1 != y + h):
logger.debug(f"Barrier {id} is not on horizontal edge")
else:
self.current_barriers.append(Barrier(id=id, position=b["position"]))
break
failed_barriers.append(id)
break
else:
logger.debug(f"Barrier {id} does not fit into any zone")
failed_barriers.append(id)
continue
response = Response(0, {})
response.results["failed_barriers"] = dbus.Array(
[dbus.UInt32(f) for f in failed_barriers],
signature="u",
variant_level=1,
)
logger.debug(f"SetPointerBarriers with response {response}")
return response.response, response.results
except Exception as e:
logger.critical(e)
return (2, {})
@dbus.service.method(
MAIN_IFACE,
in_signature="osa{sv}",
out_signature="ua{sv}",
)
def Enable(self, session_handle, app_id, options):
try:
logger.debug(f"Enable({session_handle}, {options})")
params = self.inputcapture_params
assert session_handle in self.active_session_handles
# for use in the signals
activation_id = next(serials)
barrier = self.current_barriers[0]
pos = (barrier.position[0] + 10, barrier.position[1] + 20)
if params.disable_delay > 0:
def disable():
logger.debug("emitting Disabled")
self.EmitSignal(MAIN_IFACE, "Disabled", "oa{sv}", [session_handle, {}])
GLib.timeout_add(params.disable_delay, disable)
if params.activated_delay > 0:
def activated():
logger.debug("emitting Activated")
options = {
"activation_id": dbus.UInt32(activation_id, variant_level=1),
"barrier_id": dbus.UInt32(barrier.id, variant_level=1),
"cursor_position": dbus.Struct(
pos, signature="dd", variant_level=1
),
}
self.EmitSignal(
MAIN_IFACE, "Activated", "oa{sv}", [session_handle, options]
)
GLib.timeout_add(params.activated_delay, activated)
if params.deactivated_delay > 0:
def deactivated():
logger.debug("emitting Deactivated")
options = {
"activation_id": dbus.UInt32(activation_id, variant_level=1),
"cursor_position": dbus.Struct(
pos, signature="dd", variant_level=1
),
}
self.EmitSignal(
MAIN_IFACE, "Deactivated", "oa{sv}", [session_handle, options]
)
GLib.timeout_add(params.deactivated_delay, deactivated)
if params.zones_changed_delay > 0:
def zones_changed():
logger.debug("emitting ZonesChanged")
options = {
"zone_set": dbus.UInt32(activation_id, variant_level=1),
}
self.EmitSignal(
MAIN_IFACE, "ZonesChanged", "oa{sv}", [session_handle, options]
)
GLib.timeout_add(params.zones_changed_delay, zones_changed)
except Exception as e:
logger.critical(e)
return (2, {})
@dbus.service.method(
MAIN_IFACE,
in_signature="osa{sv}",
out_signature="ua{sv}",
)
def Disable(self, session_handle, app_id, options):
try:
logger.debug(f"Disable({session_handle}, {options})")
assert session_handle in self.active_session_handles
except Exception as e:
logger.critical(e)
return (2, {})
@dbus.service.method(
MAIN_IFACE,
in_signature="osa{sv}",
out_signature="ua{sv}",
)
def Release(self, session_handle, app_id, options):
try:
logger.debug(f"Release({session_handle}, {options})")
assert session_handle in self.active_session_handles
except Exception as e:
logger.critical(e)
return (2, {})
@dbus.service.method(
MAIN_IFACE,
in_signature="osa{sv}",
out_signature="h",
)
def ConnectToEIS(self, session_handle, app_id, options):
try:
logger.debug(f"ConnectToEIS({session_handle}, {options})")
assert session_handle in self.active_session_handles
sockets = socket.socketpair()
self.eis_socket = sockets[0]
assert self.eis_socket.send(b"HELLO") == 5
fd = sockets[1]
logger.debug(f"ConnectToEis with fd {fd.fileno()}")
return dbus.types.UnixFd(fd)
except Exception as e:
logger.critical(e)
return -1