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
337 lines
9.8 KiB
Python
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
|