From f75dc339e32adf5bcf10d688c1a4391d282ae563 Mon Sep 17 00:00:00 2001 From: Fabian Arndt Date: Wed, 18 Dec 2024 00:17:52 +0100 Subject: [PATCH] Fixed unit tests - Fixed relative imports - Removed import handling via ImportError exception - Fixed CI by running via module, instead of file - Added / improved handling in __init__.py for unit tests --- .github/workflows/ci.yml | 6 +++++- __init__.py | 40 +++++++++++++++++++++++++++++++--------- checks.py | 5 +---- config.py | 7 ++----- engine.py | 1 + fix.py | 12 ++++-------- protonfixes_test.py | 16 +++++++++------- util.py | 11 ++++------- 8 files changed, 57 insertions(+), 41 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 37432194..7d5f086c 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -39,6 +39,10 @@ jobs: run: | python3 .github/scripts/check_gamefixes.py python3 .github/scripts/check_verbs.py + # We need a known parent package in the unit tests, so we need to change to the parent dir + # As "umu-protonfixes" is not a valid package name, we create a symbolic link to "protonfixes" - name: Test with unittest run: | - python3 protonfixes_test.py + cd .. + ln -s umu-protonfixes protonfixes + python3 -m protonfixes.protonfixes_test diff --git a/__init__.py b/__init__.py index 6605138c..aa329ea2 100644 --- a/__init__.py +++ b/__init__.py @@ -1,18 +1,40 @@ -"""Starts the protonfix module""" +"""Starts the protonfix module and runs fixes after pre-flight-checks""" import os import sys +import traceback -RUN_CONDITIONS = [ - 'STEAM_COMPAT_DATA_PATH' in os.environ, - 'PROTONFIXES_DISABLE' not in os.environ, - 'waitforexitandrun' in sys.argv[1], -] +from . import fix +from .logger import log -if all(RUN_CONDITIONS): - import traceback - from . import fix +def check_conditions() -> bool: + """Determine, if the actual game was executed and protonfixes isn't deactivated. + + Returns: + bool: True, if the fix should be executed. + """ + return len(sys.argv) >= 1 and \ + 'STEAM_COMPAT_DATA_PATH' in os.environ and \ + 'PROTONFIXES_DISABLE' not in os.environ and \ + 'waitforexitandrun' in sys.argv[1] + + +def check_iscriptevaluator() -> bool: + """Determine, if we were invoked while running "iscriptevaluator.exe". + + Returns: + bool: True, if we were invoked while running "iscriptevaluator.exe". + """ + return len(sys.argv) >= 2 and \ + 'iscriptevaluator.exe' in sys.argv[2] + + +if check_iscriptevaluator(): + log.debug('Skipping fix execution. We are running "iscriptevaluator.exe".') +elif not check_conditions(): + log.warn('Skipping fix execution. We are probably running an unit test.') +else: try: fix.main() diff --git a/checks.py b/checks.py index 064e0651..b560056d 100644 --- a/checks.py +++ b/checks.py @@ -1,9 +1,6 @@ """Run some tests and generate warnings for proton configuration issues""" -try: - from .logger import log -except ImportError: - from logger import log +from .logger import log def esync_file_limits() -> bool: diff --git a/config.py b/config.py index 0743be36..8032aa28 100644 --- a/config.py +++ b/config.py @@ -1,12 +1,9 @@ """Load configuration settings for protonfixes""" import os -from configparser import ConfigParser -try: - from .logger import log -except ImportError: - from logger import log +from configparser import ConfigParser +from .logger import log CONF_FILE = '~/.config/protonfixes/config.ini' diff --git a/engine.py b/engine.py index e19a777d..8c870315 100644 --- a/engine.py +++ b/engine.py @@ -2,6 +2,7 @@ import os import sys + from .logger import log diff --git a/fix.py b/fix.py index 825833a5..f3fd805e 100644 --- a/fix.py +++ b/fix.py @@ -4,17 +4,13 @@ import re import sys import csv + from functools import lru_cache from importlib import import_module -try: - from . import config - from .checks import run_checks - from .logger import log -except ImportError: - import config - from checks import run_checks - from logger import log +from . import config +from .checks import run_checks +from .logger import log try: import __main__ as protonmain diff --git a/protonfixes_test.py b/protonfixes_test.py index 83ce91b9..5bf76535 100644 --- a/protonfixes_test.py +++ b/protonfixes_test.py @@ -1,11 +1,13 @@ -import unittest +import io import os import tempfile +import unittest +import urllib.request + from pathlib import Path from unittest.mock import patch, mock_open -import io -import urllib.request -import fix + +from . import fix class TestProtonfixes(unittest.TestCase): @@ -315,7 +317,7 @@ def testGetGameNameDBFileNotFound(self): with patch('builtins.open', mock_open()) as mocked_open: mocked_open.side_effect = FileNotFoundError - with patch('fix.log') as mocked_log: # Mock the logger separately + with patch('protonfixes.fix.log') as mocked_log: # Mock the logger separately func = fix.get_game_name.__wrapped__ # Do not reference the cache result = func() self.assertEqual(result, 'UNKNOWN') @@ -329,7 +331,7 @@ def testGetGameNameDbOS(self): with patch('builtins.open', mock_open()) as mocked_open: mocked_open.side_effect = OSError - with patch('fix.log') as mocked_log: # Mock the logger separately + with patch('protonfixes.fix.log') as mocked_log: # Mock the logger separately func = fix.get_game_name.__wrapped__ # Do not reference the cache result = func() self.assertEqual(result, 'UNKNOWN') @@ -357,7 +359,7 @@ def testGetGameNameDbUnicode(self): with patch('builtins.open', mock_open()) as mocked_open: mocked_open.side_effect = UnicodeDecodeError('utf-8', b'', 0, 1, '') - with patch('fix.log') as mocked_log: # Mock the logger separately + with patch('protonfixes.fix.log') as mocked_log: # Mock the logger separately func = fix.get_game_name.__wrapped__ # Do not reference the cache result = func() self.assertEqual(result, 'UNKNOWN') diff --git a/util.py b/util.py index 6b1c0d6c..b39397cd 100644 --- a/util.py +++ b/util.py @@ -1,7 +1,6 @@ """Utilities to make gamefixes easier""" import configparser -from io import TextIOWrapper import os import sys import re @@ -12,16 +11,14 @@ import subprocess import urllib.request import functools + +from io import TextIOWrapper from socket import socket, AF_INET, SOCK_DGRAM from typing import Literal, Any, Callable, Union from collections.abc import Mapping, Generator -try: - from .logger import log - from .steamhelper import install_app -except ImportError: - from logger import log - from steamhelper import install_app +from .logger import log +from .steamhelper import install_app try: import __main__ as protonmain