input-remapper/tests/test.py

504 lines
14 KiB
Python
Raw Normal View History

2020-10-26 22:45:22 +00:00
#!/usr/bin/python3
# -*- coding: utf-8 -*-
# key-mapper - GUI for device specific keyboard mappings
2021-01-02 23:08:33 +00:00
# Copyright (C) 2021 sezanzeb <proxima@hip70890b.de>
2020-10-26 22:45:22 +00:00
#
2020-10-31 13:02:59 +00:00
# This file is part of key-mapper.
2020-10-26 22:45:22 +00:00
#
2020-10-31 13:02:59 +00:00
# key-mapper is free software: you can redistribute it and/or modify
2020-10-26 22:45:22 +00:00
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
2020-10-31 13:02:59 +00:00
# key-mapper is distributed in the hope that it will be useful,
2020-10-26 22:45:22 +00:00
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
2020-10-31 13:02:59 +00:00
# along with key-mapper. If not, see <https://www.gnu.org/licenses/>.
2020-10-26 22:45:22 +00:00
2020-10-31 13:02:59 +00:00
"""Sets up key-mapper for the tests and runs them."""
2020-10-26 22:45:22 +00:00
2020-11-29 15:21:34 +00:00
import os
2020-10-26 22:45:22 +00:00
import sys
import shutil
2020-11-19 10:40:15 +00:00
import time
2020-12-03 19:03:53 +00:00
import copy
2020-10-26 22:45:22 +00:00
import unittest
2020-11-29 00:44:14 +00:00
import subprocess
2020-11-20 20:38:59 +00:00
import multiprocessing
2020-11-28 14:43:24 +00:00
import asyncio
import evdev
import gi
gi.require_version('Gtk', '3.0')
gi.require_version('GLib', '2.0')
2020-11-16 14:39:15 +00:00
2020-11-29 15:21:34 +00:00
assert not os.getcwd().endswith('tests')
def is_service_running():
"""Check if the daemon is running."""
try:
subprocess.check_output(['pgrep', '-f', 'key-mapper-service'])
return True
except subprocess.CalledProcessError:
return False
if is_service_running():
# let tests control daemon existance
raise Exception('Expected the service not to be running already.')
# make sure the "tests" module visible
sys.path.append(os.getcwd())
2020-11-29 15:21:34 +00:00
2020-11-30 17:59:34 +00:00
# give tests some time to test stuff while the process
# is still running
EVENT_READ_TIMEOUT = 0.01
2020-11-30 21:42:53 +00:00
MAX_ABS = 2 ** 15
2020-11-29 15:21:34 +00:00
2020-11-16 14:39:15 +00:00
tmp = '/tmp/key-mapper-test'
2020-11-18 22:00:18 +00:00
uinput_write_history = []
2020-11-20 20:38:59 +00:00
# for tests that makes the injector create its processes
uinput_write_history_pipe = multiprocessing.Pipe()
2020-11-18 23:04:04 +00:00
pending_events = {}
2020-11-18 22:00:18 +00:00
2021-01-01 21:20:33 +00:00
def read_write_history_pipe():
"""convert the write history from the pipe to some easier to manage list"""
history = []
while uinput_write_history_pipe[0].poll():
event = uinput_write_history_pipe[0].recv()
history.append((event.type, event.code, event.value))
return history
# key-mapper is only interested in devices that have EV_KEY, add some
# random other stuff to test that they are ignored.
2020-12-27 10:26:07 +00:00
phys_1 = 'usb-0000:03:00.0-1/input2'
2021-01-08 16:21:51 +00:00
info_1 = evdev.device.DeviceInfo(1, 1, 1, 1)
2020-12-27 10:26:07 +00:00
fixtures = {
# device 1
'/dev/input/event11': {
2021-01-05 18:33:47 +00:00
'capabilities': {evdev.ecodes.EV_KEY: [], evdev.ecodes.EV_REL: [
evdev.ecodes.REL_WHEEL,
evdev.ecodes.REL_HWHEEL
]},
2020-12-27 10:26:07 +00:00
'phys': f'{phys_1}/input2',
'info': info_1,
2021-01-05 18:33:47 +00:00
'name': 'device 1 foo',
'group': 'device 1'
},
'/dev/input/event10': {
'capabilities': {evdev.ecodes.EV_KEY: list(evdev.ecodes.keys.keys())},
2020-12-27 10:26:07 +00:00
'phys': f'{phys_1}/input3',
'info': info_1,
2021-01-05 18:33:47 +00:00
'name': 'device 1',
'group': 'device 1'
},
'/dev/input/event13': {
'capabilities': {evdev.ecodes.EV_KEY: [], evdev.ecodes.EV_SYN: []},
2020-12-27 10:26:07 +00:00
'phys': f'{phys_1}/input1',
'info': info_1,
2021-01-05 18:33:47 +00:00
'name': 'device 1',
'group': 'device 1'
},
'/dev/input/event14': {
'capabilities': {evdev.ecodes.EV_SYN: []},
2020-12-27 10:26:07 +00:00
'phys': f'{phys_1}/input0',
'info': info_1,
2021-01-05 18:33:47 +00:00
'name': 'device 1 qux',
'group': 'device 1'
},
# device 2
'/dev/input/event20': {
'capabilities': {evdev.ecodes.EV_KEY: list(evdev.ecodes.keys.keys())},
'phys': 'usb-0000:03:00.0-2/input1',
2021-01-08 16:21:51 +00:00
'info': evdev.device.DeviceInfo(2, 1, 2, 1),
'name': 'device 2'
},
'/dev/input/event30': {
2020-12-03 19:03:53 +00:00
'capabilities': {
evdev.ecodes.EV_SYN: [],
evdev.ecodes.EV_ABS: [
evdev.ecodes.ABS_X,
evdev.ecodes.ABS_Y,
evdev.ecodes.ABS_RX,
evdev.ecodes.ABS_RY,
2020-12-03 19:03:53 +00:00
evdev.ecodes.ABS_HAT0X
2020-12-26 15:46:01 +00:00
],
evdev.ecodes.EV_KEY: [
evdev.ecodes.BTN_A
2020-12-03 19:03:53 +00:00
]
},
'phys': 'usb-0000:03:00.0-3/input1',
2021-01-08 16:21:51 +00:00
'info': evdev.device.DeviceInfo(3, 1, 3, 1),
2020-11-30 21:42:53 +00:00
'name': 'gamepad'
},
2020-11-30 21:42:53 +00:00
# device that is completely ignored
2020-11-22 20:54:09 +00:00
'/dev/input/event31': {
'capabilities': {evdev.ecodes.EV_SYN: []},
'phys': 'usb-0000:03:00.0-4/input1',
2021-01-08 16:21:51 +00:00
'info': evdev.device.DeviceInfo(4, 1, 4, 1),
2020-11-22 20:54:09 +00:00
'name': 'Power Button'
},
# key-mapper devices are not displayed in the ui, some instance
# of key-mapper started injecting apparently.
'/dev/input/event40': {
'capabilities': {evdev.ecodes.EV_KEY: list(evdev.ecodes.keys.keys())},
'phys': 'key-mapper/input1',
2021-01-08 16:21:51 +00:00
'info': evdev.device.DeviceInfo(5, 1, 5, 1),
'name': 'key-mapper device 2'
},
}
2020-11-18 22:00:18 +00:00
2020-11-18 23:04:04 +00:00
def get_events():
"""Get all events written by the injector."""
return uinput_write_history
2020-11-18 22:00:18 +00:00
def push_event(device, event):
2020-11-18 23:04:04 +00:00
"""Emit a fake event for a device.
2020-11-18 22:00:18 +00:00
Parameters
----------
device : string
2020-11-18 23:04:04 +00:00
For example 'device 1'
2020-12-06 13:23:00 +00:00
event : InputEvent
2020-11-18 22:00:18 +00:00
"""
2020-11-18 23:04:04 +00:00
if pending_events.get(device) is None:
pending_events[device] = []
pending_events[device].append(event)
2020-11-18 22:00:18 +00:00
2021-01-05 18:33:47 +00:00
def new_event(type, code, value, timestamp=None):
"""Create a new input_event."""
if timestamp is None:
timestamp = time.time()
2020-12-05 10:58:29 +00:00
2021-01-05 18:33:47 +00:00
sec = int(timestamp)
usec = timestamp % 1 * 1000000
event = evdev.InputEvent(sec, usec, type, code, value)
return event
2021-01-01 21:20:33 +00:00
2020-11-16 14:39:15 +00:00
def patch_paths():
from keymapper import paths
paths.CONFIG_PATH = '/tmp/key-mapper-test'
2020-11-16 14:39:15 +00:00
2020-11-30 17:59:34 +00:00
def patch_select():
# goes hand in hand with patch_evdev, which makes InputDevices return
# their names for `.fd`.
# rlist contains device names therefore, so select.select returns the
# name of the device for which events are pending.
import select
def new_select(rlist, *args):
ret = []
for thing in rlist:
if hasattr(thing, 'poll') and thing.poll():
# the reader receives msgs through pipes. If there is one
# ready, provide the pipe
ret.append(thing)
continue
if len(pending_events.get(thing, [])) > 0:
ret.append(thing)
2021-01-01 21:20:33 +00:00
# avoid a fast iterating infinite loop in the reader
time.sleep(0.01)
return [ret, [], []]
2020-11-30 17:59:34 +00:00
select.select = new_select
class InputDevice:
# expose as existing attribute, otherwise the patch for
# evdev < 1.0.0 will crash the test
path = None
2020-11-14 23:27:45 +00:00
def __init__(self, path):
2021-01-05 18:33:47 +00:00
if path != 'justdoit' and path not in fixtures:
raise FileNotFoundError()
2020-11-30 13:34:27 +00:00
self.path = path
2021-01-05 18:33:47 +00:00
fixture = fixtures.get(path, {})
self.phys = fixture.get('phys', 'unset')
2021-01-08 16:21:51 +00:00
self.info = fixture.get('info', evdev.device.DeviceInfo(None, None, None, None))
2021-01-05 18:33:47 +00:00
self.name = fixture.get('name', 'unset')
self.fd = self.name
2020-12-03 19:37:36 +00:00
2021-01-05 18:33:47 +00:00
# properties that exists for test purposes and are not part of
# the original object
self.group = fixture.get('group', self.name)
2021-01-01 21:20:33 +00:00
def log(self, key, msg):
print(
f'\033[90m' # color
2021-01-05 18:33:47 +00:00
f'{msg} "{self.name}" "{self.path}" {key}'
2021-01-01 21:20:33 +00:00
'\033[0m' # end style
)
2020-12-26 15:46:01 +00:00
def absinfo(self, *args):
raise Exception('Ubuntus version of evdev doesn\'t support .absinfo')
2020-11-30 20:16:58 +00:00
def grab(self):
pass
2020-11-30 20:16:58 +00:00
def read(self):
2021-01-05 18:33:47 +00:00
# the patched fake InputDevice objects read anything pending from
# that group, to be realistic it would have to check if the provided
# element is in its capabilities.
2021-01-07 16:15:12 +00:00
ret = [e.copy() for e in pending_events.get(self.group, [])]
if ret is not None:
# consume all of them
2021-01-05 18:33:47 +00:00
pending_events[self.group] = []
2020-11-18 22:00:18 +00:00
return ret
2020-11-30 17:59:34 +00:00
def read_one(self):
2021-01-05 18:33:47 +00:00
if pending_events.get(self.group) is None:
return None
2020-11-30 17:59:34 +00:00
2021-01-05 18:33:47 +00:00
if len(pending_events[self.group]) == 0:
return None
2021-01-07 16:15:12 +00:00
event = pending_events[self.group].pop(0).copy()
2021-01-01 21:20:33 +00:00
self.log(event, 'read_one')
return event
def read_loop(self):
"""Read all prepared events at once."""
2021-01-05 18:33:47 +00:00
if pending_events.get(self.group) is None:
return
2021-01-05 18:33:47 +00:00
while len(pending_events[self.group]) > 0:
2021-01-07 16:15:12 +00:00
result = pending_events[self.group].pop(0).copy()
2021-01-01 21:20:33 +00:00
self.log(result, 'read_loop')
yield result
time.sleep(EVENT_READ_TIMEOUT)
2020-11-18 22:00:18 +00:00
async def async_read_loop(self):
"""Read all prepared events at once."""
2021-01-05 18:33:47 +00:00
if pending_events.get(self.group) is None:
return
2020-11-18 23:04:04 +00:00
2021-01-05 18:33:47 +00:00
while len(pending_events[self.group]) > 0:
2021-01-07 16:15:12 +00:00
result = pending_events[self.group].pop(0).copy()
2021-01-01 21:20:33 +00:00
self.log(result, 'async_read_loop')
yield result
await asyncio.sleep(0.01)
2020-11-28 14:43:24 +00:00
def capabilities(self, absinfo=True):
2020-12-26 15:46:01 +00:00
result = copy.deepcopy(fixtures[self.path]['capabilities'])
if absinfo and evdev.ecodes.EV_ABS in result:
absinfo_obj = evdev.AbsInfo(
value=None, min=None, fuzz=None, flat=None,
resolution=None, max=MAX_ABS
)
result[evdev.ecodes.EV_ABS] = [
(stuff, absinfo_obj) for stuff in result[evdev.ecodes.EV_ABS]
]
return result
2020-11-28 14:43:24 +00:00
2020-11-07 23:54:19 +00:00
class UInput:
2021-01-05 18:33:47 +00:00
def __init__(self, events=None, name='unnamed', *args, **kwargs):
self.fd = 0
self.write_count = 0
2021-01-05 18:33:47 +00:00
self.device = InputDevice('justdoit')
self.name = name
self.events = events
2020-11-18 23:04:04 +00:00
def capabilities(self, *args, **kwargs):
2021-01-05 18:33:47 +00:00
return self.events
2020-12-06 18:54:02 +00:00
def write(self, type, code, value):
self.write_count += 1
2021-01-05 18:33:47 +00:00
event = new_event(type, code, value)
uinput_write_history.append(event)
uinput_write_history_pipe[1].send(event)
2021-01-05 18:33:47 +00:00
print(
f'\033[90m' # color
f'{(type, code, value)} written'
'\033[0m' # end style
)
2020-11-18 22:00:18 +00:00
def syn(self):
pass
2021-01-05 18:33:47 +00:00
class InputEvent(evdev.InputEvent):
def __init__(self, sec, usec, type, code, value):
self.t = (type, code, value)
super().__init__(sec, usec, type, code, value)
2021-01-07 16:15:12 +00:00
def copy(self):
return InputEvent(
self.sec,
self.usec,
self.type,
self.code,
self.value
)
2021-01-05 18:33:47 +00:00
def patch_evdev():
def list_devices():
return fixtures.keys()
2020-11-18 22:00:18 +00:00
2020-11-16 14:39:15 +00:00
evdev.list_devices = list_devices
evdev.InputDevice = InputDevice
2020-11-18 22:00:18 +00:00
evdev.UInput = UInput
2021-01-05 18:33:47 +00:00
evdev.InputEvent = InputEvent
2020-11-16 14:39:15 +00:00
2021-01-05 18:33:47 +00:00
def patch_events():
# improve logging of stuff
evdev.InputEvent.__str__ = lambda self: (
f'InputEvent{(self.type, self.code, self.value)}'
)
2020-11-28 23:28:46 +00:00
def clear_write_history():
"""Empty the history in preparation for the next test."""
while len(uinput_write_history) > 0:
uinput_write_history.pop()
while uinput_write_history_pipe[0].poll():
uinput_write_history_pipe[0].recv()
2020-12-03 19:03:53 +00:00
2020-11-16 14:39:15 +00:00
# quickly fake some stuff before any other file gets a chance to import
# the original versions
patch_paths()
patch_evdev()
2020-11-30 17:59:34 +00:00
patch_select()
2021-01-05 18:33:47 +00:00
patch_events()
2020-11-07 23:54:19 +00:00
from keymapper.logger import update_verbosity
from keymapper.dev.injector import Injector
from keymapper.config import config
2020-12-31 20:46:57 +00:00
from keymapper.dev.reader import keycode_reader
2020-12-26 15:46:01 +00:00
from keymapper.getdevices import refresh_devices
from keymapper.state import system_mapping, custom_mapping
2021-01-09 16:44:24 +00:00
from keymapper.paths import get_config_path
2020-12-31 20:46:57 +00:00
from keymapper.dev.keycode_mapper import active_macros, unreleased
2020-10-26 22:45:22 +00:00
2020-12-06 18:54:02 +00:00
# no need for a high number in tests
Injector.regrab_timeout = 0.15
2020-12-06 18:54:02 +00:00
2020-12-26 15:46:01 +00:00
_fixture_copy = copy.deepcopy(fixtures)
2021-01-07 16:15:12 +00:00
environ_copy = copy.deepcopy(os.environ)
2020-12-26 15:46:01 +00:00
def cleanup():
"""Reset the applications state."""
2021-01-05 18:33:47 +00:00
print(
f'\033[90m' # color
f'cleanup'
'\033[0m' # end style
)
2020-12-31 20:46:57 +00:00
keycode_reader.stop_reading()
2021-01-07 16:15:12 +00:00
keycode_reader.__init__()
2020-12-31 20:46:57 +00:00
if asyncio.get_event_loop().is_running():
for task in asyncio.all_tasks():
task.cancel()
2020-12-27 18:06:17 +00:00
os.system('pkill -f key-mapper-service')
2021-01-01 21:20:33 +00:00
time.sleep(0.05)
if os.path.exists(tmp):
shutil.rmtree(tmp)
2020-12-26 15:46:01 +00:00
2021-01-09 16:44:24 +00:00
config.path = os.path.join(get_config_path(), 'config.json')
config.clear_config()
config.save_config()
2020-12-26 15:46:01 +00:00
system_mapping.populate()
custom_mapping.empty()
2020-12-27 18:06:17 +00:00
custom_mapping.clear_config()
2021-01-09 16:44:24 +00:00
custom_mapping.changed = False
2020-12-26 15:46:01 +00:00
clear_write_history()
2020-12-26 15:46:01 +00:00
for key in list(active_macros.keys()):
del active_macros[key]
2020-12-31 20:46:57 +00:00
for key in list(unreleased.keys()):
del unreleased[key]
for key in list(pending_events.keys()):
del pending_events[key]
2020-12-26 15:46:01 +00:00
for path in list(fixtures.keys()):
if path not in _fixture_copy:
del fixtures[path]
for path in list(_fixture_copy.keys()):
if path not in fixtures:
fixtures[path] = _fixture_copy[path]
2020-12-27 18:06:17 +00:00
2021-01-07 16:15:12 +00:00
os.environ.update(environ_copy)
for key in list(os.environ.keys()):
if key not in environ_copy:
del os.environ[key]
2020-12-26 15:46:01 +00:00
refresh_devices()
2020-12-06 18:54:02 +00:00
2020-11-26 20:37:15 +00:00
def main():
update_verbosity(True)
cleanup()
2020-10-26 22:45:22 +00:00
modules = sys.argv[1:]
# discoverer is really convenient, but it can't find a specific test
# in all of the available tests like unittest.main() does...,
# so provide both options.
if len(modules) > 0:
2020-12-02 15:17:52 +00:00
# for example
# `tests/test.py test_integration.TestIntegration.test_can_start`
# or `tests/test.py test_integration test_daemon`
2020-10-26 22:45:22 +00:00
testsuite = unittest.defaultTestLoader.loadTestsFromNames(
[f'testcases.{module}' for module in modules]
)
else:
# run all tests by default
testsuite = unittest.defaultTestLoader.discover(
'testcases', pattern='*.py'
)
2020-11-18 23:28:45 +00:00
# add a newline to each "qux (foo.bar)..." output before each test,
# because the first log will be on the same line otherwise
original_start_test = unittest.TextTestResult.startTest
def start_test(self, test):
original_start_test(self, test)
2020-11-22 14:17:55 +00:00
print()
2020-11-18 23:28:45 +00:00
unittest.TextTestResult.startTest = start_test
unittest.TextTestRunner(verbosity=2).run(testsuite)
2020-11-26 20:37:15 +00:00
if __name__ == "__main__":
main()